compose 页面间参数正向反向传递方法

简介: compose 页面间参数正向反向传递方法

名词解释


什么是正向反向参数传递呢

正向:相当于一个Activity启动另一个Activity并向其传递参数; 反向:相当于一个Activity退出的时候向启动它的Activity回传参数;


使用SaveStateHandle实现


下面的代码是官方推荐的方法


向上级页面返回数据

/**
 * 返回指定的route并回调参数
 */
fun NavHostController.goBackRouteWithParams1(
    route: String,
    autoPop: Boolean = true,
    callback: (SavedStateHandle.() -> Unit)? = null,
) {
    val entry = getBackStackEntry(route)
    entry.savedStateHandle?.let {
        callback?.invoke(it)
    }
    if (autoPop) {
        popBackStack()
    }
}
/**
 * 回到上级页面,并回调参数
 */
fun NavHostController.goBackWithParams1(
    autoPop: Boolean = true,
    callback: (SavedStateHandle.() -> Unit)? = null,
) {
//    if (previousBackStackEntry?.arguments == null){
//        previousBackStackEntry?.arguments = Bundle()
//    }
    previousBackStackEntry?.savedStateHandle?.let {
        callback?.invoke(it)
    }
    if (autoPop) {
        popBackStack()
    }
}
复制代码

接收数据代码

val saveStateHandle = controller.currentBackStackEntryAsState().value?.savedStateHandle
saveStateHandle?.get<String>("data")//获取指定key对应的数据
复制代码


思路(此方法不是最佳答案,已经被证实不是官方推荐的方案,不看也罢)


本文主要讲反向参数传递,正向参数传递参考:正向参数传递

起先关于这个问题是没什么思路的,网上(包括google官网)也没有相关说明,以至于公司新开的项目,虽然使用compose来写的,但是反向参数传递却是自定义的一个总线框架来做的。

使用总线框架做明显是不合适的。

~~基于这样一个原因,项目完成后我决定研究一下这个问题。~~~~

我们都知道compose的页面打开关闭是使用NavHostController来进行的,那么有理由相信,如果有反向数据传递的方法必然也与NavHostController来处理。

后面的做法也简单,将所有的NavHostController中方法过一遍基本就能过滤出可能的几个。

~~通过过滤有几个方法被选中:findDestinationgetBackStackEntrypreviousBackStackEntrycontroller.currentBackStackEntryAsState()。~~~~

最终有效的方法刚好就是:getBackStackEntrypreviousBackStackEntry,通过这两个方法可以获取NavBackStackEntity,而NavBackStackEntity中有arguments:Bundle参数,通过arguments参数就可以进行数据传递。

在上级页面使用controller.currentBackStackEntryAsState()可以获取到Bundle从而拿到返回的参数。


代码


方法封装

我将回传参数的代码进行封装如下

**方法2:** `goBackWithParams` 直接返回上级页面,并回传参数
```kt
/**
 * 返回指定的route并回调参数
 */
fun NavHostController.goBackRouteWithParams(
    route: String,
    autoPop: Boolean = true,
    callback: (Bundle.() -> Unit)? = null,
) {
    getBackStackEntry(route).arguments?.let {
        callback?.invoke(it)
    }
    if (autoPop) {
        popBackStack()
    }
}
/**
 * 回到上级页面,并回调参数
 */
fun NavHostController.goBackWithParams(
    autoPop: Boolean = true,
    callback: (Bundle.() -> Unit)? = null,
) {
    previousBackStackEntry?.arguments?.let {
        callback?.invoke(it)
    }
    if (autoPop) {
        popBackStack()
    }
}
```
### 业务代码
```kt
@Composable
fun NavigateParams1View(controller: NavHostController) {
    val bundle = controller.currentBackStackEntryAsState().value
    Column(Modifier
        .fillMaxSize()
        .background(Color.Red),
        verticalArrangement = Arrangement.Center,
        horizontalAlignment = Alignment.CenterHorizontally) {
        Text(text = "展示下一级页面返回来的数据", style = TextStyle(fontSize = 20.sp))
        Box(Modifier
            .padding(horizontal = 14.dp, vertical = 20.dp)
            .size(120.dp)
            .background(color = Color.Gray, RoundedCornerShape(10.dp)),
            contentAlignment = Alignment.Center
        ) {
            Text(text = bundle?.arguments?.getString("data") ?: "未返回数据")
        }
        Button(onClick = {
            controller.navigate(navigate_param_transfer2)
        }, modifier = Modifier.padding(top = 20.dp)) {
            Text(text = "点击跳转到下一级页面")
        }
    }
}
@Composable
fun NavigateParams2View(controller: NavHostController) {
    Column(Modifier.fillMaxSize(), verticalArrangement = Arrangement.Center, horizontalAlignment = Alignment.CenterHorizontally) {
        Text(text = "展示下一级页面返回来的数据")
        Button(onClick = {
            controller.goBackRouteWithParams(navigate_param_transfer1) {
                putString("data",
                    "Hello world to you")
            }
        },modifier = Modifier.padding(top = 20.dp)) {
            Text(text = "点击跳转到下一级页面")
        }
        Button(onClick = {
            controller.goBackWithParams {
                putString("data", "Hello world to you")
            }
        },modifier = Modifier.padding(top = 20.dp)) {
            Text(text = "点击跳转到下一级页面")
        }
    }
}
```
## 实现效果
![两种方法实现反向参数传递](https://files.mdnice.com/user/15648/ee2c266e-9ace-452b-aa50-9c122ec5ea27.gif)
git: https://github.com/ananananzhuo-blog/ComposeNavigate



相关文章
|
Android开发
android Compose中沉浸式设计、导航栏、状态栏的处理
android Compose中沉浸式设计、导航栏、状态栏的处理
2705 0
android Compose中沉浸式设计、导航栏、状态栏的处理
|
Shell API Android开发
android queries属性
android queries属性
943 2
|
Dart 编译器 API
Dart笔记:Dart 库
Dart笔记:Dart 库
320 0
|
JavaScript Java Android开发
kotlin安卓在Jetpack Compose 框架下跨组件通讯EventBus
**EventBus** 是一个Android事件总线库,简化组件间通信。要使用它,首先在Gradle中添加依赖`implementation &#39;org.greenrobot:eventbus:3.3.1&#39;`。然后,可选地定义事件类如`MessageEvent`。在活动或Fragment的`onCreate`中注册订阅者,在`onDestroy`中反注册。通过`@Subscribe`注解方法处理事件,如`onMessageEvent`。发送事件使用`EventBus.getDefault().post()`。
|
缓存 数据处理 Android开发
Android经典实战之Kotlin常用的 Flow 操作符
本文介绍 Kotlin 中 `Flow` 的多种实用操作符,包括转换、过滤、聚合等,通过简洁易懂的例子展示了每个操作符的功能,如 `map`、`filter` 和 `fold` 等,帮助开发者更好地理解和运用 `Flow` 来处理异步数据流。
429 4
|
安全 JavaScript 前端开发
kotlin开发安卓app,JetPack Compose框架,给webview新增一个按钮,点击刷新网页
在Kotlin中开发Android应用,使用Jetpack Compose框架时,可以通过添加一个按钮到TopAppBar来实现WebView页面的刷新功能。按钮位于右上角,点击后调用`webViewState?.reload()`来刷新网页内容。以下是代码摘要:
|
JSON 调度 数据库
Android面试之5个Kotlin深度面试题:协程、密封类和高阶函数
本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点。文章详细解析了Kotlin中的协程、扩展函数、高阶函数、密封类及`inline`和`reified`关键字在Android开发中的应用,帮助读者更好地理解和使用这些特性。
302 1
|
JavaScript 前端开发 Android开发
kotlin安卓在Jetpack Compose 框架下使用webview , 网页中的JavaScript代码如何与native交互
在Jetpack Compose中使用Kotlin创建Webview组件,设置JavaScript交互:`@Composable`函数`ComposableWebView`加载网页并启用JavaScript。通过`addJavascriptInterface`添加`WebAppInterface`类,允许JavaScript调用Android方法如播放音频。当页面加载完成时,执行`onWebViewReady`回调。
|
Android开发 Kotlin
安卓Jetpack Compose+Kotlin, 使用ExoPlayer播放多个【本地】音频,播放完随机播放下一首,遇到播放错误,也自动播放下一首
使用Kotlin和Jetpack Compose开发的安卓应用中,实现了两个EvoPlayer同时播放res/raw目录下的音频。一个音轨播放人声(顺序播放),另一个播放背景音乐(随机播放)。每个音轨都有独立的播放和停止控制,且在播放结束或遇到错误时会自动切换到下一首。MediaPlayer置于ViewModel中,UI界面包含播放和停止按钮,控制两个音轨。每次切换音频前,还会随机调整播放速度在0.9到1.2之间。代码示例展示了如何创建ViewModel和UI以实现这一功能。
|
XML Java Android开发
Android Studio App开发之使用摄像机录制视频和从视频库中选取视频的讲解及实战(附源码)
Android Studio App开发之使用摄像机录制视频和从视频库中选取视频的讲解及实战(附源码)
940 1