Android: ListView di CheckedTextView + SQLite, visualizzazione delle righe controllate con adattatore?
-
12-11-2019 - |
Domanda
Sto cercando di implementare una lista "TODO" molto semplice. Ho scelto di utilizzare un listview di CheckedTextView (con ID CHECKED_TEXT), ListView utilizza Choice_Mode_Multiple ovviamente.
Ora, sto usando il database SQLite per memorizzare gli elementi dell'elenco.
Per riempire l'elenco a cui ho qualcosa come:
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);
.
.
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);
Questo funziona bene, ottiene tutti gli elementi nel DB e per ciascuno Articolo Crea una controllataTextVisualizza con come DBadapter di testo.
Ma ciò di cui ho davvero bisogno è ottenere il testo e lo stato controllato, e controllare le righe di conseguenza.
Diciamo che posso ottenere lo stato (come intero 0/1) dal DB usando Dbadapter.Checked .. Come posso implementare un adattatore cursore che ottiene il lavoro fatto?
Ho cercato ovunque (ad esempio Questa domanda Quale è un casino) ma non ha trovato nulla per risolvere questo problema. Wah. Sembrava così facile all'inizio.
.
@ @bert f:
Grazie, questo è il metodo che sto usando per riempire la mia 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);
}
.
Ora, come test, sto solo stampando cosa sta succedendo.
Se aggiungo 1 elemento all'elenco ciò che vedo è:
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 lo controllo ..
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
.
Lasciare da parte il fatto che dice che non è controllato, perché si spara così tante volte? Ho solo 1 articolo: I
.
Modifica: Ugly Solution
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);
}
.
con 2 ingressi (prima Second Second Not) quando carico l'elenco che ho:
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
.
applausi.
Soluzione
- .
-
Implementa un SimpleCursollow"> SimpleCursiraDapter.ViewBinder .
-
Avere il viewbinder.setviewValue () Metodo Controllo per quando è chiamato per la colonna
DbAdapter.NAME
o altrimenti averlo verificato per la visualizzazione del tipoR.id.checked_text
. -
Quando si chiamava, ma non per non la colonna di destinazione / vista, restituisce
false
in modo che l'adattatore vincoli la colonna / visualizzazione del modo in cui normalmente. -
Quando ha chiamato la colonna / vista Target, ha impostato il testo e la casella di controllo per il
(CheckedTextView) view
. Dovrebbe finire per restituiretrue
in modo tale che l'adattatore non continuerà a tentare di vincolare la vista stessa. Si noti che il cursore è disponibile per accedere ai dati della query per determinare se controllare la casella di controllo o meno (DbAdapter.CHECKED
). -
Impostare il
ViewBinder
nelSimpleCursorAdapter
tramite setviewbinder ()Ecco una delle mie implementazioni del tuo voto. Non è per Checboxes, piuttosto il suo per fare una formattazione di fantasia di una vista di testo, ma dovrebbe darti un'idea per l'approccio:
.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; } };