事件分发

学习是需要总结归纳,纸上得来终觉浅,还是需要自己总结归纳,实例观察,作为学习笔记,跟大家分享一下。。

事件分发机制的分析对象就是MotionEvent,当一个MotionEvent对象产生之后,系统需要把这个事件传递给一个view,这个传递的过程就是分发过程

MotionEvent有3个常用的类型:
事件类型:代表的是motionEvent对象的动作
ACTION_DOWN:手指按下的动作
ACTION_UP:手指离开的动作
ACTION_MOVE:手指滑动的动作

分发的过程中有3个重要的方法:
public boolean dispatchTouchEvent(MotionEvent ev)
用来进行事件的分发,如果当前的ACTION_MOVE事件传递给当前的view,那么这个方法一定会被调用,返回的结果受当前view的onInterceptTouchEvent和下级view的dispatchTouchEvent方法的影响,表示是否消耗当前的事件,返回true代表消耗,返回false代表不消耗

public boolean onInterceptTouchEvent(MotionEvent ev)
在dispatchTouchEvent内部调用,用来判断是否拦截某个事件,如果当前view拦截某个事件,那么在这个事件序列(这个motionEvent整个动作)中不会再调用此方法,返回表示是否拦截某个事件,返回true拦截

public boolean onTouchEvent(MotionEvent ev)
在dispatchTouchEvent方法中调用,用来处理点击事件,如果不处理,那么在整个事件中无法再接收到事件,返回true代表处理这个事件,false表示不处理

整个过程可以用一个伪代码来解读

public boolean dispatchTouchEvent(MotionEvent ev){
 boolean isConsumed = false;
   if(onInterceptTouchEvent(ev)){
     isCousumed = this.onTouchEvent(ev);
   }else{
      isConsumed = childView.dispatchTouchEvent(ev);
   }
   return isConsumed;
}
activity-viewgroup-view.jpg
事件最终被textview消费

它的事件类型是,ACTION_DOWN==0,ACTION_UP==1
当我们点击了view,首先是activity接收到ACTION_DOWN事件,activity传递给window,window再传递给decorview,顶级的view进行分发(activity没有onInterceptTouchEvent方法),顺序就是调用activity的dispatchTouchEvent,然后调用viewgroup的dispatchTouchEvent,根据伪代码嗲用viewgroup的InterceptTouchEvent,再调用textview的dispatchTouchEvent和onTouchEvent方法,在这里我们对textview的点击事件进行处理,接下来对ACTION_UP进行分发

事件被viewgroup处理

可以看到当textview不进行处理的时候,会调用父view的onTouchEvent把事件传递回去,直到有一个进行处理,接下里对onTouchEvent进行分发的时候只分发到viewgroup,对处理onTouchEvent的textview不进行分发

事件谁都没有处理

当所有的view都不进行处理的时候只有将事件传递给activity了,接下来所有的事件序列都不进行分发了。


当viewgroup把事件拦截下来

当viewgroup把事件拦截下来就不对textview进行分发了,拦截下来又没有进行处理,气不气!最后只能交给activity处理了

形象的比喻一下:遇到了一个问题MotionEvent,上级一级一级的交代下来dispatchTouchEvent,看看谁能做onTouchEvent,谁能做就把这整个dispatchTouchEvent事件给他,如果最底层的那个马仔能做,onTouchEvent就返回个true,接下来其他事件也是按照这么个套路下发下来,要是底层马仔做不了,onTouchEvent返回个false,那只能是他的上级做了,上级要是做不了只能老板处理了,要是分发给中间某个领导时,他说这个事那个谁马仔干不了,就不给马仔分发下去了~

接下来看看onTouchListener,onClickLitener和onTouchEvent之间的关系


onTouchListener-onClickLitener-onTouchEvent

上图的操作是给viewgroup设置onTouchListener和onClickListener,onTouchListener返回值是false
由图可知:优先级依次是onTouchListener>onTouchEvent>onTouchEvent

小结:
1.当viewgroup中任何一个方法进行处理,那么事件都不会再向下传递,并且当设置了onTouchListener,那么onTouch一定会被回调,事件如何处理还需要看onTouch的返回值,要是true,那么view的onTouchEvent不会被调用,从方法上也可以看出onTouchListener(View v, MotionEvent event)肯定跟MotionEvent 的传递有一定的关系;
2.同一事件序列是指从手指接触屏幕的那一刻起,到手离开屏幕的那一刻,这个过程中所产生的一系列事件,一般是down-move-up
3.正常情况下,一个事件序列只能由view去消费,因为一个view拦截了一个事件之后,其他的事件都会交给它进行处理,因此同一事件序列中的事件不能交给两个view进行处理
4.某个view一旦开始消费事件,如果它不处理down事件onTouchEvent返回false,那么其他的事件也不会交给它进行处理,由父view进行处理
5.如果view处理了down事件,但是并不处理其他的事件,那么这些事件它的父view无法收到,只能由activity处理
6.view的onTouchEvent默认都会消耗事件,除非是不可点击的clickable是false,view的longClickable默认是false,但是view的clickable要分情况的,Button是true,textview就是false
7.onClick会发生的前提是当前view是可点击的,并且收到了up和down事件
8.事件传递的过程是从上到下的,由外向内的,父---子,通过requestDisallowInterceptTouchEvent可以干预父类的除down事件的事件分发
以上参考了《Android开发艺术探索》

下面分析下源码:
由于开始的方法是dispatchTouchEvent,由此分发,那么我们从这开始分析,首先调到activity中的dispatchTouchEvent

/**
     * Called to process touch screen events.  You can override this to
     * intercept all touch screen events before they are dispatched to the
     * window.  Be sure to call this implementation for touch screen events
     * that should be handled normally.
     *
     * @param ev The touch screen event.
     *
     * @return boolean Return true if this event was consumed.
     */
    public boolean dispatchTouchEvent(MotionEvent ev) {
        if (ev.getAction() == MotionEvent.ACTION_DOWN) {
            onUserInteraction();
        }
        if (getWindow().superDispatchTouchEvent(ev)) {
            return true;
        }
        return onTouchEvent(ev);
    }

看到getWindow().superDispatchTouchEvent(ev)这行代码,我们知道activity-window,window只有一个实现类phonewindow

 @Override
    public boolean superDispatchTouchEvent(MotionEvent event) {
        return mDecor.superDispatchTouchEvent(event);
    }

mDecor和phoneWindow的关系已经讨论过了,在这我们看这个方法
viewGroup类中的方法
由于比较长,需要抓住几个方向点:
通过伪代码的逻辑是先进行拦截,我们先看看拦截是怎么回事

            final boolean intercepted;
            if (actionMasked == MotionEvent.ACTION_DOWN
                    || mFirstTouchTarget != null) {
                final boolean disallowIntercept = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0;
                if (!disallowIntercept) {
                    intercepted = onInterceptTouchEvent(ev);
                    ev.setAction(action); // restore action in case it was changed
                } else {
                    intercepted = false;
                }
            } else {
                // There are no touch targets and this action is not an initial down
                // so this view group continues to intercept touches.
                intercepted = true;
            }

从上述代码中我们可以看出:viewGroup在两种情况下要去判断是否要拦截,事件类型为ACTION_DOWN或者 mFirstTouchTarget != null,ACTION_DOWN这个事件很好理解,那么 mFirstTouchTarget 是什么呢
从方法后面的代码可以看出当viewgroup的子元素处理事件时,mFirstTouchTarget 会被赋值并指向子元素,也就是当viewgroup不拦截事件并交给子元素处理的时候mFirstTouchTarget !=null,所以当ACTION_MOVE和ACTION_UP事件到来时,这个条件为false,那么viewgroup的onInterceptTouchEvent不会再被调用,

继续向下看FLAG_DISALLOW_INTERCEPT这个标记位是由子view的requestDisallowInterceptTouchEvent方法啊设置的,一旦设置了之后viewgroup将无法拦截除了ACTION_DOWN以外的事件,因为ACTION_DOWN会重置FLAG_DISALLOW_INTERCEPT这个标记位,将导致子view设置的失效,对于ACTION_DOWN事件viewgroup总会首先调用自己的intercept方法来判断是否要拦截

          if (actionMasked == MotionEvent.ACTION_DOWN) {
                // Throw away all previous state when starting a new touch gesture.
                // 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.
                cancelAndClearTouchTargets(ev);
                resetTouchState();
            }
    /**
     * Resets all touch state in preparation for a new cycle.
     */
    private void resetTouchState() {
        clearTouchTargets();
        resetCancelNextUpFlag(this);
        mGroupFlags &= ~FLAG_DISALLOW_INTERCEPT;
        mNestedScrollAxes = SCROLL_AXIS_NONE;
    }

总结:就是说ACTION_DOWN事件永远是viewgroup先去判断是否拦截,然后判断是否交给子view去处理,如果不交给子view处理,那么就是拦截,如果交给子view处理,那么以后所有的事件类型都不会再调用viewgroup的拦截方法,再看看是否有标记位,如果有标记位的话那么就拦截,如果没有就不拦截

 final View[] children = mChildren;
                        for (int i = childrenCount - 1; i >= 0; i--) {
                            final int childIndex = getAndVerifyPreorderedIndex(
                                    childrenCount, i, customOrder);
                            final View child = getAndVerifyPreorderedView(
                                    preorderedList, children, childIndex);

                            // If there is a view that has accessibility focus we want it
                            // to get the event first and if not handled we will perform a
                            // normal dispatch. We may do a double iteration but this is
                            // safer given the timeframe.
                            if (childWithAccessibilityFocus != null) {
                                if (childWithAccessibilityFocus != child) {
                                    continue;
                                }
                                childWithAccessibilityFocus = null;
                                i = childrenCount - 1;
                            }

                            if (!canViewReceivePointerEvents(child)
                                    || !isTransformedTouchPointInView(x, y, child, null)) {
                                ev.setTargetAccessibilityFocus(false);
                                continue;
                            }

                            newTouchTarget = getTouchTarget(child);
                            if (newTouchTarget != null) {
                                // Child is already receiving touch within its bounds.
                                // Give it the new pointer in addition to the ones it is handling.
                                newTouchTarget.pointerIdBits |= idBitsToAssign;
                                break;
                            }

                            resetCancelNextUpFlag(child);
                            if (dispatchTransformedTouchEvent(ev, false, child, idBitsToAssign)) {
                                // Child wants to receive touch within its bounds.
                                mLastTouchDownTime = ev.getDownTime();
                                if (preorderedList != null) {
                                    // childIndex points into presorted list, find original index
                                    for (int j = 0; j < childrenCount; j++) {
                                        if (children[childIndex] == mChildren[j]) {
                                            mLastTouchDownIndex = j;
                                            break;
                                        }
                                    }
                                } else {
                                    mLastTouchDownIndex = childIndex;
                                }
                                mLastTouchDownX = ev.getX();
                                mLastTouchDownY = ev.getY();
                                newTouchTarget = addTouchTarget(child, idBitsToAssign);
                                alreadyDispatchedToNewTouchTarget = true;
                                break;
                            }
}

这段代码首先遍历viewgroup中所有的子view,判断子view是否可以接收到点击事件,2点用来衡量,是否点击的坐标在子view的区域内,子view是否在播放动画,如果子元素都满足这条件的话,那么事件就交给它来处理dispatchTransformedTouchEvent这个方法实际上就是调用子元素的dispatchTouchEvent方法

 private boolean dispatchTransformedTouchEvent(MotionEvent event, boolean cancel,
            View child, int desiredPointerIdBits) {
        final boolean handled;

        // Canceling motions is a special case.  We don't need to perform any transformations
        // or filtering.  The important part is the action, not the contents.
        final int oldAction = event.getAction();
        if (cancel || oldAction == MotionEvent.ACTION_CANCEL) {
            event.setAction(MotionEvent.ACTION_CANCEL);
            if (child == null) {
                handled = super.dispatchTouchEvent(event);
            } else {
                handled = child.dispatchTouchEvent(event);
            }
            event.setAction(oldAction);
            return handled;
        }
  newTouchTarget = addTouchTarget(child, idBitsToAssign);
  alreadyDispatchedToNewTouchTarget = true;

当子view dispatchTouchEvent返回true时,会给mFirstTouchTarget赋值并跳出循环

view的点击事件就比较简单了,当CLICKABLE和LONG_CLICKABLE其中一个为true时就会消耗这个事件,当move_up执行的时候会触发performClick方法,如果view设置了onclickListener那么performClick方法会调用onClick方法,通过setOnClickListener和setOnClickable会使属性变为true可以消耗事件。

你可能感兴趣的:(事件分发)