说明:菜鸟的学习方法只能是不断地总结,该系列文章是我在观看慕课丘吉尔的IM视频教程的一些学习记录,旨在加深自己的理解,提高自己的技术。另外,非常推荐该课程,如有兴趣可以围观该教程,肯定会很有收获,此处贴出视频教程地址:http://coding.imooc.com/learn/list/100.html
1、目的
- 封装Activity和Fragment可以为整个项目在使用Activity和Fragment将一些步骤解耦出来。此处主要封装的部分有创建和返回两部分。
- 在创建部分,封装可以将参数初始化、初始化控件、初始化数据等不同功能采用不同的方法分离开来,子类有选择的继承。
- 返回部分,需要针对Activity和Fragment之间的关系进行判断,当Fragment对返回键有响应并且拦截时执行Fragment里面的逻辑,否则执行Activity里面的逻辑。
2、Activity的封装
由于代码中自解释,因此直接贴代码了
/**
* 封装自己的Activity以在项目中统一Activity
* 分两部分封装:
* 1、封装创建过程,使之分离为初始化窗口、初始化参数、设置布局Id、初始化控件、初始化数据五个部分。
* 2、封装Activity的返回,当用户点击导航返回时直接结束当前Activity,当用户点击返回键时,判断当前
* Activity所加载的Fragment是否拦截了返回键。如果没有拦截则退出界面。
*/
public abstract class Activity extends AppCompatActivity {
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// 1、初始化窗口
initWindow();
// 2、初始化参数,如果成功则继续,否则结束当前Activity
if (initArgs(initArgs(getIntent().getExtras()))) {
// 3、设置界面布局
setContentView(getContentLayoutId());
// 4、初始化控件
initWidget();
// 5、初始化数据
initData();
} else {
finish();
}
}
/**
* 获得当前布局的Id,由子类实现
*
* @return 当前Activity的布局
*/
protected abstract int getContentLayoutId();
/**
* 初始化窗口
*/
protected void initWindow() {
}
/**
* 初始化参数
*
* @return 成功返回True,失败返回False,默认为True
*/
protected boolean initArgs(Bundle bundle) {
return true;
}
/**
* 初始化控件
*/
protected void initWidget() {
}
/**
* 初始化数据
*/
protected void initData() {
}
/**
* 当按导航键时直接退出当前Activity
*
* @return 使用父类的返回
*/
@Override
public boolean onSupportNavigateUp() {
finish();
return super.onSupportNavigateUp();
}
/**
* 当按返回键时,去遍历当前加载的Fragment,并查看Fragment是否有拦截
*/
@Override
public void onBackPressed() {
// 得到当前Activity所有的Fragment
List<android.support.v4.app.Fragment> fragments = getSupportFragmentManager().getFragments();
if (fragments != null && fragments.size() != 0)
for (Fragment fragment : fragments) {
if (fragment instanceof net.weberlisper.italker.push.common.app.Fragment) {
// 如果有拦截则直接返回
if (((net.weberlisper.italker.push.common.app.Fragment)fragment).onBackPressed())
return;
}
}
super.onBackPressed();
finish();
}
}
3、Fragment的封装
/**
* 封装Fragment,同样,需要封装Fragment的创建过程和退出过程
* 在创建的封装过程中,由于Fragment可能只是退出但并未销毁,
* 也就是说可能由onDestoryView()回到onCreateView(),
* 因此视图可以加载之前的,而不需要重新创建。
* 另一方面,某些只需要初始化一次的数据也需要另外创建。
*/
public abstract class Fragment extends android.support.v4.app.Fragment {
// 保存根视图,用于复用
private View mRoot;
Unbinder mUnbinder;
// 是否第一次初始化
private boolean mIsFirstInit = true;
@Override
public void onAttach(Context context) {
super.onAttach(context);
initArgs(getArguments());
}
/**
* 在此处初始化视图
*
* @param inflater 布局加载器
* @param container 父类控件
* @param savedInstanceState 上次保存的数据
* @return 返回加载的视图
*/
@Nullable
@Override
public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
if (mRoot == null) { // 当缓存视图为空时需要创建
View root = inflater.inflate(getContentLayoutId(), container, false);
initWidget(root);
mRoot = root;
} else {
// 当视图存在时,需要将视图从其父控件中移除,
// 以免二次加载,这是因为返回时会自动去加载到负布局
((ViewGroup) mRoot.getParent()).removeView(mRoot);
}
return mRoot;
}
@Override
public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
super.onViewCreated(view, savedInstanceState);
// 如果是第一次加载
if (mIsFirstInit) {
mIsFirstInit = false;
initFirst();
}
initData();
}
/**
* 获得当前布局的Id,由子类实现
*
* @return 当前Activity的布局
*/
protected abstract int getContentLayoutId();
/**
* 初始化参数
*/
protected void initArgs(Bundle bundle) {
}
/**
* 初始化控件
*/
protected void initWidget(View root) {
// 使用ButterKnife将当前Fragment与视图绑定在一起
mUnbinder = ButterKnife.bind(this, root);
}
/**
* 只在第一次进来时才初始化的数据
*/
protected void initFirst() {
}
/**
* 初始化数据
*/
protected void initData() {
}
/**
* 当按返回键时触发的方法
* @return 如果当前Fragment拦截返回True,否则返回False,默认为False
*/
protected boolean onBackPressed() {
return false;
}
}