Android文章收藏Android ClassAndroid 开发技术分享

android中的dispatchTouchEvent、onIn

2016-04-13  本文已影响3725人  chiaro

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

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

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的区别

<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这个方法的返回值是最简单的,及是否拦截事件,放在最前面讲。

 …… 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-事件处理

 …… 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事件中处理它。

上一篇下一篇

猜你喜欢

热点阅读