刷新列表视图中的复选框状态

时间:2011-01-17 22:44:49

标签: android listview checkbox

我有自定义行布局(list_row.xml)的列表视图,其中包含CheckBox元素:

<TableLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_height="wrap_content"
    android:layout_width="fill_parent"
    android:orientation="horizontal"
    android:weightSum="1.0">
    <TableRow>
        <TextView android:id="@+id/name"
            android:layout_weight=".85"
            android:layout_width="0dip"
            android:layout_height="wrap_content"
            android:text="sample"
            android:textSize="24sp">
        </TextView>
        <CheckBox android:id="@+id/enabled"
            android:layout_weight=".15"
            android:layout_width="0dip"
            android:layout_height="wrap_content" 
            android:focusable="false">
        </CheckBox>
    </TableRow>
</TableLayout>

在我的列表活动中,我有ListRowAdapter:

private static class ListRowAdapter extends ArrayAdapter<ListRow> implements
        CompoundButton.OnCheckedChangeListener {

    private List<ListRow> items;
    private LayoutInflater vi;
    private SparseBooleanArray mCheckStates;

    public ListRowAdapter(Context context, int textViewResourceId,
            List<ListRow> objects) {
        super(context, textViewResourceId, objects);
        this.items = objects;
        this.vi = LayoutInflater.from(context);
        this.mCheckStates = new SparseBooleanArray(objects.size());
        for(int i=0;i<mCheckStates.size();i++){
            mCheckStates.put(i,items.get(i).isEnabled());
        }
        Log.i("VIEW", "constructor");
    }

    public boolean isChecked(int position) {
        return mCheckStates.get(position, false);
    }

    public void setChecked(int position, boolean isChecked) {
        mCheckStates.put(position, isChecked);
        items.get(position).setEnabled(isChecked);
        notifyDataSetChanged();
    }

    public void toggle(int position) {
        setChecked(position, !isChecked(position));
    }

    public void onCheckedChanged(CompoundButton buttonView,
            boolean isChecked) {
        ListRow r = (ListRow) buttonView.getTag();
        r.setEnabled(isChecked);
        mCheckStates.put(items.indexOf(r), isChecked);
        Toast.makeText(getContext(), "row " + r.getName() + " changed to "+isChecked, Toast.LENGTH_SHORT).show();
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder;
        if (convertView == null) {
            Log.i("VIEW", "init holder");
            convertView = vi.inflate(R.layout.list_row, null);
            holder = new ViewHolder();
            holder.name = (TextView) convertView.findViewById(R.id.name);
            holder.enabled = (CheckBox) convertView
                    .findViewById(R.id.enabled);
            convertView.setTag(holder);
        } else {
            Log.i("VIEW", "using holder");
            holder = (ViewHolder) convertView.getTag();
        }
        ListRow row = items.get(position);
        if (row != null) {
            holder.name.setText(row.getName());
            holder.enabled.setTag(row);
            holder.enabled.setChecked(mCheckStates.get(position, false));           holder.enabled.setOnCheckedChangeListener(this);
        }
        return convertView;
    }

    static class ViewHolder {
        TextView name;
        CheckBox enabled;
    }

}

在我的Activity(扩展ListActivity)中,我执行以下操作:

private static List<ListRow> rows;
private ListRowAdapter lra;

......然后

getListView().setItemsCanFocus(false);
getListView().setTextFilterEnabled(true);
rows = initRows();
lra = new ListRowAdapter(this, R.layout.list_row, rows);
setListAdapter(lra);

滚动时工作正常。但如果我不使用mCheckStates数组,它就会失败!我只是无法理解为什么。我有items数组作为我的Activity类的成员,它包含复选框的所有状态!我使用onCheckedChanged()方法更新了它,例如我更新mCheckState。所以下面的代码必须工作:

public boolean isChecked(int position) {
    return items.get(position).isEnebled();
}

public void setChecked(int position, boolean isChecked) {
    items.get(position).setEnabled(isChecked);
    notifyDataSetChanged();
}

public void toggle(int position) {
    setChecked(position, !isChecked(position));
}

public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
    ListRow r = (ListRow) buttonView.getTag();
    r.setEnabled(isChecked);
    Toast.makeText(getContext(), "row " + r.getName() + " changed to "+isChecked, Toast.LENGTH_SHORT).show();
}

并在getView()方法中:

holder.enabled.setTag(row);holder.enabled.setChecked(row.isEnabled());

但事实并非如此。它在滚动时丢失了复选框状态,我不知道为什么。

2 个答案:

答案 0 :(得分:1)

我不确定我是否按照你在那里说过的所有内容,但请记住ListView在滚动时回收视图。这意味着您不能假设视图在getView开头处于任何默认状态。这可能是每次获得新视图时停止将复选框重置为正确状态时无效的原因。

那就是说,您是否考虑使用ListView内置的复选框工具?将其选择模式设置为CHOICE_MODE_MULTIPLE并使用其提供的布局之一,例如android.R.layout.simple_list_item_multiple_choice

答案 1 :(得分:1)

我发现因为getView正在重新绘制并回滚滚动列表项,所以需要在选中复选框时明确说明,何时不是(所选项目背景颜色的类似解决方案)在< / p>

if(condition){... set check box checked code ...} else {... set check box unchecked code ...}

复选框状态可以从一个数组派生,每次用户选中或取消选中列表视图中的框时都必须更新该数组。