首页管理 Fragment 方式

我见过很多种,写也过很多种 - 首页管理 Fragment ,这里我觉得有必要整理一下思路,为依然不是很顺手的朋友提供下思路

1. Fragment 切换方式

首页的 Fragment 是要求无缝切换的,换言之就是不能卡,要一瞬间完成,有时产品还要加上一点小动画效果

这里我们推荐 show/hide 这套组合,我们不用维护 currentFragment ,逻辑复杂, 还容易出 bug ,最好的思路是遍历 FragmentManager ,看看谁当前显示者,然后 hind,show 我们想要的就好,简单干净

    @Override
    protected void initListener() {
        super.initListener();
        navigationView.setOnNavigationItemSelectedListener(new BottomNavigationView.OnNavigationItemSelectedListener() {
            @Override
            public boolean onNavigationItemSelected(@NonNull MenuItem item) {
                switch (item.getItemId()) {
                    case R.id.menu_library:
                        switchFragment(libraryFragment);
                        break;
                    case R.id.menu_welfare:
                        if (welfareFragment == null) {
                            welfareFragment = new WelfareFragment();
                        }
                        switchFragment(welfareFragment);
                        break;
                    case R.id.menu_my:
                        if (myFragment == null) {
                            myFragment = new MyFragment();
                        }
                        switchFragment(myFragment);
                        break;
                }
                return true;
            }
        });
    private void switchFragment(Fragment fragment) {
        List<Fragment> fragments = fm.getFragments();
        for (Fragment f : fragments) {
            if (!f.equals(fragment) && !f.isHidden()) {
                fm.beginTransaction().hide(f).commit();
            }
        }
        if (fragment.isAdded()) {
            fm.beginTransaction().show(fragment).commit();
        } else {
            fm.beginTransaction().add(R.id.container, fragment).commit();
        }
    }

2. 第一个 Fragment 的处理

第一个 Fragment 我们直接在 onCreate 中初始化,addFragment 进去

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(bindLayout());

        libraryFragment = new LibraryFragment();
        addFragment(R.id.rootView, libraryFragment);
    }
    private void addFragment(int layoutID, Fragment fragment) {
        if (fragment.isAdded()) {
            fm.beginTransaction().show(fragment).commit();
        } else {
            fm.beginTransaction().add(layoutID, fragment).commit();
        }
    }

3. 首页异常关闭处理

AC 若是由系统回收,屏幕旋转或是异常关闭,会保存 app 中各种信息的,AC 堆栈数据,Fragment 堆栈数据。这里说首页,首页要是这样被关闭然后由系统再起来,那么就会恢复之前我们在首页添加的所有 Fragment 页面,并且这样页面都在显示,也就是所谓的穿透 bug

样子大概是这样的:


若是这种情况,我们需要处理 onSaveInstanceState 和 onRestoreInstanceState 2个方法

3.1 onCreate 里面 savedInstanceState == null,表示正常启动

@Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //初始化
        init();
        //根据传入的Bundle对象判断Activity是正常启动还是销毁重建
        if(savedInstanceState == null){
            //设置第一个Fragment默认选中
            setFragment(0);
        }
    }

3.2 维护一个 fragmentId 或是一个 tag(可选项)

3.3 onSaveInstanceState 中保存当前显示的 Fragment 标记,这个可以是上面自己记录的fragmentId ,也可以遍历 FragmentManage,找到其中 !isHidden(),获取其 TAG

@Override
    protected void onSaveInstanceState(Bundle outState) {
        //通过onSaveInstanceState方法保存当前显示的fragment
        outState.putInt("fragment_id",fragmentId);
        super.onSaveInstanceState(outState);
    }

3.4 onRestoreInstanceState 获取所有的 Fragment ,然后显示标记的 ,hidden 其他的

@Override
    protected void onRestoreInstanceState(Bundle savedInstanceState) {
        super.onRestoreInstanceState(savedInstanceState);
        FragmentManager mFragmentManager = getSupportFragmentManager();
        //通过FragmentManager获取保存在FragmentTransaction中的Fragment实例
        mClothesFragment = (ClothesFragment)mFragmentManager
                .findFragmentByTag("clothes_fragment");
        mFoodFragment = (FoodFragment)mFragmentManager
                .findFragmentByTag("food_fragment");
        mHotelFragment = (HotelFragment)mFragmentManager
                .findFragmentByTag("hotel_fragment");
        //恢复销毁前显示的Fragment
        setFragment(savedInstanceState.getInt("fragment_id"));
    }

具体可以参考:


4. 配合 navigationBar 的优秀写法

这里使用的是开源 navigationBar : BottomNavigation

class HomeActivity : AppCompatActivity(){

    private var lastSelectedPosition = 0

    override fun onTabSelected(position: Int) {
        lastSelectedPosition = position
        replaceFragments(position)
    }

    private fun replaceFragments(position: Int) {
        supportFragmentManager.beginTransaction().apply {
            when (position) {
                0 -> replace(R.id.home_activity_frag_container, fragment1)
                1 -> replace(R.id.home_activity_frag_container, fragment2)
                2 -> replace(R.id.home_activity_frag_container, fragment3)
                3 -> replace(R.id.home_activity_frag_container, fragment4)
                4 -> replace(R.id.home_activity_frag_container, fragment5)
                else -> replace(R.id.home_activity_frag_container, fragment6)
            }
        }.commitAllowingStateLoss()
    }

}

这个写法看着很干净,当然 replace 大家自己改改,换个方法


5. 其他

有位简 友分享了自己的 fragment 切换代码,只不过这个朋友比我们更对自己狠了一些,他想看看代码通过设计能减到多少,玉石有了一篇: 记录一次代码演变过程——35行变24行,其实很不错的,大家没事都看看,当我们平时对自己严要求之后都会有些闪光点的,记下来,因为你很快机会忘

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容