Search code examples
androidandroid-fragmentsandroid-nested-fragmentfragment-lifecycle

Nested fragments created twice on orientation change


I created this example to understand the lifecycle of android fragments at screen orientation change.

MainActivity is a container for DrawerLayout, DrawerLayout allows you to choose a fragment, which will fill the MainActivity screen.

public class MainActivity extends ActionBarActivity {
...
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    Log.d("MainActivity", "onCreate savedInstanceState = "+(savedInstanceState == null ? "null" : "not null"));

...
    mDrawerList.setOnItemClickListener(new AdapterView.OnItemClickListener() {

        @Override
        public void onItemClick(AdapterView<?> parent, View view, int position, long id) {

    if (position == 0) {
            if (viewingPosition == position) {
                mDrawerLayout.closeDrawer(mDrawerList);
                return;
            }
            getSupportFragmentManager()
                    .beginTransaction()
                    .replace(R.id.contentFrame, new ParentPagerFragment(),
                                ParentPagerFragment.TAG).commit();
           viewingPosition = 0;

    }

    if (position == 1) {
            if (viewingPosition == position) {
                mDrawerLayout.closeDrawer(mDrawerList);
                return;
            }
            getSupportFragmentManager().beginTransaction()
                .replace(R.id.contentFrame, ChildTextViewFragment.newInstance("hello fragment"), ChildTextViewFragment.TAG)
                .commit();
            viewingPosition = 1;
    }
}
...

getSupportFragmentManager()
            .beginTransaction()
            .replace(R.id.contentFrame, new ParentPagerFragment(),
                    ParentPagerFragment.TAG).commit();
...

Then I have ParentPagerFragment, ParentPagerFragment contains only a ViewPager with 3 ChildTextViewFragments.

public class ParentPagerFragment extends Fragment {
...

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    Log.d(TAG, "onCreateView savedInstanceState is "+(savedInstanceState == null ? "null" : "not null"));

    View v = inflater.inflate(R.layout.fragment_pager, container, false);

    List<ChildTextViewFragment> viewFragments = new ArrayList<>();
    viewFragments.add(ChildTextViewFragment.newInstance("Fragment1"));
    viewFragments.add(ChildTextViewFragment.newInstance("Fragment2"));
    viewFragments.add(ChildTextViewFragment.newInstance("Fragment3"));

    MyPagerAdapter mPagerAdapter = new MyPagerAdapter(getChildFragmentManager(), viewFragments);

    ViewPager mViewPager = (ViewPager) v.findViewById(R.id.pager);
    mViewPager.setAdapter(mPagerAdapter);

    return v;
}

...
}


class MyPagerAdapter extends FragmentPagerAdapter {

    List<ChildTextViewFragment> viewFragments;

    public MyPagerAdapter(FragmentManager fm, List<ChildTextViewFragment> viewFragments) {
        super(fm);
        this.viewFragments = viewFragments;
    }

    @Override
    public Fragment getItem(int index) {
        return viewFragments.get(index);

    }

    @Override
    public int getCount() {
        return viewFragments.size();
    }

    @Override
    public CharSequence getPageTitle(int position){
        if(position == 0) {
            return "Fragment1";
        } else if (position == 1) {
            return "Fragment2";
        } else {
            return "Fragment3";
        }
    }

}

ChildTextViewFragment is used only to display some text

public class ChildTextViewFragment extends Fragment {
...

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {

    View view = inflater.inflate(R.layout.fragment_text, container, false);

    TextView mTextView = (TextView) view.findViewById(R.id.textView1);
    String text = getArguments().getString(TEXT_KEY);
    mTextView.setText(text);

    Log.d(TAG, text+" :: onCreateView savedInstanceState is " + (savedInstanceState == null ? "null" : "not null"));

    return view;

}

After I run this example for the first time I got these logs messages as expected:

06-01 10:34:24.154: D/MainActivity(8426): onCreate savedInstanceState = null
06-01 10:34:24.272: D/ParentPagerFragment(8426): onCreateView savedInstanceState is null
06-01 10:34:24.389: D/ChildTextViewFragment(8426): Fragment1 :: onCreateView savedInstanceState is null
06-01 10:34:24.390: D/ChildTextViewFragment(8426): Fragment2 :: onCreateView savedInstanceState is null

The surprise appeared, when I rotated the display:

06-01 10:36:15.697: D/MainActivity(8426): onCreate savedInstanceState = not null
06-01 10:36:15.713: D/ParentPagerFragment(8426): onCreateView savedInstanceState is not null
06-01 10:36:15.716: D/ChildTextViewFragment(8426): Fragment1 :: onCreateView savedInstanceState is not null
06-01 10:36:15.717: D/ChildTextViewFragment(8426): Fragment2 :: onCreateView savedInstanceState is not null

06-01 10:36:15.718: D/ParentPagerFragment(8426): onCreateView savedInstanceState is null
06-01 10:36:15.739: D/ChildTextViewFragment(8426): Fragment1 :: onCreateView savedInstanceState is null
06-01 10:36:15.740: D/ChildTextViewFragment(8426): Fragment2 :: onCreateView savedInstanceState is null

I was expecting all fragments will be restored and displayed again (like the first 4 line of this logs shows), but I don't understand, why are all the fragments created again with savedInstanceState = null ? Is this a common behavior, or am I doing something wrong ?


Solution

  • Try adding null check for savedInstanceState in your MainActivity onCreate(Bundle savedInstanceState) method like this

            if (savedInstanceState == null) {
                 getSupportFragmentManager()
                .beginTransaction()
                .replace(R.id.contentFrame, new ParentPagerFragment(),
                        ParentPagerFragment.TAG).commit();
            }