前言
通过对ViewGroup与View的dispatchTouchEvent()源码简要分析看View的事件分发原理.
ViewGroup的dispatchTouchEvent()
局部变量handled代表是否处理,初始化为false.
1.检测View是否安全.
onFilterTouchEventForSecurity(),主要是判断View有没有被遮蔽.如果不通过返回handled.通过走下一步.
2.重置mGroupFlags,清空touchTarget.
如果是ACTIONDOWN事件,
a.重置mGroupFlags使boolean disallowIntercept = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0为false.
b.
mFirstTouchTarget是单链表结构.在addTOuchTarget()的时候被赋值,新建一个Target,next节点赋值为mFirstTouchTarget,mFirstTouchTarget赋值为新的target.
概括来说一旦有子View处理了事件序列中某一个事件,那么mFirstTouchTarget就不为空.
遍历TouchTarget链表,释放清空链表,最后mFirstTouchTarget赋值为空.
3.intercepted变量赋值,表示是否拦截.
判断actionMasked == MotionEvent.ACTION_DOWN || mFirstTouchTarget != null,为false,intercepted赋值为true.走下一步.
false意味着,当不是ACTION_DOWN事件,且mFirstTouchTarget为空(子View没有处理事件序列中的任一事件).
所以子View不处理ACTION_DOWN或者拦截了ACTION_DOWN,子View将无法得到其他的事件
true判断disallowIntercept = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0,true,intercepted赋值为false.false,intercepted = onInterceptTouchEvent(ev).
4.判断!canceled && !intercepted
canceled的分析//fixme
!canceled && !intercepted为true.遍历子View,判断子元素:1.是否可见或者在播放动画2.事件坐标是否在子View的区域
如果1和2是true调用dispatchTransformedTouchEvent().dispatchTransformedTouchEvent()在View不为空的情况下就会调用子View的dispatchTouchEvent().子View处理了alreadyDispatchedToNewTouchTarget 赋值为true;mFirstTouchTarget赋值.
!canceled && !intercepted为flase.清空状态,给子View分发Cancle事件,Target链表清空,mFirstTarget赋值为null.
这里反应出父View一旦拦截某个事件,事件序列中之后的事件都将interCepted设置为true,不会执行onInterCepted()去判断了.因为第2步的mFirstTarget为null.
开始这里我是有疑问的,外部拦截不是有move部分拦截吗,拦截了那么之后子View就没反应了?确认下代码,想了想,事件序列这个概念比较重要.move拦截了一次,那么mFirstTarget为null.就一直拦截了.
这里我又观察到一个现象,我在测试外部拦截处理滑动冲突的时候,move事件的拦截是有条件的,一开始不拦截,在ontercepted()打印了几次不拦截的日志.之后ontercepted()就没有再执行了.查看源码只有改变mGroupFlags为false才会有这种现象.于是我去内层的RecyclerView中查找是否调用requestDisallowInterceptTouchEvent()方法.发现在onTounchEvent()中就有调用,发现有滚动消耗就调用requestDisallowInterceptTouchEvent()方法使得外层mGroupFlags判定结果是false.
5.handled赋值
判断mFirstTouchTarget == null
true调用handled = dispatchTransformedTouchEvent(),执行super.dispatchTouchEvent(event)即View的dispatchTouchEvent(event).View的dispatchTouchEvent之后分析.
false根据子View的处理情况将handled赋值.
View的dispatchTouchEvent()
if (onFilterTouchEventForSecurity(event)) {
if ((mViewFlags & ENABLED_MASK) == ENABLED && handleScrollBarDragging(event)) {
result = true;
}
//noinspection SimplifiableIfStatement
ListenerInfo li = mListenerInfo;
if (li != null && li.mOnTouchListener != null
&& (mViewFlags & ENABLED_MASK) == ENABLED
&& li.mOnTouchListener.onTouch(this, event)) {
result = true;
}
if (!result && onTouchEvent(event)) {
result = true;
}
}
在判断onFilterTouchEventForSecurity后,如果View可用且mOnTouchListener.onTouch(this, event))返回true,就返回true.否则
result = onTouchEvent().