A volta preditiva, um recurso de navegação por gestos, permite que os usuários tenham uma prévia de onde o deslize para voltar os leva.
O gesto "Voltar" pode, por exemplo, mostrar uma visualização animada da tela inicial atrás do app, como apresentado na Figura 1.
A partir do Android 15, a opção para desenvolvedores de animações de volta preditiva não está mais disponível. Animações do sistema, como de volta à tela inicial, entre tarefas e atividades, agora aparecem para apps que ativaram o gesto de volta preditivo inteiramente ou em um nível de atividade.
É possível testar essa animação de retorno à tela inicial, conforme descrito na próxima seção desta página.
Para oferecer suporte ao gesto de volta preditivo, é necessário atualizar o app usando o
OnBackPressedCallback
da API compatível com versões anteriores AppCompat 1.6.0-alpha05 (AndroidX) ou uma versão mais recente, ou ainda usando a nova API da plataforma OnBackInvokedCallback
. A maioria dos apps usa a API AndroidX compatível com versões anteriores.
Essa atualização oferece um caminho de migração para interceptar corretamente a navegação de retorno,
o que envolve substituir essas interceptações em KeyEvent.KEYCODE_BACK
e todas as classes com métodos onBackPressed
, como Activity
e
Dialog
, pelas novas APIs Back do sistema.
Codelab e vídeo do Google I/O
Além de usar a documentação disponível nesta página, acesse nosso codelab. Ele apresenta a implementação de um caso de uso comum de um WebView que processa o gesto de volta preditivo usando as APIs Activity do AndroidX.
Você também pode assistir ao vídeo do Google I/O, que mostra outros exemplos de implementação das APIs do AndroidX e da plataforma.
Atualizar um app que usa a navegação de retorno padrão
Se o app não implementa nenhum comportamento personalizado para a ação "Voltar", ou seja, se é o sistema que processa essa ação, a atualização para oferecer suporte ao novo recurso é um processo simples. Ative esse recurso conforme descrito neste guia.
Se o app usa fragmentos ou o componente Navigation, faça upgrade para o AndroidX Activity 1.6.0-alpha05 ou versões mais recentes.
Atualizar um app que usa uma navegação de retorno personalizada
Existem diferentes caminhos de migração para apps que implementam um comportamento personalizado para a ação "Voltar", dependendo se o app usa o AndroidX e da maneira como ele processa a navegação de retorno.
O app usa o AndroidX | Como o app processa a navegação de retorno | Caminho de migração recomendado (link nesta página) |
Sim | APIs do AndroidX | Migrar uma implementação da ação "Voltar" existente do AndroidX |
APIs de plataforma sem suporte | Migrar para as APIs do AndroidX um aplicativo AndroidX com APIs de navegação de retorno sem suporte | |
Não | App usa APIs de plataforma sem suporte, mas é possível migrá-lo | Migrar para as APIs de plataforma um aplicativo que usa APIs de navegação de retorno sem suporte |
App usa APIs de plataforma sem suporte, mas não é possível migrá-lo | Adiar a ativação do recurso até que ele seja obrigatório |
Migrar a implementação da navegação de retorno do AndroidX
Esse é o caso de uso mais comum e recomendado. Ele se aplica a apps
novos ou antigos que processam uma navegação de retorno por gesto personalizada com
OnBackPressedDispatcher
, conforme descrito em
Oferecer navegação de retorno personalizada.
Caso seu app se encaixe nessa categoria, siga estas etapas para adicionar o suporte para o gesto de volta preditivo:
Para garantir que as APIs que já usam as APIs
OnBackPressedDispatcher
(como fragmentos e o componente Navigation) funcionem perfeitamente com o gesto de volta preditivo, faça upgrade para a AndroidX Activity 1.6.0-alpha05.// In your build.gradle file: dependencies { // Add this in addition to your other dependencies implementation "androidx.activity:activity:1.6.0-alpha05"
Ative o gesto de volta preditivo, conforme descrito nesta página.
Migrar para as APIs do AndroidX um aplicativo AndroidX com APIs de navegação de retorno sem suporte
Se o app usa as bibliotecas do AndroidX, mas implementa ou referencia APIs de navegação de retorno sem suporte, você precisa migrar para as APIs do AndroidX para conseguir oferecer suporte ao novo comportamento.
Para migrar de APIs sem suporte e passar a usar as APIs do AndroidX:
Migre a lógica de processamento da ação "Voltar" do sistema para o
OnBackPressedDispatcher
do AndroidX, implementandoOnBackPressedCallback
. Encontre instruções detalhadas em Oferecer navegação de retorno personalizada.Desative o
OnBackPressedCallback
quando estiver tudo pronto para deixar de interceptar o gesto de volta.Pare de usar
OnBackPressed
ouKeyEvent.KEYCODE_BACK
para interceptar eventos de retorno.Faça upgrade para o AndroidX Activity 1.6.0-alpha05.
// In your build.gradle file: dependencies { // Add this in addition to your other dependencies implementation "androidx.activity:activity:1.6.0-alpha05"
Depois de migrar o app, ative o gesto de volta preditivo, conforme descrito nesta página, para conferir a animação do sistema para voltar à tela inicial.
Migrar para as APIs de plataforma um aplicativo que usa APIs de navegação de retorno sem suporte
Caso não seja possível usar as bibliotecas do AndroidX no app e ele implemente ou faça
referência à navegação de retorno personalizada usando APIs sem suporte, você precisa migrar
para a API de plataforma OnBackInvokedCallback
.
Conclua as etapas a seguir para migrar das APIs sem suporte e passar a usar a API da plataforma:
Use a nova API
OnBackInvokedCallback
em dispositivos com o Android 13 ou versões mais recentes. No Android 12 ou versões anteriores, você precisará usar APIs sem suporte.Registre sua lógica de retorno personalizada em
OnBackInvokedCallback
com oonBackInvokedDispatcher
. Isso impede que a atividade em curso seja finalizada. Assim, o callback poderá reagir à ação "Voltar" quando o usuário concluir a navegação de retorno.Cancele o registro do
OnBackInvokedCallback
quando estiver tudo pronto para deixar de interceptar o gesto de volta. Caso contrário, o app pode apresentar comportamentos indesejáveis para a ação "Voltar". Por exemplo, ele pode ficar preso entre as duas visualizações, fazendo com que o usuário tenha que forçar o fechamento.Confira um exemplo de como migrar a lógica de
onBackPressed
:@Override fun onCreate() { if (BuildCompat.isAtLeastT()) { onBackInvokedDispatcher.registerOnBackInvokedCallback( OnBackInvokedDispatcher.PRIORITY_DEFAULT ) { /** * onBackPressed logic goes here. For instance: * Prevents closing the app to go home screen when in the * middle of entering data to a form * or from accidentally leaving a fragment with a WebView in it * * Unregistering the callback to stop intercepting the back gesture: * When the user transitions to the topmost screen (activity, fragment) * in the BackStack, unregister the callback by using * OnBackInvokeDispatcher.unregisterOnBackInvokedCallback * (https://developer.android.com/reference/kotlin/android/window/OnBackInvokedDispatcher#unregisteronbackinvokedcallback) */ } } }
@Override void onCreate() { if (BuildCompat.isAtLeastT()) { getOnBackInvokedDispatcher().registerOnBackInvokedCallback( OnBackInvokedDispatcher.PRIORITY_DEFAULT, () -> { /** * onBackPressed logic goes here - For instance: * Prevents closing the app to go home screen when in the * middle of entering data to a form * or from accidentally leaving a fragment with a WebView in it * * Unregistering the callback to stop intercepting the back gesture: * When the user transitions to the topmost screen (activity, fragment) * in the BackStack, unregister the callback by using * OnBackInvokeDispatcher.unregisterOnBackInvokedCallback * (https://developer.android.com/reference/kotlin/android/view/OnBackInvokedDispatcher#unregisteronbackinvokedcallback) */ } ); } }
Pare de usar
OnBackPressed
ouKeyEvent.KEYCODE_BACK
para interceptar eventos de retorno no Android 13 e versões mais recentes.Depois de migrar o app, ative o gesto de volta preditivo, conforme descrito nesta página, para que
OnBackInvokedCallback
entre em vigor.
Você pode registrar um OnBackInvokedCallback
com PRIORITY_DEFAULT
ou
PRIORITY_OVERLAY
, que não está disponível em um
OnBackPressedCallback
do AndroidX semelhante. Registrar um callback com PRIORITY_OVERLAY
é
útil em alguns casos.
Isso se aplica quando você migra
de onKeyPreIme()
e seu callback precisa receber o gesto "Voltar" em vez
de um IME aberto. Os IMEs registram callbacks com PRIORITY_DEFAULT
quando abertos.
Registre o callback com PRIORITY_OVERLAY
para garantir que
OnBackInvokedDispatcher
envie o gesto de volta ao callback, em vez
do IME aberto.
Ativar o gesto de volta preditivo
Depois de entender como atualizar o app de acordo com seu caso específico, ofereça suporte ao gesto de volta preditivo.
Para ativar esse recurso, em AndroidManifest.xml
, na tag <application>
, defina a flag
android:enableOnBackInvokedCallback
como true
.
<application
...
android:enableOnBackInvokedCallback="true"
... >
...
</application>
Se você não informar um valor, o padrão vai ser false
e vai ocorrer o seguinte:
- A animação do sistema do gesto de volta preditivo será desativada.
- O
OnBackInvokedCallback
será ignorado, mas as chamadas deOnBackPressedCallback
vão continuar funcionando.
Ativar no nível da atividade
No Android 14 e versões mais recentes, a flag android:enableOnBackInvokedCallback
permite
ativar animações preditivas do sistema no nível da atividade. Esse comportamento
facilita a migração de apps grandes com várias atividades para gestos
de volta preditivos. Com o Android 15, a volta preditiva não está mais por trás da
opção para desenvolvedores. Os apps podem ativar a volta preditiva totalmente ou no nível
da atividade.
O código abaixo mostra um exemplo de como usar o enableOnBackInvokedCallback
para
ativar a animação do sistema de volta à tela inicial da MainActivity
:
<manifest ...>
<application . . .
android:enableOnBackInvokedCallback="false">
<activity
android:name=".MainActivity"
android:enableOnBackInvokedCallback="true"
...
</activity>
<activity
android:name=".SecondActivity"
android:enableOnBackInvokedCallback="false"
...
</activity>
</application>
</manifest>
No exemplo anterior, definir android:enableOnBackInvokedCallback=true
para
".SecondActivity"
ativa a animação do sistema entre atividades.
Lembre-se das considerações abaixo ao usar a
flag android:enableOnBackInvokedCallback
:
- Configurar
android:enableOnBackInvokedCallback=false
desativa as animações de volta preditiva no nível da atividade ou do app, dependendo de onde a tag foi definida. Além disso, instrui o sistema a ignorar chamadas para a API da plataformaOnBackInvokedCallback
. No entanto, as chamadas paraOnBackPressedCallback
continuam sendo executadas porqueOnBackPressedCallback
é compatível com versões anteriores e chama a APIonBackPressed
, que não tem suporte em versões anteriores ao Android 13. - Definir a flag
enableOnBackInvokedCallback
no nível do app estabelece o valor padrão para todas as atividades nele. Você pode substituir o padrão por atividade definindo a flag no nível da atividade, conforme mostrado no exemplo de código anterior.
Práticas recomendadas de callback
Confira as práticas recomendadas para usar os callbacks de sistema com suporte:
BackHandler
(para Compose), OnBackPressedCallback
ou
OnBackInvokedCallback
.
Determinar o estado da interface que ativa e desativa cada callback
O estado da interface é a propriedade que a descreve. Recomendamos seguir estas etapas gerais.
Determine o estado da interface que ativa e desativa cada callback.
Defina esse estado usando um tipo de detentor de dados observáveis, como
StateFlow
ou o estado do Compose, e ative ou desative o callback quando o estado mudar.
Se o app estava associando a lógica de retorno às instruções condicionais, isso pode significar que você está reagindo ao evento de retorno depois que ele já ocorreu. Evite esse padrão com callbacks mais recentes. Se possível, retire o callback da instrução condicional e associe-o a um tipo de detentor de dados observáveis.
Usar callbacks de sistema para a lógica da interface
A lógica da interface determina como mostrá-la. Use callbacks de sistema para executar a lógica da interface, como exibir um pop-up ou executar uma animação.
Se o app ativar um callback de sistema, as animações preditivas não serão executadas, e você vai precisar processar o evento de retorno. Não crie callbacks apenas para executar uma lógica que não seja da interface.
Por exemplo, se você estiver interceptando eventos de retorno apenas para geração de registros, faça os registros no ciclo de vida da atividade ou do fragmento.
- Para casos atividade-atividade ou fragmento-atividade, registre se
isFinishing
emonDestroy
fortrue
no ciclo de vida da atividade. - Para casos fragmento-fragmento, registre se
isRemoving
emonDestroy
for verdadeiro no ciclo de vida da visualização do fragmento. Ou registre usando os métodosonBackStackChangeStarted
ouonBackStackChangeCommitted
emFragmentManager.OnBackStackChangedListener
.
Para o caso do Compose, faça o registro no callback onCleared()
de um ViewModel
associado ao destino do Compose. Esse é o melhor indicador para saber
quando um destino do Compose é retirado da backstack e destruído.
Criar callbacks de responsabilidade única
É possível adicionar vários callbacks ao agente. Eles são adicionados a uma pilha em que o último callback ativo adicionado processa o próximo gesto de volta com um callback por gesto de volta.
É mais fácil gerenciar o estado ativado de um callback se ele tiver uma única responsabilidade. Exemplo:
A Figura 2 mostra como é possível ter vários callbacks na pilha, cada um responsável por uma coisa. Um callback só é executado se os callbacks acima dele na pilha estiverem desativados. Neste exemplo, o callback "Tem certeza..." é ativado quando o usuário insere dados em um formulário e desativado em outros casos. O callback abre uma caixa de diálogo de confirmação quando o usuário desliza para sair do formulário.
O outro callback pode incluir um componente do Material Design que ofereça suporte à volta preditiva, uma transição do AndroidX usando as APIs Progress ou outro callback personalizado.
Um callback de childFragmentManager
é executado se os callbacks acima estiverem
desativados e a backstack para essa FragmentManager
não estiver vazia, em que
childFragmentManager
está anexado a um fragmento. Neste exemplo, esse
callback interno está desativado.
Da mesma forma, o callback interno de supportFragmentManager
será executado se os callbacks
acima estiverem desativados e a pilha não estiver vazia. Esse comportamento é consistente
ao usar FragmentManager
ou NavigationComponent
para navegação,
já que NavigationComponent
depende de FragmentManager
. Neste exemplo,
esse callback é executado se o usuário não inserir texto no formulário, fazendo com que o
callback "Tem certeza..." seja desativado.
Por fim, super.onBackPressed()
é o callback no nível do sistema, que é executado novamente se
os callbacks acima estiverem desativados. Para acionar animações do sistema, como
voltar à tela inicial, entre atividades e entre tarefas, a backstack de supportFragmentManager
precisa estar vazia para que o callback interno seja desativado.
Testar a animação do gesto de volta preditivo
Se você ainda usa o Android 13 ou o Android 14, teste a animação de retorno à tela de início mostrada na Figura 1.
Para testar a animação, siga estas etapas:
No dispositivo, acesse Configurações > Sistema > Opções do desenvolvedor.
Selecione Animações de gestos "Voltar" preditivos.
Inicie o app atualizado e use o gesto "Voltar" para testar o recurso.