dispatchTouchEvent()源码看事件分发机制

2017-04-10  本文已影响101人  xwp

前言

通过对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().

上一篇下一篇

猜你喜欢

热点阅读