问题描述
项目中遇到了一个RecyclerView内嵌套RecyclerView,内层RecyclerView的区域无法响应所在Item的点击事件的问题( RecyclerView内嵌套RecyclerView导致外层item点击不响应)。
首先,需要知道触摸事件的响应机制是怎么样的:由上至下,最下层不消费后,则由下至上;然后需要了解一下这三个方法:dispatchTouchEvent、onInterceptTouchEvent、onTouchEvent。
dispatchTouchEvent:事件分发,一般不处理,返回false,事件到onInterceptTouchEvent中处理。
onInterceptTouchEvent:事件拦截,返回true的话,则不向下传递,事件到onTouchEvent,返回false事件往下传递
onTouchEvent:返回true代表事件消费,返回false不消费,事件往上传递。
只需要内部RecyclerView用于显示,不需要任何操作的情况下,为了使外层RecyclerView的item响应,把嵌入的RecyclerView触摸事件拦截,并且不消费就行了,事件就会传递到上一层,重写嵌套的RecyclerView
解决方式(形式不同,原理相同)
方式一
viewHolder.rv.setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
return viewHolder.itemView.onTouchEvent(event);
}
});
给内层的RecyclerView设置OnTouchListener,在onTouch中处理外层RecyclerView的Item的触摸事件onTouchEvent,item已经设置了点击事件,所以在onTouchEvent的事件处理中会调用到item的点击事件。
方式二
viewHolder.rv.setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
if(event.getAction() == MotionEvent.ACTION_UP){
viewHolder.itemView.performClick();
}
return false;
}
});
依然是给内层的RecyclerView设置OnTouchListener,在onTouch中调用外层RecyclerView的Item的preformClick方法,执行外层item的点击事件。
方式三(其实就是把方式二加了些判断封装了下)
viewHolder.rv.setOnTouchListener(new ChildTouchListener(viewHolder.itemView));
public class ChildTouchListener implements View.OnTouchListener{
private float downX;
private float downY;
private float touchSlop;
private View view;
public ChildTouchListener(View view){
this.view = view;
touchSlop = ViewConfiguration.get(view.getContext()).getScaledTouchSlop();
}
@Override
public boolean onTouch(View v, MotionEvent event) {
switch (event.getActionMasked()){
case MotionEvent.ACTION_DOWN:
downX = event.getX();
downY = event.getY();
break;
case MotionEvent.ACTION_UP:
if (Math.abs(event.getY() - downY) < touchSlop && Math.abs(event.getX() - downX) < touchSlop){
view.performClick();
}
break;
}
return false;
}
}
依然是给内层的RecyclerView设置OnTouchListener,在onTouch中调用外层RecyclerView的Item的preformClick方法,执行外层item的点击事件,就是加了些判断。
总结
我们上面也说了,其实原理是相同的,里面的RecyclerView区域点击没响应,是因为RecyclerView内部重写了onTouchEvent方法,导致了上述问题的发生,然后我们给RecyclerView设置OnTouchListener,那么OnTouchListener中的onTouch方法会在onTouchEvent方法之前回调,并且需要注意onTouch方法的返回值,如果是false,onTouchEvent方法才会被调用,如果是true,那么onTouchEvent方法将不会被调用(也就是给view设置的OnTouchListener的优先级高于onTouchEvent),我们的处理中返回false以便不影响RecyclerView后续onTouchEvent中的固有逻辑。另外onTouchEvent方法中,如果当前View设置的有OnClickListener,那么它的onClick方法会被调用,这也就是方法1中直接调用了itemView的onTouchEvent方法,自然后续会调用到我们设置的OnClickListener中的onClick方法,执行我们想要的业务逻辑。