FragmentStatePagerAdapter的刷新問題

衆所周知,FragmentStatePagerAdapter是谷歌官方專門爲Fragment和ViewPager推出的Adapter,其特點是爲Fragment提供緩存,避免重複加載Fragment。這個我們從源代碼就可以看得出來:

@Override
    public Object instantiateItem(ViewGroup container, int position) {
        // If we already have this item instantiated, there is nothing
        // to do.  This can happen when we are restoring the entire pager
        // from its saved state, where the fragment manager has already
        // taken care of restoring the fragments we previously had instantiated.
        if (mFragments.size() > position) {
        // 存在緩存表示該Fragment已經加載過了
        // 值得注意的是,這個地方Fragment的緩存是與position綁定在一起的
            Fragment f = mFragments.get(position);
            if (f != null) {
                return f;
            }
        }

        if (mCurTransaction == null) {
            mCurTransaction = mFragmentManager.beginTransaction();
        }

        Fragment fragment = getItem(position);
        if (DEBUG) Log.v(TAG, "Adding item #" + position + ": f=" + fragment);
        if (mSavedState.size() > position) {
            Fragment.SavedState fss = mSavedState.get(position);
            if (fss != null) {
                //Fragment未加載,但是存在緩存的狀態信息
                fragment.setInitialSavedState(fss);
            }
        }
        while (mFragments.size() <= position) {
            mFragments.add(null);
        }
        fragment.setMenuVisibility(false);
        fragment.setUserVisibleHint(false);
        mFragments.set(position, fragment);
        mCurTransaction.add(container.getId(), fragment);

        return fragment;
    }

這就是FragmentStatePagerAdapter加載緩存的代碼,其中兩個關鍵點,我都加了中文註釋。在這裏我要講的是第一個地方,也就是Fragment是和position綁定在一起的,這樣的話就會有一個問題,當我們想要隱藏或者顯示(插入)其中某一個Fragment,我們就必須要讓所有的Fragment重新走一遍生命週期。

這是因爲,想要刷新ViewPager的Fragment必須修改getItemPosition(Object object)這個方法,使其返回POSITION_NONE,但是這樣的話,所有的Fragment都會先被remove,然後再重新add。

    /**
     * Called when the host view is attempting to determine if an item's position
     * has changed. Returns {@link #POSITION_UNCHANGED} if the position of the given
     * item has not changed or {@link #POSITION_NONE} if the item is no longer present
     * in the adapter.
     */
    public int getItemPosition(Object object) {
        return POSITION_UNCHANGED;
    }

然兒這並不是我們想要,因此就有必要尋找一種替代方案。我們依然可以從getItemPosition這個方法入手,getItemPosition返回值表示的是page的位置,也就是說我們可以用這個方法來調整page的位置,而fragment也不需要重新加載。

既然知道問題在哪,只要對症下藥就好了。修改一下FragmentStatePagerAdapter的部分邏輯即可,具體代碼如下:

  /**
 * Description: Copied from android.support.v4.app.FragmentStatePagerAdapter, Fix bug for that fragments cache not refreshed when adapter items's position changed.
 * FixBug#修復Fragment的位置更新而緩存未更新導致顯示錯誤以及crash的問題
 * Author: xuqingqi
 * E-mail: [email protected]
 * Date: 2017/12/8
 */
@Keep
public abstract class FragmentStatePagerAdapterCompat extends PagerAdapter {

    private static final String TAG = FragmentStatePagerAdapterCompat.class.getSimpleName();
    private static final boolean DEBUG = false;

    private final FragmentManager mFragmentManager;
    private FragmentTransaction mCurTransaction = null;

    private SparseArray<Fragment.SavedState> mSavedState = new SparseArray<Fragment.SavedState>();
    private SparseArray<Fragment> mFragments = new SparseArray<Fragment>();
    private Fragment mCurrentPrimaryItem = null;

    public FragmentStatePagerAdapterCompat(FragmentManager fm) {
        mFragmentManager = fm;
    }

    /**
     * Return the Fragment associated with a specified position.
     */
    public abstract Fragment getItem(int position);

    /**
     * Return a unique identifier for the item at the given position.
     *
     * <p>The default implementation returns the given position.
     * Subclasses should override this method if the positions of items can change.</p>
     *
     * @param position Position within this adapter
     * @return Unique identifier for the item at position
     */
    //添加一個方法,將Fragment和Id綁定在一起而不是position
    public int getItemId(int position) {
        return position;
    }

    @Override
    public void startUpdate(ViewGroup container) {
        if (container.getId() == View.NO_ID) {
            throw new IllegalStateException("ViewPager with adapter " + this
                    + " requires a view id");
        }
    }

    @Override
    public Object instantiateItem(ViewGroup container, int position) {
        // If we already have this item instantiated, there is nothing
        // to do.  This can happen when we are restoring the entire pager
        // from its saved state, where the fragment manager has already
        // taken care of restoring the fragments we previously had instantiated.
        int itemId = getItemId(position);
        Fragment f = mFragments.get(itemId);
        if (f != null) {
            return f;
        }

        if (mCurTransaction == null) {
            mCurTransaction = mFragmentManager.beginTransaction();
        }

        Fragment fragment = getItem(position);
        if (DEBUG) Log.v(TAG, "Adding item #" + position + ": f=" + fragment);
        Fragment.SavedState fss = mSavedState.get(itemId);
        if (fss != null) {
            fragment.setInitialSavedState(fss);
        }

        fragment.setMenuVisibility(false);
        fragment.setUserVisibleHint(false);
        mFragments.put(itemId, fragment);
        mCurTransaction.add(container.getId(), fragment);

        return fragment;
    }

    @Override
    public void destroyItem(ViewGroup container, int position, Object object) {
        Fragment fragment = (Fragment) object;

        if (mCurTransaction == null) {
            mCurTransaction = mFragmentManager.beginTransaction();
        }
        if (DEBUG) Log.v(TAG, "Removing item #" + position + ": f=" + object
                + " v=" + ((Fragment)object).getView());

        int index = mFragments.indexOfValue(fragment);
        if (index >= 0) {
            int itemId = mFragments.keyAt(index);
            mSavedState.put(itemId, fragment.isAdded()
                    ? mFragmentManager.saveFragmentInstanceState(fragment) : null);
            mFragments.remove(itemId);
        }

        mCurTransaction.remove(fragment);
    }

    @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.commitNowAllowingStateLoss();
            mCurTransaction = null;
        }
    }

    @Override
    public boolean isViewFromObject(View view, Object object) {
        return ((Fragment)object).getView() == view;
    }

    @Override
    public Parcelable saveState() {
        Bundle state = null;
        if (mSavedState.size() > 0) {
            state = new Bundle();
            for (int i=0; i< mSavedState.size(); i++) {
                int itemId = mFragments.keyAt(i);
                Fragment.SavedState ss = mSavedState.get(itemId);
                if (ss != null) {
                    String key = "s" + itemId;
                    state.putParcelable(key, ss);
                }
            }
        }
        for (int i=0; i< mFragments.size(); i++) {
            int itemId = mFragments.keyAt(i);
            Fragment f = mFragments.get(itemId);
            if (f != null && f.isAdded()) {
                if (state == null) {
                    state = new Bundle();
                }
                String key = "f" + itemId;
                mFragmentManager.putFragment(state, key, f);
            }
        }
        return state;
    }

    @Override
    public void restoreState(Parcelable state, ClassLoader loader) {
        if (state != null) {
            Bundle bundle = (Bundle)state;
            bundle.setClassLoader(loader);
            mSavedState.clear();
            mFragments.clear();
            Iterable<String> keys = bundle.keySet();
            for (String key: keys) {
                if (key.startsWith("f")) {
                    int index = Integer.parseInt(key.substring(1));
                    Fragment f = mFragmentManager.getFragment(bundle, key);
                    if (f != null) {
                        f.setMenuVisibility(false);
                        mFragments.put(index, f);
                    } else {
                        Log.w(TAG, "Bad fragment at key " + key);
                    }
                }
                else if (key.startsWith("s")) {
                    int index = Integer.parseInt(key.substring(1));
                    Parcelable parcelable = bundle.getParcelable(key);
                    if (parcelable instanceof Fragment.SavedState) {
                        mSavedState.put(index, (Fragment.SavedState) parcelable);
                    }
                }
            }
        }
    }
}  
--------------------- 

原文:https://blog.csdn.net/cuenca/article/details/78851310 
 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章