轻松实现Android自定义九宫格图案解锁
作者:shineflowers 发布时间:2023-09-20 00:32:52
标签:Android,九宫格,解锁
Android实现九宫格图案解锁,自带将图案转化成数字密码的功能,代码如下:
LockPatternView.java
package com.jackie.lockpattern;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Point;
import android.text.TextUtils;
import android.util.AttributeSet;
import android.util.TypedValue;
import android.view.MotionEvent;
import android.view.View;
import java.util.ArrayList;
import java.util.List;
/**
* Created by Jackie on 2015/12/24.
* 图案解锁
*/
public class LockPatternView 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 final int SELECTED_COLOR = 0xFF979797;
public LockPatternView(Context context) {
this(context, null);
}
public LockPatternView(Context context, 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;
}
/**
* 画线
* @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);
}
@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;
}
}
/**
* 图案 *
*/
public interface OnPatternChangeListener {
/**
* 图案改变
* @param patternPassword 图案密码
*/
void onPatternChange(String patternPassword);
/**
* 图案是否重新绘制
* @param isStarted 重新绘制
*/
void onPatternStarted(boolean isStarted);
}
}
PointView.java
package com.jackie.lockpattern;
import android.graphics.Point;
/**
* Created by Jackie on 2015/12/25.
* 自定义点对象
*/
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.java
package com.jackie.lockpattern;
import android.app.Activity;
import android.os.Bundle;
import android.widget.TextView;
public class MainActivity extends Activity implements LockPatternView.OnPatternChangeListener {
private TextView mLockPatternHint;
private LockPatternView mLockPatternView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mLockPatternHint = (TextView) findViewById(R.id.lock_pattern_hint);
mLockPatternView = (LockPatternView) findViewById(R.id.lock_pattern_view);
mLockPatternView.setOnPatternChangeListener(this);
}
@Override
public void onPatternChange(String patternPassword) {
if (patternPassword == null) {
mLockPatternHint.setText("至少5个点");
} else {
mLockPatternHint.setText(patternPassword);
}
}
@Override
public void onPatternStarted(boolean isStarted) {
if (isStarted) {
mLockPatternHint.setText("请绘制图案");
}
}
}
效果图如下:
附上源码地址:https://github.com/shineflower/LockPattern.git
0
投稿
猜你喜欢
- 之前一篇文章研究完横向二级菜单,发现其中使用了SparseArray去替换HashMap的使用.于是乎自己查了一些相关资料,自己同时对性能进
- protobuf 是 google的一个开源项目,可用于以下两种用途:(1)数据的存储(序列化和反序列化),类似于xml、json等;(2)
- 初学线程时,总是将 run 方法和 start 方法搞混,虽然二者是完全不同的两个方法,但刚开始使用时很难分清,原因就是因为初次使用时效果貌
- 对数组使用 foreachC#提供 foreach 语句。 该语句提供一种简单、明了的方法来循环访问数组或任何可枚举集合的元素。 forea
- 本文主要是用到java中的swing技术,以及JMFjar中的API,为大家分享了java音乐播放器的具体实现代码,供大家参考,具体内容如下
- 前言在Android开发过程中,我发现很多安卓源代码里应用了设计模式,比较常用的有适配器模式(各种adapter),建造者模式(Alert
- Android超清晰6.0权限申请AndPermission的具体实现代码,供大家参考,具体内容如下前言这是我经常使用的框架,原因:1.思路
- Allatori混淆技术介绍Allatori是一个Java 混淆器,它属于第二代混淆器,因此它能够全方位地保护你的知识产权。 Allator
- 准备工作这里就不说了,包括签约和申请APPID,附上微信开放平台APP开发步骤,不懂的同学可以参考这里:https://pay.weixin
- 本文实例为大家分享了C#通过NPOI导入导出数据EXCEL的具体代码,供大家参考,具体内容如下其实从数据库到服务器导入导出有很多方法,但是比
- 问题在Service层注入Mybatis的Mapper我们通常会使用@Autowired 自动注入@Autowiredprivate Pro
- Android 自定义阴影效果详解及实例Android5.X中,Google为其增加了两个属性 android:elevation=” ”
- 完美地模仿了2048游戏,是根据网友的一个2048改的。Block.javaimport javax.swing.*;import java
- 自定义TypeHandler映射JSON类型为List1. 实体类这里只展示需要映射的字段,分别在所需映射的字段和实体类上添加注解。&nbs
- java POI合并两个word文档有需要的可以将主函数中写死的地方改为一个Listimport java.io.FileInputStre
- 单纯使用GridView通用的两种给GridView 添加分割线的方法;http://stackoverflow.com/questions
- 1.简介这是一个用于实现像微信朋友圈和微博的类似的九宫格图片展示控件,通过自定义viewgroup实现,使用方便。 多图根据屏幕适配,单张图
- 方法1:以textbox为例①:先设置textbox的属性Multiline为true②:组织好显示字符串:FistLine(第一行要显示的
- 问题描述:服务器接收后台返回的报文时,提示java.lang.NegativeArraySizeException分析:这种异常返回的原因,
- 一.WebSocket简单介绍WebSocket protocol 是HTML5一种新的协议。它实现了浏览器与服务器全双工通信(full-d