处理多点触控手势

编写: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的点在移动事件中的索引值。如下:

  1. private int mActivePointerId;
  2. public boolean onTouchEvent(MotionEvent event) {
  3. ....
  4. // Get the pointer ID
  5. mActivePointerId = event.getPointerId(0);
  6. // ... Many touch events later...
  7. // Use the pointer ID to find the index of the active pointer
  8. // and fetch its position
  9. int pointerIndex = event.findPointerIndex(mActivePointerId);
  10. // Get the pointer's current position
  11. float x = event.getX(pointerIndex);
  12. float y = event.getY(pointerIndex);
  13. }

获取MotionEvent的动作

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

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

  1. int action = MotionEventCompat.getActionMasked(event);
  2. // Get the index of the pointer associated with the action.
  3. int index = MotionEventCompat.getActionIndex(event);
  4. int xPos = -1;
  5. int yPos = -1;
  6. Log.d(DEBUG_TAG,"The action is " + actionToString(action));
  7. if (event.getPointerCount() > 1) {
  8. Log.d(DEBUG_TAG,"Multitouch event");
  9. // The coordinates of the current screen contact, relative to
  10. // the responding View or Activity.
  11. xPos = (int)MotionEventCompat.getX(event, index);
  12. yPos = (int)MotionEventCompat.getY(event, index);
  13. } else {
  14. // Single touch event
  15. Log.d(DEBUG_TAG,"Single touch event");
  16. xPos = (int)MotionEventCompat.getX(event, index);
  17. yPos = (int)MotionEventCompat.getY(event, index);
  18. }
  19. ...
  20. // Given an action int, returns a string description
  21. public static String actionToString(int action) {
  22. switch (action) {
  23. case MotionEvent.ACTION_DOWN: return "Down";
  24. case MotionEvent.ACTION_MOVE: return "Move";
  25. case MotionEvent.ACTION_POINTER_DOWN: return "Pointer Down";
  26. case MotionEvent.ACTION_UP: return "Up";
  27. case MotionEvent.ACTION_POINTER_UP: return "Pointer Up";
  28. case MotionEvent.ACTION_OUTSIDE: return "Outside";
  29. case MotionEvent.ACTION_CANCEL: return "Cancel";
  30. }
  31. return "";
  32. }

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