1.解决事件冲突的主要思路
- 1.down事件首先会传递到onInterceptTouchEvent()方法
- 2.如果该ViewGroup的OnInterceptTouchEvent()在接收到down事件处理完成之后return false,那么后续的move,up等事件,将继续会先传递给该ViewGroup,之后才和down事件一样传递给最终的目标view的onTouchEvent()处理
- 3.如果该ViewGroup的onTnterceptTouchEvent()在接收到down事件处理完成之后return true,那么后续的move,up等事件,将不再传递给onInterceptTouchEvent(),而是和down事件一样传递给该ViewGroup的onTouchEvent()处理,注意,目标view
将接收不到任何事件 - 4.如果最终需要处理事件的view的onTouchEvent()返回了false ,那么该事件将被传递至其上一层次的view的onTouchEvent()处理
- 5.如果最终需要处理事件的view 的onTouchEvent()返回了true,那么后续事件将可以继续传递给该view的onTouchEvent()处理
2.主要方法:外部拦截法和内部拦截法
-
2.1外部拦截法:情景:一个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); } }
-
2.2内部拦截法:情景:一个ViewPager嵌套了一个ViewPager,两个都是左右滑动。这个时候我们可以用内部拦截法,来处理冲突。即重写子元素的dispatchTouchEvent()方法,并调用getParent().requestDisallowInterceptTouchEvent(true)是父容器不能拦截子元素需要的事件。下面来看具体代码:
public boolean dispatchTouchEvent(MotionEvent event) { ... switch (action) { case MotionEvent.ACTION_MOVE: getParent().requestDisallowInterceptTouchEvent(true); break; case MotionEvent.ACTION_MOVE: if(子元素需要处理此事件) getParent().requestDisallowInterceptTouchEvent(true); break; case MotionEvent.ACTION_UP: { break; } ... return super.dispatchTouchEvent(event); ; }
案例二:
public class HorizontalScrollViewPager extends ViewPager{
public HorizontalScrollViewPager(Context context) {
super(context);
}
public HorizontalScrollViewPager(Context context, AttributeSet attrs) {
super(context,attrs);
}
private float startX;
private float startY;
/**
* 事件分发
*/
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
// getParent().requestDisallowInterceptTouchEvent(true); //把事件传递给自己
switch(ev.getAction()) {
case MotionEvent.ACTION_DOWN:
//一定要把事件给自己
getParent().requestDisallowInterceptTouchEvent(true);
//1.记录起始坐标
startX = ev.getX();
startY = ev.getY();
break;
case MotionEvent.ACTION_MOVE:
//1.来到新的坐标
float endX = ev.getX();
float endY = ev.getY(); ev.getRawX();
//2.计算偏移量
float distanceX = endX - startX;
float distanceY = endY - startY;
//3.判断滑动方向
if(Math.abs(distanceX) > Math.abs(distanceY)) {
//水平方向滑动
//1.如果第0个位置,并且滑动方向是从左到右滑动
//getParent().requestDisallowInterceptTouchEvent(false);
if(getCurrentItem() == 0 && distanceX >0 ) {
getParent().requestDisallowInterceptTouchEvent(false);
}
// 2.如果是页签页面的最后一个位置,并且滑动方向是从右向左滑动
//getParent().requestDisallowInterceptTouchEvent(false);
else if (getCurrentItem() == (getAdapter().getCount()-1) && distanceX <0) {
getParent().requestDisallowInterceptTouchEvent(false);
}
// 3.其他中间部分
// getParent().requestDisallowInterceptTouchEvent(true);*/
else {
getParent().requestDisallowInterceptTouchEvent(true);
}
} else {
//竖值方向滑动
getParent().requestDisallowInterceptTouchEvent(false);
}
break;
case MotionEvent.ACTION_UP:
break;
}
return super.dispatchTouchEvent(ev);
}
}