Android 自定义九宫格手势锁

预览效果图如下:

主要的方法是重写View.onTouchEvent( MotionEvent event ) , 常用的三个操作:ACTION_DOWN 手指触摸屏幕 ; ACTION_UP 手指离开屏幕;

ACTION_MOVE手指在屏幕滑动。

如果该方法返回true ,表示该事件已经被View处理,不再向上层的View或Activity传递 ; 如果返回false, 表示事件未处理,继续传递。

具体代码如下:

package com.ninegrid;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
/**
 * Created by Administrator on 2017/6/24.
 */
public class SuduView extends View {
  //定义默认常量
  private static final int DEFAULT_CELL_WIDTH = 200 ;
  private static final int DEFAULT_CELL_STROKE_WIDTH = 10 ;
  private static final int DEFAULT_SPACE = 100 ;
  //九宫格数组
  private Cell mCells[] = new Cell[9] ;
  //直径
  private int mCellWidth;
  //半径
  private int mCellRadius;
  //边框宽度
  private int mCellStrokeWidth;
  //空白部分
  private int mSpace ;
  //定义画笔
  private Paint mPaintNormal ;
  private Paint mPaintSelected ;
  private float mCurrentX ;
  private float mCurrentY ;
  //判断是否结束的标识
  private boolean mFinish = false ;
  private StringBuffer mSbSelected = new StringBuffer(20);
  public SuduView(Context context) {
    super(context);
    init();
  }
  public SuduView(Context context, AttributeSet attrs) {
    super(context, attrs);
    init();
  }
  public SuduView(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    init();
  }
  private void init(){
    //初始化画笔
    mCellWidth = DEFAULT_CELL_WIDTH ;
    mCellRadius = DEFAULT_CELL_WIDTH >> 1 ;
    mCellStrokeWidth = DEFAULT_CELL_STROKE_WIDTH ;
    mSpace = DEFAULT_SPACE ;
    mPaintNormal = new Paint();
    mPaintNormal.setColor(Color.WHITE);
    mPaintNormal.setStrokeWidth(mCellStrokeWidth);
    mPaintNormal.setStyle(Paint.Style.STROKE);
    mPaintNormal.setAntiAlias(true);
    mPaintSelected = new Paint();
    mPaintSelected.setColor(Color.CYAN);
    mPaintSelected.setStrokeWidth(mCellStrokeWidth);
    mPaintSelected.setStyle(Paint.Style.STROKE);
    mPaintSelected.setAntiAlias(true);
    Cell cell ;
    float x;
    float y;
    //计算每个格子的坐标
    for( int i = 0 ; i < 9 ; i ++ ){
      x = mSpace * ( i%3 + 1 ) + mCellRadius + mCellWidth * ( i%3 ) ;
      y = mSpace * ( i/3 + 1 ) + mCellRadius + mCellWidth * ( i/3 ) ;
      cell = new Cell(x , y);
      mCells[i] = cell ;
    }
  }
  @Override
  protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    drawCell(canvas);
    drawLine(canvas);
  }
  //绘制连接线
  private void drawLine( Canvas canvas ){
    if("".equals(mSbSelected.toString())){
      return;
    }
    String[] selectedIndexs = mSbSelected.toString().split(",");
    Cell cell = mCells[Integer.valueOf(selectedIndexs[0])];
    Cell nextCell ;
    //绘制每两个格子中心点之间的连接线
    if( selectedIndexs.length > 1) {
      for (int i = 1; i < selectedIndexs.length; i++) {
        nextCell = mCells[Integer.valueOf(selectedIndexs[i])];
        canvas.drawLine(cell.getCenterX(), cell.getCenterY(), nextCell.getCenterX(), nextCell.getCenterY(), mPaintSelected);
        cell = nextCell;
      }
    }
    //绘制格子到其他空白位置的连接线
    if( !mFinish ) {
      canvas.drawLine(cell.getCenterX(), cell.getCenterY(), mCurrentX, mCurrentY, mPaintSelected);
    }
  }
  private void drawCell( Canvas canvas ){
    for ( int i = 0 ; i < 9 ; i ++ ){
      canvas.drawCircle(mCells[i].getCenterX(), mCells[i].getCenterY() , mCellRadius ,
          mCells[i].isSelected() ? mPaintSelected : mPaintNormal );
    }
  }
  //处理点击事件
  @Override
  public boolean onTouchEvent(MotionEvent event) {
    switch ( event.getAction()){
      case MotionEvent.ACTION_DOWN:
        //如果手指已经松开,则所有格子变为初始状态
        if( mFinish ){
          for ( int i = 0 ; i < 9 ; i ++ ){
            mCells[i].setSelected(false);
          }
          mFinish = false ;
          mSbSelected.delete(0,mSbSelected.length());
          invalidate();
          return false;
        }
        handleDownEvent(event);
        break;
      //松开则结束
      case MotionEvent.ACTION_UP:
        mFinish = true ;
        break;
      case MotionEvent.ACTION_MOVE:
        handleMoveEvent(event);
        break;
    }
    //表示已处理,不向上传递
    return true ;
  }
  //处理手指移动的事件
  private void handleMoveEvent( MotionEvent event ){
    int index = findCellIndex(event.getX(),event.getY());
    if( index != -1 ){
      mCells[index].setSelected(true);
      mSbSelected.append(index).append(",");
    }
    invalidate();
    mCurrentX = event.getX();
    mCurrentY = event.getY();
  }
  //处理手指按下的事件
  private void handleDownEvent( MotionEvent event){
    int index = findCellIndex(event.getX(),event.getY());
    if( index != -1 ){
      mCells[index].setSelected(true);
      mSbSelected.append(index).append(",");
      invalidate();
    }
    mCurrentX = event.getX();
    mCurrentY = event.getY();
  }
  //根据坐标判断点击的哪个格子
  private int findCellIndex( float x , float y){
    float cellX ;
    float cellY ;
    int result = -1 ;
    for( int i = 0 ; i < 9 ; i ++ ){
      if( mCells[i].isSelected()){
        continue;
      }
      //获取每个格子的坐标
      cellX = mCells[i].getCenterX();
      cellY = mCells[i].getCenterY();
      //计算按下的点到每个格子的距离
      float tempX = cellX - x ;
      float tempY = cellY - y ;
      float distance = (float) Math.sqrt(tempX * tempX + tempY * tempY);
      //如果点击的位置在某个格子的圆内
      if( distance < mCellRadius ){
        result = i ;
        break;
      }
    }
    //返回该格子的位置
    return result ;
  }
}

最后在布局文件中引用该View即可,若想实现更高的定制性,可以仿照上一篇文章重写View的onMearsure方法并增加自定义属性。

以上所述是小编给大家介绍的Android 自定义九宫格手势锁,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

(0)

相关推荐

  • Android 九宫格的实现方法

    1.xml代码: 复制代码 代码如下: <?xml version="1.0" encoding="utf-8"?>  <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"       android:orientation="vertical"      android:layout_width="fill

  • android 九宫格滑动解锁开机实例源码学习

    效果图由于网站占时不能上传,以后补上. NinePointLineView.java 复制代码 代码如下: package org.demo.custon_view; import org.demo.utils.MLog; import android.content.Context; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.Canvas; imp

  • Android实现九宫格(GridView中各项平分空间)的方法

    本文实例讲述了Android实现九宫格(GridView中各项平分空间)的方法.分享给大家供大家参考.具体如下: 项目需要做一个九宫格(也不一定是9的,4宫格.16宫格.4x3宫格...),封了 一个宫格,它能够根据为它分配的空间来自动的调节宫中各项的尺寸. 从TableLayout集成来的,因此如果你直接在设计器上使用该封装的话需要把它自动加进去的那几个TableRow删除一下. 类名为AdvancedGridView,代码如下: import android.content.Context;

  • Android开发之实现GridView支付宝九宫格

    先给大家展示下关于仿支付宝钱包首页中带有分割线的gridview,俗称九宫格 的效果图,怎么样是不是和你想象的一样啊.在你的预料之中就继续访问以下代码内容吧. 我们都知道ListView设置分割线是非常容易的,设置ListView的分割线颜色和宽度,只需要在布局中定义android:divider和android:dividerHeight属性即可.而GridView并没有这样的属性和方法,那我们改如何来做呢? 我们小编在做这个效果之前,也参考了其他的一些方案,比如说定义一个自定义的GridVi

  • 超实用的Android手势锁制作实例教程

    今天偶遇以github上gesturelock关于手势锁的一个例子(有兴趣的去搜索下看看),于是下载下来研究,无奈基本没有注释,代码上存在一些问题(当设置gravity=center_vertical无法进行手势选择,无意中发现的),于是借鉴这位仁兄的代码,自己重写写了一个,修复了一些问题,加入一些基本的自定义属性,在此先感谢这位兄弟~. 先上图,默认效果图: 当然可以自定义数量啊,颜色神马的,自定义效果图: 如果你有艺术细胞,可以给我推荐几个颜色,无奈个人审美有问题~ 1.整体思路 a.自定义

  • Android 自定义九宫格手势锁

    预览效果图如下: 主要的方法是重写View.onTouchEvent( MotionEvent event ) , 常用的三个操作:ACTION_DOWN 手指触摸屏幕 ; ACTION_UP 手指离开屏幕; ACTION_MOVE手指在屏幕滑动. 如果该方法返回true ,表示该事件已经被View处理,不再向上层的View或Activity传递 : 如果返回false, 表示事件未处理,继续传递. 具体代码如下: package com.ninegrid; import android.con

  • Android自定义UI手势密码改进版源码下载

    在之前文章的铺垫下,再为大家分享一篇:Android手势密码,附源码下载,不要错过. 源码下载:http://xiazai.jb51.net/201610/yuanma/androidLock(jb51.net).rar 先看第一张图片的布局文件 activity_main.xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://sc

  • Android自定义九宫格输入框

    本文实例为大家分享了Android自定义九宫格输入框的具体代码,供大家参考,具体内容如下 效果 实现 绘制宫格分割线 这里我们用一个RectF类型的数组来装载数据.在onSizeChanged方法中获取到控件尺寸,经过计算,将81个位置合适的矩形保存到数组中. 绘制点击效果 在onTouchEvent方法中监听手指离开事件,当手指离开,获取到当前点击区域的RectF,并将状态同样保存到一个数组中. 绘制输入内容 输入内容利用onTextChanged方法获取,同样保存到一个数组中. PS 控件中

  • 轻松实现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; i

  • Android自定义View手势密码

    Android 自定义View 当然是十分重要的,笔者这两天写了一个自定义 View 的手势密码,和大家分享分享: 首先,我们来创建一个表示点的类,Point.java: public class Point { // 点的三种状态 public static final int POINT_STATUS_NORMAL = 0; public static final int POINT_STATUS_CLICK = 1; public static final int POINT_STATUS

  • Android实现九宫格手势解锁

    本文为大家分享了Android九宫格手势解锁的具体代码,供大家参考,具体内容如下 这里是使用的开源库GestureLibray 里面有关于这个东西的介绍和接入方式,这里就不累赘了,我只是说下里面没有的. 关于这个库的使用: protected void initViews() { //设置模式 LockMode lockMode = (LockMode) getIntent().getSerializableExtra(Config.INTENT_SECONDACTIVITY_KEY); //是

  • Android实现九宫格手势密码

    本文实例为大家分享了vue + element ui实现锚点定位的具体代码,供大家参考,具体内容如下 介绍下自己编写的九宫格手势密码.先见图 思路:首先是9个格子,接着是格子连线:那么我们的步骤就有了. 1.手势监听,进行连线2.格子的状态未连接(初始状态).已连接的(没有结果前).错误状态(有结果后).(先这三个,可扩展,比如按下状态)3.自定义viewgroup作为九宫格的容器,里面包含9个view(小格子) 一.先从简单的说起吧,9个小格子以及状态 为了扩展性,不自定义view,将三个状态

  • Android自定义UI手势密码终结版

    之前写过3篇手势密码的demo,不过没有集成到真实的企业项目中,这几天正好领到一个手势密码项目,昨天刚好弄完,今天抽空整理下,目前还没有完善,有一些地方需要更改,不过基本的流程都可以跑通了. 源码下载地址:http://xiazai.jb51.net/201610/yuanma/AndroidGestureLock(jb51.net).rar 先看主界面的入口把.里面有2个button(一个是设置手势密码.一个是校验手势密码) activity_main.xml <RelativeLayout

  • Android自定义UI手势密码改进版

    接着第一个Android UI手势密码设计的基础上继续改进,效果图如下 activity_main.xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layo

  • Android自定义UI手势密码简单版

    先看看效果图: ImageLockActivity package com.example.imagelock; import com.example.view.NinePointLineView; import android.os.Bundle; import android.app.Activity; import android.view.Menu; import android.view.View; public class ImageLockActivity extends Acti

随机推荐