23

我使用 ViewPager 和 FragmentPagerAdapter 来托管三个不同的片段

[Fragment1][Fragment2][Fragment3]

如果特定任务成功,我想要实现的是用一个全新的片段 Fragment4 成功替换 Fragment1。

当我使用..

FragmentManager fragmentManager = getSupportFragmentManager();
FragmentTransaction transaction = fragmentManager.beginTransaction();
transaction.replace(R.id.fragment1_layout_id, fragment4);
transaction.commit();

..片段被漂亮地替换,显示片段4而不是片段1。虽然当我一直滑动到 Fragment3 然后回到 Fragment4 时,Fragment1 已经卷土重来。

再说一次,如果我使用..

FragmentManager fragmentManager = getSupportFragmentManager();
FragmentTransaction transaction = fragmentManager.beginTransaction();
transaction.remove(fragment1);
transaction.commit();

..Fragment1 被删除,当我回来时 Fragment4 就在那里。所以这个解决方案的问题是我无法找到一种方法来立即显示 Fragment4 因为 Fragment1 被删除,即使我尝试过:

transaction.add(fragment4);
transaction.show(fragment4);

这是我的 FragmentPagerAdapter 实现目前的样子,没有任何事务管理

public static class PagerAdapter extends FragmentPagerAdapter {

    public PagerAdapter(FragmentManager fm) {
        super(fm);
    }

    @Override
    public int getCount() {
        return VIEW_COUNT;
    }

    @Override
    public Fragment getItem(int position) {

        Fragment fragment = null;

        switch (position) {
        case 0:
            fragment = Fragment1.newInstance(context_);
            break;

        case 1:
            fragment = Fragment2.newInstance(context_);
            break;

        case 2:
            fragment = Fragment3.newInstance(context_);
            break;

        default:
            break;
        }

        return fragment;
    }
}

编辑。

所以我的问题似乎并不完全清楚。我决定删除我之前设法创建的所有意大利面,并试图声明我已经将它们关闭(参见上面的粗体文本)。

无论如何,这几乎是我在 getItem() 中尝试做的事情:

@Override
public Fragment getItem(int position) {

    Fragment fragment = null;

    switch (position) {
    case 0:

        if (!app.isUserLoggedIn) {
            if (loginFragment_ == null) {
                loginFragment_ = LoginFragment.newInstance(context_);
                transaction_ = fragmentManager_.beginTransaction();

                if (logoutFragment_ != null) {
                    transaction_.remove(logoutFragment_);
                    logoutFragment_ = null;
                }

                transaction_.add(loginFragment_, "login");
                transaction_.commit();
            }

            if (fragmentManager_.findFragmentByTag("login") == null)
                fragment = LoginFragment.newInstance(context_);
            else
                fragment = fragmentManager_.findFragmentByTag("login");

        } else {
            if (logoutFragment_ == null) {                      
                logoutFragment_ = LogoutFragment.newInstance(context_);
                transaction_ = fragmentManager_.beginTransaction();

                if (loginFragment_ != null) {
                    transaction_.remove(loginFragment_);
                    loginFragment_ = null;
                }

                transaction_.add(logoutFragment_, "logout");
                transaction_.commit();
            }

            if (fragmentManager_.findFragmentByTag("logout") == null)
                fragment = LogoutFragment.newInstance(context_);
            else
                fragment = fragmentManager_.findFragmentByTag("logout");
        }
        break;

    case 1:
        fragment = HomeFragment.newInstance(context_);
        break;

    case 2:
        fragment = RegisterFragment.newInstance(context_);
        break;

    default:
        break;
    }

    return fragment;
}

使用这段代码我什么也没做,但如果有人看到我做错了什么并想指出我正确的方向,我会非常感激!

4

5 回答 5

12

尝试使用 FragmentStatePagerAdapter

http://android-developers.blogspot.com/2011_08_01_archive.html

因为它会在您滑动视图时删除片段(也当您调用 getitem 时),因此您不需要自己显式删除它们。

我一直遇到同样的问题,它对我有用。

我尝试您的代码并删除该if (fragment!=null)块,它可以工作。

于 2011-09-26T08:00:03.853 回答
3

mAdapter.notifyDataSetChanged()就像 Maurycy 所说,每当您想告诉 ViewPager 片段已被替换时,您都需要调用。但是,您还需要覆盖getItemPosition()适配器中的抽象函数,并在使用旧的、要隐藏的片段作为参数调用时返回 POSITION_NONE。否则,ViewPager 将保留所有当前片段。

有关更多信息和示例代码,请参阅相关问题答案

于 2012-02-03T11:58:30.480 回答
0

在某些情况下,您的getItem语句不应该返回 fragment4吗?case 0

于 2011-09-16T14:32:34.427 回答
0

嘿,我已经通过在我的 FragmentPagerAdapter 中使用 List 来处理各种片段来解决这个问题,然后我的 getItem 就是这样......

    public void addScreen(String tag, Class<?> clss, Bundle args){
        if(clss!=null){
            TabInfo info = new TabInfo(tag, clss, args);
            screens.add(info);
        }
    }
@Override
    public Fragment getItem(int position) {
        TabInfo info = screens.get(position);
        return Fragment.instantiate(context, info.clss.getName(), info.args);
    }

然后我以编程方式删除/添加/重新排列列表项(片段),然后调用处理程序

mAdapter.notifyDataSetChanged()
于 2012-01-28T01:33:39.980 回答
0

尝试将 offScreenPageLimit 设置为 5(或基于您的用例)以避免重新加载片段,以便您替换的内容将保留而不是在您滑动时重新加载。

    mViewPager.setOffscreenPageLimit(5);
于 2018-02-22T14:33:54.867 回答