1.问题描述:
首先看一下:
2代表的是RecyclerView,1代表的是其他的布局。
问题场景是这样的:
外层是一个自定义的可下拉刷新的控件,它的内部嵌套一个RecyclerView(Recycleview上方还有一个布局)
出现的问题是:当RecyclerView可以上下滑动,当下拉到顶部的时候(把这里记为特殊状态),再下拉不触发RefreshLayoutView(外层)的刷新监听,如果在RecyclerView上方的那个布局下拉是能触发刷新的。
(需求是:RecyclerView可以上拉,当RecyclerView下拉滑动到顶部的时候,再下拉应该触发RefreshLayoutView刷新)
2.分析:
因为所有的事件分发都是在外层这个自定义控件中,发生在RecyclerView范围内下拉不触发刷新的原因,是由于在特殊状态下,外层没有将事件拦截,而是将事件分发给了下面的RecyclerView,RecyclerView默认将事件消费掉了。
3.解决问题:
为了解决这个问题,我在网上看了很多事件分发的原理,其中这篇文章我觉得写的不错,对我帮助挺大的:
https://www.jianshu.com/p/e99b5e8bd67b
接下来看一下问题的解决:
在外层的RefreshLayoutView的onInterceptTouchEvent方法中,在特殊状态情况下,选择将不讲事件分发,而是自己消费掉。
getChildView();
switch (ev.getAction()){
case MotionEvent.ACTION_DOWN:
downYY = ev.getY();
break;
case MotionEvent.ACTION_MOVE:
float curY = ev.getY();
float deltaY = curY - downYY;
//返回false表示已经到顶部了,并且是下拉的状态下,此时应该拦截,
if(!mChildView.canScrollVertically(-1)){//到顶部
if(deltaY >= 40){//下拉
canScrollVertically = false;
}else {
canScrollVertically = true;
}
}else {
canScrollVertically = true;
}
break;
case MotionEvent.ACTION_UP:
downY = Float.MAX_VALUE;
break;
}
if (!interceptAllMoveEvents) {//表示不向下拦截
return !disallowIntercept || !canScrollVertically;
}
// 如果设置了拦截所有move事件,即interceptAllMoveEvents为true
if (MotionEvent.ACTION_MOVE == ev.getAction()) {
return true;
}
return false;
核心是这个switch代码块
//当返回false的时候表示已经在顶部
mChildView.canScrollVertically(-1)
同时还要是有效的下拉操作的时候,将这个特殊状态的标志canScrollVertically 设为false,其他情况向下设为true。
当然了,这个可能只是适用于我遇到的这种情况,大家遇到问题的时候需要具体问题具体分析。