有这样一个需求场景,要用百度地图,且要放在Fragment中,百度地图左右拖动就会和ViewPager的滑动事件冲突,这类似两可拖动控件的冲突,网上的方法大致如下:
1、父View继承ViewPager,重写方法,如:
public class CustomViewPager extends ViewPager {
private boolean isCanScroll = true;
public CustomViewPager(Context context) {
super(context);
}
public CustomViewPager(Context context, AttributeSet attrs) {
super(context, attrs);
}
public void setScanScroll(boolean isCanScroll){
this.isCanScroll = isCanScroll;
}
@Override
public void scrollTo(int x, int y){
if (isCanScroll){
super.scrollTo(x, y);
}
}
又如:
private boolean isCanScroll = true;
public void setScanScroll(boolean isCanScroll){
this.isCanScroll = isCanScroll;
}
@Override
public boolean onTouchEvent(MotionEvent ev) {
if (isCanScroll== false) {
return false;
} else {
return super.onTouchEvent(ev);
}
}
@Override
public boolean onInterceptTouchEvent(MotionEvent ev) {
if (isCanScroll== false) {
return false;
} else {
return super.onInterceptTouchEvent(ev);
}
}
又或者其它相似的实现,然后子View实现setOnTouchListener,在UP和DOWN中,通过设定isCanScroll的相反值,调用父View重写的方法控制
2、父View不做处理,子View实现setOnTouchListener,在UP和DOWN中,分别调用下面的方法(用于通知父控件是否拦截事件):
getParent().requestDisallowInterceptTouchEvent(true or false)
然而这两种方法对于我的场景有很大弊端!因为定义了是否拦截Touch事件,当未拦截时,子View的OnTouch事件有效,但是一切换到拦截状态,子View的OnTouch事件无效,即上面的代码会作废,并会一直保持这个状态,
后来,终于给我找到一篇文章(忘了名字),里面有个方法:
@Override
protected boolean canScroll(View v, boolean checkV, int dx, int x, int y) {
// Not satisfied with this method of checking...
// working on a more robust solution
if(v.getClass().getName().equals("maps.j.b")) {
return true;
}
}
其中“maps.j.b”不知道什么含义的,不是相当于魔数么?
然后我加了个断点,调试进去一看,v.getClass()就是可触发Scroll的控件的类名,如在MapView拖动,会先后两次进来,分别是ViewPager和MapView,那“maps.j.b”这个魔数换成“com.baidu.mapapi.map.MapView”是否可以了?一试,果然!!!
于是,纯净的解决冲突方法就是:
public class ViewPagerCompat extends ViewPager {
public ViewPagerCompat(Context context) {
super(context);
}
public ViewPagerCompat(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
protected boolean canScroll(View v, boolean checkV, int dx, int x, int y) {
if(v.getClass().getName().equals("com.baidu.mapapi.map.MapView")) {
return true;
}
//if(v instanceof MapView){
// return true;
//}
return super.canScroll(v, checkV, dx, x, y);
}
}
然后用这个ViewPagerCompat代替原来的ViewPager使用即可,从之衍生,其它类似组件冲突应该也可这样处理