Android手势识别功能

现在智能手机基本都是触摸操作,点击按钮是一种交互方式,同时手势相关的操作,比如滑动等等同样是很重要的交互方式。这篇文章是对安卓手势交互相关知识点的整理和总结,主要来源基于官方文档。

触摸交互中的概念

常用事件

首先要了解一些常用的事件:
ACTION_DOWN:第一个手指按下
ACTION_UP:第一个手指抬起
ACTION_POINTER_DOWN:第二、三、四等等手指按下
ACTION_POINTER_UP: 第二、三、四等等手指抬起
ACTION_MOVE: 手指移动
ACTION_OUTSIDE:手指移出了屏幕
ACTION_CANCEL:收到前驱事件比如ACTION_DOWN后,后续事件被父控件拦截的情况下产生

上面我们可以看到,除了第一个手指有唯一的action down和action up事件触发,后续其它手指的按下和移动,都触发的是同一个事件。那么这个时候就可能涉及到对不同手指区分的逻辑处理。

MotionEvent

MotionEvent中用action code和坐标值描述了触摸运动的轨迹,action code值描述了运动状态的改变,坐标值描述了轨迹的位置和一起其它信息。
比如 ACTION_DOWN表明手指开始触碰到屏幕,X和Y的坐标轴值表明了当前的位置。

上面仅仅是基本的单指操作,但是现在很多设备都提供多指操作的功能。多个手指每个手指都被在第一次触碰屏幕的时候分配一个pointer id,直到这个手指离开相应的pointer id才变无效。当第一个手指按下时,会触发ACTION_DOWN,ACTION_MOVE一系列的事件,同时当第二个手指按下的时候,又会触发 ACTION_POINTER_DOWN事件,此后两个手指移动的时候,只会触发ACTION_MOVE事件。当一个ACTION_MOVE触发的时,通过使用 getPointerId(第几个手指) 方法去获取pointer id明确是哪一个手指,然后使用使用findPointerIndex 方法去获得pointer index,pointer index代表了这一个MotionEvent事件中哪一个是当前pointer对应的事件。

MotionEvent事件捆绑

结合上面的概念,再来说一下MotionEvent的捆绑。为了处理效率,安卓中会把MOVE动作中多个坐标点捆绑在一个MotionEvent中,对于单个手指操作,getX返回的是最近一点的坐标,getHistoricalX 返回的是之前的坐标。看下面一段代码:

 void printSamples(MotionEvent ev) {
   //获取MotionEvent中捆绑的坐标点
   final int historySize = ev.getHistorySize();
   //获取手指数目
   final int pointerCount = ev.getPointerCount();
   for (int h = 0; h < historySize; h++) {
     System.out.printf("At time %d:", ev.getHistoricalEventTime(h));
     for (int p = 0; p < pointerCount; p++) {
       System.out.printf(" pointer %d: (%f,%f)",
         ev.getPointerId(p), ev.getHistoricalX(p, h), ev.getHistoricalY(p, h));
     }
   }
 }

可以看到,一个MotionEvent中,可能包括多个手指的动作信息,以及一些历史信息。

事件分发机制

MotionEvent代表触摸后响应的事件,安卓中的视图是按照视图树构建而成的,点击之后,会生成点击事件MotionEvent并沿树传递。

与事件分发有关的方法有:
public boolean dispatchTouchEvent(MotionEvent ev) 事件分发
public boolean onInterceptTouchEvent(MotionEvent ev) 事件拦截
public boolean onTouchEvent(MotionEvent ev) 事件响应

在一个ViewGroup中通常会具有以上三个方法,可以进行事件的分发、拦截和响应,而在一个View中因为没有子View,所以只能进行事件的处理,也就只有onTouchEvent方法。

dispatchTouchEvent

事件分发的过程中,会以深度遍历的方式进行分发。分以下情况:

返回true,则事件会分发给当前View,由当前View消费。
返回false,将事件返回给父View进行消费
默认 super.dispatchTouchEvent(ev),会调用当前View的 onInterceptTouchEvent 进行拦截处理。
一般情况下,我们不会去重写view的分发过程,而是着重处理事件的拦截和响应。

onInterceptTouchEvent

如果返回true,则拦截当前事件,交由onTouchEvent处理
如果返回false,则不拦截当前事件,交由子View的dispatchTouchEvent处理
如果调用默认 super.onInterceptTouchEvent,则拦截当前事件。

onTouchEvent

如果返回false,表明当前View无法处理,之间会返回上级有上级View的onTouchEvent处理,一直 向上传递直到事件被消费。
如果返回true则会接收并消费该事件
如果返回 super.onTouchEvent(ev) 默认处理事件的逻辑和返回 false 时相同。
注意,对于View而非ViewGroup来说,只具有onTouchEvent方法。所以在一个View中,处理事件响应的典型代码如下:

public class MainActivity extends Activity {
...
// This example shows an Activity, but you would use the same approach if
// you were subclassing a View.
@Override
public boolean onTouchEvent(MotionEvent event){

  int action = MotionEventCompat.getActionMasked(event);

  switch(action) {
    case (MotionEvent.ACTION_DOWN) :
      Log.d(DEBUG_TAG,"Action was DOWN");
      return true;
    case (MotionEvent.ACTION_MOVE) :
      Log.d(DEBUG_TAG,"Action was MOVE");
      return true;
    case (MotionEvent.ACTION_UP) :
      Log.d(DEBUG_TAG,"Action was UP");
      return true;
    case (MotionEvent.ACTION_CANCEL) :
      Log.d(DEBUG_TAG,"Action was CANCEL");
      return true;
    case (MotionEvent.ACTION_OUTSIDE) :
      Log.d(DEBUG_TAG,"Movement occurred outside bounds " +
          "of current screen element");
      return true;
    default :
    //当前View不处理事件,交由上层处理。
      return super.onTouchEvent(event);
  }
}

同样,一个View处理触摸事件,还可以设置监听器onTouchListener,不过要注意的是onTouchListener的优先级比onTouch要高,如果其中返回了true,那么将不会调用onTouch方法。

手势探测

onTouch中我们可以通过MotionEvent获取触摸点的坐标信息,但是关于某些手势比如点击、滑动还需要进行我们自己的逻辑处理。在这里Android本身提供了一些手势判别的功能。这样在onTouch方法中,我们只需要把MotionEvent传递给手势监听器处理即可,同时实现接口中相应的回调方法:

private GestureDetectorCompat mDetector;
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    mDetector = new GestureDetectorCompat(this,this);
    mDetector.setOnDoubleTapListener(this);
  }
  @Override
  public boolean onTouchEvent(MotionEvent event){
    this.mDetector.onTouchEvent(event);
    // Be sure to call the superclass implementation
    return super.onTouchEvent(event);
  }

如果不需要监听那么多事件,那么可以写一个监听类继承GestureDetector.SimpleOnGestureListener并实现其中的方法。

如果要监听触摸的速度,那么可以通过VelocityTracker来监听:

    switch(action) {
      case MotionEvent.ACTION_DOWN:
        if(mVelocityTracker == null) {
          mVelocityTracker = VelocityTracker.obtain();
        }
        else {
          mVelocityTracker.clear();
        }
        mVelocityTracker.addMovement(event);
        break;
      case MotionEvent.ACTION_MOVE:
        mVelocityTracker.addMovement(event);
        mVelocityTracker.computeCurrentVelocity(1000);
        Log.d("", "X velocity: " +
            VelocityTrackerCompat.getXVelocity(mVelocityTracker,
            pointerId));
        Log.d("", "Y velocity: " +
            VelocityTrackerCompat.getYVelocity(mVelocityTracker,
            pointerId));
        break;
      case MotionEvent.ACTION_UP:
      case MotionEvent.ACTION_CANCEL:
        mVelocityTracker.recycle();
        break;

通过将MotionEvent加入VelocityTracker中,可以通过computeCurrentVelocity算出速度。
(未完待续。。。)

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • android开发之为activity增加左右手势识别示例

    android开发中为activity增加左右手势识别,如右滑关闭当前页面. 复制代码 代码如下: /* *  for左右手势 *  1.复制下面的内容到目标Activity *  2.目标Activity的onCreate()调用initGesture() *  3.目标Activity需implements OnTouchListener, OnGestureListener */   private GestureDetector mGestureDetector;   private i

  • Android基础开发之手势识别

    由于精确度等原因,手势识别在android中用的并不多,不过这并不妨碍我们来玩玩这个神奇的玩意. 在android中要使用手势,先得建立手势库,建立手势库非常简单,新建一个android sample project,建一个android示例工程,然后选择创建的android版本,完了之后看到这个界面: 选择gesturebuilder,创建成功之后把它安装到真机上,然后可以在里边添加手势,并给手势命名. 创建完gesture之后,在eclipse的file explore窗口中查看系统文件,在

  • Android View进行手势识别详解

    我们在进行Android游戏开发时会用到很多种控制,包括前面讲到的按键和轨迹球控制方式,除此之外还有手势操作.重力感应等多种控制方式需要了解掌握.本节主要为大家讲解在View中如何进行手势识别. 很多网友发现Android中手势识别提供了两个类,由于Android 1.6以下的版本比如cupcake中无法使用android.view.GestureDetector,而android.gesture.Gesture是Android 1.6开始支持的,考虑到仍然有使用Android 1.5固件的网友

  • Android应用开发中触摸屏手势识别的实现方法解析

    很多时候,利用触摸屏的Fling.Scroll等Gesture(手势)操作来操作会使得应用程序的用户体验大大提升,比如用Scroll手势在 浏览器中滚屏,用Fling在阅读器中翻页等.在Android系统中,手势的识别是通过 GestureDetector.OnGestureListener接口来实现的,不过William翻遍了Android的官方文档也没有找到一个相 关的例子,API Demo中的TouchPaint也仅仅是提到了onTouch事件的处理,没有涉及到手势. 我们先来明确一些概念

  • 深入理解Android手势识别

    对于触摸屏,其原生的消息无非按下.抬起.移动这几种,我们只需要简单重载onTouch或者设置触摸侦听器setOnTouchListener即可进行处理.不过,为了提高我们的APP的用户体验,有时候我们需要识别用户的手势,Android给我们提供的手势识别工具GestureDetector就可以帮上大忙了. 基础 GestureDetector的工作原理是,当我们接收到用户触摸消息时,将这个消息交给GestureDetector去加工,我们通过设置侦听器获得GestureDetector处理后的手

  • 札记:android手势识别功能实现(利用MotionEvent)

    摘要 本文是手势识别输入事件处理的完整学习记录.内容包括输入事件InputEvent响应方式,触摸事件MotionEvent的概念和使用,触摸事件的动作分类.多点触摸.根据案例和API分析了触摸手势Touch Gesture的识别处理的一般过程.介绍了相关的GestureDetector,Scroller和VelocityTracker.最后分析drag和scale等一些手势的识别. 输入源分类 虽然android本身是一个完整的系统,它主要运行在移动设备的特性决定了我们在它上面开的app绝大数

  • android创建手势识别示例代码

    这篇的内容使用到的是android.gesture包,具体的例子参考的是Sample中GestureBuilder程序. 1.手势创建手势创建主要用到GestureOverlayView和GestureLibrary.GestureOverlayView的父类为android.widget.FrameLayout,是手势绘图区.GestureLibrary类主要对手势进行保存.删除等操作的,存放手势的仓库.下面给出创建手势的例子,如下图,可以定义如图手势打开csdn.net 1.1.创建绘图区

  • android使用gesturedetector手势识别示例分享

    复制代码 代码如下: public class MyGestureLintener extends SimpleOnGestureListener {private Context context;public MyGestureLintener(Context context) {    super();    this.context = context;} // 单击,触摸屏按下时立刻触发/*@Overridepublic boolean onDown(MotionEvent e) {  

  • 理解Android的手势识别提高APP的用户体验

    对于触摸屏,其原生的消息无非按下.抬起.移动这几种,我们只需要简单重载onTouch或者设置触摸侦听器setOnTouchListener即可进行处理.不过,为了提高我们的APP的用户体验,有时候我们需要识别用户的手势,Android给我们提供的手势识别工具GestureDetector就可以帮上大忙了. 基础 GestureDetector的工作原理是,当我们接收到用户触摸消息时,将这个消息交给GestureDetector去加工,我们通过设置侦听器获得GestureDetector处理后的手

  • Android实现Gesture手势识别用法分析

    本文实例分析了Android实现Gesture手势识别用法.分享给大家供大家参考.具体如下: 很高兴能在Android1.6的sdk看到手势识别这一功能,之前一直在想,如何在android中实现nds游戏那样用手势(准确点应该是笔势)来控制游戏角色?现在总算看到一点曙光了,不过手势要做到笔势那样随心所欲地控制游戏人物,还有很多细节问题需要处理. 在Android1.6的模拟器里面预装了一个叫Gestures Builder的程序,这个程序就是让你创建自己的手势的(Gestures Builder

随机推荐