Правильно ли закодирован этот пользовательский CursorAdapter для ListView для Android? - PullRequest
11 голосов
/ 24 февраля 2012

Я никогда не был доволен кодом на моем собственном CursorAdapter, до сегодняшнего дня я решил пересмотреть его и исправить небольшую проблему, которая беспокоила меня долгое время (достаточно интересно, что никто из пользователей моего приложения никогда не сообщал о такомпроблема).

Вот небольшое описание моего вопроса:

Мой пользовательский CursorAdapter переопределяет newView() и bindView() вместо getView(), как большинство примеров, которые я вижу.Я использую шаблон ViewHolder между этими двумя методами.Но моя основная проблема заключалась в том, что пользовательский макет, который я использую для каждого элемента списка, содержит ToggleButton.

Проблема заключалась в том, что состояние кнопки не сохранялось, когда представление элемента списка прокручивалось вне поля зрения.а затем прокрутили обратно в поле зрения.Эта проблема существовала, потому что cursor никогда не знал, что данные базы данных изменились при нажатии ToggleButton, и он всегда извлекал одни и те же данные.Я пытался запросить курсор при нажатии ToggleButton, и это решило проблему, но это было очень медленно.

Я решил эту проблему, и я публикую весь класс здесь для обзора.Я подробно прокомментировал код для этого конкретного вопроса, чтобы лучше объяснить мои решения по кодированию.

Этот код вам подходит?Вы бы улучшили / оптимизировали или изменили бы его как-нибудь?

PS: я знаю, что CursorLoader - очевидное улучшение, но у меня нет времени, чтобы заниматься такими большими переписываниями кода в настоящее время.Это то, что у меня есть в дорожной карте.

Вот код:

public class NotesListAdapter extends CursorAdapter implements OnClickListener {

    private static class ViewHolder {
        ImageView icon;
        TextView title;
        TextView description;
        ToggleButton visibility;
    }

    private static class NoteData {
        long id;
        int iconId;
        String title;
        String description;
        int position;
    }

    private LayoutInflater mInflater;

    private NotificationHelper mNotificationHelper;
    private AgendaNotesAdapter mAgendaAdapter;

    /*
     * This is used to store the state of the toggle buttons for each item in the list
     */
    private List<Boolean> mToggleState;

    private int mColumnRowId;
    private int mColumnTitle;
    private int mColumnDescription;
    private int mColumnIconName;
    private int mColumnVisibility;

    public NotesListAdapter(Context context, Cursor cursor, NotificationHelper helper, AgendaNotesAdapter adapter) {
        super(context, cursor);

        mInflater = LayoutInflater.from(context);

        /*
         * Helper class to post notifications to the status bar and database adapter class to update
         * the database data when the user presses the toggle button in any of items in the list
         */
        mNotificationHelper = helper;
        mAgendaAdapter = adapter;

        /*
         * There's no need to keep getting the column indexes every time in bindView() (as I see in
         * a few examples) so I do it once and save the indexes in instance variables
         */
        findColumnIndexes(cursor);

        /*
         * Populate the toggle button states for each item in the list with the corresponding value
         * from each record in the database, but isn't this a slow operation?
         */
        for(mToggleState = new ArrayList<Boolean>(); !cursor.isAfterLast(); cursor.moveToNext()) {
            mToggleState.add(cursor.getInt(mColumnVisibility) != 0);
        }
    }

    @Override
    public View newView(Context context, Cursor cursor, ViewGroup parent) {
        View view = mInflater.inflate(R.layout.list_item_note, null);

        /*
         * The ViewHolder pattern is here only used to prevent calling findViewById() all the time
         * in bindView(), we only need to find all the views once
         */
        ViewHolder viewHolder = new ViewHolder();

        viewHolder.icon = (ImageView)view.findViewById(R.id.imageview_icon);
        viewHolder.title = (TextView)view.findViewById(R.id.textview_title);
        viewHolder.description = (TextView)view.findViewById(R.id.textview_description);
        viewHolder.visibility = (ToggleButton)view.findViewById(R.id.togglebutton_visibility);

        /*
         * I also use newView() to set the toggle button click listener for each item in the list
         */
        viewHolder.visibility.setOnClickListener(this);

        view.setTag(viewHolder);

        return view;
    }

    @Override
    public void bindView(View view, Context context, Cursor cursor) {
        Resources resources = context.getResources();

        int iconId = resources.getIdentifier(cursor.getString(mColumnIconName),
                "drawable", context.getPackageName());

        String title = cursor.getString(mColumnTitle);
        String description = cursor.getString(mColumnDescription);

        /*
         * This is similar to the ViewHolder pattern and it's need to access the note data when the
         * onClick() method is fired
         */
        NoteData noteData = new NoteData();

        /*
         * This data is needed to post a notification when the onClick() method is fired
         */
        noteData.id = cursor.getLong(mColumnRowId);
        noteData.iconId = iconId;
        noteData.title = title;
        noteData.description = description;

        /*
         * This data is needed to update mToggleState[POS] when the onClick() method is fired
         */
        noteData.position = cursor.getPosition();

        /*
         * Get our ViewHolder with all the view IDs found in newView()
         */
        ViewHolder viewHolder = (ViewHolder)view.getTag();

        /*
         * The Html.fromHtml is needed but the code relevant to that was stripped
         */
        viewHolder.icon.setImageResource(iconId);
        viewHolder.title.setText(Html.fromHtml(title));
        viewHolder.description.setText(Html.fromHtml(description));

        /*
         * Set the toggle button state for this list item from the value in mToggleState[POS]
         * instead of getting it from the database with 'cursor.getInt(mColumnVisibility) != 0'
         * otherwise the state will be incorrect if it was changed between the item view scrolling
         * out of view and scrolling back into view
         */
        viewHolder.visibility.setChecked(mToggleState.get(noteData.position));

        /*
         * Again, save the note data to be accessed when onClick() gets fired
         */
        viewHolder.visibility.setTag(noteData);
    }

    @Override
    public void onClick(View view) {
        /*
         * Get the new state directly from the toggle button state 
         */
        boolean visibility = ((ToggleButton)view).isChecked();

        /*
         * Get all our note data needed to post (or remove) a notification 
         */
        NoteData noteData = (NoteData)view.getTag();

        /*
         * The toggle button state changed, update mToggleState[POS] to reflect that new change
         */
        mToggleState.set(noteData.position, visibility);

        /*
         * Post the notification or remove it from the status bar depending on toggle button state
         */
        if(visibility) {
            mNotificationHelper.postNotification(
                    noteData.id, noteData.iconId, noteData.title, noteData.description);
        } else {
            mNotificationHelper.cancelNotification(noteData.id);
        }

        /*
         * Update the database note item with the new toggle button state, without the need to
         * requery the cursor (which is slow, I've tested it) to reflect the new toggle button state
         * in the list because the value was saved in mToggleState[POS] a few lines above
         */
        mAgendaAdapter.updateNote(noteData.id, null, null, null, null, visibility);
    }

    private void findColumnIndexes(Cursor cursor) {
        mColumnRowId = cursor.getColumnIndex(AgendaNotesAdapter.KEY_ROW_ID);
        mColumnTitle = cursor.getColumnIndex(AgendaNotesAdapter.KEY_TITLE);
        mColumnDescription = cursor.getColumnIndex(AgendaNotesAdapter.KEY_DESCRIPTION);
        mColumnIconName = cursor.getColumnIndex(AgendaNotesAdapter.KEY_ICON_NAME);
        mColumnVisibility = cursor.getColumnIndex(AgendaNotesAdapter.KEY_VISIBILITY);
    }

}

Ответы [ 3 ]

4 голосов
/ 29 февраля 2012

Ваше решение оптимально, и я добавлю его в свое оружие :). Может быть, я постараюсь внести небольшую оптимизацию для вызовов в базу данных.

Действительно, из-за условий задачи есть только три решения:

  1. Обновление только одной строки, запрос курсора и перерисовка всех элементов. (Прямой, грубая сила).
  2. Обновление строки, кеширование результатов и использование кеша для рисования элементов.
  3. Кэшируйте результаты, используйте кеш для рисования элементов. И когда вы оставляете это действие / фрагмент, то фиксируете результаты в базе данных.

Для третьего решения вы можете использовать SparseArray для поиска изменений.

private SparseArray<NoteData> mArrayViewHolders;

public void onClick(View view) {
     //here your logic with NoteData. 
     //start of my improve
     if (mArrayViewHolders.get(selectedPosition) == null) {
        // put the change into array
        mArrayViewHolders.put(selectedPosition, noteData);
     } else {
        // rollback the change
        mArrayViewHolders.delete(selectedPosition);
     }
     //end of my improve
     //we don't commit the changes to database.
}

Еще раз: с самого начала этот массив пуст. Когда вы переключаете кнопку в первый раз (есть изменение), вы добавляете NoteData в массив. Когда вы переключаете кнопку во второй раз (происходит откат), вы удаляете NoteData из массива. И так далее.

Когда вы закончите, просто запросите массив и поместите изменения в базу данных.

1 голос
/ 18 июня 2012

Я бы подождал, прежде чем идти в CursorLoader. Как кажется, производные CursorLoader не работают с CursorLoader.

1 голос
/ 01 марта 2012

То, что вы видите, - это просмотр использования Android.Я не думаю, что вы делаете что-то не так, снова запрашивая курсор.Просто не используйте функцию cursor.requery ().

Вместо этого всегда сначала установите переключатель в положение false, а затем задайте курсор и включите его, если необходимо.

Может быть, вы делали это, и я что-то неправильно понял, однако я не думаю, что у вас должны быть медленные результаты.

Псевдокод:

getView(){
setToggleFalse();
boolean value = Boolean.valueOf(cursor.getString('my column'));
if (value){
   setToggleTrue();
}
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...