在此問題中討論此問題Android: Wrong item checked when filtering listview。爲了總結這個問題,當使用具有CursorAdapter和過濾器的列表視圖時,在過濾器被移除之後,在過濾列表上選擇的項目將失去它們的選擇,相反,未過濾列表中該位置上的項目將被選中。使用光標適配器實現帶有多個使用篩選器的選擇列表視圖
使用上面的鏈接的問題,我們應該在哪裏把代碼標記的複選框的代碼示例。我相信它應該在CustomCursorAdapter的getView()方法中,但我不確定。另外,我們如何訪問HashSet來保存自定義適配器類中的所有selectedIds,因爲它將在保存列表的主活動中進行初始化和修改。
我的活動實現的ListView
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.selectfriends);
Log.v(TAG, "onCreate called") ;
selectedIds = new ArrayList<String>() ;
selectedLines = new ArrayList<Integer>() ;
mDbHelper = new FriendsDbAdapter(this);
mDbHelper.open() ;
Log.v(TAG, "database opened") ;
Cursor c = mDbHelper.fetchAllFriends();
startManagingCursor(c);
Log.v(TAG, "fetchAllFriends Over") ;
String[] from = new String[] {mDbHelper.KEY_NAME};
int[] to = new int[] { R.id.text1 };
final ListView listView = getListView();
Log.d(TAG, "Got listView");
// Now initialize the adapter and set it to display using our row
adapter =
new FriendsSimpleCursorAdapter(this, R.layout.selectfriendsrow, c, from, to);
Log.d(TAG, "we have got an adapter");
// Initialize the filter-text box
//Code adapted from https://stackoverflow.com/questions/1737009/how-to-make-a-nice-looking-listview-filter-on-android
filterText = (EditText) findViewById(R.id.filtertext) ;
filterText.addTextChangedListener(filterTextWatcher) ;
/* Set the FilterQueryProvider, to run queries for choices
* that match the specified input.
* Code adapted from https://stackoverflow.com/questions/2002607/android-how-to-text-filter-a-listview-based-on-a-simplecursoradapter
*/
adapter.setFilterQueryProvider(new FilterQueryProvider() {
public Cursor runQuery(CharSequence constraint) {
// Search for friends whose names begin with the specified letters.
Log.v(TAG, "runQuery Constraint = " + constraint) ;
String selection = mDbHelper.KEY_NAME + " LIKE '%"+constraint+"%'";
mDbHelper.open();
Cursor c = mDbHelper.fetchFriendsWithSelection(
(constraint != null ? constraint.toString() : null));
return c;
}
});
setListAdapter(adapter);
Log.d(TAG, "setListAdapter worked") ;
listView.setItemsCanFocus(false);
listView.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE);
// listView.setOnItemClickListener(mListener);
Button btn;
btn = (Button)findViewById(R.id.buttondone);
mDbHelper.close();
}
@Override
protected void onListItemClick(ListView parent, View v, int position, long id) {
String item = (String) getListAdapter().getItem(position);
Toast.makeText(this, item + " selected", Toast.LENGTH_LONG).show();
//gets the Bookmark ID of selected position
Cursor cursor = (Cursor)parent.getItemAtPosition(position);
String bookmarkID = cursor.getString(0);
Log.d(TAG, "mListener -> bookmarkID = " + bookmarkID);
Log.d(TAG, "mListener -> position = " + position);
// boolean currentlyChecked = checkedStates.get(position);
// checkedStates.set(position, !currentlyChecked);
if (!selectedIds.contains(bookmarkID)) {
selectedIds.add(bookmarkID);
selectedLines.add(position);
} else {
selectedIds.remove(bookmarkID);
selectedLines.remove(position);
}
}
private TextWatcher filterTextWatcher = new TextWatcher() {
public void afterTextChanged(Editable s) {
}
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
public void onTextChanged(CharSequence s, int start, int before, int count) {
Log.v(TAG, "onTextChanged called. s = " + s);
adapter.getFilter().filter(s);
}
};
@Override
protected void onDestroy() {
super.onDestroy();
filterText.removeTextChangedListener(filterTextWatcher);
}
我的自定義光標適配器:
public class FriendsSimpleCursorAdapter extends SimpleCursorAdapter implements Filterable {
private static final String TAG = "FriendsSimpleCursorAdapter";
private final Context context ;
private final String[] values ;
private final int layout ;
private final Cursor cursor ;
static class ViewHolder {
public CheckedTextView checkedText ;
}
public FriendsSimpleCursorAdapter(Context context, int layout, Cursor c,
String[] from, int[] to) {
super(context, layout, c, from, to);
this.context = context ;
this.values = from ;
this.layout = layout ;
this.cursor = c ;
Log.d(TAG, "At the end of the constructor") ;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
Log.d(TAG, "At the start of rowView. position = " + position) ;
View rowView = convertView ;
if(rowView == null) {
Log.d(TAG, "rowView = null");
try {
LayoutInflater inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
rowView = inflater.inflate(layout, parent, false);
Log.d(TAG, "rowView inflated. rowView = " + rowView);
ViewHolder viewHolder = new ViewHolder() ;
viewHolder.checkedText = (CheckedTextView) rowView.findViewById(R.id.text1) ;
rowView.setTag(viewHolder);
}
catch (Exception e) {
Log.e(TAG, "exception = " + e);
}
}
ViewHolder holder = (ViewHolder) rowView.getTag();
int nameCol = cursor.getColumnIndex(FriendsDbAdapter.KEY_NAME) ;
String name = cursor.getString(nameCol);
holder.checkedText.setText(name);
Log.d(TAG, "At the end of rowView");
return rowView;
}
}
根據這個鏈接,http://codereview.stackexchange.com/questions/1057/android-custom-cursoradapter-design我們最好重寫newView和bindView。 – rohitmishra 2012-02-22 19:11:42