已检查的RadioButton回收

时间:2019-03-02 12:47:42

标签: android android-recyclerview android-radiogroup

我正在以编程方式将无线电组添加到我的recyclerview中,并且工作正常。 但是当我检查并滚动recyclerview时,它会丢失选中的收音机。

我已经看到了许多解决方案的方法和示例,但是我无法实现。已经连续几天了。

我正在将选中的单选保存在模型中,如下面的代码所示。

适配器:

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

    holder.priceGroup.removeAllViews();
    holder.priceGroup.setOnCheckedChangeListener(null);

    int id = (position+1)*100;
    checklistModel = mChecklists.get(position);
    holder.packageName.setText(checklistModel.getTitle());

    for(String price : checklistModel.getQuestions()){
        RadioButton rb = new RadioButton(NROptionLineAdapter.this.context);
        rb.setId(id++);
        rb.setText(price);
        holder.priceGroup.addView(rb);
    }
    holder.priceGroup.check(checklistModel.getSelectedId());

    holder.priceGroup.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
        @Override
        public void onCheckedChanged(RadioGroup group, int checkedId) {
            checklistModel.setSelectedId(checkedId);
            Log.d(TAG, "onCheckedChanged: " + checkedId);
        }
    });

}

持有人

    OnNROptionListener onNROptionListener;

    public NROptionLineHolder(View itemView, OnNROptionListener onNROptionListener) {
        super(itemView);

        packageName = itemView.findViewById(R.id.package_name);
        priceGroup = itemView.findViewById(R.id.price_grp);

//        priceGroup.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
//            @Override
//            public void onCheckedChanged(RadioGroup radioGroup, int i) {
//
//                Log.d(TAG, "onCheckedChanged: " + radioGroup.getCheckedRadioButtonId() + " " + i);
//            }
//        });

        this.onNROptionListener = onNROptionListener;
        itemView.setOnClickListener(this);

    }

    @Override
    public void onClick(View v) {
        onNROptionListener.onNROptionClick(getAdapterPosition());
    }

    public interface OnNROptionListener {
        void onNROptionClick(int position);
    }
}

编辑1-广播组

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="wrap_content">
    <TextView
        android:id="@+id/package_name"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintTop_toTopOf="parent" />
    <RadioGroup
        android:id="@+id/price_grp"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintTop_toBottomOf="@+id/package_name"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        android:orientation="horizontal"/>
</android.support.constraint.ConstraintLayout>

编辑2

根据要求,这是我的ChecklistActivity中的重要代码

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

    intent = getIntent();
    size = intent.getIntExtra("size", 0);
    nr = intent.getIntExtra("nr", 0);

    Log.d(TAG, "Checklist Activity - Qtd Questões: " + size);
    Log.d(TAG, "Checklist Activity - NR: " + nr);

    btnSaveCheck = findViewById(R.id.btnSaveChecklist);

    mRecyclerView = findViewById(R.id.package_lst);
    setupRecycler();

    btnSaveCheck.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            Toast.makeText(getApplicationContext(), "Sucesso", Toast.LENGTH_SHORT).show();
        }
    });
}

private void setupRecycler() {

    LinearLayoutManager layoutManager = new LinearLayoutManager(this);
    mRecyclerView.setLayoutManager(layoutManager);

    setupList();

    mAdapter = new NROptionLineAdapter(data, this, getApplication());
    mRecyclerView.setAdapter(mAdapter);

}

private void setupList(){
    data = new ArrayList<>();

    class setupList extends AsyncTask<Void, Void, List<MRNrOption>> {
        @Override
        protected void onPreExecute() {
            super.onPreExecute();
        }

        @Override
        protected List<MRNrOption> doInBackground(Void... voids) {

            list = DatabaseClient
                    .getInstance(getApplicationContext())
                    .getAppDatabase()
                    .mrNrOptionDAO()
                    .loadAllByNRId(nr);
            return list;

        }

        @Override
        protected void onPostExecute(List<MRNrOption> list) {
            super.onPostExecute(list);

            List<String> priceList = new ArrayList<>();
            priceList.add("Sim");
            priceList.add("Não");
            priceList.add("Não se Aplica");

            for (int i=0; i<list.size(); i++) {
                Log.d(TAG, "NRs Activity - Adding To List: " + list.get(i).getTitle());
                data.add(new Checklist(
                        list.get(i).getTitle(),
                        priceList)
                );

                mAdapter.notifyDataSetChanged();
            }

        }
    }

    setupList lm = new setupList();
    lm.execute();
}

RecyclerView with RadioButtons

编辑3-重要

RadioGroups和RadioButtons是以编程方式生成的,因为我从服务器获取了所有问题,而问题的数量则取决于用户先前所做的选择,这就是为什么我需要这种方式。

编辑4

GIF to enhance the problem visualization

编辑5-清单模型类

    public class Checklist {

    String title;
    List<String> questions;
    boolean isRadioButtonAdded;
    int selectedId;

    public Checklist(String title, List<String> questions) {
        this.title = title;
        this.questions = questions;
    }

    public Checklist(){}

    public boolean getIsAdded(){
        return isRadioButtonAdded;
    }

    public void setIsAdded(boolean isAdded){
        this.isRadioButtonAdded = isAdded;
    }

    public String getTitle() {
        return title;
    }

    public void setTitle(String title) {
        this.title = title;
    }

    public List<String> getQuestions() {
        return questions;
    }

    public void setQuestions(List<String> questions) {
        this.questions = questions;
    }

    public int getSelectedId() {
        return selectedId;
    }

    public void setSelectedId(int selectedId) {
        this.selectedId = selectedId;
    }
}

7 个答案:

答案 0 :(得分:0)

在发生onClick事件时,将单选按钮的选中/未选中状态保存到模型中(即,列表中的项目应具有此字段)。绑定ViewHolder时,请确保将复选框的值设置为模型中保存的值。

答案 1 :(得分:0)

我看到checklistModel是在onBindViewHolder方法之外声明的:

checklistModel = mChecklists.get(position);

因此,当您滚动时,类变量checklistModel正在使用触发onBindViewHolder的最后一个位置的清单模型进行更新。

因此,当您点击其中一个单选按钮时,无论处于哪个位置,它都会更新触发onBindViewHolder的最后一个位置的清单模型。

您可以通过许多不同的方法来解决此问题,一种方法是在checklistModel内将onBindViewHolder声明为final:

final Checklist checklistModel = mChecklists.get(position);

答案 2 :(得分:0)

唯一的可行解决方案是将RecyclerView设置为在ViewHolder中不可回收。

this.setIsRecyclable(false);

答案 3 :(得分:0)

尽管我不确定这是否能解决您的问题,但是作为一种优化做法,您也应该将监听器附加到onCreateViewHolder而不是onBindViewHolder上,这样可以防止多个为侦听器创建对象。

为什么不在onCreateViewHolder

中移动此代码?

您拥有的视图支架内的该块:

priceGroup.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
        @Override
        public void onCheckedChanged(RadioGroup group, int checkedId) {
            checklistModel.setSelectedId(checkedId);
            Log.d(TAG, "onCheckedChanged: " + checkedId);
        }
    });

答案 4 :(得分:0)

尝试将您的setOnCheckedChangeListener代码移至ViewHolder并在此处更新mCheckList


priceGroup.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
    @Override
    public void onCheckedChanged(RadioGroup radioGroup, int checkedId) {
        mCheckList.get(getAdapterPosition()).setSelectedId(checkedId);
        Log.d(TAG, "onCheckedChanged: " + radioGroup.getCheckedRadioButtonId() + " " + i);
}});

主要问题是您没有更新正确的项目状态。当您单击单选按钮时,它将仅更新最后调用的项onBindViewHolder,因为checklistModel仅保留最后一个引用。要解决此问题,您始终需要在侦听器内部访问mainList。

答案 5 :(得分:0)

recyclerview的工作方式是,当您向下或向上滚动直到视图不可见时,它将存储状态,但是当您向下滚动或向上滚动recyclerview时,它将破坏该行,并将新行放入该行中position.so整个行的视图将被破坏。

该问题的解决方案是,向正在使用的列表中添加另一个变量,并且当单选按钮更改状态时,将存储数据并继续。 在您的数据模型类中是这样的 然后在模型类中定义recyclerview使用的列表。

Boolean is stateclicked;
int state position;

,在OnBindViewholder中,您可以获取此数据的值。如果该值为null,则不单击该值;如果单击,则将布尔值更改为yes,然后将该状态的值放入整数

答案 6 :(得分:-1)

下面的代码怎么样?

适配器:

由于视图已被回收,我认为唯一的id是一个错误。

int id = (position+1)*100;

int id = 1;