注:本文涉及的demo的地址:https://github.com/absfree/TouchDispatch

1. 触摸动作及事件序列

(1)触摸事件的动作

触摸动作一共有三种:ACTION_DOWN、ACTION_MOVE、ACTION_UP。当用户手指接触屏幕时,便产生一个动作为ACTION_DOWN的触摸事件,此时若用户的手指立即离开屏幕,会产生一个动作为ACTION_UP的触摸事件;若用户手指接触屏幕后继续滑动,当滑动距离超过了系统中预定义的距离常数,则产生一个动作为ACTION_MOVE的触摸事件,系统中预定义的用来判断用户手指在屏幕上的滑动是否是一个ACTION_MOVE动作的这个距离常量叫做TouchSlop,可通过ViewConfiguration.get(getContext()).getScaledTouchSlop()获取。

(2)事件序列

当用户的手指接触屏幕,在屏幕上滑动,又离开屏幕,这个过程会产生一系列触摸事件:ACTION_DOWN-->若干个ACTION_MOVE-->ACTION_UP。这一系列触摸事件即为一个事件序列。

2. 触摸事件的分发

(1)概述

当产生了一个触摸时间后,系统要负责把这个触摸事件给一个View(TargetView)来处理,touch事件传递到TargetView的过程即为touch事件的分发。

触摸事件的分发顺序:Activity-->顶级View-->顶级View的子View-->. . .-->Target View

触摸事件的响应顺序:TargetView --> TargetView的父容器 --> . . . -->顶级View -->Activity

(2)toush事件分发的具体过程

a. Activity对touch事件的分发

当用户手指接触屏幕时,便产生了一个touch事件,封装了touch事件的MotionEvent最先被传递给当前Activity,Activity的dispatchTouchEvent方法负责touch事件的分发。分发touch事件的实际工作由当前Activity的Window完成,而Window会将touch事件传递给DecorView(当前用户界面顶级View)。Activity的dispatchTouchEvent方法代码如下:

  1. 1 public boolean dispatchTouchEvent(MotionEvent ev) {
  2. 2 if (ev.getAction() == MotionEvent.ACTION_DOWN) {
  3. 3 onUserInteraction();
  4. 4 }
  5. 5 if (getWindow().superDispatchTouchEvent(ev)) {
  6. 6 return true;
  7. 7 }
  8. 8 return onTouchEvent(ev);
  9. 9 }

根据以上代码可以知道,touch事件会交由Window的superDispatchTouchEvent进行分发,若这个方法返回true,意味touch事件的分发过程结束,返回false则说明经过层层分发,没有子View对这个事件进行处理,即所有子View的onTouchEvent方法都返回false(即这个touch事件没有被“消耗”)。这时会调用Activity的onTouchEvent方法来处理这个touch事件。

在Window的superDispatchTouchEvent方法中,首先会把touch事件分发给DecorView,因为它是当前用户界面的顶级View。Window的superDispatchTouchEvent方法如下:

  1. 1 public abstract boolean superDispatchTouchEvent(MotionEvent ev);

是个抽象方法,这个方法由Window的实现类PhoneWindow实现,PhoneWindow的superDispatchTouchEvent方法的代码如下:

  1. 1 public boolean superDispatchTouchEvent(MotionEvent ev) {
  2. 2 return mDecor.superDispatchTouchEvent(event);
  3. 3 }

由以上代码可得,PhoneWindow的superDispatchTouchEvent方法实际上是通过DecorView的superDispatchTouchEvent方法来完成自己的工作,也就是说,当前Activity的Window直接将这个touch事件传递给了DecorView。也就是说,目前touch事件已经经过了如下的分发:Activity-->Window-->DecorView。

b. 顶级View对touch事件的分发

经过Activity与Window的分发,现在touch事件已经被传递到了DecorView的dispatchTouchEvent方法中。DecorView本质上是一个ViewGroup(更具体的说是FrameLayout),ViewGroup的dispatchTouchEvent方法所做的工作可以分为如下几个阶段,第一个阶段的主要代码如下:

  1. 1 //Handle an initial down.
  2. 2 if (actionMasked == MotionEvent.ACTION_DOWN) {
  3. 3 //Throw away all previous state when starting a new touch gesture.
  4. 4 //The framework may have dropped the up or cancel event for the previous gesture due to an app switch, ANR, or some other state change.
  5. 5 cancelAndClearTouchTargets(ev);
  6. 6 resetTouchState();
  7. 7 }

第一阶段的主要工作有俩:一是在第6行的resetTouchState方法中完成了对FLAG_DISALLOW_INTERCEPT标记的重置;二是第5行的cancelAndClearTouchTargets方法会清除当前MotionEvent的touch target。关于FLAG_DISALLOW_INTERCEPT标记和touch target,在下文会有相关说明。

第二阶段的主要工作是决定当前ViewGroup是否拦截本次的touch事件,主要代码如下:

  1. 1 //Check for interception.
  2. 2 final boolean intercepted;
  3. 3 if (actionMasked == MotionEvent.ACTION_DOWM || mFirstTouchTarget != null) {
  4. 4 final boolean disallowIntercept = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0;
  5. 5 if (!disallowIntercept) {
  6. 6 intercepted = onInterceptTouchEvent(ev);
  7. 7 ev.setAction(action); //restore action in case it was changed
  8. 8 } else {
  9. 9 intercepted = false;
  10. 10 }
  11. 11 } else {
  12. 12 //There are no touch targets and this action is not an initial down so this view group continues to intercept touches.
  13. 13 intercept =true;
  14. 14 }

由以上代码我们可以知道,当一个touch事件被传递到ViewGroup时,会先判断这个touch事件的动作是否是ACTION_DOWN,如果这个事件是ACTION_DOWN或者mFirstTouchTarget不为null,就会根据FLAG_DISALLOW_INTERCEPT标记决定是否拦截这个touch事件。那么mFirstTouchTarget是什么呢?当touch事件被ViewGroup的子View成功处理时,mFirstTouchTarget就会被赋值为成功处理touch事件的View,也就是上面提高的touch target。

总结一下上述代码的流程:在子View不干预ViewGroup的拦截的情况下(上述代码中的disallowIntercept为false),若当前事件为ACTION_DOWN或者mFirstTouchTarget不为空,则会调用ViewGroup的onInterceptTouchEvent方法来决定最终是否拦截此事件;否则(没有TargetView并且此事件不是ACTION_DOWN),当前ViewGroup就拦截下此事件。 一旦ViewGroup拦截了某次touch事件,那么mFirstTouchTarget就不会被赋值,因此当再有ACTION_MOVE或是ACTION_UP传递到该ViewGroup时,mTouchTarget就为null,所以上述代码第3行的条件就为false,ViewGroup会拦截下来。由此可得到的结论是:一旦ViewGroup拦截了某次事件,则同一事件序列中的剩余事件也会它默认被拦截而不会再询问是否拦截(即不会再调用onInterceptTouchEvent)。

这里存在一种特殊情形,就是子View通过requestDisallowInterceptTouchEvent方法设置父容器的FLAG_DISALLOW_INTERCEPT为true,这个标记指示是否不允许父容器拦截,为true表示不允许。这样做能够禁止父容器拦截除ACTION_DOWN以外的所有touch事件。之所以不能够拦截ACTION_DOWN事件,是因为每当ACTION_DOWN事件到来时,都会重置FLAG_DISALLOW_INTERCEPT这个标记位为默认值(false),所以每当开始一个新touch事件序列(即到来一个ACTION_DOWN动作),都会通过调用onInterceptTouchEven询问ViewGroup是否拦截此事件。当ACTION_DOWN事件到来时,重置标记位的工作是在上面的第一阶段完成的。

接下来,会进入第三阶段的工作:

  1. 1 final boolean canceled = resetCancelNextUpFlag(this) || actionMasked == MotionEvent.ACTION_CANCEL;
  2. 2 final boolean split = (mGroupFlags & FLAG_SPLIT_MOTION_EVENTS) != 0;
  3. 3 TouchTarget newTouchTarget = null;
  4. 4 boolean alreadyDispatchedToNewTouchTarget = false;
  5. 5 if (!canceled && !intercepted) {
  6. 6 // 不是ACTION_CANCEL并且不拦截
  7. 7 if (actionMasked == MotionEvent.ACTION_DOWN) {
  8. 8 // 若当前事件为ACTION_DOWN则去寻找这次事件新出现的touch target
  9. 9 final int actionIndex = ev.getActionIndex(); // always 0 for down
  10. 10
  11. 11 ...
  12. 12
  13. 13 final int childrenCount = mChildrenCount;
  14. 14 if (newTouchTarget == null && childrenCount != 0) {
  15. 15 // 根据触摸的坐标寻找能够接收这个事件的touch target
  16. 16 final float x = ev.getX(actionIndex);
  17. 17 final float y = ev.getY(actionIndex);
  18. 18
  19. 19 final View[] children = mChildren;
  20. 20 // 遍历所有子View
  21. 21 for (int i = childrenCount - 1; i >= 0; i--) {
  22. 22 final int childIndex = i;
  23. 23 final View child = children[childIndex];
  24. 24 // 寻找可接收这个事件并且touch事件坐标在其区域内的子View
  25. 25 if (!canViewReceivePointerEvents(child) || !isTransformedTouchPointInView(x, y, child, null)) {
  26. 26 continue;
  27. 27 }
  28. 28
  29. 29 newTouchTarget = getTouchTarget(child); // 找到了符合条件的子View,赋值给newTouchTarget
  30. 30 if (newTouchTarget != null) {
  31. 31 //Child is already receiving touch within its bounds.
  32. 32 //Give it the new pointer in addition to ones it is handling.
  33. 33 newTouchTarget.pointerIdBits |= idBitsToAssign;
  34. 34 break;
  35. 35 }
  36. 36 resetCancelNextUpFlag(child);
  37. 37 // 把ACTION_DOWN事件传递给子组件进行处理
  38. 38 if (dispatchTransformedTouchEvent(ev, false, child, idBitsToAssign)) {
  39. 39 //Child wants to receive touch within its bounds.
  40. 40 mLastTouchDownTime = ev.getDownTime();
  41. 41 if (preorderedList != null) {
  42. 42 //childIndex points into presorted list, find original index
  43. 43 for (int j=0;j<childrenCount;j++) {
  44. 44 if (children[childIndex]==mChildren[j]) {
  45. 45 mLastTouchDownIndex=j;
  46. 46 break;
  47. 47 }
  48. 48 }
  49. 49 } else {
  50. 50 mLastTouchDownIndex = childIndex;
  51. 51 }
  52. 52 mLastTouchDownX = ev.getX();
  53. 53 mLastTouchDownY = ev.getY();
  54. 54 //把mFirstTouchTarget赋值为newTouchTarget,此子View成为新的touch事件的起点
  55. 55 newTouchTarget = addTouchTarget(child, idBitsToAssign);
  56. 56 alreadyDispatchedToNewTouchTarget = true;
  57. 57 break;
  58. 58 }
  59. 59 }
  60. 60 }
  61. 61 }
  62. 62 }

当ViewGroup不拦截本次事件,则touch事件会分发给它的子View进行处理,相关代码从第21行开始:遍历所有ViewGroup的子View,寻找能够处理此touch事件的子View,若一个子View不在播放动画并且touch事件坐标位于其区域内,则该子View能够处理此touch事件,并且会把该子View赋值给newTouchTarget。

若当前遍历到的子View能够处理此touch事件,就会进入第38行的dispatchTransformedTouchEvent方法,该方法实际上调用了子View的dispatchTouchEvent方法。dispatchTransformedTouchEvent方法中相关的代码如下:

  1. 1 if (child == null) {
  2. 2 handled = super.dispatchTouchEvent(event);
  3. 3 } else {
  4. 4 handled = child.dispatchTouchEvent(event);
  5. 5 }

若dispatchTransformedTouchEvent方法传入的child参数不为null,则会调用child(即处理touch事件的子View)的dispatchTouchEvent方法。若该子View的dispatchTouchEvent方法返回true,则dispatchTransformedTouchEvent方法也会返回true,则表示成功找到了一个处理该事件的touch target,会在第55行把newTouchTarget赋值给mFirstTouchTarget(这一赋值过程是在addTouchTarget方法内部完成的),并跳出对子View遍历的循环。若子View的dispatchTouchEvent方法返回false,ViewGroup就会把事件分发给下一个子View。

若遍历了所有子View后,touch事件都没被处理(该ViewGroup没有子View或是所有子View的dispatchTouchEvent返回false),ViewGroup会自己处理touch事件,相关代码如下:

  1. 1 if (mFirstTouchTarget == null) {
  2. 2 handled = dispatchTransformedTouchEvent(ev, canceled, null, TouchTarget.ALL_POINTER_IDS);
  3. 3 }

由以上代码可知,ViewGroup自己处理touch事件时,会调用dispatchTransformedTouchEvent方法,传入的child参数为null。根据上文的分析,传入的chid为null时,会调用super.dispatchTouchEvent方法,即调用父类的dispatchTouchEvent方法,父类的dispatchTouchEvent方法中又包含了对onTouchEvent方法的调用,根据多态,实际上会回过头来调用当前View的onTouchEvent方法。(关于这点的详细说明请参考这篇文章:

c. View(不包含ViewGroup)对touch事件的处理

View的dispatchTouchEvent方法的主要代码如下:

  1. 1 public boolean dispatchTouchEvent(MotionEvent event) {
  2. 2 boolean result = false;
  3. 3 . . .
  4. 4
  5. 5 if (onFilterTouchEventForSecurity(event)) {
  6. 6 //noinspection SimplifiableIfStatement
  7. 7 ListenerInfo li = mListenerInfo;
  8. 8 if (li != null && li.mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED
  9. 9 && li.mOnTouchListener.onTouch(this, event)) {
  10. 10 result = true;
  11. 11 }
  12. 12
  13. 13 if (!result && onTouchEvent(event)) {
  14. 14 result = true;
  15. 15 }
  16. 16 . . .
  17. 17 return result;
  18. 18 }

由上述代码可知,View对touch事件的处理过程如下:由于View不包含子元素,所以它只能自己处理事件。它首先会判断是否设置了OnTouchListener,若设置了,会调用onTouch方法,若onTouch方法返回true(表示该touch事件已经被消耗),则不会再调用onTouchEvent方法;若onTouch方法返回false或没有设置OnTouchListener,则会调用onTouchEvent方法,onTouchEvent对touch事件进行具体处理的相关代码如下:

  1. 1 if (((viewFlags & CLICKABLE) == CLICKABLE || (viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE)) {
  2. 2 switch (event.getAction()) {
  3. 3 case MotionEvent.ACTION_UP:
  4. 4 boolean prepressed = (mPrivateFlags & PFLAG_PREPRESSED) != 0;
  5. 5 if ((mPrivateFlags & PFLAG_PRESSED) != 0 || prepressed) {
  6. 6 . . .
  7. 7 if (!mHasPerformedLongPress) {
  8. 8 //This is a tap, so remove the longpress check
  9. 9 removeLongPressCallback();
  10. 10
  11. 11 //Only perform take click actions if we were in the pressed state
  12. 12 if (!focusTaken) {
  13. 13 //Use a Runnable and post this rather than calling performClick directly.
  14. 14 //This lets other visual state of the view update before click actions start.
  15. 15 if (mPerformClick == null) {
  16. 16 mPerformClck = new PeformClick();
  17. 17 }
  18. 18 if (!post(mPerformClick)) {
  19. 19 performClick();
  20. 20 }
  21. 21 }
  22. 22 }
  23. 23 . . .
  24. 24 }
  25. 25 break;
  26. 26 }
  27. 27 . . .
  28. 28 return true;
  29. 29 }

由以上代码可知,只要View的CLICKABLE属性和LONG_CLICKABLE属性有一个为true(View的CLICKABLE属性和具体View有关,LONG_CLICKABLE属性默认为false,setOnClikListener和setOnLongClickListener会分别自动将以上俩属性设为true),那么这个View就会消耗这个touch事件,即使这个View处于DISABLED状态。若当前事件是ACTION_UP,还会调用performClick方法,该View若设置了OnClickListener,则performClick方法会在其内部调用onClick方法。performClick方法代码如下:

  1. 1 public boolean performClick() {
  2. 2 final boolean result;
  3. 3 final ListenerInfo li = mListenerInfo;
  4. 4 if (li != null && li.mOnClickListener != null) {
  5. 5 playSoundEffect(SoundEffectConstants.CLICK);
  6. 6 li.mOnClickListener.onClick(this);
  7. 7 result = true;
  8. 8 } else {
  9. 9 result = false;
  10. 10 }
  11. 11 sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_CLICKED);
  12. 12 return result;
  13. 13 }

3. 结合demo详解事件分发的各种情况

注:本文的demo参考自以下博文: http://hukai.me/android-deeper-touch-event-dispatch-process/

首先我们来看一下本文所使用的demo的布局情况:

以此为例,触摸事件的分发及响应顺序如下:

  • 分发:MainActivity -> ParentLayout -> ChildLayout -> CustomButton
  • 响应:CustomButton -> ChildLayotu -> ParentLayout -> MainActivity

从上面我们可以看到,分发的顺序是自顶向下,而响应的顺序是自底向上。我们可以把触摸事件看做是一项任务,而MainActivity看做是一个公司的老总,CustomButton看做底层的员工,这样来理解以下的叙述会更加容易。我们运行demo,然后在CustomButton点击一下而后松开,然后观察Log输出。

在讲解正常情况前,我们先来看假如把CustomButton删去会发生什么。删去CustomButton后的布局如下:

我们点击一下内层的ChildLayout并松开,可以得到以下日志输出:

这种情况是ChildLayout没有子View,根据我们上面讲解的这段代码:

  1. 1 if (child == null) {
  2. 2 handled = super.dispatchTouchEvent(event);
  3. 3 } else {
  4. 4 handled = child.dispatchTouchEvent(event);
  5. 5 }

此时会调用super.dispatchTouchEvent(event)。对于ChildLayout来说,它的父类是LinearLayout,所以会调用LinearLayout的dispatchTouchEvent方法,我们知道dispatchTouchEvent方法中包含着对onTouchEvent方法的调用,那么在LinearLayout的dispatchTouchEvent方法中对onTouchEvent方法的调用实际上会调用ChildLayout的onTouchEvent方法(根据多态)。也就是说,当一个ViewGroup没有子View或所有的子View都没消耗触摸事件时,实际上它会调用自己的onTouchEvent方法。

接下来,我们再恢复删除的CustomButton,针对各个情况进行具体的讲解。

(1)正常情况

在本例中,触摸事件分发响应的正常情况就是CustomButton消耗了触摸事件,然后层层向上返回true,最后到了MainAcitivity那便知道这次触摸事件被成功消耗了。这就好比老总下发任务,经过各级经理下发到员工后,员工圆满完成了任务,然后逐级上报最后汇报给了老总。我们来看一下日志输出:

(2)MainActivity的dispatchTouchEvent方法中对ACTION_DOWN返回true

这种情况下,ACTION_DOWN传到MainActivity的dispatchTouchEvent方法时,直接返回了true(被丢弃),因此没有找到TargetView,则后续的ACITON_MOVE、ACTION_UP会直接交由MainActivity的onTouchEvent处理,而它的onTouchEvent默认会返回false(没有能力消耗触摸事件)。我们来看一下日志输出:

(3)ParentLayout的dispatchTouchEvent方法中对ACTION_DOWN返回true

这种情况下,也是无法找到TargetView。由于没有TargetView,后续的ACTION_MOVE和ACTION_UP就只能下发到ParentLayout层了,而它无法消耗触摸事件。我们来看一下日志输出:

需要注意的是,触摸事件回传时,只有ParentLayout层和Activity层能够收到onTouchEvent的回调(假如ParentLayout还有一个父View,则该父View收不到onTouchEvent回调)。

(4)Activity的dispatchTouchEvent方法中对ACTION_MOVE返回true

这种情况下,ACTION_DOWN和ACTION_UP会被正常下发并被消耗,而ACTION_MOVE会在MainActivity即停止分发,日志如下:

注意,这里由于滑动距离未达到TouchSlop,所以并没有产生ACTION_MOVE事件,关于对ACTION_MOVE的处理大家可以参考(2)中对ACTION_DOWN的处理(直接在MainActivity层被丢弃)。

(5)ParentLayout的onInterceptTouchEvent方法中对ACTION_DOWN返回true

这种情况下,ACTION_DOWN分发到ParentLayout时被拦截,则会交给ParentLayout的onTouchEvent处理,而由于未找到TargetView,后续的ACTION_MOVE、ACTION_DOWN事件便无从下发,在MainActivity层就会停止下发。注意到onInterceptTouchEvent方法与dispatchTouchEvent方法的区别,前者若对ACTION_DOWN返回true,则同一事件序列的后续事件便在MainActivity层终止下发;后者若对ACTION_DOWN返回true,后续ACTION_MOVE及ACTION_UP会在返回true的那层终止下发,且回传时只有返回true的那层以及MainActivity层的onTouchEvent方法会被回掉。日志输出如下:

(6)ChildLayout的onInterceptTouchEvent方法中对ACTION_DOWN返回true

这种情况下,ACTION_DOWN会交由ChildLayout的onTouchEvent方法处理,而它默认会返回false,于是开始回传,经过ParentLayout、MainAcitivity均无法消费该事件。后续的事件会在MainAcitivity层便停止下发。日志如下:

(7)ChildLayout的onInterceptTouchEvent方法中对ACTION_MOVE返回true

这种情况下,ACTION_DOWN会被正常分发到CustomButton并被消费,CustomButton会成为TargetView。而ACTION_MOVE会被ChildLayout拦截,这时会产生一个ACTION_CANCEL消息(表示后续事件过不来了),这个消息会被分发给CustomButton,CustomButton会消耗它。后面若还有ACTION_MOVE会交给ChildLayout的onTouchEvent,由于无法消耗,会返回false。后续的ACTION_UP只能分发到ChildLayout。这种情况要和拦截ACTION_DOWN区分开,由于有TargetView,所以ACTION_UP能传递到拦截ACTION_MOVE的那层而不是终止在MainAcitivity层。日志输出如下:

  1. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [dispatchTouchEvent] -> ACTION_DOWN
  2. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [dispatchTouchEvent] -> ACTION_DOWN
  3. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [onInterceptTouchEvent] -> ACTION_DOWN
  4. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [onInterceptTouchEvent] return false
  5. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [dispatchTouchEvent] -> ACTION_DOWN
  6. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [onInterceptTouchEvent] -> ACTION_DOWN
  7. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [onInterceptTouchEvent] return false
  8. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch D/CustomButton: [dispatchTouchEvent] -> ACTION_DOWN
  9. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch D/CustomButton: [onTouchEvent] -> ACTION_DOWN
  10. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch I/CustomButton: [onTouchEvent] return true
  11. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch I/CustomButton: [dispatchTouchEvent] return true
  12. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [dispatchTouchEvent] return true
  13. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [dispatchTouchEvent] return true
  14. 04-05 03:15:40.756 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [dispatchTouchEvent] return true
  15. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [dispatchTouchEvent] -> ACTION_MOVE
  16. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [dispatchTouchEvent] -> ACTION_MOVE
  17. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [onInterceptTouchEvent] -> ACTION_MOVE
  18. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [onInterceptTouchEvent] return false
  19. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [dispatchTouchEvent] -> ACTION_MOVE
  20. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [onInterceptTouchEvent] -> ACTION_MOVE, return true
  21. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch D/CustomButton: [dispatchTouchEvent] -> ACTION_CANCEL
  22. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch D/CustomButton: [onTouchEvent] -> ACTION_CANCEL
  23. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch I/CustomButton: [onTouchEvent] return true
  24. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch I/CustomButton: [dispatchTouchEvent] return true
  25. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [dispatchTouchEvent] return true
  26. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [dispatchTouchEvent] return true
  27. 04-05 03:15:40.872 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [dispatchTouchEvent] return true
  28. 04-05 03:15:40.892 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [dispatchTouchEvent] -> ACTION_MOVE
  29. 04-05 03:15:40.892 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [dispatchTouchEvent] -> ACTION_MOVE
  30. 04-05 03:15:40.892 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [onInterceptTouchEvent] -> ACTION_MOVE
  31. 04-05 03:15:40.892 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [onInterceptTouchEvent] return false
  32. 04-05 03:15:40.892 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [dispatchTouchEvent] -> ACTION_MOVE
  33. 04-05 03:15:40.892 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [onTouchEvent] -> ACTION_MOVE
  34. 04-05 03:15:40.892 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [onTouchEvent] return false
  35. 04-05 03:15:40.892 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [dispatchTouchEvent] return false
  36. 04-05 03:15:40.892 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [dispatchTouchEvent] return false
  37. 04-05 03:15:40.892 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [onTouchEvent] -> ACTION_MOVE
  38. 04-05 03:15:40.892 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [onTouchEvent] return false
  39. 04-05 03:15:40.892 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [dispatchTouchEvent] return false
  40. 04-05 03:15:40.908 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [dispatchTouchEvent] -> ACTION_MOVE
  41. 04-05 03:15:40.908 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [dispatchTouchEvent] -> ACTION_MOVE
  42. 04-05 03:15:40.908 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [onInterceptTouchEvent] -> ACTION_MOVE
  43. 04-05 03:15:40.908 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [onInterceptTouchEvent] return false
  44. 04-05 03:15:40.908 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [dispatchTouchEvent] -> ACTION_MOVE
  45. 04-05 03:15:40.908 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [onTouchEvent] -> ACTION_MOVE
  46. 04-05 03:15:40.908 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [onTouchEvent] return false
  47. 04-05 03:15:40.908 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [dispatchTouchEvent] return false
  48. 04-05 03:15:40.908 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [dispatchTouchEvent] return false
  49. 04-05 03:15:40.908 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [onTouchEvent] -> ACTION_MOVE
  50. 04-05 03:15:40.908 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [onTouchEvent] return false
  51. 04-05 03:15:40.908 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [dispatchTouchEvent] return false
  52. 04-05 03:15:40.924 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [dispatchTouchEvent] -> ACTION_MOVE
  53. 04-05 03:15:40.924 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [dispatchTouchEvent] -> ACTION_MOVE
  54. 04-05 03:15:40.924 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [onInterceptTouchEvent] -> ACTION_MOVE
  55. 04-05 03:15:40.924 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [onInterceptTouchEvent] return false
  56. 04-05 03:15:40.924 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [dispatchTouchEvent] -> ACTION_MOVE
  57. 04-05 03:15:40.924 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [onTouchEvent] -> ACTION_MOVE
  58. 04-05 03:15:40.924 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [onTouchEvent] return false
  59. 04-05 03:15:40.924 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [dispatchTouchEvent] return false
  60. 04-05 03:15:40.924 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [dispatchTouchEvent] return false
  61. 04-05 03:15:40.932 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [onTouchEvent] -> ACTION_MOVE
  62. 04-05 03:15:40.932 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [onTouchEvent] return false
  63. 04-05 03:15:40.932 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [dispatchTouchEvent] return false
  64. 04-05 03:15:40.940 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [dispatchTouchEvent] -> ACTION_MOVE
  65. 04-05 03:15:40.940 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [dispatchTouchEvent] -> ACTION_MOVE
  66. 04-05 03:15:40.940 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [onInterceptTouchEvent] -> ACTION_MOVE
  67. 04-05 03:15:40.940 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [onInterceptTouchEvent] return false
  68. 04-05 03:15:40.944 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [dispatchTouchEvent] -> ACTION_MOVE
  69. 04-05 03:15:40.944 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [onTouchEvent] -> ACTION_MOVE
  70. 04-05 03:15:40.944 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [onTouchEvent] return false
  71. 04-05 03:15:40.944 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [dispatchTouchEvent] return false
  72. 04-05 03:15:40.944 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [dispatchTouchEvent] return false
  73. 04-05 03:15:40.944 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [onTouchEvent] -> ACTION_MOVE
  74. 04-05 03:15:40.948 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [onTouchEvent] return false
  75. 04-05 03:15:40.948 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [dispatchTouchEvent] return false
  76. 04-05 03:15:40.972 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [dispatchTouchEvent] -> ACTION_MOVE
  77. 04-05 03:15:40.972 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [dispatchTouchEvent] -> ACTION_MOVE
  78. 04-05 03:15:40.972 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [onInterceptTouchEvent] -> ACTION_MOVE
  79. 04-05 03:15:40.972 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [onInterceptTouchEvent] return false
  80. 04-05 03:15:40.972 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [dispatchTouchEvent] -> ACTION_MOVE
  81. 04-05 03:15:40.972 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [onTouchEvent] -> ACTION_MOVE
  82. 04-05 03:15:40.976 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [onTouchEvent] return false
  83. 04-05 03:15:40.976 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [dispatchTouchEvent] return false
  84. 04-05 03:15:40.976 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [dispatchTouchEvent] return false
  85. 04-05 03:15:40.976 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [onTouchEvent] -> ACTION_MOVE
  86. 04-05 03:15:40.980 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [onTouchEvent] return false
  87. 04-05 03:15:40.980 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [dispatchTouchEvent] return false
  88. 04-05 03:15:40.980 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [dispatchTouchEvent] -> ACTION_UP
  89. 04-05 03:15:40.984 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [dispatchTouchEvent] -> ACTION_UP
  90. 04-05 03:15:40.984 30055-30055/com.yxy.zlp.touchdispatch D/ParentLayout: [onInterceptTouchEvent] -> ACTION_UP
  91. 04-05 03:15:40.984 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [onInterceptTouchEvent] return false
  92. 04-05 03:15:40.984 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [dispatchTouchEvent] -> ACTION_UP
  93. 04-05 03:15:40.984 30055-30055/com.yxy.zlp.touchdispatch D/ChildLayout: [onTouchEvent] -> ACTION_UP
  94. 04-05 03:15:40.988 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [onTouchEvent] return false
  95. 04-05 03:15:40.988 30055-30055/com.yxy.zlp.touchdispatch I/ChildLayout: [dispatchTouchEvent] return false
  96. 04-05 03:15:40.988 30055-30055/com.yxy.zlp.touchdispatch I/ParentLayout: [dispatchTouchEvent] return false
  97. 04-05 03:15:40.988 30055-30055/com.yxy.zlp.touchdispatch D/MainActivity: [onTouchEvent] -> ACTION_UP
  98. 04-05 03:15:40.992 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [onTouchEvent] return false
  99. 04-05 03:15:40.992 30055-30055/com.yxy.zlp.touchdispatch I/MainActivity: [dispatchTouchEvent] return false

总结一下:若某一层的dispatchTouchEvent方法对ACTION_DOWN返回true,则被认为该层能够消耗触摸事件,因此后续的事件会传到该层;而某一层的onInterceptTouchEvent方法返回true的话,由于它的onTouchEvent返回false进而导致它的dispatchTouchEvent方法返回false,则该层会被认为不能消耗触摸事件,所以后续事件只会到Activity层。

4. 参考资料

(1)《Android开发艺术探索》

(2)Touch事件分发响应机制

一个demo让你彻底理解Android触摸事件的并发的更多相关文章

  1. 一个demo让你彻底理解Android中触摸事件的分发

    注:本文涉及的demo的地址:https://github.com/absfree/TouchDispatch 1. 触摸动作及事件序列 (1)触摸事件的动作 触摸动作一共有三种:ACTION_DOW ...

  2. 对于android触摸事件模型的一些理解

    body{ font-family: "Microsoft YaHei UI","Microsoft YaHei",SimSun,"Segoe UI& ...

  3. android自定义控件(9)-Android触摸事件分发机制

    触摸事件的传递机制:   首先是最外层的viewgroup接收到事件,然后调用会调用自己的dispatchTouchEvent方法.如果在ACTION_DOWN的时候dispatchTouchEven ...

  4. 初识Android触摸事件传递机制

    前言 今天总结的一个知识点是Andorid中View事件传递机制,也是核心知识点,相信很多开发者在面对这个问题时候会觉得困惑,另外,View的另外一个难题滑动冲突,比如在ScrollView中嵌套Li ...

  5. Android触摸事件的应用

    前言 上一篇讲了Android触摸事件的传递机制,具体可以看这里 初识Android触摸事件传递机制.既然知道Android中触摸事件的传递分发,那么它能解决什么样的问题,在我们实际开发中如何应用,这 ...

  6. Android触摸事件传递机制

    简单梳理一下Android触摸事件传递机制的知识点. 一.View与ViewGroup的关系 View和ViewGroup二者的继承关系如下图所示: View是Android中最基本的一种UI组件,它 ...

  7. iOS 和 Android 触摸事件传递

    先看文章,写得很好 ios 触摸事件传递 http://www.cnblogs.com/Quains/p/3369132.html 另外一篇 http://blog.csdn.net/yongyinm ...

  8. Android触摸事件

    简单介绍: 做了一个语音发送UI的小demo. 按下显示语音窗体,依据音量调节UI音量显示,上划至窗体显示取消发送. 原理: 1:获取什么事件来运行操作: 给Button加入setOnTouchLis ...

  9. 【知识梳理1】Android触摸事件机制

    前言 随着科学技术的发展,智能手机早已成为我们当代人身边不可缺少的"伙伴"之中的一个,堪比对象女友.每天我们对着手机反复的做着点击.滑动操作,而手机则随着我们的操作给我们展示她的精 ...

随机推荐

  1. 禁止复制放在js文件中

    1.使右键和复制失效 方法1: 在网页中加入以下代码: 复制代码代码如下: <script language="Javascript"> document.oncont ...

  2. 自动生成查找组件的lua代码

    本篇主要解决的问题是使用lua脚本编写unity业务逻辑时,自动生成一些查找组件及绑定控件事件的lua代码! 现在很多unity项目都是用ulua作为热更新解决方案,因此需要用lua来写相关的逻辑,经 ...

  3. iOS相册中图片按照时间排序

    ios相册默认是按照时间从过去到现在排列,图片顺序有正序和逆序,group可以用以下方法来选择顺序 /** @param NSIndexSet 需要获取的相册中图片范围 @param NSEnumer ...

  4. Linux命令小结:crontab/netstat/iostat/sar

    crontab cron可以设定在指定的时间运行任务. 1.查看定时任务 [root@client1 ~]# crontab -l -u root */1 * * * * date >> ...

  5. [Tools] Vim插件管理

    我们在使用插件的时候,都不希望插件安装的很杂乱,它不是一个看不见的黑盒,也为了下次方便在其它地方安装. 由于要方便插件管理,于是有了 Vundle,以下做些介绍: 1. 一个插件管理器, 自己本身也是 ...

  6. Subliem Text 3 的安装和使用

    前两天将Sublime Text3简单的看了看,发现是好经典的开发工具.... 1. sublime Text安装:www.sublimetext.com 此时的版本是: Build 3103 可用的 ...

  7. JQuery初体验

    虽然做b/s也有一年半了,但是还没怎么认真的去看JQuery,趁自己生病的这几天,恶补一下JQuery方面的知识,保持学习的态度,内容很简单,聊以自慰一下>_<.废话不多说,直接上代码了. ...

  8. Numpy 学习之路(1)——数组的创建

    数组是Numpy操作的主要对象,也是python数据分析的主要对象,本系列文章是本人在学习Numpy中的笔记. 文章中以下都基于以下方式的numpy导入: import numpy as np fro ...

  9. 一个Angular模块中可以声明哪些组件?

    一个Angular模块中可以声明哪些组件? (1) controller        控制器 (2) directive                指令 (3) function         ...

  10. 关于MYSQL四种引擎

    你能用的数据库引擎取决于mysql在安装的时候是如何被编译的.要添加一个新的引擎,就必须重新编译MYSQL.在缺省情况下,MYSQL支持三个引擎:ISAM.MYISAM和HEAP.另外两种类型INNO ...