签到APP:android入门级小项目,Node.js 提供服务端接口。

###1、项目描述
  实验室小伙伴们经过APP链接实验室路由器,比对路由器Mac地址进行签到。此外小伙伴们还可经过APP进行请假,上传请假理由、日期等信息,便于实验室平常管理。html

###2、APP截图 用户登陆 输入图片说明 输入图片说明 输入图片说明 输入图片说明 输入图片说明node

###3、相关github开源项目 感谢开源:android

/*卡片布局*/
compile 'com.android.support:cardview-v7:+'
/*meterial风格对话框*/
compile 'com.afollestad.material-dialogs:core:0.9.1.0'
/*Meterial UI控件*/
compile 'com.github.navasmdc:MaterialDesign:1.5[@aar](https://my.oschina.net/AAR)'
compile 'com.wdullaer:materialdatetimepicker:2.5.0'
/*悬浮按钮*/
compile 'com.getbase:floatingactionbutton:1.10.1'
/*输入框*/
compile 'com.rengwuxian.materialedittext:library:2.1.4'
/*访问网络工具类*/
compile 'com.lzy.net:okgo:2.0.0'
/*Gson Json转化成bean工具类*/
compile 'com.google.code.gson:gson:2.3.1'
/*andrroid工具包*/
compile 'com.blankj:utilcode:1.3.3'

###4、数据库设计   主要设计三个对象:用户、请假记录、签到记录
  **约束:**一个用户天天只能有一条签到记录,同时用户签到过允许再次签到。
  **解决方法:**给每条签到记录经过用户名加日期生成一个unique_id可预见字段,用户签到时查询数据库是否存在存在此字段的记录。
输入图片说明
其中版本更新代码目前并无实现。git

###5、关键代码 5.一、比对路由器Mac地址:github

/*比对路由器的MAC地址*/
    public boolean checkLabMac(){
        //获取WiFIMac地址
        if(NetworkUtils.isWifiConnected(MainActivity.this)){
            WifiManager wifi = (WifiManager) getSystemService(Context.WIFI_SERVICE);
            WifiInfo info = wifi.getConnectionInfo();
            str_mac=info.getBSSID();
            if(!str_mac.equals(Constant.E412_MAC)){
                signSuccess("非实验室WIFI,没法签到!");
                return false;
            }else {
                return true;
            }
        }else{
            signSuccess("WIFI未链接,请先链接实验室WIFI!");
            return false;
        }
    }

5.二、访问网络与Gson解析数据:
   访问网络工具类:compile 'com.lzy.net:okgo:2.0.0'Gson Json, 地址:https://github.com/jeasonlzy/okhttp-OkGo
转化成bean工具类:compile 'com.google.code.gson:gson:2.3.1' Gson转换Json到Bean须要创建相应的Bean与Json字段一一对应,否则会出错。
若是Json和Bean字段对应不上,或者服务端传过来的字段命名不规范,加一个 @SerializedName,解决方案:http://blog.csdn.net/bzy601638015/article/details/32916281 输入图片说明web

经过谷歌浏览器插件PostMan模拟登陆 输入图片说明chrome

这里以用户登陆访问网络为例:
 /*用户名密码校验*/
    public void checkUser(){
        /*构造请求体*/
        HashMap<String, String> params = new HashMap<>();
        params.put("username", str_username);
        params.put("password", str_password);
        JSONObject jsonObject = new JSONObject(params);
        /*发送登陆请求*/
        OkGo.post(Api.LOGIN)//
                .tag(this)//
                .upJson(jsonObject.toString())//
                .execute(new StringCallback() {
                    @Override
                    public void onSuccess(String s, Call call, Response response) {
                        /*关闭提示框*/
                        /* int code = conn.getResponseCode();//返回码200请求成功,若是请求码不是200,则提示服务器出错*/
                        login=new ResponseLogin();
                        login= JsonUtils.fromJson(s,ResponseLogin.class);
                        md.dismiss();
                        if(login.getStatus().equals(Constant.SUCCESS)){
                            /*若是勾选了记住密码,且登陆成功,就保存用户名密码*/
                            if(checkBox.isCheck()){
                              /*记住用户名密码*/
                                PreferencesUtils.putString(LoginActivity.this,"username",str_username);
                                PreferencesUtils.putString(LoginActivity.this,"password",str_password);
                            }
                            toActivity(MainActivity.class);
                            Toast.makeText(LoginActivity.this,"登录成功",Toast.LENGTH_SHORT).show();
                            LoginActivity.this.finish();
                        }else{
                            if(login.getMsg().equals(Constant.ERROR_SYSTEM)){
                                Toast.makeText(LoginActivity.this,"系统错误",Toast.LENGTH_SHORT).show();
                                return;
                            }if(login.getMsg().equals(Constant.ERROR_USERNAME)){
                                Toast.makeText(LoginActivity.this,"用户不存在",Toast.LENGTH_SHORT).show();
                                return;
                            }if(login.getMsg().equals(Constant.ERROR_PASSWORD)){
                                Toast.makeText(LoginActivity.this,"密码错误",Toast.LENGTH_SHORT).show();
                                return;
                            }
                        }
                    }
                });
    }

JSON对应Bean,服务端值放回成功或者失败信息,只有两个字段。

public class ResponseLogin implements Serializable {
    private String status;
    private String msg;
    public void setStatus(String status) {
        this.status = status;
    }
    public String getStatus() {
        return status;
    }

    public void setMsg(String msg) {
        this.msg = msg;
    }
    public String getMsg() {
        return msg;
    }
}

5.三、使用时间选择器问题:
  使用时间选择器是,样例代码是之间经过Activity实现结构,而后重写方法,若是用这种方式一个Activity中貌似只能用一个Dialog选择时间(不知道本身理解是否是对的,欢迎指正),而这里有两个位置须要选择时间。因此选择匿名内部类的方式实现选择时间功能。数据库

btn_startTime.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //弹出日期选择对话框
                Calendar now = Calendar.getInstance();
                DatePickerDialog dialog= DatePickerDialog.newInstance(new DatePickerDialog.OnDateSetListener(){
                  @Override
                  public void onDateSet(DatePickerDialog view, int year, int monthOfYear, int dayOfMonth) {
                      String date =year+"-"+(monthOfYear+1)+"-"+dayOfMonth;
                      startTime.setText(date);
                      }},now.get(Calendar.YEAR),
                        now.get(Calendar.MONTH),
                        now.get(Calendar.DAY_OF_MONTH));
                dialog.show(getFragmentManager(), "Datepickerdialog");
            }
        });
        btn_endTime.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //弹出日期选择对话框
                Calendar now = Calendar.getInstance();
                DatePickerDialog dialog= DatePickerDialog.newInstance(new DatePickerDialog.OnDateSetListener(){
                      @Override
                      public void onDateSet(DatePickerDialog view, int year, int monthOfYear, int dayOfMonth) {
                      String date =year+"-"+(monthOfYear+1)+"-"+dayOfMonth;
                      endTime.setText(date);
                      }},now.get(Calendar.YEAR),
                        now.get(Calendar.MONTH),
                        now.get(Calendar.DAY_OF_MONTH));
                dialog.show(getFragmentManager(), "Datepickerdialog");
            }
        });

5.四、关于Toolbar菜单字体颜色背景颜色问题:
  使用Toorbar的时候,它的菜单字体为白色,菜单背景为灰白色,蛮难看。
  解决方案:http://www.cnblogs.com/oyjt/p/4762640.htmljson

5.五、关于仿知乎悬浮按钮问题:
  主页上仿知乎悬浮按钮,github上例子的布局是相对布局,可是本身经常使用的是线性布局,想着把它用在线性布局中,给它调位置一直没成功,只能将就的把主页改为相对布局吧,效果也还不错。
可给按钮添加图标以及颜色:浏览器

<com.getbase.floatingactionbutton.FloatingActionButton
            android:id="@+id/action_a"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            fab:fab_colorNormal="@color/blue"
            fab:fab_title="我要请假"
            fab:fab_icon="@drawable/holiday"
            fab:fab_colorPressed="@color/white_pressed"/>

地址:https://github.com/futuresimple/android-floating-action-button

5.六、下拉刷新上拉加载更多:
  下拉刷新再本项目中貌似并无多大意义,由于用户每次查看本身的记录的时候,只弄了个上拉分页加载更多。注意传入当前页数出现的问题。 上拉加载更多参考代码:https://github.com/wangnaiwen/RecyclerViewRefresh

5.七、引入CheckBox时,默认选中不起做用:
  发现materialdesign是代码提示自动引入的,
输入图片说明
须要将其修改成:修改成 xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:materialdesign="http://schemas.android.com/apk/res-auto" 这是自定义属性引入有问题形成的。
参考个人另外一个小结:http://www.javashuo.com/article/p-hcgxabto-z.html

5.八、本地测试时,Natapp内外网映射工具:
  开机重启后natapp映射的本地路径就变了,天天都得修改,若是你没有关机习惯倒也无所谓。注意用node.js作服务端时须要在www文件中将默认的3000端口改为80端口。 输入图片说明

###6、最后相关网址分享: 阿里狂拽酷炫的图标库:地址:http://www.iconfont.cn/plus
**API测试工具PostMan:**下载地址: http://chromecj.com/web-development/2014-09/60/download.html
**内外网映射工具:**作服务端开发时APP需访问我本地电脑,这是能够经过免费的内外网映射,不须要经过阿里云测试(微信公众号开发也能够经过此服务 )。 地址:https://natapp.cn/
项目源码: https://github.com/dpc761218914/SignIn
Node.js服务端源码: https://github.com/dpc761218914/SignIn_Server

相关文章
相关标签/搜索