Jetpack Compose Navigation - pass argument to startDestination(Jetpack合成导航-将参数传递给startDestination)
问题描述
我正在构建的应用程序使用带有路线的组合导航。挑战在于起始目标是动态的。
这里是一个最小的示例:
class MainActivity : ComponentActivity()
{
override fun onCreate(savedInstanceState: Bundle?)
{
super.onCreate(savedInstanceState)
setContent {
val navController = rememberNavController()
NavHost(
navController = navController,
startDestination = "dynamic/1", // doesn't work
// startDestination = "static", // workaround
) {
composable(
route = "dynamic/{$ARG_ID}",
arguments = listOf(navArgument(ARG_ID) { type = NavType.StringType }),
) {
val id = it.arguments?.getString(ARG_ID)
Text("dynamic route, received argument: $id!")
}
// part of the workaround
// composable(
// route = "static",
// ) {
// LaunchedEffect(this) {
// navController.navigate("dynamic/1")
// }
// }
}
}
}
companion object
{
const val ARG_ID = "id"
}
}
应用程序崩溃,
java.lang.IllegalArgumentException: navigation destination route/1 is not a direct child of this NavGraph
仅当";动态";路由用作起始目标时才会出现此问题。这可以通过使用startDestination = "static"
进行验证。
虽然";静态";路由解决方法可行,但我正在寻找不使用它的解决方案,因为它会混淆代码,还会在后端堆栈中创建一个额外的条目。
->;Full code sample复制问题
相关SO问题
- Navigation Architecture Component- Passing argument data to the startDestination-答案似乎不适用于撰写导航。
- Pass an argument to a nested navigation graph in Jetpack Compose-没有应答。
- Compose Navigation - navigation destination ... is not a direct child of this NavGraph-接受的答案不能解决问题。
编辑:
我想强调的是,原始样本过去不包含";静态&可组合。我添加";静态";Composable只是为了让startDestination
正常工作,并证明可以导航到";Dynamic";Composable。
更新:
即使切换到query parameter syntax for optional arguments、提供默认值并在没有任何参数的情况下设置开始目标也不起作用。
以下变体
NavHost(
navController = navController,
startDestination = "dynamic",
) {
composable(
route = "dynamic?$ARG_ID={$ARG_ID}",
arguments = listOf(navArgument(ARG_ID) { type = NavType.StringType; defaultValue = "1" }),
) {
val id = it.arguments?.getString(ARG_ID)
Text("dynamic route, received argument: $id!")
}
}
导致异常
java.lang.IllegalArgumentException: navigation destination dynamic is not a direct child of this NavGraph
推荐答案
全部归功于ianhanniballake,他在评论中向我解释了解决方案。我将在这里展示我的代码示例的工作版本。
对我来说最大的洞察是:
startDestination
在模式匹配的意义上不能与可组合的route
匹配,但它必须是完全相同的字符串。
这意味着参数不能直接通过startDestination
设置,而必须通过参数的defaultValue
设置。
以下是工作示例:
class MainActivity : ComponentActivity()
{
override fun onCreate(savedInstanceState: Bundle?)
{
super.onCreate(savedInstanceState)
setContent {
val navController = rememberNavController()
NavHost(
navController = navController,
// 1st change: Set startDestination to the exact string of route
startDestination = "dynamic/{$ARG_ID}", // NOT "dynamic/1", provide arguments via defaultValue
) {
composable(
route = "dynamic/{$ARG_ID}",
// 2nd change: Set startDestination argument via defaultValue
arguments = listOf(navArgument(ARG_ID) { type = NavType.StringType; defaultValue = "1" }),
) {
val id = it.arguments?.getString(ARG_ID)
Text("dynamic route, received argument: $id!")
}
}
}
}
companion object
{
const val ARG_ID = "id"
}
}
该方法同样适用于以查询参数形式提供的参数。
老实说,我认为这是一个很小的限制,因为开始路线现在规定了必须是defaultValue
。我可能希望设置不同的defaultValue
或根本不设置。然而,在大多数情况下,这应该是最优雅的解决方案。
这篇关于Jetpack合成导航-将参数传递给startDestination的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!
本文标题为:Jetpack合成导航-将参数传递给startDestination
- 想使用ViewPager,无法识别android.support.*? 2022-01-01
- 在测试浓缩咖啡时,Android设备不会在屏幕上启动活动 2022-01-01
- Android - 拆分 Drawable 2022-01-01
- 使用自定义动画时在 iOS9 上忽略 edgesForExtendedLayout 2022-01-01
- 用 Swift 实现 UITextFieldDelegate 2022-01-01
- Android viewpager检测滑动超出范围 2022-01-01
- Android - 我如何找出用户有多少未读电子邮件? 2022-01-01
- MalformedJsonException:在第1行第1列路径中使用JsonReader.setLenient(True)接受格式错误的JSON 2022-01-01
- 如何检查发送到 Android 应用程序的 Firebase 消息的传递状态? 2022-01-01
- android 4中的android RadioButton问题 2022-01-01