Mudanças de comportamento: apps destinados ao Android 15 ou mais recente

Assim como nas versões anteriores, o Android 15 inclui mudanças de comportamento que podem afetar seu app. As mudanças de comportamento a seguir se aplicam exclusivamente a apps destinados ao Android 15 ou versões mais recentes. Caso seu app seja direcionado ao Android 15 ou a versões mais recentes, faça modificações para oferecer suporte a esses comportamentos de forma adequada, quando aplicável.

Consulte também a lista de mudanças de comportamento que afetam todos os apps executados no Android 15, independente da targetSdkVersion do seu app.

Principal recurso

O Android 15 modifica ou expande vários recursos principais do sistema Android.

Mudanças nos serviços em primeiro plano

Estamos fazendo as seguintes mudanças nos serviços em primeiro plano com o Android 15.

Comportamento de tempo limite do serviço em primeiro plano da sincronização de dados

O Android 15 introduz um novo comportamento de tempo limite no dataSync para apps destinados ao Android 15 (nível 35 da API) ou versões mais recentes. Esse comportamento também se aplica ao novo tipo de serviço em primeiro plano mediaProcessing.

O sistema permite que os serviços dataSync de um app sejam executados por um total de 6 horas em um período de 24 horas. Depois disso, o sistema chama o método Service.onTimeout(int, int) do serviço em execução (introduzido no Android 15). No momento, o serviço tem alguns segundos para chamar Service.stopSelf(). Quando Service.onTimeout() é chamado, o serviço não é mais considerado um serviço em primeiro plano. Se o serviço não chamar Service.stopSelf(), o sistema vai gerar uma exceção interna. A exceção é registrada no Logcat com a seguinte mensagem:

Fatal Exception: android.app.RemoteServiceException: "A foreground service of
type dataSync did not stop within its timeout: [component name]"

Para evitar problemas com essa mudança de comportamento, faça uma ou mais das seguintes ações:

  1. Faça com que o serviço implemente o novo método Service.onTimeout(int, int). Quando o app receber o callback, chame stopSelf() em alguns segundos. Se você não parar o app imediatamente, o sistema vai gerar uma falha.
  2. Os serviços dataSync do app não podem ser executados por mais de 6 horas em um período de 24 horas (a menos que o usuário interaja com o app, redefinindo o timer).
  3. Só inicie serviços em primeiro plano dataSync como resultado da interação direta do usuário. Como o app está em primeiro plano quando o serviço é iniciado, ele tem seis horas completas após o app ir para o segundo plano.
  4. Em vez de usar um serviço em primeiro plano dataSync, use uma API alternativa.

Se os serviços em primeiro plano dataSync do app tiverem sido executados por seis horas nas últimas 24 horas, não será possível iniciar outro serviço em primeiro plano dataSync a menos que o usuário tenha trazido o app para o primeiro plano, o que redefine o timer. Se você tentar iniciar outro serviço em primeiro plano dataSync, o sistema vai gerar ForegroundServiceStartNotAllowedException com uma mensagem de erro como "O limite de tempo já foi esgotado para o tipo de serviço em primeiro plano dataSync".

Teste

Para testar o comportamento do app, ative os timeouts de sincronização de dados mesmo que o app não esteja segmentado para o Android 15, desde que esteja sendo executado em um dispositivo Android 15. Para ativar os tempos limite, execute o comando adb:

adb shell am compat enable FGS_INTRODUCE_TIME_LIMITS your-package-name

Você também pode ajustar o período de tempo limite para facilitar o teste do comportamento do app quando o limite for atingido. Para definir um novo período de tempo limite, execute o seguinte comando adb:

adb shell device_config put activity_manager data_sync_fgs_timeout_duration duration-in-milliseconds

Novo tipo de serviço em primeiro plano de processamento de mídia

O Android 15 apresenta um novo tipo de serviço em primeiro plano, mediaProcessing. Esse tipo de serviço é adequado para operações como transcodificação de arquivos de mídia. Por exemplo, um app de mídia pode fazer o download de um arquivo de áudio e precisar convertê-lo para um formato diferente antes de reproduzi-lo. Você pode usar um serviço em primeiro plano mediaProcessing para garantir que a conversão continue mesmo quando o app estiver em segundo plano.

O sistema permite que os serviços mediaProcessing de um app sejam executados por um total de 6 horas em um período de 24 horas. Depois disso, o sistema chama o método Service.onTimeout(int, int) do serviço em execução (introduzido no Android 15). Nesse momento, o serviço tem alguns segundos para chamar Service.stopSelf(). Se o serviço não chamar Service.stopSelf(), o sistema vai gerar uma exceção interna. A exceção é registrada no Logcat com a seguinte mensagem:

Fatal Exception: android.app.RemoteServiceException: "A foreground service of
type mediaProcessing did not stop within its timeout: [component name]"

Para evitar a exceção, siga um destes procedimentos:

  1. Faça com que o serviço implemente o novo método Service.onTimeout(int, int). Quando o app receber o callback, chame stopSelf() dentro de alguns segundos. Se você não interromper o app imediatamente, o sistema vai gerar uma falha.
  2. Verifique se os serviços mediaProcessing do app não são executados por mais de um total de seis horas em qualquer período de 24 horas, a menos que o usuário interaja com o app, redefinindo o timer.
  3. Só inicie serviços em primeiro plano mediaProcessing como resultado da interação direta do usuário. Como o app está em primeiro plano quando o serviço é iniciado, ele tem seis horas completas após o app ir para o segundo plano.
  4. Em vez de usar um serviço em primeiro plano mediaProcessing, use uma API alternativa, como o WorkManager.

Se os serviços em primeiro plano mediaProcessing do app tiverem sido executados por 6 horas nas últimas 24 horas, não será possível iniciar outro serviço em primeiro plano mediaProcessing a menos que o usuário tenha trazido o app para o primeiro plano (o que redefine o timer). Se você tentar iniciar outro serviço mediaProcessing em primeiro plano, o sistema vai gerar ForegroundServiceStartNotAllowedException com uma mensagem de erro como "O limite de tempo já se esgotou para o tipo de serviço em primeiro plano mediaProcessing".

Para mais informações sobre o tipo de serviço mediaProcessing, consulte Mudanças nos tipos de serviço em primeiro plano do Android 15: processamento de mídia.

Teste

Para testar o comportamento do app, ative os timeouts de processamento de mídia, mesmo que o app não seja direcionado ao Android 15 (desde que esteja sendo executado em um dispositivo Android 15). Para ativar os tempos limite, execute o seguinte comando adb:

adb shell am compat enable FGS_INTRODUCE_TIME_LIMITS your-package-name

Você também pode ajustar o período de tempo limite para facilitar o teste do comportamento do app quando o limite for atingido. Para definir um novo período de tempo limite, execute o seguinte comando adb:

adb shell device_config put activity_manager media_processing_fgs_timeout_duration duration-in-milliseconds

Restrições em broadcast receivers BOOT_COMPLETED que iniciam serviços em primeiro plano

There are new restrictions on BOOT_COMPLETED broadcast receivers launching foreground services. BOOT_COMPLETED receivers are not allowed to launch the following types of foreground services:

If a BOOT_COMPLETED receiver tries to launch any of those types of foreground services, the system throws ForegroundServiceStartNotAllowedException.

Testing

To test your app's behavior, you can enable these new restrictions even if your app is not targeting Android 15 (as long as the app is running on an Android 15 device). Run the following adb command:

adb shell am compat enable FGS_BOOT_COMPLETED_RESTRICTIONS your-package-name

To send a BOOT_COMPLETED broadcast without restarting the device, run the following adb command:

adb shell am broadcast -a android.intent.action.BOOT_COMPLETED your-package-name

Restrições para iniciar serviços em primeiro plano enquanto um app tem a permissão SYSTEM_ALERT_WINDOW

Anteriormente, se um app tivesse a permissão SYSTEM_ALERT_WINDOW, ele poderia iniciar um serviço em primeiro plano mesmo que estivesse em segundo plano (conforme discutido em isenção de restrições de início em segundo plano).

Se um app for destinado ao Android 15, essa isenção será mais restrita. Agora o app precisa ter a permissão SYSTEM_ALERT_WINDOW e também ter uma janela de sobreposição visível. Ou seja, o app precisa primeiro abrir uma janela TYPE_APPLICATION_OVERLAY e a janela precisa estar visível antes de iniciar um serviço em primeiro plano.

Se o app tentar iniciar um serviço em primeiro plano em segundo plano sem atender a esses novos requisitos (e não tiver outra isenção), o sistema vai gerar uma ForegroundServiceStartNotAllowedException.

Se o app declarar a permissão SYSTEM_ALERT_WINDOW e iniciar serviços em primeiro plano em segundo plano, ele poderá ser afetado por essa mudança. Se o app receber uma ForegroundServiceStartNotAllowedException, verifique a ordem das operações e verifique se ele já tem uma janela de sobreposição ativa antes de tentar iniciar um serviço em primeiro plano em segundo plano. Você pode conferir se a janela de sobreposição está visível chamando View.getWindowVisibility() ou substituir View.onWindowVisibilityChanged() para receber uma notificação sempre que a visibilidade mudar.

Teste

Para testar o comportamento do app, ative essas novas restrições mesmo que ele não seja direcionado ao Android 15, desde que esteja sendo executado em um dispositivo Android 15. Para ativar essas novas restrições na inicialização de serviços em primeiro plano em segundo plano, execute este comando adb:

adb shell am compat enable FGS_SAW_RESTRICTIONS your-package-name

Mudanças em quando os apps podem modificar o estado global do modo Não perturbe

以 Android 15(API 级别 35)及更高版本为目标平台的应用无法再更改设备上的勿扰 (DND) 功能的全局状态或政策(无论是通过修改用户设置还是关闭勿扰模式)。相反,应用必须提供 AutomaticZenRule,系统会将其与现有的“最严格的政策优先”方案合并为一个全局政策。对之前会影响全局状态的现有 API 的调用(setInterruptionFiltersetNotificationPolicy)会导致创建或更新隐式 AutomaticZenRule,该 AutomaticZenRule 会根据这些 API 调用的调用周期开启和关闭。

请注意,只有当应用调用 setInterruptionFilter(INTERRUPTION_FILTER_ALL) 并希望该调用停用之前由其所有者激活的 AutomaticZenRule 时,此更改才会影响可观察到的行为。

Mudanças na API OpenJDK

Android 15 将继续更新 Android 的核心库,以与最新 OpenJDK LTS 版本中的功能保持一致。

以下变更可能会影响以 Android 15(API 级别 35)为目标平台的应用的兼容性:

  • 对字符串格式化 API 进行了更改:现在,使用以下 String.format()Formatter.format() API 时,对实参索引、标志、宽度和精度的验证要求变得更加严格:

    例如,当使用参数索引 0(格式字符串中的 %0)时,系统会抛出以下异常:

    IllegalFormatArgumentIndexException: Illegal format argument index = 0
    

    在这种情况下,可以使用实参索引 1(格式字符串中的 %1)来解决此问题。

  • Arrays.asList(...).toArray() 的组件类型的更改:使用 Arrays.asList(...).toArray() 时,所得数组的组件类型现在是 Object,而不是底层数组元素的类型。因此,以下代码会抛出 ClassCastException

    String[] elements = (String[]) Arrays.asList("one", "two").toArray();
    

    在这种情况下,为了在生成的数组中保留 String 作为组件类型,您可以改用 Collection.toArray(Object[])

    String[] elements = Arrays.asList("two", "one").toArray(new String[0]);
    
  • 语言代码处理方面的变化:使用 Locale API 时,希伯来语、意第绪语和印度尼西亚语的语言代码不再转换为其过时的形式(希伯来语:iw、意第绪语:ji 和印度尼西亚语:in)。指定这些语言区域的语言代码时,请改用 ISO 639-1 中的代码(希伯来语:he、意第绪语:yi 和印度尼西亚语:id)。

  • 对随机整数序列的更改:根据 https://bugs.openjdk.org/browse/JDK-8301574 中所做的更改,以下 Random.ints() 方法现在返回的数字序列与 Random.nextInt() 方法返回的数字序列不同:

    一般来说,此更改不应导致应用行为中断,但您的代码不应期望从 Random.ints() 方法生成的序列与 Random.nextInt() 相匹配。

新的 SequencedCollection API 可能会影响您应用的兼容性,具体取决于您是否在应用的 build 配置中更新 compileSdk 以使用 Android 15(API 级别 35)

  • kotlin-stdlib 中的 MutableList.removeFirst()MutableList.removeLast() 扩展函数发生冲突

    Java 中的 List 类型会映射到 Kotlin 中的 MutableList 类型。 由于 List.removeFirst()List.removeLast() API 已在 Android 15(API 级别 35)中引入,因此 Kotlin 编译器会将函数调用(例如 list.removeFirst())静态解析为新的 List API,而不是 kotlin-stdlib 中的扩展函数。

    如果应用重新编译时将 compileSdk 设置为 35,并将 minSdk 设置为 34 或更低值,然后在 Android 14 及更低版本上运行该应用,则会抛出运行时错误:

    java.lang.NoSuchMethodError: No virtual method
    removeFirst()Ljava/lang/Object; in class Ljava/util/ArrayList;
    

    Android Gradle 插件中现有的 NewApi lint 选项可以捕获这些新的 API 用法。

    ./gradlew lint
    
    MainActivity.kt:41: Error: Call requires API level 35 (current min is 34): java.util.List#removeFirst [NewApi]
          list.removeFirst()
    

    为了修复运行时异常和 lint 错误,可以在 Kotlin 中将 removeFirst()removeLast() 函数调用分别替换为 removeAt(0)removeAt(list.lastIndex)。如果您使用的是 Android Studio Ladybug | 2024.1.3 或更高版本,它还会针对这些错误提供快速修复选项。

    如果已停用 lint 选项,请考虑移除 @SuppressLint("NewApi")lintOptions { disable 'NewApi' }

  • 与 Java 中的其他方法发生冲突

    现有类型中添加了新方法,例如 ListDeque。这些新方法可能与具有相同名称和实参类型的其他接口和类中的方法不兼容。如果方法签名发生不兼容的冲突,javac 编译器会输出 build 时错误。例如:

    错误示例 1:

    javac MyList.java
    
    MyList.java:135: error: removeLast() in MyList cannot implement removeLast() in List
      public void removeLast() {
                  ^
      return type void is not compatible with Object
      where E is a type-variable:
        E extends Object declared in interface List
    

    错误示例 2:

    javac MyList.java
    
    MyList.java:7: error: types Deque<Object> and List<Object> are incompatible;
    public class MyList implements  List<Object>, Deque<Object> {
      both define reversed(), but with unrelated return types
    1 error
    

    错误示例 3:

    javac MyList.java
    
    MyList.java:43: error: types List<E#1> and MyInterface<E#2> are incompatible;
    public static class MyList implements List<Object>, MyInterface<Object> {
      class MyList inherits unrelated defaults for getFirst() from types List and MyInterface
      where E#1,E#2 are type-variables:
        E#1 extends Object declared in interface List
        E#2 extends Object declared in interface MyInterface
    1 error
    

    如需修复这些 build 错误,实现这些接口的类应使用兼容的返回类型替换相应方法。例如:

    @Override
    public Object getFirst() {
        return List.super.getFirst();
    }
    

Segurança

O Android 15 inclui mudanças que promovem a segurança do sistema para proteger apps e usuários contra apps maliciosos.

Versões TLS restritas

O Android 15 restringe o uso das versões 1.0 e 1.1 do TLS. Essas versões foram descontinuadas no Android, mas agora não são mais permitidas para apps destinados ao Android 15.

Início de atividades em segundo plano protegidas

O Android 15 protege os usuários contra apps maliciosos e oferece mais controle sobre os dispositivos, adicionando mudanças que impedem que apps maliciosos em segundo plano tragam outros apps para o primeiro plano, elevando os privilégios e abusando da interação do usuário. Os lançamentos de atividades em segundo plano estão restritos desde Android 10 (nível 29 da API)

Outras mudanças

Além da restrição para correspondência de UID, estas outras mudanças também foram incluídas:

  • Os criadores de PendingIntent vão bloquear o início de atividades em segundo plano por padrão. Isso ajuda a evitar que os apps criem acidentalmente um PendingIntent que podem ser usados indevidamente por usuários maliciosos.
  • Não levar um app para o primeiro plano, a menos que o remetente PendingIntent permite. Essa mudança tem como objetivo impedir que apps maliciosos abusem da capacidade de iniciar atividades em segundo plano. Por padrão, os apps não podem trazer a pilha de tarefas para o primeiro plano, a menos que o criador permita privilégios de inicialização de atividades em segundo plano ou que o remetente tenha privilégios de inicialização de atividades em segundo plano.
  • Controlar como a principal atividade de uma pilha de tarefas pode concluir a tarefa. Se o principal atividade finaliza uma tarefa, o Android volta para a tarefa que ativo pela última vez. Além disso, se uma atividade não principal terminar a tarefa, o Android voltar para a tela inicial não vai bloquear o final desse objeto atividades.
  • Impedir o lançamento de atividades arbitrárias de outros apps na sua própria tarefa. Essa mudança impede que apps maliciosos façam phishing pelos usuários criando atividades que parecem ser de outros apps.
  • Impeça que janelas não visíveis sejam consideradas para atividades em segundo plano é lançado. Isso ajuda a evitar que apps maliciosos abusem de inicializações de atividade em segundo plano para mostrar conteúdo indesejado ou malicioso aos usuários.

Intents mais seguras

O Android 15 apresenta novas medidas de segurança opcionais para tornar as intents mais seguras e mais robusto. Essas mudanças têm como objetivo evitar possíveis vulnerabilidades e o uso indevido de intents que podem ser exploradas por apps maliciosos. Existem duas Principais melhorias na segurança de intents no Android 15:

  • Combinar filtros de intent de destino:as intents que visam componentes específicos precisam corresponder com precisão às especificações do filtro de intent do destino. Se você enviar uma intent para iniciar a atividade de outro app, o componente de intent de destino precisa estar alinhado com os filtros de intent declarados da atividade de recebimento.
  • As intents precisam ter ações: as intents sem uma ação não vão mais corresponder a nenhum filtro de intent. Isso significa que as intents usadas para iniciar atividades ou serviços precisam ter uma ação claramente definida.

Para verificar como o app responde a essas mudanças, use StrictMode no app. Para conferir registros detalhados sobre violações de uso de Intent, adicione o seguinte método:

Kotlin

fun onCreate() {
    StrictMode.setVmPolicy(VmPolicy.Builder()
        .detectUnsafeIntentLaunch()
        .build()
    )
}

Java

public void onCreate() {
    StrictMode.setVmPolicy(new VmPolicy.Builder()
            .detectUnsafeIntentLaunch()
            .build());
}

Experiência do usuário e interface do sistema

O Android 15 inclui algumas mudanças que visam criar uma experiência do usuário mais consistente e intuitiva.

Mudanças no encarte da janela

Há duas mudanças relacionadas aos engastes de janela no Android 15: o engaste de borda a borda é forçado por padrão, e também há mudanças de configuração, como a configuração padrão das barras do sistema.

Aplicação de ponta a ponta

Os apps são de ponta a ponta por padrão em dispositivos com Android 15 se o app for destinado ao Android 15 (nível 35 da API).

Um app destinado ao Android 14 que não é de ponta a ponta em um dispositivo Android 15.


Um app direcionado ao Android 15 (nível 35 da API) e que é de ponta a ponta em um dispositivo Android 15. Este app usa principalmente componentes do Material 3 para Compose que aplicam encartes automaticamente. Essa tela não é afetada negativamente pela restrição de ponta a ponta do Android 15.

Essa é uma mudança incompatível que pode afetar negativamente a interface do seu app. As mudanças afetam as seguintes áreas da interface:

  • Barra de navegação com alça de gesto
    • Transparente por padrão.
    • O deslocamento da parte de baixo está desativado, então o conteúdo é mostrado por trás da barra de navegação do sistema, a menos que encartes sejam aplicados.
    • setNavigationBarColor e R.attr#navigationBarColor estão descontinuados e não afetam a navegação por gestos.
    • setNavigationBarContrastEnforced e R.attr#navigationBarContrastEnforced continuam sem efeito na navegação por gestos.
  • Navegação com três botões
    • Opacidade definida como 80% por padrão, com a cor possivelmente correspondendo ao plano de fundo da janela.
    • Deslocamento da parte de baixo desativado para que o conteúdo seja mostrado por trás da barra de navegação do sistema, a menos que encartes sejam aplicados.
    • setNavigationBarColor e R.attr#navigationBarColor são definidos para corresponder ao plano de fundo da janela por padrão. O plano de fundo da janela precisa ser um drawable de cor para que esse padrão seja aplicado. Essa API está descontinuada, mas continua afetando a navegação com três botões.
    • setNavigationBarContrastEnforced e R.attr#navigationBarContrastEnforced são verdadeiros por padrão, o que adiciona um plano de fundo 80% opaco na navegação com três botões.
  • Barra de status
    • Transparente por padrão.
    • O deslocamento da parte de cima é desativado para que o conteúdo seja mostrado por trás da barra de status, a menos que encartes sejam aplicados.
    • setStatusBarColor e R.attr#statusBarColor foram descontinuados e não têm efeito no Android 15.
    • setStatusBarContrastEnforced e R.attr#statusBarContrastEnforced foram descontinuados, mas ainda têm um efeito no Android 15.
  • Corte da tela
    • O layoutInDisplayCutoutMode de janelas não flutuantes precisa ser LAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS. SHORT_EDGES, NEVER e DEFAULT são interpretados como ALWAYS para que os usuários não vejam uma barra preta causada pelo corte da tela e apareçam de ponta a ponta.

O exemplo a seguir mostra um app antes e depois de ser direcionado ao Android 15 (nível 35 da API) e antes e depois de aplicar encartes.

Um app destinado ao Android 14 que não é de ponta a ponta em um dispositivo Android 15.
Um app que tem como destino o Android 15 (nível 35 da API) e é de ponta a ponta em um dispositivo Android 15. No entanto, muitos elementos agora estão ocultos pela barra de status, pela barra de navegação de três botões ou pelo corte da tela devido às imposições de ponta a ponta do Android 15. A interface oculta inclui a barra de apps superior do Material 2, botões de ação flutuantes e itens de lista.
Um app que é direcionado ao Android 15 (nível 35 da API), é de ponta a ponta em um dispositivo Android 15 e aplica encartes para que a interface não fique oculta.
O que verificar se o app já é de ponta a ponta

Se o app já for de ponta a ponta e aplicar encartes, você não será muito afetado, exceto nos seguintes cenários. No entanto, mesmo que você não ache que foi afetado, recomendamos que teste seu app.

  • Você tem uma janela não flutuante, como um Activity que usa SHORT_EDGES, NEVER ou DEFAULT em vez de LAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS. Se o app falhar na inicialização, isso pode ser devido à tela de apresentação. Você pode fazer upgrade da dependência core splashscreen para 1.2.0-alpha01 ou uma versão mais recente ou definir window.attributes.layoutInDisplayCutoutMode = WindowManager.LayoutInDisplayCutoutMode.always.
  • Pode haver telas com menos tráfego e interface obstruída. Verifique se essas telas menos visitadas não têm uma interface do usuário ocluída. As telas com menos tráfego incluem:
    • Telas de onboarding ou login
    • Páginas de configurações
O que verificar se o app ainda não é de ponta a ponta

Se o app ainda não estiver de ponta a ponta, é provável que você seja afetado. Além dos cenários para apps que já são de ponta a ponta, considere o seguinte:

  • Se o app usar componentes do Material 3 ( androidx.compose.material3) no Compose, como TopAppBar, BottomAppBar e NavigationBar, é provável que esses componentes não sejam afetados porque processam encartes automaticamente.
  • Se o app estiver usando componentes do Material 2 ( androidx.compose.material) no Compose, eles não vão processar encartes automaticamente. No entanto, você pode conseguir acesso aos encartes para fazer a aplicação manual. No androidx.compose.material 1.6.0 e versões mais recentes, use o parâmetro windowInsets para aplicar os encartes manualmente para BottomAppBar, TopAppBar, BottomNavigation e NavigationRail. Da mesma forma, use o parâmetro contentWindowInsets para Scaffold.
  • Se o app usar views e componentes do Material Design (com.google.android.material), a maioria dos componentes do Material baseados em views, como BottomNavigationView, BottomAppBar, NavigationRailView ou NavigationView, processa encartes e não exige mais trabalho. No entanto, você precisará adicionar android:fitsSystemWindows="true" se estiver usando AppBarLayout.
  • Para elementos combináveis personalizados, aplique os encartes manualmente como padding. Se o conteúdo estiver em um Scaffold, use os valores de padding Scaffold de encartes. Caso contrário, aplique o padding usando um dos WindowInsets.
  • Se o app estiver usando visualizações e BottomSheet, SideSheet ou contêineres personalizados, aplique padding usando ViewCompat.setOnApplyWindowInsetsListener. Para RecyclerView, aplique padding usando esse listener e também adicione clipToPadding="false".
O que verificar se o app precisa oferecer proteção de plano de fundo personalizada

Se o app precisar oferecer proteção de plano de fundo personalizada para a navegação com três botões ou a barra de status, ele deverá colocar um elemento combinável ou uma visualização atrás da barra de sistema usando WindowInsets.Type#tappableElement() para receber a altura da barra de navegação com três botões ou WindowInsets.Type#statusBars.

Outros recursos de ponta a ponta

Consulte os guias Visualizações de ponta a ponta e Compose de ponta a ponta para mais considerações sobre a aplicação de encartes.

APIs descontinuadas

As APIs a seguir estão descontinuadas, mas não desativadas:

As seguintes APIs estão descontinuadas e desativadas:

Configuração estável

Se o app for direcionado ao Android 15 (API de nível 35) ou versões mais recentes, Configuration não vai mais excluir as barras de sistema. Se você usar o tamanho da tela na classe Configuration para cálculo de layout, substitua por alternativas melhores, como um ViewGroup, WindowInsets ou WindowMetricsCalculator adequado, dependendo da sua necessidade.

O Configuration está disponível desde a API 1. Normalmente, ele é obtido de Activity.onConfigurationChanged. Ele fornece informações como densidade, orientação e tamanhos da janela. Uma característica importante sobre os tamanhos de janela retornados de Configuration é que ele excluía as barras de sistema.

O tamanho da configuração é usado normalmente para seleção de recursos, como /res/layout-h500dp, e ainda é um caso de uso válido. No entanto, o uso para cálculo de layout sempre foi desencorajado. Se você estiver fazendo isso, pare agora. Substitua o uso de Configuration por algo mais adequado, dependendo do seu caso de uso.

Se você usar para calcular o layout, use um ViewGroup adequado, como CoordinatorLayout ou ConstraintLayout. Se você usar para determinar a altura da barra de navegação do sistema, use WindowInsets. Se quiser saber o tamanho atual da janela do app, use computeCurrentWindowMetrics.

A lista a seguir descreve os campos afetados por essa mudança:

O atributo "elegantTextHeight" tem como padrão o valor "true".

For apps targeting Android 15 (API level 35), the elegantTextHeight TextView attribute becomes true by default, replacing the compact font used by default with some scripts that have large vertical metrics with one that is much more readable. The compact font was introduced to prevent breaking layouts; Android 13 (API level 33) prevents many of these breakages by allowing the text layout to stretch the vertical height utilizing the fallbackLineSpacing attribute.

In Android 15, the compact font still remains in the system, so your app can set elegantTextHeight to false to get the same behavior as before, but it is unlikely to be supported in upcoming releases. So, if your app supports the following scripts: Arabic, Lao, Myanmar, Tamil, Gujarati, Kannada, Malayalam, Odia, Telugu or Thai, test your app by setting elegantTextHeight to true.

elegantTextHeight behavior for apps targeting Android 14 (API level 34) and lower.
elegantTextHeight behavior for apps targeting Android 15.

A largura da TextView muda para formas de letras complexas

Nas versões anteriores do Android, algumas fontes cursivas ou linguagens que têm modelagem complexa podiam desenhar as letras na área do caractere anterior ou seguinte. Em alguns casos, essas letras eram cortadas na posição inicial ou final. No Android 15 e versões mais recentes, uma TextView aloca largura para desenhar espaço suficiente para essas letras e permite que os apps solicitem mais paddings à esquerda para evitar recortes.

Como essa mudança afeta a forma como um TextView decide a largura, o TextView aloca mais largura por padrão se o app for destinado ao Android 15 (nível 35 da API) ou mais recente. Para ativar ou desativar esse comportamento, chame a API setUseBoundsForWidth em TextView.

Como adicionar o padding à esquerda pode causar um desalinhamento nos layouts atuais, ele não é adicionado por padrão nem mesmo para apps direcionados ao Android 15 ou mais recente. No entanto, é possível adicionar um padding extra para evitar o corte chamando setShiftDrawingOffsetForStartOverhang.

Os exemplos a seguir mostram como essas mudanças podem melhorar o layout de texto para algumas fontes e idiomas.

Layout padrão para texto em inglês em uma fonte cursiva. Algumas das letras estão cortadas. Este é o XML correspondente:

<TextView
    android:fontFamily="cursive"
    android:text="java" />
Layout do mesmo texto em inglês com largura e padding adicionais. Este é o XML correspondente:

<TextView
    android:fontFamily="cursive"
    android:text="java"
    android:useBoundsForWidth="true"
    android:shiftDrawingOffsetForStartOverhang="true" />
Layout padrão para texto em tailandês. Algumas das letras estão cortadas. Este é o XML correspondente:

<TextView
    android:text="คอมพิวเตอร์" />
Layout do mesmo texto tailandês com largura e padding adicionais. Confira o XML correspondente:

<TextView
    android:text="คอมพิวเตอร์"
    android:useBoundsForWidth="true"
    android:shiftDrawingOffsetForStartOverhang="true" />

Altura da linha padrão sensível à localidade para EditText

在较低版本的 Android 中,文本布局会拉伸文本的高度,以满足与当前语言区域匹配的字体的行高。例如,如果内容是日语,由于日语字体的行高略高于拉丁字体,因此文本的高度会略高。不过,尽管行高存在这些差异,但无论使用的是哪种语言区域,EditText 元素的大小都是统一的,如下图所示:

三个框,表示可以包含英语 (en)、日语 (ja) 和缅甸语 (my) 文本的 EditText 元素。EditText 的高度相同,即使这些语言的行高各不相同。

对于以 Android 15(API 级别 35)为目标平台的应用,现在为 EditText 预留了最小行高,以匹配指定语言区域的参考字体,如下图所示:

三个框,表示可以包含英语 (en)、日语 (ja) 和缅甸语 (my) 文本的 EditText 元素。EditText 的高度现在包含足够的空间来容纳这些语言字体的默认行高。

如有需要,您的应用可以将 useLocalePreferredLineHeightForMinimum 属性指定为 false,以恢复之前的行为;您的应用还可以在 Kotlin 和 Java 中使用 setMinimumFontMetrics API 设置自定义最小垂直指标。

Câmera e mídia

O Android 15 faz as seguintes mudanças no comportamento de mídia e câmera para apps direcionados ao Android 15 ou versões mais recentes.

Restrições ao solicitar o foco de áudio

Os apps direcionados ao Android 15 (nível 35 da API) precisam ser o app principal ou executar um serviço em primeiro plano para solicitar o foco de áudio. Se um app tentar solicitar o foco quando não atender a um desses requisitos, a chamada vai retornar AUDIOFOCUS_REQUEST_FAILED.

Saiba mais sobre o foco de áudio em Gerenciar o foco de áudio.

Atualização das restrições não SDK

O Android 15 inclui listas atualizadas de interfaces não SDK restritas com base na colaboração com desenvolvedores Android e nos testes internos mais recentes. Antes de restringirmos interfaces não SDK, sempre que possível, garantimos que haja alternativas públicas disponíveis.

Caso seu app não seja destinado ao Android 15, é possível que algumas dessas mudanças não afetem você imediatamente. No entanto, embora seja possível que seu app acesse algumas interfaces não SDK dependendo do nível da API de destino do app, o uso de qualquer método ou campo não SDK sempre apresenta um alto risco de corromper o app.

Se você não souber se seu app usa interfaces não SDK, poderá testá-lo para descobrir. Se ele depende de interfaces não SDK, comece a planejar uma migração para alternativas SDK. No entanto, entendemos que alguns apps têm casos de uso válidos para interfaces não SDK. Se você não encontrar uma alternativa para deixar de usar uma interface não SDK em um recurso no seu app, solicite uma nova API pública.

To learn more about the changes in this release of Android, see Updates to non-SDK interface restrictions in Android 15. To learn more about non-SDK interfaces generally, see Restrictions on non-SDK interfaces.