CheckBox ใน RecyclerView คอยตรวจสอบรายการต่างๆ


93

นี่คือ XML สำหรับรายการของฉันใน RecyclerView

<android.support.v7.widget.CardView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:card_view="http://schemas.android.com/apk/res-auto"
    android:id="@+id/cvItems"
    android:layout_height="wrap_content"
    android:layout_width="fill_parent"
    android:layout_margin="2dp"
    card_view:cardElevation="0dp"
    card_view:contentPadding="0dp"
    card_view:cardBackgroundColor="#FFFFFF"
    >

    <LinearLayout
        android:orientation="horizontal"
        android:layout_height="fill_parent"
        android:layout_width="fill_parent">
        <TextView
            android:layout_width="0dip"
            android:layout_height="match_parent"
            android:layout_weight="0.8"
            android:id="@+id/tvContent"
            android:textSize="15dp"
            android:paddingLeft="5dp"
            android:paddingRight="5dp" />
        <CheckBox
            android:id="@+id/cbSelect"
            android:layout_width="0dip"
            android:layout_weight="0.2"
            android:layout_height="match_parent"
            android:button="@drawable/cb_checked"
            android:gravity="center_horizontal"
            android:textAlignment="center"
            android:layout_gravity="center_horizontal" />
    </LinearLayout>
</android.support.v7.widget.CardView>

และนี่คืออะแดปเตอร์ RecyclerView ที่ขยายเลย์เอาต์ด้านบนสำหรับแต่ละรายการ:

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

    private ArrayList<ObjectIncome> myItems = new ArrayList<>();

    public AdapterTrashIncome(ArrayList<ObjectIncome> getItems, Context context){
        try {
            mContext = context;
            myItems = getItems;
        }catch (Exception e){
            Log.e(FILE_NAME, "51: " + e.toString());
            e.printStackTrace();
        }
    }

    public class ViewHolder extends RecyclerView.ViewHolder {
        public TextView tvContent;
        public CheckBox cbSelect;

        public ViewHolder(View v) {
            super(v);
            tvContent = (TextView) v.findViewById(R.id.tvContent);
            cbSelect = (CheckBox) v.findViewById(R.id.cbSelect);
        }
    }

    @Override
    public void onBindViewHolder(ViewHolder holder, final int position) {
        final ObjectIncome objIncome = myItems.get(position);
        String content = "<b>lalalla</b>";
        holder.tvContent.setText(Html.fromHtml(content));
    }
}

ปัญหาคือสมมติว่าฉันมี 10 รายการใน RecyclerView เมื่อฉันเลือกช่องทำเครื่องหมายในรายการ 1,2,3 แล้วฉันก็เลื่อน RecyclerView ลงมาทันใดนั้นก็มีการเลือกบางรายการเช่นรายการ 8,9 และเมื่อฉันเลื่อนขึ้นอีกครั้งจะมีการตรวจสอบรายการที่ 1 และ 3 แต่ไม่ใช่รายการที่ 2 มีความคิดอย่างไร


ลองใช้ไลบรารีนี้ดู ViewStates ช่วยบันทึกสถานะเมื่อเลื่อน
Vitaly

คำตอบ:


167

นั่นเป็นพฤติกรรมที่คาดหวัง คุณไม่ได้ตั้งค่าช่องทำเครื่องหมายของคุณถูกเลือกหรือไม่ คุณกำลังเลือกหนึ่งรายการและผู้ถือ View จะเลือกไว้ คุณสามารถเพิ่มตัวแปรบูลีนลงในออบเจ็กต์ ObjectIncome ของคุณและรักษาสถานะการเลือกรายการของคุณได้

คุณอาจดูตัวอย่างของฉัน คุณสามารถทำสิ่งนี้ได้:

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

    private ArrayList<ObjectIncome> myItems = new ArrayList<>();

    public AdapterTrashIncome(ArrayList<ObjectIncome> getItems, Context context){
        try {
            mContext = context;
            myItems = getItems;
            }catch (Exception e){
            Log.e(FILE_NAME, "51: " + e.toString());
            e.printStackTrace();
        }
    }

    public class ViewHolder extends RecyclerView.ViewHolder {
        public TextView tvContent;
        public CheckBox cbSelect;

        public ViewHolder(View v) {
            super(v);
            tvContent = (TextView) v.findViewById(R.id.tvContent);
            cbSelect = (CheckBox) v.findViewById(R.id.cbSelect);
        }
    }

    @Override
    public void onBindViewHolder(ViewHolder holder, final int position) {
        final ObjectIncome objIncome = myItems.get(position);
        String content = "<b>lalalla</b>";
        holder.tvContent.setText(Html.fromHtml(content));

        //in some cases, it will prevent unwanted situations
        holder.cbSelect.setOnCheckedChangeListener(null);

        //if true, your checkbox will be selected, else unselected
        holder.cbSelect.setChecked(objIncome.isSelected());

        holder.cbSelect.setOnCheckedChangeListener(new OnCheckedChangeListener() {
                @Override
                public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
                    //set your object's last status
                    objIncome.setSelected(isChecked);
            }
        });

    }
}

19
มันไม่ได้ผล คุณต้องเขียนholder.cbSelect.setOnCheckedChangeListener(null);ก่อนholder.cbSelect.setChecked(objIncome.isSelected())
Jemshit Iskenderov

2
มีเหตุผลว่าทำไมการตั้งค่า holder.cbSelect.setOnCheckedChangeListener (null); ได้ผล?
Deb

4
@oguzhand สวัสดีฉันลองใช้วิธีแก้ปัญหาของคุณแล้ว แต่มันไม่ได้ผลทั้งแบบมีหรือไม่มีการตั้งค่าตัวฟังเป็น null
Abbas

3
@oguzhand นี่คือรหัสจากonBindViewHolder. @Override public void onBindViewHolder(final ItemHolder holder, int position) { holder.checkBox.setOnCheckedChangeListener(null); holder.checkBox.setSelected(list.get(position).isSelected()); holder.checkBox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { @Override public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) { list.get(holder.getAdapterPosition()).setSelected(isChecked); } });
Abbas

1
@Suisse คุณต้องรักษาสถานะช่องทำเครื่องหมายในวัตถุเนื่องจาก ViewHolder เป็นเพียงตัวยึด หากคุณมี 100 รายการคุณมีเพียงประมาณ 6-7 รายการ (ขึ้นอยู่กับขนาดหน้าจอและเค้าโครง) ViewHolders และใช้วัตถุทั้งหมดเหล่านี้ในวงจร
OğuzhanDöngül

22

ในระยะสั้นเป็นเพราะการรีไซเคิลมุมมองและนำกลับมาใช้อีกครั้ง!

คุณจะหลีกเลี่ยงสิ่งนั้นได้อย่างไร:

1. ในการonBindViewHolderตรวจสอบว่าคุณควรเลือกหรือยกเลิกการเลือกช่อง อย่าลืมใส่ทั้ง if และ else

if (...)
    holder.cbSelect.setChecked(true);
else
    holder.cbSelect.setChecked(false);
  1. ใส่กล่องกาเครื่องหมายฟัง! เมื่อใดก็ตามที่รูปปั้นที่ตรวจสอบเปลี่ยนไปให้อัปเดตวัตถุที่เกี่ยวข้องในmyItemsอาร์เรย์ของคุณด้วย! ดังนั้นเมื่อใดก็ตามที่มีการแสดงมุมมองใหม่มันจะอ่านรูปปั้นใหม่ล่าสุดของวัตถุ

จุดที่สองของคุณคือกุญแจสำคัญ แม้ว่าจะทำงานได้ดีที่สุดในสถานการณ์ แต่เมื่อชุดข้อมูลเริ่มต้นมีข้อมูลเกี่ยวกับสถานะที่ตรวจสอบ (ซึ่งเป็นกรณีของฉัน)
Attila Orosz

1
นี่คือคำตอบที่ตรงประเด็นและถูกต้องมากขึ้น SetCheck สำหรับทั้งจริงและเท็จใน onBindViewHolder เป็นกุญแจสำคัญ
Beeing Jk

ในกรณีของฉันฉันต้องบันทึกข้อมูลในโมเดลข้อมูลด้วยค่าเริ่มต้นisChecked falseสำหรับชุดข้อมูลทั้งหมดเมื่อเริ่มต้นจากนั้นonCheckChangedฉันก็อัปเดตisCheckedค่าเป็นtrueหรือfalseและตามที่บอกไว้ในคำตอบที่ใช้ว่าตรวจสอบหรือไม่
Ali Tamoor

20

ใช้สิ่งนี้เฉพาะในกรณีที่คุณมีรายการจำนวน จำกัด ในมุมมองผู้ใช้งานของคุณ
ฉันลองใช้ค่าบูลีนในโมเดลและคงสถานะช่องทำเครื่องหมายไว้ แต่มันไม่ได้ช่วยอะไรในกรณีของฉัน สิ่งที่ใช้ได้ผลสำหรับฉันคือthis.setIsRecyclable (false);

public class ComponentViewHolder extends RecyclerView.ViewHolder {
    public MyViewHolder(View itemView) {
        super(itemView);
        ....
        this.setIsRecyclable(false);
    }

สามารถดูคำอธิบายเพิ่มเติมได้ที่นี่https://developer.android.com/reference/android/support/v7/widget/RecyclerView.ViewHolder.html#isRecyclable ()

หมายเหตุ:นี่เป็นวิธีแก้ปัญหาชั่วคราว หากต้องการใช้อย่างถูกต้องคุณสามารถอ้างถึงเอกสารที่ระบุว่า"ควรจับคู่การโทรไปยัง setIsRecyclable () เสมอ (การเรียกหนึ่งครั้งไปยัง setIsRecyclabe (false) ควรจับคู่กับการเรียกใช้ setIsRecyclable (true) ในภายหลัง) คู่ของการโทรอาจซ้อนกัน เนื่องจากรัฐถูกนับเป็นการอ้างอิงภายใน " ฉันไม่รู้ว่าจะทำอย่างไรในรหัสหากมีใครสามารถให้รหัสเพิ่มเติมเกี่ยวกับเรื่องนี้ได้


ช่วยอธิบายวิธีการใช้งานหน่อยได้ไหม?
UserName_Untold

44
มันไม่สิ้นเปลืองตรรกะที่อยู่เบื้องหลังรีไซเคิล View?
Eren

3
ฉันลองใช้สิ่งนี้ด้วยรายการยาวซึ่งแก้ไขปัญหาการตรวจสอบแบบสุ่ม แต่เมื่อฉันเลื่อนลงและเลื่อนขึ้นอีกครั้งด้วยรายการยาวช่องทำเครื่องหมายที่เลือกไว้จะหายไป :(
SonDang

2
ไม่ใช่ความคิดที่ดีที่จะทำให้มุมมองที่ไม่สามารถรีไซเคิลได้เนื่องจากจะทำให้หน่วยความจำหมดไปและคุณจะสูญเสียประโยชน์ส่วนใหญ่จากมุมมองรีไซเคิล
Arthur

ฉันเห็นด้วยกับพวกคุณ @ eren130 และ Arthur ฉันได้แก้ไขโพสต์แล้วและจะขอบคุณมากถ้าเราสามารถหาวิธีใช้ setIsRecyclable (จริง / เท็จ) ได้ อย่างถูกต้อง
Rana Ranvijay Singh

13

เพียงเพิ่มสองวิธีการแทนที่ของ RecyclerView

@Override
public long getItemId(int position) {
    return position;
}

@Override
public int getItemViewType(int position) {
    return position;
}

2
อย่าทำแบบนั้น !! มันจะข้ามกลไกการรีไซเคิลของรีไซเคิลดูและสูญเสียจุดทั้งหมดในการใช้งาน
Hanoch Moreno

1
ไม่มันจะไม่ส่งคืนตำแหน่งที่แน่นอนของทุกมุมมองรีไซเคิลในที่ยึดมุมมอง
Harish Reddy

1
Harish บางทีฉันอาจจะขาดอะไรไป แต่เท่าที่ฉันรู้การทำเช่นนี้คุณจะบอกอะแด็ปเตอร์ว่าจำนวนประเภทรายการคือจำนวนรายการ ความหมายคือไม่สามารถรีไซเคิลสิ่งของใด ๆ ได้เนื่องจากไม่มีมุมมองที่เหมือนกัน ง่ายต่อการทดสอบ เพียงแค่บันทึกการอ้างอิง viewHolder.itemView ภายใน onBindViewHolder และดูว่ามี viewHolders สองรายการที่มีการอ้างอิงมุมมองเดียวกันหรือไม่ การทดสอบควรอยู่ในรายชื่อที่ยาวนานเพื่อให้ระบบรีไซเคิลถูกดำเนินการ
Hanoch Moreno

3
มันตื่นขึ้นมาอย่างไร้ที่ติช่วยวันของฉัน
Kundan

5
ในกรณีที่คุณมีรายการรีไซเคิลมากกว่า 100 รายการโซลูชันนี้จะโหลดรายการทั้งหมดพร้อมกันซึ่งอาจทำให้ OutOfMemoryException ในกรณีที่คุณมีรูปภาพหมายความว่ามิฉะนั้นโซลูชันนี้จะสมบูรณ์แบบ @Kundan
Harish Reddy

11

คุณสามารถใช้คลาสโมเดลเพื่อติดตามช่องทำเครื่องหมายของรายการรีไซเคิลแต่ละรายการ อ้างอิงแบบเต็มมาจาก: RecyclerView Checkbox Android

setTagและgetTagใช้เพื่อติดตามสถานะช่องทำเครื่องหมาย ตรวจสอบลิงค์อ้างอิงแบบเต็มสำหรับข้อมูลเพิ่มเติม นอกจากนี้ยังสอนวิธีการส่งรายการการตรวจสอบเพื่อNEXTACTIVITY

สร้างแบบจำลอง

public class Model {

    private boolean isSelected;
    private String animal;

    public String getAnimal() {
        return animal;
    }

    public void setAnimal(String animal) {
        this.animal = animal;
    }

    public boolean getSelected() {
        return isSelected;
    }

    public void setSelected(boolean selected) {
        isSelected = selected;
    }
}

สร้าง integer.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <integer name="btnplusview">1</integer>
    <integer name="btnpluspos">2</integer>
</resources>

ในที่สุดอะแดปเตอร์จะมีลักษณะดังนี้:

 import android.content.Context;
 import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
    import android.view.View;
 import android.view.ViewGroup;
 import android.widget.CheckBox;
 import android.widget.TextView;
 import android.widget.Toast;

 import java.util.ArrayList;


  public class CustomAdapter extends RecyclerView.Adapter<CustomAdapter.MyViewHolder> {

private LayoutInflater inflater;
public static ArrayList<Model> imageModelArrayList;
private Context ctx;

public CustomAdapter(Context ctx, ArrayList<Model> imageModelArrayList) {

    inflater = LayoutInflater.from(ctx);
    this.imageModelArrayList = imageModelArrayList;
    this.ctx = ctx;
}

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

    View view = inflater.inflate(R.layout.rv_item, parent, false);
    MyViewHolder holder = new MyViewHolder(view);

    return holder;
}

@Override
public void onBindViewHolder(final CustomAdapter.MyViewHolder holder, int position) {

    holder.checkBox.setText("Checkbox " + position);
    holder.checkBox.setChecked(imageModelArrayList.get(position).getSelected());
    holder.tvAnimal.setText(imageModelArrayList.get(position).getAnimal());

   // holder.checkBox.setTag(R.integer.btnplusview, convertView);
    holder.checkBox.setTag(position);
    holder.checkBox.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {

            Integer pos = (Integer) holder.checkBox.getTag();
            Toast.makeText(ctx, imageModelArrayList.get(pos).getAnimal() + " clicked!", Toast.LENGTH_SHORT).show();

            if (imageModelArrayList.get(pos).getSelected()) {
                imageModelArrayList.get(pos).setSelected(false);
            } else {
                imageModelArrayList.get(pos).setSelected(true);
            }
        }
    });


}

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

class MyViewHolder extends RecyclerView.ViewHolder {

    protected CheckBox checkBox;
    private TextView tvAnimal;

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

        checkBox = (CheckBox) itemView.findViewById(R.id.cb);
        tvAnimal = (TextView) itemView.findViewById(R.id.animal);
    }

}

}


3

การใช้ Kotlin สิ่งเดียวที่แก้ปัญหานี้สำหรับฉันคือการล้างOnCheckedChangeListenerค่าก่อนตั้งค่าตัวแปรจากนั้นสร้างใหม่OnCheckedChangeListenerafter checkedได้รับการตั้งค่า

ฉันทำสิ่งต่อไปนี้ในไฟล์ RecyclerView.ViewHolder

task.setOnCheckedChangeListener(null)
task.isChecked = item.status
task.setOnCheckedChangeListener { _: CompoundButton, checked: Boolean ->
    item.status = checked
    ...
    do more stuff
    ...
}

นี่คือการทำงานที่สมบูรณ์แบบ ฉันไม่ทำไม แต่มันใช้ได้เฉพาะเมื่อใครก็ตามที่ใช้ KOTLIN!
Aditya S.

2

ตามที่ระบุไว้ข้างต้นสถานะที่ตรวจสอบของวัตถุควรรวมอยู่ในคุณสมบัติของวัตถุ ในบางกรณีคุณอาจต้องเปลี่ยนสถานะการเลือกอ็อบเจ็กต์โดยคลิกที่อ็อบเจ็กต์นั้นเองและให้ CheckBox แจ้งเกี่ยวกับสถานะจริง (ไม่ว่าจะเลือกหรือไม่ได้เลือก) จากนั้นช่องทำเครื่องหมายจะใช้สถานะของวัตถุที่ตำแหน่งจริงของอะแด็ปเตอร์ที่กำหนดซึ่งเป็น (โดยค่าเริ่มต้น / ในกรณีส่วนใหญ่) ตำแหน่งขององค์ประกอบในรายการ

ตรวจสอบตัวอย่างด้านล่างซึ่งอาจเป็นประโยชน์

import android.content.Context;
import android.graphics.Bitmap;
import android.net.Uri;
import android.provider.MediaStore;
import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.CheckBox;
import android.widget.CompoundButton;
import android.widget.ImageView;

import java.io.File;
import java.io.IOException;
import java.util.List;

public class TakePicImageAdapter extends RecyclerView.Adapter<TakePicImageAdapter.ViewHolder>{
    private Context context;
    private List<Image> imageList;

    public TakePicImageAdapter(Context context, List<Image> imageList) {
        this.context = context;
        this.imageList = imageList;
    }

    @Override
    public TakePicImageAdapter.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view= LayoutInflater.from(context).inflate(R.layout.image_item,parent,false);
        return new ViewHolder(view);
    }

    @Override
    public void onBindViewHolder(final TakePicImageAdapter.ViewHolder holder, final int position) {
        File file=new File(imageList.get(position).getPath());
        try {
            Bitmap bitmap= MediaStore.Images.Media.getBitmap(context.getContentResolver(), Uri.fromFile(file));
            holder.image.setImageBitmap(bitmap
            );
        } catch (IOException e) {
            e.printStackTrace();
        }
        holder.selectImage.setOnCheckedChangeListener(null);
        holder.selectImage.setChecked(imageList.get(position).isSelected());
        holder.selectImage.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
            @Override
            public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
                holder.selectImage.setChecked(isChecked);
                imageList.get(position).setSelected(isChecked);
            }
        });
        holder.image.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (imageList.get(position).isSelected())
                {
                    imageList.get(position).setSelected(false);
                    holder.selectImage.setChecked(false);
                }else
                {
                    imageList.get(position).setSelected(true);
                    holder.selectImage.setChecked(true);
                }
            }
        });

    }

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

    public class ViewHolder extends RecyclerView.ViewHolder {
        public ImageView image;public CheckBox selectImage;
        public ViewHolder(View itemView) {
            super(itemView);
            image=(ImageView)itemView.findViewById(R.id.image);
            selectImage=(CheckBox) itemView.findViewById(R.id.ch);

        }
    }
}


2

ในกรณีของฉันสิ่งนี้ได้ผล

@Override
public void onViewRecycled(MyViewHolder holder) {
    holder.checkbox.setChecked(false); // - this line do the trick
    super.onViewRecycled(holder);
}

2

ใช้อาร์เรย์เพื่อเก็บสถานะของรายการ

ในอะแด็ปเตอร์ให้ใช้ Map หรือSparseBooleanArray (ซึ่งคล้ายกับแผนที่ แต่เป็นคู่คีย์ - ค่า int และ boolean) เพื่อเก็บสถานะของรายการทั้งหมดในรายการของเราจากนั้นใช้คีย์และค่าเพื่อเปรียบเทียบ เมื่อสลับสถานะที่เลือก

ในอะแดปเตอร์สร้างไฟล์ SparseBooleanArray

// sparse boolean array for checking the state of the items

    private SparseBooleanArray itemStateArray= new SparseBooleanArray();

จากนั้นในตัวจัดการการคลิกรายการonClick()ใช้สถานะของรายการใน itemStateArray เพื่อตรวจสอบก่อนที่จะสลับนี่คือตัวอย่าง

        @Override
        public void onClick(View v) {
            int adapterPosition = getAdapterPosition();
            if (!itemStateArray.get(adapterPosition, false)) {
                mCheckedTextView.setChecked(true);
                itemStateArray.put(adapterPosition, true);
            }
            else  {
                mCheckedTextView.setChecked(false);
                itemStateArray.put(adapterPosition, false);
            }
        }

นอกจากนี้ยังใช้อาร์เรย์บูลีนแบบกระจัดกระจายเพื่อตั้งค่าสถานะที่ตรวจสอบเมื่อมุมมองถูกผูกไว้

@Override
public void onBindViewHolder(ViewHolder holder, int position) {
    holder.bind(position);
}

@Override
public int getItemCount() {
    if (items == null) {
        return 0;
    }
    return items.size();
}

 void loadItems(List<Model> tournaments) {
    this.items = tournaments;
    notifyDataSetChanged();
}


class ViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {

    CheckedTextView mCheckedTextView;

    ViewHolder(View itemView) {
        super(itemView);
        mCheckedTextView = (CheckedTextView) itemView.findViewById(R.id.checked_text_view);
        itemView.setOnClickListener(this);
    }

    void bind(int position) {
        // use the sparse boolean array to check
        if (!itemStateArray.get(position, false)) {
            mCheckedTextView.setChecked(false);}
        else {
            mCheckedTextView.setChecked(true);
        }
    }

และอะแดปเตอร์สุดท้ายจะเป็นเช่นนี้


1

คุณต้องแยกการโต้ตอบ onBindViewHolder (ตรรกะ) กับ CheckBox และการโต้ตอบของผู้ใช้ด้วยช่องทำเครื่องหมาย ฉันใช้ OnCheckedChangeListener สำหรับการโต้ตอบของผู้ใช้ (เห็นได้ชัด) และ ViewHolder.bind () สำหรับตรรกะนั่นคือเหตุผลที่คุณต้องตั้งค่าตัวฟังที่ตรวจสอบแล้วเป็น null ก่อนตั้งค่าตัวยึดและหลังจากที่ผู้ถือพร้อมแล้วให้กำหนดค่าตัวฟังที่ตรวจสอบแล้วสำหรับการโต้ตอบของผู้ใช้

boolean[] checkedStatus = new boolean[numberOfRows];

@Override
        public void onBindViewHolder(final RecyclerView.ViewHolder holder, int position) {
        final ViewHolderItem itemHolder = (ViewHolderItem) holder;

        //holder.bind should not trigger onCheckedChanged, it should just update UI
        itemHolder.checkBox.setOnCheckedChangeListener(null);

        itemHolder.bind(position);

        itemHolder.checkBox.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
            @Override
            public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
                if (isChecked) {
                    checkedStatus[holder.getAdapterPosition()] = true;
                    performCheckedActions(); //your logic here
                } else {
                    checkedStatus[holder.getAdapterPosition()] = false;
                    performUncheckedActions(); //your logic here
                }
            }
        });
    }

public void bind(int position) {
            boolean checked = checkedStatus[position];
            if (checked) {
                checkBox.setChecked(false);
            } else {
                checkBox.setChecked(true);
            }
        }

1

ผมขอแนะนำให้ไม่ได้ที่จะใช้ในcheckBox.setOnCheckedChangeListener recyclerViewAdapterเนื่องจากในการเลื่อน RecyclerView checkBox.setOnCheckedChangeListenerจะถูกยิงโดยอะแด็ปเตอร์ มันไม่ปลอดภัย ให้ใช้checkBox.setOnClickListenerเพื่อโต้ตอบกับอินพุตของผู้ใช้แทน

ตัวอย่างเช่น:

     public void onBindViewHolder(final ViewHolder holder, int position) {
        /*
         .
         .
         .
         .
         .
         .
        */

        holder.checkBoxAdapterTasks.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                boolean isChecked =  holder.checkBoxAdapterTasks.isChecked();
                if(isChecked){
                    //checkBox clicked and checked
                }else{
                    //checkBox clicked and unchecked
                }

            }
        });

    }

1

ปัญหาของโซลูชันนี้ที่ฉันพบคือโดยการสร้างอาร์เรย์ส่วนกลางแบบคงที่และใช้ใน ADAPER CLASS "onBindViewHolder" ซึ่งฉันได้สร้างสิ่งที่จำเป็นทั้งหมด / วัตถุส่วนกลาง

public class RVAdapter extends RecyclerView.Adapter<RVAdapter.PersonViewHolder> {
private Context context;
public static class PersonViewHolder extends RecyclerView.ViewHolder {

    CardView cv;
    TextView question,category;
    TextView personAge;
    ImageView upvote;
    Button b1;
    public static int k;
    private int visibleThreshold = 5;
    public static int i=0;
     static int  check[]; //Static array
    PersonViewHolder(View itemView,int i) {
        super(itemView);
        if(i==PersonViewHolder.k)
        {
            b1=(Button)itemView.findViewById(R.id.loadmore);

        }
        else
        {
            cv = (CardView)itemView.findViewById(R.id.cv);
            question = (TextView)itemView.findViewById(R.id.question);
            category = (TextView)itemView.findViewById(R.id.text_categ);
            personAge = (TextView)itemView.findViewById(R.id.text1);
            upvote = (ImageView)itemView.findViewById(R.id.upvote);

        }

    }

}

ที่นี่ (ในผู้สร้างของ RVADAPTER CLASS) ฉันกำหนดขนาดให้กับอาร์เรย์เท่ากับขนาดของ / ไม่ใช่ของรายการที่ฉันจะแสดงในมุมมองผู้รีไซเคิล

List<Person> persons;

RVAdapter(List<Person> persons){
    this.persons = persons;
    PersonViewHolder.check=new int[persons.size()];
    PersonViewHolder.k=persons.size();
}

BindViewHolder, I, ใช้แนวคิดนี้กับปุ่มเมื่อฉันคลิกที่ปุ่มภาพพื้นหลังของปุ่มจะเปลี่ยนไป ออบเจ็กต์ของปุ่มที่ฉันใช้คือชื่อเป็น "upvote" เนื่องจาก "i" ถือตำแหน่งของแต่ละรายการในมุมมองผู้รีไซเคิลฉันใช้มันเป็นดัชนีของอาร์เรย์ซึ่งทำงานเป็นค่าสถานะและเป็นตัวติดตามสถานะขององค์ประกอบ

@Override
public void onBindViewHolder(final PersonViewHolder personViewHolder, final int i) {
    if(i==PersonViewHolder.k) {
        personViewHolder.b1.setText("load more");

    }
    else
     {
        personViewHolder.question.setText(persons.get(i).name);
        personViewHolder.personAge.setText(persons.get(i).age);

         if(personViewHolder.check[i]==0)
         {personViewHolder.upvote.setBackgroundResource(R.drawable.noupvote);
         }
         else
         {
             personViewHolder.upvote.setBackgroundResource(R.drawable.upvote);

         }

         personViewHolder.upvote.setOnClickListener(new View.OnClickListener() {
             @Override
             public void onClick(View v) {
                 if(personViewHolder.check[i]==0)
                 {personViewHolder.check[i]=1;
                     personViewHolder.upvote.setBackgroundResource(R.drawable.upvote);


                 }
                 else
                 {personViewHolder.check[i]=0;
                     personViewHolder.upvote.setBackgroundResource(R.drawable.noupvote);

                 }


             }
         });
        // personViewHolder.personPhoto.setImageResource(persons.get(i).photoId);
    }

}

1

ฉันมีปัญหาเดียวกัน เมื่อฉันคลิกที่ปุ่มสลับของรายการในจุดรีไซเคิลของฉันที่ตรวจสอบแล้วปุ่ม Toggle ปรากฏในทุกรายการที่ 10 (ตัวอย่างเช่นหากคลิกในรายการที่มีดัชนี 0 รายการที่มีดัชนี 9, 18, 27 ก็จะถูกคลิกด้วย) ประการแรกรหัสของฉันใน onBindViewHolder คือ:

if (newsItems.get(position).getBookmark() == 1) {
            holder.getToggleButtonBookmark().setChecked(true);
        }

แต่แล้วฉันก็เพิ่มคำสั่งอื่น

if (newsItems.get(position).getBookmark() == 1) {
            holder.getToggleButtonBookmark().setChecked(true);
//else statement prevents auto toggling
        } else{
            holder.getToggleButtonBookmark().setChecked(false);
        }

และปัญหาได้รับการแก้ไข


ขอบคุณ. ส่วนอื่นจะล้างช่องทำเครื่องหมายหากตรวจสอบตามค่าเริ่มต้นเมื่อรีไซเคิลดูมุมมองเดียวกัน
Adarsh ​​Vijayan P

1

โอเคมีคำตอบมากมายที่นี่ฉันจะโพสต์รหัสของฉันและฉันจะอธิบายสิ่งที่ฉันทำ ... มันอาจจะช่วยรุ่นน้องอย่างฉันก็ได้: D

1- วัตถุประสงค์:

เราจะสร้างรายการRecyclerViewที่มีCheckBoxและRadioButtonสิ่งนี้:

ป้อนคำอธิบายภาพที่นี่ 2- รุ่นคลาส

public class ModelClass {
private String time;
private boolean checked;
private boolean free;
private boolean paid;

public TherapistScheduleModel(String time, boolean checked, boolean free, boolean paid) {
    this.time = time;
    this.checked = checked;
    this.free = free;
    this.paid = paid;
}

public boolean isFree() {
    return free;
}

public void setFree(boolean free) {
    this.free = free;
}

public boolean isPaid() {
    return paid;
}

public void setPaid(boolean paid) {
    this.paid = paid;
}

public String getTime() {
    return time;
}

public void setTime(String time) {
    this.time = time;
}

public boolean getChecked() {
    return checked;
}

public void setChecked(boolean checked) {
    this.checked= checked;
}
}

อะแดปเตอร์ 3-My Amazing

public class MyAdapter extends RecyclerView.Adapter<MyAdapter.MyViewHolder> {
private Context context;
private ListAllListeners listAllListeners;
private ArrayList<ModelClass> mDataList;

public MyAdapter(Context context, ArrayList<ModelClass> mDataList,
                             ListAllListeners listAllListeners) {
    this.mDataList = mDataList;
    this.listAllListeners = listAllListeners;
    this.context = context;
}

@NonNull
@Override
public MyViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
    LayoutInflater inflater = LayoutInflater.from(parent.getContext());
    View view = inflater.inflate(R.layout.single_view, parent, false);
    return new MyViewHolder(view);
}

@Override
public int getItemCount() {
    if (mDataList != null)
        return mDataList.size();
    else
        return 0;
}

@Override
public void onBindViewHolder(@NonNull final MyViewHolder holder, final int position) {
     //important to:
    //setOnCheckedChangeListener to 'null'
    holder.checkBoxTime.setOnCheckedChangeListener(null);
    holder.freeRB.setOnCheckedChangeListener(null);
    holder.paidRB.setOnCheckedChangeListener(null);

    //Check Box
            holder.checkBoxTime.setText(mDataList.get(holder.getAdapterPosition()).getTime());
    //here we check if the item is checked or not from the model.
    if(mDataList.get(holder.getAdapterPosition()).getChecked())
        holder.checkBoxTime.setChecked(true);
    else
        holder.checkBoxTime.setChecked(false);

    holder.checkBoxTime.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
        @Override
        public void onCheckedChanged(CompoundButton compoundButton, boolean b) {
            if (b) {
                mDataList.get(holder.getAdapterPosition()).setChecked(true);
                listAllListeners.onItemCheck(holder.checkBoxTime.getText().toString(), holder.getAdapterPosition());
            }
            else {
                mDataList.get(holder.getAdapterPosition()).setChecked(false);
                listAllListeners.onItemUncheck(holder.checkBoxTime.getText().toString(), holder.getAdapterPosition());
            }
        }
    });

    //Radio Buttons

    if(mDataList.get(holder.getAdapterPosition()).isFree())
        holder.freeRB.setChecked(true);
    else
        holder.freeRB.setChecked(false);
    holder.freeRB.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
        @Override
        public void onCheckedChanged(CompoundButton compoundButton, boolean b) {
            if (b) {
                mDataList.get(holder.getAdapterPosition()).setFree(true);
                listAllListeners.onFreeCheck(holder.freeRB.getText().toString(), holder.getAdapterPosition());
            } else {
                mDataList.get(holder.getAdapterPosition()).setFree(false);
                listAllListeners.onFreeUncheck(holder.freeRB.getText().toString(), holder.getAdapterPosition());
            }
        }
    });

   //***and so on to paidRB***

}//end onBindViewHolder()

public interface ListAllListeners {
//here is a list of clicked listeners to use them as you want ;).
//you can get a list of checked or unChecked of all 
        void onItemCheck(String checkBoxName, int position);
        void onItemUncheck(String checkBoxName, int position);
        void onFreeCheck(String name, int pos);
        void onFreeUncheck(String name, int pos);
        void onPaidCheck(String name, int pos);
        void onPaidUncheck(String name, int pos);
    }

    class MyViewHolder extends RecyclerView.ViewHolder {

        CheckBox checkBoxTime;
        RadioButton freeRB, paidRB;

        MyViewHolder(View itemView) {
            super(itemView);
            checkBoxTime = itemView.findViewById(R.id.timeCheckBox);
            freeRB = itemView.findViewById(R.id.freeRadioBtn);
            paidRB = itemView.findViewById(R.id.paidRadioBtn);
        }
    }//end class MyViewHolder

    }//end class

3- ในกิจกรรมคุณจะได้รับสิ่งนี้:

myAdapter= new MyAdapter(getActivity().getApplicationContext(), mDataList,
                new MyAdapter.ListAllListeners() {

                    @Override
                    public void onItemCheck(String checkBoxName, int position) {
                        Toast.makeText(getActivity(), "" + checkBoxName + "  " + position, Toast.LENGTH_SHORT).show();
                    }

                    @Override
                    public void onItemUncheck(String checkBoxName, int position) {
                        Toast.makeText(getActivity(), "" + checkBoxName + "  " + position, Toast.LENGTH_SHORT).show();
                    }

                    @Override
                    public void onFreeCheck(String name, int position) {

                        Toast.makeText(getActivity(), "" + name + "  " + position, Toast.LENGTH_SHORT).show();
                    }

                    @Override
                    public void onFreeUncheck(String name, int position) {

                        Toast.makeText(getActivity(), "" + name + "  " + position, Toast.LENGTH_SHORT).show();
                    }

                    @Override
                    public void onPaidCheck(String name, int position) {

                        Toast.makeText(getActivity(), "" + name + "  " + position, Toast.LENGTH_SHORT).show();
                    }

                    @Override
                    public void onPaidUncheck(String name, int position) {

                        Toast.makeText(getActivity(), "" + name + "  " + position, Toast.LENGTH_SHORT).show();
                    }
                });

0

ฉันมีปัญหาเดียวกันในรายการ RecyclerView พร้อมสวิตช์และแก้ไขโดยใช้คำตอบ @oguzhand แต่ด้วยรหัสนี้ในรายการ CheckChangeListener:

if (buttonView.isPressed()) {
    if (isChecked) {
        group.setSelected(true);
    } else {
        group.setSelected(false);
    }
}else{
    if (isChecked) {
        buttonView.setChecked(false);
    } else {
        buttonView.setChecked(true);
    }
}

(โดยที่ 'group' คือเอนทิตีที่ฉันต้องการเลือก / ยกเลิกการเลือก)


0

คลาสสาธารณะ TagYourDiseaseAdapter ขยาย RecyclerView.Adapter {ส่วนตัว ReCyclerViewItemClickListener mRecyclerViewItemClickListener; บริบทส่วนตัว mContext;

List<Datum> deviceList = Collections.emptyList();

/**
 * Initialize the values
 *
 * @param context : context reference
 * @param devices : data
 */

public TagYourDiseaseAdapter(Context context, List<Datum> devices,
                             ReCyclerViewItemClickListener mreCyclerViewItemClickListener) {
    this.mContext = context;
    this.deviceList = devices;
    this.mRecyclerViewItemClickListener = mreCyclerViewItemClickListener;
}


/**
 * @param parent   : parent ViewPgroup
 * @param viewType : viewType
 * @return ViewHolder
 * <p>
 * Inflate the Views
 * Create the each views and Hold for Reuse
 */
@Override
public TagYourDiseaseAdapter.OrderHistoryViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {

    View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_tag_disease, parent, false);
    TagYourDiseaseAdapter.OrderHistoryViewHolder myViewHolder = new TagYourDiseaseAdapter.OrderHistoryViewHolder(view);
    return myViewHolder;
}


/**
 * @param holder   :view Holder
 * @param position : position of each Row
 *                 set the values to the views
 */
@Override
public void onBindViewHolder(final TagYourDiseaseAdapter.OrderHistoryViewHolder holder, final int position) {
    Picasso.with(mContext).load(deviceList.get(position).getIconUrl()).into(holder.document);
    holder.name.setText(deviceList.get(position).getDiseaseName());

    holder.radioButton.setOnCheckedChangeListener(null);
    holder.radioButton.setChecked(deviceList.get(position).isChecked());

    //if true, your checkbox will be selected, else unselected
    //holder.radioButton.setChecked(objIncome.isSelected());

    holder.radioButton.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() {
        @Override
        public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
            deviceList.get(position).setChecked(isChecked);
        }
    });


}

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


/**
 * Create The view First Time and hold for reuse
 * View Holder for Create and Hold the view for ReUse the views instead of create again
 * Initialize the views
 */

public class OrderHistoryViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener {
    ImageView document;
    TextView name;
    CheckBox radioButton;

    public OrderHistoryViewHolder(View itemView) {
        super(itemView);
        document = itemView.findViewById(R.id.img_tag);
        name = itemView.findViewById(R.id.text_tag_name);
        radioButton = itemView.findViewById(R.id.rdBtn_tag_disease);
        radioButton.setOnClickListener(this);
        //this.setIsRecyclable(false);
    }


    @Override
    public void onClick(View view) {
        mRecyclerViewItemClickListener.onItemClickListener(this.getAdapterPosition(), view);
    }
}

}


0

สิ่งนี้จะเกิดขึ้นเมื่อใช้setOnCheckedChangeListenerแทนการใช้งานนั้นsetObClickListenerและภายในที่จับง่ายนี้:

   if (list.get(position).isCheck())
            {
                list.get(position).setCheck(false);
            }
            else
            {
                list.get(position).setCheck(true);
            }

หมายเหตุ: ในแบบจำลองรายการของคุณเพิ่มตัวแปรบูลีนหนึ่งตัวพร้อมชื่อcheckและตั้งค่า getter และ setter สำหรับสิ่งนั้นในกรณีข้างต้นของฉันคือ setCheck และ isCheck

หวังว่าจะช่วยได้บ้างถ้าใช่ + โหวตให้คำตอบนี้



0

นี่เป็นเพราะการสร้างมุมมองซ้ำแล้วซ้ำเล่าตัวเลือกที่ดีที่สุดคือล้างแคชก่อนตั้งค่าอะแดปเตอร์

recyclerview.setItemViewCacheSize(your array.size());

0

ตัวอย่างที่สมบูรณ์
คลาสสาธารณะ ChildAddressAdapter ขยาย RecyclerView.Adapter <ChildAddressAdapter.CartViewHolder> {

private Activity context;
private List<AddressDetail> addressDetailList;
private int selectedPosition = -1;

public ChildAddressAdapter(Activity context, List<AddressDetail> addressDetailList) {
    this.context = context;
    this.addressDetailList = addressDetailList;
}

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

    LayoutInflater inflater = LayoutInflater.from(context);
    View myView = inflater.inflate(R.layout.address_layout, parent, false);
    return new CartViewHolder(myView);
}

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

    holder.adress_checkbox.setOnClickListener(view -> {
        selectedPosition = holder.getAdapterPosition();
        notifyDataSetChanged();
    });

    if (selectedPosition==position){
        holder.adress_checkbox.setChecked(true);
    }
    else {
        holder.adress_checkbox.setChecked(false);
    }


}

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

class CartViewHolder extends RecyclerView.ViewHolder
{
    TextView address_text,address_tag;
    CheckBox adress_checkbox;

    CartViewHolder(View itemView) {
        super(itemView);
        address_text = itemView.findViewById(R.id.address_text);
        address_tag = itemView.findViewById(R.id.address_tag);
        adress_checkbox = itemView.findViewById(R.id.adress_checkbox);
    }
}

}


-1

สิ่งที่ได้ผลสำหรับฉันคือทำให้ผู้ฟังเป็นโมฆะใน viewHolder เมื่อมุมมองกำลังจะถูกรีไซเคิล ( onViewRecycled):

 override fun onViewRecycled(holder: AttendeeViewHolder) {
            super.onViewRecycled(holder)
            holder.itemView.hasArrived.setOnCheckedChangeListener(null);
            holder.itemView.edit.setOnClickListener { null }
        }
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.