Android滑动事件冲突

首先,我们假设这样一个场景:一个ViewPager里面嵌套一个ViewPager,内部滑动方向和外部滑动方向一样时,该怎么解决这一冲突呢? 
针对滑动冲突这里给出两种解决方案:外部拦截法,内部拦截法。

外部拦截法

情景:一个ViewPager嵌套了一个Listview,一个是左右滑动,一个上下滑动。这个时候我们可以用外部拦截法,来处理冲突。在父容器ViewPager中,重写onInterceptTouchEvent()方法,判断当左右滑动时就拦截事件,上下滑动就不拦截,将事件交由子元素Listview来处理。首先我们需要重写一个ViewPager,叫MyViewPager,然后重写onInterceptTouchEvent()方法。具体代码如下:

复制代码
 1 public class MyViewPager extends ViewPager {
 2     private int startX;
 3     private int startY;
 4     public MyViewPager(Context context) {
 5         super(context);
 6     }
 7 
 8 
 9     @Override
10     public boolean onInterceptTouchEvent(MotionEvent ev) {
11         switch (ev.getAction())
12         {
13             case MotionEvent.ACTION_DOWN:
14                 startX= (int) ev.getX();
15                 startY= (int) ev.getY();
16                 break;
17             case MotionEvent.ACTION_MOVE:
18 
19                 int dX= (int) (ev.getX()-startX);
20                 int dY= (int) (ev.getY()-startX);
21                 if(Math.abs(dX)>Math.abs(dY)){//左右滑动
22                     return true;
23                 }else {//上下滑动
24                     return false;
25                 }
26             case MotionEvent.ACTION_UP:
27                 break;
28         }
29         return super.onInterceptTouchEvent(ev);
30     }
31 }
复制代码

这样就解决这种情况下的滑动冲突, 程序演示入下图: 

上述代码是外部拦截的典型逻辑,只需要重写onInterceptTouchEvent()方法,修改父容器当前需要的事件即可。

 

内部拦截法

情景:一个ViewPager嵌套了一个ViewPager,两个都是左右滑动。这个时候我们可以用内部拦截法,来处理冲突。即重写子元素的dispatchTouchEvent()方法,并调用getParent().requestDisallowInterceptTouchEvent(true)是父容器不能拦截子元素需要的事件。下面来看具体代码:

复制代码
 1 public boolean dispatchTouchEvent(MotionEvent event) {
 2         ...
 3 
 4         switch (action) {
 5             case MotionEvent.ACTION_MOVE:
 6                         getParent().requestDisallowInterceptTouchEvent(true);
 7 
 8                 break;
 9             case MotionEvent.ACTION_MOVE:
10                 if(子元素需要处理此事件)
11                             getParent().requestDisallowInterceptTouchEvent(true);
12 
13                 break;
14             case MotionEvent.ACTION_UP: {
15                 break;
16         }
17         ...
18         return super.dispatchTouchEvent(event);
19 ;
20     }
复制代码

当然,还需要修改父容器的onInterceptTouchEvent()方法,代码如下:

复制代码
 1 @Override
 2     public boolean onInterceptTouchEvent(MotionEvent ev) {
 3 
 4             int action=ev.getAction();
 5             if(action==MotionEvent.ACTION_DOWN){
 6                 return false;
 7             }else {
 8                 return true;
 9             }
10         }
复制代码

运行结果如图:

以上就是两种解决滑动冲突的解决方案。

好文要顶 关注我 收藏该文
1
0
 
« 上一篇:Activity的启动模式
» 下一篇:Android消息机制之Handler
posted @ 2016-03-07 12:55 Vonnie_Jade 阅读(11310) 评论(3) 编辑 收藏


#1楼 2016-03-07 19:28 码魂  
为什么不引入官方支持库....
#2楼[楼主] 2016-03-08 18:42 Vonnie_Jade  
@ 码魂
引入支持库是可以解决,但是可以在自己处理中,了解一下Android的事件分发机制
#3楼 2016-06-12 08:53 糊涂虫jarrod  
用支持库怎么解决呢?我在scrollview里面套了一个pulltorefresh的listview 有冲突
发表评论

昵称:

评论内容:
引用 粗体 链接 缩进 代码 图片

posted @ 2019-04-13 20:38  天涯海角路  阅读(226)  评论(0)    收藏  举报