8

文档中,我可以传递字符串、整数等。但是如何在导航中传递对象?

注意:如果我将参数类型设置为 parcelable ,那么应用程序会以java.lang.UnsupportedOperationException: Parcelables don't support default values..

composable(
    "vendor/details/{vendor}",
        arguments = listOf(navArgument("vendor") {
            type = NavType.ParcelableType(Vendor::class.java)
        })
) {
// ...
}
4

2 回答 2

8

以下解决方法基于navigation-compose版本2.4.0-alpha05


我找到了 2 个传递对象的解决方法。

1. 将对象转换为 JSON 字符串:

在这里,我们可以使用对象的 JSON 字符串表示来传递对象。

示例代码:

val ROUTE_USER_DETAILS = "user-details/user={user}"


// Pass data (I am using Moshi here)
val user = User(id = 1, name = "John Doe") // User is a data class.

val moshi = Moshi.Builder().build()
val jsonAdapter = moshi.adapter(User::class.java).lenient()
val userJson = jsonAdapter.toJson(user)

navController.navigate(
    ROUTE_USER_DETAILS.replace("{user}", userJson)
)


// Receive Data
NavHost {
    composable(ROUTE_USER_DETAILS) { backStackEntry ->
        val userJson =  backStackEntry.arguments?.getString("user")
        val moshi = Moshi.Builder().build()
        val jsonAdapter = moshi.adapter(User::class.java).lenient()
        val userObject = jsonAdapter.fromJson(userJson)

        UserDetailsView(userObject) // Here UserDetailsView is a composable.
    }
}


// Composable function/view
@Composable
fun UserDetailsView(
    user: User
){
    // ...
}

2.使用传递对象NavBackStackEntry

在这里,我们可以使用 传递数据navController.currentBackStackEntry和使用 接收数据navController.previousBackStackEntry

示例代码:

val ROUTE_USER_DETAILS = "user-details/{user}"


// Pass data
val user = User(id = 1, name = "John Doe") // User is a parcelable data class.

navController.currentBackStackEntry?.arguments?.putParcelable("user", user)
navController.navigate(ROUTE_USER_DETAILS)


// Receive data
NavHost {
    composable(ROUTE_USER_DETAILS) { backStackEntry ->
        val userObject = navController.previousBackStackEntry?.arguments?.getParcelable<User>("user")
        
        UserDetailsView(userObject) // Here UserDetailsView is a composable.
    }
}


// Composable function/view
@Composable
fun UserDetailsView(
    user: User
){
    // ...
}

重要提示:如果我们在导航时弹出回栈,第二个解决方案将不起作用。

于 2021-04-17T00:11:36.140 回答
2

Parcelables 当前不支持默认值,因此您需要将对象作为字符串值传递。是的,这是一种解决方法。因此,我们可以将该对象转换为 JSON(字符串)并通过导航传递,然后将该 JSON 解析回目的地的对象,而不是将对象本身作为 Parcelize 对象传递。您可以使用GSON进行对象到 json 字符串的转换...

Json 到对象

fun <A> String.fromJson(type: Class<A>): A {
    return Gson().fromJson(this, type)
}

对象到 Json 字符串

fun <A> A.toJson(): String? {
    return Gson().toJson(this)
}

用户 NavType.StringType 而不是 NavType.ParcelableType..

composable("detail/{item}",
            arguments = listOf(navArgument("item") {
                type = NavType.StringType
            })
        ) {
            it.arguments?.getString("item")?.let { jsonString ->
                val user = jsonString.fromJson(User::class.java)
                DetailScreen( navController = navController, user = user )
            }
          }

现在通过传递字符串导航..

  val userString = user.toJson()
  navController.navigate(detail/$userString")

编辑:您可以导航的 Json-String 也有限制。如果 Json-String 的长度太长,那么 NavController 将无法识别您的 Composable Route 最终会引发异常......另一种解决方法是使用全局变量并在导航之前设置其值......然后传递这个值作为可组合函数中的参数..

 var globalUser : User? = null // global object somewhere in your code
    .....
    .....
    // While Navigating
    click { user->
            globalUser = user
            navController.navigate(detail/$userString")
          }

    // Composable
    composable( "detail") {
            DetailScreen(
                navController = navController,
                globalUser)
             }

注意:-> ViewModels 也可以用来实现这一点..

于 2021-06-16T20:40:44.090 回答