2018-08-29 android如何编写登录界面

思路,每个APP都会有一个登录界面,用来注册用户那么数据会保存在哪里呢?肯定是在服务端的数据库上,那么就需要OKHttp3去访问数据库得到我们想要的数据,这里就不演示了,用简单的sharedpreferences去存储,代替数据库的功能。废话不多说,下面开始

1.首先是XML的布局文件,我这个登录和注册时通用的,只需要改变下布局,就能实现登录,和注册界面。




    

        
    

    

        

            

                

                
            

            

            

        


        


            

                

                

            

            

            

            
        

        

            


            
        

        

代码看似很多,实际上也就是要实现这两张图效果


Screenshot_20180829-115910.png
Screenshot_20180829-115913.png

重点在于如何实现手机登录格式为3-4-4和切换界面布局 以及第三方登录和存储用户信息.
1.实现手机格式3-4-4,是目前主流的格式输入,实现思想是通过对Edtext的监听,以及去修改TextWatcher()等一系列回调函数来实现的

private void edittextlistener() {
        //账号输入框事件监听
        muserPNEdtext.addTextChangedListener(new TextWatcher() {
            //这个方法被调用,说明在S字符串中,从start位置开始的count个字符即将被长度为after的新文本
            //所取代,在这个方法里面改变s,会报错.
            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {

            }

            //这个方法被调用,说明在s字符串中,从start位置开始的count个字符取代了长度为before的旧文本,
            //在这个方法里改变s,会报错。
            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
                StringBuffer stringBuffer = new StringBuffer(s);
                if (before > 0) {
                    if (stringBuffer.length() == 4 || stringBuffer.length() == 9) {
                        stringBuffer.deleteCharAt(stringBuffer.length() - 1);
                        muserPNEdtext.setText(stringBuffer.toString());
                        muserPNEdtext.setSelection(stringBuffer.length());
                    }
                }
            }

            //这个方法被调用,说明s字符串位置已经改变
            @Override
            public void afterTextChanged(Editable s) {
                /**
                 * 判断是否显示删除图标
                 */
                if (s.toString().length() == 0) {
                    mIv_PhoneNumber_clean.setVisibility(View.INVISIBLE);
                } else {
                    mIv_PhoneNumber_clean.setVisibility(View.VISIBLE);
                }

                StringBuffer stringBuffer = new StringBuffer(s);
                if (s.length() >= 4) {
                    char[] chars = s.toString().toCharArray();
                    if (chars[3] != ' ') {
                        stringBuffer.insert(3, ' ');
                        muserPNEdtext.setText(stringBuffer.toString());
                        muserPNEdtext.setSelection(stringBuffer.length());
                    }
                }
                if (s.length() >= 9) {
                    char[] chars = s.toString().toCharArray();
                    if (chars[8] != ' ') {
                        stringBuffer.insert(8, ' ');
                        muserPNEdtext.setText(stringBuffer.toString());
                        muserPNEdtext.setSelection(stringBuffer.length());
                    }
                }

            }
        });

这段代码我写来一个方法private void edittextlistener()去实现对EditText的监听
主要方法EditText.addTextChangedListener();里面传入new TextWatcher(),会回调三个函数,三个函数的具体作用我也在代码中写明,剩下的就改写public void afterTextChanged(Editable s){}这个函数就行了,注意,使用此方法一定要将Edittext文本设置为不可编辑,因为我还是无法解决Selection的位置问题,如果不设置的话,可能在操作的时候会光标错乱。
2.切换界面布局,这个问题比较简单,首先我们要获取可以改变布局控件的点击次数然后对2取余就行了

private int mEyesType = 0;//设置点击计数器,判断当前密码状态
private int layoutChange = 0;//设置点击计数器,判断当前布局状态
private Handler mHandler = new Handler(new Handler.Callback() {
        @Override
        public boolean handleMessage(Message msg) {
            switch (msg.what) {
                case 0x01://获取输入框焦点时,让下划线背景颜色改变
                    mView_phonebg_line.setBackgroundResource(R.color.colorGoldFFC125);
                    mView_password_line.setBackgroundResource(R.color.colorGrayE8E8E8);
                    break;
                case 0x02://获取密码框焦点时,让下划线背景颜色改变
                    mView_phonebg_line.setBackgroundResource(R.color.colorGrayE8E8E8);
                    mView_password_line.setBackgroundResource(R.color.colorGoldFFC125);
                    break;
                case 0x03:
                    mEduserPhoneGetFocus();//点击手机图标也能让输入框获取到焦点
                    break;
                case 0x04:
                    mEduserPasswordGetFocus();//点击密码图标也能让输入框获取到焦点
                    break;
                case 0x05://通过点击次数判断是否显示密码,此时不显示
                    mIv_eyes_type.setImageResource(R.drawable.user_icon_eye_close_blue);
                    muserPAPEdtext.setTransformationMethod(PasswordTransformationMethod.getInstance());
                    break;
                case 0x06://通过点击次数判断是否显示密码,此时显示
                    mIv_eyes_type.setImageResource(R.drawable.user_icon_eye_open_blue);
                    muserPAPEdtext.setTransformationMethod(HideReturnsTransformationMethod.getInstance());
                    break;
                case 0x07://通过点击次数判断是否改变格局,此时不改变
                    muserpasswordRelayout.setVisibility(View.VISIBLE);
                    mTv_Choose_type.setText(R.string.registered_msg2);
                    mLogin_btn.setText(R.string.btn_registered_msg2);
                    mLogin_btn.setBackground(getResources().getDrawable(R.drawable.shape_btn_light_yellow_bg));
                    break;
                case 0x08://通过点击次数判断是否改变格局,此时改变
                    muserpasswordRelayout.setVisibility(View.GONE);
                    mTv_Choose_type.setText(R.string.registered_msg1);
                    mLogin_btn.setText(R.string.btn_registered_msg1);
                    mLogin_btn.setBackground(getResources().getDrawable(R.drawable.shape_btn_khakil_bg));
                    break;
                default:
                    break;
            }

            return false;
        }
    });


   /**
     * 通过点击判断布局格式
     */
    private void LayoutChange() {
        if (layoutChange % 2 == 0) {
            Message message = mHandler.obtainMessage(0x07);
            mHandler.sendMessage(message);
        } else {
            Message message = mHandler.obtainMessage(0x08);
            mHandler.sendMessage(message);
        }
        layoutChange++;
    }

这段代码首先获取layoutChange的被点击次数,然后对2取余判断当前应该显示界面的状态,然后将变更状态交由handler去异步处理,UI的操作我是尽量能用异步就用异步,这样不会给主线程带来卡顿.实现效果就是点击 改变界面状态,上面图片已经显示了,我就不再举例了.
3.第三方登录
转载https://www.jianshu.com/p/133d84042483 大家可以先看看第三方登录的介绍
4.储存用户信息

       //储存用户信息
       pref = PreferenceManager.getDefaultSharedPreferences(mContext);
       editor = pref.edit();
       editor.putString("phoneNumber", "187 **** 5216");
       editor.putString("password", "123456");
       editor.apply();
      //取出用户信息
       userPhoneNumber = pref.getString("phoneNumber", null);
       PhonePassword = pref.getString("password", null);
       muserPNEdtext.setText(userPhoneNumber);
       muserPNEdtext.setSelection(userPhoneNumber.length());

这里是使用SharedPreference去储存了一个用户手机号和输入密码,当然,在实际操作中不可能这样去操作,实际操作首先要验证用户手机号码是否存在,如果存在,只有登录成功后,才能去存储用户的手机号码,而且,密码是不存储的,为了用户的信息安全。
如果要了解更多SharedPreferencene内容可以看https://www.jianshu.com/p/ca3a8476edd4这篇文章
经过上面的代码,这个登录界面基本上就算成型了,接下来是业务逻辑内容。假设用户输入了手机号码,可是,我们怎么能判断是手机号码,而且,我们如何判断用户是否是正确输入手机号码,用户输入手机号码是正确之后,我们还需要对应当前界面进行不同的逻辑操作,下面,让我们跟随代码一起来理清这些逻辑。

public class PhoneUtil {
    //判断是否符合电话号码

    /**
     * 判断电话号码是否符合格式.
     *
     * @param inputText the input text
     * @return true, if is phone
     */
    public static boolean isPhone(String inputText) {
        Pattern p = Pattern.compile("^((14[0-9])|(13[0-9])|(15[0-9])|(18[0-9])|(17[0-9]))\\d{8}$");
        Matcher m = p.matcher(inputText);
        return m.matches();
    }
}

这里给大家介绍一个工具类,用来判断是否是手机号码,具体为什么能实现,我也不是很清楚。


 case 0x07://通过点击次数判断是否改变格局,此时不改变
         muserpasswordRelayout.setVisibility(View.VISIBLE);
         isViewRegisteredType = true;//是登录界面
         mTv_Choose_type.setText(R.string.registered_msg2);
         mLogin_btn.setText(R.string.btn_registered_msg2);
         mLogin_btn.setBackground(getResources().getDrawable(R.drawable.shape_btn_light_yellow_bg));
                    break;
case 0x08://通过点击次数判断是否改变格局,此时改变
         muserpasswordRelayout.setVisibility(View.GONE);
         isViewRegisteredType = false;//是注册界面
         mTv_Choose_type.setText(R.string.registered_msg1);
         mLogin_btn.setText(R.string.btn_registered_msg1);
                 mLogin_btn.setBackground(getResources().getDrawable(R.drawable.shape_btn_khakil_bg));
                    break;

这段代码,我使用了布尔值isViewRegisteredType作为我判断当前界面的标准,当然,这段代码这是根据点击事件来控制的,如果没有点击事件的话,isViewRegisteredType就必须有一个默认的布尔值,至于是默认值是是什么,就根据当前你进入的页面去决定,这个也不难

 private void checklogin() {//登录时对手机号码的判断
        String number = muserPNEdtext.getText().toString();
        String phonePassword = muserPAPEdtext.getText().toString();
        String phonenumber = number.replace(" ", "");
        String password = muserPAPEdtext.toString();
        Toast toast = ToastUtil.setMyToast(mContext, ToastUtil.PROMPT,
                "手机号码格式不对", Toast.LENGTH_SHORT);
        Toast toast1 = ToastUtil.setMyToast(mContext, ToastUtil.WARING,
                "这不是手机号码", Toast.LENGTH_SHORT);
        Toast toast2 = ToastUtil.setMyToast(mContext, ToastUtil.ERROR,
                "请输入密码,长度在6-20之间", Toast.LENGTH_SHORT);
        Toast toast3 = ToastUtil.setMyToast(mContext, ToastUtil.PROMPT,
                "用户不存在", Toast.LENGTH_SHORT);

        if (isViewRegisteredType) {//判断是什么界面,true为登录界面,false为注册界面
            if (isUserPhoneCheck) {//判断手机号是否是13位
                if (PhoneUtil.isPhone(phonenumber)) {//判断手机号是否是手机号
                    if (isPasswordinit) {//判断密码是否输入且位数是否在6-20之间
                        if (number.equals(userPhoneNumber)) {//判断账户是否存在
                            if (phonePassword.equals(PhonePassword)) {
                                Toast.makeText(mContext, "成功", Toast.LENGTH_SHORT).show();
                            } else {
                                aReturn.onCheckResultReturn();
                            }
                        } else {
                            toast3.show();
                        }
                    } else {
                        toast2.show();
                    }
                } else {
                    toast1.show();
                }
            } else {
                toast.show();
            }

        } else {
            if (isUserPhoneCheck) {//判断手机号是否有13位
                if (PhoneUtil.isPhone(phonenumber)) {
                    if (number.equals(userPhoneNumber)) {//判断账户是否存在
                        ToastUtil.setToastNormal(mContext, "下一步", Toast.LENGTH_SHORT);
                    } else {
                        toast3.show();
                    }
                } else {
                    toast1.show();
                }
            } else {
                toast.show();
            }
        }

    }

这样当我们获取到了当前界面状态时候,就能进一步判断用户的输入状态,进而可以判断密码是否正确,这里又有一个问题,当用户密码和账户完全正确时,我们肯定要跳转Activity,但是我这里是用的Fragment,那么Fragment如何去跳转Activity呢?我们知道,普通的跳转是用Intent,这里,推荐大家去看这篇文章https://blog.csdn.net/androidkwd/article/details/52670520
因为我是用Fragment写的,所以会有点麻烦,大体思路是用接口实现回调方法,Fragment具体要实现的方法,可以回调给Fragment所在的Activity里面去执行,这样,可以很完美的实现Activity和Fragment之间的连接.

这样,我们的登录界面就算是完成了,感谢大家的阅读。

你可能感兴趣的:(2018-08-29 android如何编写登录界面)