我正在尝试恢复操作模式在方向更改时的状态。但它不能正常工作。按照以下步骤的顺序,在3和4中,所选内容将损坏:
长按项目-项目将突出显示,上下文操作栏将显示,标题显示“1选定”。看起来不错。
把电话转到横向-没有变化。看起来不错。
将手机调回纵向-项目未突出显示,上下文操作栏已消失
长按项目-项目将突出显示,上下文操作栏显示,标题显示“0选定”myListView.getCheckedItemCount()
inonSaveInstanceState()
在3返回0。问题就从这里开始。
我在片段中做错了什么(使用支持库)?
@Override
public void onSaveInstanceState(Bundle outState) {
//check if any items are selected
if (myListView.getCheckedItemCount() > 0) {
//get the list of selected items and convert it to an int Array
//because SparseBooleanArray cannot be stored in a bundle
SparseBooleanArray selectedItems = myListView.getCheckedItemPositions();
int[] selectedItems_intArray = new int[myListView.getCheckedItemCount()];
for (int i = 0; i < selectedItems.size(); i++) {
if (selectedItems.valueAt(i) == false)
continue;
selectedItems_intArray[i] = selectedItems.keyAt(i);
}
outState.putIntArray(KEY_CHECKED_ITEMS, selectedItems_intArray);
}
}
@Override
public void onViewStateRestored(Bundle savedInstanceState) {
super.onViewStateRestored(savedInstanceState);
if (savedInstanceState != null) {
int[] checkedItems = savedInstanceState.getIntArray(KEY_CHECKED_ITEMS);
if (checkedItems != null) {
actionMode = ((ActionBarActivity) getActivity()).startSupportActionMode(new ContextualActionBarActionModeCallBack());
for (int i = 0; i < checkedItems.length; i++) {
myListView.setItemChecked(checkedItems[i], true);
}
actionMode.setTitle(myListView.getCheckedItemCount() + " selected");
}
}
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
myListView = (ListView) getActivity().findViewById(R.id.myListView);
adpt = new myCustomCursorAdapter(getActivity());
myListView.setAdapter(adpt);
//Choice mode is allowed only after a long click
//disabling it on first time load
myListView.setChoiceMode(ListView.CHOICE_MODE_NONE);
myListView.setOnItemLongClickListener(new OnItemLongClickListener(){
@Override
public boolean onItemLongClick(AdapterView<?> parent, View view, int position, long id){
if(actionMode == null){
//Start the action mode
actionMode = ((ActionBarActivity)getActivity()).startSupportActionMode(new ContextualActionBarActionModeCallBack());
myLisVIew.setItemChecked(position, true);
actionMode.setTitle(myListView.getCheckedItemCount() + " selected");
return true;
}
else return false;
}
});
}
@Override
public void onResume(){
super.onResume();
getLoaderManager().initLoader(0,null,this);
}
private class ContextualActionBarActionModeCallBack implements ActionMode.Callback{
@Override
public boolean onCreateActionMode(ActionMode mode, Menu menu) {
MenuInflater inflater = mode.getMenuInflater();
inflater.inflate(R.menu.mycontextmenu, menu);
myListView.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE);
}
...
...
}
ListView布局元素:
<ListView
android:id="@+id/myListView"
android:layout_height="match_parent"
android:layout_width="match_parent"
android:layout_alignParentTop="true"
android:listSelector="@android:color/transparent"
android:stackFromBottom="true"
android:layout_above="@id/layout_input"
android:divider="#00000000"
/>
最佳答案
问题是,每次在方向更改时创建列表视图时,列表视图的选择模式都会重置为默认值CHOICE_MODE_NONE
,因为列表视图布局文件中未显式定义选择模式。android的源代码-http://androidxref.com/4.4.2_r2/xref/frameworks/base/core/java/android/widget/AbsListView.java#815。
解决方法是添加:
android:choiceMode="multipleChoice"
到列表视图布局。
这也意味着不需要手动将列表视图选项存储在
onSaveInstanceState()
中并在onViewStateRestored()
中检索。它是由片段自动完成的。所以删除这两个函数。只需在onResume
中启动操作模式:@Override
public void onResume() {
super.onResume();
getLoaderManager().initLoader(0,null,this);
if (myListView.getCheckedItemCount() > 0 && actionMode == null) {
actionMode = ((ActionBarActivity) getActivity()).startSupportActionMode(new ContextualActionBarActionModeCallBack());
actionMode.setTitle(myListView.getCheckedItemCount() + " selected");
}
}
注意:如果您已经实现了
LoaderManager.LoaderCallBacks
接口,并且您需要在方向改变后访问适配器项,请在适配器加载了数据源后在onLoadFinished()
中执行此操作。在onResume()
中执行会导致NPE
。