Android:CheckedTextView + SQLite的ListView,用Adapter显示选中的行?

时间:2011-02-11 23:00:59

标签: android sqlite listview cursor checkedtextview

我正在尝试实现一个非常简单的“待办事项”列表。 我选择使用CheckedTextView的ListView(带有id checked_text),ListView当然使用CHOICE_MODE_MULTIPLE。

现在,我正在使用SQLite数据库来存储列表项。

要填写列表,我有类似的内容:

cursor = dbHelper.fetchAll();
startManagingCursor(cursor);

String[] from = new String[] {  DbAdapter.NAME };
int[] to = new int[] { R.id.checked_text };

SimpleCursorAdapter notes = new SimpleCursorAdapter(this,
R.layout.todo_multi_row, cursor, from, to);
listView.setAdapter(notes);

这工作正常,它获取数据库中的所有项目以及每个项目 item它创建一个CheckedTextView,文本为DbAdapter.NAME。

但我真正需要的是同时获取文本和检查状态, 并相应地检查行。

假设我可以使用DB从状态(整数0/1)获取 DbAdapter.CHECKED ..如何实现获取的游标适配器 干完了吗?

我到处搜索(例如this question这是一团糟),但没有找到解决这个问题的方法。华。 起初看起来很容易。

谢谢,
斯特凡诺


@Bert F:
谢谢,这是我用来填充我的列表的方法:

private void fillData() {
        Log.d("LIST","fill");
        cursor = dbHelper.fetchAllTodos();
        startManagingCursor(cursor);

        String[] from = new String[] { TodoDbAdapter.NAME };
        int[] to = new int[] { R.id.checked_text };

        SimpleCursorAdapter notes = new SimpleCursorAdapter(this,
                R.layout.todo_multi_row, cursor, from, to);

        final SimpleCursorAdapter.ViewBinder mViewBinder =
            new SimpleCursorAdapter.ViewBinder() {
                @Override
                public boolean setViewValue(
                        final View view,
                        final Cursor cursor,
                        final int columnIndex) {

                    Log.d("LIST",view +" "+cursor+" "+columnIndex);

                    CheckedTextView item = (CheckedTextView) view;
                    Log.d("LIST","NAME: "+item.getText()+" State: "+item.isChecked());

                    return false;
                }
            };

        notes.setViewBinder(mViewBinder);
        listView.setAdapter(notes);
    }

现在,作为测试,我只是打印出正在发生的事情 如果我在列表中添加1个元素,我看到的是:

02-12 14:45:35.913: DEBUG/LIST(22621): fill
02-12 14:45:35.933: DEBUG/LIST(22621): android.widget.CheckedTextView@479da398 android.database.sqlite.SQLiteCursor@479d98f0 2
02-12 14:45:35.933: DEBUG/LIST(22621): NAME:  State: false
02-12 14:45:35.933: DEBUG/LIST(22621): android.widget.CheckedTextView@479da398 android.database.sqlite.SQLiteCursor@479d98f0 2
02-12 14:45:35.943: DEBUG/LIST(22621): NAME: first item State: false
02-12 14:45:36.013: DEBUG/LIST(22621): android.widget.CheckedTextView@479dc6c0 android.database.sqlite.SQLiteCursor@479d98f0 2
02-12 14:45:36.013: DEBUG/LIST(22621): NAME:  State: false
02-12 14:45:36.223: DEBUG/LIST(22621): android.widget.CheckedTextView@479dc6c0 android.database.sqlite.SQLiteCursor@479d98f0 2
02-12 14:45:36.223: DEBUG/LIST(22621): NAME: first item State: false

如果我检查它..

02-12 14:53:33.123: DEBUG/LIST(22621): android.widget.CheckedTextView@479dc6c0 android.database.sqlite.SQLiteCursor@479d98f0 2
02-12 14:53:33.123: DEBUG/LIST(22621): NAME: first item State: false
02-12 14:53:33.123: DEBUG/LIST(22621): android.widget.CheckedTextView@479da398 android.database.sqlite.SQLiteCursor@479d98f0 2
02-12 14:53:33.123: DEBUG/LIST(22621): NAME: first item State: false

撇开它说它没有被检查的事实,为什么它被解雇了这么多次?我只有一个项目:我


编辑:丑陋的解决方案

private void fillData() {
        Log.d("LIST","fill");
        cursor = dbHelper.fetchAllTodos();
        startManagingCursor(cursor);

        String[] from = new String[] { TodoDbAdapter.NAME };
        int[] to = new int[] { R.id.checked_text };

        SimpleCursorAdapter notes = new SimpleCursorAdapter(this,
                R.layout.todo_multi_row, cursor, from, to);

        final SimpleCursorAdapter.ViewBinder mViewBinder =
            new SimpleCursorAdapter.ViewBinder() {
                @Override
                public boolean setViewValue(
                        final View view,
                        final Cursor cursor,
                        final int columnIndex) {

                     final int checkedIndex =
                         cursor.getColumnIndexOrThrow(
                                 TodoDbAdapter.CHECKED);

                     Log.d("LIST","VIEW: "+view+" NAME: "+cursor.getString(columnIndex)+" "+cursor.getInt(checkedIndex));

                    return false;
                }
            };

        notes.setViewBinder(mViewBinder);
        listView.setAdapter(notes);
    }

当我加载列表时,有2个条目(第一个选择第二个没有):

02-12 15:59:48.613: DEBUG/LIST(23533): fill
02-12 15:59:48.643: DEBUG/LIST(23533): VIEW: android.widget.CheckedTextView@47a30ec0 NAME: test 1
02-12 15:59:48.653: DEBUG/LIST(23533): VIEW: android.widget.CheckedTextView@47a30ec0 NAME: LOL 0
02-12 15:59:48.683: DEBUG/LIST(23533): VIEW: android.widget.CheckedTextView@47a30ec0 NAME: test 1
02-12 15:59:48.683: DEBUG/LIST(23533): VIEW: android.widget.CheckedTextView@47a331f8 NAME: LOL 0
02-12 15:59:48.713: DEBUG/LIST(23533): VIEW: android.widget.CheckedTextView@47a35770 NAME: test 1
02-12 15:59:48.713: DEBUG/LIST(23533): VIEW: android.widget.CheckedTextView@47a35770 NAME: LOL 0
02-12 15:59:48.783: DEBUG/LIST(23533): VIEW: android.widget.CheckedTextView@47a35770 NAME: test 1
02-12 15:59:48.783: DEBUG/LIST(23533): VIEW: android.widget.CheckedTextView@47a35770 NAME: LOL 0

干杯。

1 个答案:

答案 0 :(得分:2)

  1. 实施SimpleCursorAdapter.ViewBinder

  2. ViewBinder.setViewValue()方法检查其调用DbAdapter.NAME列的时间,或者让它检查R.id.checked_text - 类型视图。

  3. 当它调用而不是目标列/视图时,让它返回false,以便适配器以正常方式绑定列/视图会。

  4. 当它调用目标列/视图时,让它设置(CheckedTextView) view的文本和复选框。它应该最终返回true,以便适配器不会继续尝试绑定视图本身。请注意,光标可用于访问查询数据以确定是否选中复选框(DbAdapter.CHECKED)。

  5. 通过setViewBinder()

  6. ViewBinder中设置SimpleCursorAdapter

    这是我的一个ViewBinder实现。它不是用于checboxes,而是用于对文本视图进行一些奇特的格式化,但它应该让你对这种方法有所了解:

    private final SimpleCursorAdapter.ViewBinder mViewBinder =
        new SimpleCursorAdapter.ViewBinder() {
            @Override
            public boolean setViewValue(
                    final View view,
                    final Cursor cursor,
                    final int columnIndex) {
                final int latitudeColumnIndex =
                    cursor.getColumnIndexOrThrow(
                            LocationDbAdapter.KEY_LATITUDE);
                final int addressStreet1ColumnIndex =
                    cursor.getColumnIndexOrThrow(
                            LocationDbAdapter.KEY_ADDRESS_STREET1);
    
                if (columnIndex == latitudeColumnIndex) {
    
                    final String text = formatCoordinates(cursor);
                    ((TextView) view).setText(text);
                    return true;
    
                } else if (columnIndex == addressStreet1ColumnIndex) {
    
                    final String text = formatAddress(cursor);
                    ((TextView) view).setText(text);
                    return true;
    
                }
    
                return false;
            }
        };