Android Dev精选案例Android

Fragment可见性及懒加载终极解决方案

2021-03-02  本文已影响0人  张坤的笔记

Fragment 有很多种使用方法,官方并没有提供一个统一的 api 来处理 Fragment 的可见性判断和回调,导致在不同的使用场景下需要使用不同的方法来判断 Fragment 的可见性。网上已经有很多讲 Fragment 可见性的文章,但是大部分文章覆盖的使用场景不够全面,有些文章的用法也过时了,因此本人梳理了当前 Fragment 的各种使用场景,提供了一个统一的 api 来处理 Fragment 的可见性。

一般使用场景

在Activity中直接使用

在 xml 文件中声明 Fragment,或者在代码中通过 FragmentTransaction 的 add 或 replace 动态载入 Fragment。这两种情况下都只要监听 Fragment 的 onResume 和 onPause 方法就能判断 Fragment 的可见性。

override fun onResume() {
    super.onResume()
 
    determineFragmentVisible()
}

override fun onPause() {
    super.onPause()
    
    determineFragmentInvisible()
}

使用show和hide控制显示和隐藏

Google 在 androidx.fragment 1.2.0 中新增了一个 FragmentContainerView,用来替代 FlameLayout 做为 Fragment 的容器,在下文中将使用 FragmentContainerView 作为 Fragment 的容器。

老的用法

通过 FragmentTransaction 的 add 将 Fragment 添加到 FragmentManager 后,Fragment 的生命周期会跟随绑定的 Activity 或父 Fragment 走到 onResume,这个时候,只要所依附的 Activity 或父 Fragment 的生命周期不发生变化,通过 FragmentTransaction 的 show 和 hide 方法控制 Fragment 的显示和隐藏并不会改变 Fragment 的生命周期,这个时候需要监听 onHiddenChanged 判断 Fragment 的可见性。

一般情况下,将 Fragment add 到 FragmentManager 的过程是在 Activity 中的 onCreate 回调中进行的,第一次回调 onHiddenChanged 是在 Fragment 回调 onCreateView 之前。如果需要在 Fragment 第一次可见的时候进行 UI 操作,就会出错,为了避免出错,需要结合 Fragment 的 onResume 和 onPause 判断 Fragment 的可见性。

override fun onHiddenChanged(hidden: Boolean) {
    super.onHiddenChanged(hidden)

    if (hidden) {
        determineFragmentInvisible()
    } else {
        determineFragmentVisible()
    }
}

override fun onResume() {
    super.onResume()
 
    determineFragmentVisible()
}

override fun onPause() {
    super.onPause()
    
    determineFragmentInvisible()
}

AndroidX用法

调用了 hide 后,接着调用 setMaxLifecycle(fragment, Lifecycle.State.STARTED),Fragment 生命周期会走到 onPause。调用 show 方法后,接着调用 setMaxLifecycle(fragment, Lifecycle.State.RESUMED),Fragment 生命周期会走到 onPause。这样只要监听 Fragment 的 onResume 和 onPause 方法就能判断 Fragment 的可见性。

override fun onResume() {
    super.onResume()
 
    determineFragmentVisible()
}

override fun onPause() {
    super.onPause()
    
    determineFragmentInvisible()
}

在ViewPager中使用

老的用法

在 support 和 androidx.fragment 1.0.0,通过监听 setUserVisibleHint 判断Fragment 的可见性。如果将 Fragment add 到 FragmentManager 的过程是在 Activity 中的 onCreate 回调中进行的,第一次回调 setUserVisibleHint 也是在 Fragment 回调 onCreateView 之前,也需要结合 Fragment 的 onResume 和 onPause 判断 Fragment 的可见性。

override fun setUserVisibleHint(isVisibleToUser: Boolean) {
    super.setUserVisibleHint(isVisibleToUser)
    
    if (isVisibleToUser) {
        determineFragmentVisible()
    } else {
        determineFragmentInvisible()
    }
}

override fun onResume() {
    super.onResume()
 
    determineFragmentVisible()
}

override fun onPause() {
    super.onPause()
    
    determineFragmentInvisible()
}

AndroidX用法

谷歌从 androidx.fragment 1.1.0 中开始,对 FragmentPagerAdapter 和 FragmentStatePagerAdapter 进行了调整,支持使用 setMaxLifecycle 控制 Fragment 的生命周期,只需要创建 Adpter 的时候, Behavior 选择 BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT

public static final int BEHAVIOR_SET_USER_VISIBLE_HINT = 0;
public static final int BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT = 1;

public void setPrimaryItem(@NonNull ViewGroup container, int position, @NonNull Object object) {
    Fragment fragment = (Fragment)object;
    if (fragment != mCurrentPrimaryItem) {
        if (mCurrentPrimaryItem != null) {
            ...
            if (mBehavior == BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT) {
                ...
                mCurTransaction.setMaxLifecycle(mCurrentPrimaryItem, Lifecycle.State.STARTED);
            } else {
                mCurrentPrimaryItem.setUserVisibleHint(false);
            }
        }
        ...
        if (mBehavior == BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT) {
            ...
            mCurTransaction.setMaxLifecycle(fragment, Lifecycle.State.RESUMED);
        } else {
            fragment.setUserVisibleHint(true);
        }
        ...
    }
}

这样只要监听 Fragment 的 onResume 和 onPause 方法就能判断 Fragment 的可见性。

override fun onResume() {
    super.onResume()
 
    determineFragmentVisible()
}

override fun onPause() {
    super.onPause()
    
    determineFragmentInvisible()
}

在ViewPager2中使用

在 ViewPager2 中使用 Fragment 时,使用的适配器是 FragmentStateAdapter,FragmentStateAdapter 内部使用 FragmentMaxLifecycleEnforcer ,FragmentMaxLifecycleEnforcer 也是通过 setMaxLifecycle 控制 Fragment 的生命周期

class FragmentStateAdapter {

    public void onAttachedToRecyclerView(@NonNull RecyclerView recyclerView) {
        ...
        mFragmentMaxLifecycleEnforcer = new FragmentMaxLifecycleEnforcer();
        ...
    }

    public void onDetachedFromRecyclerView(@NonNull RecyclerView recyclerView) {
        ...
        mFragmentMaxLifecycleEnforcer = null;
    }

    class FragmentMaxLifecycleEnforcer {

        void updateFragmentMaxLifecycle(boolean dataSetChanged) {
            ...
            for (int ix = 0; ix < mFragments.size(); ix++) {
                ...
                if (itemId != mPrimaryItemId) {
                    transaction.setMaxLifecycle(fragment, STARTED);
                } else {
                    toResume = fragment; // itemId map key, so only one can match the predicate
                }
                ...
            }
            if (toResume != null) { // in case the Fragment wasn't added yet
                transaction.setMaxLifecycle(toResume, RESUMED);
            }
            ...
        }
    }
}

这样只要监听 Fragment 的 onResume 和 onPause 方法就能判断 Fragment 的可见性。

override fun onResume() {
    super.onResume()
 
    determineFragmentVisible()
}

override fun onPause() {
    super.onPause()
    
    determineFragmentInvisible()
}

具体实现

IFragmentVisibility 中定义 Fragment 可见性相关方法:

interface IFragmentVisibility {

    /**
     * Fragment可见时调用。
     */
    fun onVisible() {}

    /**
     * Fragment不可见时调用。
     */
    fun onInvisible() {}

    /**
     * Fragment第一次可见时调用。
     */
    fun onVisibleFirst() {}

    /**
     * Fragment可见时(第一次除外)调用。
     */
    fun onVisibleExceptFirst() {}

    /**
     * Fragment当前是否对用户可见
     */
    fun isVisibleToUser(): Boolean
}

Fragment可见

Fragment 可见受到几个因素影响:Fragment 是否处于 RESUMED 状态、Fragment 是否显示、Fragment Hint 是否对用户可见,判断Fragment可见性可能会被连续调用多次,如果当前已经对用户可见,则不进行判断可见性。

// Fragment当前是否对用户可见。
private var mIsFragmentVisible = false

// Fragment当前是否是第一次对用户可见。
private var mIsFragmentVisibleFirst = true

private fun determineFragmentVisible() {
    if (isResumed && !isHidden && userVisibleHint && !mIsFragmentVisible) {
        mIsFragmentVisible = true
        onVisible()
        if (mIsFragmentVisibleFirst) {
            mIsFragmentVisibleFirst = false
            onVisibleFirst()
        } else {
            onVisibleExceptFirst()
        }
    }
}

Fragment不可见

当 Fragment 处于可见状态,调用一次 determineFragmentInvisible 方法,Fragment 就变成不可见了。

private fun determineFragmentInvisible() {
    if (mIsFragmentVisible) {
        mIsFragmentVisible = false
        onInvisible()
    }
}

Fragment嵌套

老的用法

从日志中可以看到,Fragment-1 和 Fragment-1-1 处于可见状态,但是奇怪的是 Fragment-2-1 也处于可见状态,这不符合逻辑,判断可见性逻辑还有待优化的地方。

分析日志可知,所有的 Fragment 生命周期都走到了onResume,但是 Fragment-2、Fragment-1-2、Fragment-2-2 因为 isHidden = true,判断出是不可见状态。Fragment-2-1 是 isHidden = false,但是 Fragment-2 是 isHidden = true,从逻辑上父 Fragment 不可见,子 Fragment 也应该不可见。所以在判断 Fragment 是否可见的时候,还要考虑父 Fragment 是否可见(如果存在父 Fragment)。

当从 Fragment-1 切换到 Fragment-2 后,可以看到,Fragment-1 不可见,Fragment-2 可见,但是本应该不可见的 Fragment-1-1 还是可见,本应该可见的 Fragment-2-1 还是不可见,说明判断可见性逻辑还有待优化的地方。

从 Fragment-1 切换到 Fragment-2,这两者的 onHiddenChanged 被调用了,所以它们的可见性发生了变化。Fragment-1-1 和 Fragment-2-1 没有任何操作,但是它们的可见性也应该随着父Fragment 可见性发生变化而变化,所以应该在父 Fragment 可见性变化的时候重新判断一次子 Fragment 的可见性。

AndroidX用法

全部使用 setMaxLifecycle 控制 Fragment 生命周期,可以看到 Fragment 的可见性判断是正确的。

从 Fragment-1 切换到 Fragment-2,可见性判断还是正确的。

子 Fragment 的生命周期会根据所绑定的 Activity 或父 Fragment 的生命周期变化而变化,setMaxLifecycle 改变了父 Fragment 的生命周期,子 Fragment 的生命周期自然就跟着变化了。所以,仅监听 Fragment 的 onResume 和 onPause 就能判断 Fragment 的可见性,不需要调整判断逻辑。

具体实现

在 determineFragmentVisible 中增加判断父 Fragment 是否可见的代码:

private fun determineFragmentVisible() {
    val parent = parentFragment
    if (parent != null && parent is VisibilityFragment) {
        if (!parent.isVisibleToUser()) {
            // 父Fragment不可见,子Fragment也一定不可见
            return
        }
    }
    ...
}

在 determineFragmentVisible 和 determineFragmentInvisible 增加判断子 Fragment 的可见性代码:

private fun determineFragmentVisible() {
    ...
    if (isResumed && !isHidden && userVisibleHint && !mIsFragmentVisible) {
        ...
        determineChildFragmentVisible()
    }
}

private fun determineFragmentInvisible() {
    if (mIsFragmentVisible) {
        ...
        determineChildFragmentInvisible()
    }
}

private fun determineChildFragmentVisible() {
    childFragmentManager.fragments.forEach {
        if (it is VisibilityFragment) {
            it.determineFragmentVisible()
        }
    }
}

private fun determineChildFragmentInvisible() {
    childFragmentManager.fragments.forEach {
        if (it is VisibilityFragment) {
            it.determineFragmentInvisible()
        }
    }
}

懒加载

在实现了上述功能后,对于需要懒加载功能的 Fragment,只需要重写 onVisibleFirst,在里面加载数据就可以了。

总结

对于全部使用 setMaxLifecycle 控制 Fragment 生命周期的代码,Fragment 的可见性判断相对比较简单,只要监听 Fragment 的 onResume 和 onPause 方法就能判断 Fragment 的可见性。

对于老的用法或者老的用法和 setMaxLifecycle 混用的代码,Fragment 可见性判断不仅要考虑使用方式,也要考虑父 Fragment 的可见性,同时自身可见性改变的时候,也要主动调用子 Fragment 判断可见性的代码。

项目地址

fragment-visibility,觉得用起来很爽的,请不要吝啬你的 Star !

参考

如何判断Fragment是否对用户可见

Fragment新功能,setMaxLifecycle了解一下

Androidx 下 Fragment 懒加载的新实现

上一篇 下一篇

猜你喜欢

热点阅读