美文网首页
dispatchTouchEvent()源码看事件分发机制

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

作者: xwp | 来源:发表于2017-04-10 13:35 被阅读101次

    前言

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

    相关文章

      网友评论

          本文标题:dispatchTouchEvent()源码看事件分发机制

          本文链接:https://www.haomeiwen.com/subject/tmrsattx.html