美文网首页
笔记(四)——事件分发机制

笔记(四)——事件分发机制

作者: 木溪bo | 来源:发表于2020-03-24 01:00 被阅读0次

    ——》个人平时笔记,看到的同学欢迎指正错误,文中多处摘录于各大博主精华、书籍

    1、事件分发机制:整个事件分发是一个U形传递的,递归传递。图解 Android 事件分发机制

    image

    一个事件是指:一个ACTION_DOWN事件或ACTION_MOVE事件或ACTION_UP事件等。它们合称同一个事件序列。事件是分开执行传递的,在顺利的条件下ACTION_DOWN、ACTION_MOVE、ACTION_UP事件分别按这个顺序,依次跑完U行事件流程。

    • 对于ACTION_DOWN事件有以下特性
      (1)、dispatchTouchEvent返回 false 的含义应该是事件停止往子View传递和分发,并往上层父控件的onTouchEvent 回溯,之后上层父控件的onTouchEvent开始从下往上回传,直到某个更上一层的onTouchEvent return true消费事件而终止传递。其中如果是activity的话,dispatchTouchEvent return false | ture都是消费;最内部底层view的dispatchTouchEvent return super.dispatchTouchEvent()则会将事件传递给当前view的onTouchEvent 。
      (2)、onTouchEvent返回false | super就比较简单了,它就是不消费事件,并让事件继续从下往上向一层父控件的方向传递,直到return true消费掉事件终止传递。
      (3)、ViewGroup 和View的这些方法的super.xxxx()默认实现就是会让整个事件按照U形完整走完。中间不做任何改动,不回溯、不终止,每个环节都走到 dispatchTouchEvent--->onInterceptTouchEvent--->dispatchTouchEvent--->onTouchEvent。
      (4)、onInterceptTouchEvent拦截方法,类似一个开关,当return true时该事件则被当前控件消费了,拦截该事件,不再往下传递了;而return falsereturn super.xxxxxx()不会拦截事件,继续往下传递事件,保证U形路径畅通。
      down事件流程1
    down事件流程2

    重要注意:

    (1)、如果一个View或ViewGroup的onTouchEvent不消耗ACTION_DOWN事件返回了false,那么它就不会再接收同一事件序列中的ACTION_MOVE、ACTION_UP等事件,并且将整个事件交给它的上一层父元素去处理。如果返回true就消费事件终止传递。

    (2)、dispatchTouchEvent()与onInterceptTouchEvent()依据上图情况可以得到相似结果。如果ACTION_DOWN都没有接收到,同一事件序列的ACTION_MOVE、ACTION_UP就不会再被接收了。但是onInterceptTouchEvent()比较特殊,当拦截事件ACTION_DOWN返回true时,同一事件序列ACTION_MOVE、ACTION_UP并不会再传递到onInterceptTouchEvent(),而是直接跳过onInterceptTouchEvent(),直接传递到当前view的onTouchEvent()中。反之onInterceptTouchEvent()不拦截事件ACTION_DOWN,后面的ACTION_MOVE、ACTION_UP才能交由它处理。

    2、所有ListView、RecycleView、ScrollView等可以滚动的控件,当一页显示不完数据时都会消费掉onTouchEvent,所以父View的onTouchEvent就接收不到事件了。

    小结:dispatchTouchEvent(),onInterceptTouchEvent(),onTouchEvent()所示最后打印输出所知Down、Move、UP都是一个一个对应执行。但是要是ACTION_DOWN都没有接收到事件,后面的ACTION_MOVE、ACTION_UP事件序列就也不能接收到。

    3、《Android开发艺术探索》中提到:

    绘制及分发的顺序流程:

    Activity->Window(PhoneWindow实体类)->ViewRoot(ViewRootImpl)->DecorView->ViewGroup——》View(最底部转折点)——》ViewGroup->DecorView->ViewRoot(ViewRootImpl)->Window(PhoneWindow实体类)->Activity

    关于事件传递的机制,这里给出一些结论,根据这些结论可以更好地理解整个传递机制,如下所示。

    (1)、同一个事件序列是指从手指接触屏幕的那一刻起,到手指离开屏幕的那一刻结束,在这个过程中所产生的一系列事件,这个事件序列以down事件开始,中间含有数量不定的move事件,最终以up事件结束。

    (2)、正常情况下,一个事件序列只能被一个View拦截且消耗。这一条的原因可以参考(3),因为一旦一个元素拦截了某此事件,那么同一个事件序列内的所有事件都会直接交给它处理,因此同一个事件序列中的事件不能分别被两个View同时处理,但是通过特殊手段可以做到,比如:一个view1将本该自己处理的事件通过onTouchEvent()强行传递给view2的onTouchEvent()的处理。

    (3)、某个View一旦决定拦截(拦截ACTION_DOWN返回true),那么这一个事件序列都只能由它来处理(如果事件序列能够传递给它的话),并且它的onInterceptTouchEvent不会再被调用。这条也很好理解,就是说当一个View决定拦截一个事件后,那么系统会把同一个事件序列内的其他事件都直接交给它的onTouchEvent()来处理,因此就不用再调用这个View的onInterceptTouchEvent去询问它是否要拦截了。

    (4)、某个View一旦开始处理事件,如果它不消耗ACTION_DOWN事件(onTouchEvent返回了false),那么同一事件序列中的其他事件都不会再交给它来处理,并且事件将重新交由它的父元素去处理,即父元素的onTouchEvent会被调用。意思就是事件一旦交给一个View处理,那么它就必须消耗掉,否则同一事件序列中剩下的事件就不再交给它来处理了,这就好比上级交给程序员一件事,如果这件事没有处理好,短期内上级就不敢再把事情交给这个程序员做了,二者是类似的道理。

    (5)、如果View不消耗除ACTION_DOWN以外的其他事件,那么这个点击事件会消失,此时父元素的onTouchEvent并不会被调用,并且当前View可以持续收到后续的事件,最终这些消失的点击事件会传递给Activity处理。(特别记下)

    image

    (6)、ViewGroup默认不拦截任何事件。Android源码中ViewGroup的onInterceptTouchEvent方法默认返回false。

    (7)、View没有onInterceptTouchEvent方法,一旦有点击事件传递给它,那么它的onTouchEvent方法就会被调用。

    (8)、View的onTouchEvent默认都会消耗事件(返回true),除非它是不可点击的(clickable 和longClickable同时为false)。View的longClickable属性默认都为false,clickable属性要分情况,比如Button的clickable属性默认为true,而TextView的clickable属性默认为false。

    (9)、View的enable属性不影响onTouchEvent的默认返回值。哪怕一个View是disable状态的,只要它的clickable或者longClickable有一个为true,那么它的onTouchEvent就返回true,就会消费事件。

    比如Button是可点击的,TextView是不可点击的。通过setClickable和setLongClickable可以分别改变View的CLICKABLE和LONG_CLICKABLE属性。另外,setOnClickListener会自动将View的CLICKABLE设为true,setOnLongClickListener则会自动将View的LONG_CLICKABLE设为true,这一点从源码中可以看出来。

     源码:public void setOnClickListener(OnClickListener l) {
             if (!isClickable()) {
                     setClickable(true);
             }
             getListenerInfo().mOnClickListener = l;
         }
     
         public void setOnLongClickListener(OnLongClickListener l) {
             if (!isLongClickable()) {
                     setLongClickable(true);
             }
            getListenerInfo().mOnLongClickListener = l;
     }
    

    (10)、onClick会发生的前提是当前View是可点击的,并且它收到了ACTION_DOWN和ACTION_UP的事件。优先级:onTouchListener > onTouchEvent > OnClickListener

    (11)、事件传递过程是由外向内的,即事件总是先传递给父元素,然后再由父元素分发给子View,通过requestDisallowInterceptTouchEvent方法可以在子元素中干预父元素的事件分发过程,但是ACTION_DOWN事件除外。(若要干预ACTION_DOW的话也只能改父viewGroup为拦截状态返回true,如果这样的话子view本身就接收不到任何事件序列了,就更谈不上requestDisallowInterceptTouchEvent能够干预父元素事件了)(有解释说因为ACTION_DOWN事件方法里,会清除所有的标志位——View的事件分发机制和滑动冲突解决方案

    4、下图理解(图解 Android 事件分发机制一文中):事件为U型传递,ViewGroup2在onTouchEvent消费事件,事件序列都返回true,事件分发到此为止;ViewGroup2既然能消费事件,则它的下层子View的onTouchEvent的ACTION_DOWN必定是不消费返回false的,而返回了false则后面子View的ACTION_MOVE和ACTION_UP等事件序列就不再被接收了,直接分发至父ViewGroup2,所以才有这样的蓝色箭头走向

    我们在ViewGroup 2 的onTouchEvent 返回true消费这次事件

    红色的箭头代表ACTION_DOWN 事件的流向

    蓝色的箭头代表ACTION_MOVE 和 ACTION_UP 事件的流向

    image

    5、解决滑动冲突的方式:外部拦截法和内部拦截法 参考-View的事件分发机制和滑动冲突解决方案

    外部拦截法:是指点击事情都先经过父容器的拦截处理,如果父容器需要此事件就拦截,如果不需要此事件就不拦截,这样就可以解决滑动冲突的问题,这种方法比较符合点击事件的分发机制。外部拦截法需要重写父容器的onInterceptTouchEvent方法,在方法内做相应的拦截即可。

    ACTION_UP事件,这里必须要返回false,假设事件交由子元素处理,如果父容器在ACTION_UP时返回了true,就会导致子元素无法接收到ACTION_UP事件,这个时候子元素中的onClick事件就无法触发,但是父容器比较特殊,一旦它开始拦截任何一个事件,那么后续的事件都会交给它来处理,而ACTION_UP作为最后一个事件也必定可以传递给父容器,即便父容器的onInterceptTouchEvent方法在ACTION_UP时返回了false。

    外部拦截法,父ViewGroup伪代码如下: 
     public boolean onInterceptTouchEvent(MotionEvent event) {
             boolean intercepted = false;
             int x = (int) event.getX();
             int y = (int) event.getY();
     
             switch (event.getAction()) {
              case MotionEvent.ACTION_DOWN: {
                     intercepted = false;
                     break;
              }
              case MotionEvent.ACTION_MOVE: {
                        if (父容器需要当前点击事件) {
                             intercepted = true;
                        } else {
                             intercepted = false;
                        }
                      break;
              }
             case MotionEvent.ACTION_UP: {
                     intercepted = false;
                     break;
             }
             default:
                     break;
             }
             mLastXIntercept = x;
             mLastYIntercept = y;
             return intercepted;
          }
    

    内部拦截法:是指父容器不拦截任何事件,所有的事件都传递给子元素,如果子元素需要此事件就直接消耗掉,否则就交由父容器进行处理,这种方法和Android中的事件分发机制不一致,需要配合requestDisallowInterceptTouchEvent方法才能正常工作。

    内部拦截法,子View伪代码如下: 
     public boolean dispatchTouchEvent(MotionEvent event) { 
             int x = (int) event.getX();
             int y = (int) event.getY();
    
             switch (event.getAction()) {
                case MotionEvent.ACTION_DOWN: {
                      //参数true,使父ViewGroup不拦截
                     parent.requestDisallowInterceptTouchEvent(true);
                     break;
                }
                case MotionEvent.ACTION_MOVE: {
                     int deltaX = x -mLastX;
                     int deltaY = y -mLastY;
                     if (父容器需要此类点击事件)) {
                        //参数false,让父ViewGroup拦截事件
                             parent.requestDisallowInterceptTouchEvent(false);
                     }
                     break;
                 }
                 case MotionEvent.ACTION_UP: {
                     break;
                 }
                default:
                    break;
              }
             mLastX = x;
             mLastY = y;
             return super.dispatchTouchEvent(event);
         }
     
     父ViewGroup伪代码:
     
     @Override
       public boolean onInterceptTouchEvent(MotionEvent event) {
           int action = event.getAction();
           if (action == MotionEvent.ACTION_DOWN) {
               //因为子view需要事件,父ViewGroup的down都不拦截,让这一系列事件往下传递
               return false;
           } else {
               return true;
           }
       }
    

    相关文章

      网友评论

          本文标题:笔记(四)——事件分发机制

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