欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android開發(fā)筆記之:ListView刷新順序的問題詳解

 更新時間:2013年05月21日 10:00:19   作者:  
本篇文章是對Android中ListView刷新順序的問題進行了詳細的分析介紹,需要的朋友參考下
背景
一個典型的ListView,每個Item顯示一個TextView,代表一個Task,需要實現(xiàn)二個編輯方式:一個是用CheckBox來標識任務(wù)已經(jīng)完成,另一個要實現(xiàn)的編輯是刪除任務(wù)。對于完成的CheckBox就直接放在布局中就可,但對于刪除不想使用ContextMenu來實現(xiàn)編輯,對于像iOS中那樣的列表,它的刪除都是通過對列表中每個項目的手勢來觸發(fā)。這個實現(xiàn)起來并不難,可以用一個ViewSwitcher,Checkbox和刪除按扭是放入其中,讓ViewSwitcher來控制顯示哪一個,正常情況下顯示Checkbox,隱藏刪除按扭,然后當點擊Item時就顯示刪除按扭,隱藏Checkbox,這樣也更符合操作習慣,可以一個一個條目的刪除。
實現(xiàn)起來的方式如下:
復(fù)制代碼 代碼如下:

public class ListOrderActivity extends Activity {
    private ListView mTaskList;
    private EditText mAddTaskEditor;
    private LayoutInflater mFactory;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.list_activity);

        mFactory = LayoutInflater.from(getApplication());
        mTaskList = (ListView) findViewById(R.id.task_list);
        final View headerView = mFactory.inflate(R.layout.header_view, null);
        mTaskList.addHeaderView(headerView);
        mAddTaskEditor = (EditText) headerView.findViewById(R.id.task_editor);
        mAddTaskEditor.setOnKeyListener(new OnKeyListener() {
            @Override
            public boolean onKey(View view, int keycode, KeyEvent event) {
         if (keycode == KeyEvent.KEYCODE_DPAD_CENTER || keycode == KeyEvent.KEYCODE_ENTER) {
             // finish editing
             final InputMethodManager inputManager = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
             inputManager.hideSoftInputFromWindow(getCurrentFocus().getWindowToken(), 0);
             final String text = mAddTaskEditor.getText().toString();
             if (!TextUtils.isEmpty(text)) {
          final ContentValues values = new ContentValues(1);
          values.put(TaskColumns.TASK, text);
          values.put(TaskColumns.TYPE, Task.TYPE_TODAY);
          getContentResolver().insert(Task.CONTENT_URI, values);
             }
             mAddTaskEditor.setText("");
         }
         return false;
            }
        });
        final Cursor cursor = getContentResolver().query(Task.CONTENT_URI, Task.PROJECTION, TaskColumns.TYPE + " = " + Task.TYPE_TODAY, null, null);
        final TaskAdapter adapter = new TaskAdapter(getApplication(), cursor);
        mTaskList.setAdapter(adapter);
    }

    private class TaskAdapter extends CursorAdapter {
        private Cursor mCursor;

        public TaskAdapter(Context context, Cursor c) {
            super(context, c);
            mCursor = c;
        }

        @Override
        public void bindView(View view, Context context, Cursor cursor) {
            if (view  == null) {
                view = mFactory.inflate(R.layout.today_task_item, null);
            }
            final ViewSwitcher switcher = (ViewSwitcher) view.findViewById(R.id.action_switcher);
//            if (switcher.getDisplayedChild() == 1) {
//         switcher.clearAnimation();
//         switcher.showPrevious();
//         switcher.clearAnimation();
//            }
            final CheckBox toggle = (CheckBox) view.findViewById(R.id.action_toggle_done);
            final short done = cursor.getShort(ProjectionIndex.DONE);
            final int id = cursor.getInt(ProjectionIndex.ID);
            toggle.setOnCheckedChangeListener(null);
            toggle.setChecked(done != 0);
            toggle.setOnCheckedChangeListener(new OnCheckedChangeListener() {
         @Override
         public void onCheckedChanged(CompoundButton view, boolean checked) {
             final Uri uri = ContentUris.withAppendedId(Task.CONTENT_URI, id);
             final ContentValues values = new ContentValues(1);
             values.put(TaskColumns.DONE, checked ? 1 : 0);
             getContentResolver().update(uri, values, null, null);
         }
            });
            view.setOnClickListener(new OnClickListener() {
         @Override
         public void onClick(View v) {
             switcher.showNext();
             if (switcher.getDisplayedChild() == 0) {
          switcher.getInAnimation().setAnimationListener(null);
          return;
             }
             final ImageView delete = (ImageView) v.findViewById(R.id.action_delete_task);
             delete.setOnClickListener(new OnClickListener() {
          @Override
          public void onClick(View v) {
              switcher.getInAnimation().setAnimationListener(new AnimationListener() {
           @Override
           public void onAnimationEnd(Animation animation) {
               switcher.getInAnimation().setAnimationListener(null);
               final Uri uri = ContentUris.withAppendedId(Task.CONTENT_URI, id);
               getContentResolver().delete(uri, null, null);
           }

           @Override
           public void onAnimationRepeat(Animation animation) {
           }

           @Override
           public void onAnimationStart(Animation animation) {
           }
              });
              switcher.showPrevious();
          }
      });
         }
     });
            TextView task = (TextView) view.findViewById(R.id.task);
            final String taskContent = cursor.getString(ProjectionIndex.TASK);
            if (done != 0) {
         final Spannable style = new SpannableString(taskContent);
         style.setSpan(new StrikethroughSpan(), 0, taskContent.length(), Spannable.SPAN_INCLUSIVE_INCLUSIVE);
         style.setSpan(new StyleSpan(Typeface.ITALIC) , 0, taskContent.length(), Spannable.SPAN_INCLUSIVE_INCLUSIVE);
         task.setText(style);
         task.setTextAppearance(getApplication(), R.style.done_task_item_text);
            } else {
         task.setText(taskContent);
         task.setTextAppearance(getApplication(), R.style.task_item_text);
            }
        }
        @Override
        public View newView(Context context, Cursor cursor, ViewGroup parent) {
            View view = mFactory.inflate(R.layout.today_task_item, null);
            return view;
        }

        @Override
        public void onContentChanged() {
            mCursor.requery();
        }
    }
}

復(fù)制代碼 代碼如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical"
    android:background="#f0f0f0"
    android:paddingBottom="5dip"
    android:paddingLeft="12dip"
    android:paddingRight="12dip"
    android:paddingTop="5dip" >
    <ListView
        android:id="@+id/task_list"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        android:divider="@color/divider"
        android:dividerHeight="0.6dip" />
</LinearLayout>

復(fù)制代碼 代碼如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
  xmlns:android="http://schemas.android.com/apk/res/android"
  android:orientation="horizontal"
  android:layout_width="fill_parent"
  android:layout_height="wrap_content"
  android:layout_gravity="center"
  android:gravity="center">
    <ViewSwitcher android:id="@+id/action_switcher"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:gravity="center"
        android:inAnimation="@anim/action_switcher_in"
        android:outAnimation="@anim/action_switcher_out">
     <CheckBox android:id="@+id/action_toggle_done"
         android:layout_width="wrap_content"
         android:layout_height="wrap_content"
         android:gravity="center"
         android:layout_gravity="center" />
     <ImageView android:id="@+id/action_delete_task"
         android:src="@drawable/ic_delete"
         android:layout_width="48dip"
         android:layout_height="48dip"
         android:contentDescription="@string/delete_description"
         android:gravity="center"
         android:layout_gravity="center"
         android:scaleType="center" />
    </ViewSwitcher>
 <TextView android:id="@+id/task"
           style="@style/task_item_text" />
</LinearLayout>



問題
但這有一個問題,就是如果其中某個條目是處于刪除狀態(tài),這時再添加一個新任務(wù),或者點擊另外條目的Checkbox時,條目的狀態(tài)會錯亂,本來處于正常狀態(tài)的條目會處于刪除狀態(tài)!
原因分析
最開始以為是數(shù)據(jù)問題,因為事件的處理都是匿名的類,可能會指向不正確的外部數(shù)據(jù),通過打印調(diào)試發(fā)現(xiàn)所有數(shù)據(jù)都是對的。最后通過在bindView方法中加LOG信息發(fā)現(xiàn)了原因:每次ListView刷新bindView的順序并不相同,原來處在第3的子View,刷新后可能被放在第1位置。ViewSwitcher的顯示狀態(tài)是它自己維護的,也就是說沒有在View的外部保存其應(yīng)該顯示的狀態(tài),所以當數(shù)據(jù)發(fā)生變化(Checkbox會引發(fā)數(shù)據(jù)變化)刷新列表時,原來處于刪除狀態(tài)的子View(可能在第4位置)現(xiàn)在可能變成了第2位置,造成了第二個處于刪除狀態(tài),而第四個處于正常狀態(tài)。
解決方案
這個問題沒有完美解決方法,只能做一個Workaround的方法:那就是每次刷新bindView時把刪除狀態(tài)清掉,都換成默認狀態(tài),這樣至少不會出現(xiàn)狀態(tài)混亂的狀況。但是,還是會看到刪除會閃一下。
要想完全解決這個問題就是避免在有其他方式導(dǎo)致數(shù)據(jù)變化時使用這種設(shè)計,這種設(shè)計僅適用于:整個列表僅有刪除,沒有其他方式能導(dǎo)致列表會刷新時,這時每當刪除時,直接把子View從ListView中移除,就不會出現(xiàn)混亂了。

同時也說明為什么我們每次bindView時要重新給子View添數(shù)據(jù),而不是僅當創(chuàng)建子View添數(shù)據(jù)。因為每次刷新bindView時順序并不一定是先前的順序,所以一定要重新添數(shù)據(jù)。而數(shù)據(jù)通常是與View分享開來,或是在數(shù)據(jù)庫中,或是其他形式,會以特定的順序存在,它不會因為View的刷新而改變,所以為了不使用戶感覺狀態(tài)錯亂,就必須要重新按照數(shù)據(jù)的順序來給View填充數(shù)據(jù)。

相關(guān)文章

  • OpenGL ES著色器使用詳解(二)

    OpenGL ES著色器使用詳解(二)

    這篇文章主要為大家詳細介紹了OpenGL ES著色器的使用方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-05-05
  • Android實現(xiàn)時間倒計時功能

    Android實現(xiàn)時間倒計時功能

    這篇文章主要為大家詳細介紹了Android實現(xiàn)時間倒計時功能,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-10-10
  • android鬧鈴簡單實現(xiàn)

    android鬧鈴簡單實現(xiàn)

    本文給大家分享的是一段簡單的實現(xiàn)Android系統(tǒng)的鬧鈴的代碼,非常實用,想做Android開發(fā)的小伙伴們可以參考下。
    2015-03-03
  • Android studio 如何刪除項目 module

    Android studio 如何刪除項目 module

    本篇文章主要介紹了Android studio 如何刪除項目module的相關(guān)知識,具有很好的參考價值。下面跟著小編一起來看下吧
    2017-05-05
  • Android自定義懸浮按鈕效果

    Android自定義懸浮按鈕效果

    這篇文章主要為大家詳細介紹了Android自定義懸浮按鈕效果,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-09-09
  • Android學習教程之日歷庫使用(15)

    Android學習教程之日歷庫使用(15)

    這篇文章主要為大家詳細介紹了Android學習教程之日歷庫使用的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2016-11-11
  • Android編程實現(xiàn)播放MP3功能示例

    Android編程實現(xiàn)播放MP3功能示例

    這篇文章主要介紹了Android編程實現(xiàn)播放MP3功能,結(jié)合實例形式分析了Android播放MP3功能的界面布局與功能實現(xiàn)相關(guān)操作技巧,需要的朋友可以參考下
    2017-02-02
  • Android實現(xiàn)隱藏狀態(tài)欄和標題欄

    Android實現(xiàn)隱藏狀態(tài)欄和標題欄

    這篇文章主要介紹了Android實現(xiàn)隱藏狀態(tài)欄和標題欄的相關(guān)資料,需要的朋友可以參考下
    2015-06-06
  • android之listview懸浮topBar效果

    android之listview懸浮topBar效果

    這篇文章主要為大家詳細介紹了android之listview懸浮topBar效果的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-02-02
  • Android app啟動時黑屏或者白屏的原因及解決辦法

    Android app啟動時黑屏或者白屏的原因及解決辦法

    這篇文章主要介紹了Android app啟動時黑屏或者白屏的原因及解決辦法的相關(guān)資料,需要的朋友可以參考下
    2016-09-09

最新評論