Андроид:ListView из CheckedTextView + SQLite, отображать проверенные строки с помощью адаптера?
-
12-11-2019 - |
Вопрос
Я пытаюсь реализовать очень простой список "задач".Я решил использовать ListView из CheckedTextView (с идентификатором 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);
Это работает нормально, он получает все элементы в базе данных и для каждого элемента создает CheckedTextView с as text DbAdapter.NAME.
Но что мне действительно нужно, так это получить как текст, так и проверенное состояние и соответствующим образом проверить строки.
Допустим, я могу получить состояние (как целое число 0/1) из базы данных, используя DBAdapter.проверен..как я могу реализовать адаптер курсора, который выполняет эту работу?
Я искал везде (например этот вопрос что является беспорядком), но не нашел ничего, что могло бы решить эту проблему.Вах.Сначала это казалось таким простым.
Спасибо,
Стефано
@Берт Ф:
Спасибо, это метод, который я использую для заполнения своего списка:
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 пункт: я
РЕДАКТИРОВАТЬ:уродливое решение
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
Ваше здоровье.
Решение
Внедрить SimpleCursorAdapter.Привязка к просмотру.
Иметь возможность ViewBinder.setViewValue() метод проверяет, когда он вызывается для
DbAdapter.NAME
столбец или же попросите его проверить наличиеR.id.checked_text
-вид типа.Когда он звонил, но не для нет целевой столбец/представление, пусть он вернет
false
так что адаптер выполнит привязку столбца/ представления так, как это было бы обычно.Когда он вызывал целевой столбец /представление, попросите его установить текст и флажок для
(CheckedTextView) view
.В конечном итоге он должен вернутьсяtrue
таким образом, этот адаптер не будет продолжать пытаться привязать само представление.Обратите внимание, что курсор доступен для доступа к данным запроса, чтобы определить, устанавливать флажок или нет (DbAdapter.CHECKED
).Установите свой
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;
}
};