Android - сохранение объектов в фрагменте пользовательского интерфейса

В моем проекте я изначально использовал эту схему:

Активность A (UI) > Фрагмент B (не-пользовательский интерфейс) > Адаптер + AsyncTask

  • B сохраняется setRetainInstance(true) и обновляется A UI
  • A экземпляр B с getFragmentManager()

Но теперь мне нужно использовать фрагменты пользовательского интерфейса, поэтому я подумал использовать эту схему:

Активность A > Фрагмент B (UI) > Фрагмент C (не-пользовательский интерфейс) > Адаптер + AsyncTask

  • C сохраняется setRetainInstance(true) и обновляется B UI
  • B создать экземпляр C с помощью getChildFragmentManager()

Но это кажется невозможным:

Caused by: java.lang.IllegalStateException:
    Can't retain fragements that are nested in other fragments

Я попытался создать экземпляр C с getFragmentManager() вместо getChildFragmentManager(), но у меня были некоторые проблемы.


Какова наилучшая практика сохранения объектов и управления AsyncTasks во фрагменте?

Ответ 1

Наконец-то я разработаю небольшой вспомогательный класс:

public class RetainFragment<F extends Fragment> extends Fragment {

    private static <F extends Fragment> String tag(F ui, Object id) {
        return ui.getClass().getName() + id.toString();
    }

    public static <F extends Fragment> Fragment get(F ui, Object id) {
        return ui.getFragmentManager().findFragmentByTag(tag(ui, id));
    }

    public F ui;
    public Object id;

    @Override
    public void onCreate(Bundle state) {
        super.onCreate(state);
        setRetainInstance(true);
    }

    public void link(F ui, Object id) {
        this.id = id;
        this.ui = ui;
        if (get(ui, id) == null) {
            ui.getFragmentManager()
              .beginTransaction()
              .add(this, tag(ui, id))
              .commit();
        }
    }

    public void unlink() {
        if (get(ui, id) != null) {
            ui.getFragmentManager()
              .beginTransaction()
              .remove(this)
              .commit();
        }
        ui = null;
        id = null;
    }

}

И теперь, в фрагментах пользовательского интерфейса, я пишу только:

public class UiFragment extends Fragment {

    private static class Retain extends RetainFragment<UiFragment> {
        private MyAsyncTask task;   // Objects to retain over
        private MyAdapter adapter;  // configuration changes
    }

    private Retain retain;

    @Override
    public void onAttach(Activity activity) {
        super.onAttach(activity);
        retain = (Retain) Retain.get(this, 0);
        if (retain == null) {
            retain = new Retain();
            // Retained objects initialization
            retain.adapter = new MyAdapter();
        }
        retain.link(this, 0);
    }

    @Override
    public void onDetach() {
        if (isRemoving()) {
            // Retained objects terminate
            if (retain.task != null) {
                retain.task.cancel(true);
            }
            retain.unlink();
        }
        super.onDetach();
    }

    private void update() {
        if (!isDetached()) {
            // Update UI fragment
        }
    }

}

И в retain.task.onPostExecute():

retain.ui.update(); // Reference to the last ui fragment instance linked

И запустить задачу при запуске фрагмента пользовательского интерфейса:

private static class Retain extends RetainFragment<UiFragment> {
    private MyAsyncTask task;   // Objects to retain over
    private MyAdapter adapter;  // configuration changes

    @Override
    public void onCreate(Bundle state) {
        super.onCreate(state);
        ui.onRetainCreated();
    }
}

private void onRetainCreated() {
    retain.task = new MyAsyncTask() {
        @Override
        protected void onPostExecute(Void result) {
            retain.ui.update();
        }
    };
    retain.task.execute();
}