Inner Recyclerview未收到点击事件

时间:2015-08-30 11:02:43

标签: android android-recyclerview motionevent

我有一个父级回收者视图,里面装满了卡片和每张卡片内部的回收者视图。

卡片内部回收视图的滚动已被禁用,但这也影响了内部回收者视图接收点击事件的能力。

为了禁用滚动,我跟着一个非常类似的接近Lucas Crawford的回答,建议您创建一个自定义的recyclerview类并覆盖dispatchTouchEvent:Disable Scrolling in child Recyclerview android

我的班级看起来像这样:

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注册点击事件?

1 个答案:

答案 0 :(得分:4)

所以经过大约一天试图解决这个问题后,我终于设法解决了这个问题。

完成所有这些步骤后,您最终应该:

  • 外部回收者视图不受内部回收者视图的滚动事件的阻碍 - 在内部回收者视图中滚动被禁用。当您想要使用折叠工具栏方法同时使用外部和内部回收站视图时,这非常有用。
  • 可以点击内部回收站视图中的元素(如果无法点击,则显示项目列表的内部回收站视图用途如何。)

首先,我在Android上观看了一些关于触控主题的视频:https://www.youtube.com/watch?v=SYoN-OvdZ3M&list=PLonJJ3BVjZW6CtAMbJz1XD8ELUs1KXaTD&index=19

我从这里获得了视频链接: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的子进程处理这些事件。该事件必须从viewgroup(自定义recyclelerview(ScrollThroughRecyclerView))转到recyclerview中的视图。

对于MOVE和CANCEL,我们返回true表示内部的recyclerview已处理这些事件,并且事件可以返回到父级,这将允许外部的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,我们返回false,因为我们希望内部recyclelerview中的子项处理这些事件(从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设置到您正在侦听触摸事件的视图上。对我而言,由于我正在收听recyclelerview整条线路上的点击,因此我在rl上设置了触摸侦听器,它表示在recyclerview中显示的行的relativeLayout。

touchlistener不会接收视图组在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;
    }
};