[英]How to save custom ListFragment state with orientation change
我更彻底地希望这个问题实际上会更容易理解。
活动目的 :允许用户从图库中选择图像; 在ListFragment中显示图像的缩略图以及标题用户给出的图像; 用户完成操作后,保存每个图像的uri和标题,而用户的名字就是该图像的集合。
问题 :旋转设备时,FragmentList将丢失用户已经选择的所有图像和标题,即列表的所有行均丢失。
尝试解决问题 :
实现RetainedFragment以在设备旋转时保存List集合。 以前我还没有做过这件事,并且想过:“啊,适配器在创建时被喂入一个空白的List集合。我将保存列表的状态,然后在调用Activity的onCreate时,我可以将保留的List喂入Adapter构造函数,并且会工作的。” 但事实并非如此。
然后我想:“当然不能用,您还没有通知适配器更改!” 所以我把adapter.notifyDataSetChanged()
放在onCreate中。 这没用。
然后,我将adapter.notifyDataSetChanged()
移到onStart
以为我可能需要在活动的生命周期中稍后通知适配器。 没用
注意:我在同一应用程序中有另一个活动,该活动使用相同的自定义ListViewFragment,并且通过设备方向更改来保留ListFragment的状态。 该活动有两个原则上的区别:片段被硬编码到.xml中(我认为这不会有所作为,除了也许Android原生保存的.xml片段不同于以编程方式添加的片段外); 并且该活动使用Loader和LoaderManager并从我构建的提供程序(从我的SQLite数据库收集数据)中获取其数据。 看这两个活动之间的差异是什么使我想到“您没有以某种方式适当地处理向适配器提供的数据”,并启发了我在设备旋转时使用RetainedFragment保存List集合。
...这促使我思考如何解决,就像Android在其关于LoaderManager的Loader页面上所说的那样:
“与活动或片段相关联的抽象类,用于管理一个或多个Loader实例。这有助于应用程序与活动或片段生命周期一起管理运行时间较长的操作;最常见的用法是与CursorLoader一起使用,但是应用程序自由编写自己的加载程序以加载其他类型的数据。”
我想到的是“正在加载其他类型的数据”部分,“我可以使用LoaderManager加载列表数据吗?我对此避讳的两个原因:1)我已经(至少在概念上)应该可以工作; 2) )我不认为目前根本不是一个“长时间运行的操作”。
研究方向
StackOverflow傻瓜式方法,可在方向改变时处理片段
一劳永逸,如何正确地将Fragments的实例状态保存在后台堆栈中?
保存后退堆栈片段。
未在下面粘贴的代码中显示,但是我的活动动态创建了其他三个片段,如果savedInstanceState !=null
并且在不对onSaveInstanceState()
做任何工作的情况下保存了这些片段的状态,则使用以下内容(这就是为什么感觉像我的问题不在于在onSaveInstanceState
做某事,因为Android会处理保存我的其他片段状态,因此它也不应该使用ListFragment来做吗?似乎应该这样做)。
if(savedInstanceState.containsKey(AddActivity_Frag1.F1_TAG)){ frag1 = (AddActivity_Frag1)getFragmentManager().getFragment(savedInstanceState, AddActivity_Frag1.F1_TAG); }
我的查询周围的许多StackOverflow问题似乎都与如何通过更改方向保存ListFragment的滚动位置有关,但是我不需要这样做(尽管我确实读过它们,旨在寻找可能有所帮助的技巧)。
活动-删除了许多(希望与之无关)的内容:
public class AddActivity extends Activity{
// data collection
List<ImageBean> beanList;
// adapter
AddCollectionAdapter adapter;
// ListViewFragment tag
private static final String LVF_TAG = "list fragment tag";
// fragment handles
ListViewFragment listFrag;
// Handles images; LruCache for bitmapes
ImageHandler imageHandler;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_add2);
// Create ImageHandler that holds LruCache
imageHandler = new ImageHandler(this, getFragmentManager());
// Obtain retained List<ImageBean> or create new List<ImageBean>.
RetainedFragment retainFragment = RetainedFragment.findOrCreateRetainFragment(getFragmentManager());
beanList = retainFragment.list;
if(beanList == null){
beanList = new ArrayList<ImageBean>();
retainFragment.list = beanList;
}
// create fragments
if(savedInstanceState == null){
listFrag = new ListViewFragment();
FragmentTransaction ft = getFragmentManager().beginTransaction();
ft.add(R.id.add_fragFrame, listFrag, LVF_TAG);
ft.commit();
}else{
listFrag = (ListViewFragment)getFragmentManager().findFragmentByTag(LVF_TAG);
}
// create adapter
adapter = new AddCollectionAdapter(this, beanList);
// set list fragment adapter
listFrag.setListAdapter(adapter);
}
@Override
protected void onStart() {
// TESTING: If device orientation has changed List<ImageBean> was saved
// with a RetainedFragment. Seed the adapter with the retained
// List.
adapter.notifyDataSetChanged();
super.onStart();
}
@Override
protected void onSaveInstanceState(Bundle outState) {
// Android automatically saves visible fragments here. (?)
super.onSaveInstanceState(outState);
}
/*
* ImageBean.
*/
public static class ImageBean{
private String collectionName; // Title of image collection
private String imageUri; // Image URI as a string
private String imageTitle; // Title given to image
public ImageBean(String name, String uri, String title){
collectionName = name;
imageUri = uri;
imageTitle = title;
}
public String getCollectionName() {
return collectionName;
}
public String getImageUri() {
return imageUri;
}
public String getImageTitle() {
return imageTitle;
}
}
/*
* Called when user is finished selecting images.
*
* Performs a bulk insert to the Provider.
*/
private void saveToDatabase() {
int arraySize = beanList.size();
final ContentValues[] valuesArray = new ContentValues[arraySize];
ContentValues values;
String imageuri;
String title;
int counter = 0;
for(ImageBean image : beanList){
imageuri = image.getImageUri();
title = image.getImageTitle();
values = new ContentValues();
values.put(CollectionsTable.COL_NAME, nameOfCollection);
values.put(CollectionsTable.COL_IMAGEURI, imageuri);
values.put(CollectionsTable.COL_TITLE, title);
values.put(CollectionsTable.COL_SEQ, counter +1);
valuesArray[counter] = values;
counter++;
}
AsyncTask<Void, Void, Void> task = new AsyncTask<Void, Void, Void>() {
@Override
protected Void doInBackground(Void... arg0) {
getContentResolver().bulkInsert(CollectionsContentProvider.COLLECTIONS_URI, valuesArray);
return null;
}
@Override
protected void onPostExecute(Void result) {
// End this activity.
finish();
}
};
task.execute();
}
public ImageHandler getImageHandler(){
return imageHandler;
}
}
class RetainedFragment extends Fragment{
private static final String TAG = "RetainedFragment";
// data to retain
public List<AddActivity.ImageBean> list;
public static RetainedFragment findOrCreateRetainFragment(FragmentManager fm){
RetainedFragment fragment = (RetainedFragment)fm.findFragmentByTag(TAG);
if(fragment == null){
fragment = new RetainedFragment();
fm.beginTransaction().add(fragment, TAG);
}
return fragment;
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setRetainInstance(true);
}
}
ListFragment:
public class ListViewFragment extends ListFragment {
ListFragListener listener;
public interface ListFragListener{
public void listFragListener(Cursor cursor);
}
@Override
public void onCreate(Bundle savedInstanceState) {
// Retain this fragment across configuration change
setRetainInstance(true);
super.onCreate(savedInstanceState);
}
@Override
public void onAttach(Activity activity) {
super.onAttach(activity);
// Set listener
if(activity instanceof ListFragListener){
listener = (ListFragListener)activity;
}else{
//Instantiating activity does not implement ListFragListener.
}
}
@Override
public void onListItemClick(ListView listView, View v, int position, long id) {
// no action necessary
}
}
适配器:
public class AddCollectionAdapter extends BaseAdapter {
// data collection
List<ImageBean> beanList;
// layout inflator
private LayoutInflater inflater;
// context
Context context;
public AddCollectionAdapter(Context context, List<ImageBean> beanList){
this.context = context;
this.beanList = beanList;
inflater = (LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
}
@Override
public int getCount() {
return beanList.size();
}
@Override
public Object getItem(int position) {
return beanList.get(position);
}
@Override
public long getItemId(int arg0) {
// collection not from database nor is going directly to database; this is useless.
return 0;
}
// holder pattern
private class ViewHolder{
ImageView imageView;
TextView titleView;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder;
View xmlTemplate = convertView;
if(xmlTemplate == null){
//inflate xml
xmlTemplate = inflater.inflate(R.layout.frag_listview_row, null);
// initilaize ViewHolder
holder = new ViewHolder();
// get views that are inside the xml
holder.imageView = (ImageView)xmlTemplate.findViewById(R.id.add_lvrow_image);
holder.titleView = (TextView)xmlTemplate.findViewById(R.id.add_lvrow_title);
// set tag
xmlTemplate.setTag(holder);
}else{
holder = (ViewHolder)xmlTemplate.getTag();
}
// Get image details from List<ImageBean>
ImageBean bean = beanList.get(position);
String imageUri = bean.getImageUri();
String title = bean.getImageTitle();
// Set Holder ImageView bitmap; Use parent activity's ImageHandler to load image into Holder's ImageView.
((AddActivity)context).getImageHandler().loadBitmap(imageUri, holder.imageView, Constants.LISTVIEW_XML_WIDTH, Constants.LISTVIEW_XML_HEIGHT);
// Set Holder's TextView.
holder.titleView.setText(title);
// return view
return xmlTemplate;
}
}
解决了。 将日志语句放在重要位置后,我发现RetainedFragment的列表始终为空。 挠头后在RetainedFragment中注意到了这一点:
fm.beginTransaction().add(fragment, TAG);
我错过了commit()
!
在我添加完状态之后,现在通过配置更改来保留状态。
有关保存在试验和灾难中发现的ListFragment状态的更多信息:
如果您通过以下方式添加片段:
if(savedInstanceState == null){
listFrag = new ListViewFragment();
// programmatically add fragment to ViewGroup
FragmentTransaction ft = getFragmentManager().beginTransaction();
ft.add(R.id.add_fragFrame, listFrag, LVF_TAG);
}
然后这些都可以在else
工作:
1) This one works because Android takes care of saving the Fragment:
listFrag = (ListViewFragment)getFragmentManager().findFragmentByTag(LVF_TAG);
2) This one works because the fragment was specifically saved into bundle in
onSaveInstanceState:
listFrag = (ListViewFragment)getFragmentManager().getFragment(savedInstanceState, LVF_TAG);
为了使编号2起作用,这发生在onSaveInstanceState()中:
@Override
protected void onSaveInstanceState(Bundle outState) {
super.onSaveInstanceState(outState);
getFragmentManager().putFragment(outState, LVF_TAG, listFrag);
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.