2021-10-14 Android app 九宫图案解锁实现

本文介绍了一种自定义View的方式,用于实现Android的九宫格手势解锁功能。通过创建NineLockView类,实现了绘制九宫格节点、监听用户触摸事件并解析手势路径,最终回调接口传递解锁结果。涉及到的关键技术包括自定义View的绘制、触摸事件处理以及接口设计。
摘要由CSDN通过智能技术生成

一、效果图如下。

 二、废话不多说,直接read the fuck code。

       1、代码预览

        2、src\main\java\com\giada\ninelock\MainActivity.java

package com.giada.ninelock;

import androidx.appcompat.app.AppCompatActivity;

import android.os.Bundle;
import android.util.Log;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity implements NineLockListener {
    private NineLockView mNineLockView = null;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mNineLockView=(NineLockView)findViewById(R.id.nine);
        mNineLockView.setLockListener(this);
    }
    @Override
   public void  onLockResult(int[] result)
    {

        StringBuffer stringBuffer = new  StringBuffer();
        for(int i :result)
        {
            stringBuffer.append(Integer.toString(i) );
            Log.i("NineLockView","for i="+i);
        }
        Toast.makeText(getApplicationContext(), "onLockResult:"+stringBuffer.toString(), Toast.LENGTH_SHORT).show();
    }

    @Override
    public void onError()
    {
        System.out.println("Please draw again!");
        Toast.makeText(getApplicationContext(), "Please draw again!", Toast.LENGTH_SHORT).show();
    }
}

        3、src\main\java\com\giada\ninelock\NineLockListener.java

package com.giada.ninelock;

public interface NineLockListener {
    void onLockResult(int[] result);
    void onError();
}

        4、src\main\java\com\giada\ninelock\NineLockView.java

package com.giada.ninelock;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Point;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.widget.Toast;

import androidx.annotation.Nullable;

import java.util.ArrayList;
import java.util.List;

public class NineLockView extends View {

    private Paint paint;
    private boolean isInit=true;//是否初始化
    private boolean isDrawEnd=false;//是否画最后的点与触摸点之间的线
    private int offsetX;//x轴偏移量
    private int offsetY;//y轴偏移量
    private int step;//节点间的距离
    private int radius=40;//节点圆圈的半径大小
    private int lastPointX;//最后一个节点的 x 坐标
    private int lastPointY;//最后一个节点的 y 坐标
    private float linearEndX;//触摸点的最终位置坐标
    private float linearEndY;
    private List<Point> code=new ArrayList<Point>();//选中点的集合
    private NineLockListener lockListener;

    public NineLockView(Context context,@Nullable AttributeSet attrs) {
        super(context, attrs);
        init();
    }

    private void init(){
        paint=new Paint();
        paint.setColor(Color.parseColor("#ff0000"));
        paint.setAntiAlias(true);
        paint.setStrokeWidth(1);
        paint.setStrokeJoin(Paint.Join.ROUND);
        paint.setStrokeCap(Paint.Cap.ROUND);
    }

    public void setLockListener(NineLockListener lockListener) {
        this.lockListener = lockListener;
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        if(isInit){
            isInit=false;
            int w=getWidth();
            int h=getHeight();
            if(w>h){
                offsetX=(w-h)/2;
                w=h;
            }else {
                offsetY=(h-w)/2;
                h=w;
            }
            step=w/4;
            code.clear();
            lastPointY=0;
            lastPointX=0;
        }
        for(int i=1;i<4;i++){
            for(int j=1;j<4;j++){
                canvas.drawCircle(
                        offsetX+step*i,
                        offsetY+step*j,
                        radius,
                        paint
                );
            }
        }
        paint.setStrokeWidth(8);//设置连线的宽度
        if(code.size()>=1){
            for(int i=1;i<code.size();i++){
                canvas.drawLine(code.get(i-1).x*step+offsetX,
                        code.get(i-1).y*step+offsetY,
                        code.get(i).x*step+offsetX,
                        code.get(i).y*step+offsetY,
                        paint);
            }
            lastPointX=code.get(code.size()-1).x*step+offsetX;
            lastPointY=code.get(code.size()-1).y*step+offsetY;
        }
        if(isDrawEnd && lastPointX!=0 && lastPointY!=0 && linearEndX > 9 && linearEndY>9){
            canvas.drawLine(lastPointX,lastPointY,linearEndX,linearEndY,paint);
        }


    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        Log.i("NineLockView","onTouchEvent");
        switch (event.getAction()){
            case MotionEvent.ACTION_DOWN:
                isInit=true;
                isDrawEnd=true;
                Log.i("NineLockView","MotionEvent.ACTION_DOWN");
                invalidate();
                standLinear(event.getX(),event.getY());
                break;
            case MotionEvent.ACTION_MOVE:
                Log.i("NineLockView","ACTION_MOVE");
                standLinear(event.getX(),event.getY());
                break;
            case MotionEvent.ACTION_UP:
                isDrawEnd=false;
                Log.i("NineLockView","ACTION_UP");
                setResultCode();
                invalidate();
                break;
        }
        return super.onTouchEvent(event);
    }

    /**
     * 根据手的滑动判断是否选中某个节点
     * 当触摸点与节点值之间的距离小于 radius 时默认选中
     * @param x
     * @param y
     */
    private void standLinear(float x,float y){
        boolean isStand=false;
        for(int i=1;i<4;i++){
            for(int j=1;j<4;j++){
                float tx=offsetX+step*i-x;
                float ty=offsetY+step*j-y;
                if(Math.sqrt(tx*tx+ty*ty)<=radius){
                    isStand=true;
                    if(code.size()==0){
                        linearEndX=i;
                        linearEndY=j;
                        code.add(new Point(i,j));
                    }else {
                        Point last=code.get(code.size() - 1);
                        //如果当前点与记录的最后一个点重复,则不再重复添加
                        if (last.x == i && last.y == j) {
                            linearEndX = x;
                            linearEndY = y;
                        } else {
                            //添加记录点,如果在竖直或水平方向上中间间隔一个点则间隔的点也要加上
                            //未考虑对角线间隔的情况
                            if(i==last.x && j==last.y+2){
                                code.add(new Point(i,j-1));
                            }else if(i==last.x && j==last.y-2){
                                code.add(new Point(i,j+1));
                            }else if(j==last.y && i==last.x+2){
                                code.add(new Point(i-1,j));
                            }else if(j==last.y && i==last.x-2){
                                code.add(new Point(i+1,j));
                            }
                            code.add(new Point(i,j));
                        }
                    }
                    break;
                }
            }
            if(isStand){
                break;
            }
        }
        if(!isStand){
            linearEndX=x;
            linearEndY=y;
        }
        invalidate();
    }

    private void setResultCode(){
        if(lockListener!=null){
            if(code.size()==0){
                lockListener.onError();
            }else {
                int[] result = new int[code.size()];
                for (int i = 0; i < code.size(); i++) {
                    result[i] = (code.get(i).y - 1 )* 3 + code.get(i).x;
                }
                Log.i("NineLockView","onLockResult");
                lockListener.onLockResult(result);
            }
        }
    }

}

       5、src\main\res\layout\activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:gravity="center"
    android:layout_height="match_parent" >

    <com.giada.ninelock.NineLockView
        android:id="@+id/nine"
        android:layout_width="500dp"
        android:layout_height="500dp"
        android:background="#00BCD4"
        android:clickable="true"/>

</LinearLayout>

       6、src\main\AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.giada.ninelock">

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/Theme.Ninelock">
        <activity android:name=".MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

三、参考文章



https://www.jb51.net/article/80265.htm

https://www.jb51.net/article/76073.htm



自定义View----Android九宫格手势密码解锁_JerryloveEmily平-CSDN博客

android 九宫格图案解锁_禾枇-余的博客-CSDN博客

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值