Не удается сохранить вложенные фрагменты

есть ли другой способ сохранения состояния вложенного фрагмента ? Или если мы не должны этого делать, то почему ? Спасибо !

02-13 11:42:43.258: E/AndroidRuntime(7167): java.lang.IllegalStateException: Can't retain fragements that are nested in other fragments
02-13 11:42:43.258: E/AndroidRuntime(7167):     at android.support.v4.app.Fragment.setRetainInstance(Fragment.java:742)

4 ответов


можно использовать FragmentManager.saveFragmentInstanceState(Fragment) для получения состояния фрагмента. Возвращаемое значение реализует Parcelable, поэтому вы можете поместить его в пакет.

для восстановления можно указать состояние после создания фрагмента с помощью Fragment.setInitialSavedState(Fragment.SavedState).


С библиотеки поддержки 20+ (https://code.google.com/p/android/issues/detail?id=74222), есть ошибка с экземпляром recreation для дочерних фрагментов, есть исправление для него - http://ideaventure.blogspot.com.au/2014/10/nested-retained-fragment-lost-state.html

код с веб-страницы (добавьте это в свой родительский фрагмент) -

private FragmentManager childFragmentManager() {//!!!Use this instead of getFragmentManager, support library from 20+, has a bug that doesn't retain instance of nested fragments!!!!
        if(mRetainedChildFragmentManager == null) {
            mRetainedChildFragmentManager = getChildFragmentManager();
        }
        return mRetainedChildFragmentManager;
    }

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);

        if (mRetainedChildFragmentManager != null) {
            //restore the last retained child fragment manager to the new
            //created fragment
            try {
                Field childFMField = Fragment.class.getDeclaredField("mChildFragmentManager");
                childFMField.setAccessible(true);
                childFMField.set(this, mRetainedChildFragmentManager);
            } catch (NoSuchFieldException e) {
                e.printStackTrace();
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            }
        }
    }
 @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setRetainInstance(true);

    }

: mChildFrgamentManager воссоздается (https://code.google.com/p/android/issues/detail?id=74222)
Workaround: сохранение mChildFrgamentManager если фрагмент setRetainInstance(true):

@Override
public void onAttach(Activity activity) {
    super.onAttach(activity);

    if (getRetainInstance()) {
        if (mRetainedChildFragmentManager != null) {
            try {
                Field childFMField = Fragment.class.getDeclaredField("mChildFragmentManager");
                childFMField.setAccessible(true);
                childFMField.set(this, mRetainedChildFragmentManager);
            } catch (NoSuchFieldException e) {
                e.printStackTrace();
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            }
        } else {
            mRetainedChildFragmentManager = getChildFragmentManager();
        }
    }
}

предупреждение: этот код setRetainInstace стоит перед onAttach.

P. S: это немного улучшенная версия ответа @attels.


Это больше не ограничение с последней библиотекой поддержки, после этого AOSP commit.

ниже есть сообщение:

разрешить setRetainInstance (true) для вложенных фрагментов

сохранить произвольно вложенные фрагменты через изменения конфигурации как объекты nonconfiguration. Это позволяет использовать retain-instance дочерние фрагменты как произвольные непрозрачные зависимости внутри других фрагментарный.