从 recyclerview 项目中保存模型 class 中选定的 RadioButton 文本
Saving selected RadioButton's text in model class from recyclerview item
从 RecyclerView
项开始,我正在尝试将 selected RadioButton 的文本保存在模型 class 中。当我从第一项 select a RadioButton
时,它的文本被正确保存。问题是,RadioButton 在第 8 项相同位置的文本也会自动保存。如果我 select 第 2 项的单选按钮,第 9 项的文本也会自动保存,依此类推。如何解决这个问题呢?
onBindViewHolder 方法如下:
@Override
public void onBindViewHolder(@NonNull final ViewHolder holder, final int position) {
...
holder.radioGroup.setTag(position);
holder.radioGroup.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
@Override
public void onCheckedChanged(RadioGroup group, int checkedId) {
int radioButtonID = group.getCheckedRadioButtonId();
RadioButton radioButton = (RadioButton) group.findViewById(radioButtonID);
int clickedPos = (Integer) group.getTag();
models.get(clickedPos).setChecked(radioButtonID);
if (radioButtonID > 0){
models.get(clickedPos).setSelectedAns(radioButton.getText().toString());
}
}
});
holder.radioGroup.check(models.get(position).getChecked());
Log.d("TAG", "At position " + position + " selected : " + models.get(position).getSelectedAns());
}
问题在于,尽管代码表面上看起来是正确的,但实际上发生的是当您调用 holder.radioGroup.check()
时,它会触发您的 onCheckedChanged()
事件处理程序,就像用户调用发起它。
由于视图被回收,位置 0 的视图将被重新用于列表中的位置 8。因此,在 onBindViewHolder()
中对 check()
的调用将调用 onCheckedChanged()
,位置 0 中选中的单选按钮仍处于选中状态(即 checkedId
和 radioGroup. getCheckedRadioButtonId()
将 return 在位置 0 使用视图时检查的单选按钮的 ID)。
真正的症结在于
models.get(clickedPos).setChecked(radioButtonID);
考虑答案的第一段,您会发现这将(错误地)更新位置 8 处的模型项,其中 radioButtonID
是在位置 0 处使用此视图时选中的。
解决此问题的一种方法是区分用户启动的更改和绑定启动的更改。例如,您可以通过向 ViewHolder
添加一个字段来指示视图当前是否绑定。
class ViewHolder extends RecyclerView.ViewHolder{
TextView selectedAnswer;
RadioGroup radioGroup;
boolean isBinding;
ViewHolder(View itemView) {
super(itemView);
radioGroup = itemView.findViewById(R.id.radioGroup);
radioGroup.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
@Override
public void onCheckedChanged(RadioGroup group, int checkedId) {
int position = getAdapterPosition();
RadioButton radioButton = (RadioButton) group.findViewById(checkedId);
/* Only update the model when onCheckedChange() was initiated by the user clicking
a radio button, not when the adapter is binding the view. In that scenario, we
are only interested in passing information FROM the model TO the view. */
if( !isBinding ) {
models.get(position).setChecked(checkedId);
models.get(position).setSelectedAns(radioButton != null ? radioButton.getText().toString() : "");
}
selectedAnswer.setText( models.get(position).getSelectedAns() );
}
});
...
}
}
@Override
public void onBindViewHolder(@NonNull final ViewHolder holder, final int position) {
holder.isBinding = true;
...
/* When calling check() here, we invoke onCheckedChanged(), which will
update the textview that displays the selected answer - so no need to call
holder.selectedAnswer.setText( models.get(position).getSelectedAns() )
from here */
holder.radioGroup.check(models.get(position).getChecked());
holder.isBinding = false;
}
从 RecyclerView
项开始,我正在尝试将 selected RadioButton 的文本保存在模型 class 中。当我从第一项 select a RadioButton
时,它的文本被正确保存。问题是,RadioButton 在第 8 项相同位置的文本也会自动保存。如果我 select 第 2 项的单选按钮,第 9 项的文本也会自动保存,依此类推。如何解决这个问题呢?
onBindViewHolder 方法如下:
@Override
public void onBindViewHolder(@NonNull final ViewHolder holder, final int position) {
...
holder.radioGroup.setTag(position);
holder.radioGroup.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
@Override
public void onCheckedChanged(RadioGroup group, int checkedId) {
int radioButtonID = group.getCheckedRadioButtonId();
RadioButton radioButton = (RadioButton) group.findViewById(radioButtonID);
int clickedPos = (Integer) group.getTag();
models.get(clickedPos).setChecked(radioButtonID);
if (radioButtonID > 0){
models.get(clickedPos).setSelectedAns(radioButton.getText().toString());
}
}
});
holder.radioGroup.check(models.get(position).getChecked());
Log.d("TAG", "At position " + position + " selected : " + models.get(position).getSelectedAns());
}
问题在于,尽管代码表面上看起来是正确的,但实际上发生的是当您调用 holder.radioGroup.check()
时,它会触发您的 onCheckedChanged()
事件处理程序,就像用户调用发起它。
由于视图被回收,位置 0 的视图将被重新用于列表中的位置 8。因此,在 onBindViewHolder()
中对 check()
的调用将调用 onCheckedChanged()
,位置 0 中选中的单选按钮仍处于选中状态(即 checkedId
和 radioGroup. getCheckedRadioButtonId()
将 return 在位置 0 使用视图时检查的单选按钮的 ID)。
真正的症结在于
models.get(clickedPos).setChecked(radioButtonID);
考虑答案的第一段,您会发现这将(错误地)更新位置 8 处的模型项,其中 radioButtonID
是在位置 0 处使用此视图时选中的。
解决此问题的一种方法是区分用户启动的更改和绑定启动的更改。例如,您可以通过向 ViewHolder
添加一个字段来指示视图当前是否绑定。
class ViewHolder extends RecyclerView.ViewHolder{
TextView selectedAnswer;
RadioGroup radioGroup;
boolean isBinding;
ViewHolder(View itemView) {
super(itemView);
radioGroup = itemView.findViewById(R.id.radioGroup);
radioGroup.setOnCheckedChangeListener(new RadioGroup.OnCheckedChangeListener() {
@Override
public void onCheckedChanged(RadioGroup group, int checkedId) {
int position = getAdapterPosition();
RadioButton radioButton = (RadioButton) group.findViewById(checkedId);
/* Only update the model when onCheckedChange() was initiated by the user clicking
a radio button, not when the adapter is binding the view. In that scenario, we
are only interested in passing information FROM the model TO the view. */
if( !isBinding ) {
models.get(position).setChecked(checkedId);
models.get(position).setSelectedAns(radioButton != null ? radioButton.getText().toString() : "");
}
selectedAnswer.setText( models.get(position).getSelectedAns() );
}
});
...
}
}
@Override
public void onBindViewHolder(@NonNull final ViewHolder holder, final int position) {
holder.isBinding = true;
...
/* When calling check() here, we invoke onCheckedChanged(), which will
update the textview that displays the selected answer - so no need to call
holder.selectedAnswer.setText( models.get(position).getSelectedAns() )
from here */
holder.radioGroup.check(models.get(position).getChecked());
holder.isBinding = false;
}