Android: ListView CheckedTextView + SQLite, отображать проверенные строки с помощью адаптера? - PullRequest
2 голосов
/ 12 февраля 2011

Я пытаюсь реализовать очень простой список задач.Я решил использовать ListView для CheckedTextView (с идентификатором check_text), конечно, ListView использует CHOICE_MODE_MULTIPLE.

Теперь я использую базу данных SQLite для хранения элементов списка.

Для заполненияlist У меня есть что-то вроде:

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);

Это работает нормально, он получает все элементы в БД и для каждого элемента создает CheckedTextView с текстом DbAdapter.NAME.

Но что мне действительно нужно, так это получить и текст, и проверенное состояние, и соответственно проверить строки.

Допустим, я могу получить состояние (в виде целого числа 0/1) из БД с помощью DbAdapter.CHECKED .. как я могу реализовать адаптер курсора, который выполняет свою работу?

Я искал везде (например, этот вопрос , который является беспорядком), но не нашел ничего, чтобы решить эту проблему.Вах.Поначалу это казалось таким простым.

Спасибо,
Стефано


@ Берт 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

Оставим в стороне тот факт, что он говорит, что он не проверен, почему он стреляет так много раз?У меня просто есть 1 элемент: I


РЕДАКТИРОВАТЬ: некрасивое решение

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 Ответ

2 голосов
/ 12 февраля 2011
  1. Реализация SimpleCursorAdapter.ViewBinder .

  2. Имейте метод ViewBinder.setViewValue () , чтобы проверить, когда он вызывается для столбца DbAdapter.NAME, или же проверьте его для представления R.id.checked_text -типа. *

  3. Когда он вызывается, но не для , а не целевого столбца / представления, он должен возвращать false, чтобы адаптер связывал столбец / просматривал, как обычно.

  4. Когда он вызывал целевой столбец / представление, он должен установить текст и флажок для (CheckedTextView) view. В конечном итоге он должен вернуть true, чтобы адаптер не пытался связывать само представление. Обратите внимание, что курсор доступен для доступа к данным запроса, чтобы определить, стоит ли устанавливать флажок или нет (DbAdapter.CHECKED).

  5. Установите ViewBinder в SimpleCursorAdapter с помощью setViewBinder ()

Вот одна из моих реализаций ViewBinder. Это не для чекбоксов, а для того, чтобы сделать какое-то необычное форматирование текстового представления, но это должно дать вам некоторое представление о подходе:

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;
        }
    };
...