【Orbit-mvi】Android MVI 框架推荐与示例,简洁优雅的实现MVI代码结构

项目介绍

示例程序

  • 预览
  • 添加依赖项到build.gradle
scss 复制代码
implementation("org.orbit-mvi:orbit-core:9.0.0")
implementation("org.orbit-mvi:orbit-viewmodel:9.0.0")

代码结构

  1. 为页面创建一个状态类XXState,包含所有状态信息,注意state中的字段是val的,不可变的,每次页面刷新都是发出一个新的state对象
kotlin 复制代码
data class MyState(
    val isLike: Boolean = false, // 是否点赞
    val likeCount: Int = 0, // 点赞数
)
  1. 为页面创建一个副作用类XXEffect,一些一次性事件,类如Toast, Loading都放在这里
kotlin 复制代码
sealed class MyEffect {
    data class Loading(val show: Boolean, val text: String? = null) : MyEffect()
    data class Toast(val text: String) : MyEffect()
}
  1. 为页面创建ViewModel, 实现ContainerHost接口
kotlin 复制代码
class MyViewModel : ViewModel(), ContainerHost<MyState, MyEffect> {
    override val container = container<MyState, MyEffect>(MyState(likeCount = 232))

    fun like() = intent {
        if (state.isLike) {
            postSideEffect(MyEffect.Toast("已点赞,不可重复点赞")) // 发送toast事件
            return@intent
        }
        postSideEffect(MyEffect.Loading(show = true, text = "请稍候...")) // 发送loading
        delay(500) // 模拟网络请求
        reduce {
            // 在此处生成新的state对象, 以便UI观察这个新state而改变
            state.copy(isLike = true, likeCount = state.likeCount + 1)
        }
        postSideEffect(MyEffect.Loading(show = false)) // 关闭loading
        postSideEffect(MyEffect.Toast("点赞成功"))
    }
}
  1. 实现页面逻辑, 此处为Activity
kotlin 复制代码
// 监听state变化和sideEffect事件
viewModel.observe(this, state = ::render, sideEffect = ::handleEffect)

// 点击事件,此处无需过度封装为一个Intent, 直接调viewmodel函数就行
binding.tvLike.setOnClickListener {
    viewModel.like()
}

// 渲染UI
fun render(state: MyState) {
    ...
    binding.textView.text = state.likeCount.toString() // 更新点赞数
}

// 处理副作用
fun handleEffect(effect: MyEffect) {
    when (effect) {
        is MyEffect.Toast -> {
            Toast.makeText(this, effect.text, Toast.LENGTH_SHORT).show()
        }
    }
}

总结

  • 可以看到, 框架已经帮我们封装好了Kotlin协程和Flow的实现,我们只需要在viewModel中使用intent{}即可, 并且侵入性较小,只需要viewModel实现一个接口就行
  • 对于副作用,例如Toast,使用postSideEffect()发出新的Effect对象
  • 对于页面刷新,例如变更点赞数,使用reduce{},将state.copy()出一个新的对象并发出
  • 本实例程序源码: Github
相关推荐
柿蒂2 小时前
聊聊SliverPersistentHeader优先消费滑动的设计
android·flutter
假装多好1233 小时前
android三方调试几个常用命令
android·1024程序员节·三方,gms
侧耳4294 小时前
android11禁止安装apk
android·java·1024程序员节
JohnnyDeng945 小时前
ArkTs-Android 与 ArkTS (HarmonyOS) 存储目录全面对比
android·harmonyos·arkts·1024程序员节
王嘉俊9255 小时前
HarmonyOS 超级终端与服务卡片开发:打造无缝多设备交互体验
华为·架构·harmonyos·arkts·1024程序员节
2501_915918415 小时前
iOS 26 查看电池容量与健康状态 多工具组合的工程实践
android·ios·小程序·https·uni-app·iphone·webview
limingade5 小时前
手机摄像头如何识别体检的色盲检查图的数字和图案(下)
android·1024程序员节·色盲检查图·手机摄像头识别色盲图案·android识别色盲检测卡·色盲色弱检测卡
文火冰糖的硅基工坊5 小时前
[嵌入式系统-150]:智能机器人(具身智能)内部的嵌入式系统以及各自的功能、硬件架构、操作系统、软件架构
android·linux·算法·ubuntu·机器人·硬件架构
小马哥编程6 小时前
【软考架构】架构风格:RAG知识库是属于软件八大架构风格中的哪一个,黑板架构风格 ?规则系统体系风格?
大数据·计算机网络·架构·1024程序员节
2501_915909067 小时前
iOS 架构设计全解析 从MVC到MVVM与使用 开心上架 跨平台发布 免Mac
android·ios·小程序·https·uni-app·iphone·webview