android-手势密码

引子

手势密码,移动开发中的常用功能点,看起来高大上,其实挺简单的。

本文提供 我自定义的 手势密码控件布局,以及使用方法,首先附上github地址:https://github.com/18598925736/EazyGesturePwdLayoutDemo

 

实际效果动态图

 

设置手势密码:

 

设置手势密码,当 前后两次的手势不一样时

 

 

校验手势密码-当5次都错时:

 

校验手势密码-当5次之内输入正确时

 

重新设置手势(之前设置过,现在需要修改手势密码)

 

源码解析

首先说下开发思路:

上面的图里面,我们主要看到了9个圆点,以及随着手势而产生的线条;

9个圆点,其实就是 自定义的View,如果你运行demo,把手放上去的画,你会发现原点会出现圆环背景,这是在自定义的时候加上的功能,至于圆环的颜色宽度神马的,你开心的话自己就行了。

至于线条,其实是 通过在一个自定义ViewGroup上重写onToucheEvent监测 down,move和up来绘制的,9个圆点是被放置(用的 addView)在这个自定义ViewGroup里面,排布的方式看看源码应该能明白;

特别说明一下这里有个坑:

在绘制线条的时候,我发现 我绘制出来的线条总是被9个圆点覆盖,经过多方查询,最终得出结论:这是ViewGroup的绘制机制导致的,它默认的绘制顺序,是先绘制 background,然后是自己,然后是子,最后是装饰;

看起来很抽象是吧?

看源码;

最下方这个英语翻译过来,就是我刚才说的意思,由于后绘制的会覆盖先绘制的,所以,线条被子覆盖也是正常的。

但是,这不是我想要的效果,问题是不是无解了呢?

也不是,只是大路不通,要走小路了;

还是看 View.java源码:

 

发现,在绘制的第四步,DrawChildren中,调用的方法是dispatchDraw(canvas); 

那我如果在绘制子之后,再画线,是不是可以让线条覆盖子。

所以,我重写了这个方法,执行super.dispatchDraw()先保持原有逻辑,并且在执行我自己的绘制来画线;

 

 OK,坑 解释完毕。

自定义控件的源码:

业内人士应该没有什么看不懂的,毕竟我这个注释已经是详细得令人发指了(●´∀`●)....

 

首先是那9个圆点:

  1 package com.example.gesture_password_study.gesture_pwd.custom;
  2 
  3 import android.content.Context;
  4 import android.content.res.TypedArray;
  5 import android.graphics.Canvas;
  6 import android.graphics.Paint;
  7 import android.support.annotation.Nullable;
  8 import android.util.AttributeSet;
  9 import android.view.View;
 10 
 11 import com.example.gesture_password_study.R;
 12 
 13 
 14 /**
 15  * 手势密码专用的圆形控件
 16  */
 17 public class GestureLockCircleView extends View {
 18 
 19     public GestureLockCircleView(Context context) {
 20         this(context, null);
 21     }
 22 
 23     public GestureLockCircleView(Context context, @Nullable AttributeSet attrs) {
 24         this(context, attrs, 0);
 25     }
 26 
 27     public GestureLockCircleView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
 28         super(context, attrs, defStyleAttr);
 29         dealAttr(context, attrs);
 30         initPaint();
 31     }
 32 
 33     private void dealAttr(Context context, AttributeSet attrs) {
 34         TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.GestureLockCircleView);
 35 
 36         if (ta != null) {
 37             try {
 38                 circleFillColor = ta.getColor(R.styleable.GestureLockCircleView_gestureCircleFillColor, 0x00FE6665);
 39                 circleRadius = ta.getDimension(R.styleable.GestureLockCircleView_gestureCircleRadius, 0);
 40 
 41                 hasRoundBorder = ta.getBoolean(R.styleable.GestureLockCircleView_hasRoundBorder, false);
 42                 roundBorderColor = ta.getColor(R.styleable.GestureLockCircleView_roundBorderColor, 0x00FE6665);
 43                 roundBorderWidth = ta.getDimension(R.styleable.GestureLockCircleView_roundBorderWidth, 0);
 44             } catch (Exception e) {
 45 
 46             } finally {
 47                 ta.recycle();
 48             }
 49         }
 50     }
 51 
 52 
 53     private int minWidth = 50, minHeight = 50;
 54 
 55     /**
 56      * 重写onMeasure设定最小宽高
 57      *
 58      * @param widthMeasureSpec
 59      * @param heightMeasureSpec
 60      */
 61     @Override
 62     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
 63         setMeasuredDimension(widthMeasureSpec, heightMeasureSpec);
 64         int widthMode = MeasureSpec.getMode(widthMeasureSpec);
 65         int widthSize = MeasureSpec.getSize(widthMeasureSpec);
 66         int heightMode = MeasureSpec.getMode(heightMeasureSpec);
 67         int heightSize = MeasureSpec.getSize(heightMeasureSpec);
 68 
 69         if (widthMode == MeasureSpec.AT_MOST && heightMode == MeasureSpec.AT_MOST) {
 70             setMeasuredDimension(minWidth, minHeight);
 71         } else if (widthMode == MeasureSpec.AT_MOST) {
 72             setMeasuredDimension(minWidth, heightSize);
 73         } else if (heightMode == MeasureSpec.AT_MOST) {
 74             setMeasuredDimension(widthSize, minHeight);
 75         }
 76     }
 77 
 78     @Override
 79     protected void onDraw(Canvas canvas) {
 80         super.onDraw(canvas);
 81         int width = getWidth();
 82         int height = getHeight();
 83 
 84         float centerX = width / 2;
 85         float centerY = height / 2;
 86 
 87         if (hasRoundBorder) {
 88             canvas.drawCircle(centerX, centerY, roundBorderWidth, paint_border);
 89         }
 90         canvas.drawCircle(centerX, centerY, circleRadius, paint_inner);
 91 
 92     }
 93 
 94     private Paint paint_inner, paint_border;
 95 
 96 
 97     private boolean hasRoundBorder;
 98     private int roundBorderColor;
 99     private float roundBorderWidth;
100 
101     /**
102      * 设置内圈的颜色和半径
103      *
104      * @param circleFillColor
105      * @param circleRadius
106      */
107     public void setInnerCircle(int circleFillColor, float circleRadius) {
108         this.circleFillColor = circleFillColor;
109         this.circleRadius = circleRadius;
110         initPaint();
111         postInvalidate();
112     }
113 
114     public void setBorderRound(boolean hasRoundBorder, int roundBorderColor, float roundBorderWidth) {
115         this.hasRoundBorder = hasRoundBorder;
116         this.roundBorderColor = roundBorderColor;
117         this.roundBorderWidth = roundBorderWidth;
118         initPaint();
119         postInvalidate();
120     }
121 
122 
123     private int circleFillColor;
124     private float circleRadius;
125 
126     private void initPaint() {
127         paint_inner = new Paint();
128         paint_inner.setColor(circleFillColor);
129         paint_inner.setAntiAlias(true);//抗锯齿
130         paint_inner.setStyle(Paint.Style.FILL);//FILL填充,stroke描边
131 
132         paint_border = new Paint();
133         paint_border.setColor(roundBorderColor);
134         paint_border.setAntiAlias(true);//抗锯齿
135         paint_border.setStyle(Paint.Style.FILL);//FILL填充,stroke描边
136     }
137 
138     //3个状态
139     public static final int STATUS_NOT_CHECKED = 0x01;
140     public static final int STATUS_CHECKED = 0x02;
141     public static final int STATUS_CHECKED_ERR = 0x03;
142 
143     public void switchStatus(int status) {
144         switch (status) {
145             case STATUS_CHECKED:
146                 circleFillColor = getResources().getColor(R.color.colorChecked);
147                 roundBorderColor = getResources().getColor(R.color.colorRoundBorder);
148                 break;
149             case STATUS_CHECKED_ERR:
150                 circleFillColor = getResources().getColor(R.color.colorCheckedErr);
151                 roundBorderColor = getResources().getColor(R.color.colorRoundBorderErr);
152                 break;
153             case STATUS_NOT_CHECKED:// 普通状态
154             default://以及缺省状态
155                 //没有外框,内圈为灰色
156                 circleFillColor = getResources().getColor(R.color.colorNotChecked);
157                 roundBorderColor = getResources().getColor(R.color.transparent);
158                 break;
159         }
160         initPaint();
161         postInvalidate();
162     }
163 
164 }

 

然后是外层的布局:

  1 package com.example.gesture_password_study.gesture_pwd.custom;
  2 
  3 import android.content.Context;
  4 import android.content.res.TypedArray;
  5 import android.graphics.Canvas;
  6 import android.graphics.Paint;
  7 import android.graphics.Path;
  8 import android.graphics.Point;
  9 import android.graphics.Rect;
 10 import android.util.AttributeSet;
 11 import android.util.Log;
 12 import android.view.MotionEvent;
 13 import android.view.View;
 14 import android.widget.RelativeLayout;
 15 
 16 import com.example.gesture_password_study.R;
 17 
 18 import java.util.ArrayList;
 19 import java.util.List;
 20 
 21 /**
 22  * 手势密码绘制 控件;
 23  */
 24 public class EasyGestureLockLayout extends RelativeLayout {
 25 
 26     //全局变量统一管理
 27     private Context mContext;
 28     private boolean hasRoundBorder;//按键是否允许有圆环外圈
 29     private boolean ifAllowInteract;//是否允许有事件交互
 30     private Paint currentPaint;//当前使用的画笔
 31     private Paint paint_correct, paint_error;//画线用的两种颜色的画笔
 32     private GestureLockCircleView[] gestureCircleViewArr = null;//用数组来保存所有按键
 33     private int mCount = 4;// 方阵的行数(列数等同)
 34     private int mGesturePasswordViewWidth;//每一个按键的边长(因为宽高相同)
 35     private int mWidth, mHeight;//本layout的宽高
 36     private int childStartIndex, childEndIndex;//画轨迹线(密码轨迹)的时候,需要指定子的起始和结束 index
 37     private float marginRate = 0.2f;//缩小MotionEvent到达时的密码键选中的判定范围,这里的0.2的意思是,原本10*10的判定范围,现在,缩小到6*6,其他4,被两头平分
 38     private boolean ifAllowDrawLockPath = false;//因为有可能存在,down的时候没有点在任何一个键位的范围之内,所以必须用这个变量来控制是否进行绘制
 39     private int guideLineStartX, guideLineStartY, guideLineEndX, guideLineEndY;//引导线(正在画手势,但是尚未或者无法形成轨迹线的时候,会出现)的起始和终止坐标
 40     private int downX, downY;//MotionEvent的down事件坐标
 41     private int movedX, movedY;//MotionEvent的move事件坐标
 42     private Path lockPath = new Path();//密码的图形路径.用于绘制轨迹线
 43     private List<Integer> lockPathArr;//手势密码路径,用于输出到外界以及核对密码
 44     private int minLengthOfPwd = 4;//密码最少位数
 45 
 46     private int mModeStatus = -1;
 47     private List<Integer> checkPwd;//外界传入的需要核对的密码
 48     private int maxAttemptTimes = 5;//允许解锁的最大尝试次数,有必要的话,给他设置一个set方法,或者弄一个自定义属性
 49     private int currentAttemptTime = 1;// 当前尝试次数
 50 
 51     private int resetCurrentTime = 0;//当用户重新设置密码,这个值将会被重置
 52     private List<Integer> tempPwd;//用于重新设置密码
 53     private boolean ifCheckOnErr = false;//当前是否检测密码曾失败过
 54 
 55     //常量
 56     public static final int STATUS_RESET = 0x01;//本类状态:重新设置,此状态下会允许用户绘制两次手势,而且必须相同,绘制完成之后,返回密码值出去;
 57     // 如果第二次绘制和第一次绘制不同,则强制重新绘制
 58     public static final int STATUS_CHECK = 0x02;//本类状态:校验密码,此状态下,要求外界传入密码,然后给予用户若干尝试解锁的次数,
 59     // 如果规定次数之内,密码相同,则返回解锁成功;
 60     // 如果规定次数之内,都没有绘制出正确密码,则返回解锁失败;
 61 
 62     //************* 构造函数 *****************************
 63     public EasyGestureLockLayout(Context context) {
 64         this(context, null);
 65     }
 66 
 67     public EasyGestureLockLayout(Context context, AttributeSet attrs) {
 68         this(context, attrs, 0);
 69     }
 70 
 71     public EasyGestureLockLayout(Context context, AttributeSet attrs, int defStyleAttr) {
 72         super(context, attrs, defStyleAttr);
 73         dealAttr(context, attrs);
 74         init(context);
 75     }
 76 
 77     //************* 属性值获取 *****************************
 78     private void dealAttr(Context context, AttributeSet attrs) {
 79         TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.EasyGestureLockLayout);
 80 
 81         if (ta != null) {
 82             try {
 83                 hasRoundBorder = ta.getBoolean(R.styleable.EasyGestureLockLayout_ifChildHasBorder, false);
 84                 mCount = ta.getInteger(R.styleable.EasyGestureLockLayout_count, 3);
 85 
 86                 ifAllowInteract = ta.getBoolean(R.styleable.EasyGestureLockLayout_ifAllowInteract, false);
 87             } catch (Exception e) {
 88 
 89             } finally {
 90                 ta.recycle();
 91             }
 92         }
 93     }
 94 
 95     //************* 重写方法 *****************************
 96     @Override
 97     protected void onMeasure(int widthSpec, int heightSpec) {
 98         super.onMeasure(widthSpec, heightSpec);
 99 
100         //取测量之后的宽和高
101         mWidth = MeasureSpec.getSize(widthSpec);
102         mHeight = MeasureSpec.getSize(heightSpec);
103         //强行将绘图使用的宽高置为  测量宽高中的较小值, 因为绘图不能超出边界
104         mHeight = mWidth = mWidth < mHeight ? mWidth : mHeight;
105 
106         // 初始化mGestureLockViews
107         if (gestureCircleViewArr == null) {
108             gestureCircleViewArr = new GestureLockCircleView[mCount * mCount];//用数组来保存 “按键”
109             mGesturePasswordViewWidth = mWidth / mCount;//等分,不需要留间隙, 因为圆形控件会自己留空隙
110 
111             //利用相对布局的参数来放置子元素
112             for (int i = 0; i < gestureCircleViewArr.length; i++) {
113                 //初始化每个GestureLockView
114                 gestureCircleViewArr[i] = getCircleView(mHeight);
115                 gestureCircleViewArr[i].setId(i + 1);
116                 LayoutParams lockerParams = new LayoutParams(
117                         mGesturePasswordViewWidth, mGesturePasswordViewWidth);
118 
119                 // 不是每行的第一个,则设置位置为前一个的右边
120                 if (i % mCount != 0) {
121                     lockerParams.addRule(RelativeLayout.RIGHT_OF,
122                             gestureCircleViewArr[i - 1].getId());
123                 }
124                 // 从第二行开始,设置为上一行同一位置View的下面
125                 if (i > mCount - 1) {
126                     lockerParams.addRule(RelativeLayout.BELOW,
127                             gestureCircleViewArr[i - mCount].getId());
128                 }
129                 lockerParams.setMargins(0, 0, 0, 0);
130                 addView(gestureCircleViewArr[i], lockerParams);
131             }
132         }
133 
134     }
135 
136     /**
137      * 实验结果,在这里onDraw,绘制出来的线,总是会被子元素覆盖,
138      *
139      * @param canvas
140      */
141     @Override
142     protected void onDraw(Canvas canvas) { //闹半天,这个onDraw没有执行
143         super.onDraw(canvas);
144         //奇怪,为何不执行onDraw
145         // 一般情况下,viewGroup都不会执行onDraw,因为它本身是一个容器,容器不具有自我绘制功能;
146         //图像的表现,和绘制的顺序有关系;
147         Log.d("onDrawTag", "onDraw");
148     }
149 
150     /**
151      * 然而,由这个方法进行绘制,线,则会覆盖"子";
152      *
153      * @param canvas
154      */
155     @Override
156     public void dispatchDraw(Canvas canvas) {
157         super.dispatchDraw(canvas);//这一步居然就是绘制 “子”, 具体看View.java 的 19195行
158         Log.d("onDrawTag", "dispatchDraw");//那么, 等children画完了之后,再画线,就名正言顺了。⊙︿⊙ 一头包。明白了
159         if (gestureCircleViewArr != null && ifAllowInteract) {
160             drawLockPath(canvas);
161             drawMovingPath(canvas);
162         }
163     }
164 
165     //************* 模式设置 *****************************
166 
167     public int getCurrentMode() {
168         return mModeStatus;
169     }
170 
171     /**
172      * 切换到Reset模式,重新设置手势密码;
173      * 此模式下,不需要入参。设置完成之后,会执行回调GestureEventCallback.onResetFinish(pwd);
174      */
175     public void switchToResetMode() {
176         mModeStatus = STATUS_RESET;
177     }
178 
179     /**
180      * 切换到 校验模式;
181      * 这个模式需要传入原始密码,以及最大尝试的次数;
182      * <p>
183      * 尝试解锁成功,或者超过了最大尝试次数都没有成功,就会执行回调GestureEventCallback.onCheckFinish(boolean succeedOrFailed);
184      *
185      * @param pwd
186      * @param maxAttemptTimes
187      */
188     public void switchToCheckMode(List<Integer> pwd, int maxAttemptTimes) {
189         if (pwd == null || maxAttemptTimes <= 0) {
190             Log.e("switchToCheckMode", "参数错误,pwd不能为空,而且 maxAttemptTimes必须大于0");
191             return;
192         }
193         this.currentAttemptTime = 1;
194         this.mModeStatus = STATUS_CHECK;
195         this.maxAttemptTimes = maxAttemptTimes;
196         this.checkPwd = copyPwd(pwd);
197     }
198 
199     //****************************以下全是业务代码**************************
200     private int background_color = 0xff4790FF;
201     private int background_color_transparent = 0x00000000;
202 
203     /**
204      * 初始化画笔,
205      *
206      * @param context
207      */
208     private void init(Context context) {
209         mContext = context;
210         setClickable(true);//为了顺利接收事件,需要开启click;因为你如果不设置,,就只能收到down,其他的一概收不到
211         setBackgroundColor(background_color_transparent);//设置透明色;这里如果不设置,onDraw将不会执行;原因:这是一个ViewGroup,本身是容器,不具备自我绘制功能,但是这里设置了背景色,就说明有东西需要绘制,onDraw就会执行;
212 
213         paint_correct = new Paint();
214         paint_correct.setStyle(Paint.Style.STROKE);
215         paint_correct.setAntiAlias(true);
216         paint_correct.setColor(getResources().getColor(R.color.colorChecked));
217 
218         paint_error = new Paint();
219         paint_error.setStyle(Paint.Style.STROKE);
220         paint_error.setAntiAlias(true);
221         paint_error.setColor(getResources().getColor(R.color.colorCheckedErr));
222 
223         initLockPathArr();
224         currentPaint = paint_correct;// 默认使用的画笔
225     }
226 
227     /**
228      * 构建单个圆
229      *
230      * @param wh 边长
231      * @return
232      */
233     private GestureLockCircleView getCircleView(int wh) {
234         GestureLockCircleView gestureCircleView = new GestureLockCircleView(mContext);
235 
236         double s = Math.pow(mCount, 3) + 0.5f;//除法系数,用于计算内圆的半径; 行数的3次方,并且转为浮点型
237         gestureCircleView.setInnerCircle(getResources().getColor(R.color.colorChecked), (float) (wh / s));
238 
239         paint_correct.setStrokeWidth((float) (wh / s) * 0.2f);
240         paint_error.setStrokeWidth((float) (wh / s) * 0.2f);
241 
242         //内圆颜色,内圆半径
243         s = Math.pow(mCount, 2) + 0.5f;//除法系数,用于计算外圆的半径;行数的2次方,并且转为浮点型
244         gestureCircleView.setBorderRound(hasRoundBorder, getResources().getColor(R.color.colorChecked), (float) (wh / s));//是否有边框,外圆颜色,外圆半径
245         gestureCircleView.switchStatus(GestureLockCircleView.STATUS_NOT_CHECKED);
246         return gestureCircleView;
247     }
248 
249     /**
250      * 重置所有按键为 notChecked 状态
251      */
252     private void resetAllCircleBtn() {
253         if (gestureCircleViewArr == null) return;
254         for (int i = 0; i < gestureCircleViewArr.length; i++) {
255             gestureCircleViewArr[i].switchStatus(GestureLockCircleView.STATUS_NOT_CHECKED);
256         }
257     }
258 
259     //*************************手势密码路径的管理***********************************************
260     private void initLockPathArr() {
261         lockPathArr = new ArrayList<>();
262     }
263 
264     /**
265      * 增加一个密码数字
266      *
267      * @param p
268      */
269     private void addPwd(int p) {
270         if (!checkRepetition(p)) {
271             lockPathArr.add(p);
272         }
273     }
274 
275     private void resetPwd() {
276         if (lockPathArr == null)
277             lockPathArr = new ArrayList<>();
278         else
279             lockPathArr.clear();
280     }
281 
282     /**
283      * 绘制密码“轨迹线”
284      *
285      * @param canvas
286      */
287     private void drawLockPath(Canvas canvas) {
288         canvas.drawPath(lockPath, currentPaint);
289     }
290 
291     /**
292      * 重置引导线的起/终 坐标值
293      */
294     private void resetMovingPathCoordinate() {
295         guideLineStartX = 0;
296         guideLineStartY = 0;
297         guideLineEndX = 0;
298         guideLineEndY = 0;
299     }
300 
301     /**
302      * 绘制引导线
303      */
304     private void drawMovingPath(Canvas canvas) {
305         if (guideLineStartX != 0 && guideLineStartY != 0)//只有当起始位置不是0的时候,才进行绘制
306             canvas.drawLine(guideLineStartX, guideLineStartY, guideLineEndX, guideLineEndY, currentPaint);
307     }
308 
309     /**
310      * 辅助方法,获得一个View的中心位置
311      *
312      * @param v
313      * @return
314      */
315     private Point getCenterPoint(View v) {
316         Rect rect = new Rect();
317         v.getHitRect(rect);
318         int x = rect.left + v.getWidth() / 2;
319         int y = rect.top + v.getHeight() / 2;
320         return new Point(x, y);
321     }
322 
323     /**
324      * 判断当前点击的点位置是不是在子元素范围之内
325      *
326      * @param x
327      * @param y
328      * @param v
329      * @return
330      */
331     private boolean ifClickOnView(int x, int y, View v) {
332         Rect r = new Rect();
333         v.getHitRect(r);
334 
335         //判定点是不是在view范围内,根据业务需求,要给view一个判定的间隙,比如 5*5的View,判定范围只能是3*3
336         //以原来的矩阵为基础,重新定一个判定范围,范围暂时定位原来的80%
337         //真正的判定区域的矩阵范围
338 
339         int w = v.getWidth();
340         int h = v.getHeight();
341 
342         int realLeft = (int) (r.left + marginRate * w);
343         int realTop = (int) (r.top + marginRate * h);
344         int realRight = (int) (r.right - marginRate * w);
345         int realBottom = (int) (r.bottom - marginRate * h);
346 
347         Rect rect1 = new Rect(realLeft, realTop, realRight, realBottom);
348 
349         if (rect1.contains(x, y)) {
350             return true;
351         }
352         return false;
353     }
354 
355     /**
356      * 根据点坐标,返回当前点在哪个密码键的范围内,直接返回View对象
357      *
358      * @param x
359      * @param y
360      * @return
361      */
362     private GestureLockCircleView getClickedChild(int x, int y) {
363         for (GestureLockCircleView v : gestureCircleViewArr) {
364             if (ifClickOnView(x, y, v)) {//
365                 return v;
366             }
367         }
368         return null;
369     }
370 
371     /**
372      * 根据点坐标,返回当前点在哪个密码键的范围内,直接返回View对象的id
373      *
374      * @param x
375      * @param y
376      * @return
377      */
378     private int getClickedChildIndex(int x, int y) {
379         for (int i = 0; i < gestureCircleViewArr.length; i++) {
380             View v = gestureCircleViewArr[i];
381             if (ifClickOnView(x, y, v)) {//
382                 return i;
383             }
384         }
385         return -1;
386     }
387 
388     /**
389      * 检查密码值是否重复
390      *
391      * @return
392      */
393     private boolean checkRepetition(int pwd) {
394         return lockPathArr.contains(pwd);
395     }
396 
397     /**
398      * 手势绘制
399      *
400      * @param event
401      * @return
402      */
403     @Override
404     public boolean onTouchEvent(MotionEvent event) {
405         if (ifAllowInteract)//只有设置了允许事件交互,才往下执行
406             switch (event.getAction()) {
407                 case MotionEvent.ACTION_DOWN:
408                     onToast("", ColorHolder.COLOR_GRAY);
409                     downX = (int) event.getX();
410                     downY = (int) event.getY();
411                     ifAllowDrawLockPath = false;
412                     GestureLockCircleView current = getClickedChild(downX, downY);
413                     if (current != null) {//如果当前按下的点,没有在任何一个按键范围之内
414                         ifAllowDrawLockPath = true;
415 
416                         if (ifCheckOnErr)
417                             current.switchStatus(GestureLockCircleView.STATUS_CHECKED_ERR);
418                         else
419                             current.switchStatus(GestureLockCircleView.STATUS_CHECKED);//down的时候,将当前这个按键设置为checked
420 
421                         childStartIndex = getClickedChildIndex(downX, downY);
422                         //记录手势密码
423                         lockPath.reset();
424                         resetPwd();
425                         addPwd(childStartIndex);
426                         //path处理
427                         Point startP = getCenterPoint(gestureCircleViewArr[childStartIndex]);
428                         if (startP != null) {//因为如果
429                             lockPath.moveTo(startP.x, startP.y);
430                             //引导线的起始坐标
431                             guideLineStartX = startP.x;
432                             guideLineStartY = startP.y;
433                         } else {
434                             Log.d("tagpx", "1");
435                         }
436                     } else {
437                         //如果第一次点下去,就是在 键位的空隙里面。那么,就不用绘制了
438                         Log.d("tagpx", "2");
439                     }
440 
441                     break;
442                 case MotionEvent.ACTION_MOVE:
443                     if (ifAllowDrawLockPath) {
444                         movedX = (int) event.getX();
445                         movedY = (int) event.getY();
446                         childEndIndex = getClickedChildIndex(movedX, movedY);
447 
448                         //-1表示没有找到对应的区域
449                         boolean flag1 = childStartIndex != -1 && childEndIndex != -1;//没有获取到正确的对应区域
450                         boolean flag2 = childStartIndex != childEndIndex;//在同一个区域内不需要画线
451                         boolean flag3 = checkRepetition(childEndIndex);//不允许密码值重复,这里要检查当前这个区域是不是已经在lockPathArr里面
452 
453                         if (flag1 && flag2 && !flag3) {//如果起点终点都在区域之内,那么就直接绘制“轨迹线”
454                             Point endP = getCenterPoint(gestureCircleViewArr[childEndIndex]);
455                             GestureLockCircleView cur = getClickedChild(movedX, movedY);
456                             if (ifCheckOnErr)
457                                 cur.switchStatus(GestureLockCircleView.STATUS_CHECKED_ERR);
458                             else
459                                 cur.switchStatus(GestureLockCircleView.STATUS_CHECKED);
460 
461                             addPwd(childEndIndex);
462                             lockPath.lineTo(endP.x, endP.y);
463 
464                             guideLineStartX = endP.x;
465                             guideLineStartY = endP.y;
466                         }
467                         guideLineEndX = movedX;
468                         guideLineEndY = movedY;
469                         postInvalidate();//刷新视图
470                     }
471                     break;
472                 case MotionEvent.ACTION_UP:
473                 case MotionEvent.ACTION_CANCEL:
474                     if (ifAllowDrawLockPath) {
475                         resetMovingPathCoordinate(); // up的时候,要清除引导线
476                         lockPath.reset(); //同时要清除轨迹线
477                         postInvalidate();//刷新本layout
478                         resetAllCircleBtn();//up的时候,把所有按键全部设置为notChecked,
479                         onSwipeFinish();
480                         if (lockPathArr.size() >= minLengthOfPwd) {
481                             if (mModeStatus == STATUS_RESET) {//如果处于reset模式下,执行rest的回调
482                                 onReset();
483                             } else if (mModeStatus == STATUS_CHECK) {//检查模式下,执行onCheck
484                                 onCheck();
485                             } else {
486                                 throw new RuntimeException("异常模式,请正确调用switchToCheckMode/switchToResetMode!");
487                             }
488                         } else {
489                             onToast(String.format(ToastStrHolder.swipeTooLittlePointStr, minLengthOfPwd), ColorHolder.COLOR_RED);
490                         }
491                     }
492                     break;
493                 default:
494                     break;
495             }
496         return super.onTouchEvent(event);
497     }
498 
499     private void onSwipeFinish() {
500         if (mGestureEventCallback == null) return;
501         mGestureEventCallback.onSwipeFinish(copyPwd(lockPathArr));
502     }
503 
504     private void onReset() {
505         if (mGestureEventCallback == null) return;
506         if (resetCurrentTime == 0) {//第一次绘制,赋值给tempPwd
507             tempPwd = copyPwd(lockPathArr);
508             resetCurrentTime++;
509             onToast(ToastStrHolder.tryAgainStr, ColorHolder.COLOR_GRAY);
510         } else {
511             try {
512                 boolean s = compare(tempPwd, lockPathArr);
513                 if (s) {
514                     onToast(ToastStrHolder.successStr, ColorHolder.COLOR_GRAY);
515                     mGestureEventCallback.onResetFinish(copyPwd(lockPathArr));//执行回调
516                 } else {
517                     onToast(ToastStrHolder.notSameStr, ColorHolder.COLOR_RED);
518                 }
519             } catch (RuntimeException e) {
520                 e.printStackTrace();
521             }
522         }
523     }
524 
525     /**
526      * 初始化当前的绘制次数
527      */
528     public void initCurrentTimes() {
529         resetCurrentTime = 0;
530     }
531 
532     private void onCheck() {
533         if (mGestureEventCallback == null) return;
534         boolean compareRes = compare(checkPwd, lockPathArr); //对比当前密码和外界传入的密码
535         if (currentAttemptTime <= maxAttemptTimes) {//如果还能继续尝试解锁,那么
536             if (compareRes) {//如果成功
537                 mGestureEventCallback.onCheckFinish(compareRes);//直接返回结果
538 
539                 currentAttemptTime = 1;
540                 currentPaint = paint_correct;
541                 ifCheckOnErr = false;
542             } else {//否则,提示
543                 int remindTime = maxAttemptTimes - currentAttemptTime;
544                 if (remindTime > 0) {
545                     onToast(String.format(ToastStrHolder.wrongPwdInputStr, remindTime), ColorHolder.COLOR_RED);
546 
547                     currentPaint = paint_error;
548                     ifCheckOnErr = true;
549                 } else {
550                     mGestureEventCallback.onCheckFinish(compareRes);//直接返回结果
551                 }
552                 currentAttemptTime++;
553             }
554         } else {//如果已经不能尝试, 无论是否成功,都要返回结果
555             mGestureEventCallback.onCheckFinish(compareRes);
556             currentAttemptTime = 1;
557         }
558     }
559 
560     private void onSwipeMore() {
561         if (mGestureEventCallback == null) return;
562         mGestureEventCallback.onSwipeMore();
563     }
564 
565     private void onToast(String s, int color) {
566         if (mGestureEventCallback == null) return;
567         mGestureEventCallback.onToast(s, color);
568     }
569 
570     /**
571      * 提供一个方法,绘制密码点,但是只绘制 圆圈,不绘制引导线和轨迹线
572      */
573     public void refreshPwdKeyboard(List<Integer> pwd) {
574         try {
575             for (int i = 0; i < mCount * mCount; i++) {//先把所有的点都设置为notChecked
576                 gestureCircleViewArr[i].switchStatus(GestureLockCircleView.STATUS_NOT_CHECKED);
577             }
578 
579             if (null != pwd)
580                 for (int i = 0; i < pwd.size(); i++) {//再把密码中的点,设置为checked
581                     gestureCircleViewArr[pwd.get(i)].switchStatus(GestureLockCircleView.STATUS_CHECKED);
582                 }
583         } catch (IndexOutOfBoundsException e) {
584             //这里有可能发生数组越界,因为 本类的各个对象时相互独立的,方阵行数可能不同
585             e.printStackTrace();
586         }
587     }
588 
589     //*************************下面业务对接***********************************************
590     public interface GestureEventCallback {
591         /**
592          * 当滑动结束,无论模式,只要滑动之后发现upEvent就执行
593          */
594         void onSwipeFinish(List<Integer> pwd);
595 
596         /**
597          * 当重新设置密码成功的时候,将密码返回出去
598          *
599          * @param pwd 设置的密码
600          */
601         void onResetFinish(List<Integer> pwd);
602 
603         /**
604          * 如果当前模式是 check模式,则用这个方法来返回check的结果
605          *
606          * @param succeedOrFailed 校验是否成功
607          */
608         void onCheckFinish(boolean succeedOrFailed);
609 
610         /**
611          * 如果当前滑动的密码格子数太少(比如设置了至少滑动4格,却只滑了2格)
612          */
613         void onSwipeMore();
614 
615         /**
616          * 当需要给外界反馈信息的时候
617          *
618          * @param s     信息内容
619          * @param color 有必要的话,传字体颜色给外界
620          */
621         void onToast(String s, int color);
622     }
623 
624     /**
625      * 反馈给外界的回调
626      */
627     private GestureEventCallback mGestureEventCallback;
628 
629     public void setGestureFinishedCallback(GestureEventCallback gestureFinishedCallback) {
630         this.mGestureEventCallback = gestureFinishedCallback;
631     }
632 
633     public static class GestureEventCallbackAdapter implements GestureEventCallback {
634 
635         @Override
636         public void onSwipeFinish(List<Integer> pwd) {
637 
638         }
639 
640         @Override
641         public void onResetFinish(List<Integer> pwd) {
642 
643         }
644 
645         @Override
646         public void onCheckFinish(boolean succeedOrFailed) {
647 
648         }
649 
650         @Override
651         public void onSwipeMore() {
652 
653         }
654 
655         @Override
656         public void onToast(String s, int color) {
657 
658         }
659     }
660 
661     //*************************下面是辅助方法以及辅助内部类***********************************************
662 
663     /**
664      * 辅助方法,复制一份密码对象,因为如果直接把当前对象的密码返回出去,则外界使用的全部都是同一个对象,这个对象可能随时变化,外层逻辑无法对比密码值
665      */
666     private List<Integer> copyPwd(List<Integer> pwd) {
667         List<Integer> copyOne = new ArrayList<>();
668         for (int i = 0; i < pwd.size(); i++) {
669             copyOne.add(pwd.get(i));
670         }
671         return copyOne;
672     }
673 
674     /**
675      * 对比两个list是否内容完全相同
676      */
677     private boolean compare(List<Integer> list1, List<Integer> list2) throws RuntimeException {
678 
679         if (list1 == null || list2 == null) {
680             throw new RuntimeException("存在list为空,不执行对比");
681         }
682 
683         if (list1.size() != list2.size())//size长度都不同,就不用比了
684             return false;
685 
686         for (int i = 0; i < list1.size(); i++) {
687             if (list1.get(i) != list2.get(i)) {
688                 return false;
689             }
690         }
691         return true;
692     }
693 
694 
695     public class ColorHolder {
696         public static final int COLOR_RED = 0xffFF3232;
697         public static final int COLOR_GRAY = 0xff999999;
698         public static final int COLOR_YELLOW = 0xffF8A916;
699     }
700 
701     public class ToastStrHolder {
702         public static final String successStr = "绘制成功";
703         public static final String tryAgainStr = "请再次绘制手势密码";
704         public static final String notSameStr = "与首次绘制不一致,请再次绘制";
705         public static final String forYourSafetyStr = "为了您的账户安全,请设置手势密码";
706         public static final String swipeTooLittlePointStr = "请最少连接%s个点";
707         public static final String wrongPwdInputStr = "输入错误,您还可以输入%s次";
708     }
709 }

 

 

 具体使用方法:


只展示一个例子,这是设置手势密码的界面,红色的代码就是你需要自己编写的;

package com.example.gesture_password_study.gesture_pwd;


import android.os.Bundle;
import android.support.annotation.Nullable;
import android.view.View;
import android.widget.TextView;
import android.widget.Toast;

import com.example.gesture_password_study.R;
import com.example.gesture_password_study.gesture_pwd.base.GestureBaseActivity;
import com.example.gesture_password_study.gesture_pwd.custom.EasyGestureLockLayout;

import java.util.List;

/**
 * 手势密码 设置界面
 */
public class GesturePwdSettingActivity extends GestureBaseActivity {

    EasyGestureLockLayout layout_small;
    TextView tv_go;
    TextView tv_redraw;
    EasyGestureLockLayout layout_parent;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.layout_gesture_pwd_setting);
        initView();
        initLayoutView();
    }

    private void initView() {
        tv_go = findViewById(R.id.tv_go);
        layout_parent = findViewById(R.id.layout_parent);
        layout_small = findViewById(R.id.layout_small);
        tv_redraw = findViewById(R.id.tv_redraw);
    }


    protected void initLayoutView() {
        
        //写个适配器
        EasyGestureLockLayout.GestureEventCallbackAdapter adapter = new EasyGestureLockLayout.GestureEventCallbackAdapter() {
            @Override
            public void onSwipeFinish(List<Integer> pwd) {
                layout_small.refreshPwdKeyboard(pwd);//通知另一个小密码盘,将密码点展示出来,但是不展示轨迹线
                tv_redraw.setVisibility(View.VISIBLE);
            }

            @Override
            public void onResetFinish(List<Integer> pwd) {// 当密码设置完成
                savePwd(showPwd("showGesturePwdInt", pwd));//保存密码到本地
                Toast.makeText(GesturePwdSettingActivity.this, "密码已保存", Toast.LENGTH_SHORT).show();
            }

            @Override
            public void onCheckFinish(boolean succeedOrFailed) {
                String str = succeedOrFailed ? "解锁成功" : "解锁失败";
                Toast.makeText(GesturePwdSettingActivity.this, str, Toast.LENGTH_SHORT).show();
                if (succeedOrFailed) {//如果解锁成功,则切换到set模式
                    layout_parent.switchToResetMode();
                } else {
                    onCheckFailed();
                }
            }

            @Override
            public void onSwipeMore() {
                //执行动画
                animate(tv_go);
            }

            @Override
            public void onToast(String s, int textColor) {
                tv_go.setText(s);
                if (textColor != 0)
                    tv_go.setTextColor(textColor);

                if (textColor == 0xffFF3232) {
                    animate(tv_go);
                }
            }
        };

        layout_parent.setGestureFinishedCallback(adapter);

        //使用rest模式
        layout_parent.switchToResetMode();

        tv_redraw.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                layout_parent.initCurrentTimes();
                tv_redraw.setVisibility(View.INVISIBLE);
                layout_small.refreshPwdKeyboard(null);
                tv_go.setText("请重新绘制");
            }
        });
    }



}

 

它的布局xml:

layout_gesture_pwd_setting.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@android:color/white"
    android:gravity="center_horizontal"
    android:orientation="vertical">

    <TextView
        android:id="@+id/tv_skip"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="right"
        android:layout_marginBottom="16dp"
        android:layout_marginRight="20dp"
        android:layout_marginTop="40dp"
        android:text="--"
        android:textColor="@color/color_v"
        android:textSize="15sp" />


    <com.example.gesture_password_study.gesture_pwd.custom.EasyGestureLockLayout
        android:id="@+id/layout_small"
        android:layout_width="@dimen/small_grid_width"
        android:layout_height="@dimen/small_grid_width"
        app:count="3"
        app:ifAllowInteract="false"
        app:ifChildHasBorder="false" />

    <TextView
        android:id="@+id/tv_go"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="24dp"
        android:text="为了您的账户安全,请设置手势密码"
        android:textColor="#F8A916"
        android:textSize="13sp" />

    <com.example.gesture_password_study.gesture_pwd.custom.EasyGestureLockLayout
        android:id="@+id/layout_parent"
        android:layout_width="@dimen/big_grid_width"
        android:layout_height="@dimen/big_grid_width"
        android:layout_marginTop="64dp"
        app:count="3"
        app:ifAllowInteract="true"
        app:ifChildHasBorder="true" />

    <TextView
        android:id="@+id/tv_redraw"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="10dp"
        android:text="重新绘制"
        android:textColor="@color/color_v"
        android:textSize="15sp"
        android:visibility="invisible"/>


</LinearLayout>

count属性,是控制 密码盘的 方阵宽度,目前是3,所以呈现出来就是3*3;

你可以换成4,5,6···随意,只要你没有密集恐惧症.```````````

 

 ===========================================

 欧拉,源码解读就到这里,也没什么复杂的东西。

想起之前面试的时候有一个大佬问我的问题, 自定义ViewGroup能不能在里面同时放置子View并且还能对自身进行绘制。

当时一脸懵逼,不知道什么意思,···· 现在知道了。

自定义ViewGroup,然后addView。。。然后还 onDraw··自己。

 

 

 

喜欢的大佬可以下载源码,欢迎留言讨论···

posted @ 2018-09-05 10:24  波澜不惊x  阅读(1857)  评论(0编辑  收藏  举报