package com.xjf.repository.view.locuspassword; import android.content.Context; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Paint; import android.graphics.Paint.Style; import android.graphics.Path; import android.util.AttributeSet; import android.util.Log; import android.view.MotionEvent; import android.view.View; import java.util.ArrayList; import java.util.List; import java.util.Timer; import java.util.TimerTask; /** * ----------------------------------------------------------------- * User:xijiufu * Email:xjfsml@163.com * Version:1.0 * Time:2016/10/18--9:40 * Function: 屏幕锁 * ModifyHistory: * ----------------------------------------------------------------- */ public class LocusPasswordView extends View { private float width = 0; private float height = 0; //是否已经缓存 private boolean isCache = false; //画笔 private Paint mPaint = new Paint(Paint.ANTI_ALIAS_FLAG); //九宫格的点 private Point[][] mPoints = new Point[3][3]; //圆半径 private float dotRadius = 0; //选中的点集合 private List<Point> sPoints = new ArrayList<Point>(); private boolean checking = false; private long CLEAR_TIME = 1000; private int pwdMaxLen = 9; private int pwdMinLen = 4; private boolean isTouch = true; private Paint arrowPaint; private Paint linePaint; private Paint selectedPaint; private Paint errorPaint; private Paint normalPaint; //错误颜色 private int errorColor = 0xffea0945; //选中颜色 private int selectedColor = 0xff0596f6; private int outterSelectedColor = 0xff8cbad8; private int outterErrorColor = 0xff901032; //小圆点颜色 private int dotColor = 0xffd9d9d9; // private int outterDotColor = 0xff929292; public LocusPasswordView(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); } public LocusPasswordView(Context context, AttributeSet attrs) { super(context, attrs); } public LocusPasswordView(Context context) { super(context); } @Override public void onDraw(Canvas canvas) { System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>onDraw"); if (!isCache) { initCache(); } drawToCanvas(canvas); } private void drawToCanvas(Canvas canvas) { boolean inErrorState = false; for (int i = 0; i < mPoints.length; i++) { for (int j = 0; j < mPoints[i].length; j++) { Point p = mPoints[i][j]; if (p.state == Point.STATE_CHECK) { // selectedPaint.setColor(outterSelectedColor); selectedPaint.setStyle(Style.STROKE);//描边 selectedPaint.setColor(dotColor); // canvas.drawCircle(p.x, p.y, dotRadius, selectedPaint); selectedPaint.setColor(selectedColor); selectedPaint.setStyle(Style.FILL);//实心 canvas.drawCircle(p.x, p.y, dotRadius / 4, selectedPaint); } else if (p.state == Point.STATE_CHECK_ERROR) { inErrorState = true; // errorPaint.setColor(outterErrorColor); errorPaint.setColor(dotColor); // canvas.drawCircle(p.x, p.y, dotRadius, errorPaint); errorPaint.setStyle(Style.FILL);//实心 errorPaint.setColor(errorColor); canvas.drawCircle(p.x, p.y, dotRadius / 4, errorPaint); } else { normalPaint.setColor(dotColor); // canvas.drawCircle(p.x, p.y, dotRadius, normalPaint); normalPaint.setStyle(Style.FILL);//实心 normalPaint.setColor(outterDotColor); canvas.drawCircle(p.x, p.y, dotRadius / 4, normalPaint); } } } if (inErrorState) { arrowPaint.setColor(errorColor); linePaint.setColor(errorColor); } else { arrowPaint.setColor(selectedColor); linePaint.setColor(selectedColor); } // 画线 if (sPoints.size() > 0) { int tmpAlpha = mPaint.getAlpha(); Point tp = sPoints.get(0); for (int i = 1; i < sPoints.size(); i++) { Point p = sPoints.get(i); drawLine(tp, p, canvas, linePaint); drawArrow(canvas, arrowPaint, tp, p, dotRadius / 4, 38); tp = p; } if (this.movingNoPoint) { drawLine(tp, new Point(moveingX, moveingY, -1), canvas, linePaint); } mPaint.setAlpha(tmpAlpha); } } /*** * 绘制两点连接 * * @param start * @param end * @param canvas * @param paint */ private void drawLine(Point start, Point end, Canvas canvas, Paint paint) { double d = MathUtil.distance(start.x, start.y, end.x, end.y); float rx = (float) ((end.x - start.x) * dotRadius / 4 / d); float ry = (float) ((end.y - start.y) * dotRadius / 4 / d); canvas.drawLine(start.x + rx, start.y + ry, end.x - rx, end.y - ry, paint); } /** * 绘制箭头 * * @param canvas * @param paint * @param start * @param end * @param arrowHeight * @param angle */ private void drawArrow(Canvas canvas, Paint paint, Point start, Point end, float arrowHeight, int angle) { /*double d = MathUtil.distance(start.x, start.y, end.x, end.y); float sin_B = (float) ((end.x - start.x) / d); float cos_B = (float) ((end.y - start.y) / d); float tan_A = (float) Math.tan(Math.toRadians(angle)); float h = (float) (d - arrowHeight - dotRadius * 1.1); float l = arrowHeight * tan_A; float a = l * sin_B; float b = l * cos_B; float x0 = h * sin_B; float y0 = h * cos_B; float x1 = start.x + (h + arrowHeight) * sin_B; float y1 = start.y + (h + arrowHeight) * cos_B; float x2 = start.x + x0 - b; float y2 = start.y + y0 + a; float x3 = start.x + x0 + b; float y3 = start.y + y0 - a; Path path = new Path(); path.moveTo(x1, y1); path.lineTo(x2, y2); path.lineTo(x3, y3); path.close(); canvas.drawPath(path, paint); */ } /** * 初始化缓存 */ private void initCache() { width = this.getWidth(); height = this.getHeight(); float x = 0; float y = 0; if (width > height) { x = (width - height) / 2; width = height; } else { y = (height - width) / 2; height = width; } int leftPadding = 15; float dotPadding = width / 3 - leftPadding; float middleX = width / 2; float middleY = height / 2; mPoints[0][0] = new Point(x + middleX - dotPadding, y + middleY - dotPadding, 1); mPoints[0][1] = new Point(x + middleX, y + middleY - dotPadding, 2); mPoints[0][2] = new Point(x + middleX + dotPadding, y + middleY - dotPadding, 3); mPoints[1][0] = new Point(x + middleX - dotPadding, y + middleY, 4); mPoints[1][1] = new Point(x + middleX, y + middleY, 5); mPoints[1][2] = new Point(x + middleX + dotPadding, y + middleY, 6); mPoints[2][0] = new Point(x + middleX - dotPadding, y + middleY + dotPadding, 7); mPoints[2][1] = new Point(x + middleX, y + middleY + dotPadding, 8); mPoints[2][2] = new Point(x + middleX + dotPadding, y + middleY + dotPadding, 9); // Log.d("jerome", "canvas width:" + width); dotRadius = width / 10; isCache = true; initPaints(); } /** * 初始化画笔 */ private void initPaints() { arrowPaint = new Paint(); arrowPaint.setColor(selectedColor); arrowPaint.setStyle(Style.FILL); arrowPaint.setAntiAlias(true); linePaint = new Paint(); linePaint.setColor(selectedColor); linePaint.setStyle(Style.STROKE); linePaint.setAntiAlias(true); linePaint.setStrokeWidth(dotRadius / 9); selectedPaint = new Paint(); selectedPaint.setStyle(Style.STROKE); selectedPaint.setAntiAlias(true); selectedPaint.setStrokeWidth(dotRadius / 6); errorPaint = new Paint(); errorPaint.setStyle(Style.STROKE); errorPaint.setAntiAlias(true); errorPaint.setStrokeWidth(dotRadius / 6); normalPaint = new Paint(); normalPaint.setStyle(Style.STROKE); normalPaint.setAntiAlias(true); normalPaint.setStrokeWidth(dotRadius / 9); } /** * @param index * @return */ public int[] getArrayIndex(int index) { int[] ai = new int[2]; ai[0] = index / 3; ai[1] = index % 3; return ai; } /** * 检查选中的点 * * @param x * @param y * @return */ private Point checkSelectPoint(float x, float y) { for (int i = 0; i < mPoints.length; i++) { for (int j = 0; j < mPoints[i].length; j++) { Point p = mPoints[i][j]; if (MathUtil.checkInRound(p.x, p.y, dotRadius, (int) x, (int) y)) { return p; } } } return null; } /** * 重置 */ private void reset() { for (Point p : sPoints) { p.state = Point.STATE_NORMAL; } sPoints.clear(); this.enableTouch(); } /** * @param p * @return */ private int crossPoint(Point p) { // reset if (sPoints.contains(p)) { if (sPoints.size() > 2) { // if (sPoints.get(sPoints.size() - 1).index != p.index) { return 2; } } return 1; // } else { return 0; // } } /** * 跳点 * @param point */ private void addPoint(Point point) { if (sPoints.size() > 0) { Point lastPoint = sPoints.get(sPoints.size() - 1); int dx = Math.abs(lastPoint.getColNum() - point.getColNum()); int dy = Math.abs(lastPoint.getRowNum() - point.getRowNum()); if ((dx > 1 || dy > 1) && (dx == 0 || dy == 0 || dx == dy)) { // if ((dx > 1 || dy > 1) && (dx != 2 * dy) && (dy != 2 * dx)) { int middleIndex = (point.index + lastPoint.index) / 2 - 1; Point middlePoint = mPoints[middleIndex / 3][middleIndex % 3]; if (middlePoint.state != Point.STATE_CHECK) { middlePoint.state = Point.STATE_CHECK; sPoints.add(middlePoint); } } } this.sPoints.add(point); } /** * @return */ private String toPointString() { if (sPoints.size() >= pwdMinLen && sPoints.size() <= pwdMaxLen) { StringBuffer sf = new StringBuffer(); for (Point p : sPoints) { sf.append(p.index); } return sf.toString(); } else { return ""; } } boolean movingNoPoint = false; float moveingX, moveingY; @Override public boolean onTouchEvent(MotionEvent event) { // if (!isTouch) { return false; } movingNoPoint = false; float ex = event.getX(); float ey = event.getY(); boolean isFinish = false; boolean redraw = false; Point p = null; switch (event.getAction()) { case MotionEvent.ACTION_DOWN: // // System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>ACTION_DOWN"); if (task != null) { task.cancel(); task = null; Log.d("task", "touch cancel()"); } // reset(); p = checkSelectPoint(ex, ey); if (p != null) { checking = true; } break; case MotionEvent.ACTION_MOVE: if (checking) { p = checkSelectPoint(ex, ey); if (p == null) { movingNoPoint = true; moveingX = ex; moveingY = ey; } } System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>ACTION_MOVE"); break; case MotionEvent.ACTION_UP: p = checkSelectPoint(ex, ey); checking = false; isFinish = true; System.out.println(">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>ACTION_UP"); break; } if (!isFinish && checking && p != null) { int rk = crossPoint(p); if (rk == 2) // { // reset(); // checking = false; movingNoPoint = true; moveingX = ex; moveingY = ey; redraw = true; } else if (rk == 0) // { p.state = Point.STATE_CHECK; addPoint(p); redraw = true; } // rk == 1 } // if (redraw) { } if (isFinish) { if (this.sPoints.size() == 1) { this.reset(); } else if (sPoints.size() < pwdMinLen || sPoints.size() > pwdMaxLen) { error(); markError(300); if (mCompleteListener != null) { mCompleteListener.onError(); } } else if (mCompleteListener != null) { this.disableTouch(); if (mCompleteListener != null) { mCompleteListener.onComplete(toPointString()); } } } this.postInvalidate(); return true; } /** * */ private void error() { for (Point p : sPoints) { p.state = Point.STATE_CHECK_ERROR; } } public void markError() { markError(CLEAR_TIME); } public void markError(final long time) { for (Point p : sPoints) { p.state = Point.STATE_CHECK_ERROR; } this.clearPassword(time); } public void enableTouch() { isTouch = true; } public void disableTouch() { isTouch = false; } private Timer timer = new Timer(); private TimerTask task = null; public void clearPassword() { clearPassword(CLEAR_TIME); } public void clearPassword(final long time) { if (time > 1) { if (task != null) { task.cancel(); Log.d("task", "clearPassword cancel()"); } postInvalidate(); task = new TimerTask() { public void run() { reset(); postInvalidate(); } }; Log.d("task", "clearPassword schedule(" + time + ")"); timer.schedule(task, time); } else { reset(); postInvalidate(); } } // private OnCompleteListener mCompleteListener; /** * @param mCompleteListener */ public void setOnCompleteListener(OnCompleteListener mCompleteListener) { this.mCompleteListener = mCompleteListener; } public interface OnCompleteListener { public void onComplete(String password); public void onError(); } }