Реализация списка с множественным выбором с фильтром с использованием адаптера курсора

Эта проблема обсуждается в этом вопросе Android: при фильтрации списка просмотра отмечен неправильный элемент. Подводя итог проблеме, при использовании представления списка с CursorAdapter и фильтром элементы, выбранные в отфильтрованном списке, теряют свой выбор после удаления фильтра, и вместо этого выбираются элементы в этой позиции в нефильтрованном списке.

Используя образец кода в связанном вопросе выше, где мы должны поместить код, чтобы отметить флажки. Я считаю, что это должно быть в методе getView () CustomCursorAdapter, но я не уверен. Кроме того, как получить доступ к 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;

}

}


person rohitmishra    schedule 22.02.2012    source источник
comment
Согласно этой ссылке, codereview.stackexchange.com/questions/1057 / нам лучше переопределить newView и bindView.   -  person rohitmishra    schedule 22.02.2012


Ответы (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. у вас есть ошибка, из-за которой вы не получаете действительный access_token, если на устройстве установлено приложение FB. Удалил приложение FB и получил FC в getFriends(). Решил, обернув его область действия runOnUiThread(new Runnable...).

Ошибка, которую вы получаете, не имеет ничего общего с плохой фильтрацией, плохими состояниями флажка ... Вы получаете ее, потому что пытаетесь получить доступ к курсору перед его запросом (уже закрытым). Похоже, вы закрываете курсор до того, как адаптер его использует. Проверено добавлением:

mDbHelper = new FriendsDbAdapter(context);

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

в область getView в SelectFriendsAdapter.

После того, как вы добавите это, он не будет FC, и вы можете начать заботиться о своем фильтре. Убедитесь, что курсор не закрыт, и в основном, если вы управляете им с помощью startManagingCursor(), нет необходимости закрывать его вручную.

Надеюсь, ты сможешь взять это отсюда!

person Lior Iluz    schedule 22.02.2012
comment
Спасибо большое за вашу помощь. У меня все еще возникают проблемы с setFilterQueryProvider. Использование того же помощника базы данных, что и для SimpleCursorAdapter, дает мне ошибку Invalid Statement в fillWindow (). В моем списке действий я использовал экземпляр mDbHelper, который является моим DatabaseHelper. Я вызываю c = mDbHelper.fetchAllFriends (), за которым следует startManagingCursor () в onCreate моего listActivity. Должен ли я объявить новый экземпляр mDbHelper для вызова в runQuery ()? Я не могу найти лучший способ справиться с этим. - person rohitmishra; 24.02.2012
comment
@movingahead Я не уверен, что понимаю, как вам помочь и как ваш текущий вопрос связан с тем, что вы описали в сообщении OP, НО, после полученной ошибки, кажется, что правильный способ - создать экземпляр mDbHelper в onCreate (новый mDbHelper () или что-то еще), а затем вызовите метод fetchFriends. См. Здесь: 1. stackoverflow.com/questions/4195089/ 2. stackoverflow.com/questions/4258493/ 3. stackoverflow.com/questions/9049542/ - person Lior Iluz; 25.02.2012
comment
Я перешел по всем ссылкам, на которые вы указали. Я получаю ту или иную ошибку курсора. В любом случае, спасибо за помощь. - person rohitmishra; 25.02.2012
comment
@movingahead очень плохо, что это не решено ... Думаю, я мог бы больше помочь, если бы увидел ваш код или, по крайней мере, ту часть, которая включает строку, в которой трассировка стека указывает на ошибку. - person Lior Iluz; 25.02.2012
comment
Весь мой код находится на github.com/movingahead/GroupBanker/tree / development / src / me / Соответствующие файлы для этой ошибки: SelectFriends.java, SelectFriendsAdapter.java и FriendsDbAdapter.java. Точная строка для ошибки: github.com/movingahead/GroupBanker/commit/ Будет здорово, если вы увидите, если я делаю очевидные вещи. - person rohitmishra; 25.02.2012
comment
@movingahead См. обновление в моем ответе (нижняя часть после редактирования). Надеюсь, это приведет вас к правильному решению. - person Lior Iluz; 25.02.2012
comment
Большое спасибо. Я попробую это реализовать. - person rohitmishra; 25.02.2012