[关闭]
@xujun94 2016-07-17T16:15:43.000000Z 字数 7358 阅读 1129

常用的自定义View例子三(MultiInterfaceView多界面处理)

最近在做项目的时候,刚开始没有考虑空界面,错误界面的处理,一开始是想为每个界面在布局文件中都添加一个错误界面,空界面,但仔细一想,这样的工作量太大了,而且也不方便处理,于是我想能不能做出一个自定义控件出来,想了挺久,终于做出来了,现在将其分享出来,有什么不足的请各位指点。

本文固定链接:https://www.zybuluo.com/xujun94/note/421494
源码下载地址:
转载请注明原博客地址:

老规矩,废话不多说,大家先来看一下效果图

大家先来看一下源码

  1. package com.szl.loadinngpagedemo.view;
  2. import android.content.Context;
  3. import android.util.AttributeSet;
  4. import android.view.View;
  5. import android.widget.Button;
  6. import android.widget.FrameLayout;
  7. import com.szl.loadinngpagedemo.R;
  8. /***
  9. * 创建了自定义帧布局 把baseFragment 一部分代码 抽取到这个类中
  10. *
  11. * @author itcast
  12. */
  13. public abstract class LoadingPage extends FrameLayout {
  14. public static final int STATE_UNKOWN = 0;
  15. public static final int STATE_LOADING = 1;
  16. public static final int STATE_ERROR = 2;
  17. public static final int STATE_EMPTY = 3;
  18. public static final int STATE_SUCCESS = 4;
  19. public int state = STATE_UNKOWN;
  20. private Context mContext;
  21. private View loadingView;// 加载中的界面
  22. private View errorView;// 错误界面
  23. private View emptyView;// 空界面
  24. private View successView;// 加载成功的界面
  25. public LoadingPage(Context context) {
  26. this(context,null,0);
  27. }
  28. public LoadingPage(Context context, AttributeSet attrs) {
  29. this(context, attrs,0);
  30. }
  31. public LoadingPage(Context context, AttributeSet attrs, int defStyle) {
  32. super(context, attrs, defStyle);
  33. mContext = context;
  34. init();
  35. }
  36. private void init() {
  37. loadingView = createLoadingView(); // 创建了加载中的界面
  38. if (loadingView != null) {
  39. this.addView(loadingView, new LayoutParams(
  40. LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT));
  41. }
  42. errorView = createErrorView(); // 加载错误界面
  43. if (errorView != null) {
  44. this.addView(errorView, new LayoutParams(
  45. LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT));
  46. }
  47. emptyView = createEmptyView(); // 加载空的界面
  48. if (emptyView != null) {
  49. this.addView(emptyView, new LayoutParams(
  50. LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT));
  51. }
  52. /**
  53. * 注意这里依赖是就初始化成功的界面,也可以在状态成功的时候才初始化成功的界面
  54. */
  55. successView = createSuccessView();
  56. if(successView!=null){
  57. this.addView(successView, new LayoutParams(
  58. LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT));
  59. }
  60. showPage();// 根据不同的状态显示不同的界面
  61. }
  62. // 根据不同的状态显示不同的界面
  63. private void showPage() {
  64. if (loadingView != null) {
  65. loadingView.setVisibility(state == STATE_UNKOWN
  66. || state == STATE_LOADING ? View.VISIBLE : View.INVISIBLE);
  67. }
  68. if (errorView != null) {
  69. errorView.setVisibility(state == STATE_ERROR ? View.VISIBLE
  70. : View.INVISIBLE);
  71. }
  72. if (emptyView != null) {
  73. emptyView.setVisibility(state == STATE_EMPTY ? View.VISIBLE
  74. : View.INVISIBLE);
  75. }
  76. if (state == STATE_SUCCESS) {
  77. if (successView == null) {
  78. successView = createSuccessView();
  79. this.addView(successView, new LayoutParams(
  80. LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT));
  81. }
  82. successView.setVisibility(View.VISIBLE);
  83. } else {
  84. if (successView != null) {
  85. successView.setVisibility(View.INVISIBLE);
  86. }
  87. }
  88. }
  89. /* 创建了空的界面 */
  90. private View createEmptyView() {
  91. View view = View.inflate(mContext, R.layout.loadpage_empty,
  92. null);
  93. return view;
  94. }
  95. /* 创建了错误界面 */
  96. private View createErrorView() {
  97. View view = View.inflate(mContext, R.layout.loadpage_error,
  98. null);
  99. Button page_bt = (Button) view.findViewById(R.id.page_bt);
  100. page_bt.setOnClickListener(new OnClickListener() {
  101. @Override
  102. public void onClick(View v) {
  103. }
  104. });
  105. return view;
  106. }
  107. /* 创建加载中的界面 */
  108. private View createLoadingView() {
  109. View view = View.inflate(mContext,
  110. R.layout.loadpage_loading, null);
  111. return view;
  112. }
  113. /**
  114. * 枚举类,对应相应的状态码,用来表示各种状态
  115. */
  116. public enum LoadResult {
  117. loading(1),error(2), empty(3), success(4);
  118. int value;
  119. LoadResult(int value) {
  120. this.value = value;
  121. }
  122. public int getValue() {
  123. return value;
  124. }
  125. }
  126. /**
  127. * 注意请求成功后必须调用show方法,会根据这个压面显示相应的数据
  128. * @param loadResult
  129. */
  130. public void show(LoadResult loadResult) {
  131. state = loadResult.getValue();
  132. showPage();
  133. }
  134. /***
  135. * 创建成功的界面
  136. *
  137. * @return
  138. */
  139. public abstract View createSuccessView();
  140. }

思路解析

  1. private View loadingView;// 加载中的界面
  2. private View errorView;// 错误界面
  3. private View emptyView;// 空界面
  4. private View successView;// 加载成功的界面
  5. errorView = createErrorView(); // 加载错误界面
  6. emptyView = createEmptyView(); // 加载空的界面
  7. successView = createSuccessView();//加载成功的界面

注意在init()方法里面我们可以知道 createSuccessView()是一个抽象方法,因为每个Activity或Fragment的成功界面一般是变异羊的,我们交给子类自己去实现

  1. public int state = STATE_UNKOWN;
  2. loadingView.setVisibility(state == STATE_UNKOWN
  3. || state == STATE_LOADING ? View.VISIBLE : View.INVISIBLE);
  4. errorView.setVisibility(state == STATE_ERROR ? View.VISIBLE
  5. : View.INVISIBLE);
  6. -------

效果图如下:

当然我们只需调用void show(LoadResult loadResult)这个方法即可根据相应的状态显示相应的界面

LoadingPage的源码分析到此为止


下面我们来看一下我们是怎样结合Fragment来是用的,首先我们抽取一个BaseFragemnt,源码如下

  1. public abstract class BaseFragment extends Fragment {
  2. protected LoadingPage mLoadingPage;
  3. Context mContext;
  4. Activity mActivity;
  5. @Override
  6. public View onCreateView(LayoutInflater inflater, ViewGroup container,
  7. Bundle savedInstanceState) {
  8. mContext=getActivity();
  9. mActivity=getActivity();
  10. if (mLoadingPage == null) { // 之前的frameLayout 已经记录了一个爹了 爹是之前的ViewPager
  11. mLoadingPage = new LoadingPage(this.mContext) {
  12. @Override
  13. public View createSuccessView() {
  14. return BaseFragment.this.createSuccessView();
  15. }
  16. };
  17. } else {
  18. ViewUtils.removeParent(mLoadingPage);// 移除frameLayout之前的爹
  19. }
  20. return mLoadingPage; // 拿到当前viewPager 添加这个framelayout
  21. }
  22. @Override
  23. public void onActivityCreated(@Nullable Bundle savedInstanceState) {
  24. super.onActivityCreated(savedInstanceState);
  25. initData();
  26. }
  27. protected void initData() {
  28. }
  29. /***
  30. * 创建成功的界面
  31. *
  32. * @return
  33. */
  34. public abstract View createSuccessView();
  35. public void show(LoadResult loadResult) {
  36. if (mLoadingPage != null) {
  37. mLoadingPage.show(loadResult);
  38. }
  39. }
  40. /**
  41. * 校验数据
  42. */
  43. public LoadResult checkData(List datas) {
  44. if (datas == null) {
  45. return LoadResult.error;// 请求服务器失败
  46. } else {
  47. if (datas.size() == 0) {
  48. return LoadingPage.LoadResult.empty;
  49. } else {
  50. return LoadingPage.LoadResult.success;
  51. }
  52. }
  53. }
  54. }

BaseFragment的代码很简单,就是将我们的LoadPager这个自定义View设置为根布局,然后创建成功的界面由子类自己去实现

下面我们来看一下他的子类ErrorFragment的代码

  1. public class ErrorFragment extends BaseFragment {
  2. @Override
  3. public View createSuccessView() {
  4. return View.inflate(mContext,R.layout.fragemnt_test,null);
  5. }
  6. @Override
  7. protected void initData() {
  8. mLoadingPage.postDelayed(new Runnable() {
  9. @Override
  10. public void run() {
  11. show(LoadingPage.LoadResult.error);
  12. }
  13. },2500);
  14. }
  15. }

我们可以看到我们所做的就是创建自己的成功界面,如果想显示别的界面,就调用void show(LoadResult loadResult)这个方法,这个我们延时之后调用 show(LoadingPage.LoadResult.error);来显示错误界面

看ErrorFragment的代码以后,我们可以轻易地想到我们的EmptyFragment是怎样操作的,没错就是调用show(LoadingPage.LoadResult.empty);这个方法而已

  1. public class EmptyFragment extends BaseFragment {
  2. @Override
  3. public View createSuccessView() {
  4. return View.inflate(mContext,R.layout.fragemnt_test,null);
  5. }
  6. @Override
  7. protected void initData() {
  8. mLoadingPage.postDelayed(new Runnable() {
  9. @Override
  10. public void run() {
  11. show(LoadingPage.LoadResult.empty);
  12. }
  13. },2500);
  14. }
  15. }

同理SuccessFragemnt的代码你也能轻易的想象得到,这里就不贴出来了

下面我们来看一下BaseActivity的代码,下面只给出主要代码,详细的代码请点击 源码下载地址:

  1. /**
  2. * 博客地址:http://blog.csdn.net/gdutxiaoxu
  3. * @author xujun
  4. * @time 2016-6-29 14:52.
  5. */
  6. public abstract class BaseActivity extends AppCompatActivity {
  7. protected Context mContext;
  8. protected ProgressDialog dialog;
  9. protected LoadingPage mLoadingPage;
  10. @Override
  11. protected void onCreate(Bundle savedInstanceState) {
  12. super.onCreate(savedInstanceState);
  13. requestWindowFeature(Window.FEATURE_NO_TITLE);
  14. initWindows();
  15. // base setup
  16. mContext = this;
  17. if (mLoadingPage == null) {
  18. mLoadingPage = new LoadingPage(this) {
  19. @Override
  20. public View createSuccessView() {
  21. return BaseActivity.this.createSuccessView();
  22. }
  23. };
  24. } else {
  25. ViewUtils.removeParent(mLoadingPage);
  26. }
  27. setContentView(mLoadingPage);
  28. dialog = new ProgressDialog(this);
  29. initListener();
  30. initData();
  31. }
  32. public void show(LoadingPage.LoadResult loadResult){
  33. if(mLoadingPage!=null){
  34. mLoadingPage.show(loadResult);
  35. }
  36. }
  37. /**
  38. * 创造成功的页面
  39. *
  40. * @return
  41. */
  42. protected abstract View createSuccessView();
  43. ----

总结

思路其实很简单,一开始给LoadPager初始化各种布局,包括错误界面,加载中的界面,空界面,其中成功的界面交友子类自己去实现,如果我们想显示别的界面的话,我们只需要调用void show(LoadResult loadResult)这个方法而已

待改进的地方

  1. 由于时间关系,没有给错误界面和空界面统一集成一个自定义控件,这样我们可以利用自定义属性统一处理要显示界面的信息
  2. 没有提供更换空界面,错误界面的方法,这个很简单,大家需要的话就自己去实现就好,这里我就实现了,有时间的话会统一处理这些问题,大家有兴趣的话可以关注我github上面仓库的变化。

源码下载地址:

转载请注明原博客地址:

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注