android 事件分发机制

来源:互联网 发布:如何玩淘宝众筹 编辑:程序博客网 时间:2024/06/05 19:13

本次做了一个小的仿qq主页面的小demo,用的是viewpager+fragment+recycleview,但是在处理禁止侧滑与recycleview的item长按删除时出现事件冲突,故,又重新温习了一下事件分发过程也记录一下小心得。
一般都知道事件分发主要会有三个函数dispatchTouchEvent、onInterceptTouchEvent、 onTouchEvent。除此之外还有一个监听里面的OnTouch。
一、Touch 事件分析

▐ 事件分发:public boolean dispatchTouchEvent(MotionEvent ev)

Touch 事件发生时 Activity 的 dispatchTouchEvent(MotionEvent ev) 方法会以隧道方式(从根元素依次往下传递直到最内层子元素或在中间某一元素中由于某一条件停止传递)将事件传递给最外层 View 的 dispatchTouchEvent(MotionEvent ev) 方法,并由该 View 的 dispatchTouchEvent(MotionEvent ev) 方法对事件进行分发。dispatchTouchEvent 的事件分发逻辑如下:
如果 return true,事件会分发给当前 View 并由 dispatchTouchEvent 方法进行消费,所以整个分发响应过程中,到了该dispatchTouchEvent则终止了,同时事件会停止向下传递;
如果 return false,事件分发分为两种情况:(类似责任链向上传递,直到Acitivity层)
如果当前 View 获取的事件直接来自 Activity,则会将事件返回给 Activity 的 onTouchEvent 进行消费;
如果当前 View 获取的事件来自外层父控件,则会将事件返回给父 View 的 onTouchEvent 进行消费。
如果返回系统默认的 super.dispatchTouchEvent(ev),事件会自动的分发给当前 View 的 onInterceptTouchEvent 方法。
▐ 事件拦截:public boolean onInterceptTouchEvent(MotionEvent ev)

在外层 View 的 dispatchTouchEvent(MotionEvent ev) 方法返回系统默认的 super.dispatchTouchEvent(ev) 情况下,事件会自动的分发给当前 View 的 onInterceptTouchEvent 方法。onInterceptTouchEvent 的事件拦截逻辑如下:

如果 onInterceptTouchEvent 返回 true,则表示将事件进行拦截,并将拦截到的事件交由当前 View 的 onTouchEvent 进行处理;
如果 onInterceptTouchEvent 返回 false,则表示将事件放行,当前 View 上的事件会被传递到子 View 上,再由子 View 的 dispatchTouchEvent 来开始这个事件的分发;如果该viewgroup没有子view,则该viewgroup**先去执行acitivity中触摸监听中的ontouch方法,若ontouch返回true,则相当于该viewgroup层的dispatchTouchEvent返回true,事件将不会再分发下去直接在ontouch中消费完。当ontouch返回值为false,则继续执行viewgroup中的OntouchEvent方法。**
如果 onInterceptTouchEvent 返回 super.onInterceptTouchEvent(ev),事件默认不会被拦截,并将事件继续分发下去由子view处理。
▐ 事件响应:public boolean onTouchEvent(MotionEvent ev)

在 dispatchTouchEvent 返回 super.dispatchTouchEvent(ev) 并且 onInterceptTouchEvent 返回 true 或返回 super.onInterceptTouchEvent(ev) 的情况下 onTouchEvent 会被调用。onTouchEvent 的事件响应逻辑如下:

如果事件传递到当前 View 的 onTouchEvent 方法,而该方法返回了 false,那么这个事件会从当前 View 向上传递,并且都是由上层 View 的 onTouchEvent 来接收,如果传递到上面的 onTouchEvent 也返回 false,这个事件就会“消失”,而且接收不到下一次事件。
如果返回了 true 则会接收并消费该事件。
如果返回 super.onTouchEvent(ev) 默认处理事件的逻辑和返回 false 时相同。
到这里,与 Touch 事件相关的三个方法就分析完毕了。下面的内容会通过各种不同的的测试案例来验证上文中三个方法对事件的处理逻辑。
这里先附上相关代码:在自定义控件MyViewpager中重写了几个事件触发方法,目的是为了禁止侧滑,这里主要目的是测试事件分发过程,其他忽略。代码如下:
/**
* Created by lzy on 2017/2/22.
*/

public class MyViewPager extends ViewPager {

private boolean noScroll = false;public MyViewPager(Context context, AttributeSet attrs) {    super(context, attrs);    // TODO Auto-generated constructor stub}public MyViewPager(Context context) {    super(context);}public void setNoScroll(boolean noScroll) {    this.noScroll = noScroll;}@Overridepublic void scrollTo(int x, int y) {    super.scrollTo(x, y);}@Overridepublic boolean onTouchEvent(MotionEvent arg0) {    /* return false;//super.onTouchEvent(arg0); */    Log.i(MyViewPager.class.getSimpleName(), " onTouchEvent" + " event = " + arg0);  if (noScroll){     return false;      //return  super.onTouchEvent(arg0); }   else        return true;}@Overridepublic boolean onInterceptTouchEvent(MotionEvent arg0) {    Log.i(MyViewPager.class.getSimpleName(), " onInterceptTouchEvent" + " event = " + arg0);   if (noScroll) {       //return false;       return true;    }    else        return super.onInterceptTouchEvent(arg0);}@Overridepublic void setCurrentItem(int item, boolean smoothScroll) {    super.setCurrentItem(item, smoothScroll);}@Overridepublic void setCurrentItem(int item) {    super.setCurrentItem(item);}

}
`在activity中主要事件方法有:
public void InitViewPager(){
mPager = (MyViewPager)findViewById(R.id.vp_main);
mPager .setOnTouchListener( new View.OnTouchListener()
{
@Override
public boolean onTouch(View v, MotionEvent event)
{
Log.i(TAG, ” InitViewPager onTouch ” + event );
return false; //修改为true
}
});
mPager.setNoScroll(true);
}
当onInterceptTouchEvent返回值为true即事件被截断的时候调用顺序如下:先调用activity中该viewpager的触摸监听的ontouch方法返回false之后继续调用onTouchEvent消费事件

当onInterceptTouchEvent返回值为true,activity中该viewpager的触摸监听的ontouch方法返回true时,即activity中,触摸事件被它消费了,不再继续交给onTouchEvent处理
这里写图片描述
当onInterceptTouchEvent返回值为false即事件不会被截断的时候调用顺序如下,先分发到底下View执行长按事件,不会继续执行viewgroup层的ontouch()与ontouchevent()
这里写图片描述

2 0