Android Action Bar Tab with scrollview made duplicate view after orientation change

前端 未结 5 1821
执笔经年
执笔经年 2021-01-04 10:52

I have a very simple code where I use Action Bar with tab fragments. It works fine after load, but after orientation change it goes crazy. The old fragment also visible (why

相关标签:
5条回答
  • 2021-01-04 11:00

    I found a usable answer in other question.

    I need to modify my TabListener (I moved it into my Main activity class as inner class):

    private class TabListener<T extends Fragment> implements android.app.ActionBar.TabListener
        {
            private Fragment mFragment;
            private final Activity mActivity;
            private final String mTag;
            private final Class<T> mClass;
    
            /**
             * Constructor used each time a new tab is created.
             * 
             * @param activity
             *            The host Activity, used to instantiate the fragment
             * @param tag
             *            The identifier tag for the fragment
             * @param clz
             *            The fragment's Class, used to instantiate the fragment
             */
            public TabListener(final Activity activity, final String tag, final Class<T> clz)
            {
                mActivity = activity;
                mTag = tag;
                mClass = clz;
            }
    
            @Override
            public void onTabReselected(final Tab tab, final FragmentTransaction ft)
            {
                // User selected the already selected tab. Usually do nothing.
            }
    
            @Override
            public void onTabSelected(final Tab tab, final FragmentTransaction ft)
            {
                mFragment = mActivity.getFragmentManager().findFragmentByTag(mTag);
                if (mFragment == null)
                {
                    mFragment = Fragment.instantiate(mActivity, mClass.getName());
                    ft.add(android.R.id.content, mFragment, mTag);
                }
                else
                {
                    ft.attach(mFragment);
                }
            }
    
            @Override
            public void onTabUnselected(final Tab tab, final FragmentTransaction ft)
            {
                if (mFragment != null)
                {
                    ft.detach(mFragment);
                }
            }
        }
    

    So before I add fragment(again), I check it existance (and get its reference) and if it exists I attach it only.

    0 讨论(0)
  • 2021-01-04 11:04
    public void onTabSelected(Tab tab, FragmentTransaction ft)
        {
            // Check if the fragment is already initialized
            if (mFragment == null)
            {
                if(ft.findFragmentById(android.R.id.content) == null){
                // If not, instantiate and add it to the activity
                mFragment = Fragment.instantiate(mActivity, mClass.getName());
                ft.add(android.R.id.content, mFragment, mTag);
                }
            } else
            {
                // If it exists, simply attach it in order to show it
                ft.attach(mFragment);
            }
    
        }
    

    Something like that, because your problem is that your adding the same fragment twice, we just have to find where...

    0 讨论(0)
  • 2021-01-04 11:12

    Try using ft.replace(R.id.content, mFragment) in place of ft.attach(mFragment); in onTabSelected function

    0 讨论(0)
  • 2021-01-04 11:13

    I found a very simple solution to avoid fragments duplication:

        public void onTabSelected(ActionBar.Tab tab, FragmentTransaction ft) {
            Fragment currentFragment = getFragmentManager().findFragmentByTag(CURRENT_FRAGMENT_TAG);
            if (currentFragment == null || !currentFragment.getClass().equals(mFragment.getClass())) {
                ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_OPEN);
                ft.add(android.R.id.content, mFragment, CURRENT_FRAGMENT_TAG);
            }
        }
    
        public void onTabUnselected(ActionBar.Tab tab, FragmentTransaction ft) {
            ft.setTransition(FragmentTransaction.TRANSIT_FRAGMENT_CLOSE);
            ft.remove(mFragment);
        }
    

    The key of the solution is in the condition:

    currentFragment == null || !currentFragment.getClass().equals(mFragment.getClass())
    

    This condition is valid ONLY if the classes of the fragments are different. In case you have different instances of the same class you have to put an extra attribute in your fragments to recognize his function (or the association with and to make the condition !currentFragment.getClass().equals(mFragment.getClass()) true: for example you can use the FragmentTransaction tag feature.

    Bye, Alex.

    0 讨论(0)
  • 2021-01-04 11:14

    I solved this by just looking up the fragment in the tab listener constructor.

    public class TabListener<T extends Fragment> implements ActionBar.TabListener 
    {
        private Fragment fragment;
        private final SherlockFragmentActivity activity;
        private final String tag;
        private final Class<T> clazz;
    
        public TabListener(SherlockFragmentActivity activity, String tag, Class<T> clazz) 
        {
            this.activity = activity;
            this.tag = tag;
            this.clazz = clazz;
    
            FragmentManager manager = ((SherlockFragmentActivity) activity).getSupportFragmentManager();
            fragment = manager.findFragmentByTag(tag);
        }
    ...
    }
    
    0 讨论(0)
提交回复
热议问题