现在大多使用Fragment的场景有两种,一种是一个APP只有一个Activity+很多Fragment。另一种是APP中的每一个模块是一个Activity + 多个Fragment。最近需要用到Fragment来替代部分Activity。所以思考一下有什么方案。
实现的需求
- Fragment 能像Activity那样有一个栈,然后跳转的时候会把Fragment加入到栈中,按返回键的时候能按顺序回退栈中的Fragment。
- 退回到第一个Fragment的时候再按返回键就finish Activity。
- 在跳转的时候和回退的时候能切换标题。
- 在任意的Fragment中都有返回第一个Fragment的方法。
设计是这样的,首先定义一个接口:
public interface IFragmentCallBack {
/**
* 从一个Fragment 跳到另一个Fragment
*
* @param tag
* @param current
*/
void jump(String tag, BaseFragment current);
/**
* 返回第一个Fragment
*/
void home();
}
定义一个BaseFragmentActivity实现这个接口,在onCreate方法中:
//处理异常退出恢复情况
if (savedInstanceState != null) {
List<Fragment> fragments = getSupportFragmentManager().getFragments();
if (fragments != null && fragments.size() > 0) {
boolean showFlag = false;
FragmentTransaction ft = getSupportFragmentManager().beginTransaction();
for (int i = fragments.size() - 1; i >= 0; i--) {
Fragment fragment = fragments.get(i);
if (fragment != null) {
if (!showFlag) {
ft.show(fragments.get(i));
showFlag = true;
} else {
ft.hide(fragments.get(i));
}
}
}
ft.commit();
}
}
上面代码是处理Activity异常退出时恢复Fragment并且防止Fragment显示重叠的。
实现的接口的方法中,有home()
和jump()
,home的主要用能是跳回第一个Fragment,jump是用来通过tag跳转Fragment的,
home 实现如下:
@Override
public void home() {
while (getSupportFragmentManager().getBackStackEntryCount() != 1) {
getSupportFragmentManager().popBackStackImmediate();
}
}
jump 实现如下
@Override
public void jump(String tag, BaseFragment current) {
FragmentTransaction transaction = getSupportFragmentManager().beginTransaction();
HashMap<String, FragmentModule> modules = getFragmentModule();
BaseFragment target = (BaseFragment) getSupportFragmentManager().findFragmentByTag(tag);
if (target == null) {
target = modules.get(tag).getFragment();
transaction.hide(current)
.add(getContentId(), target, tag)
.addToBackStack(null)
.commit();
} else {
transaction.hide(current).show(target).commit();
}
setTitle(modules.get(tag).getTitle());
}
这里使用了add
、show
、hide
方法,把Fragment添加到BackStack中,方便后面回退。上面使用了FragmentModule
和 getFragmentModule
、getContentId
,getFragmentModule
是抽象方法需要子类自己实现,getContentId
也是抽象的,用于提供显示Framgent的布局id。
FragmentModule:
public class FragmentModule {
public BaseFragment fragment;
public String title;
public FragmentModule(BaseFragment fragment, String title) {
this.fragment = fragment;
this.title = title;
}
public BaseFragment getFragment() {
return fragment;
}
public String getTitle() {
return title;
}
}
需要添加回退功能。
/**
* 后退的时候按栈回退
*/
@Override
public void onBackPressed() {
if (getSupportFragmentManager().getBackStackEntryCount() != 1) {
getSupportFragmentManager().popBackStackImmediate();
} else {
super.onBackPressed();
}
}
回退的时候使用popBackStackImmediate
方法出栈,注意这里不能使用popBackStack
否则 getBackStackEntryCount
可能返回不正确的值。
接下来需要设置页面默认的Fragment
,也就是第一个Fragment
,在onCreate
中调用:
FragmentModule defaultModule = getDefaultModule();
setTitle(defaultModule.getTitle());
getSupportFragmentManager().beginTransaction().add(getContentId(), defaultModule.getFragment(),
tag).addToBackStack(null).commit();
getDefaultModule
是一个抽象方法,用于返回默认的Fragment
。
上面基本上就搭建完毕。
使用的时候,Activity
继承BaseFragmentActivity
,然后实现getDefaultMoudle()
、getContentId
、getFragmentModule
方法
getContentId
返回布局的ID,Fragment
会在该布局中显示getFragmentModule
返回一个HashMap
,key
为fragment
的tag
,值为FragmentModule
。getDefaultMoudle
直接new 一个FragmentModule
即可。
最后,Fragment
需要提供一个注入IFragmentCallback
的构造方法。然后在Fragment
中调用 callback.jump(tag,this)
就可以跳转了,callback.home()
可以回到第一个页面。
如果回退的时候需要改变标题需要重写Fragment的onHiddenChanged(boolean hidden)
方法。
如果这些Fragment中还需要嵌套的话,比如一个Fragment(A)里面有ViewPager,ViewPager中又嵌套了几个Fragment的话,则A在调度ViewPager中的Fragment的时候,需要使用getChildFragmentManager
,不能直接getSupportFragmentManager
,否则会出现Fragment重叠的问题。
这种方案没有在Fragment需要加载刷新大量数据下实现过,应用于登陆、注册这些静态模块没有问题,如果要使用在复杂场景下,可能会出现未知的坑~~~~