Android MVVMHabit开发库基本使用

概述

目前,android流行的MVC、MVP模式的开发框架很多,然而一款基于MVVM模式开发框架却很少。MVVMHabit是以谷歌DataBinding+LiveData+ViewModel框架为基础,整合Okhttp+RxJava+Retrofit+Glide等流行模块,加上各种原生控件自定义的BindingAdapter,让事件与数据源完美绑定的一款容易上瘾的实用性MVVM快速开发框架。从此告别findViewById(),告别setText(),告别setOnClickListener()…

框架特点

  • 快速开发

    只需要写项目的业务逻辑,不用再去关心网络请求、权限申请、View的生命周期等问题,撸起袖子就是干。

  • 维护方便

    MVVM开发模式,低耦合,逻辑分明。Model层负责将请求的数据交给ViewModel;ViewModel层负责将请求到的数据做业务逻辑处理,最后交给View层去展示,与View一一对应;View层只负责界面绘制刷新,不处理业务逻辑,非常适合分配独立模块开发。

  • 流行框架

    retrofit+okhttp+rxJava负责网络请求;gson负责解析json数据;glide负责加载图片;rxlifecycle负责管理view的生命周期;与网络请求共存亡;rxbinding结合databinding扩展UI事件;rxpermissions负责Android 6.0权限申请;material-dialogs一个漂亮的、流畅的、可定制的material design风格的对话框。

  • 数据绑定

    满足google目前控件支持的databinding双向绑定,并扩展原控件一些不支持的数据绑定。例如将图片的url路径绑定到ImageView控件中,在BindingAdapter方法里面则使用Glide加载图片;View的OnClick事件在BindingAdapter中方法使用RxView防重复点击,再把事件回调到ViewModel层,实现xml与ViewModel之间数据和事件的绑定(框架里面部分扩展控件和回调命令使用的是@kelin原创的)。

  • 基类封装

    专门针对MVVM模式打造的BaseActivity、BaseFragment、BaseViewModel,在View层中不再需要定义ViewDataBinding和ViewModel,直接在BaseActivity、BaseFragment上限定泛型即可使用。普通界面只需要编写Fragment,然后使用ContainerActivity盛装(代理),这样就不需要每个界面都在AndroidManifest中注册一遍。

  • 全局操作

    1. 全局的Activity堆栈式管理,在程序任何地方可以打开、结束指定的Activity,一键退出应用程序。
    2. LoggingInterceptor全局拦截网络请求日志,打印Request和Response,格式化json、xml数据显示,方便与后台调试接口。
    3. 全局Cookie,支持SharedPreferences和内存两种管理模式。
    4. 通用的网络请求异常监听,根据不同的状态码或异常设置相应的message。
    5. 全局的异常捕获,程序发生异常时不会崩溃,可跳入异常界面重启应用。
    6. 全局事件回调,提供RxBus、Messenger两种回调方式。
    7. 全局任意位置一行代码实现文件下载进度监听(暂不支持多文件进度监听)。
    8. 全局点击事件防抖动处理,防止点击过快。

使用

启用databinding

在主工程app的build.gradle的android {}中加入:

dataBinding {    enabled true}

依赖Library

从远程依赖:

在根目录的build.gradle中加入

allprojects {    repositories {        ...        google()        jcenter()        maven { url 'https://jitpack.io' }    }}

在主项目app的build.gradle中依赖

dependencies {    ...    implementation 'com.github.goldze:MVVMHabit:3.1.4'}

配置AndroidManifest

添加权限:

配置Application:

继承mvvmhabit中的BaseApplication,或者调用

BaseApplication.setApplication(this);

来初始化你的Application

可以在你的自己AppApplication中配置

//是否开启日志打印KLog.init(true);//配置全局异常崩溃操作CaocConfig.Builder.create()    .backgroundMode(CaocConfig.BACKGROUND_MODE_SILENT) //背景模式,开启沉浸式    .enabled(true) //是否启动全局异常捕获    .showErrorDetails(true) //是否显示错误详细信息    .showRestartButton(true) //是否显示重启按钮    .trackActivities(true) //是否跟踪Activity    .minTimeBetweenCrashesMs(2000) //崩溃的间隔时间(毫秒)    .errorDrawable(R.mipmap.ic_launcher) //错误图标    .restartActivity(LoginActivity.class) //重新启动后的activity    //.errorActivity(YourCustomErrorActivity.class) //崩溃后的错误activity    //.eventListener(new YourCustomEventListener()) //崩溃后的错误监听    .apply();

新建文件

三个文件LoginActivty.javaLoginViewModel.javaactivity_login.xml

关联ViewModel

在activity_login.xml中关联LoginViewModel。

variable - type:类的全路径 variable - name:变量名
和传统的布局文件不同,databinding的布局文件的跟标签为layout,他有两个子标签,分别为data和原始布局标签,也就是说在原来的布局基础上多加了一层layout,然后里面又多了一个data标签。在写完布局文件之后别忘记MakeProject一下,以便系统自动生成对应的类,类名为布局文件的名字首字母大写。
<layout xmlns:android="http://schemas.android.com/apk/res/android">    <data>        <variable            type="com.goldze.mvvmhabit.ui.login.LoginViewModel"            name="viewModel"        />    data>    .....layout>
继承BaseActivity

LoginActivity继承BaseActivity

public class LoginActivity extends BaseActivity<ActivityLoginBinding, LoginViewModel> {    //ActivityLoginBinding类是databinding框架自定生成的,对activity_login.xml    @Override    public int initContentView(Bundle savedInstanceState) {        return R.layout.activity_login;    }    @Override    public int initVariableId() {        return BR.viewModel;    }    @Override    public LoginViewModel initViewModel() {        //View持有ViewModel的引用,如果没有特殊业务处理,这个方法可以不重写        return ViewModelProviders.of(this).get(LoginViewModel.class);    }}

重写BaseActivity的二个抽象方法

initContentView() 返回界面layout的id initVariableId() 返回变量的id,对应activity_login中name=“viewModel”,就像一个控件的id,可以使用R.id.xxx,这里的BR跟R文件一样,由系统生成,使用BR.xxx找到这个ViewModel的id。

选择性重写initViewModel()方法,返回ViewModel对象

@Overridepublic LoginViewModel initViewModel() {    //View持有ViewModel的引用,如果没有特殊业务处理,这个方法可以不重写    return ViewModelProviders.of(this).get(LoginViewModel.class);}

注意: 不重写initViewModel(),默认会创建LoginActivity中第二个泛型约束的LoginViewModel,如果没有指定第二个泛型,则会创建BaseViewModel

继承BaseViewModel

LoginViewModel继承BaseViewModel

public class LoginViewModel extends BaseViewModel {    public LoginViewModel(@NonNull Application application) {        super(application);    }    ....}复制代码

BaseViewModel与BaseActivity通过LiveData来处理常用UI逻辑,即可在ViewModel中使用父类的showDialog()、startActivity()等方法。在这个LoginViewModel中就可以尽情的写你的逻辑了!

数据绑定

1、传统绑定

绑定用户名:

在LoginViewModel中定义

//用户名的绑定public ObservableField<String> userName = new ObservableField<>("");

在用户名EditText标签中绑定

android:text="@={viewModel.userName}"

这样一来,输入框中输入了什么,userName.get()的内容就是什么,userName.set("")设置什么,输入框中就显示什么。 注意: @符号后面需要加=号才能达到双向绑定效果;userName需要是public的,不然viewModel无法找到它。

点击事件绑定:

在LoginViewModel中定义

//登录按钮的点击事件public View.OnClickListener loginOnClick = new View.OnClickListener() {    @Override    public void onClick(View v) {    }};

在登录按钮标签中绑定

android:onClick="@{viewModel.loginOnClick}"

2、自定义绑定

使用自定义的点击事件绑定。

在LoginViewModel中定义

//登录按钮的点击事件public BindingCommand loginOnClickCommand = new BindingCommand(new BindingAction() {    @Override    public void call() {    }});

在activity_login中定义命名空间

xmlns:binding="http://schemas.android.com/apk/res-auto"

在登录按钮标签中绑定

binding:onClickCommand="@{viewModel.loginOnClickCommand}"

使用这种形式的绑定,在原本事件绑定的基础之上,带有防重复点击的功能,1秒内多次点击也只会执行一次操作。如果不需要防重复点击,可以加入这条属性

binding:isThrottleFirst="@{Boolean.TRUE}"

自定义方法设置重复点击时间onClickCommand

使用@BindingAdapter注解来标明这是一个绑定方法。在方法中使用了RxView来增强view的clicks事件,.throttleFirst()限制订阅者在指定的时间内重复执行,最后通过BindingCommand将事件回调出去,就好比有一种拦截器,在点击时先做一下判断,然后再把事件沿着他原有的方向传递。

//防重复点击间隔(秒)public static final int CLICK_INTERVAL = 1;/*** requireAll 是意思是是否需要绑定全部参数, false为否* View的onClick事件绑定* onClickCommand 绑定的命令,* isThrottleFirst 是否开启防止过快点击*/@BindingAdapter(value = {"onClickCommand", "isThrottleFirst"}, requireAll = false)public static void onClickCommand(View view, final BindingCommand clickCommand, final boolean isThrottleFirst) {    if (isThrottleFirst) {        RxView.clicks(view)        .subscribe(new Consumer<Object>() {            @Override            public void accept(Object object) throws Exception {                if (clickCommand != null) {                    clickCommand.execute();                }            }        });    } else {        RxView.clicks(view)        .throttleFirst(CLICK_INTERVAL, TimeUnit.SECONDS)//1秒钟内只允许点击1次        .subscribe(new Consumer<Object>() {            @Override            public void accept(Object object) throws Exception {                if (clickCommand != null) {                    clickCommand.execute();                }            }        });    }}

更多相关文章

  1. [置顶] Android Touch 事件传递机制
  2. 详细讲解下Hook技术,以Hook点击事件来示范
  3. android 事件派发流程详解
  4. Android otto 事件总线的使用(使用场景)

随机推荐

  1. Android中XML绘图:Bitmap/Shape/Seletor和
  2. 同一功能在Android不同版本进行兼容的方
  3. Android从零撸美团(四) - 美团首页布局解
  4. 横竖屏切换【Android】
  5. Android 安全
  6. android EditText的自动换行和对Enter键
  7. Android简易音乐播放器之播放列表实现(第
  8. Android进阶笔记10:Android(安卓)万能适配
  9. 昨日看了一本书《Android技术内幕.系统卷
  10. Android(安卓)退出Activity