胡凯 · 更新于 2018-10-20 21:00:47

处理多点触控手势

编写:Andrwyw - 原文:http://developer.android.com/training/gestures/multi.html

多点触控手势是指在同一时间有多点(手指)触碰屏幕。本节课程讲述,如何检测涉及多点的触摸手势。

追踪多点

当多个手指同时触摸屏幕时,系统会产生如下的触摸事件:

  • ACTION_DOWN - 针对触摸屏幕的第一个点。此事件是手势的开端。第一触摸点的数据在MotionEvent中的索引总是0。
  • ACTION_POINTER_DOWN - 针对第一点后,出现在屏幕上额外的点。这个点的数据在MotionEvent中的索引,可以通过getActionIndex()获得。
  • ACTION_MOVE - 在按下手势期间发生变化。
  • ACTION_POINTER_UP - 当非主要点(non-primary pointer)离开屏幕时,发送此事件。
  • ACTION_UP - 当最后一点离开屏幕时发送此事件。

我们可以通过各个点的索引以及id,单独地追踪MotionEvent中的每个点。

  • IndexMotionEvent把各个点的信息都存储在一个数组中。点的索引值就是它在数组中的位置。大多数用来与点交互的MotionEvent函数都是以索引值而不是点的ID作为参数的。
  • ID:每个点也都有一个ID映射,该ID映射在整个手势期间一直存在,以便我们单独地追踪每个点。

每个独立的点在移动事件中出现的次序是不固定的。因此,从一个事件到另一个事件,点的索引值是可以改变的,但点的ID在它的生命周期内是保证不会改变的。使用getPointerId()可以获得一个点的ID,在手势随后的移动事件中,就可以用该ID来追踪这个点。对于随后一系列的事件,可以使用findPointerIndex()函数,来获得对应给定ID的点在移动事件中的索引值。如下:

private int mActivePointerId;

public boolean onTouchEvent(MotionEvent event) {
    ....
    // Get the pointer ID
    mActivePointerId = event.getPointerId(0);

    // ... Many touch events later...

    // Use the pointer ID to find the index of the active pointer
    // and fetch its position
    int pointerIndex = event.findPointerIndex(mActivePointerId);
    // Get the pointer's current position
    float x = event.getX(pointerIndex);
    float y = event.getY(pointerIndex);
}

获取MotionEvent的动作

我们应该总是使用getActionMasked()函数(或者用MotionEventCompat.getActionMasked()这个兼容版本更好)来获取MotionEvent的动作(action)。与旧的getAction()函数不同的是,getActionMasked()是设计用来处理多点触摸的。它会返回执行过的动作的掩码值,不包括点的索引位。然后,我们可以使用getActionIndex()来获得与该动作关联的点的索引值。这在接下来的代码段中可以看到。

Note: 这个样例使用的是MotionEventCompat类。该类在Support Library中。我们应该使用MotionEventCompat类,来提供对更多平台的支持。需要注意的一点是,MotionEventCompat并不是MotionEvent类的替代品。准确来说,它提供了一些静态工具类函数,我们可以把MotionEvent对象作为参数传递给这些函数,来得到与事件相关的动作。

int action = MotionEventCompat.getActionMasked(event);
// Get the index of the pointer associated with the action.
int index = MotionEventCompat.getActionIndex(event);
int xPos = -1;
int yPos = -1;

Log.d(DEBUG_TAG,"The action is " + actionToString(action));

if (event.getPointerCount() > 1) {
    Log.d(DEBUG_TAG,"Multitouch event");
    // The coordinates of the current screen contact, relative to
    // the responding View or Activity.
    xPos = (int)MotionEventCompat.getX(event, index);
    yPos = (int)MotionEventCompat.getY(event, index);

} else {
    // Single touch event
    Log.d(DEBUG_TAG,"Single touch event");
    xPos = (int)MotionEventCompat.getX(event, index);
    yPos = (int)MotionEventCompat.getY(event, index);
}
...

// Given an action int, returns a string description
public static String actionToString(int action) {
    switch (action) {

        case MotionEvent.ACTION_DOWN: return "Down";
        case MotionEvent.ACTION_MOVE: return "Move";
        case MotionEvent.ACTION_POINTER_DOWN: return "Pointer Down";
        case MotionEvent.ACTION_UP: return "Up";
        case MotionEvent.ACTION_POINTER_UP: return "Pointer Up";
        case MotionEvent.ACTION_OUTSIDE: return "Outside";
        case MotionEvent.ACTION_CANCEL: return "Cancel";
    }
    return "";
}

关于多点触摸的更多内容以及示例,可以查看拖拽与缩放章节。

上一篇: 滚动手势动画 下一篇: 拖拽与缩放