FragmentStatePagerAdapter 是否在方向改变时保存片段状态?

cha*_*ser 5 android android-fragments android-viewpager fragmentstatepageradapter

我有 3 个片段需要在 ViewPager 中。这些片段将保存从数据库中检索到的动态信息。我知道在方向改变时,活动和片段会被破坏和重新创建。但我对它的名字印象是,FragmentStatePagerAdapter 将保存片段的状态。显然,我错了,因为每次我对片段做一些事情,然后改变方向时,片段就会恢复到它在布局 xml 文件中的布局方式。

在调试时,我注意到在方向更改时,从未调用 Adapter 的 getItem() 方法 - 这意味着它没有重新创建。那么片段状态是如何恢复到其原始状态的呢?

如何使用 FragmentStatePagerAdapter 保存片段状态?

请注意,我一直在关注本教程并使用他们版本的SmartFragmentStatePagerAdapter.java类来动态管理片段。

以下是我的示例代码。

PageLoader.java - 此接口允许 MainActivity 在运行时动态管理片段页面的加载。

public interface PageLoader {
    void loadPage(int from, int target);
}
Run Code Online (Sandbox Code Playgroud)

主活动.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);
    }
}
Run Code Online (Sandbox Code Playgroud)

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;

    }
}
Run Code Online (Sandbox Code Playgroud)

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);
    }
}
Run Code Online (Sandbox Code Playgroud)

活动_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>
Run Code Online (Sandbox Code Playgroud)

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>
Run Code Online (Sandbox Code Playgroud)

请告诉我如何使用 FragmentStatePagerAdapter 来保存片段的“状态”。我今天从早上 9 点到晚上 9 点一直在网上搜索……12 个小时……我真的需要一些帮助来解决这个问题。提前致谢!

kri*_*son 3

编辑试试这个:

将另一个实例变量添加到您的片段中:

    private String text;   // this is part of saved state
Run Code Online (Sandbox Code Playgroud)

设置此变量loadPage

    @Override
    public void loadPage(int from, int target) {
        text = txtView.getText().toString() + "\nThis message was created from" + from + " to " + target;
        txtView.setText(text);
    }
Run Code Online (Sandbox Code Playgroud)

覆盖onSaveInstanceState以保存此变量:

    @Override
    public void onSaveInstanceState(Bundle outState);

        outState.putString("text", text);
        super.onSaveInstanceState(outState);
    }
Run Code Online (Sandbox Code Playgroud)

然后使用此变量恢复状态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;
    }
Run Code Online (Sandbox Code Playgroud)

每当发生配置更改等情况时,您希望片段中的某些内容保持不变,这就是您跟踪状态,然后保存并恢复它的方式。


这是您用于onSaveInstanceState片段和活动的地方。

这是您将重写以保存任何必要状态的方法。您在片段中所做的任何更改(如果您希望在配置更改时重新创建)都必须保存,然后在onCreate或期间恢复onCreateView

因此,如果您尝试恢复在 中创建的文本loadPage,则可以为该文本创建一个类级字符串,将其设置在 中loadPage,将其保存在onSaveInstanceState覆盖中,然后从onCreateView参数中恢复savedInstanceState

现在,关键是:您注意到getItem在配置更改后不会调用适配器上的内容。但是您是否注意到您的碎片仍然在那里(即使它不是您留下的方式)?请记住,该活动有一个FragmentManager正在管理片段及其事务。当活动进行配置更改时,它会保存其状态。所有FragmentManager活动片段都是该状态的一部分。adapter.getItem然后以不调用的方式恢复碎片。

事实证明,这SmartFragmentPagerAdapter并不是那么聪明。它无法registeredFragments在配置更改后重新创建其数组,因此它实际上不是很有用。我会劝阻你不要使用它。

ViewPager那么,当片段的标签自用时,如何将事件发送到页外片段呢?

我使用的技术是定义事件侦听器接口,并将片段注册为活动的侦听器。当我触发事件时,它是通过调用活动上的方法来通知其活动侦听器的。我在这个答案中给出了一个非常完整的例子。