使用游标适配器实现带有过滤器的多选列表视图

Posted

技术标签:

【中文标题】使用游标适配器实现带有过滤器的多选列表视图【英文标题】:Implementing a listview with multiple select with filter using a Cursor Adapter 【发布时间】:2012-03-13 02:01:47 【问题描述】:

这个问题在这个问题android: Wrong item checked when filtering listview 中讨论过。总结一下这个问题,当使用带有 CursorAdapter 和过滤器的列表视图时,在过滤器列表中选择的项目在过滤器被删除后会丢失它们的选择,而是在未过滤列表中该位置的项目被选中。

使用上面链接问题中的代码示例,我们应该将代码放在哪里来标记复选框。我相信它应该在 CustomCursorAdapter 的 getView() 方法中,但我不确定。另外,我们如何访问自定义适配器类中保存所有 selectedIds 的 HashSet,因为它将在保存列表的主活动中进行初始化和修改。

我实现 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://***.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://***.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;


【问题讨论】:

根据这个链接,codereview.stackexchange.com/questions/1057/… 我们最好覆盖 newView 和 bindView。 【参考方案1】:

我做了什么解决了它:

cur = 光标。 c = simplecursoradapter 内部光标。

在我的 MainActivity 中:

(members)
static ArrayList<Boolean> checkedStates = new ArrayList<Boolean>();
static HashSet<String> selectedIds = new HashSet<String>();
static HashSet<Integer> selectedLines = new HashSet<Integer>();

在listView中的onItemClickListener:

if (!selectedIds.contains(bookmarkID)) 

    selectedIds.add(bookmarkID);
    selectedLines.add(position);


 else 

     selectedIds.remove(bookmarkID);
     selectedLines.remove(position);



 if (selectedIds.isEmpty()) 
    //clear everything
        selectedIds.clear();
        checkedStates.clear();      
        selectedLines.clear();

        //refill checkedStates to avoid force close bug - out of bounds
        if (cur.moveToFirst()) 
            while (!cur.isAfterLast())     
                MainActivity.checkedStates.add(false);

                cur.moveToNext();
            
                               

 

在我添加的 SimpleCursorAdapter 中(都在 getView 中):

// fill the checkedStates array with amount of bookmarks (prevent OutOfBounds Force close)
        if (c.moveToFirst()) 
            while (!c.isAfterLast())   
                MainActivity.checkedStates.add(false);
                c.moveToNext();
            
        

和:

String bookmarkID = c.getString(0);
        CheckedTextView markedItem = (CheckedTextView) row.findViewById(R.id.btitle);
        if (MainActivity.selectedIds.contains(new String(bookmarkID))) 
            markedItem.setChecked(true);
            MainActivity.selectedLines.add(pos);

         else 
            markedItem.setChecked(false);
            MainActivity.selectedLines.remove(pos);
        

希望这会有所帮助...您当然需要根据自己的需要对其进行调整。

编辑:

已下载FB SDK,无法通过FB登录。如果设备上安装了 FB 应用程序,您将无法获得有效的 access_token 错误。删除了 FB 应用程序并在getFriends() 中获得了 FC。通过使用runOnUiThread(new Runnable...) 包装其范围来解决它。

您得到的错误与错误的过滤、错误的复选框状态无关......您得到它是因为您在查询之前尝试访问光标(已经关闭)。 您似乎在适配器使用光标之前关闭了它。通过添加验证它:

mDbHelper = new FriendsDbAdapter(context);

        mDbHelper.open() ;
        cursor = mDbHelper.fetchAllFriends();

到 SelectFriendsAdapter 中的 getView 范围。

添加后,它不会 FC 并且您可以开始处理您的过滤器。 确保光标没有关闭,基本上如果您使用startManagingCursor() 管理它,则无需手动关闭它。

希望你能从这里拿走它!

【讨论】:

非常感谢您的帮助。我仍然在 setFilterQueryProvider 中遇到麻烦。使用与 SimpleCursorAdapter 相同的数据库助手给我一个“fillWindow() 错误中的无效语句” 在我的 listactivity 中,我使用了一个 mDbHelper 实例,它是我的 DatabaseHelper。我调用了 c = mDbHelper.fetchAllFriends(),然后在我的 listActivity 的 onCreate 中调用了 startManagingCursor()。我应该为 runQuery() 中的调用声明一个新的 mDbHelper 实例吗?我无法找到解决此问题的首选方法。 @movingahead 我不确定我是否了解如何为您提供帮助以及您当前的问题与您在 OP 帖子中描述的内容有什么关系,但是,在您收到错误之后,这似乎是正确的方法是在 onCreate(new mDbHelper() 或其他)中实例化 mDbHelper,然后调用 fetchFriends。见这里:1.***.com/questions/4195089/… 2.***.com/questions/4258493/… 3.***.com/questions/9049542/… 我已关注您指向的所有链接。我收到一个或另一个光标错误。无论如何,感谢您的帮助。 @movingahead 太糟糕了,它没有解决......我想如果我看到你的代码或至少包含堆栈跟踪作为错误的行的部分,我将能够提供更多帮助. 我的整个代码在github.com/movingahead/GroupBanker/tree/development/src/me/… 这个错误的相关文件是SelectFriends.java、SelectFriendsAdapter.java 和FriendsDbAdapter.java 错误的确切行是github.com/movingahead/GroupBanker/commit/… 如果可以的话那就太好了看看我是否犯了一些明显的错误。【参考方案2】:

adapter.setFilterQueryProvider(new FilterQueryProvider() 
            public Cursor runQuery(CharSequence constraint) 

                return db.fetchdatabyfilter(constraint.toString(), "name");

                
        );

【讨论】:

以上是关于使用游标适配器实现带有过滤器的多选列表视图的主要内容,如果未能解决你的问题,请参考以下文章

列表视图中的多选

创建过滤数据表的多选下拉列表

NetSuite Advanced PDF - 使用记录中的多选字段过滤项目表列表

如何使用自定义列表视图在edittext过滤器中获取空间?可以使用简单的适配器吗?

vue多选,数据量大怎么

使用带复选框的多选下拉菜单搜索或过滤 jquery 数据表中的列