美文网首页
Android View事件处理

Android View事件处理

作者: BarbaraBear | 来源:发表于2017-05-02 14:36 被阅读0次
事件的处理对象们

Android中View的事件处理用的是设计模式中的职责链模式。整个职责链中的处理对象是这样的:Activity->ViewGroup->View。

事件处理的三个重要阶段

这三类事件处理对象对事件的处理主要有三个阶段,对应三个重要的方法:

  1. 事件分发:boolean dispatchTouchEvent(MotionEvent ev)
    让当前处理对象决定是由自己来消费事件,还是将事件交给子View来处理。Activity、ViewGroup、View都有这个方法。
  2. 事件拦截:boolean onInterceptTouchEvent(MotionEvent ev)
    让处理对象决定要不要继续把时间传递给子View,还是自己消费掉算了 。只有ViewGroup有这个方法。
  3. 事件消费:boolean onTouchEvent(MotionEvent ev)
    决定是否消费掉事件,如何消费。Activity、ViewGroup、View都有这个方法。

这三个方法不同的返回值会影响事件的传递,其实还是挺复杂的。今天趁着劳动节,我来做做体力劳动,自定义一个ViewGroup和一个View,通过日志打印出这三个方法在不同的返回值下的调用。

onTouchEvent
all onTouchEvent false
viewGroup onTouchEvent true, view onTouchEvent false
  • 返回false或者是super.onTouchEvent时,表示当前View不想消费事件。则会逐级调用其父控件的onTouchEvent方法,直到调用的父控件的onTouchEvent返回true为止,即直到有父控件消费掉事件为止,或者是最终被Activity消费掉事件。这就是典型的职责链模式
  • 当某个事件在最终被某个对象的onTouchEvent消费掉后,这个事件之后连续的事件都会在分发到那个对象后,直接被它的onTouchEvent消费掉,而不会继续传递给子View了。从日志当中可以看出,View和ViewGroup的onTouchEvent都返回false,不消费事件,ACTION_DOWN事件最终被Activity消费掉。这此后ACTION_UP在被dispatch到Activity之后,就直接调用Activity的onTouchEvent,不会继续往下传递给子View了。同样的,当ViewGroup消费ACTION_DOWN事件后,接下来的ACTION_MOVE, ACTION_UP都会在dispatch到ViewGroup后,调用ViewGroup的onTouchEvent
onTouchEvent true
  • onTouchEvent返回true时,表示当前View想要消费掉事件。连续的所有的事件都会逐层被分发到当前View后,调用onTouchEvent方法
onInterceptTouchEvent
ViewGroup onInterceptTouchEvent=true, onTouchEvent=true
ViewGroup onInterceptTouchEvent=true, onTouchEvent=false
  • onInterceptTouchEvent返回true时,表示当前ViewGroup想拦截事件。此时会调用ViewGroup的onTouchEvent方法,它的子View会收到ACTION_CANCEL事件。如果ViewGroup的onTouchEvent返回true,则接下来的事件都会直接交给ViewGroup的onTouchEvent去处理,不会再调用onInterceptTouchEvent了。也就是ViewGroup从某个点拦截住事件,并且消费掉事件后,就可以直接处理接下来的事件而不需要再次拦截了。如果ViewGroup的onTouchEvent返回false,那么事件会被逐级向上传给它的父View的onTouchEvent去处理,并且此后连续的事件都不会传递给当前的ViewGroup了,也就是说当前ViewGroup再也没有机会收到接下来的事件了。因此,一般在自定义ViewGroup时,onInterceptTouchEvent返回true开始拦截事件时,都要让onTouchEvent返回true,并在onTouchEvent处理接下来的事件。
ViewGroup onInterceptTouchEvent=false
  • onInterceptTouchEvent返回false或者super.onInterceptTouchEvent时,表示当前ViewGroup不拦截事件,事件分发到子View
  • 当还没有确定事件讲会被哪个View处理时,事件在分发阶段都会调用onInterceptTouchEvent,但是一旦事件的处理对象明确后,onInterceptTouchEvent讲不会再被调用。这就是为什么,当最底层的ViewonTouchEvent返回true消费掉事件后,接下来的事件不知道到底会被谁处理,因此被下发到底层View的过程中还是会调用ViewGroup的onInterceptTouchEvent。而当事件被中途拦截,或者会被上层的某个父View处理后,ViewGroup的onInterceptTouchEvent都不会被调用。
dispatchTouchEvent
Paste_Image.png
  • 返回'true',事件无疾而终,接下来的连续的事件也无疾而终无人处理。
Paste_Image.png
  • 返回false,事件也不会继续往下传递,但是会被逐级向上传递给父view的onTouchEvent去处理。
  • 返回super.dispatchTouchEvent时,事件才会正常的往下传递给子View。一开始学习的时候,我有一个误区,认为这三个方法要么返回true,要么返回false,返回true的时候是自己处理事件,返回false的时候是让子view去处理事件。其实不是这样的,返回true的时候确实是让自己来处理事件,但是必须要调用super.dispatchTouchEvent才会把事件传递给子View进行处理。
  • 参考简书其它朋友写的文章

ViewGroup的dispatchTouchEvent是真正在执行“分发”工作,而View的dispatchTouchEvent方法,并不执行分发工作,或者说它分发的对象就是自己。一般情况下,我们不该在普通View内重写dispatchTouchEvent方法,因为它并不执行分发逻辑。当Touch事件到达View时,我们该做的就是是否在onTouchEvent事件中处理它。

相关文章

  • Android View事件处理

    事件的处理对象们 Android中View的事件处理用的是设计模式中的职责链模式。整个职责链中的处理对象是这样的:...

  • Android事件分发

    android的事件分发主要在view和viewGroup中,那么android是怎么进行事件处理的通过源码来一探...

  • 事件分发—初体验

    1.View 事件分发初体验 View 事件在 Android 中也是很重要的一块,通过对事件的处理可以完成一系列...

  • Android嵌套滑动讲解

    在Android的事件分发机制里面,当一个View决定消耗事件流时,其它的View就不能再处理这个事件流的了,所以...

  • View源码-Touch事件

    在Android-27中查看源码: 首先我们来查看单个View的触摸事件的处理,在View的dispatchTou...

  • Android事件分发机制

    准备工作 对于Android事件处理分为两类: 对于Activity和View: 只有两种事件:分发: dispa...

  • dispatchTouchEvent、onInterceptTo

    简述 1.View对触摸事件的处理机制2.ViewGroup对触摸事件的处理机制3.android对触摸体系的概念...

  • Android 注解

    Android 注解一、ButterKnife优势:1.强大的View绑定和Click事件处理功能,简化代码,提升...

  • Android 事件分发篇——边看源码边理解onTouch与on

    Android事件分发对于每一个Android Developer来说是必须理解的知识点,是自定义View、处理滑...

  • Android View事件分发机制

    Android View 事件分发机制

网友评论

      本文标题:Android View事件处理

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