【Android】保存Fragment切換狀態



前言

一般頻繁切換Fragment會導致頻繁的釋放和創建,如果Fragment比較臃腫體驗就非常不好了,這里分享一個方法。

 

聲明

歡迎轉載,但請保留文章原始出處:) 
博客園:http://www.cnblogs.com
農民伯伯: http://over140.cnblogs.com 

 

正文

一、應用場景

1、不使用ViewPager

2、不能用replace來切換Fragment,會導致Fragment釋放(調用onDestroyView)

 

二、實現

1、xml

< LinearLayout  xmlns:android ="http://schemas.android.com/apk/res/android"
    android:layout_width
="match_parent"
    android:layout_height
="match_parent"
    android:orientation
="vertical"   >

     < FrameLayout
        
android:id ="@+id/container"
        android:layout_width
="match_parent"
        android:layout_height
="0dip"
        android:layout_weight
="1.0"   >
     </ FrameLayout >

     < RadioGroup
        
android:id ="@+id/main_radio"
        android:layout_width
="fill_parent"
        android:layout_height
="wrap_content"
        android:layout_gravity
="bottom"
        android:gravity
="bottom"
        android:layout_marginBottom
="-6dp"
        android:orientation
="horizontal"   >

         < RadioButton
            
android:id ="@+id/radio_button0"
            style
="@style/main_tab_bottom"
            android:drawableTop
="@drawable/bottom_1"   />

         < RadioButton
            
android:id ="@+id/radio_button1"
            style
="@style/main_tab_bottom"
            android:drawableTop
="@drawable/bottom_2"   />

         < RadioButton
            
android:id ="@+id/radio_button2"
            style
="@style/main_tab_bottom"
            android:drawableTop
="@drawable/bottom_3"   />

         < RadioButton
            
android:id ="@+id/radio_button3"
            style
="@style/main_tab_bottom"
            android:drawableTop
="@drawable/bottom_4"   />

         < RadioButton
            
android:id ="@+id/radio_button4"
            style
="@style/main_tab_bottom"
            android:drawableTop
="@drawable/bottom_5"   />
     </ RadioGroup >

</ LinearLayout >

代碼說明:

非常常見的底部放5個RadioButton,點擊切換不同的Fragment。

 

2、Activity

為RadioButton設置setOnCheckedChangeListener事件,其他代碼:

    @Override
     public  void onCheckedChanged(CompoundButton buttonView,  boolean isChecked) {
         if (isChecked) {
            Fragment fragment = (Fragment) mFragmentPagerAdapter.instantiateItem(mContainer, buttonView.getId());
            mFragmentPagerAdapter.setPrimaryItem(mContainer, 0, fragment);
            mFragmentPagerAdapter.finishUpdate(mContainer);
        } 
    }

     private FragmentPagerAdapter mFragmentPagerAdapter =  new FragmentPagerAdapter(getSupportFragmentManager()) {

        @Override
         public Fragment getItem( int position) {
             switch (position) {
             case R.id.radio_button1:
                 return  new Fragment1();
             case R.id.radio_button2:
                 return  new Fragment2();
             case R.id.radio_button3:
                 return  new Fragment3();
             case R.id.radio_button4:
                 return  new Fragment4();
             case R.id.radio_button0:
             default:
                 return  new Fragment0();
            }
        }

        @Override
         public  int getCount() {
             return 5;
        }
    };

代碼說明:

instantiateItem從FragmentManager中查找Fragment,找不到就getItem新建一個,setPrimaryItem設置隱藏和顯示,最后finishUpdate提交事務。

mContainer就是xml中的FrameLayout。 

 

三、FragmentPagerAdapter核心代碼

    @Override
     public Object instantiateItem(ViewGroup container,  int position) {
         if (mCurTransaction ==  null) {
            mCurTransaction = mFragmentManager.beginTransaction();
        }

         final  long itemId = getItemId(position);

         //  Do we already have this fragment?
        String name = makeFragmentName(container.getId(), itemId);
        Fragment fragment = mFragmentManager.findFragmentByTag(name);
         if (fragment !=  null) {
             if (DEBUG) Log.v(TAG, "Attaching item #" + itemId + ": f=" + fragment);
            mCurTransaction.attach(fragment);
        }  else {
            fragment = getItem(position);
             if (DEBUG) Log.v(TAG, "Adding item #" + itemId + ": f=" + fragment);
            mCurTransaction.add(container.getId(), fragment,
                    makeFragmentName(container.getId(), itemId));
        }
         if (fragment != mCurrentPrimaryItem) {
            fragment.setMenuVisibility( false);
            fragment.setUserVisibleHint( false);
        }

         return fragment;
    }

    @Override
     public  void destroyItem(ViewGroup container,  int position, Object object) {
         if (mCurTransaction ==  null) {
            mCurTransaction = mFragmentManager.beginTransaction();
        }
         if (DEBUG) Log.v(TAG, "Detaching item #" + getItemId(position) + ": f=" + object
                + " v=" + ((Fragment)object).getView());
        mCurTransaction.detach((Fragment)object);
    }

    @Override
     public  void setPrimaryItem(ViewGroup container,  int position, Object object) {
        Fragment fragment = (Fragment)object;
         if (fragment != mCurrentPrimaryItem) {
             if (mCurrentPrimaryItem !=  null) {
                mCurrentPrimaryItem.setMenuVisibility( false);
                mCurrentPrimaryItem.setUserVisibleHint( false);
            }
             if (fragment !=  null) {
                fragment.setMenuVisibility( true);
                fragment.setUserVisibleHint( true);
            }
            mCurrentPrimaryItem = fragment;
        }
    }

    @Override
     public  void finishUpdate(ViewGroup container) {
         if (mCurTransaction !=  null) {
            mCurTransaction.commitAllowingStateLoss();
            mCurTransaction =  null;
            mFragmentManager.executePendingTransactions();
        }
    }

 FragmentPagerAdapter是support包自帶的類。

 

四、注意 

之前自己模擬ViewPager用attach、setMenuVisibility、setUserVisibleHint來控制Fragment的顯示隱藏,經常會出現Fragment重疊現象,非常頭疼,換了這個之后目前沒有發現重疊現象。

 

五、文章后期維護

2013-12-01  上傳示例代碼:http://files.cnblogs.com/over140/SampleFragmentSwitch.zip

 

    @Override
     public  void setMenuVisibility( boolean menuVisible) {
         super.setMenuVisibility(menuVisible);
         if ( this.getView() !=  null)
             this.getView().setVisibility(menuVisible ? View.VISIBLE : View.GONE);
    }

重新做例子時發現自己也出不來效果了,后來發現少了這段代碼。

 

2014-01-08 想實現本文的效果還是推薦直接使用ViewPager,通過自定義ViewPager禁用掉左右滑動和自動銷毀即可,根據評論來看非正常情況下重影現象還是挺嚴重的。 

 

結束

需要多看看源碼,才能很好的解決問題。 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM