MVI架构3--实战示例:我的收藏页面

这个例子展示一个简单的

"我的收藏"页面:包含加载数据、点击收藏(更新状态)和弹出提醒(副作用)。

1. 定义三要素(MVI 核心)

复制代码
// 1. UiState: 页面长什么样?(持久的)
data class MineUiState(
    val isLoading: Boolean = false,
    val items: List<String> = emptyList(),
    val isFavorite: Boolean = false
) : UiState

// 2. Intent: 用户想干什么?(输入)
sealed class MineIntent : Intent {
    object FetchData : MineIntent()
    object ToggleFavorite : MineIntent()
}

// 3. UiEffect: 发生一次就消失的动作(输出)
sealed class MineUiEffect : UiEffect {
    data class ShowToast(val message: String) : MineUiEffect()
}

2. ViewModel 实现(逻辑大脑)

复制代码
class MineViewModel : SimpleViewModel<MineUiState, MineIntent, MineUiEffect>() {

    override fun createInitialState() = MineUiState()

    override suspend fun handleIntent(intent: MineIntent) {
        when (intent) {
            is MineIntent.FetchData -> {
                setState { copy(isLoading = true) }
                delay(1000) // 模拟网络请求
                setState { copy(isLoading = false, items = listOf("歌单1", "歌单2")) }
            }
            is MineIntent.ToggleFavorite -> {
                val newState = !currentState.isFavorite
                setState { copy(isFavorite = newState) }
                
                // 发送一次性副作用
                val msg = if (newState) "已添加到收藏" else "已取消收藏"
                sendEffect(MineUiEffect.ShowToast(msg))
            }
        }
    }
}

3. Compose UI 实现(渲染与消费)

复制代码
@Composable
fun MineScreen(viewModel: MineViewModel) {
    val uiState by viewModel.uiState.collectAsStateWithLifecycle()
    val context = LocalContext.current

    // 【核心】监听 UiEffect:这里只会在 Effect 发送时执行一次
    LaunchedEffect(Unit) {
        viewModel.uiEffect.collect { effect ->
            when (effect) {
                is MineUiEffect.ShowToast -> {
                    Toast.makeText(context, effect.message, Toast.LENGTH_SHORT).show()
                }
            }
        }
    }

    // 布局渲染
    Column(modifier = Modifier.fillMaxSize().padding(16.dp)) {
        if (uiState.isLoading) {
            CircularProgressIndicator()
        } else {
            // 根据 UiState 显示红心颜色
            Icon(
                imageVector = if (uiState.isFavorite) Icons.Filled.Favorite else Icons.Filled.FavoriteBorder,
                contentDescription = null,
                modifier = Modifier.clickable { 
                    // 发送 Intent
                    viewModel.sendIntent(MineIntent.ToggleFavorite) 
                },
                tint = if (uiState.isFavorite) Color.Red else Color.Gray
            )
            
            uiState.items.forEach { Text(text = it) }
        }

        Button(onClick = { viewModel.sendIntent(MineIntent.FetchData) }) {
            Text("刷新数据")
        }
    }
}

为什么这样设计

  1. 旋转屏幕测试:当你点击收藏弹出 Toast 后旋转屏幕,界面会依据 uiState.isFavorite 保持红心状态,但 Toast 不会再弹出来,因为 LaunchedEffect 里的 collect 只响应新发射的 Effect。
  2. 代码位置:所有的 if-else 逻辑都在 ViewModel 里的 handleIntent,Compose 页面非常干净。
  3. 单向流动:UI (Intent) -> ViewModel (Logic) -> UI (State/Effect)。
相关推荐
zh_xuan6 小时前
Android compose测试数据双向绑定
android·compose
ChoSeitaku1 天前
Git实战|协作开发|分支设计规范|Git实践
git·设计规范
stevenzqzq3 天前
Compose 调用层参数设计规范(基于默认值复用原则)
设计规范·compose
stevenzqzq3 天前
Compose 三层结构设计规范1(基于Slot API)
设计规范·compose
bug攻城狮4 天前
SpringBoot 脚手架搭建指南:从零构建企业级开发框架
java·spring boot·后端·架构·系统架构·设计规范
Watermelo6175 天前
【前端实战】构建 Vue 全局错误处理体系,实现业务与错误的清晰解耦
前端·javascript·vue.js·信息可视化·性能优化·前端框架·设计规范
KANGBboy5 天前
埋点设计规范
设计规范
小湘西6 天前
图的分类大全
设计规范
九硕智慧建筑一体化厂家7 天前
DDC:看似普通的存在,在楼宇自控系统中却主宰智能建筑高效运行?
大数据·运维·人工智能·网络协议·制造·设计规范