12

我可以先将片段添加到视图,然后将其“分离”,然后将其“重新附加”到另一个视图吗?

在代码中,我想:

fragOne one = new fragOne();
getSupportFragmentManager().beginTransaction()
        .add(R.id.left, one, "tag").commit();

getSupportFragmentManager().beginTransaction()
        .detach(one).commit();      // or .remove(), or .addToBackStack(null).remove()

getSupportFragmentManager().executePendingTransactions();

getSupportFragmentManager().beginTransaction()
        .add(R.id.right, one).commit();

但它会引发错误:

04-05 13:28:03.492: E/AndroidRuntime(7195): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.trybackstack/com.example.trybackstack.MainActivity}: java.lang.IllegalStateException: Can't change container ID of fragment fragOne{40523130 #0 id=0x7f080000 tag}: was 2131230720 now 2131230721

感谢帮助!

4

6 回答 6

17

我遇到了同样的问题,但我发现我真正需要的是重新设置片段的视图而不是片段本身,从而避免任何片段管理器事务。

View vv = fragment.getView();
ViewGroup parent = (ViewGroup)vv.getParent();
parent.removeView(vv);
newparent.addView(vv, layoutParams);
于 2014-06-18T07:01:39.753 回答
14

在尝试了类似问题的所有答案之后,看起来我找到了解决问题的方法。

第一个问题 -在尝试将片段添加到另一个容器之前,您确实必须提交执行删除事务。谢谢你去中殿的回答

但这并非每次都有效。第二个问题是回栈。它以某种方式阻止了交易。

因此,对我有用的完整代码如下所示:

manager.popBackStackImmediate(null, manager.POP_BACK_STACK_INCLUSIVE);
manager.beginTransaction().remove(detailFragment).commit();
manager.executePendingTransactions();
manager.beginTransaction()
    .replace(R.id.content, masterFragment, masterTag)
    .add(R.id.detail, detailFragment, activeTag)
    .commit();              
于 2014-01-24T09:26:49.493 回答
8

我想您现在应该已经弄清楚了,但是我没有看到任何令人满意的答案。因此,我将其发布给将来可能会参考此内容的其他人。

如果要将片段从一个视图移动到另一个视图,请执行以下操作:

android.app.FragmentTransaction fragmentTransaction = fragmentManager.beginTransaction();
fragmentTransaction.remove(fragment1);
fragmentTransaction.commit();
fragmentManager.executePendingTransactions();
fragmentTransaction = fragmentManager.beginTransaction();
fragmentTransaction.add(R.id.containerToMoveTo, fragment1);
fragmentTransaction.commit();

这样您就不必复制片段。

于 2013-07-21T17:41:58.910 回答
3

请检查解决方案,如果要保存旧片段的状态,则需要创建同一片段的新实例并使用旧片段的状态实例化它。

 FragmentTransaction ft = mFragmentManager.beginTransaction();
    ft.remove(one);
    Fragment newInstance = fetchOldState(one);
    ft.add(R.id.right, newInstance);
    ft.commit();


//TO fetch the old state
    private Fragment fetchOldState(Fragment f)
        {
            try {
                Fragment.SavedState oldState= mFragmentManager.saveFragmentInstanceState(f);

                Fragment newInstance = f.getClass().newInstance();
                newInstance.setInitialSavedState(oldState);

                return newInstance;
            }
            catch (Exception e) // InstantiationException, IllegalAccessException
            {

            }
        }
于 2013-04-05T09:05:46.903 回答
0

我也遇到过这个问题。有时移动片段有效,有时您必须创建一个新实例。如果片段一直处于“isRemoving”状态,移动片段似乎不起作用。通过在后台堆栈中有一个片段似乎也可以防止被删除。

于 2014-01-07T11:43:00.577 回答
0

加于颜。尤金的回答。确保不要对事务应用任何转换,因为这些似乎会延迟片段的分离。

于 2014-04-22T20:58:11.343 回答