Android 自定义View九宫格手势解锁

自定义view

public class SudokuView extends View {
    /**
     * 圆的画笔
     */
    private Paint mCirclePaint;
    /**
     * 线的画笔
     */
    private Paint mLinePaint;
    /**
     * 圆心数组
     */
    private PointView[][] mPointViewArray = new PointView[3][3];
    /**
     * 保存选中点的集合
     */
    private List<PointView> mSelectedPointViewList;


    /**
     * 解锁图案的边长
     */
    private int mPatternWidth;

    /**
     * 图案监听器
     */
    private OnPatternChangeListener mOnPatternChangeListener;

    /**
     * 半径
     */
    private float mRadius;

    /**
     * 每个圆圈的下标
     */
    private int mIndex = 1;

    /**
     * 第一个点是否选中
     */
    private boolean mIsSelected;
    /**
     * 是否绘制结束
     */
    private boolean mIsFinished;

    /**
     * 正在滑动并且没有任何点选中
     */
    private boolean mIsMovingWithoutCircle = false;

    private float mCurrentX, mCurrentY;

    /**
     * 正常状态的颜色
     */
    private static final int NORMAL_COLOR = 0xFF70DBDB;
    /**
     * 选中状态的颜色
     */
    private static int SELECTED_COLOR = 0xFF979797;

    public SudokuView(Context context) {
        super(context);
    }

    public SudokuView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        mCirclePaint = new Paint();
        mCirclePaint.setAntiAlias(true);
        mCirclePaint.setDither(true);
        mCirclePaint.setColor(NORMAL_COLOR);
        mCirclePaint.setStyle(Paint.Style.FILL);

        mLinePaint = new Paint();
        mLinePaint.setAntiAlias(true);
        mLinePaint.setDither(true);
        mLinePaint.setStrokeWidth(20);
        mLinePaint.setColor(SELECTED_COLOR);
        mLinePaint.setStyle(Paint.Style.STROKE);

        mRadius = TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 20, getResources().getDisplayMetrics());

        mSelectedPointViewList = new ArrayList<>();
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        //取屏幕长和宽中的较小值作为图案的边长
        mPatternWidth = Math.min(getMeasuredWidth(), getMeasuredHeight());
        setMeasuredDimension(mPatternWidth, mPatternWidth);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        //画圆
        drawCircle(canvas);

        //将选中的圆重新绘制一遍,将选中的点和未选中的点区别开来
        for (PointView pointView : mSelectedPointViewList) {
            mCirclePaint.setColor(SELECTED_COLOR);
            canvas.drawCircle(pointView.x, pointView.y, mRadius, mCirclePaint);
            mCirclePaint.setColor(NORMAL_COLOR);  //每重新绘制一个,将画笔的颜色重置,保证不会影响其他圆的绘制
        }

        //点与点画线
        if (mSelectedPointViewList.size() > 0) {
            Point pointViewA = mSelectedPointViewList.get(0);  //第一个选中的点为A点
            for (int i = 0; i < mSelectedPointViewList.size(); i++) {
                Point pointViewB = mSelectedPointViewList.get(i);  //其他依次遍历出来的点为B点
                drawLine(canvas, pointViewA, pointViewB);
                pointViewA = pointViewB;
            }

            //点与鼠标当前位置绘制轨迹
            if (mIsMovingWithoutCircle & !mIsFinished) {
                drawLine(canvas, pointViewA, new PointView((int)mCurrentX, (int)mCurrentY));
            }
        }

        super.onDraw(canvas);
    }
    /**
     * 画圆
     * @param canvas  画布
     */
    private void drawCircle(Canvas canvas) {
        //初始化点的位置
        for (int i = 0; i < mPointViewArray.length; i++) {
            for (int j = 0; j < mPointViewArray.length; j++) {
                //圆心的坐标
                int cx = mPatternWidth / 4 * (j + 1);
                int cy = mPatternWidth / 4 * (i + 1);

                //将圆心放在一个点数组中
                PointView pointView = new PointView(cx, cy);
                pointView.setIndex(mIndex);
                mPointViewArray[i][j] = pointView;
                canvas.drawCircle(cx, cy, mRadius, mCirclePaint);
                mIndex++;
            }
        }

        mIndex = 1;
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        mCurrentX = event.getX();
        mCurrentY = event.getY();
        PointView selectedPointView = null;

        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                //重新绘制
                if (mOnPatternChangeListener != null) {
                    mOnPatternChangeListener.onPatternStarted(true);
                }

                mSelectedPointViewList.clear();
                mIsFinished = false;

                selectedPointView = checkSelectPoint();

                if (selectedPointView != null) {
                    //第一次按下的位置在圆内,被选中
                    mIsSelected = true;
                }
                break;
            case MotionEvent.ACTION_MOVE:
                if (mIsSelected) {
                    selectedPointView = checkSelectPoint();
                }

                if (selectedPointView == null) {
                    mIsMovingWithoutCircle = true;
                }
                break;
            case MotionEvent.ACTION_UP:
                mIsFinished = true;
                mIsSelected = false;
                break;
        }

        //将选中的点收集起来
        if (!mIsFinished && mIsSelected && selectedPointView != null) {
            if (!mSelectedPointViewList.contains(selectedPointView)) {
                mSelectedPointViewList.add(selectedPointView);
            }
        }

        if (mIsFinished) {
            if (mSelectedPointViewList.size() == 1) {
                mSelectedPointViewList.clear();
            } else if (mSelectedPointViewList.size() < 5 && mSelectedPointViewList.size() > 0) {
                //绘制错误
                if (mOnPatternChangeListener != null) {
                    mOnPatternChangeListener.onPatternChange(null);
                }
            } else {
                //绘制成功
                String patternPassword = "";
                if (mOnPatternChangeListener != null) {
                    for (PointView pointView : mSelectedPointViewList) {
                        patternPassword += pointView.getIndex();
                    }

                    if (!TextUtils.isEmpty(patternPassword)) {
                        mOnPatternChangeListener.onPatternChange(patternPassword);
                    }
                }
            }
        }

        invalidate();
        return true;
    }

    /**
     * 判断当前按下的位置是否在圆心数组中
     *
     * @return 返回选中的点
     */
    private PointView checkSelectPoint() {
        for (int i = 0; i < mPointViewArray.length; i++) {
            for (int j = 0; j < mPointViewArray.length; j++) {
                PointView pointView = mPointViewArray[i][j];
                if (isWithinCircle(mCurrentX, mCurrentY, pointView.x, pointView.y, mRadius)) {
                    return pointView;
                }
            }
        }

        return null;
    }

    /**
     * 判断点是否在圆内
     *
     * @param x      点X轴坐标
     * @param y      点Y轴坐标
     * @param cx     圆心X坐标
     * @param cy     圆心Y坐标
     * @param radius 半径
     * @return true表示在圆内,false表示在圆外
     */
    private boolean isWithinCircle(float x, float y, float cx, float cy, float radius) {
        //如果点和圆心的距离小于半径,则证明点在圆内
        if (Math.sqrt(Math.pow(x - cx, 2) + Math.pow(y - cy, 2)) <= radius) {
            return true;
        }

        return false;
    }

    /**
     * 设置图案监听器
     */
    public void setOnPatternChangeListener(OnPatternChangeListener onPatternChangeListener) {
        if (onPatternChangeListener != null) {
            this.mOnPatternChangeListener = onPatternChangeListener;
        }
    }

    /**
     * 画线
     *
     * @param canvas 画布
     * @param pointA 第一个点
     * @param pointB 第二个点
     */
    private void drawLine(Canvas canvas, Point pointA, Point pointB) {
        canvas.drawLine(pointA.x, pointA.y, pointB.x, pointB.y, mLinePaint);
    }

    /**
     * 手势图案错误时改变选中的的颜色为红色
     *
     */
    public void error(){
        SELECTED_COLOR = Color.RED;
        mLinePaint.setColor(SELECTED_COLOR);
    }
    /**
     * 手势图案正确时改变选中的的颜色为绿色
     *
     */
    public void access(){
        SELECTED_COLOR = Color.GREEN;
        mLinePaint.setColor(SELECTED_COLOR);
    }
    /**
     * 恢复颜色及布局
     */
    public void clear(){
        mIsFinished = false;
        SELECTED_COLOR = 0xFF979797;
        mLinePaint.setColor(SELECTED_COLOR);
        mSelectedPointViewList.clear();
        invalidate();
    }
    /**
     * 图案监听器
     */
    public interface OnPatternChangeListener {
        /**
         * 图案改变
         *
         * @param patternPassword 图案密码
         */
        void onPatternChange(String patternPassword);

        /**
         * 图案是否重新绘制
         *
         * @param isStarted 重新绘制
         */
        void onPatternStarted(boolean isStarted);
    }
}

圆点类

public class PointView extends Point {
    public int index;

    public PointView(int x, int y) {
        super(x, y);
    }

    public int getIndex() {
        return index;
    }

    public void setIndex(int index) {
        this.index = index;
    }
}

mainActivity

public class MainActivity extends AppCompatActivity implements SudokuView.OnPatternChangeListener {
    public SudokuView sudokuView;
    public TextView textView;
    public Handler handler;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        sudokuView = findViewById(R.id.sudoku_view);
        textView = findViewById(R.id.tv_password);
        sudokuView.setOnPatternChangeListener(this);
        handler = new Handler();
    }

    @Override
    public void onPatternChange(String patternPassword) {
        if (patternPassword == null) {//判断密码
            textView.setText("至少5个点");
            sudokuView.error();
        }else if (patternPassword.equals("123654789")){
            textView.setText(patternPassword);
            sudokuView.access();
        }else{
            textView.setText(patternPassword);
            sudokuView.error();
        }
        // 延迟500毫秒清除绘制的手势点
        handler.postDelayed(new Runnable() {
            public void run() {
                // do something
                textView.setText("");
                sudokuView.clear();
            }

        }, 500);

    }

    @Override
    public void onPatternStarted(boolean isStarted) {
        if (isStarted) {
            textView.setText("请绘制图案");
        }else{

        }
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值