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

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

public class MyViewPager extends ViewPager {
    private int startX;
    private int startY;
    public MyViewPager(Context context) {
        super(context);
    }


    @Override
    public boolean onInterceptTouchEvent(MotionEvent ev) {
        switch (ev.getAction())
        {
            case MotionEvent.ACTION_DOWN:
                startX= (int) ev.getX();
                startY= (int) ev.getY();
                break;
            case MotionEvent.ACTION_MOVE:

                int dX= (int) (ev.getX()-startX);
                int dY= (int) (ev.getY()-startX);
                if(Math.abs(dX)>Math.abs(dY)){//左右滑动
                    return true;
                }else {//上下滑动
                    return false;
                }
            case MotionEvent.ACTION_UP:
                break;
        }
        return super.onInterceptTouchEvent(ev);
    }
}

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

android so冲突解决 安卓事件冲突_上下滑动

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

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

a. 先要在MotionEvent.ACTION_DOWN:的时候使用mHorizontalEx2.requestDisallowInterceptTouchEvent(true);不允许父类拦截,这样的话,下一个事件到来时,就交给子控件了。
b. 然后在MotionEvent.ACTION_MOVE: 根据业务逻辑判断,当该事件为父类需要的处理时,调用mHorizontalEx2.requestDisallowInterceptTouchEvent(false);来决定父控件拦截事件。

下面来看具体代码:

public boolean dispatchTouchEvent(MotionEvent event) {
        ...

        switch (action) {
            case MotionEvent.ACTION_DOWN:  
                      getParent().requestDisallowInterceptTouchEvent(true);
                break;
            case MotionEvent.ACTION_MOVE:
                if(父控件需要处理此事件)     
                                        getParent().requestDisallowInterceptTouchEvent(false);                   
                break;
            case MotionEvent.ACTION_UP: {
                break;
        }
        ...
        return super.dispatchTouchEvent(event);
;
    }

当然,还需要修改父容器的onInterceptTouchEvent()方法,代码如下:
a、我们要重写父控件的onInterceptTouchEvent,在ACTION_DOWN的时候返回false,负责的话子View调用requestDisallowInterceptTouchEvent也将无能为力。
b、还有就是其他事件的话都返回true,这样就把能否拦截事件的权利交给了子View。

@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {

int action=ev.getAction();
        if(action==MotionEvent.ACTION_DOWN){
            return false;
        }else {
            return true;
        }
    }

运行结果如图:

android so冲突解决 安卓事件冲突_android so冲突解决_02

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