FragmentStatePagerAdapter 是否在方向更改时保存片段状态?
Does FragmentStatePagerAdapter save fragment state on orientation change?
我有 3 个片段需要放在 ViewPager 中。这些片段将保存从数据库中检索到的动态信息。我知道在方向改变时,activity 和片段会被销毁并重新创建。但我对它的名字印象深刻,FragmentStatePagerAdapter 将保存片段的状态。显然,我错了,因为每次我对片段做一些事情,然后改变方向,片段都会恢复到它在布局 xml 文件中的布局。
在调试时,我注意到在方向更改时,适配器的 getItem() 方法从未被调用 - 这意味着它没有被重新创建。那么碎片状态是怎么恢复到原来状态的呢?
如何使用 FragmentStatePagerAdapter 保存片段状态?
请注意,我一直在关注此 tutorial and used their version of the SmartFragmentStatePagerAdapter.java class 以动态管理片段。
下面是我的示例代码。
PageLoader.java - 此接口允许 MainActivity 在 运行 时间动态管理片段页面的加载。
public interface PageLoader {
void loadPage(int from, int target);
}
MainActivity.java
public class MainActivity extends AppCompatActivity implements PageLoader {
MyPagerAdapter adapter;
DirectionalViewPager pager;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// Get the ViewPager and set it's PagerAdapter so that it can display items
pager = (DirectionalViewPager) findViewById(R.id.vpPager);
adapter = new MyPagerAdapter(getSupportFragmentManager());
pager.setOffscreenPageLimit(5);
pager.setAdapter(adapter);
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
int position = pager.getCurrentItem();
if (position > 0) pager.setCurrentItem(position - 1);
return true;
}
@Override
public void loadPage(int from, int target) {
PageLoader fragment = (PageLoader) adapter.getRegisteredFragment(target);
fragment.loadPage(from, target);
}
}
MyPagerAdapter.java
public class MyPagerAdapter extends SmartFragmentStatePagerAdapter {
private static final int NUM_ITEMS = 4;
public MyPagerAdapter(FragmentManager fragmentManager) {
super(fragmentManager);
}
// Returns total number of pages
@Override
public int getCount() {
return NUM_ITEMS;
}
// Returns the fragment to display for that page
@Override
public Fragment getItem(int position) {
switch (position) {
case 0: // Fragment # 0 - This will show Frag1
return Frag1.newInstance(position, Frag1.class.getSimpleName());
case 1: // Fragment # 0 - This will show Frag1 different title
return Frag1.newInstance(position, Frag1.class.getSimpleName());
case 2: // Fragment # 1 - This will show Frag2
return Frag2.newInstance(position, Frag2.class.getSimpleName());
default:
return Frag3.newInstance(position, Frag3.class.getSimpleName());
}
}
// Returns the page title for the top indicator
@Override
public CharSequence getPageTitle(int position) {
return "Page " + position;
}
}
Frag1.java Frag2.java Frag3.java - 这些都是一样的,除了编号。
public class Frag1 extends Fragment implements PageLoader {
// Store instance variables
private String title;
private int page;
private TextView txtView;
// newInstance constructor for creating fragment with arguments
public static Frag1 newInstance(int page, String title) {
Frag1 fragmentFirst = new Frag1();
Bundle args = new Bundle();
args.putInt("someInt", page);
args.putString("someTitle", title);
fragmentFirst.setArguments(args);
return fragmentFirst;
}
// Store instance variables based on arguments passed
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
page = getArguments().getInt("someInt", 0);
title = getArguments().getString("someTitle");
}
// Inflate the view for the fragment based on layout XML
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.frag1, container, false);
txtView = (TextView) view.findViewById(R.id.txt_frag1);
txtView.setText(page + " - " + title);
Button btn = (Button) view.findViewById(R.id.btn_frag1);
btn.setOnClickListener( new View.OnClickListener() {
@Override
public void onClick(View v) {
PageLoader activity = (PageLoader) getActivity();
activity.loadPage(page, page+1);
}
});
return view;
}
@Override
public void loadPage(int from, int target) {
txtView.setText(txtView.getText() + "\nThis message was created from" + from + " to " + target);
}
}
activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<com.example.someone.smartfragmentstatepageradapter.custom.DirectionalViewPager
android:id="@+id/vpPager"
android:layout_width="match_parent"
android:layout_height="wrap_content">
</com.example.someone.smartfragmentstatepageradapter.custom.DirectionalViewPager>
</LinearLayout>
frag1.xml frag2.xml frag3.xml - 除了编号
之外,这些都是相同的
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:background="#cc2">
<TextView
android:id="@+id/txt_frag1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:text="txt_frag1"
/>
<Button
android:id="@+id/btn_frag1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:text="btn_frag1"
android:textSize="26dp" />
</LinearLayout>
请告诉我如何使用 FragmentStatePagerAdapter 来保存 "State" 片段。我今天从上午 9 点到晚上 9 点一直在网上搜索……12 个小时……我真的需要一些帮助来解决这个问题。提前致谢!
编辑试试这个:
向您的片段添加另一个实例变量:
private String text; // this is part of saved state
在 loadPage
中设置此变量:
@Override
public void loadPage(int from, int target) {
text = txtView.getText().toString() + "\nThis message was created from" + from + " to " + target;
txtView.setText(text);
}
覆盖 onSaveInstanceState
以保存此变量:
@Override
public void onSaveInstanceState(Bundle outState);
outState.putString("text", text);
super.onSaveInstanceState(outState);
}
然后使用这个变量恢复TextView
状态:
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
if (savedInstanceState != null) {
// not null means we are restoring the fragment
text = savedInstanceState.getString("text");
} else {
text = "" + page + " - " + title;
}
View view = inflater.inflate(R.layout.frag1, container, false);
txtView = (TextView) view.findViewById(R.id.txt_frag1);
txtView.setText(text);
Button btn = (Button) view.findViewById(R.id.btn_frag1);
btn.setOnClickListener( new View.OnClickListener() {
@Override
public void onClick(View v) {
PageLoader activity = (PageLoader) getActivity();
activity.loadPage(page, page+1);
}
});
return view;
}
任何时候您希望片段中的某些内容在发生配置更改等事情时保持不变,这就是您跟踪状态、然后保存和恢复它的方式。
这是您将 onSaveInstanceState
用于片段和活动的地方。
这是您要覆盖以保存任何必要状态的方法。您在片段中更改的任何您希望在配置更改时重新创建的内容都必须保存,然后在 onCreate
或 onCreateView
期间恢复。
因此,如果您尝试恢复在 loadPage
中创建的文本,您将为该文本创建一个 class 级别的字符串,将其设置在 loadPage
中,保存在 onSaveInstanceState
覆盖中,然后在 onCreateView
中从 savedInstanceState
参数恢复。
现在更重要的是:您注意到适配器上的 getItem
在配置更改后未被调用。但是你有没有注意到你的片段还在那里(即使它不是你离开的方式)?请记住,activity 有一个 FragmentManager
管理片段及其事务。当 activity 进行配置更改时,它会保存其状态。 FragmentManager
和所有活动片段都是该状态的一部分。然后以不调用 adapter.getItem
的方式恢复片段。
事实证明,SmartFragmentPagerAdapter
并不是那么聪明。它不能在配置更改后重新创建其 registeredFragments
数组,因此它真的不是很有用。我不鼓励你使用它。
那么,当 ViewPager
已将片段的标签据为己有时,如何向页外片段发送事件?
我使用的技术是定义事件侦听器接口,并让片段注册为 activity 的侦听器。当我触发一个事件时,它是通过在 activity 上调用一个方法来通知其活动侦听器的。我在 .
中给出了一个非常完整的例子
我有 3 个片段需要放在 ViewPager 中。这些片段将保存从数据库中检索到的动态信息。我知道在方向改变时,activity 和片段会被销毁并重新创建。但我对它的名字印象深刻,FragmentStatePagerAdapter 将保存片段的状态。显然,我错了,因为每次我对片段做一些事情,然后改变方向,片段都会恢复到它在布局 xml 文件中的布局。
在调试时,我注意到在方向更改时,适配器的 getItem() 方法从未被调用 - 这意味着它没有被重新创建。那么碎片状态是怎么恢复到原来状态的呢?
如何使用 FragmentStatePagerAdapter 保存片段状态?
请注意,我一直在关注此 tutorial and used their version of the SmartFragmentStatePagerAdapter.java class 以动态管理片段。
下面是我的示例代码。
PageLoader.java - 此接口允许 MainActivity 在 运行 时间动态管理片段页面的加载。
public interface PageLoader {
void loadPage(int from, int target);
}
MainActivity.java
public class MainActivity extends AppCompatActivity implements PageLoader {
MyPagerAdapter adapter;
DirectionalViewPager pager;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// Get the ViewPager and set it's PagerAdapter so that it can display items
pager = (DirectionalViewPager) findViewById(R.id.vpPager);
adapter = new MyPagerAdapter(getSupportFragmentManager());
pager.setOffscreenPageLimit(5);
pager.setAdapter(adapter);
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
int position = pager.getCurrentItem();
if (position > 0) pager.setCurrentItem(position - 1);
return true;
}
@Override
public void loadPage(int from, int target) {
PageLoader fragment = (PageLoader) adapter.getRegisteredFragment(target);
fragment.loadPage(from, target);
}
}
MyPagerAdapter.java
public class MyPagerAdapter extends SmartFragmentStatePagerAdapter {
private static final int NUM_ITEMS = 4;
public MyPagerAdapter(FragmentManager fragmentManager) {
super(fragmentManager);
}
// Returns total number of pages
@Override
public int getCount() {
return NUM_ITEMS;
}
// Returns the fragment to display for that page
@Override
public Fragment getItem(int position) {
switch (position) {
case 0: // Fragment # 0 - This will show Frag1
return Frag1.newInstance(position, Frag1.class.getSimpleName());
case 1: // Fragment # 0 - This will show Frag1 different title
return Frag1.newInstance(position, Frag1.class.getSimpleName());
case 2: // Fragment # 1 - This will show Frag2
return Frag2.newInstance(position, Frag2.class.getSimpleName());
default:
return Frag3.newInstance(position, Frag3.class.getSimpleName());
}
}
// Returns the page title for the top indicator
@Override
public CharSequence getPageTitle(int position) {
return "Page " + position;
}
}
Frag1.java Frag2.java Frag3.java - 这些都是一样的,除了编号。
public class Frag1 extends Fragment implements PageLoader {
// Store instance variables
private String title;
private int page;
private TextView txtView;
// newInstance constructor for creating fragment with arguments
public static Frag1 newInstance(int page, String title) {
Frag1 fragmentFirst = new Frag1();
Bundle args = new Bundle();
args.putInt("someInt", page);
args.putString("someTitle", title);
fragmentFirst.setArguments(args);
return fragmentFirst;
}
// Store instance variables based on arguments passed
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
page = getArguments().getInt("someInt", 0);
title = getArguments().getString("someTitle");
}
// Inflate the view for the fragment based on layout XML
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.frag1, container, false);
txtView = (TextView) view.findViewById(R.id.txt_frag1);
txtView.setText(page + " - " + title);
Button btn = (Button) view.findViewById(R.id.btn_frag1);
btn.setOnClickListener( new View.OnClickListener() {
@Override
public void onClick(View v) {
PageLoader activity = (PageLoader) getActivity();
activity.loadPage(page, page+1);
}
});
return view;
}
@Override
public void loadPage(int from, int target) {
txtView.setText(txtView.getText() + "\nThis message was created from" + from + " to " + target);
}
}
activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<com.example.someone.smartfragmentstatepageradapter.custom.DirectionalViewPager
android:id="@+id/vpPager"
android:layout_width="match_parent"
android:layout_height="wrap_content">
</com.example.someone.smartfragmentstatepageradapter.custom.DirectionalViewPager>
</LinearLayout>
frag1.xml frag2.xml frag3.xml - 除了编号
之外,这些都是相同的<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:background="#cc2">
<TextView
android:id="@+id/txt_frag1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:text="txt_frag1"
/>
<Button
android:id="@+id/btn_frag1"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:text="btn_frag1"
android:textSize="26dp" />
</LinearLayout>
请告诉我如何使用 FragmentStatePagerAdapter 来保存 "State" 片段。我今天从上午 9 点到晚上 9 点一直在网上搜索……12 个小时……我真的需要一些帮助来解决这个问题。提前致谢!
编辑试试这个:
向您的片段添加另一个实例变量:
private String text; // this is part of saved state
在 loadPage
中设置此变量:
@Override
public void loadPage(int from, int target) {
text = txtView.getText().toString() + "\nThis message was created from" + from + " to " + target;
txtView.setText(text);
}
覆盖 onSaveInstanceState
以保存此变量:
@Override
public void onSaveInstanceState(Bundle outState);
outState.putString("text", text);
super.onSaveInstanceState(outState);
}
然后使用这个变量恢复TextView
状态:
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
Bundle savedInstanceState) {
if (savedInstanceState != null) {
// not null means we are restoring the fragment
text = savedInstanceState.getString("text");
} else {
text = "" + page + " - " + title;
}
View view = inflater.inflate(R.layout.frag1, container, false);
txtView = (TextView) view.findViewById(R.id.txt_frag1);
txtView.setText(text);
Button btn = (Button) view.findViewById(R.id.btn_frag1);
btn.setOnClickListener( new View.OnClickListener() {
@Override
public void onClick(View v) {
PageLoader activity = (PageLoader) getActivity();
activity.loadPage(page, page+1);
}
});
return view;
}
任何时候您希望片段中的某些内容在发生配置更改等事情时保持不变,这就是您跟踪状态、然后保存和恢复它的方式。
这是您将 onSaveInstanceState
用于片段和活动的地方。
这是您要覆盖以保存任何必要状态的方法。您在片段中更改的任何您希望在配置更改时重新创建的内容都必须保存,然后在 onCreate
或 onCreateView
期间恢复。
因此,如果您尝试恢复在 loadPage
中创建的文本,您将为该文本创建一个 class 级别的字符串,将其设置在 loadPage
中,保存在 onSaveInstanceState
覆盖中,然后在 onCreateView
中从 savedInstanceState
参数恢复。
现在更重要的是:您注意到适配器上的 getItem
在配置更改后未被调用。但是你有没有注意到你的片段还在那里(即使它不是你离开的方式)?请记住,activity 有一个 FragmentManager
管理片段及其事务。当 activity 进行配置更改时,它会保存其状态。 FragmentManager
和所有活动片段都是该状态的一部分。然后以不调用 adapter.getItem
的方式恢复片段。
事实证明,SmartFragmentPagerAdapter
并不是那么聪明。它不能在配置更改后重新创建其 registeredFragments
数组,因此它真的不是很有用。我不鼓励你使用它。
那么,当 ViewPager
已将片段的标签据为己有时,如何向页外片段发送事件?
我使用的技术是定义事件侦听器接口,并让片段注册为 activity 的侦听器。当我触发一个事件时,它是通过在 activity 上调用一个方法来通知其活动侦听器的。我在