Android 3步搞定事件分发机制,再也不用担心onTouch和onTouchEvent&dispatchTouchEvent

来源:互联网 发布:中国地质大学武汉知乎 编辑:程序博客网 时间:2024/06/05 19:39

       事件分发机制分为2种:View事件的分发和ViewGroup事件分发机制
先看简单的View事件分发机制
//子控件的ontouch方法影响子控件的函数//onTouch====onTouchEvent====onClick;/** * 检验view的事件分发顺序,点击---dispatch-  Ontouch返回值为ture  不执行---ontouchEvent---onclick */button1.setOnTouchListener(new View.OnTouchListener() {    @Override    public boolean onTouch(View v, MotionEvent event) {        Log.d("TAG", "button1  on touch"+event.getAction());        return true;    }});/** * 检验view的事件分发顺序, 点击---dispatch-  Ontouch返回值为false执行---ontouchEvent---onclick */button2.setOnTouchListener(new View.OnTouchListener() {    @Override    public boolean onTouch(View v, MotionEvent event) {        Log.d("TAG", "button1  on touch" + event.getAction());        return false;    }});


然后我们来看一下View中dispatchTouchEvent方法的源码:

public boolean dispatchTouchEvent(MotionEvent event) {    if (mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED &&            mOnTouchListener.onTouch(this, event)) {        return true;    }    return onTouchEvent(event);}


1、整个View的事件转发流程是:

View.dispatchEvent->View.setOnTouchListener->View.onTouchEvent

在dispatchTouchEvent中会进行OnTouchListener的判断,如果OnTouchListener不为null且返回true,则表示事件被消费,onTouchEvent不会被执行;否则执行onTouchEvent。

onClick方法是在onTouchEvent方法里调用的


2个Button的onTouch返回值验证了以下:

onTouchListener的onTouch方法优先级比onTouchEvent高,会先触发。

假如onTouch方法返回false会接着触发onTouchEvent,反之onTouchEvent方法不会被调用。


Button和ImageView效果不一样:一个是自带点击,一个是要自己控制点击

onTouch能够得到执行需要两个前提条件,第一mOnTouchListener的值不能为空,第二当前点击的控件必须是enable的。因此如果你有一个控件是非enable的,那么给它注册onTouch事件将永远得不到执行。对于这一类控件,如果我们想要监听它的touch事件,就必须通过在该控件中重写onTouchEvent方法来实现。

/**ImageView默认是不能点击事件的,要想点击的话必须手动设置*/imageView.setClickable(true);imageView.setOnClickListener(new View.OnClickListener() {    @Override    public void onClick(View v) {        Log.e("TAG","imageView setOnTouchListener");    }});
ViewGruop的事件分发:
多了一个拦截事件的方法:onInterceptTouchEvent

比较好的形容:
其中Activity和View控件(TextView)拥有分派和处理事件方法,View容器(LinearLayout)具有分派,拦截,处理事件方法。
这里也有个比喻:领导都会把任务向下分派,一旦下面的人把事情做不好,就不会再把后续的任务交给下面的人来做了,只能自己亲自做,
如果自己也做不了,就只能告诉上级不能完成任务,上级又会重复他的过程。
另外,领导都有权利拦截任务,对下级隐瞒该任务,而直接自己去做,如果做不成,也只能向上级报告不能完成任务。

/**默认为false,不拦截子控件的监听*/@Overridepublic boolean onInterceptTouchEvent(MotionEvent ev) {    return false;}

public boolean dispatchTouchEvent(MotionEventev);  //用来分派eventpublic boolean onInterceptTouchEvent(MotionEventev);//用来拦截eventpublic boolean onTouchEvent(MotionEventev);//用来处理event


总结:

1. Android事件分发是先传递到ViewGroup,再由ViewGroup传递到View的。

2. 在ViewGroup中可以通过onInterceptTouchEvent方法对事件传递进行拦截,onInterceptTouchEvent方法返回true代表不允许事件继续向子View传递,返回false代表不对事件进行拦截,默认返回false。

3.拦截的好好处在于调用谁的dispatchTouchEvent的方法,谁出来点击事件

4. 子View中如果将传递的事件消费掉,ViewGroup中将无法接收到任何事件。

5.在ViewGroup中onInterceptTouchEvent方法若反回false,那么触屏事件会继续向下传递,
但如果没有子View去处理这个事件,即子view的onTouchEvent没有返回True
则最后还是由ViewGroup去处理这个事件,也就又执行了自己的onTouchEvent。

下面是方法总结:
OnTouch方法是在Activity里面设置的监听事件


1. onTouch和onTouchEvent有什么区别,又该如何使用?

从源码中可以看出,这两个方法都是在View的dispatchTouchEvent中调用的,onTouch优先于onTouchEvent执行。如果在onTouch方法中通过返回true将事件消费掉,onTouchEvent将不会再执行。


思考的问题:

1.View的dispatchTouchEvent重写了会怎么样,View的onTouchEvent重写了会怎样?

2.ViewGroup的dispatchTouchEvent重写了会怎么样,ViewGroup的onTouchEvent重写了会怎样?

3.ListView上的button,点击button2个控件同时要有相应,应该怎么处理?

4.


事件的传递顺序:

G----->A----->B----->C----->D----->E----->F
AS源代码不知道什么原因传不上去委屈

参考博客:
http://blog.csdn.net/guolin_blog/article/details/9097463
http://blog.csdn.net/lmj623565791/article/details/38960443
事件分发机制分为2种:View事件的分发和ViewGroup事件分发机制
先看简单的View事件分发机制
//子控件的ontouch方法影响子控件的函数//onTouch====onTouchEvent====onClick;/** * 检验view的事件分发顺序,点击---dispatch-  Ontouch返回值为ture  不执行---ontouchEvent---onclick */button1.setOnTouchListener(new View.OnTouchListener() {    @Override    public boolean onTouch(View v, MotionEvent event) {        Log.d("TAG", "button1  on touch"+event.getAction());        return true;    }});/** * 检验view的事件分发顺序, 点击---dispatch-  Ontouch返回值为false执行---ontouchEvent---onclick */button2.setOnTouchListener(new View.OnTouchListener() {    @Override    public boolean onTouch(View v, MotionEvent event) {        Log.d("TAG", "button1  on touch" + event.getAction());        return false;    }});


然后我们来看一下View中dispatchTouchEvent方法的源码:

public boolean dispatchTouchEvent(MotionEvent event) {    if (mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED &&            mOnTouchListener.onTouch(this, event)) {        return true;    }    return onTouchEvent(event);}


1、整个View的事件转发流程是:

View.dispatchEvent->View.setOnTouchListener->View.onTouchEvent

在dispatchTouchEvent中会进行OnTouchListener的判断,如果OnTouchListener不为null且返回true,则表示事件被消费,onTouchEvent不会被执行;否则执行onTouchEvent。

onClick方法是在onTouchEvent方法里调用的


2个Button的onTouch返回值验证了以下:

onTouchListener的onTouch方法优先级比onTouchEvent高,会先触发。

假如onTouch方法返回false会接着触发onTouchEvent,反之onTouchEvent方法不会被调用。


Button和ImageView效果不一样:一个是自带点击,一个是要自己控制点击

onTouch能够得到执行需要两个前提条件,第一mOnTouchListener的值不能为空,第二当前点击的控件必须是enable的。因此如果你有一个控件是非enable的,那么给它注册onTouch事件将永远得不到执行。对于这一类控件,如果我们想要监听它的touch事件,就必须通过在该控件中重写onTouchEvent方法来实现。

/**ImageView默认是不能点击事件的,要想点击的话必须手动设置*/imageView.setClickable(true);imageView.setOnClickListener(new View.OnClickListener() {    @Override    public void onClick(View v) {        Log.e("TAG","imageView setOnTouchListener");    }});
ViewGruop的事件分发:
多了一个拦截事件的方法:onInterceptTouchEvent

比较好的形容:
其中Activity和View控件(TextView)拥有分派和处理事件方法,View容器(LinearLayout)具有分派,拦截,处理事件方法。
这里也有个比喻:领导都会把任务向下分派,一旦下面的人把事情做不好,就不会再把后续的任务交给下面的人来做了,只能自己亲自做,
如果自己也做不了,就只能告诉上级不能完成任务,上级又会重复他的过程。
另外,领导都有权利拦截任务,对下级隐瞒该任务,而直接自己去做,如果做不成,也只能向上级报告不能完成任务。

/**默认为false,不拦截子控件的监听*/@Overridepublic boolean onInterceptTouchEvent(MotionEvent ev) {    return false;}

public boolean dispatchTouchEvent(MotionEventev);  //用来分派eventpublic boolean onInterceptTouchEvent(MotionEventev);//用来拦截eventpublic boolean onTouchEvent(MotionEventev);//用来处理event




总结:

1. Android事件分发是先传递到ViewGroup,再由ViewGroup传递到View的。

2. 在ViewGroup中可以通过onInterceptTouchEvent方法对事件传递进行拦截,onInterceptTouchEvent方法返回true代表不允许事件继续向子View传递,返回false代表不对事件进行拦截,默认返回false。

3.拦截的好好处在于调用谁的dispatchTouchEvent的方法,谁出来点击事件

4. 子View中如果将传递的事件消费掉,ViewGroup中将无法接收到任何事件。

5.在ViewGroup中onInterceptTouchEvent方法若反回false,那么触屏事件会继续向下传递,
但如果没有子View去处理这个事件,即子view的onTouchEvent没有返回True
则最后还是由ViewGroup去处理这个事件,也就又执行了自己的onTouchEvent。

下面是方法总结:
OnTouch方法是在Activity里面设置的监听事件



1. onTouch和onTouchEvent有什么区别,又该如何使用?

从源码中可以看出,这两个方法都是在View的dispatchTouchEvent中调用的,onTouch优先于onTouchEvent执行。如果在onTouch方法中通过返回true将事件消费掉,onTouchEvent将不会再执行。


思考的问题:

1.View的dispatchTouchEvent重写了会怎么样,View的onTouchEvent重写了会怎样?

2.ViewGroup的dispatchTouchEvent重写了会怎么样,ViewGroup的onTouchEvent重写了会怎样?

3.ListView上的button,点击button2个控件同时要有相应,应该怎么处理?

4.


事件的传递顺序:

G----->A----->B----->C----->D----->E----->F





0 0