首先来看一个栗子吧~
自定义了一个ViewGroup,名叫HorizontalScrollView,里面的子View横向排布,在ViewGroup上左右滑动时里面的子View会跟着滑动,手指抬起时,根据页面滑动的距离动态滑动到相应子View(手指抬起时哪个子View占据空间大则滑动到那个子View)。
主要复写了ViewGroup的onMeasure、onLayout、onTouchEvent方法:
① 复写onMeasure方法:
这里要手动调一次measureChildren方法来测量子View的宽高,因为super.onMeasure中调用的是View的onMeasure方法,只会测量自身的宽高。
② 复写onLayout方法:
在onLayout方法,需要调用layout方法来确定子View的位置,这里我将子View横向排放。这里还需要获取子View最左边的边界和最右边的边界(后面滑动判断边界时会用到)。
③ 复写onTouchEvent方法:
在Move状态中,有三种情况:
1.向右滑动过程中若已滑动到左边界,则调用scrollTo方法停留在左边界
2.向左滑动过程中若已滑动到右边界,则调用scrollTo方法停留在右边界
3.其余情况,调用scrollBy方法进行滑动,手机滑动多少页面就滑动多少
在up状态时,若当前子View的宽度在界面中出现一半以上(我在布局中设置每个子View都是match_parent,子View完整展示时刚好是界面的宽度),则调用startScroll方法平滑地滑动到相应的子View。
在调用startScroll方法进行平滑移动时,需要复写computeScroll方法,如下所示
好了,在布局中给HorizontalScrollView添加三个ListView作为子View,如下所示
想像一下效果,左右滑动的时候ListView之间会进行切换,上下滑动的时候ListView也能正常滑动,美滋滋~运行一下看看咯^:^
额,不管怎么滑动,都是展示的是第一个ListView额,给跪orz。
其实这是一个滑动冲突的问题,要解决滑动冲突,首先要了解Android中的事件传递机制。关于这方面的知识,可以参看郭神的《事件分发机制》相关文章,文章里讲的很详细~
分析原因
出现上面情况的原因是,ViewGroup默认会将滑动事件分发给它的子View处理,而它的子View将滑动事件消耗了。所以,HorizontalViewGroup中的onTouchEvent中虽然做了滑动处理的相关逻辑,但是它根本就没有执行。
如何解决
解决的方法有两种,分别是:
1.外部拦截法
2.内部拦截法
在事件传递的流程中,父View在接收到事件后,其实可以对事件进行拦截处理,不让它传递给子View,查看ViewGroup类中dispatchTouchEvent方法中的源码(API 25),可以看到下面这段代码
可以看到,在里面调用了onInterceptToucheEvent方法来判断是否对事件进行拦截,因此可以通过重写onInterceptTouchEvent方法来对需要处理的事件进行拦截,这就是外部拦截的方法。
还有一个优先级更高的操作,那就是disallowIntercept标志,若disallowIntercept标记为true,则onInterceptTouchEvent方法根本就不会执行,那么这个flag要怎么设置呢?可以看到它和FLAG_DISALLOW_INTERCEPT有关,而FLAG_DISALLOW_INTERCEPT标志可以通过调用父View的requestDisallowInterceptTouchEvent来设置:
①父View.requestDisallowInterceptTouchEvent(true),则父View无法拦截;
②父View.requestDisallowInterceptTouchEvent(false),则父view可以在onInteceptTouchEvent方法中进行事件拦截。
而这个requestDisallowInterceptToucheEvent方法可以在子View中执行,当需要父View拦截时,调用上面的①,而当不需要父View拦截时,调用上面的②,这就是内部拦截法。
外部拦截法实践
重写父View的onInterceptTouchEvent方法,在move状态时判断,若x方向滑动的距离大于y方向滑动的距离,则拦截该事件进行处理,代码如下:
运行一下,看一下效果:(左右滑动时切换子View,每个列表都能正常地上下滑动和点击)
外部拦截成功(阴影是播放器的哈,不必在意)~
内部拦截法实践
1.重写父View的onInterceptTouchEvent方法,默认拦截除了down以外的所有事件。
2.自定义ListView,重写dispatchTouchEvent方法:
①在down时,不允许父View拦截
②在move时,若x方向滑动的距离大于y方向滑动的距离,允许父View拦截
代码如下:
运行一下看效果:
效果和外部拦截一样,内部拦截也成功啦~
回顾
外部拦截和内部拦截的效果是一样的,具体应用时要看拦截的逻辑在哪边写比较方便。在本例中外部拦截比内部拦截逻辑更清晰简洁,因此选用外部拦截会更好~
项目代码:滑动冲突解决实践