热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android仿支付宝自定义密码输入框及安全键盘(密码键盘)

这篇文章主要介绍了Android仿支付宝自定义密码输入框及安全键盘(密码键盘),需要的朋友可以参考下

 0、前言

 之前做过的项目里有运用到一个支付场景:用户办理业务时需要输入交易密码,并且可根据平台下发的支付方式进行选择。这与支付宝的密码输入方式十分相似,如果使用Android系统或者第三方软件的键盘,会有密码泄露的风险。因此,大多数的应用软件使用的是自定义的密码输入框及安全键盘。

 由于密码输入方式需要实现一个从底部弹出的效果,因此总体上决定采用BottomSheetDialog来进行封装,同时为了提高安全性,还应该随机生成键盘上的数字,界面如下图所示:

  

 首先新建一个PasswordInputView类,将需要使用到的Context对象、支付金额、可支持的支付方式等数据,作为该类构造方法的参数进行传递。下文还将提到该类有一个回调方法,当用户输入的密码满足六位时,可以在回调方法中获取密码并显示出来。PasswordInputView类的构造方法如下所示:

public PasswordInputView(Context context, String payMoney, List payWayList) { 
 this.cOntext= context; 
 this.payMOney= payMoney; 
 this.payWayList = payWayList; 
 payPwdDialog = new BottomSheetDialog(context); 
 View view = LayoutInflater.from(context).inflate(R.layout.dialog_pay_pwd, null, false); 
 initStep1(view); 
 initStep2(view); 
 llyPwdInputView = (LinearLayout) view.findViewById(R.id.lly_pwd_input_view); 
 llyPayWaySelect = (LinearLayout) view.findViewById(R.id.lly_pay_way_select); 
 showStep1(); // 显示第一页 
} 

1、自定义密码输入框

 因为不能明文显示输入的密码,所以使用“●”来代替每位密码。自定义密码输入框涉及到的自定义属性,主要包括:输入框的大小、颜色、圆角半径以及密码圆点的大小、颜色、半径。因此,自定义属性attrs.xml文件如下所示:

<&#63;xml version="1.0" encoding="utf-8"&#63;> 
 
  
   
   
   
   
   
   
   
  
 

 接下来就需要去绘制自定义控件了。首先获取自定义属性,然后在onDraw()中进行绘制,代码如下所示:

package com.syd.paypwddialogdemo; 
import static android.graphics.Paint.ANTI_ALIAS_FLAG; 
import android.content.Context; 
import android.content.res.Resources; 
import android.content.res.TypedArray; 
import android.graphics.Canvas; 
import android.graphics.Color; 
import android.graphics.Paint; 
import android.graphics.RectF; 
import android.support.v7.widget.AppCompatEditText; 
import android.util.AttributeSet; 
/** 
 * 自定义密码输入框 
 */ 
public class PasswordEditText extends AppCompatEditText { 
 private int textLength; 
 private int borderColor; 
 private float borderWidth; 
 private float borderRadius; 
 private int passwordLength; 
 private int passwordColor; 
 private float passwordWidth; 
 private float passwordRadius; 
 private Paint passwordPaint = new Paint(ANTI_ALIAS_FLAG); 
 private Paint borderPaint = new Paint(ANTI_ALIAS_FLAG); 
 private final int defaultCOntMargin= 5; 
 private final int defaultSplitLineWidth = 3; 
 public PasswordEditText(Context context, AttributeSet attrs) { 
  super(context, attrs); 
  final Resources res = getResources(); 
  final int defaultBorderColor = res.getColor(R.color.colorGray); 
  final float defaultBorderWidth = res.getDimension(R.dimen.default_ev_border_width); 
  final float defaultBorderRadius = res.getDimension(R.dimen.default_ev_border_radius); 
  final int defaultPasswordLength = res.getInteger(R.integer.default_ev_password_length); 
  final int defaultPasswordColor = res.getColor(R.color.colorBlack); 
  final float defaultPasswordWidth = res.getDimension(R.dimen.default_ev_password_width); 
  final float defaultPasswordRadius = res.getDimension(R.dimen.default_ev_password_radius); 
  TypedArray a = context.getTheme().obtainStyledAttributes(attrs, R.styleable.PasswordEditText, 0, 0); 
  try { 
   borderColor = a.getColor(R.styleable.PasswordEditText_borderColor, defaultBorderColor); 
   borderWidth = a.getDimension(R.styleable.PasswordEditText_borderWidth, defaultBorderWidth); 
   borderRadius = a.getDimension(R.styleable.PasswordEditText_borderRadius, defaultBorderRadius); 
   passwordLength = a.getInt(R.styleable.PasswordEditText_passwordLength, defaultPasswordLength); 
   passwordColor = a.getColor(R.styleable.PasswordEditText_passwordColor, defaultPasswordColor); 
   passwordWidth = a.getDimension(R.styleable.PasswordEditText_passwordWidth, defaultPasswordWidth); 
   passwordRadius = a.getDimension(R.styleable.PasswordEditText_passwordRadius, defaultPasswordRadius); 
  } finally { 
   a.recycle(); 
  } 
  borderPaint.setStrokeWidth(borderWidth); 
  borderPaint.setColor(borderColor); 
  passwordPaint.setStrokeWidth(passwordWidth); 
  passwordPaint.setStyle(Paint.Style.FILL); 
  passwordPaint.setColor(passwordColor); 
 } 
 @Override 
 protected void onDraw(Canvas canvas) { 
  int width = getWidth(); 
  int height = getHeight(); 
  RectF rect = new RectF(0, 0, width, height); 
  borderPaint.setColor(borderColor); 
  canvas.drawRoundRect(rect, borderRadius, borderRadius, borderPaint); 
  RectF rectIn = new RectF(rect.left + defaultContMargin, rect.top + defaultContMargin, 
    rect.right - defaultContMargin, rect.bottom - defaultContMargin); 
  borderPaint.setColor(Color.WHITE); 
  canvas.drawRoundRect(rectIn, borderRadius, borderRadius, borderPaint); 
  borderPaint.setColor(borderColor); 
  borderPaint.setStrokeWidth(defaultSplitLineWidth); 
  for (int i = 1; i 

2、安全键盘的实现

 安全键盘主要是通过GridView来实现,上文提到为了保证安全性,在安全键盘初始化的时候,应该随机生成键盘上的数字,代码如下所示:

/** 
 * 初始化密码键盘 
 */ 
private void initKeyboard() { 
 final int number = 10; 
 int[] keys = new int[number]; 
 for (int i = 0; i <10; i++) { 
  keys[i] = i; 
 } 
 // 随机生成键盘数字 
 Random random = new Random(); 
 for (int i = 0; i (); 
 for (int i = 0; i <12; i++) { 
  Map map = new HashMap<>(); 
  if (i <9) { 
   map.put("num", String.valueOf(keys[i])); 
  } else if (i == 9) { 
   map.put("num", ""); 
  } else if (i == 10) { 
   map.put("num", String.valueOf(keys[9])); 
  } else if (i == 11) { 
   map.put("num", ""); 
  } 
  numList.add(map); 
 } 
 KeyAdapter keyAdapter = new KeyAdapter(context, numList, handler); 
 gvKeyboard.setAdapter(keyAdapter); 
} 

 安全键盘点击事件的处理,是在适配器KeyAdapter的构造方法中传入Handler对象,通过收发消息的方式在PasswordInputView类中处理的,代码如下所示:

holder.btnKey.setOnClickListener(new View.OnClickListener() { 
 @Override 
 public void onClick(View v) { 
  Message msg = new Message(); 
  msg.what = Constants.KEYBOARD_INPUT; 
  msg.obj = position; 
  handler.sendMessage(msg); 
 } 
}); 

 Handler对象在PasswordInputView类中定义,主要用于处理安全键盘的点击事件,代码如下所示:

Handler handler = new Handler() { 
 @Override 
 public void dispatchMessage(Message msg) { 
  switch (msg.what) { 
   case Constants.KEYBOARD_INPUT: 
    int position = (int) msg.obj; 
    if (position <11 && position != 9) { 
     // 点击0-9按键 
     password = etPwd.getText().append(numList.get(position).get("num")).toString(); 
     etPwd.setText(password); 
    } else { 
     if (position == 11) { 
      // 点击退格键 
      if (!TextUtils.isEmpty(password) && !password.equals("")) { 
       password = etPwd.getText().delete(password.length() - 1, password.length()).toString(); 
       etPwd.setText(password); 
      } 
     } 
    } 
    break; 
  } 
 } 
}; 

 为了方便外部获取到用户输入的密码,设计一个回调接口OnPwdInputListener,并在PasswordInputView类中为回调接口创建一个set方法,代码如下所示:

package com.syd.paypwddialogdemo; 
public interface OnPwdInputListener { 
 void onPwdInput(String password); 
} 

 当PasswordEditText控件的TextWatcher对象监听到输入的密码满足六位时,调用回调方法,将密码作为参数进行传递,代码如下所示:

textWatcher = new TextWatcher() { 
 @Override 
 public void afterTextChanged(Editable s) { 
  if (etPwd.getText().length() == 6) { 
   onPwdInputListener.onPwdInput(etPwd.getText().toString()); 
  } 
 } 
}; 
etPwd.addTextChangedListener(textWatcher); 

在外部调用set方法,创建OnPwdInputListener对象,重写回调方法,即可获取到用户输入的密码,代码如下所示:

pwdInputView.setOnPwdInputListener(new OnPwdInputListener() { 
 @Override 
 public void onPwdInput(String password) { 
  Toast.makeText(MainActivity.this, password, Toast.LENGTH_SHORT).show(); 
 } 
}); 

3、结语

 以上介绍了自定义密码输入框及安全键盘的大致实现思路,对源码感兴趣的小伙伴可以点击下载Demo,查看具体的实现过程及演示效果。


推荐阅读
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 本文讲述了如何通过代码在Android中更改Recycler视图项的背景颜色。通过在onBindViewHolder方法中设置条件判断,可以实现根据条件改变背景颜色的效果。同时,还介绍了如何修改底部边框颜色以及提供了RecyclerView Fragment layout.xml和项目布局文件的示例代码。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • Android系统移植与调试之如何修改Android设备状态条上音量加减键在横竖屏切换的时候的显示于隐藏
    本文介绍了如何修改Android设备状态条上音量加减键在横竖屏切换时的显示与隐藏。通过修改系统文件system_bar.xml实现了该功能,并分享了解决思路和经验。 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • 本文讨论了在Spring 3.1中,数据源未能自动连接到@Configuration类的错误原因,并提供了解决方法。作者发现了错误的原因,并在代码中手动定义了PersistenceAnnotationBeanPostProcessor。作者删除了该定义后,问题得到解决。此外,作者还指出了默认的PersistenceAnnotationBeanPostProcessor的注册方式,并提供了自定义该bean定义的方法。 ... [详细]
  • 如何在服务器主机上实现文件共享的方法和工具
    本文介绍了在服务器主机上实现文件共享的方法和工具,包括Linux主机和Windows主机的文件传输方式,Web运维和FTP/SFTP客户端运维两种方式,以及使用WinSCP工具将文件上传至Linux云服务器的操作方法。此外,还介绍了在迁移过程中需要安装迁移Agent并输入目的端服务器所在华为云的AK/SK,以及主机迁移服务会收集的源端服务器信息。 ... [详细]
  • 本文介绍了如何使用python从列表中删除所有的零,并将结果以列表形式输出,同时提供了示例格式。 ... [详细]
  • Linux环境变量函数getenv、putenv、setenv和unsetenv详解
    本文详细解释了Linux中的环境变量函数getenv、putenv、setenv和unsetenv的用法和功能。通过使用这些函数,可以获取、设置和删除环境变量的值。同时给出了相应的函数原型、参数说明和返回值。通过示例代码演示了如何使用getenv函数获取环境变量的值,并打印出来。 ... [详细]
  • 本文介绍了南邮ctf-web的writeup,包括签到题和md5 collision。在CTF比赛和渗透测试中,可以通过查看源代码、代码注释、页面隐藏元素、超链接和HTTP响应头部来寻找flag或提示信息。利用PHP弱类型,可以发现md5('QNKCDZO')='0e830400451993494058024219903391'和md5('240610708')='0e462097431906509019562988736854'。 ... [详细]
author-avatar
zoey
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有