Compatibilidade com cortes da tela

Testar o Compose
O Jetpack Compose é o kit de ferramentas de interface recomendado para Android. Aprenda a trabalhar com cortes de tela no Compose.

Um corte da tela é uma área em alguns dispositivos que se estende até a superfície da tela. Ele permite uma experiência de ponta a ponta, além de fornecer espaço para sensores importantes na parte frontal do dispositivo.

O Android oferece suporte a cortes de tela em dispositivos com o Android 9 (nível 28 da API) e versões mais recentes. No entanto, os fabricantes de dispositivos também podem oferecer suporte a cortes da tela em dispositivos com o Android 8.1 ou versões anteriores.

Este documento descreve como implementar suporte para dispositivos com cortes, incluindo como trabalhar com a área de corte, ou seja, o retângulo de ponta a ponta na superfície da tela que contém o corte.

Uma imagem mostrando um exemplo de corte da tela na parte superior central
Figura 1. 1 Corte da tela.

Escolher como o app processa áreas de corte

Se você não quiser que seu conteúdo se sobreponha a uma área de corte, geralmente basta garantir que ele não se sobreponha à barra de status e à barra de navegação. Se você estiver renderizando na área de corte, use WindowInsetsCompat.getDisplayCutout() para extrair um objeto DisplayCutout que contenha os encartes seguros e a caixa delimitadora para cada corte. Essas APIs permitem verificar se o conteúdo se sobrepõe ao corte para que você possa reposicionar, se necessário.

Você também pode determinar se o conteúdo está disposto atrás da área de corte. O atributo de layout da janela layoutInDisplayCutoutMode controla como o conteúdo é renderizado na área de corte. É possível definir layoutInDisplayCutoutMode como um dos seguintes valores:

Você pode definir o modo de corte de forma programática ou definindo um estilo na sua atividade. O exemplo abaixo define um estilo para aplicar o atributo LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES à atividade.

<style name="ActivityTheme">
  <item name="android:windowLayoutInDisplayCutoutMode">
    shortEdges <!-- default, shortEdges, or never -->
  </item>
</style>

As seções a seguir descrevem os diferentes modos de corte com mais detalhes.

Comportamento padrão

Por padrão, no modo retrato sem sinalizações especiais definidas, a barra de status em um dispositivo com corte é redimensionada para ter pelo menos a mesma altura do corte, e o conteúdo é mostrado na área abaixo. No modo paisagem ou tela cheia, a janela do app recebe o efeito letterbox para que nenhum conteúdo seja exibido na área de corte.

Renderizar conteúdo em áreas de corte de borda curta

Para alguns conteúdos, como vídeos, fotos, mapas e jogos, a renderização na área de corte pode ser uma ótima maneira de oferecer uma experiência mais imersiva e de ponta a ponta para os usuários. Com LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES, o conteúdo se estende até a área de corte na borda curta da tela no retrato e no modo paisagem, independente de as barras do sistema estarem ocultas ou visíveis. Ao usar esse modo, verifique se nenhum conteúdo importante se sobrepõe à área de corte.

A imagem abaixo é um exemplo de LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES para um dispositivo no modo retrato:

Uma imagem que mostra o conteúdo sendo renderizado na área de corte no modo retrato
Figura 2. Renderização de conteúdo na área de corte no modo retrato.

A imagem abaixo é um exemplo de LAYOUT_IN_DISPLAY_CUTOUT_MODE_SHORT_EDGES para um dispositivo no modo paisagem:

Uma imagem que mostra o conteúdo sendo renderizado na área de corte no modo paisagem
Figura 3. Renderização de conteúdo na área de corte no modo paisagem.

Nesse modo, a janela se estende por baixo dos cortes na borda curta da tela nos modos retrato e paisagem, independente de a janela estar ocultando as barras do sistema.

Considera-se que há um corte no canto da borda curta:

Uma imagem mostrando um dispositivo com um corte de canto
Figura 4. Um dispositivo com um corte de canto.

Nunca renderize conteúdo na área de corte da tela

Com LAYOUT_IN_DISPLAY_CUTOUT_MODE_NEVER, a janela nunca pode se sobrepor à área de corte.

Veja a seguir um exemplo de LAYOUT_IN_DISPLAY_CUTOUT_MODE_NEVER no retrato:

Uma imagem mostrando o LAYOUT_IN_DISPLAY_CUTOUT_MODE_NEVER no modo retrato
Figura 5. Exemplo de LAYOUT_IN_DISPLAY_CUTOUT_MODE_NEVER para o modo retrato.

Veja a seguir um exemplo de LAYOUT_IN_DISPLAY_CUTOUT_MODE_NEVER no modo paisagem:

Uma imagem mostrando o LAYOUT_IN_DISPLAY_CUTOUT_MODE_NEVER para o modo paisagem
Figura 6. Exemplo de LAYOUT_IN_DISPLAY_CUTOUT_MODE_NEVER no modo paisagem.

Práticas recomendadas para a compatibilidade com corte de tela

Ao trabalhar com cortes da tela, considere o seguinte:

  • Esteja atento ao posicionamento dos elementos essenciais da IU. Não deixe a área de corte ocultar textos, controles ou outras informações importantes.
  • Não coloque nem estenda elementos interativos que exigem reconhecimento por toque na área de corte. A sensibilidade ao toque pode ser menor na área de corte.
  • Sempre que possível, use WindowInsetsCompat para extrair a altura da barra de status e determinar o padding adequado a ser aplicado ao conteúdo. Evite fixar a altura da barra de status no código, porque isso pode resultar em conteúdo sobreposto ou cortado.

    Uma imagem mostrando conteúdo cortado na parte de cima devido à configuração incorreta de encartes
    Figura 7. Use WindowInsetsCompat para evitar sobrepor ou cortar o conteúdo.
  • Use View.getLocationInWindow() para determinar quanto espaço da janela o app está usando. Não presuma que o app esteja usando a janela inteira e não use View.getLocationOnScreen().

  • Use os modos de corte shortEdges ou never se o app precisar entrar e sair do modo imersivo. O comportamento de corte padrão pode fazer com que o conteúdo no app seja renderizado na área de corte enquanto as barras de sistema estão presentes, mas não no modo imersivo. Isso resulta no movimento do conteúdo para cima e para baixo durante as transições, como demonstrado no exemplo a seguir.

    Uma imagem que mostra conteúdo indo para cima e para baixo durante as transições.
    Figura 8. Exemplo de conteúdo que se move para cima e para baixo durante as transições.
  • No modo imersivo, tenha cuidado ao usar as coordenadas de janela e tela, já que o app não usa a tela inteira quando está com efeito letterbox. Por causa do efeito letterbox, as coordenadas da origem da tela não são as mesmas que as coordenadas da origem da janela. Você pode transformar as coordenadas da tela nas coordenadas da visualização conforme necessário usando getLocationOnScreen(). A imagem a seguir mostra como as coordenadas diferem quando o conteúdo tem efeito letterbox:

    Uma imagem mostrando as coordenadas de janela e tela quando o conteúdo está com efeito letterbox.
    Figura 9. Coordenadas de janela versus tela quando o conteúdo tem efeito letterbox.
  • Ao processar MotionEvent, use MotionEvent.getX() e MotionEvent.getY() para evitar problemas de coordenadas semelhantes. Não use MotionEvent.getRawX() ou MotionEvent.getRawY().

Testar como seu conteúdo é renderizado

Teste todas as telas e experiências do seu app. Teste em dispositivos com diferentes tipos de corte, se possível. Caso você não tenha um dispositivo com corte, pode simular configurações comuns de corte em qualquer dispositivo ou emulador com o Android 9 ou versões mais recentes fazendo o seguinte:

  1. Ative as Opções do desenvolvedor.
  2. Na tela Opções do desenvolvedor, role para baixo até a seção Desenho e selecione Simular uma tela com corte.
  3. Selecione o tipo de corte.

    Uma imagem mostrando como simular um corte da tela no emulador
    Figura 10. Opções do desenvolvedor para testar como o conteúdo é renderizado.

Outros recursos