android中的dispatchTouchEvent、onInterceptTouchEvent和onTouchEvent

android中,触摸事件的传递过程主要涉及三个方法:dispatchTouchEvent、onInterceptTouchEvent和onTouchEvent。
  详细了解这三个方法的作用首先要了解以下几个知识点:

  • android中的Touch事件都是从ACTION_DOWN开始的:
    单指:ACTION_DOWN->ACTION_MOVE->ACTION_UP;
    多指:ACTION_DOWN->ACTION_POINTER_DOWN->ACTION_MOVE->ACTION_POINTER_UP->ACTION_UP
  • 一次完整的触摸事件中,Down和Up都只有一个,Move有若干个,可以为0个。当触摸事件被拦截时,Up可能是0个。
  • View在ViewGroup内,ViewGroup也可以在其他ViewGroup内,这时候把内部的ViewGroup当成View来分析。

知道了以上基本知识点以后,就可以开始了~

Activity、ViewGroup、View里的回调方法

在Activity里,有两个回调方法:

public boolean dispatchTouchEvent(MotionEvent ev);    
public boolean onTouchEvent(MotionEvent ev);    

在ViewGroup里,有三个回调方法:

public boolean dispatchTouchEvent(MotionEvent ev);    
public boolean onInterceptTouchEvent(MotionEvent ev);    
public boolean onTouchEvent(MotionEvent ev);  

在View里,和Activity相同,同样有两个回调方法:

public boolean dispatchTouchEvent(MotionEvent ev);    
public boolean onTouchEvent(MotionEvent ev);    

总结起来就是:

  • 和事件分发相关的方法共有三个:dispatchTouchEvent、onInterceptTouchEvent和onTouchEvent
  • dispatchTouchEvent和onTouchEvent在Activity、ViewGroup和View中均存在
  • 只有ViewGroup中有onInterceptTouchEvent方法

dispatchTouchEvent、onInterceptTouchEvent和onTouchEvent的区别

  • dispatchTouchEvent:这个方法用来分发TouchEvent。
  • onInterceptTouchEvent:这个方法用来拦截TouchEvent。
  • onTouchEvent:这个方法用来处理TouchEvent。
      一个简单的图可以表示这三个方法的执行过程:


    3.png

      举个例子:使用下面这段代码的xml布局

<com.chiaro.view.ChiaroLinearLayout    
      android:id="@+id/chiaroLinearLayout"    
      xmlns:android="http://schemas.android.com/apk/res/android"    
      android:layout_width="match_parent"    
      android:background="#999999"    
      android:padding="80dp"    
      android:layout_height="match_parent">    

      <com.chiaro.view.ChiaroTextView        
            android:id="@+id/chiaroTextView"        
            android:layout_width="match_parent"        
            android:layout_height="match_parent"        
            android:gravity="center"        
            android:text="TextView"        
            android:background="#00dddd"        
            android:textSize="32sp"/>
</com.chiaro.view.ChiaroLinearLayout>

这个布局长这样:节点层次很简单,一个LinearLayout中添加了一个TextView。


4.png

  MainActivity.java的代码:

// import的包不贴了……
public class MainActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        Log.e("demo","MainActivity-----------onTouchEvent--------------" + event.toString());
        return super.onTouchEvent(event);
    }
    @Override
    public boolean dispatchTouchEvent(MotionEvent event) {
        Log.e("demo","MainActivity-----------dispatchTouchEvent--------" + event.toString());
        return super.dispatchTouchEvent(event);
    }
}

ChiaroLinearLayout.java的代码:

// import的包不贴了……
public class ChiaroLinearLayout extends LinearLayout {
    // 构造方法省略不贴了……
    @Override
    public boolean onInterceptTouchEvent(MotionEvent event) {
        Log.e("demo", "ChiaroLinearLayout-----onInterceptTouchEvent-----" + event.toString());
        return super.onInterceptTouchEvent(event);
    }
    @Override
    public boolean dispatchTouchEvent(MotionEvent event) {
        Log.e("demo","ChiaroLinearLayout-----dispatchTouchEvent--------" + event.toString());
        return super.dispatchTouchEvent(event);
    }
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        Log.e("demo","ChiaroLinearLayout-----onTouchEvent--------------" + event.toString());
        return super.onTouchEvent(event);
    }
}

ChiaroTextView.java的代码:

// import的包不贴了……
public class ChiaroTextView extends TextView {
    // 构造方法省略不贴了……
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        Log.e("demo", "ChiaroTextView---------onTouchEvent--------------" + event.toString());
        return super.onTouchEvent(event);
    }
    @Override
    public boolean dispatchTouchEvent(MotionEvent event) {
        Log.e("demo","ChiaroTextView---------dispatchTouchEvent--------" + event.toString());
        return super.dispatchTouchEvent(event);
    }
}

可以看出,这段代码只是简单的打出所有的log。直接运行并点击一下TextView可以看到log如下。可以看到,这个ACTION_DOWN事件一直传递到了ChiaroTextView,但是最终是被MainActivity的onTouchEvent处理的,但是ACTION_UP只传递到了MainActivity,最终也是由MainActivity处理的。

 …… E/demo: MainActivity-----------dispatchTouchEvent--------MotionEvent { action=ACTION_DOWN ……
 …… E/demo: ChiaroLinearLayout-----dispatchTouchEvent--------MotionEvent { action=ACTION_DOWN ……
 …… E/demo: ChiaroLinearLayout-----onInterceptTouchEvent-----MotionEvent { action=ACTION_DOWN ……
 …… E/demo: ChiaroTextView---------dispatchTouchEvent--------MotionEvent { action=ACTION_DOWN ……
 …… E/demo: ChiaroTextView---------onTouchEvent--------------MotionEvent { action=ACTION_DOWN ……
 …… E/demo: ChiaroLinearLayout-----onTouchEvent--------------MotionEvent { action=ACTION_DOWN ……
 …… E/demo: MainActivity-----------onTouchEvent--------------MotionEvent { action=ACTION_DOWN ……
 …… E/demo: MainActivity-----------dispatchTouchEvent--------MotionEvent { action=ACTION_UP ……
 …… E/demo: MainActivity-----------onTouchEvent--------------MotionEvent { action=ACTION_UP ……

详细分析

onInterceptTouchEvent-事件拦截

onInterceptTouchEvent这个方法的返回值是最简单的,及是否拦截事件,放在最前面讲。

  • 如果返回值是true,代表事件在当前的viewGroup中会被处理,向下传递之路被截断(所有子控件将没有机会参与Touch事件),同时把事件传递给当前的控件的onTouchEvent()继续进行传递或处理。
  • 如果返回值是false,即不拦截当前传递来的事件,会继续向下传递,把事件交给子控件的onInterceptTouchEvent()。
      如果将ChiaroLinearLayout中的onInterceptTouchEvent方法的返回值改为true,则log为:
 …… E/demo: MainActivity-----------dispatchTouchEvent--------MotionEvent { action=ACTION_DOWN ……
 …… E/demo: ChiaroLinearLayout-----dispatchTouchEvent--------MotionEvent { action=ACTION_DOWN ……
 …… E/demo: ChiaroLinearLayout-----onInterceptTouchEvent-----MotionEvent { action=ACTION_DOWN ……
// 这里的ChiaroTextView没了  因为被父控件ChairoLinearLayout拦截了
 …… E/demo: ChiaroLinearLayout-----onTouchEvent--------------MotionEvent { action=ACTION_DOWN ……
 …… E/demo: MainActivity-----------onTouchEvent--------------MotionEvent { action=ACTION_DOWN ……
 …… E/demo: MainActivity-----------dispatchTouchEvent--------MotionEvent { action=ACTION_UP ……
 …… E/demo: MainActivity-----------onTouchEvent--------------MotionEvent { action=ACTION_UP ……

可以看到,这个事件在ChiaroLinearLayout就被打断了,没有继续传递给ChiaroTextView,而是由ChiaroLinearLayout的onTouchEvent继续传递给MainActivity的OnTouchEvent,最终由MainActivity的OnTouchEvent处理了。

onTouchEvent-事件处理

  • 如果返回值是true,表示消费(consume)了这个事件。以ACTION_DOWN为例,如果某个控件的onTouchEvent返回值为true,则后续的n个ACTION_MOVE与1个ACTION_UP都会逐层传递到这个控件的onTouchEvent进行处理。
  • 由于触摸事件都是连续的,所以这个地方感谢网友【@再不侃】的提问,要特别说明一下。如果ACTION_MOVE传递到子控件,而子控件的onTouchEvent返回值是false,即没有处理该ACTION_MOVE事件,则后续的ACTION_UP就不会传到该子控件来了。这个原因就造成了最开始没有对代码进行任何变更时,ACTION_DOWN事件一直传递到了ChiaroTextView,但是最终是被MainActivity的onTouchEvent处理的,而且ACTION_UP只传递到了MainActivity,最终也是由MainActivity处理的。的情况。
  • 这里要注意是逐层,也就是说每层的拦截器还是可以拦截到后续的ACTION_MOVE与ACTION_UP。如果后续的ACTION_MOVE与ACTION_UP被某层的拦截器拦截,则后续的事件将不会再传递给之前处理onTouchEvent的子控件,而是逐层传递给由拦截消息的这个控件的onTouchEvent函数进行处理,并且会向其之前接收事件的子控件发送一个ACTION_CANCEL,表示后续事件被取消了。
  • 如果返回值是false,则会将ACTION_DOWN传递给其父ViewGroup的onTouchEvent进行处理,直到由哪一层ViewGroup消费了ACTION_DOWN事件为止。
      如果讲上面的代码还原,并且将ChiaroTextView的onTouchEvent方法的返回值改为true,则log:
 …… E/demo: MainActivity-----------dispatchTouchEvent--------MotionEvent { action=ACTION_DOWN …… 
 …… E/demo: ChiaroLinearLayout-----dispatchTouchEvent--------MotionEvent { action=ACTION_DOWN …… 
 …… E/demo: ChiaroLinearLayout-----onInterceptTouchEvent-----MotionEvent { action=ACTION_DOWN …… 
 …… E/demo: ChiaroTextView---------dispatchTouchEvent--------MotionEvent { action=ACTION_DOWN …… 
 …… E/demo: ChiaroTextView---------onTouchEvent--------------MotionEvent { action=ACTION_DOWN …… 
 …… E/demo: MainActivity-----------dispatchTouchEvent--------MotionEvent { action=ACTION_UP …… 
 …… E/demo: ChiaroLinearLayout-----dispatchTouchEvent--------MotionEvent { action=ACTION_UP …… 
 …… E/demo: ChiaroLinearLayout-----onInterceptTouchEvent-----MotionEvent { action=ACTION_UP …… 
 …… E/demo: ChiaroTextView---------dispatchTouchEvent--------MotionEvent { action=ACTION_UP …… 
 …… E/demo: ChiaroTextView---------onTouchEvent--------------MotionEvent { action=ACTION_UP …… 

可以看到,所有的事件都传递给ChiaroTextView处理了。包括ACTION_DOWN和ACTION_UP。

dispatchTouchEvent-事件分发

dispatchTouchEvent比较复杂,可以按照下面这张图分析:ViewGroup和View组成了一棵树形结构,最顶层为Activity的ViewGroup,下面有若干的ViewGroup节点,每个节点之下又有若干的ViewGroup节点或者View节点,依次类推。


5.png

  当一个Touch事件依次下发,下发的过程是调用子View(ViewGroup)的dispatchTouchEvent方法实现的。简单来说,就是ViewGroup遍历它包含着的子View,调用每个View的dispatchTouchEvent方法,而当子View为ViewGroup时,又会通过调用ViewGroup的dispatchTouchEvent方法继续调用其内部的View的dispatchTouchEvent方法。上述例子中的消息下发顺序是这样的:①-②-⑤-⑥-⑦-③-④。
  dispatchTouchEvent方法只负责事件的分发,它拥有boolean类型的返回值,当返回为true时,顺序下发会中断。在上述例子中如果⑤的dispatchTouchEvent返回结果为true,那么⑥-⑦-③-④将都接收不到本次Touch事件。
  ViewGroup的dispatchTouchEvent是真正在执行“分发”工作,而View的dispatchTouchEvent方法,并不执行分发工作,或者说它分发的对象就是自己,决定是否把touch事件交给自己处理,而处理的方法,便是onTouchEvent事件,事实上子View的dispatchTouchEvent方法真正执行的代码是这样的:

public boolean dispatchTouchEvent(MotionEvent ev){
     ....//其他处理,在此不管
     return onTouchEvent(event); 
}

一般情况下,我们不该在普通View内重写dispatchTouchEvent方法,因为它并不执行分发逻辑。当Touch事件到达View时,我们该做的就是是否在onTouchEvent事件中处理它。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 151,688评论 1 330
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 64,559评论 1 273
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 101,749评论 0 226
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 42,581评论 0 191
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 50,741评论 3 271
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 39,684评论 1 192
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,122评论 2 292
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 29,847评论 0 182
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 33,441评论 0 228
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 29,939评论 2 232
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,333评论 1 242
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 27,783评论 2 236
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,275评论 3 220
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 25,830评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,444评论 0 180
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 34,553评论 2 249
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 34,618评论 2 249

推荐阅读更多精彩内容