RecyclerView पर क्लिक करें


583

किसी का उपयोग RecyclerViewकरने के लिए एक onClickListenerमें आइटम सेट करने के लिए एक रास्ता मिल गया है RecyclerView? मैंने प्रत्येक आइटम के लिए प्रत्येक लेआउट के लिए एक श्रोता को स्थापित करने के बारे में सोचा, लेकिन लगता है कि थोड़ी बहुत परेशानी है मुझे यकीन है कि RecyclerViewइस onClickघटना के लिए सुनने का एक तरीका है, लेकिन मैं इसे समझ नहीं पा रहा हूं।



4
RecyclerViewDemo मुझे यह परियोजना बहुत अच्छी लगी , और अधिक मुक्त क्रिया प्रदान करने के लिए इंटरफेस का उपयोग किया।
मेवएक्स

2
इस ट्यूटोरियल की जाँच करें wiki.workassis.com/android-recyclerview-example
Bikesh M

सरल और प्रयोग करने में आसान - github.com/rahulkapoor1/ClickableAdapter
राहुल

stackoverflow.com/a/31199564/5788247
शाओमु

जवाबों:


477

जैसा कि एपीआई के मौलिक रूप से बदल गया है, यह मुझे आश्चर्य नहीं करेगा अगर आप OnClickListenerप्रत्येक आइटम के लिए बनाएं । हालांकि यह बहुत परेशानी की बात नहीं है। आपके कार्यान्वयन में RecyclerView.Adapter<MyViewHolder>, आपके पास होना चाहिए:

private final OnClickListener mOnClickListener = new MyOnClickListener();

@Override
public MyViewHolder onCreateViewHolder(final ViewGroup parent, final int viewType) {
    View view = LayoutInflater.from(mContext).inflate(R.layout.myview, parent, false);
    view.setOnClickListener(mOnClickListener);
    return new MyViewHolder(view);
}

onClickविधि:

@Override
public void onClick(final View view) {
    int itemPosition = mRecyclerView.getChildLayoutPosition(view);
    String item = mList.get(itemPosition);
    Toast.makeText(mContext, item, Toast.LENGTH_LONG).show();
}

32
क्या होगा अगर मैं इसके क्लिक पर एक पंक्ति को हटाना चाहता हूं?
पीयूष कुकडिय़ा

20
मुझे यह उत्तर आपके द्वारा लिंक किए गए से बेहतर लगता है। जो इसे संभालने के लिए एक जेस्चर श्रोता और हिट बॉक्स डिटेक्शन लिखना चाहता है। Google--
लो-टैन

17
getChildPosition (दृश्य) पदावनत विधि है
जिगर

45
क्या आप बता रहे हैं कि आपके स्निपेट किस वर्ग में हैं? onClick(), से संबंधित OnClickListenerकिसी भी दृश्य के साथ संलग्न किया जा सकता है। यहाँ पूरा सवाल है जो! कृपया एक प्रयास करें, एक विधि न केवल एक नाम से पहचानी जाती है, बल्कि कम से कम एक वर्ग द्वारा भी पहचानी जाती है। यदि आप नहीं कहते हैं कि आपका कोड एडॉप्टर में जोड़ा जाना है, तो आपका उत्तर वास्तव में मदद नहीं करता है।
विंस

16
OnClick विधि उस टुकड़े में होनी चाहिए जो RecyclerView रखती है? क्योंकि आप अपने MyOnClickListener वर्ग में mRecyclerView का संदर्भ देते हैं। आपको उस वस्तु का संदर्भ कहां से मिला?
सीड जूल

605

यहाँ एक के लिए लागू करने के लिए एक बेहतर और कम कसकर युग्मित तरीका OnClickListenerहै RecyclerView

उपयोग का स्निपेट:

RecyclerView recyclerView = findViewById(R.id.recycler);
recyclerView.addOnItemTouchListener(
    new RecyclerItemClickListener(context, recyclerView ,new RecyclerItemClickListener.OnItemClickListener() {
      @Override public void onItemClick(View view, int position) {
        // do whatever
      }

      @Override public void onLongItemClick(View view, int position) {
        // do whatever
      }
    })
);

RecyclerItemClickListener कार्यान्वयन:

import android.content.Context;
import android.support.v7.widget.RecyclerView;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.view.View;


public class RecyclerItemClickListener implements RecyclerView.OnItemTouchListener {
  private OnItemClickListener mListener;

  public interface OnItemClickListener {
    public void onItemClick(View view, int position);

    public void onLongItemClick(View view, int position);
  }

  GestureDetector mGestureDetector;

  public RecyclerItemClickListener(Context context, final RecyclerView recyclerView, OnItemClickListener listener) {
    mListener = listener;
    mGestureDetector = new GestureDetector(context, new GestureDetector.SimpleOnGestureListener() {
        @Override
        public boolean onSingleTapUp(MotionEvent e) {
            return true;
        }

        @Override
        public void onLongPress(MotionEvent e) {
            View child = recyclerView.findChildViewUnder(e.getX(), e.getY());
            if (child != null && mListener != null) {
                mListener.onLongItemClick(child, recyclerView.getChildAdapterPosition(child));
            }
        }
    });
}

  @Override public boolean onInterceptTouchEvent(RecyclerView view, MotionEvent e) {
    View childView = view.findChildViewUnder(e.getX(), e.getY());
    if (childView != null && mListener != null && mGestureDetector.onTouchEvent(e)) {
      mListener.onItemClick(childView, view.getChildAdapterPosition(childView));
      return true;
    }
    return false;
  }

  @Override public void onTouchEvent(RecyclerView view, MotionEvent motionEvent) { }

  @Override
  public void onRequestDisallowInterceptTouchEvent (boolean disallowIntercept){}
}

85
इससे कोई सुराग नहीं मिलेगा कि किस बटन या दृश्य (आइटम के भीतर) पर क्लिक किया गया था। लेकिन समग्र आइटम क्लिक के लिए, यह ठीक है।
निरंजन

27
ngen की टिप्पणी सही है - यदि आप आइटम दृश्य के भीतर किसी विशिष्ट आइटम पर एक ऑनक्लिक श्रोता को संलग्न करना चाहते हैं, तो आपको संभवतः ऐसा करना चाहिए कि आपके एडॉप्टर में, ऑनबाइंड व्यू फ़ोल्डर में, या अपने व्यूहोल्डर में ही।
जैकब तबक

16
इस पद्धति के साथ एक समस्या यह है कि जब आप कुछ ऐसा कर रहे हैं जिसमें कुछ सौ मिलीसेकंड लगते हैं जैसे कि एक गतिविधि शुरू करना। जब आइटम पर क्लिक किया जाता है, और इसमें एक क्लिक किया गया एनीमेशन होता है, तो एनीमेशन पर क्लिक करने और देखने के बीच ध्यान देने योग्य देरी होती है।
गाक 2

20
यह समाधान बेहद भद्दा लगता है, 'क्लिक' को संसाधित करते समय कुछ देरी होती है और यह अनुभव ठीक नहीं होता है
वॉइल

6
किसी ने मुझे समझा सकता है कि क्यों GestureDetector के साथ इस समाधान को बेहतर तत्व के रूप में "setOnClickListener" onBindViewHolder () के अनुसार एडेप्टर में तत्व का उपयोग करने से बेहतर समाधान माना जाता है? इस तरह कम से कम मुझे GestureDetector समाधान की तुलना में कम कोड की आवश्यकता है। स्पष्टीकरण के लिए धन्यवाद।
ई-प्रकृति

115

मैं इसे इस तरह से करता हूं, बिना अनुचित वर्गों, डिटेक्टरों आदि के हमारे एडाप्टर के अंदर सरल कोड। पहले प्रस्तुत किए गए की तुलना में longClick के लिए विशेष रूप से बेहतर समाधान।

public class PasswordAdapter extends RecyclerView.Adapter<PasswordAdapter.ViewHolder> {
    private static ClickListener clickListener;

    public static class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener, View.OnLongClickListener {
        TextView name;

        public ViewHolder(View itemView) {
            super(itemView);
            itemView.setOnClickListener(this);
            itemView.setOnLongClickListener(this);
            name = (TextView) itemView.findViewById(R.id.card_name);
        }

        @Override
        public void onClick(View v) {
            clickListener.onItemClick(getAdapterPosition(), v);
        }

        @Override
        public boolean onLongClick(View v) {
            clickListener.onItemLongClick(getAdapterPosition(), v);
            return false;
        }
    }

    public void setOnItemClickListener(ClickListener clickListener) {
        PasswordAdapter.clickListener = clickListener;
    }

    public interface ClickListener {
        void onItemClick(int position, View v);
        void onItemLongClick(int position, View v);
    }
}

फिर टुकड़ा या गतिविधि के अंदर, बस मारो:

PasswordAdapter mAdapter = ...;

mAdapter.setOnItemClickListener(new PasswordAdapter.ClickListener() {
    @Override
    public void onItemClick(int position, View v) {
        Log.d(TAG, "onItemClick position: " + position);
    }

    @Override
    public void onItemLongClick(int position, View v) {
        Log.d(TAG, "onItemLongClick pos = " + position);
    }
});

OurAdapter.clickListener के बजाय मुझे लगता है कि Marurban का अर्थ है PasswordAdapter.clickListener और इसके बजाय TrainingAdapter.ClickListener (), का उपयोग करना चाहिए PasswordAdapter.ClickListener ()। इसके अलावा, महान समाधान मैरबन! मेरे लिए काम किया
nommer

onItemClick श्रोता लंबी क्लिक पर भी सक्रिय हो रहा है।
राशद.जेड

10
सर्वश्रेष्ठ प्रदर्शन के लिए ViewHolder स्थिर होना चाहिए। यह एक अच्छा जवाब नहीं है।
गिल्बर्टो इबारा

10
स्टैटिक क्लिकलिस्टर मेमोरी लीक बनाता है। श्रोता को गैर-स्थिर घोषित करें और इसके बजाय ViewHolder पर बाँधें।
ब्लेडडेकर

1
@AJW सबसे आसान तरीका है श्रोता को जल्दी से जोड़ना और उसे एडॉप्टर कंस्ट्रक्टर और फिर व्यूहोल्डर कंस्ट्रक्टर के तर्क के रूप में पास करना।
ब्लेडकार्ड

97

इसी तरह के एक प्रश्न की जाँच करें @ कॉमन्सवेयर की टिप्पणी इस पर लिंक करती है , जो OnClickListenerइंटरफ़ेस को लागू करता है viewHolder

यहाँ एक सरल उदाहरण है ViewHolder:

    TextView textView;//declare global with in adapter class

public static class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {

      private ViewHolder(View itemView) {
            super(itemView);
            itemView.setOnClickListener(this);
            textView = (TextView)view.findViewById(android.R.id.text1);

      }

      @Override
      public void onClick(View view) {
            Toast.makeText(view.getContext(), "position = " + getLayoutPosition(), Toast.LENGTH_SHORT).show();

         //go through each item if you have few items within recycler view
        if(getLayoutPosition()==0){
           //Do whatever you want here

        }else if(getLayoutPosition()==1){ 
           //Do whatever you want here         

        }else if(getLayoutPosition()==2){

        }else if(getLayoutPosition()==3){

        }else if(getLayoutPosition()==4){

        }else if(getLayoutPosition()==5){

        }

        //or you can use For loop if you have long list of items. Use its length or size of the list as 
        for(int i = 0; i<exampleList.size(); i++){

        }


      }
  }

Adapterतो इस तरह दिखता है:

    @Override
    public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view =LayoutInflater.from(parent.getContext()).inflate(android.R.layout.simple_list_item_1, parent, false);

        return new ViewHolder(view);
    }

14
हालांकि यह गलत दिशा में जाता है। यदि आपको किसी अन्य खंड / गतिविधि में एक पुनर्नवीनीकरण का उपयोग करने की आवश्यकता है, तो एडेप्टर / देखने वाला अब जो कुछ भी है, उसके बजाय क्लिक के तर्क को परिभाषित कर रहा है। क्लिक को वास्तव में प्रत्यक्ष या अप्रत्यक्ष रूप से युक्त उदाहरण द्वारा नियंत्रित किया जाना चाहिए। ऐसा लगता है कि एक स्पर्श श्रोता इस बारे में जाने का एकमात्र प्रभावी तरीका है।
ian.shaun.thomas

18
@ अक्सर मुझे नहीं लगता कि यह समस्या कैसे है। यदि व्यूहॉर्डर में ही यह जानकारी सम्‍मिलित है कि तत्व को क्लिक करने पर कौन से डेटा का चयन किया जाता है, तो कंटेनर की एकमात्र जिम्मेदारी N तत्वों का प्रदर्शन है, बजाय इसके कि मैं एक विशिष्ट i-th तत्व के चयन से निपटता हूं एन तत्वों का। ListViewअगर आप सिर्फ साधारण से अधिक पंक्तियों के जटिल वस्तुओं की एक सूची बनाने के लिए पहले से ही समस्याग्रस्त था। यदि आपको सूची में आइटम पर क्लिक करने के आधार पर विभिन्न घटनाओं को संभालने की आवश्यकता है तो क्या होगा? तुम तो गए। यहाँ, आपके पास प्रत्येक श्रोता है।
एपिकपांडाफ़ोर्स 15

2
मैं इस पर @EpicPandaForce से सहमत हूं। मुझे यह उत्तर भी आया कि RecyclerViews में क्लिक पर हैंडलिंग पर कॉमन्सवेयर की पुस्तक के खंड को पढ़ने के बाद यह मेरे लिए बहुत अच्छा काम किया है।
एडमएमसी 331

1
findViewById का उपयोग केवल दृश्य धारक निर्माण पर किया जा रहा है (इसलिए onCreateViewHolder)। व्यू होल्डर का उद्देश्य है कि हर बार जब कोई वस्तु धारक के लिए बाध्य हो, तो उसे देखने से रोका जा सके, न कि प्रारंभिक निर्माण पर।
अटक

2
getPosition()पदावनत, उपयोग getAdapterPosition()या getLayoutPosition()इसके बजाय है।
के नीरज लाल

94

जैकब तबक के जवाब (उनके लिए +1) के आधार पर, मैं श्रोता को जोड़ने में सक्षम था:

import android.content.Context;
import android.support.v7.widget.RecyclerView;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.view.View;

public class RecyclerItemClickListener implements RecyclerView.OnItemTouchListener {
    public interface OnItemClickListener {
        void onItemClick(View view, int position);

        void onItemLongClick(View view, int position);
    }

    private OnItemClickListener mListener;

    private GestureDetector mGestureDetector;

    public RecyclerItemClickListener(Context context, final RecyclerView recyclerView, OnItemClickListener listener) {
        mListener = listener;

        mGestureDetector = new GestureDetector(context, new GestureDetector.SimpleOnGestureListener() {
            @Override
            public boolean onSingleTapUp(MotionEvent e) {
                return true;
            }

            @Override
            public void onLongPress(MotionEvent e) {
                View childView = recyclerView.findChildViewUnder(e.getX(), e.getY());

                if (childView != null && mListener != null) {
                    mListener.onItemLongClick(childView, recyclerView.getChildAdapterPosition(childView));
                }
            }
        });
    }

    @Override
    public boolean onInterceptTouchEvent(RecyclerView view, MotionEvent e) {
        View childView = view.findChildViewUnder(e.getX(), e.getY());

        if (childView != null && mListener != null && mGestureDetector.onTouchEvent(e)) {
            mListener.onItemClick(childView, view.getChildAdapterPosition(childView));
        }

        return false;
    }

    @Override
    public void onTouchEvent(RecyclerView view, MotionEvent motionEvent) {
    }

    @Override
    public void onRequestDisallowInterceptTouchEvent(boolean disallowIntercept) {
    }
}

तो आप इसे इस तरह से उपयोग कर सकते हैं:

recyclerView.addOnItemTouchListener(new RecyclerItemClickListener(getActivity(), recyclerView, new RecyclerItemClickListener.OnItemClickListener() {
    @Override
    public void onItemClick(View view, int position) {
        // ...
    }

    @Override
    public void onItemLongClick(View view, int position) {
        // ...
    }
}));

1
मेरे ऐप के अनुकूल होने के लिए मुझे कुछ समय लगा, और जावा / एंड्रॉइड मानकों के लिए कोड और स्वरूप संरेखित किया, लेकिन अंततः यह ठीक काम किया।
milosmns

आप लंबी क्लिक के लिए एक एनीमेशन कैसे जोड़ सकते हैं?
जॉन

1
@ Eng.Fouad क्लिक प्रभाव इसके उपयोग से काम नहीं कर रहा है, हालांकि अगर हम एडॉप्टर में क्लिक क्लिक श्रोता सेट करते हैं तो प्रभाव इसके समाधान का कोई काम नहीं करता है
ingsaurabh

4
आपको ओवरराइड भी करना चाहिएpublic void onRequestDisallowInterceptTouchEvent (boolean disallowIntercept){}
पाओलो रोटोलो सेप

किसी कारण से यह क्लिक किए गए दृश्य को वापस नहीं करता है। मेरे मामले में मैं एक चेकबॉक्स पर क्लिक करता हूं लेकिन दिया गया दृश्य कुछ अलग है। जिस तरह से मैं जांच करता हूं:view.getId() == R.id.selection
dVaffection

63

इसी से मेरा काम बना है। संलग्न OnClickListenerकरने के लिए onBindView। मैं वास्तव में नहीं जानता कि क्या यह प्रदर्शन को प्रभावित करेगा, लेकिन यह थोड़ा कोड के साथ ठीक काम करता है।

public void onBindViewHolder(ViewHolder holder, final int position) {
    holder.view.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Toast.makeText(context, "Recycle Click" + position, Toast.LENGTH_SHORT).show();
            }
    });
}

5
मेरे मामले holder.itemView.setOnClickListener (..) में
D4rWiNS

18
मेरे मामले में 'देखने' के बजाय इसे 'आइटम व्यू' नाम दिया गया है, बस शुरुआती लोगों के लिए कह रहा हूं, उनमें से कुछ को समझ में नहीं आता कि वे क्या कॉपी कर रहे हैं, और यह पता नहीं लगा सकते कि क्यों काम नहीं करता है
D4rWiNS

3
इसकी लेआउट स्थिति
बुबुन्यो न्यावर

1
मेरे लिए यह केवल अंतिम दिखाई देने वाली वस्तु के लिए लागू होता है क्लिक की गई वस्तु नहीं
Nasz Njoka Sr.

2
यह समाधान मेरे लिए सबसे अच्छा लगता है, लेकिन प्रदर्शन के थर्मस में यह @bolot के उत्तर से कैसे भिन्न होता है? उन्होंने View.OnClickListener को सीधे ViewHolder क्लास पर लागू किया है, जो ऑनक्लीक विधि को अलग से थूकता है। सबसे अच्छा समाधान कौन सा होगा?
एडमंड टैमास

46

यह मेरे लिए गतिविधि में आइटम क्लिक श्रोता पर होने के लिए बहुत कठिन था और आइटम क्लिक श्रोता पर ट्रिगर नहीं होने वाले आइटम के एकल दृश्य के लिए क्लिक श्रोता होना चाहिए। जैकब तबक के जवाब के साथ खेलने के बाद मैं आइटम के लिए उसके जवाब का सम्मान करता हूं यदि आइटम के अंदर कोई अन्य स्पर्श क्रियाएं प्रस्तुत नहीं की जाती हैं।

मेरे पास एक कस्टम OnClickListenerइंटरफ़ेस है जो आइटम क्लिक इवेंट पर है जो क्लिक किए गए आइटम के दृश्य और एडॉप्टर से आइटम स्थिति रखता है। मैं इसका एक उदाहरण कंस्ट्रक्टर में प्रस्तुत करता हूं (या यह सेटर के साथ हो सकता है) और इसे व्यू होल्डर कंटेनर क्लिक श्रोता से संलग्न करें।

मेरे पास एडॉप्टर में अन्य क्लिक श्रोता भी हैं (व्यू होल्डर में हो सकते हैं) जो कंटेनर से करंट व्यू क्लिक को हैंडल करेगा।

 public class MyRecyclerAdapter extends RecyclerView.Adapter<MyViewHolder> {

private ArrayList<String> mData;
private OnItemClickListener mOnItemClickListener;

public interface OnItemClickListener {
    public void onItemClick(View view, int position);
}

public MyRecyclerAdapter(ArrayList<String> itemsData,
        OnItemClickListener onItemClickListener) {
    mOnItemClickListener = onItemClickListener;
    this.mData = itemsData;
}

@Override
public MyViewHolder onCreateViewHolder(ViewGroup parent,
        int viewType) {

    View layoutView = LayoutInflater.from(mContext).inflate(
            R.layout.list_item, parent, false);

    final MyViewHolder viewHolder = new MyViewHolder(layoutView);

    viewHolder.container.setOnClickListener(new OnClickListener() {

        @Override
        public void onClick(View v) {
            mOnItemClickListener.onItemClick(v, viewHolder.getAdapterPosition());
        }
    });

    viewHоlder.button.setOnClickListener(new OnClickListener() {

        @Override
        public void onClick(View v) {
            //do button click work here with
            // mData.get( viewHolder.getAdapterPosition() );
        }
    });

    return viewHolder;
}

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

गतिविधि में आपको उदाहरण से गुजरकर एडेप्टर को इनिशियलाइज़ करना होगा OnItemClickListener

public class FeedActivity extends ActionBarActivity {

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    ...

    RecyclerView recyclerView = (RecyclerView) findViewById(R.id.my_recycler_view);

    .....

    MyRecyclerAdapter adapter = new MyRecyclerAdapter(new ArrayList<String>(), new OnItemClickListener() {

        @Override
        public void onItemClick(View view, int position) {

            ///list item was clicked
        }
    });

    recyclerView.setLayoutManager(new LinearLayoutManager(this));
    recyclerView.setAdapter(mFeedsAdapter);
}

और मेरा ViewHolder

public class MyViewHolder extends RecyclerView.ViewHolder {

public Button button;
public View container;

public MyViewHolder(View itemLayoutView) {
    super(itemLayoutView);

    container = itemLayoutView;
    button = (Button) itemLayoutView.findViewById(R.id.button);
}}

इस दृष्टिकोण ने मेरे लिए अच्छा काम किया - इसका मतलब है कि आप ग्रिड दृश्य के अंदर की बजाय गतिविधि में क्लिक को संभाल सकते हैं
पत्तीकार

3
मैं आपको पैसा कहां से दूं। 4 घंटे की खोज और फिर मैंने आपकी पोस्ट पढ़ी। आपने इसे पोस्ट करने के लिए जो समय लिया, उसके लिए धन्यवाद।
ब्रैंडन

3
यह बेवकूफ लगता है लेकिन, क्या यह किसी भी मुद्दे की तरह स्मृति या दक्षता है? क्योंकि हम हर बार क्लिक श्रोता को असाइन कर रहे हैं, एक दृश्य सही खींचा जा रहा है?
rgv

1
@ राघव हाँ हर बार दृश्य बाँध रहा है एक नया क्लिक श्रोता सौंपा गया है, लेकिन यह मानक प्रवाह है। लेकिन mOnItemClickListener को केवल एक बार एडॉप्टर के लिए केवल एक बार बनाया गया है और इसकी स्थिति प्रत्येक आइटम पर अलग-अलग है;)
सर NIkolay Cesar 18

3
ऐप प्रदर्शन को मारने और मेमोरी अति प्रयोग बनाने के लिए बढ़िया उदाहरण! इस उदाहरण के साथ आपको सैकड़ों बेकार वस्तुओं को बनाने के लिए बस स्क्रॉल की आवश्यकता है, जो कि एक पल में बेकार हो जाएगा और इसे गारबेज के रूप में एक्सेस करना होगा। @ नोबडॉग सही है। कोई ClickListeners onBindView पर नहीं बनाया जा सकता है। एक क्लिक श्रोता को एक बार onCreateView (या होल्डर कंस्ट्रक्टर) में बनाना होगा, और इस एडॉप्टर के साथ स्क्रीन की आवश्यकता होने पर मौजूद होगा।
मायकोला टिचियाना

36

यह वही है जो मैंने समाप्त कर दिया है, अगर कोई इसे उपयोगी पाता है:

public static class ViewHolder extends RecyclerView.ViewHolder {

    public ViewHolder(View item) {

        super(item);
        item.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                Log.d("RecyclerView", "onClick:" + getAdapterPosition());
            }
        });

    }
}

स्रोत: http://blog.csdn.net/jwzhangjie/article/details/36868515


मैंने जिन सभी संस्करणों की कोशिश की, उनमें से यही एक है जिसे मैंने काम किया है। लेकिन क्या यह ठीक है, इसे इस तरह से करना है? या वहाँ एक बेहतर तरीका है, सबसे अच्छा अभ्यास की तरह?
मथायस लोइली

1
जहां तक ​​मैं बता सकता हूं, यह एकमात्र तरीका था। अधिक जानकारी के लिए इसे देखें! stackoverflow.com/a/24933117/1508887
फ़िफ़र भेड़

मैं ऑनक्लिक विधि के अंदर से नया टुकड़ा कैसे खोल सकता हूं?
हैश

5
getAdapterPosition () की जगह getPosition ()
कैनेडी

27

मैं के लिए अच्छा समाधान है RecyclerViewकी onItemClickListenerवस्तुओं और subitems के लिए

स्टेप 1- एक इंटरफेस बनाएं

public interface OnRecyclerViewItemClickListener
{
    /**
     * Called when any item with in recyclerview or any item with in item
     * clicked
     * 
     * @param position
     *            The position of the item
     * @param id
     *            The id of the view which is clicked with in the item or
     *            -1 if the item itself clicked
     */
    public void onRecyclerViewItemClicked(int position, int id);
}

स्टेप 2- इसके बाद इसे एडेप्टर की onBindViewHolderविधि में निम्न तरीके से उपयोग करें

/**
     * Custom created method for Setting the item click listener for the items and items with in items
     * @param listener OnRecyclerViewItemClickListener 
     */
    public void setOnItemClickListener(OnRecyclerViewItemClickListener listener)
    {
        this.listener = listener;
    }

    @Override
    public void onBindViewHolder(ViewHolder viewHolder, final int position)
    {

        // viewHolder.albumBg.setBackgroundResource(_itemData[position]
        // .getImageUrl());

        viewHolder.albumName.setText(arrayList.get(position).getName());
        viewHolder.artistName.setText(arrayList.get(position).getArtistName());
        String imgUrl = arrayList.get(position).getThumbImageUrl();

        makeImageRequest(imgUrl, viewHolder);
        viewHolder.parentView.setOnClickListener(new View.OnClickListener()
        {

            @Override
            public void onClick(View v)
            {
                listener.onRecyclerViewItemClicked(position, -1);
            }
        });
        viewHolder.settingButton.setOnClickListener(new View.OnClickListener()
        {

            @Override
            public void onClick(View v)
            {
                listener.onRecyclerViewItemClicked(position, v.getId());
            }
        });

    }

    // class to hold a reference to each item of RecyclerView
    public static class ViewHolder extends RecyclerView.ViewHolder
    {

        public TextView albumName, artistName;
        public ImageView albumIcon, settingButton;
        public LinearLayout parentView;

        public ViewHolder(View itemLayoutView)
        {
            super(itemLayoutView);
            // albumBg = (LinearLayout) itemLayoutView
            // .findViewById(R.id.albumDlbg);
            albumName = (TextView) itemLayoutView.findViewById(R.id.albumName);
            artistName = (TextView) itemLayoutView
                    .findViewById(R.id.artistName);
            albumIcon = (ImageView) itemLayoutView.findViewById(R.id.albumIcon);
            parentView = (LinearLayout) itemLayoutView
                    .findViewById(R.id.albumDlbg);
            settingButton = (ImageView) itemLayoutView
                    .findViewById(R.id.settingBtn);
        }

    }

चरण 3- गतिविधि या टुकड़े में पुनर्नवीनीकरण दृश्य ढूंढें और सेटअप करें जहां आप इसका उपयोग कर रहे हैं

recyclerView = (RecyclerView) rootview.findViewById(R.id.vmtopsongs);

        lm = new LinearLayoutManager(mActivity);
        lm.setOrientation(LinearLayoutManager.VERTICAL);
        recyclerView.setLayoutManager(lm);
        recyclerView.addItemDecoration(
                new HorizontalDividerItemDecoration.Builder(getActivity())
                        .paint(Utils.getPaint()).build());
        PopularSongsadapter mAdapter = new PopularSongsadapter(gallery,
                mActivity, true);
        // set adapter
        recyclerView.setAdapter(mAdapter);
        mAdapter.setOnItemClickListener(this);
        // set item animator to DefaultAnimator
        recyclerView.setItemAnimator(new DefaultItemAnimator());

चरण 4- अंत में गतिविधि या टुकड़े में इंटरफ़ेस लागू करें जहां आप रिसाइकलवे का उपयोग कर रहे हैं

@Override
    public void onRecyclerViewItemClicked(int position, int id)
    {
        if(id==-1){
            Toast.makeText(mActivity, "complete item clicked", Toast.LENGTH_LONG).show();
        }else{
            Toast.makeText(mActivity, "setting button clicked", Toast.LENGTH_LONG).show();
        }
    }

2
यह सबसे अच्छा समाधान है क्योंकि हमें आमतौर पर हमारे onItemClickListener पर गतिविधि / टुकड़े से बहुत सारे गुणों की आवश्यकता होती है, और इसे देखने वाले से प्रत्यक्ष कॉल करने के लिए कोई मतलब नहीं है (मुझे प्रत्येक एडाप्टर के लिए एक कस्टम निर्माता बनाना होगा)। वैसे, आपका कोड थोड़ा अधिक है, मैं आपको सबसे महत्वपूर्ण भाग (setOnItemClickListener) में कटौती करने का सुझाव देता हूं।
sagits

यह उप-आइटमों से पंक्ति / आइटम को अलग करने के लिए एक अच्छा सरल तरीका है
निगेल सैवेज

17

मैंने जो किया था यह रहा। यह समाधान पुनर्नवीनीकरण आइटम और दृश्य दोनों पर क्लिक करता है और पुनर्नवीनीकरण आइटम (आंतरिक विचार) पर विचार करता है।

मैं अपनी पसंद के विचारों पर viewHolder टैग करता हूं:

public RecyclerViewAdapter.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
    View itemView = LayoutInflater.from(parent.getContext()).inflate(R.layout.layout_item, null);
    ViewHolder viewHolder = new ViewHolder(itemView);

    itemView.setOnClickListener( this);
    itemView.setOnLongClickListener(this);
    viewHolder.imageIV.setOnClickListener(this);
    viewHolder.imageIV.setOnLongClickListener(this);

    viewHolder.imageIV.setTag(viewHolder);
    itemView.setTag(viewHolder);

    return viewHolder;
}

और मैं onClick () विधि (onLongClick समान है) में स्थिति को पुनः प्राप्त करने के लिए धारक.getPosition () का उपयोग करता हूं:

public void onClick(View view) {
    ViewHolder holder = (ViewHolder) view.getTag();
    int position = holder.getPosition();

    if (view.getId() == holder.imageIV.getId()){
        Toast.makeText(context, "imageIV onClick at" + position, Toast.LENGTH_SHORT).show();
    } else {
        Toast.makeText(context, "RecyclerView Item onClick at " + position, Toast.LENGTH_SHORT).show();
    }
}

GetChildPosition के साथ एक संस्करण भी काम करता है। कृपया ध्यान दें कि आंतरिक दृश्यों के लिए, onClick () उपयोग में:

int position = recyclerView.getChildPosition((View)view.getParent());

मेरे दिमाग में, इस समाधान का लाभ यह है कि जब कोई छवि पर क्लिक करता है, तो केवल ऑनक्लिक () छवि श्रोता को बुलाया जाता है, जब मैंने एक रिसाइक्लर व्यू आइटम के लिए याकूब के समाधान को जोड़ा और आंतरिक दृश्य के लिए मेरा समाधान RecyclerView आइटम दृश्य onclick ( ) भी कहा जाता है (जब छवि पर क्लिक करें)।


11

ऐसा करने का बहुत आसान तरीका है। बस क्लिक इन पर आवेदन करेंonBindViewHolder रूट ।

एडॉप्टर के लिए यह आपका विचार है,

<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:card_view="http://schemas.android.com/apk/res-auto"
    android:id="@+id/linearlayout"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

        <TextView
            android:id="@+id/textview"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginBottom="1dp"
            android:textSize="15sp" />
</LinearLayout>

फिर अपने एडॉप्टर में निम्नलिखित करें

//get the layout and make view holder
@Override
public RVAdapter.ViewHolder1 onCreateViewHolder(ViewGroup parent, int viewType) {

    View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.adapter_layout, null);
    ViewHolder1 viewHolder = new ViewHolder1(view);
    return viewHolder;
}

@Override
public void onBindViewHolder(RVAdapter.ViewHolder1 holder, int position) {

    //apply on click on your root view
    holder.linearlayout.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            //Do on click stuff
        }
    });
}

//make references to views in layout including root view
public class ViewHolder1 extends RecyclerView.ViewHolder {

    protected LinearLayout linearlayout = null
    protected TextView textview = null;

    public CareerLinksViewHolder(View itemView) {
        super(itemView);

        this.linearlayout = (LinearLayout) itemView.findViewById(R.id.linearlayout);
        this.tvCompName = (TextView) itemView.findViewById(R.id.textview);
    }
}

9

आप एक पारित कर सकते हैं clickListenerकरने के लिएAdapter

अपने में Activity:

private View.OnClickListener mItemClick = new View.OnClickListener() {

    @Override
    public void onClick(View v) {
        Intent intent = null;
        int position = list.getChildPosition(v);
        switch (position) {
            case 0:
                intent = new Intent(MainActivity.this, LeakCanaryActivity.class);
                break;
            case 1:
                intent = new Intent(MainActivity.this, ButterKnifeFragmentActivity.class);
                break;
        }
        if (intent != null) {
            MainActivity.this.startActivity(intent);
        }
    }
};

फिर इसे पास करें Adapter:

MainAdapter mainAdapter = new MainAdapter(this, mItemClick);

में Adapterकी onCreateViewHolder:

 @Override
public MainAdapter.ViewHolder onCreateViewHolder(ViewGroup viewGroup, int position) {
    View itemView = activity.getLayoutInflater().inflate(R.layout.main_adapter_item, viewGroup, false);
    ViewHolder holder = new ViewHolder(itemView);
    itemView.setOnClickListener(mItemClick);
    return holder;
}

आप holderइरादे से मूल्यों को कैसे पार करते हैं?
RoCk RoCk

8

मैंने Android के लिए एक लाइट वेटेड लाइब्रेरी विकसित की है, आप https://github.com/ChathuraHettiarachchi/RecycleClick पर जा सकते हैं

और निम्नलिखित नमूने के लिए का पालन करें

RecycleClick.addTo(YOUR_RECYCLEVIEW).setOnItemClickListener(new RecycleClick.OnItemClickListener() {
            @Override
            public void onItemClicked(RecyclerView recyclerView, int position, View v) {
                // YOUR CODE
            }
        });

पुष्टि कर सकते हैं, यह पुस्तकालय महान है! उपयोग करने के लिए बहुत आसान है, अच्छी नौकरी!
पेटकरोडर्मैक

7

आप अपने ViewHolder वर्ग के लिए OnClickListener को लागू कर सकते हैं

public static class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
        public Item item
        @InjectView(R.id.tv_title)
        public TextView tvTitle;
        @InjectView(R.id.rl_row)
        public RelativeLayout rlRow;

        public ViewHolder(View v) {
            super(v);
            ButterKnife.inject(this, v);
            v.setOnClickListener(this);
        }

        @Override
        public void onClick(View view) {
            Log.e("item title",item.getTitle());
        }
    }

और onBindViewHolder ने आपके व्यू होल्डर का आइटम सेट किया

public void onBindViewHolder(ViewHolder holder, int position) {
        holder.tvTitle.setText(objects.get(position).getTitle());
        holder.item = objects.get(position);
        }

7

बहुत सरल और प्रभावी तरीका।

View.OnClickListenerदृश्य धारक के अंदर इंटरफ़ेस लागू करने या आपकी गतिविधि में इंटरफ़ेस और कार्यान्वयन को लागू करने के बजाय - मैंने OnClickListenerकार्यान्वयन पर सरल के लिए इस कोड का उपयोग किया।

public static class SimpleStringRecyclerViewAdapter
            extends RecyclerView.Adapter<SimpleStringRecyclerViewAdapter.ViewHolder> {

        // Your initializations goes here...
        private List<String> mValues;

        public static class ViewHolder extends RecyclerView.ViewHolder {

            //create a variable mView
            public final View mView;

            /*All your row widgets goes here
            public final ImageView mImageView;
            public final TextView mTextView;*/

            public ViewHolder(View view) {
                super(view);
                //Initialize it here
                mView = view;

                /* your row widgets initializations goes here
                mImageView = (ImageView) view.findViewById(R.id.avatar);
                mTextView = (TextView) view.findViewById(android.R.id.text1);*/
            }
        }

        public String getValueAt(int position) {
            return mValues.get(position);
        }

        public SimpleStringRecyclerViewAdapter(Context context, List<String> items) {

            mBackground = mTypedValue.resourceId;
            mValues = items;
        }

        @Override
        public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
            View view = LayoutInflater.from(parent.getContext())
                    .inflate(R.layout.list_item, parent, false);
            view.setBackgroundResource(mBackground);
            return new ViewHolder(view);
        }

        @Override
        public void onBindViewHolder(final ViewHolder holder, int position) {
            holder.mBoundString = mValues.get(position);
            holder.mTextView.setText(mValues.get(position));

            //Here it is simply write onItemClick listener here
            holder.mView.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    Context context = v.getContext();
                    Intent intent = new Intent(context, ExampleActivity.class);

                    context.startActivity(intent);
                }
            });
        }

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

3
यह एक सामान्य बुरा अभ्यास है क्योंकि यह बहुत आसान लगता है। आप हर बार एक नए आंतरिक श्रोता का निर्माण कर रहे हैं, जब एक वस्तु को बाँधा जाता है (और एक पुनर्नवीनीकरण में प्रत्येक आइटम को बहुत बार बाध्य किया जा सकता है) जो कि खराब है) a) प्रदर्शन b) कचरा संग्रह। इसके बजाय आपको कहीं एक श्रोता बनाना चाहिए (देखने वाले में, एडॉप्टर में, ऊपरी टुकड़े या गतिविधि में) और बस इसे व्यूअर कंस्ट्रक्टर में आइटम्स ऑनक्रीट व्यूहॉर्ड में जोड़ें। इसमें यह पता लगाने की जटिल जटिलता है कि किस आइटम पर क्लिकर था, लेकिन टैग के माध्यम से या धारक को अधिक जानकारी जोड़कर आसानी से हल किया जा सकता है।
GuillermoMP

6

अब तक पोस्ट किए गए सभी उत्तर महान समाधान हैं, हालांकि यदि आप बहुत सारे कार्यान्वयन विवरणों से निपटना नहीं चाहते हैं, और बस यह चाहते हैं कि यह कैसे ListView के समान काम करता है, तो मैं टू-व्यू-व्यू का उपयोग करने की सिफारिश करूंगा, जैसा कि यहां देखा गया है:

https://github.com/lucasr/twoway-view

ध्यान दें कि यह कार्यान्वयन वस्तुओं पर लंबे समय तक प्रेस का समर्थन करता है, साथ ही साथ दबाए गए राज्यों के लिए समर्थन (जो कुछ महत्वपूर्ण है कि इस प्रश्न के अन्य समाधानों की कमी है)।

यदि आप पूरी लाइब्रेरी का उपयोग नहीं करना चाहते हैं, तो ClickItemTouchListener वर्ग पर एक नज़र डालें , जिसे ज़रूरत पड़ने पर स्टैंडअलोन के रूप में इस्तेमाल किया जा सकता है। एकमात्र मुद्दा जो मुझे इसके साथ मिला था वह लंबे प्रेस + स्क्रॉलिंग के साथ है, ऐसा लगता है कि गलत व्यवहार है।


6

यदि आप अलग-अलग आइटम पर क्लिक इवेंट को पकड़ना चाहते हैं, तो बस कक्षा OnClickListenerमें लागू ViewHolderकरें और फिर व्यक्तिगत विचारों या संपूर्ण पर श्रोताओं को क्लिक करेंitemView

निम्नलिखित उदाहरण से पता चलता है

public  class ContactViewHolder extends RecyclerView.ViewHolder implements OnClickListener
    {
        TextView txt_title,txt_name,txt_email;

        public ContactViewHolder(View itemView) 
        {
            super(itemView);
            txt_title = (TextView)itemView.findViewById(R.id.txt_title);
            txt_name  = (TextView)itemView.findViewById(R.id.txt_name);
            txt_email = (TextView)itemView.findViewById(R.id.txt_email);

            txt_name.setOnClickListener(this);
            txt_email.setOnClickListener(this);
            itemView.setOnClickListener(this);
        }

        @Override
        public void onClick(View v) {
            // TODO Auto-generated method stub

            if(v == itemView)
            {
                Toast.makeText(RecyclerDemoActivity.this, "Visiting Card Clicked is ==>"+txt_name.getText(), Toast.LENGTH_SHORT).show();
            }

            if(v == txt_name)
            {
                Toast.makeText(RecyclerDemoActivity.this, "Name ==>"+txt_name.getText(), Toast.LENGTH_SHORT).show();
            }

            if(v == txt_email)
            {
                Toast.makeText(RecyclerDemoActivity.this, "Email ==>"+txt_email.getText(), Toast.LENGTH_SHORT).show();
            }
        }

    }
} 

5

मेरे लिए, यह सबसे अच्छा तरीका है:

class YourRecyclerAdapter extends RecyclerView.Adapter<ContactViewHolder> implements View.OnClickListener { 
  ...
  @Override
  public void onClick(View view) {
        int itemPosition = vRecycle.getChildPosition(view);
        //And use itemPosition to get the item from your collection. This way you dont restrain the ViewHolder with a OnClick callback
    }
  ...
}


5

के RecyclerViewपास नहीं हैOnClickListener और इसे स्वयं लागू करना होगा।

जब आप आइटम दृश्य पर क्लिक करते हैं, तो मुझे एक विधि के साथ एक OnItemClickListenerइंटरफ़ेस जोड़ना पसंद है । इस प्रकार एक आइटम पर क्लिक के प्रबंधन की जिम्मेदारी बाहर औरAdapteronClickViewHolderViewHolderAdapter । गतिविधि या टुकड़ा होगा जो तय करेगा कि क्या करना है

श्रोता और श्रोता ऑब्जेक्ट में एक इंटरफ़ेस जोड़ें।

public class ItemsAdapter extends RecyclerView.Adapter<ItemsAdapter.ViewHolder> {

  ...

  private static OnItemClickListener onItemClickListener;

  ...

  public static interface OnItemClickListener {
      public void onItemClick(View view, int position);
  }

  ...
}

हम आइटम के रूट दृश्य के क्लिक को कैप्चर करते हैं और जब कॉलबैक onClickएडॉप्टर पर श्रोता कॉल को ट्रिगर करता है।

public class ItemsAdapter extends RecyclerView.Adapter<ItemsAdapter.ViewHolder> {

  ...

  private static OnItemClickListener onItemClickListener;

  ...

  public static interface OnItemClickListener {
      public void onItemClick(View view, int position);
  }

  ...

  public static class ViewHolder extends RecyclerView.ViewHolder {
      public ImageView imageView;

      public ViewHolder(View itemRootView) {
          super(itemRootView);
          imageView = (ImageView) itemRootView.findViewById(R.id.itemImage);

          itemRootView.setOnClickListener(new View.OnClickListener() {
              @Override
              public void onClick(View view) {
                  int position  = ViewHolder.super.getAdapterPosition();
                  onItemClickListener.onItemClick(view,position);
              }
          });
      }
  }
}

चूंकि हमारे मामले में गतिविधि या टुकड़ा, टुकड़ा, हम एडेप्टर को एक श्रोता प्रदान करते हैं और कॉलबैक पर क्लिक करके हम चयनित आइटम को स्थिति से प्राप्त करेंगे और आइटम की एक विस्तृत गतिविधि खोली जाएगी।

public class ItemsFragment extends Fragment {
    ...
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
       ...    
        ((ItemsAdapter) adapter).setOnItemClickListener(new ItemsAdapter.OnItemClickListener() {
            @Override
            public void onItemClick(View view, int position) {
                //Do something when an item has been clicked
            }
        });
        ...
    }
...
}

5

दुर्भाग्य से RecyclerViewकुछ सुविधाएँ याद आ रही हैं जो ListViewअंतर्निहित थीं। उदाहरण के लिए OnItemClickListenerकिसी आइटम पर क्लिक करने पर उस ट्रिगर को जोड़ने की क्षमता । RecyclerViewआपको OnClickListenerअपने एडेप्टर में सेट करने की अनुमति देता है , लेकिन अपने कॉलिंग कोड से एडॉप्टर और उस पर क्लिक करने वाले श्रोता को पास करना ViewHolder, एक साधारण आइटम क्लिक को पकड़ने के लिए जटिल है।

public class ItemClickSupport {
private final RecyclerView mRecyclerView;
private OnItemClickListener mOnItemClickListener;
private OnItemLongClickListener mOnItemLongClickListener;
private View.OnClickListener mOnClickListener = new View.OnClickListener() {
    @Override
    public void onClick(View v) {
        if (mOnItemClickListener != null) {
            RecyclerView.ViewHolder holder = mRecyclerView.getChildViewHolder(v);
            mOnItemClickListener.onItemClicked(mRecyclerView, holder.getAdapterPosition(), v);
        }
    }
};
private View.OnLongClickListener mOnLongClickListener = new View.OnLongClickListener() {
    @Override
    public boolean onLongClick(View v) {
        if (mOnItemLongClickListener != null) {
            RecyclerView.ViewHolder holder = mRecyclerView.getChildViewHolder(v);
            return mOnItemLongClickListener.onItemLongClicked(mRecyclerView, holder.getAdapterPosition(), v);
        }
        return false;
    }
};
private RecyclerView.OnChildAttachStateChangeListener mAttachListener
        = new RecyclerView.OnChildAttachStateChangeListener() {
    @Override
    public void onChildViewAttachedToWindow(View view) {
        if (mOnItemClickListener != null) {
            view.setOnClickListener(mOnClickListener);
        }
        if (mOnItemLongClickListener != null) {
            view.setOnLongClickListener(mOnLongClickListener);
        }
    }

    @Override
    public void onChildViewDetachedFromWindow(View view) {

    }
};

private ItemClickSupport(RecyclerView recyclerView) {
    mRecyclerView = recyclerView;
    mRecyclerView.setTag(R.id.item_click_support, this);
    mRecyclerView.addOnChildAttachStateChangeListener(mAttachListener);
}

public static ItemClickSupport addTo(RecyclerView view) {
    ItemClickSupport support = (ItemClickSupport) view.getTag(R.id.item_click_support);
    if (support == null) {
        support = new ItemClickSupport(view);
    }
    return support;
}

public static ItemClickSupport removeFrom(RecyclerView view) {
    ItemClickSupport support = (ItemClickSupport) view.getTag(R.id.item_click_support);
    if (support != null) {
        support.detach(view);
    }
    return support;
}

public ItemClickSupport setOnItemClickListener(OnItemClickListener listener) {
    mOnItemClickListener = listener;
    return this;
}

public ItemClickSupport setOnItemLongClickListener(OnItemLongClickListener listener) {
    mOnItemLongClickListener = listener;
    return this;
}

private void detach(RecyclerView view) {
    view.removeOnChildAttachStateChangeListener(mAttachListener);
    view.setTag(R.id.item_click_support, null);
}

public interface OnItemClickListener {

    void onItemClicked(RecyclerView recyclerView, int position, View v);
}

public interface OnItemLongClickListener {

    boolean onItemLongClicked(RecyclerView recyclerView, int position, View v);
}
}

आपको R.id.item_click_supportids.xml का उपयोग करके परिभाषित करने की आवश्यकता है:

 <?xml version="1.0" encoding="utf-8"?>
 <resources>
  <item name="item_click_support" type="id" />
 </resources>

परिणामी कोड क्लिक श्रोता अब इस तरह दिखता है:

ItemClickSupport.addTo(mRecyclerView).setOnItemClickListener(new ItemClickSupport.OnItemClickListener() {
@Override
public void onItemClicked(RecyclerView recyclerView, int position, View v) {
    // do it
}
});

Recyclerview क्लिक्स के बारे में संक्षिप्त विवरण के लिए कृपया इस littlerobots_blog पर एक नज़र डालें


5

आप आसानी से अपने ViewHolder वर्ग में setOnClickListener को परिभाषित कर सकते हैं :

public class ViewHolder extends RecyclerView.ViewHolder {
    TextView product_name;

    ViewHolder(View itemView) {
        super(itemView);
        product_name = (TextView) itemView.findViewById(R.id.product_name);
        itemView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                int itemPosition = getLayoutPosition();
                Toast.makeText(getApplicationContext(), itemPosition + ":" + String.valueOf(product_name.getText()), Toast.LENGTH_SHORT).show();
            }
        });
    }
}

5

यहाँ है कि मैंने क्या किया और अधिक पढ़ें और यहाँ डाउनलोड करें

उसी को यहाँ जोड़ रहा है

CustomItemClickListener.java

public interface CustomItemClickListener {
 public void onItemClick(View v, int position);
}

ItemsListAdapter.java

public class ItemsListAdapter extends RecyclerView.Adapter<ItemsListAdapter.ViewHolder> {
ArrayList<ItemListSingleItem> data;

Context mContext;
CustomItemClickListener listener;

@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
    View mView = LayoutInflater.from(parent.getContext()).inflate(R.layout.items_list_single_item, parent, false);
    final ViewHolder mViewHolder = new ViewHolder(mView);
    mView.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            listener.onItemClick(v, mViewHolder.getAdapterPosition());
        }
    });
    return mViewHolder;
}

@Override
public void onBindViewHolder(ViewHolder holder, int position) {
    holder.itemTitle.setText(Html.fromHtml(data.get(position).getTitle()));
    if (!TextUtils.isEmpty(data.get(position).getThumbnailURL())) {
      // I Love picasso library :) http://square.github.io/picasso/
        Picasso.with(mContext).load(data.get(position).getThumbnailURL()).error(R.drawable.ic_no_image).
                placeholder(R.drawable.ic_no_image).
                transform(new RoundedCornersTransformation(5, 0)).
                into(holder.thumbnailImage);
    } else {
        holder.thumbnailImage.setImageResource(R.drawable.ic_no_image);
    }
}


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

public ItemsListAdapter(Context mContext, ArrayList<ItemsListSingleItem> data, CustomItemClickListener listener) {
    this.data = data;
    this.mContext = mContext;
    this.listener = listener;
}

public static class ViewHolder extends RecyclerView.ViewHolder {
    public TextView itemTitle;
    public ImageView thumbnailImage;

    ViewHolder(View v) {
        super(v);
        itemTitle = (TextView) v
                .findViewById(R.id.post_title);
        thumbnailImage = (ImageView) v.findViewById(R.id.post_thumb_image);
    }
 }
}

4

ऊपर दिए गए अधिकांश उत्तरों से, वे अपने onclicklisteners को अलग-अलग मदों पर सेट करते प्रतीत होते हैं। हालाँकि, प्रस्ताव के बारे में समाधान बहुत सरल है, लेकिन अभी तक बहुतों के लिए सहज नहीं है। कई लोग भूल रहे हैं कि अन्य घटक हमेशा एक मूल घटक में होते हैं जो सूची या पुनर्नवीनीकरण विचारों में आइटम प्रदर्शित करने के लिए उपयोग किया जाता है। यह समाधान इस माता-पिता के दृश्य के लिए एक एकल ऑन्कलिक श्रोता स्थापित करने के बारे में है और बारी खेला जाता है। समाधान में सूची या पुनर्नवीनीकरण दृश्य से क्लिक किए जा रहे आइटम की स्थिति को पारित करने का एक तरीका भी शामिल है। यहां, हमारा मुख्य रूटव्यू एंड्रॉइड सपोर्ट लाइब्रेरी से एक कार्ड व्यू है। यहाँ नमूना कोड है

public class ListAdapter extends RecyclerView.Adapter<ListAdapter.ViewHolder> {

public static final String LOG_TAG = ListAdapter.class.getSimpleName();
private Cursor mDataset;
private Context mContext;
private ViewHolder mViewHolder;

// Provide a suitable constructor (depends on the kind of dataset)
public ListAdapter(Context context, Cursor Dataset) {
    mDataset = Dataset;
    mContext = context;
}

// Create new views (invoked by the layout manager)
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {

    // create a new view
    View v = LayoutInflater.from(parent.getContext())
            .inflate(R.layout.list_business_view, parent, false);

    mViewHolder = new ViewHolder(v);
    return mViewHolder;
}

public void setData(Cursor newdata) {
    this.mDataset = newdata;
}

// Replace the contents of a view (invoked by the layout manager)
@Override
public void onBindViewHolder(ViewHolder holder, int position) {
//Bind data to other items here. To save time, i have ommited that.
           //here is where we attach a click listerner for an item in the recycler list rather than for each element of a given item.
            holder.card.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    Toast.makeText(mContext, " Just cliked item at position " + itemPosition, Toast.LENGTH_LONG).show();

            }
        });

    }
}

// Return the size of your dataset (invoked by the layout manager)
@Override
public int getItemCount() {
    if (null != mDataset) {
        return mDataset.getCount();
    }
    return 0;

}


// Provide a reference to the views for each data item
// Complex data items may need more than one view per item, and
// you provide access to all the views for a data item in a view holder
public static class ViewHolder extends RecyclerView.ViewHolder{
    // each data item is just a string in this case
    public final TextView mBusinesssName; // View for the business name
    public final TextView mBusinessCategory; //View for the category name
    public final ImageView businessImage; // View for the business category image Image
    public final TextView mBusinessDistance; // View for the distance
    public final CardView card;

    public ViewHolder(View view) {
        super(view);
        mBusinesssName = (TextView) view.findViewById(R.id.list_item_name_textview);
        mBusinessCategory = (TextView) view.findViewById(R.id.list_item_category_textview);
        mBusinessDistance = (TextView) view.findViewById(R.id.list_item_dist_textview);
        businessImage = (ImageView) view.findViewById(R.id.list_item_icon);
        card = (CardView) view.findViewById(R.id.card_view);

    }
}
}

4

नर्मदान के उत्तर का कोटलिन कार्यान्वयन :

mRecyclerView.addOnItemTouchListener(object  : RecyclerItemClickListener(this, mRecyclerView,object :RecyclerItemClickListener.OnItemClickListener{
            override fun onItemClick(view: View, position: Int) {

            }

            override fun onLongItemClick(view: View?, position: Int) {

            }
}){})

RecyclerItemClickListener.java:

import android.content.Context
import android.support.v7.widget.RecyclerView
import android.view.GestureDetector
import android.view.MotionEvent
import android.view.View


open class RecyclerItemClickListener(context: Context, recyclerView: RecyclerView, private val mListener: OnItemClickListener?) : RecyclerView.OnItemTouchListener {

    private var mGestureDetector: GestureDetector

    interface OnItemClickListener {
        fun onItemClick(view: View, position: Int)

        fun onLongItemClick(view: View?, position: Int)
    }

    init {
        mGestureDetector = GestureDetector(context, object : GestureDetector.SimpleOnGestureListener() {
            override fun onSingleTapUp(e: MotionEvent): Boolean {
                return true
            }

            override fun onLongPress(e: MotionEvent) {
                val child = recyclerView.findChildViewUnder(e.x, e.y)
                if (child != null && mListener != null) {
                    mListener.onLongItemClick(child, recyclerView.getChildAdapterPosition(child))
                }
            }
        })
    }

    override fun onInterceptTouchEvent(view: RecyclerView, e: MotionEvent): Boolean {
        val childView = view.findChildViewUnder(e.x, e.y)
        if (childView != null && mListener != null && mGestureDetector.onTouchEvent(e)) {
            mListener.onItemClick(childView, view.getChildAdapterPosition(childView))
            return true
        }
        return false
    }

    override fun onTouchEvent(view: RecyclerView, motionEvent: MotionEvent) {}

    override fun onRequestDisallowInterceptTouchEvent(disallowIntercept: Boolean) {}
}

3
public class MyViewHolder extends RecyclerView.ViewHolder {
        public TextView title, year, genre;

        public MyViewHolder(View view) {
            super(view);
            title = (TextView) view.findViewById(R.id.title);
            genre = (TextView) view.findViewById(R.id.genre);
            year = (TextView) view.findViewById(R.id.year);
            view.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    Toast.makeText(context, ""+getAdapterPosition(), Toast.LENGTH_SHORT).show();
                }
            });
        }
    }

3

यहाँ मेरे कस्टम एडॉप्टर के लिए पूरा कोड है। यह कोड "list_item" नाम की xml फ़ाइल में परिभाषित सूची आइटम के साथ पंक्तियों को बढ़ाएगा। यह संबंधित पदों के साथ सभी सूची आइटम पंक्तियों पर क्लिक इवेंट भी करेगा।

public class MyCustomAdapter extends RecyclerView.Adapter`<`AdapterMyCustomAdapter.ViewHolder> {

    public static class ViewHolder extends RecyclerView.ViewHolder implements OnClickListener {
        public onItemClickListener mListener;
        public ViewHolder(View v, onItemClickListener listener) {
            super(v);
            mListener =listener;
            v.setOnClickListener(this);
        }

        @Override
        public void onClick(View v) {
            mListener.onRecyclerItemClick(v, getPosition());
        }

        public static interface onItemClickListener {
            public void onRecyclerItemClick(View view , int position);
        }
    }

    @Override
    public int getItemCount() {
        return 5;
    }

    @Override
    public void onBindViewHolder(ViewHolder holder, int pos) {      

    }

    @Override
    public ViewHolder onCreateViewHolder(ViewGroup parent, int position) {
        View v = LayoutInflater.from(parent.getContext())
                .inflate(R.layout.list_item, parent, false);

    /* here list_item is an xml file we want to inflate ...it is same as we do in case of listview for customization.*/

        MyCustomAdapter.ViewHolder vh = new ViewHolder(v, new MyCustomAdapter.ViewHolder.onItemClickListener() {

            @Override
            public void onRecyclerItemClick(View view, int position) {
                System.out.println("clicked on list item at position " +position);
            } 
        });
        return vh;
    }
}

2

हम जावा कमजोर संदर्भों का उपयोग करके ऐसा कर सकते हैं। शब्दार्थ, व्यू होल्डर वह होता है जिसे क्लिक ईवेंट का जवाब देना चाहिए या उसे सही उत्तरदाता को सौंपना चाहिए।

हमारे लक्ष्य:

  1. व्यूहोल्डर को घटनाओं के जवाब देने वाले वर्ग के बारे में कुछ भी नहीं पता होना चाहिए सिवाय इसके कि वह एक निश्चित इंटरफ़ेस लागू करता है।
  2. क्लिक हैंडलर को पुन: क्लिक किए जाने वाले दृश्य के RecyclerView में स्थान प्राप्त करना चाहिए।
  3. हमें यह समझने में सक्षम होना चाहिए कि कौन सा है देखने में कि दृश्य धारक में दृश्य क्लिक किया गया था।
  4. सभी घटकों के बीच ढीली युग्मन बनाए रखें और किसी भी चक्र को बनाए रखने का कारण न बनें।

कदम:

  1. क्लिक प्रतिक्रियाओं को संभालने के लिए एक इंटरफ़ेस बनाएँ।

  2. गतिविधि में इस इंटरफ़ेस को लागू करें जो क्लिक को संभाल लेगा।

  3. कमजोर संदर्भ और इसे सेट करने वाले कंस्ट्रक्टर को पकड़ने के लिए RecyclerView Adapter में एक सदस्य चर जोड़ें।

  4. RecyclerView ViewHolder में ऐसा ही करें और स्थिति का ट्रैक रखने के लिए एक सदस्य चर जोड़ें।

  5. ViewHolder में किसी भी दृश्य पर अपने श्रोताओं को क्लिक पर सेट करें, फिर उन्हें संभालने के लिए उत्तरदाता को कॉलबैक करें।

  6. बाइंडिंग करते समय स्थिति सेट करने के लिए अपना onBindViewHolder तरीका बदलें।

  7. ViewHolder के नीचे उत्तरदाता पास करें।

  8. उत्तरदाता में, अब आप getId () का उपयोग करके यह देख सकते हैं कि किस दृश्य पर क्लिक किया गया था।

और यहाँ एक Gist है, तो आप देख सकते हैं कि यह सब एक साथ कैसे फिट होता है: RecyclerView हैंडलिंग पर क्लिक करें


2

मुझे पता है कि बहुत सारे उत्तर हैं, लेकिन मुझे लगा कि मैं इसे लागू करने के साथ ही इसे लागू कर सकता हूं। ( मेरे द्वारा पूछे गए एक अन्य प्रश्न पर पूर्ण विवरण पाया जा सकता है )।

इसलिए, एक क्लिक श्रोता को जोड़ने के लिए, आपके आंतरिक ViewHolderवर्ग को लागू करने की आवश्यकता है View.OnClickListener। ऐसा इसलिए है क्योंकि आप निर्माणकर्ता OnClickListenerके itemViewपैरामीटर को सेट करेंगे ViewHolder। मे तुम्हें दिखाता हूँ की मेरा क्या मतलब हैं:

public class ExampleClickViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {

    TextView text1, text2;

    ExampleClickViewHolder(View itemView) {
        super(itemView);

        // we do this because we want to check when an item has been clicked:
        itemView.setOnClickListener(this);

        // now, like before, we assign our View variables
        title = (TextView) itemView.findViewById(R.id.text1);
        subtitle = (TextView) itemView.findViewById(R.id.text2);
    }

    @Override
    public void onClick(View v) {
        // The user may not set a click listener for list items, in which case our listener
        // will be null, so we need to check for this
        if (mOnEntryClickListener != null) {
            mOnEntryClickListener.onEntryClick(v, getLayoutPosition());
        }
    }
}

केवल अन्य चीजें जो आपको जोड़ने की आवश्यकता हैं, आपके Adapterऔर एक सेटर विधि के लिए एक कस्टम इंटरफ़ेस हैं :

private OnEntryClickListener mOnEntryClickListener;

public interface OnEntryClickListener {
    void onEntryClick(View view, int position);
}

public void setOnEntryClickListener(OnEntryClickListener onEntryClickListener) {
    mOnEntryClickListener = onEntryClickListener;
}

तो आपका नया, क्लिक-सपोर्टिंग Adapter पूरा हो गया है।

अब, इसका उपयोग करते हैं ...

    ExampleClickAdapter clickAdapter = new ExampleClickAdapter(yourObjects);
    clickAdapter.setOnEntryClickListener(new ExampleClickAdapter.OnEntryClickListener() {
        @Override
        public void onEntryClick(View view, int position) {
            // stuff that will happen when a list item is clicked
        }
    });

यह मूल रूप से है कि आप एक सामान्य सेट कैसे करेंगे Adapter , सिवाय इसके कि आप अपनी सेटर विधि का उपयोग करते हैं जिसे आपने नियंत्रित किया है कि आप क्या करेंगे जब आपका उपयोगकर्ता किसी विशेष सूची आइटम पर क्लिक करेगा।

आप GitHub पर इस Gist पर बने उदाहरणों के एक सेट के माध्यम से भी देख सकते हैं:

https://gist.github.com/FarbodSalamat-Zadeh/7646564f48ee708c1582c013e1de4f07


आपको getLayoutPosition () के स्थान पर getAdapterPosition () का उपयोग करना चाहिए
BladeCoder

2

यही मैं पुन: उपयोग करता हूं OnClickListener

  public class TestAdapter extends RecyclerView.Adapter<TestAdapter.MyviewHolder>
                                         implements View.OnClickListener

ViewHoder में मदरआउट के माता-पिता को लें

  public class MyviewHolder extends RecyclerView.ViewHolder {

       LinearLayout linearLayout_item;

        public MyviewHolder(View itemView) {
            super(itemView);
            linearLayout_item=itemView.findViewById(R.id.linearLayout_item);
        }
    }

स्थिति के रूप में onBindViewHolder सेट टैग

   @Override
    public void onBindViewHolder(MyviewHolder holder, int position) {

       holder.linearLayout_item.setTag(position);
       holder.linearLayout_item.setOnClickListener(this);
    }

और ओनक्लिक में

 @Override
public void onClick(View v) {

    int position = (int) v.getTag();
    switch (v.getId()) {
        case R.id.linearLayout_item:

            // do some thing with position 

            break;
    }
}

2

मेरे लिए ऐसा करने का साफ तरीका यही है।

एडॉप्टर कंस्ट्रक्टर

private class EnvironmentTypeRecyclerViewAdapter extends RecyclerView.Adapter<EnvironmentTypeRecyclerViewAdapter.ViewHolder>
{
     private final EnvironmentTypeRecyclerViewAdapterListener mEnvironmentTypeRecyclerViewAdapterListener;
     private List<Environment> mEnvironmentsData;

     public class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener
     {
         public ViewHolder(View v)
         {
             super(v);
             v.setOnClickListener(this);

         }

         @Override
         public void onClick(View v)
         {
              Environment environment = mEnvironmentsData.get(getAdapterPosition());
              if (mEnvironmentTypeRecyclerViewAdapterListener != null && environment != null) {
                      mEnvironmentTypeRecyclerViewAdapterListener.onListItemSelected(environment);      
              }
        }

        public EnvironmentTypeRecyclerViewAdapter(List<SmallCellEnvironment> environments, EnvironmentTypeRecyclerViewAdapterListener environmentTypeRecyclerViewAdapterListener)
        {
            mEnvironmentTypeRecyclerViewAdapterListener = environmentTypeRecyclerViewAdapterListener;
            mEnvironmentsData = environments;
        }
}

लिंक्ड इंटरफ़ेस

private interface EnvironmentTypeRecyclerViewAdapterListener
{
    void onListItemSelected(Environment environment);
}
हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.