Android系统中View的onTouchEvent和OnTouch区别

以自定义的TestButton为例。

我们可以通过重写onTouchEvent方法来处理诸如down move up的消息:


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public class TestButton extends Button {
 
     public TestButton(Context context) {
         super (context);
         // TODO Auto-generated constructor stub
     }
     
     public TestButton(Context context, AttributeSet attributeSet) {
         super (context, attributeSet);
         // TODO Auto-generated constructor stub
     }
     
     @Override
     public boolean onTouchEvent(MotionEvent event) {
         boolean value = super .onTouchEvent(event);
         System.out.println( "super.onTouchEvent: " + value+ " event: " + event.getAction());
         return value;
     }
也可以通过实现OnTouchListener的接口,然后设置TestButton的onTouchListener可以达到同样的目的

1
2
3
4
5
6
7
class OnTouchListenerTest implements View.OnTouchListener{
     @Override
     public boolean onTouch(View v, MotionEvent event) {
         return false ;
     }
     
}

1
2
3
TestButton b = (TestButton)findViewById(R.id.button);
OnTouchListenerTest listener = new OnTouchListenerTest();
b.setOnTouchListener(listener);
但上述两种监听有什么区别呢?

先看一下Android源码中对于View中dispatchTouchEvent的实现:


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
public boolean dispatchTouchEvent(MotionEvent event){
     ... ...
     if (onFilterTouchEventForSecurity(event)){
         ListenerInfo li = mListenerInfo;
         if (li != null && li.mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED
             && li.mOnTouchListener.onTouch( this , event)) {
             return true ;
         }
         if (onTouchEvent(event)){
             return true ;
         }
     }
     ... ...
     return false ;
}

可以看到onTouchListener的接口的优先级是要高于onTouchEvent的,假若onTouchListener中的onTouch方法返回true,

表示此次事件已经被消费了,那onTouchEvent是接收不到消息的。

因为Button的performClick是利用onTouchEvent实现,假若onTouchEvent没有被调用到,那么Button的Click事件也无法响应。


综合来讲:

1、onTouch方法是View的 OnTouchListener接口中定义的方法。

2、OnTouch()先执行,只有当OnTouch()返回为false时才触发OnTouchEvent(),反之onTouchEvent方法不会被调用。

3、点击事件是在OnTouchEvent()中执行的。因此,Touch事件先执行,click事件后执行。

4、内置诸如click事件的实现等等都基于onTouchEvent事件,假如onTouch返回true,这些事件将不会被触发。


View的触摸事件最先执行的是dispatchTouchEvent。

View源码中dispatchTouchEvent中所做操作:
 * a 判断mOnTouchListener(监听事件)是否为null
 * b 判断当前的控件是否可用(enable)
 * c 判断view的onTouch。
 * d 如果以上一个返回为false。那么就会调用onTouchEvent


Android系统中,想要一个View去响应Touch事件,实现既能响应up事件,又能响应down事件,那么返回onTouch()方法的返回值必须为true。如果为false,则只能响应down事件,而不能响应up事件。因为如果返回为true说明该控件自身能消费掉该事件,false则表示控件自身不能消费该事件,事件向下传递。

在down事件如果能消费(处理)当前事件。那么在up的时候也会把事件传递给当前的view
​在down事件处理不了当前事件。那么在up的时候。也不会把事件传递给当前的view


点击事件是在touch事件中执行的。


Android系统中事件分发是从ViewGroup往下分发给View的。Android系统中先添加的View在布局下面,后添加的View在布局上面。

通过追踪Activity中的dispatchTouchEvent方法,发现追踪到了DecorView父类FrameLayout的dispatchTouchEvent方法,但是FrameLayout中没有dispatchTouchEvent方法,继续追踪到FrameLayout的父类ViewGroup的dispatchTouchEvent方法。所以可见Android系统中事件分发是从ViewGroup往下分发给View的。

你可能感兴趣的:(Android)