Android:ListView de CheckedTextView + SQLite, exibe linhas verificadas com Adapter?
-
12-11-2019 - |
Pergunta
Estou tentando implementar uma lista de tarefas muito simples.Optei por usar um ListView de CheckedTextView (com idchecked_text), o ListView usa CHOICE_MODE_MULTIPLE claro.
Agora estou usando o banco de dados SQLite para armazenar os itens da lista.
Para preencher a lista tenho algo como:
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);
Isso está funcionando bem, ele obtém todos os itens no banco de dados e, para cada item, cria um checkEdTextView com o texto dbadapter.name.
Mas o que eu realmente preciso é obter o texto e o estado verificado e verificar as linhas de acordo.
Digamos que eu possa obter o estado (como inteiro 0/1) do banco de dados usando DbAdapter.CHECKED..como posso implementar um adaptador de cursor que obtém o trabalho feito?
Eu procurei em todos os lugares (por exemplo essa questão o que é uma bagunça), mas não encontrei nada para resolver esse problema.Uau.Parecia tão fácil no início.
Obrigado,
Stefano
@Bert F:
Obrigado, este é o método que estou usando para preencher minha lista:
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);
}
Agora, como teste, estou apenas imprimindo o que está acontecendo.
Se eu adicionar 1 elemento à lista, o que vejo é:
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
E se eu verificar..
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
Deixando de lado o fato de que diz que não está verificado, por que está disparando tantas vezes?Só tenho 1 item:I
EDITAR:solução feia
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);
}
Com 2 entradas (a primeira selecionada, depois não), quando carrego a lista, tenho:
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
Saúde.
Solução
Implementar um SimpleCursorAdapter.ViewBinder.
Tenha o ViewBinder.setViewValue() método verifica quando é chamado para o
DbAdapter.NAME
coluna ou então peça para verificar oR.id.checked_text
-tipo de visualização.Quando ligou, mas não para não a coluna/visualização de destino, faça com que ela retorne
false
para que o adaptador vincule a coluna/visualização como faria normalmente.Quando ele chamou a coluna/visualização de destino, defina o texto e a caixa de seleção para o
(CheckedTextView) view
.Deve acabar voltandotrue
para que o adaptador não continue tentando vincular a visualização em si.Observe que o cursor está disponível para acessar os dados da consulta para determinar se a caixa de seleção deve ser marcada ou não (DbAdapter.CHECKED
).Defina seu
ViewBinder
na tuaSimpleCursorAdapter
através da setViewBinder()
Aqui está uma das minhas implementações do ViewBinder.Não é para caixas de seleção, mas sim para fazer uma formatação sofisticada de uma visualização de texto, mas deve lhe dar uma ideia da abordagem:
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;
}
};