ScrollView中事件监听和GestureDetector手势监听冲突

来源:互联网 发布:学习数据库系统工程师 编辑:程序博客网 时间:2024/06/08 19:35

在项目的开发中,同时用到了在屏幕上向左滑动或者向右滑动和ScrollView的滚动滑动,但是实际上两个同时存在的时候,并且要在ScrollView上左滑或者右滑时,屏幕的手势监听不起作用;如果手势的滑动不在ScrollView上,而两个监听事件属于屏幕的不同区域时,并不会出现某个监听不起作用。这时解决方法就是要在Activity的dispatchTouchEvent事件中对GestureDetector手势监听重新分发一下,主要的代码如下:

@Override      public boolean dispatchTouchEvent(MotionEvent ev) {          switch (ev.getAction()) {        case MotionEvent.ACTION_DOWN:            System.out.println("main---dispatchTouchEvent---DOWN");            break;        case MotionEvent.ACTION_UP:            System.out.println("main---dispatchTouchEvent---UP");            break;        }        mGestureDetector.onTouchEvent(ev); //对手势的监听重新分发        return super.dispatchTouchEvent(ev);      }

红框地方手势左右滑动可以切换,但是在篮框上手势没有任何作用
红框地方手势左右滑动可以切换,但是在篮框上手势没有任何作用
但是在Activity的dispatchTouchEvent事件中对GestureDetector手势监听重新分发以后,整个屏幕都可以进行手势的监听,ScrollView的上下滑动也没有影响。

Android 中 Touch 事件的分发和消费机制

Android 中与 Touch 事件相关的方法包括:dispatchTouchEvent(MotionEvent ev)、onInterceptTouchEvent(MotionEvent ev)、onTouchEvent(MotionEvent ev);能够响应这些方法的控件包括:ViewGroup 及其子类、Activity。方法与控件的对应关系如下所示:
这里写图片描述
**

  1. 需要注意的是 View 对 dispatchTouchEvent(MotionEvent ev)
    、onInterceptTouchEvent(MotionEvent ev) 的响应的前提是可以向该 View 中添加子
    View,如果当前的 View 已经是一个最小的单元 View(比如 TextView),那么就无法向这个最小 View 中添加子
    View,也就无法向子 View 进行事件的分发和拦截,所以它没有 dispatchTouchEvent(MotionEvent ev)
    和 onInterceptTouchEvent(MotionEvent ev),只有 onTouchEvent(MotionEvent
    ev)。
  2. onInterceptTouchEvent(MotionEvent ev)只有在ViewGroup中有,Activity没有这个函数。如果拦截掉,就不会传送到子View中去。

**
事件分发:public boolean dispatchTouchEvent(MotionEvent ev)
Touch 事件发生时 Activity 的 dispatchTouchEvent(MotionEvent ev) 方法会以隧道方式(从根元素依次往下传递直到最内层子元素或在中间某一元素中由于某一条件停止传递)将事件传递给最外层 View 的 dispatchTouchEvent(MotionEvent ev) 方法,并由该 View 的 dispatchTouchEvent(MotionEvent ev) 方法对事件进行分发。dispatchTouchEvent 的事件分发逻辑如下:

  1. 如果 return true,事件会分发给当前 View 并由 dispatchTouchEvent
    方法进行消费,同时事件会停止向下传递;
  2. 如果 return false,事件分发分为两种情况: 如果当前 View 获取的事件直接来自 Activity,则会将事件返回给 Activity 的 onTouchEvent 进行消费; 如果当前 View 获取的事件来自外层父控件,则会将事件返回给父 View 的 onTouchEvent 进行消费。
  3. 如果返回系统默认的super.dispatchTouchEvent(ev),事件会自动的分发给当前 View 的onInterceptTouchEvent 方法;如果是Activity中的返回super.dispatchTouchEvent(ev)则对应的再由子 View 的 dispatchTouchEvent 来开始这个事件的分发。

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

  1. 如果 onInterceptTouchEvent 返回 true,则表示将事件进行拦截,并将拦截到的事件交由当前 View 的
    onTouchEvent 进行处理;
  2. 如果 onInterceptTouchEvent 返回 false,则表示将事件放行,当前 View 上的事件会被传递到子 View上,再由子 View 的 dispatchTouchEvent 来开始这个事件的分发;
  3. 如果 onInterceptTouchEvent 返回super.onInterceptTouchEvent(ev),事件默认会被拦截,并将拦截到的事件交由当前 View 的onTouchEvent 进行处理。

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

  1. 如果事件传递到当前 View 的 onTouchEvent 方法,而该方法返回了 false,那么这个事件会从当前 View 向上传递,并且都是由上层 View 的 onTouchEvent 来接收。
  2. 如果传递到上面的 onTouchEvent 也返回 false,这个事件就会“消失”,而且接收不到下一次事件。
  3. 如果返回了 true 则会接收并消费该事件。 如果返回 super.onTouchEvent(ev) 默认处理事件的逻辑和返回false 时相同。
阅读全文
1 0
原创粉丝点击