内部 Recyclerview 未收到点击事件
Inner Recyclerview not receiving click event
我有一个父 RecyclerView,里面装满了卡片,每张卡片都有一个内部 RecyclerView。
卡片内的内部 recyclerview 的滚动已被禁用,但这也影响了内部 recyclerview 接收点击事件的能力。
要禁用滚动,我采用了与 Lucas Crawford 此处的回答非常相似的方法,建议您创建自定义 recyclerview class 并覆盖 dispatchTouchEvent:Disable Scrolling in child Recyclerview android
我的 class 看起来像这样:
public class ScrollThroughRecyclerView extends RecyclerView {
private boolean isOnClick;
public ScrollThroughRecyclerView(Context context) {
super(context);
}
public ScrollThroughRecyclerView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public ScrollThroughRecyclerView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
final int action = ev.getAction();
final int actionMasked = action & MotionEvent.ACTION_MASK;
Log.e("actionmasked", "" + actionMasked);
switch (actionMasked) {
case MotionEvent.ACTION_DOWN:
Log.e("motionDown", "onclick: "+isOnClick);
isOnClick = true;
break;
case MotionEvent.ACTION_MOVE:
isOnClick = false;
Log.e("motionMove", "onclick: "+isOnClick);
return true;
case MotionEvent.ACTION_UP:
Log.e("motionUp", "onclick: "+isOnClick);
if (isOnClick) {
return super.dispatchTouchEvent(ev);
}
break;
default:
return true;
}
return true;
}
但是,它从不注册点击事件,但滚动被正确禁用。
如何让内部recyclerview注册点击事件?
所以经过大约一天的时间尝试解决这个问题,我终于设法解决了这个问题。
完成所有这些步骤后,您应该得到:
- 不受内部回收器视图滚动事件阻碍的外部回收器视图 - 禁用内部回收器视图中的滚动。这在您想要将折叠工具栏方法与外部和内部 recyclerview 一起使用的情况下很有用。
- 可以点击内部recyclerview中的元素(如果无法点击,内部recyclerview有什么用,显示项目列表)。
首先,我在android看了几个关于触摸的视频:https://www.youtube.com/watch?v=SYoN-OvdZ3M&list=PLonJJ3BVjZW6CtAMbJz1XD8ELUs1KXaTD&index=19
我从这里获得了视频 link:Android: Difference between onInterceptTouchEvent and dispatchTouchEvent?
现在,我必须自定义我的 onDispatchTouchEvent:
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
final int action = ev.getAction();
final int actionMasked = action & MotionEvent.ACTION_MASK;
Log.e("actionmasked", "" + actionMasked);
switch (actionMasked) {
case MotionEvent.ACTION_DOWN:
return super.dispatchTouchEvent(ev);
case MotionEvent.ACTION_MOVE:
return true;
case MotionEvent.ACTION_CANCEL:
return true;
case MotionEvent.ACTION_UP:
return super.dispatchTouchEvent(ev);
default:
return super.dispatchTouchEvent(ev);
}
我已经在 DOWN 和 UP 以及默认情况下调用了 super.dispatchTouchEvent(ev),因为我们希望内部 recyclerview 的 child 来处理这些事件。事件必须从这个自定义回收视图 (ScrollThroughRecyclerView) 的视图组到回收视图中的视图。
对于 MOVE 和 CANCEL,我们 return 正确地说内部 recyclerview 已经处理了那些事件并且事件可以 return 返回到 parent 这将允许外部 recyclerview正确滚动。这不会妨碍折叠工具栏的应用程序行为。
现在我们需要一个自定义的 onInterceptTouchEvent 方法:
@Override
public boolean onInterceptTouchEvent(MotionEvent e) {
final int action = e.getAction();
final int actionMasked = action & MotionEvent.ACTION_MASK;
if (actionMasked == MotionEvent.ACTION_DOWN) {
return false; //intercepted by the viewgroup and passed down to child
} else if (actionMasked == MotionEvent.ACTION_UP) {
return false; //intercepted by the viewgroup and passed down to child
}
return super.onInterceptTouchEvent(e);
}
对于 UP 和 DOWN,我们 return false 因为我们希望内部 recyclerview 中的 child 处理这些事件(从 UP 和 DOWN,我们可以确定 recyclerview 中的哪个项目是实际点击了)。
对于其他一切,我们使用默认行为,所以我调用了:super.onInterceptTouchEvent(e)
现在在 recyclerview 适配器中:
@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, final int position) {
if (holder instanceof PostViewHolder) {
((PostViewHolder) holder).rl.setOnTouchListener(onTouchListener);
}
在您正在收听 recyclerview 中的触摸事件的视图上设置一个 touchlistener。对我来说,因为我正在监听 recyclerview 整行的点击,所以我在 rl 上设置了触摸监听器,它代表 recyclerview 中显示的行的 relativeLayout。
touchlistener不会收到viewgroup在onInterceptTouchEvent方法中传递过来的DOWN和UP动作事件。
因为我们只有一个 DOWN 和一个 UP 运动事件,检测点击可能比一般的 setOnClickListener 方法更乏味。此外,由于您使用的是 Touch,因此 Touch 实际上会覆盖 onClickListener,并且不会在 onClickListener 上调用任何内容。要通过 onTouchListener 检测对 recyclerview 中项目的点击,您将需要此方法:
View.OnTouchListener onTouchListener = new View.OnTouchListener() {
private float startX;
private float startY;
private static final int CLICK_ACTION_THRESHHOLD = 5;
@Override
public boolean onTouch(View v, MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
startX = event.getX();
startY = event.getY();
break;
case MotionEvent.ACTION_UP: {
float endX = event.getX();
float endY = event.getY();
if (isAClick(startX, endX, startY, endY)) {
Log.e("UserClick", "user has clicked");// WE HAVE A CLICK!!
}
break;
}
default:
return true;
}
return true;
}
private boolean isAClick(float startX, float endX, float startY, float endY) {
float differenceX = Math.abs(startX - endX);
float differenceY = Math.abs(startY - endY);
if (differenceX > CLICK_ACTION_THRESHHOLD || differenceY > CLICK_ACTION_THRESHHOLD) {
return false;
}
return true;
}
};
我有一个父 RecyclerView,里面装满了卡片,每张卡片都有一个内部 RecyclerView。
卡片内的内部 recyclerview 的滚动已被禁用,但这也影响了内部 recyclerview 接收点击事件的能力。
要禁用滚动,我采用了与 Lucas Crawford 此处的回答非常相似的方法,建议您创建自定义 recyclerview class 并覆盖 dispatchTouchEvent:Disable Scrolling in child Recyclerview android
我的 class 看起来像这样:
public class ScrollThroughRecyclerView extends RecyclerView {
private boolean isOnClick;
public ScrollThroughRecyclerView(Context context) {
super(context);
}
public ScrollThroughRecyclerView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public ScrollThroughRecyclerView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
final int action = ev.getAction();
final int actionMasked = action & MotionEvent.ACTION_MASK;
Log.e("actionmasked", "" + actionMasked);
switch (actionMasked) {
case MotionEvent.ACTION_DOWN:
Log.e("motionDown", "onclick: "+isOnClick);
isOnClick = true;
break;
case MotionEvent.ACTION_MOVE:
isOnClick = false;
Log.e("motionMove", "onclick: "+isOnClick);
return true;
case MotionEvent.ACTION_UP:
Log.e("motionUp", "onclick: "+isOnClick);
if (isOnClick) {
return super.dispatchTouchEvent(ev);
}
break;
default:
return true;
}
return true;
}
但是,它从不注册点击事件,但滚动被正确禁用。
如何让内部recyclerview注册点击事件?
所以经过大约一天的时间尝试解决这个问题,我终于设法解决了这个问题。
完成所有这些步骤后,您应该得到:
- 不受内部回收器视图滚动事件阻碍的外部回收器视图 - 禁用内部回收器视图中的滚动。这在您想要将折叠工具栏方法与外部和内部 recyclerview 一起使用的情况下很有用。
- 可以点击内部recyclerview中的元素(如果无法点击,内部recyclerview有什么用,显示项目列表)。
首先,我在android看了几个关于触摸的视频:https://www.youtube.com/watch?v=SYoN-OvdZ3M&list=PLonJJ3BVjZW6CtAMbJz1XD8ELUs1KXaTD&index=19
我从这里获得了视频 link:Android: Difference between onInterceptTouchEvent and dispatchTouchEvent?
现在,我必须自定义我的 onDispatchTouchEvent:
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
final int action = ev.getAction();
final int actionMasked = action & MotionEvent.ACTION_MASK;
Log.e("actionmasked", "" + actionMasked);
switch (actionMasked) {
case MotionEvent.ACTION_DOWN:
return super.dispatchTouchEvent(ev);
case MotionEvent.ACTION_MOVE:
return true;
case MotionEvent.ACTION_CANCEL:
return true;
case MotionEvent.ACTION_UP:
return super.dispatchTouchEvent(ev);
default:
return super.dispatchTouchEvent(ev);
}
我已经在 DOWN 和 UP 以及默认情况下调用了 super.dispatchTouchEvent(ev),因为我们希望内部 recyclerview 的 child 来处理这些事件。事件必须从这个自定义回收视图 (ScrollThroughRecyclerView) 的视图组到回收视图中的视图。
对于 MOVE 和 CANCEL,我们 return 正确地说内部 recyclerview 已经处理了那些事件并且事件可以 return 返回到 parent 这将允许外部 recyclerview正确滚动。这不会妨碍折叠工具栏的应用程序行为。
现在我们需要一个自定义的 onInterceptTouchEvent 方法:
@Override
public boolean onInterceptTouchEvent(MotionEvent e) {
final int action = e.getAction();
final int actionMasked = action & MotionEvent.ACTION_MASK;
if (actionMasked == MotionEvent.ACTION_DOWN) {
return false; //intercepted by the viewgroup and passed down to child
} else if (actionMasked == MotionEvent.ACTION_UP) {
return false; //intercepted by the viewgroup and passed down to child
}
return super.onInterceptTouchEvent(e);
}
对于 UP 和 DOWN,我们 return false 因为我们希望内部 recyclerview 中的 child 处理这些事件(从 UP 和 DOWN,我们可以确定 recyclerview 中的哪个项目是实际点击了)。
对于其他一切,我们使用默认行为,所以我调用了:super.onInterceptTouchEvent(e)
现在在 recyclerview 适配器中:
@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, final int position) {
if (holder instanceof PostViewHolder) {
((PostViewHolder) holder).rl.setOnTouchListener(onTouchListener);
}
在您正在收听 recyclerview 中的触摸事件的视图上设置一个 touchlistener。对我来说,因为我正在监听 recyclerview 整行的点击,所以我在 rl 上设置了触摸监听器,它代表 recyclerview 中显示的行的 relativeLayout。
touchlistener不会收到viewgroup在onInterceptTouchEvent方法中传递过来的DOWN和UP动作事件。
因为我们只有一个 DOWN 和一个 UP 运动事件,检测点击可能比一般的 setOnClickListener 方法更乏味。此外,由于您使用的是 Touch,因此 Touch 实际上会覆盖 onClickListener,并且不会在 onClickListener 上调用任何内容。要通过 onTouchListener 检测对 recyclerview 中项目的点击,您将需要此方法:
View.OnTouchListener onTouchListener = new View.OnTouchListener() {
private float startX;
private float startY;
private static final int CLICK_ACTION_THRESHHOLD = 5;
@Override
public boolean onTouch(View v, MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
startX = event.getX();
startY = event.getY();
break;
case MotionEvent.ACTION_UP: {
float endX = event.getX();
float endY = event.getY();
if (isAClick(startX, endX, startY, endY)) {
Log.e("UserClick", "user has clicked");// WE HAVE A CLICK!!
}
break;
}
default:
return true;
}
return true;
}
private boolean isAClick(float startX, float endX, float startY, float endY) {
float differenceX = Math.abs(startX - endX);
float differenceY = Math.abs(startY - endY);
if (differenceX > CLICK_ACTION_THRESHHOLD || differenceY > CLICK_ACTION_THRESHHOLD) {
return false;
}
return true;
}
};