Os elementos compartilhados tornam as transições entre telas mais suaves e envolventes, criando uma conexão visual que orienta o usuário. Este guia demonstra como usar as APIs de elementos compartilhados com as bibliotecas Navigation 3 e Navigation 2 do Jetpack.
O snippet a seguir inclui elementos combináveis DetailsScreen e HomeScreen que servem como destinos entre os quais os usuários podem navegar. Em cada tela, o
sharedElement modificador é usado na imagem e no texto para que
cada um desses elementos seja animado de forma independente entre as telas.
@Composable fun DetailsScreen( id: Int, snack: Snack, sharedTransitionScope: SharedTransitionScope, animatedVisibilityScope: AnimatedVisibilityScope, onBackPressed: () -> Unit ) { with(sharedTransitionScope) { Column( modifier = Modifier .fillMaxSize() .clickable { onBackPressed() }, ) { Image( painterResource(id = snack.image), contentDescription = snack.description, contentScale = ContentScale.Crop, modifier = Modifier .sharedElement( sharedTransitionScope.rememberSharedContentState(key = "image-$id"), animatedVisibilityScope = animatedVisibilityScope ) .aspectRatio(1f) .fillMaxWidth() ) Text( text = snack.name, fontSize = 18.sp, modifier = Modifier .sharedElement( sharedTransitionScope.rememberSharedContentState(key = "text-$id"), animatedVisibilityScope = animatedVisibilityScope ) .fillMaxWidth(), ) } } } @Composable fun HomeScreen( sharedTransitionScope: SharedTransitionScope, animatedVisibilityScope: AnimatedVisibilityScope, onItemClick: (Int) -> Unit, ) { LazyColumn( modifier = Modifier .fillMaxSize() .padding(8.dp), verticalArrangement = Arrangement.spacedBy(8.dp) ) { itemsIndexed(listSnacks) { index, item -> Row( modifier = Modifier .fillMaxWidth() .clickable { onItemClick(index) }, ) { Spacer(modifier = Modifier.width(8.dp)) with(sharedTransitionScope) { Image( painterResource(id = item.image), contentDescription = item.description, contentScale = ContentScale.Crop, modifier = Modifier .sharedElement( sharedTransitionScope.rememberSharedContentState(key = "image-$index"), animatedVisibilityScope = animatedVisibilityScope ) .size(100.dp) ) Spacer(modifier = Modifier.width(8.dp)) Text( item.name, fontSize = 18.sp, modifier = Modifier .align(Alignment.CenterVertically) .sharedElement( sharedTransitionScope.rememberSharedContentState(key = "text-$index"), animatedVisibilityScope = animatedVisibilityScope, ) ) } } } } }
Navigation 3
Para usar as APIs de elementos compartilhados com a Navigation 3, primeiro é necessário incluir o
NavDisplay em um SharedTransitionLayout. Em seguida, transmita o
fornecido SharedTransitionScope aos elementos combináveis da tela.
Para o AnimatedVisibilityScope, use o
LocalNavAnimatedContentScope local de composição que fornece o
AnimatedContentScope do AnimatedContent que NavDisplay
usa internamente para animar entre cenas.
@Composable fun SharedElement_Nav3() { SharedTransitionLayout { val backStack = rememberNavBackStack(HomeRoute) // Note: NavDisplay accepts a `sharedTransitionScope` parameter, which is used to animate // NavEntry instances between scenes. This parameter *isn't* required for shared element // or shared bounds transitioning elements between different NavEntry, as demonstrated in // this sample. // See https://developer.android.com/guide/navigation/navigation-3/animate-destinations#transition-nav-entries NavDisplay( modifier = Modifier.safeDrawingPadding(), backStack = backStack, entryProvider = entryProvider { entry<HomeRoute> { HomeScreen( sharedTransitionScope = this@SharedTransitionLayout, animatedVisibilityScope = LocalNavAnimatedContentScope.current, onItemClick = { backStack.add(DetailsRoute(it)) }) } entry<DetailsRoute> { detailsRoute -> val id = detailsRoute.item val snack = listSnacks[id] DetailsScreen( id = id, snack = snack, sharedTransitionScope = this@SharedTransitionLayout, animatedVisibilityScope = LocalNavAnimatedContentScope.current, onBackPressed = { backStack.removeLastOrNull() }, ) } }) } }
Navigation 2
Para usar as APIs de elementos compartilhados com a Navigation 2, primeiro é necessário incluir o app's
NavHost em um SharedTransitionLayout. Em seguida, transmita o SharedTransitionScope fornecido aos elementos combináveis da tela.
O parâmetro content do builder composable usa
AnimatedContentScope como um receptor. Portanto, é possível usar this@composable para
referenciar esse escopo.
@Composable fun SharedElement_Nav2() { SharedTransitionLayout { val navController = rememberNavController() NavHost( navController = navController, startDestination = "home", modifier = Modifier.safeDrawingPadding() ) { composable("home") { HomeScreen( sharedTransitionScope = this@SharedTransitionLayout, animatedVisibilityScope = this@composable, onItemClick = { navController.navigate("details/$it") }) } composable( "details/{item}", arguments = listOf(navArgument("item") { type = NavType.IntType }) ) { backStackEntry -> val id = backStackEntry.arguments?.getInt("item") ?: 0 val snack = listSnacks[id] DetailsScreen( id = id, snack = snack, sharedTransitionScope = this@SharedTransitionLayout, animatedVisibilityScope = this@composable, onBackPressed = { navController.popBackStack() } ) } } } }
Volta preditiva com elementos compartilhados
Para usar a volta preditiva com elementos compartilhados, siga estas etapas:
Todas as versões da Navigation 3 oferecem suporte à volta preditiva. Para a Navigation 2, use a versão
2.8.0-alpha02donavigation-composeou mais recente:[versions] androidx-navigation = "2.8.0-alpha02" # Or newer [libraries] androidx-navigation-compose = { module = "androidx.navigation:navigation-compose", version.ref = "androidx-navigation" }dependencies { implementation(libs.androidx.navigation.compose) }As animações de volta preditiva são ativadas por padrão em dispositivos com o Android 15 (nível 35 da API) ou mais recente. Para dispositivos com o Android 14 (nível 34 da API), é necessário ativar a configuração de volta preditiva nas opções do desenvolvedor.
Se o app for direcionado ao Android 14 ou versões anteriores, adicione
android:enableOnBackInvokedCallback="true"aos elementos<application>ou específicos<activity>no arquivoAndroidManifest.xml. Essa flag não é necessária se o app for direcionado ao Android 15 ou versões mais recentes.<manifest xmlns:android="http://schemas.android.com/apk/res/android"> <application ... android:enableOnBackInvokedCallback="true"> </application> </manifest>