我见过很多种,写也过很多种 - 首页管理 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行,其实很不错的,大家没事都看看,当我们平时对自己严要求之后都会有些闪光点的,记下来,因为你很快机会忘