iis服务器助手广告广告
返回顶部
首页 > 资讯 > 移动开发 >android使用DataBinding来设置空状态
  • 861
分享到

android使用DataBinding来设置空状态

databindingAndroid 2022-06-06 11:06:34 861人浏览 泡泡鱼
摘要

写在前面 在平时的开发之中,我们需要对于数据加载的情况进行展示: 空数据 网络异常 加载中等等情况 现在设置页面状态的方式有多种,由于笔者近期一直在使用databi

写在前面

在平时的开发之中,我们需要对于数据加载的情况进行展示:

空数据 网络异常 加载中等等情况

现在设置页面状态的方式有多种,由于笔者近期一直在使用databinding,而数据绑定通过改变模型来展示view的方式和状态页的设置也满契合的。

所以这里就讲讲使用databinding来设置Android中的各种状态页。很简单,先看看效果


首先

在app的build.gradle文件中开启databinding


android{
  ...
  dataBinding {
    enabled = true
  }
}

我们先定义一些用于状态的注解EmptyState



@IntDef({NORMAL, PROGRESS, EMPTY, NET_ERROR, NOT_AVaiLABLE})
@Retention(RetentionPolicy.SOURCE)
public @interface EmptyState {
  int NORMAL = -1; //正常
  int PROGRESS = -2;//显示进度条
  int EMPTY = 11111; //列表数据为空
  int NET_ERROR = 22222; //网络未连接
  int NOT_AVAILABLE = 33333; //服务器不可用
  //...各种页面的空状态,可以自己定义、添加
}

再自定义一个异常EmptyException用于显示我们需要的状态信息



public class EmptyException extends Exception {
  private int code;
  public EmptyException(@EmptyState int code) {
    super();
    this.code = code;
  }
  @EmptyState
  public int getCode() {
    return code;
  }
  public void setCode(@EmptyState int code) {
    this.code = code;
  }
}

现在,大多数展示状态页的控件都会提供

加载中的进度条 错误信息 空状态 ...

所以我们的目标也是显示这些

布局

以数据绑定的形式进行布局,使用StateModel来控制状态页展示的消息


<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="Http://schemas.android.com/apk/res/android"
  >
  <data>
    <import type="android.view.View"/>
    <variable
      name="stateModel"
      type="com.ditclear.app.state.StateModel"/>
  </data>
  <RelativeLayout
    android:id="@+id/rv_empty_view"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@color/background"
    android:clickable="true"
    android:focusableInTouchMode="true"
    android:visibility="@{stateModel.empty?View.VISIBLE:View.GoNE}">
    <android.support.v4.widget.ContentLoadingProgressBar
      style="?android:attr/progressBarStyle"
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:layout_centerInParent="true"
      android:visibility="@{stateModel.progress?View.VISIBLE:View.GONE}"/>
    <LinearLayout
      android:layout_width="match_parent"
      android:layout_height="match_parent"
      android:layout_alignParentBottom="true"
      android:layout_alignParentLeft="true"
      android:layout_alignParentStart="true"
      android:gravity="center"
      android:orientation="vertical"
      android:visibility="@{stateModel.progress?View.INVISIBLE:View.VISIBLE}">
      <ImageView
        android:id="@+id/none_data"
        android:layout_width="345dp"
        android:layout_height="180dp"
        android:scaleType="fitCenter"
        android:src="@{stateModel.emptyIconRes}"/>
      <TextView
        android:layout_width="wrap_content"
        android:layout_height="25dp"
        android:layout_below="@+id/none_data"
        android:layout_centerHorizontal="true"
        android:text="@{stateModel.currentStateLabel}"
        android:textSize="16sp"/>
    </LinearLayout>
  </RelativeLayout>
</layout>

布局文件中有几个方法

empty 用于控制状态页是显示还是隐藏,数据加载正常(即状态为NORMAL)的时候隐藏,否则展示 isProgress 是否显示加载中,如果显示进度条(即状态为PROGRESS),就隐藏异常页 emptyIconRes 显示状态的图片信息 currentStateLabel 显示状态的文字消息

我们定义状态的ViewModel ,就叫StateModel,来控制状态



public class StateModel extends BaseObservable {
  private Context mContext = MyApp.instance();
  @EmptyState
  private int emptyState = EmptyState.NORMAL;
  private boolean empty;
  public int getEmptyState() {
    return emptyState;
  }
  
  public void setEmptyState(@EmptyState int emptyState) {
    this.emptyState = emptyState;
    notifyChange();
  }
  
  public boolean isProgress() {
    return this.emptyState == EmptyState.PROGRESS;
  }
  
  public void bindThrowable(Throwable e) {
    if (e instanceof EmptyException) {
      @EmptyState
      int code = ((EmptyException) e).getCode();
      setEmptyState(code);
    }
  }
  public boolean isEmpty() {
    return this.emptyState != EmptyState.NORMAL;
  }
  
  @Bindable
  public String getCurrentStateLabel() {
    switch (emptyState) {
      case EmptyState.EMPTY:
        return mContext.getString(R.string.no_data);
      case EmptyState.net_ERROR:
        return mContext.getString(R.string.please_check_net_state);
      case EmptyState.NOT_AVAILABLE:
        return mContext.getString(R.string.server_not_avaliabe);
      default:
        return mContext.getString(R.string.no_data);
    }
  }
  
  @Bindable
  public Drawable getEmptyIconRes() {
    switch (emptyState) {
      case EmptyState.EMPTY:
        return ContextCompat.getDrawable(mContext, R.drawable.ic_visibility_off_green_400_48dp);
      case EmptyState.NET_ERROR:
        return ContextCompat.getDrawable(mContext, R.drawable.ic_signal_wifi_off_green_400_48dp);
      case EmptyState.NOT_AVAILABLE:
        return ContextCompat.getDrawable(mContext, R.drawable.ic_cloud_off_green_400_48dp);
      default:
        return ContextCompat.getDrawable(mContext, R.drawable.ic_visibility_off_green_400_48dp);
    }
  }
}

很普通的视图模型,主要有几个用于判断状态显示的方法

bindThrowable 根据异常显示状态 setEmptyState 方法用来设置当前的状态,通过notifyChange来通知布局文件改变

下面讲讲实际运用:

在activity或者fragment布局中,添加状态页的布局


<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto">
  <data>
    <import type="android.view.View"/>
    <variable
      name="stateModel"
      type="com.ditclear.app.state.StateModel"/>
  </data>
  <com.ditclear.app.ScrollChildSwipeRefreshLayout
    android:id="@+id/refresh_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <RelativeLayout
      android:id="@+id/container"
      android:layout_width="match_parent"
      android:layout_height="match_parent"
      android:orientation="vertical">
      <android.support.v4.widget.NestedScrollView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:fillViewport="false"
        android:overScrollMode="always"
        android:visibility="@{stateModel.empty?View.GONE:View.VISIBLE}">
        <TextView
          android:id="@+id/content_tv"
          android:layout_width="match_parent"
          android:layout_height="match_parent"
          android:gravity="center"/>
      </android.support.v4.widget.NestedScrollView>
      <include
        layout="@layout/widget_layout_empty"
        app:stateModel="@{stateModel}"/>
    </RelativeLayout>
  </com.ditclear.app.ScrollChildSwipeRefreshLayout>
</layout>

最后在activity或者fragment中我们只需要通过state.bindThrowable()和state.setEmptyState()方法便可以轻松设置各种各样的状态。


loadData().subscribe(new Subscriber<List<Contributor>>() {
  @Override
  public void onStart() {
    super.onStart();
    if (!mMainBinding.refreshLayout.isRefreshing()) {
      mStateModel.setEmptyState(EmptyState.PROGRESS);
    }
  }
  @Override
  public void onCompleted() {
    mStateModel.setEmptyState(EmptyState.NORMAL);
  }
  @Override
  public void onError(Throwable e) {
    mMainBinding.refreshLayout.setRefreshing(false);
    mStateModel.bindThrowable(e);
    Toast.makeText(MainActivity.this, mStateModel.getCurrentStateLabel(), Toast.LENGTH_SHORT).show();
  }
  @Override
  public void onNext(List<Contributor> contributors) {
    mMainBinding.refreshLayout.setRefreshing(false);
    if (contributors == null || contributors.isEmpty()) {
      onError(new EmptyException(EmptyState.EMPTY));
    } else {
      mMainBinding.contentTv.setText(contributors.toString());
    }
  }
});

写在最后

对于要使用数据来控制视图状态的,使用databinding实在是一个事半功倍的方式。而且也十分容易理解。

最后demo地址:StateBinding_jb51.rar

您可能感兴趣的文章:C#数据绑定(DataBinding)简单实现方法C#中DataBindings用法实例分析Android dataBinding与ListView及事件详解Android DataBinding的官方双向绑定示例Android 开发之dataBinding与ListView及事件Android 属性动画原理与DataBindingDataBinding onClick的七种点击方式


--结束END--

本文标题: android使用DataBinding来设置空状态

本文链接: https://www.lsjlt.com/news/28273.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作