輕松實(shí)現(xiàn)Android自定義九宮格圖案解鎖
Android實(shí)現(xiàn)九宮格圖案解鎖,自帶將圖案轉(zhuǎn)化成數(shù)字密碼的功能,代碼如下:
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 {
/**
* 圓的畫(huà)筆
*/
private Paint mCirclePaint;
/**
* 線的畫(huà)筆
*/
private Paint mLinePaint;
/**
* 圓心數(shù)組
*/
private PointView[][] mPointViewArray = new PointView[3][3];
/**
* 保存選中點(diǎn)的集合
*/
private List<PointView> mSelectedPointViewList;
/**
* 解鎖圖案的邊長(zhǎng)
*/
private int mPatternWidth;
/**
* 圖案監(jiān)聽(tīng)器
*/
private OnPatternChangeListener mOnPatternChangeListener;
/**
* 半徑
*/
private float mRadius;
/**
* 每個(gè)圓圈的下標(biāo)
*/
private int mIndex = 1;
/**
* 第一個(gè)點(diǎn)是否選中
*/
private boolean mIsSelected;
/**
* 是否繪制結(jié)束
*/
private boolean mIsFinished;
/**
* 正在滑動(dòng)并且沒(méi)有任何點(diǎn)選中
*/
private boolean mIsMovingWithoutCircle = false;
private float mCurrentX, mCurrentY;
/**
* 正常狀態(tài)的顏色
*/
private static final int NORMAL_COLOR = 0xFF70DBDB;
/**
* 選中狀態(tài)的顏色
*/
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);
//取屏幕長(zhǎng)和寬中的較小值作為圖案的邊長(zhǎng)
mPatternWidth = Math.min(getMeasuredWidth(), getMeasuredHeight());
setMeasuredDimension(mPatternWidth, mPatternWidth);
}
@Override
protected void onDraw(Canvas canvas) {
//畫(huà)圓
drawCircle(canvas);
//將選中的圓重新繪制一遍,將選中的點(diǎn)和未選中的點(diǎn)區(qū)別開(kāi)來(lái)
for (PointView pointView : mSelectedPointViewList) {
mCirclePaint.setColor(SELECTED_COLOR);
canvas.drawCircle(pointView.x, pointView.y, mRadius, mCirclePaint);
mCirclePaint.setColor(NORMAL_COLOR); //每重新繪制一個(gè),將畫(huà)筆的顏色重置,保證不會(huì)影響其他圓的繪制
}
//點(diǎn)與點(diǎn)畫(huà)線
if (mSelectedPointViewList.size() > 0) {
Point pointViewA = mSelectedPointViewList.get(0); //第一個(gè)選中的點(diǎn)為A點(diǎn)
for (int i = 0; i < mSelectedPointViewList.size(); i++) {
Point pointViewB = mSelectedPointViewList.get(i); //其他依次遍歷出來(lái)的點(diǎn)為B點(diǎn)
drawLine(canvas, pointViewA, pointViewB);
pointViewA = pointViewB;
}
//點(diǎn)與鼠標(biāo)當(dāng)前位置繪制軌跡
if (mIsMovingWithoutCircle & !mIsFinished) {
drawLine(canvas, pointViewA, new PointView((int)mCurrentX, (int)mCurrentY));
}
}
super.onDraw(canvas);
}
/**
* 畫(huà)圓
* @param canvas 畫(huà)布
*/
private void drawCircle(Canvas canvas) {
//初始化點(diǎn)的位置
for (int i = 0; i < mPointViewArray.length; i++) {
for (int j = 0; j < mPointViewArray.length; j++) {
//圓心的坐標(biāo)
int cx = mPatternWidth / 4 * (j + 1);
int cy = mPatternWidth / 4 * (i + 1);
//將圓心放在一個(gè)點(diǎn)數(shù)組中
PointView pointView = new PointView(cx, cy);
pointView.setIndex(mIndex);
mPointViewArray[i][j] = pointView;
canvas.drawCircle(cx, cy, mRadius, mCirclePaint);
mIndex++;
}
}
mIndex = 1;
}
/**
* 畫(huà)線
* @param canvas 畫(huà)布
* @param pointA 第一個(gè)點(diǎn)
* @param pointB 第二個(gè)點(diǎn)
*/
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) {
//第一次按下的位置在圓內(nèi),被選中
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;
}
//將選中的點(diǎn)收集起來(lái)
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) {
//繪制錯(cuò)誤
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;
}
/**
* 判斷當(dāng)前按下的位置是否在圓心數(shù)組中
* @return 返回選中的點(diǎn)
*/
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;
}
/**
* 判斷點(diǎn)是否在圓內(nèi)
* @param x 點(diǎn)X軸坐標(biāo)
* @param y 點(diǎn)Y軸坐標(biāo)
* @param cx 圓心X坐標(biāo)
* @param cy 圓心Y坐標(biāo)
* @param radius 半徑
* @return true表示在圓內(nèi),false表示在圓外
*/
private boolean isWithinCircle(float x, float y, float cx, float cy, float radius) {
//如果點(diǎn)和圓心的距離小于半徑,則證明點(diǎn)在圓內(nèi)
if (Math.sqrt(Math.pow(x - cx, 2) + Math.pow(y- cy, 2)) <= radius) {
return true;
}
return false;
}
/**
* 設(shè)置圖案監(jiān)聽(tīng)器
*/
public void setOnPatternChangeListener(OnPatternChangeListener onPatternChangeListener) {
if (onPatternChangeListener != null) {
this.mOnPatternChangeListener = onPatternChangeListener;
}
}
/**
* 圖案監(jiān)聽(tīng)器
*/
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.
* 自定義點(diǎn)對(duì)象
*/
public class PointView extends Point {
//用于轉(zhuǎn)化密碼的下標(biāo)
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個(gè)點(diǎn)");
} else {
mLockPatternHint.setText(patternPassword);
}
}
@Override
public void onPatternStarted(boolean isStarted) {
if (isStarted) {
mLockPatternHint.setText("請(qǐng)繪制圖案");
}
}
}
效果圖如下:


附上源碼地址:https://github.com/shineflower/LockPattern.git
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android 九宮格的實(shí)現(xiàn)方法
- android 九宮格滑動(dòng)解鎖開(kāi)機(jī)實(shí)例源碼學(xué)習(xí)
- Android實(shí)現(xiàn)九宮格(GridView中各項(xiàng)平分空間)的方法
- Android打造流暢九宮格抽獎(jiǎng)活動(dòng)效果
- Android實(shí)現(xiàn)九宮格解鎖
- Android實(shí)現(xiàn)九宮格橫向左右滑動(dòng)
- Android開(kāi)發(fā)之實(shí)現(xiàn)GridView支付寶九宮格
- Android編程之九宮格實(shí)現(xiàn)方法實(shí)例分析
- 輕松實(shí)現(xiàn)安卓(Android)九宮格解鎖
- Android實(shí)現(xiàn)圖片九宮格
相關(guān)文章
Android編程實(shí)現(xiàn)ImageView圖片拋物線動(dòng)畫(huà)效果的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)ImageView圖片拋物線動(dòng)畫(huà)效果的方法,實(shí)例分析了Android實(shí)現(xiàn)拋物線運(yùn)動(dòng)的算法原理與相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-10-10
Android實(shí)現(xiàn)商城購(gòu)物車功能的實(shí)例代碼
最近公司項(xiàng)目做商城模塊,需要實(shí)現(xiàn)購(gòu)物車功能,主要實(shí)現(xiàn)了單選、全選,金額合計(jì),商品刪除,商品數(shù)量加減等功能,這篇文章主要介紹了Android實(shí)現(xiàn)商城購(gòu)物車功能,需要的朋友可以參考下2019-06-06
EditText限制輸入數(shù)字,精確到小數(shù)點(diǎn)后1位的設(shè)置方法
下面小編就為大家?guī)?lái)一篇EditText限制輸入數(shù)字,精確到小數(shù)點(diǎn)后1位的設(shè)置方法。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-04-04
Android中加載網(wǎng)絡(luò)資源時(shí)的優(yōu)化可使用(線程+緩存)解決
Android 中加載網(wǎng)絡(luò)資源時(shí)的優(yōu)化;基本的思路是線程+緩存來(lái)解決,具體解決思路如下,有類似情況的朋友可以參考下哈2013-06-06
Android中監(jiān)聽(tīng)未接來(lái)電的2種方法
這篇文章主要介紹了Android中監(jiān)聽(tīng)未接來(lái)電的2種方法,本文講解了使用廣播接收器 BrocastReceiver和使用 PhoneStateListener二種方法,需要的朋友可以參考下2015-04-04
Android開(kāi)發(fā)實(shí)現(xiàn)的計(jì)時(shí)器功能示例
這篇文章主要介紹了Android開(kāi)發(fā)實(shí)現(xiàn)的計(jì)時(shí)器功能,涉及Android開(kāi)發(fā)中的計(jì)時(shí)器相關(guān)組件布局、調(diào)用、事件響應(yīng)等相關(guān)操作技巧,需要的朋友可以參考下2019-04-04
Android自定義Chronometer實(shí)現(xiàn)短信驗(yàn)證碼秒表倒計(jì)時(shí)功能
這篇文章主要介紹了Android自定義ChronometerView實(shí)現(xiàn)類似秒表倒計(jì)時(shí),短信驗(yàn)證碼倒計(jì)時(shí)功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11
Android WebView打開(kāi)網(wǎng)頁(yè)一片空白的解決方法
這篇文章主要介紹了Android WebView打開(kāi)網(wǎng)頁(yè)一片空白的解決方法,試了很多方法,最后記錄一下,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-12-12

