12

有什么办法可以设置onClickListener一个RecyclerView吗?

我有一个RecyclerView和一些孩子,并OnClickListener在 parent 上设置一个RecyclerView。但是,onClick当我单击该视图时不会触发。请参见下面的示例代码——我们希望获得对父项的点击,而不是对子项的点击。在这种情况下,我们不关心对项目的点击。

我试过做setFocusable(false),,,setClickable(false)setOnClickListener(null)孩子无济于事。无论如何,我不认为孩子们从父母那里窃取点击,因为当我点击没有孩子的区域时,点击也不会注册。

package com.formagrid.hellotest;

import android.app.Activity;
import android.os.Bundle;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

import java.util.Arrays;
import java.util.List;

public class HelloActivity extends Activity {

    private RecyclerView mRecyclerView;
    private RecyclerAdapter mAdapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_hello);

        mRecyclerView = (RecyclerView) findViewById(R.id.recycler_view);
        mRecyclerView.setLayoutManager(new LinearLayoutManager(this));
        mAdapter = new RecyclerAdapter(Arrays.asList("hi", "this", "is", "some", "text"));
        mRecyclerView.setAdapter(mAdapter);
        mRecyclerView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Log.d("patricia", view.toString());
            }
        });
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
    }

    @Override
    protected void onPause() {
        super.onPause();
    }

    @Override
    protected void onResume() {
        super.onResume();
    }

    public class RecyclerAdapter extends RecyclerView.Adapter<RecyclerAdapter.Holder> {

        public class Holder extends RecyclerView.ViewHolder {

            protected TextView textView;

            public Holder(TextView itemView) {
                super(itemView);
                this.textView = itemView;
            }

        }

        private List<String> contents;

        public RecyclerAdapter(List<String> contents) {
            this.contents = contents;
        }

        @Override
        public Holder onCreateViewHolder(ViewGroup parent, int viewType) {
            return new Holder(new TextView(parent.getContext()));
        }

        @Override
        public void onBindViewHolder(Holder holder, int position) {
            holder.textView.setText(contents.get(position));
        }

        @Override
        public int getItemCount() {
            return contents.size();
        }

    }

}
4

4 回答 4

7

有什么办法可以设置onClickListener一个RecyclerView吗?

不,也就是说,您可以设置一个OnClickListener,但RecyclerView永远不会调用它。RecyclerView拦截所有触摸事件,但从不调用performClick(),这就是View调用其侦听器的方式。

但是,您可以OnClickListener使用 anOnTouchListener和 a模拟 an GestureDetector。对于GestureDetector的监听器,我们可以使用 a SimpleOnGestureListener,只实现onSingleTapUp()方法。

class ClickListener extends GestureDetector.SimpleOnGestureListener {
    @Override
    public boolean onSingleTapUp(MotionEvent e) {
        Toast.makeText(HelloActivity.this, "Clicked", 0).show();
        return true;
    }
};

然后我们只需要从 an 中喂入GestureDetectors MotionEventOnTouchListener并检查 return 来决定是否消费该事件,以免干扰滚动、拖动等。

final GestureDetector detector = new GestureDetector(HelloActivity.this, new ClickListener());

mRecyclerView.setOnTouchListener(new OnTouchListener() {
        @Override
        public boolean onTouch(View v, MotionEvent event) {
            if(detector.onTouchEvent(event)) {
                return true;
            }
            return false;
        }
    }
);

请注意,上述解决方案几乎只适用于 "simple" RecyclerView,就像问题中描述和给出的那样。如果您正在使用更复杂的项目处理,例如拖放或滑动,我们将需要在触摸事件链的上游处理我们的手势检测。

为此,我们可以子类化并在方法RecyclerView中执行检测。dispatchTouchEvent()如果检测到单击,我们只需调用performClick(),这将触发RecyclerView's OnClickListener

public class ClickableRecyclerView extends RecyclerView {

    private final GestureDetectorCompat detector;

    public ClickableRecyclerView(Context context) {
        this(context, null);
    }

    public ClickableRecyclerView(Context context, AttributeSet attrs) {
        super(context, attrs);
        detector = new GestureDetectorCompat(context, new ClickListener());
    }

    private class ClickListener extends GestureDetector.SimpleOnGestureListener {
        @Override
        public boolean onSingleTapUp(MotionEvent e) {
            performClick();
            return true;
        }
    };

    @Override
    public boolean dispatchTouchEvent(MotionEvent e) {
        detector.onTouchEvent(e);
        return super.dispatchTouchEvent(e);
    }
}

只需使用此子类代替您的常规RecyclerView,并像往常一样在其上设置一个OnClickListener。可能需要其他构造函数,具体取决于您如何实例化它。

于 2016-07-08T01:02:08.680 回答
3

这就是我在 Kotlin 风格中所做的

fun RecyclerView.enableClickListener(){
    val gesture = object : GestureDetector.SimpleOnGestureListener(){
        override fun onSingleTapConfirmed(e: MotionEvent?): Boolean {
            this@enableClickListener.performClick()
            return super.onSingleTapConfirmed(e)
        }
    }
    val detector = GestureDetector(this.context, gesture)
    this.setOnTouchListener { v, event -> detector.onTouchEvent(event) }
}

这就是如何使用它

yourRecyclerView.apply {
        enableClickListener()
        setOnClickListener {
           // Do what you want  ...
        }
    }

享受 :)

于 2020-01-09T10:15:03.243 回答
0

这是一个技巧,它是开放的评论。

您可以在一个FrameLayout. 第一个是 the RecyclerView,第二个是 aView

    <FrameLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content">
        <androidx.recyclerview.widget.RecyclerView
            android:id="@+id/recycler"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"/>
        <View
            android:id="@+id/over_view"
            android:layout_width="match_parent"
            android:layout_height="match_parent"
            />
    </FrameLayout>

由于它的 a将位于FrameLayoutthe 的View顶部,RecyclerView并且您可以将 an 设置onClickListener为 theView并且它的行为就好像它是RecyclerView被点击的

over_view.setOnClickListener {
    .....
}
于 2020-02-04T12:11:26.167 回答
-1

RecycleView 上的点击事件?试试这样:

//set android:descendantFocusability="blocksDescendants" on parent layout 
//then setOnClickListener

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
**android:descendantFocusability="blocksDescendants"**
android:layout_height="match_parent">

<android.support.v7.widget.RecyclerView
  android:layout_width="match_parent"
  android:layout_height="match_parent">
</android.support.v7.widget.RecyclerView>

</LinearLayout>
于 2016-07-08T02:21:36.533 回答