事件处理---滑动冲突解决

首先来看一个栗子吧~


自定义了一个ViewGroup,名叫HorizontalScrollView,里面的子View横向排布,在ViewGroup上左右滑动时里面的子View会跟着滑动,手指抬起时,根据页面滑动的距离动态滑动到相应子View(手指抬起时哪个子View占据空间大则滑动到那个子View)。


主要复写了ViewGroup的onMeasure、onLayout、onTouchEvent方法:

① 复写onMeasure方法

事件处理---滑动冲突解决_第1张图片

这里要手动调一次measureChildren方法来测量子View的宽高,因为super.onMeasure中调用的是View的onMeasure方法,只会测量自身的宽高。


② 复写onLayout方法

事件处理---滑动冲突解决_第2张图片

在onLayout方法,需要调用layout方法来确定子View的位置,这里我将子View横向排放。这里还需要获取子View最左边的边界和最右边的边界(后面滑动判断边界时会用到)。


③ 复写onTouchEvent方法

事件处理---滑动冲突解决_第3张图片
事件处理---滑动冲突解决_第4张图片

在Move状态中,有三种情况:

1.向右滑动过程中若已滑动到左边界,则调用scrollTo方法停留在左边界

2.向左滑动过程中若已滑动到右边界,则调用scrollTo方法停留在右边界

3.其余情况,调用scrollBy方法进行滑动,手机滑动多少页面就滑动多少

在up状态时,若当前子View的宽度在界面中出现一半以上(我在布局中设置每个子View都是match_parent,子View完整展示时刚好是界面的宽度),则调用startScroll方法平滑地滑动到相应的子View。

在调用startScroll方法进行平滑移动时,需要复写computeScroll方法,如下所示

事件处理---滑动冲突解决_第5张图片

好了,在布局中给HorizontalScrollView添加三个ListView作为子View,如下所示

事件处理---滑动冲突解决_第6张图片

想像一下效果,左右滑动的时候ListView之间会进行切换,上下滑动的时候ListView也能正常滑动,美滋滋~运行一下看看咯^:^

事件处理---滑动冲突解决_第7张图片

额,不管怎么滑动,都是展示的是第一个ListView额,给跪orz。


其实这是一个滑动冲突的问题,要解决滑动冲突,首先要了解Android中的事件传递机制。关于这方面的知识,可以参看郭神的《事件分发机制》相关文章,文章里讲的很详细~



分析原因

出现上面情况的原因是,ViewGroup默认会将滑动事件分发给它的子View处理,而它的子View将滑动事件消耗了。所以,HorizontalViewGroup中的onTouchEvent中虽然做了滑动处理的相关逻辑,但是它根本就没有执行。



如何解决

解决的方法有两种,分别是:

1.外部拦截法

2.内部拦截法


在事件传递的流程中,父View在接收到事件后,其实可以对事件进行拦截处理,不让它传递给子View,查看ViewGroup类中dispatchTouchEvent方法中的源码(API 25),可以看到下面这段代码

事件处理---滑动冲突解决_第8张图片

可以看到,在里面调用了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方向滑动的距离,则拦截该事件进行处理,代码如下:

事件处理---滑动冲突解决_第9张图片

运行一下,看一下效果:(左右滑动时切换子View,每个列表都能正常地上下滑动和点击)

事件处理---滑动冲突解决_第10张图片

外部拦截成功(阴影是播放器的哈,不必在意)~



内部拦截法实践

1.重写父View的onInterceptTouchEvent方法,默认拦截除了down以外的所有事件。

2.自定义ListView,重写dispatchTouchEvent方法:

①在down时,不允许父View拦截

②在move时,若x方向滑动的距离大于y方向滑动的距离,允许父View拦截

代码如下:

事件处理---滑动冲突解决_第11张图片
事件处理---滑动冲突解决_第12张图片

运行一下看效果:

事件处理---滑动冲突解决_第13张图片

效果和外部拦截一样,内部拦截也成功啦~



回顾

外部拦截和内部拦截的效果是一样的,具体应用时要看拦截的逻辑在哪边写比较方便。在本例中外部拦截比内部拦截逻辑更清晰简洁,因此选用外部拦截会更好~


项目代码:滑动冲突解决实践

你可能感兴趣的:(事件处理---滑动冲突解决)