Implementar o tema escuro

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

Figura 1. Tema escuro.

O tema escuro está disponível no Android 10 (API de nível 29) e versões mais recentes. Ele tem os seguintes benefícios:

  • Reduz significativamente o consumo de energia, dependendo da tecnologia da tela do dispositivo.
  • Melhora a visibilidade para usuários com baixa visão e para aqueles sensíveis a luzes intensas.
  • Facilita o uso do dispositivo em ambientes com pouca luz.

O tema escuro se aplica à IU do sistema Android e aos apps em execução no dispositivo.

Há três maneiras de ativar o tema escuro no Android 10 e versões mais recentes:

  • Use a configuração do sistema acessando Configurações > Tela > Tema para ativar o tema escuro.
  • Use o bloco Configurações rápidas para trocar de tema na bandeja de notificações, quando ativado.
  • Em dispositivos Pixel, ative o modo Economia de bateria para ativar o tema escuro ao mesmo tempo. Outros dispositivos podem não ser compatíveis com esse comportamento.

Para ver instruções sobre como aplicar um tema escuro ao conteúdo baseado na Web usando um componente WebView, consulte Escurecer conteúdo da Web com WebView.

Oferecer suporte ao tema escuro no app

Para oferecer suporte ao tema escuro, configure o tema do seu app, normalmente encontrado em res/values/styles.xml, para herdar de um tema DayNight:

<style name="AppTheme" parent="Theme.AppCompat.DayNight">

Você também pode usar o tema escuro do Material Components:

<style name="AppTheme" parent="Theme.MaterialComponents.DayNight">

Isso vincula o tema principal do app às sinalizações do modo noturno controlado pelo sistema e oferece um tema escuro padrão ao app quando ele está ativado.

Temas e estilos

Evite usar cores ou ícones codificados para uso em um tema claro. Use atributos de tema ou recursos qualificados para a noite.

Dois atributos de tema são mais importantes para o tema escuro:

  • ?android:attr/textColorPrimary: uma cor de texto de uso geral. Ela é quase preta no tema claro e quase branca nos temas escuros. Contém um estado desativado.
  • ?attr/colorControlNormal: uma cor de ícone para fins gerais. Contém um estado desativado.

Recomendamos o uso de componentes do Material Design, já que o sistema de temas coloridos deles, como os atributos de tema ?attr/colorSurface e ?attr/colorOnSurface, proporciona fácil acesso a cores adequadas. É possível personalizar esses atributos no seu tema.

Mudar temas no app

É possível permitir que os usuários mudem o tema do app enquanto ele está em execução. Estas são as opções recomendadas:

  • Claro
  • Escuro
  • Padrão do sistema (a opção padrão recomendada)

Essas opções são mapeadas diretamente para os modos AppCompat.DayNight:

Para mudar o tema, faça o seguinte:

Forçar modo escuro

O Android 10 oferece o recurso Forçar modo escuro para que os desenvolvedores implementem rapidamente um tema escuro sem definir explicitamente um tema DayNight.

O recurso "Forçar modo escuro" analisa cada visualização do seu app com tema claro e aplica um tema escuro automaticamente antes de ele ser exibido na tela. Você pode usar uma combinação de Forçar modo escuro e implementação nativa para reduzir o tempo necessário para implementar o tema escuro.

Os apps precisam ativar o "Forçar modo escuro" configurando android:forceDarkAllowed="true" no tema da atividade. Esse atributo é definido em todos os temas claros fornecidos pelo sistema e pelo AndroidX, como Theme.Material.Light. Ao usar o Force Dark, teste seu app completamente e exclua as visualizações conforme necessário.

Caso o app use um tema escuro, como Theme.Material, o recurso Forçar modo escuro não será aplicado. Da mesma forma, se o tema do app for herdado de um tema DayNight, o recurso Forçar modo escuro não será aplicado devido à troca automática de temas.

Desativar o Forçar modo escuro em uma visualização

O recurso "Forçar modo escuro" pode ser controlado em visualizações específicas com o atributo de layout android:forceDarkAllowed ou com setForceDarkAllowed().

Conteúdo da Web

Para ver informações sobre como usar temas escuros no conteúdo baseado na Web, consulte Escurecer conteúdo da Web com WebView. Para conferir um exemplo de tema escuro aplicado a uma WebView, consulte a demonstração do WebView no GitHub .

Práticas recomendadas

As seções a seguir demonstram as práticas recomendadas para implementar temas escuros.

Notificações e widgets

Para superfícies de IU que você exibe no dispositivo, mas não controla diretamente, verifique se as visualizações usadas refletem o tema do app host. Dois exemplos são notificações e widgets da tela de início.

Notificações

Use os modelos de notificação fornecidos pelo sistema, como MessagingStyle. Isso significa que o sistema é responsável por aplicar o estilo de visualização correto.

Widgets e visualizações de notificações personalizadas

Para widgets de tela de início ou se o app usa visualizações personalizadas de conteúdo de notificação, teste o conteúdo nos temas claro e escuro.

Os problemas mais comuns a serem observados incluem:

  • Supondo que a cor do plano de fundo seja sempre clara.
  • Fixar cores do texto no código.
  • Definir uma cor de plano de fundo codificada ao usar a cor de texto padrão.
  • Usar um ícone drawable de uma cor estática.

Em todos esses casos, use atributos de tema apropriados em vez de cores fixadas no código.

Telas de inicialização

Se o app tiver uma tela de inicialização personalizada, talvez seja necessário modificá-la para que reflita o tema selecionado.

Remova todas as cores fixadas no código, como as cores de plano de fundo definidas programaticamente como branco. Em vez disso, use o atributo de tema ?android:attr/colorBackground.

Mudanças de configuração

Quando o tema do app muda, seja pela configuração do sistema ou do AppCompat, ele aciona uma mudança de configuração uiMode. Isso significa que as atividades são recriadas automaticamente.

Em alguns casos, pode ser necessário que um app processe a mudança de configuração. Por exemplo, é recomendável atrasar uma mudança de configuração quando um vídeo está sendo exibido.

Um app pode processar a implementação do tema escuro declarando que cada Activity pode processar a mudança de configuração do uiMode:

<activity
    android:name=".MyActivity"
    android:configChanges="uiMode" />

Quando um Activity declara que processa mudanças de configuração, o método onConfigurationChanged() dele é chamado quando há uma mudança de tema.

Para verificar qual é o tema atual, os apps podem executar códigos como este:

Kotlin

val currentNightMode = configuration.uiMode and Configuration.UI_MODE_NIGHT_MASK
when (currentNightMode) {
    Configuration.UI_MODE_NIGHT_NO -> {} // Night mode is not active, we're using the light theme.
    Configuration.UI_MODE_NIGHT_YES -> {} // Night mode is active, we're using dark theme.
}

Java

int currentNightMode = configuration.uiMode & Configuration.UI_MODE_NIGHT_MASK;
switch (currentNightMode) {
    case Configuration.UI_MODE_NIGHT_NO:
        // Night mode is not active, we're using the light theme
        break;
    case Configuration.UI_MODE_NIGHT_YES:
        // Night mode is active, we're using dark theme
        break;
}