A partir do nível 26 da API Android, as notificações persistentes são necessárias para serviços em primeiro plano. Esse requisito tem como objetivo impedir que você oculte serviços que possam exigir muito dos recursos do sistema, incluindo a bateria. Esse requisito cria um problema em potencial: se um app com vários serviços em primeiro plano não gerenciar a notificação com cuidado para que ela seja compartilhada entre todos os serviços, poderá haver várias notificações persistentes não dispensáveis, o que leva a uma confusão indesejada na lista ativa de notificações.
Esse problema se torna mais desafiador quando você usa SDKs, como o SDK do Navigation, que executam serviços em primeiro plano independentemente do app que tem notificações persistentes independentes, dificultando a consolidação.
Para resolver esses problemas, o SDK do Navigation v1.11 introduziu uma API simples para ajudar a gerenciar notificações persistentes no app, inclusive no SDK.

Componentes
O gerenciador de serviços em primeiro plano fornece um wrapper para a classe de serviço em primeiro plano do Android e a classe de notificação persistente. A principal função desse wrapper é aplicar a reutilização do ID de notificação para que a notificação seja compartilhada entre todos os serviços em primeiro plano que usam o gerenciador.

O SDK do Navigation contém métodos estáticos para inicializar e receber o singleton ForegroundServiceManager. Esse singleton só pode ser inicializado uma vez durante a vida útil do SDK do Navigation. Consequentemente, se você usar uma das chamadas de inicialização (initForegroundServiceManagerMessageAndIntent() ou initForegroundServiceManagerProvider()), envolva-a com um bloco try-catch caso esse caminho seja reinserido. O SDK do Navigation
gera uma exceção de execução se você chamar um dos métodos mais de uma vez, a menos que você
primeiro limpe todas as referências ao ForegroundServiceManager e chame
clearForegroundServiceManager() antes de cada chamada subsequente.
Os quatro parâmetros de initForegroundServiceManagerMessageAndIntent() são application, notificationId, defaultMessage e resumeIntent. Se os três últimos parâmetros forem nulos, a notificação será a notificação padrão do SDK do Navigation. Ainda é possível ocultar outros serviços em primeiro plano no app por trás dessa notificação. O parâmetro notificationId especifica o ID de notificação que deve ser usado para a notificação. Se for nulo, um valor arbitrário será usado. Você pode defini-lo explicitamente para evitar conflitos com outras notificações, como as de outro SDK. A defaultMessage é uma string exibida quando o sistema não está navegando. O resumeIntent é um intent que é disparado quando a notificação é clicada. Se o resumeIntent for nulo, os cliques na notificação serão ignorados.
Os três parâmetros de initForegroundServiceManagerProvider() são application, notificationId e notificationProvider. Se os dois últimos parâmetros forem nulos, a notificação será a notificação padrão do SDK do Navigation. O parâmetro notificationId especifica o ID de notificação que deve ser usado para a notificação. Se for nulo, um valor arbitrário será usado. Você pode defini-lo explicitamente para evitar conflitos com outras notificações, como as de outro SDK. Se o notificationProvider estiver definido, o provedor será sempre responsável por gerar a notificação a ser renderizada.
O método getForegroundServiceManager() do SDK do Navigation retorna o singleton do gerenciador de serviços em primeiro plano. Se você ainda não gerou um, ele é o equivalente a chamar initForegroundServiceManagerMessageAndIntent() com parâmetros nulos para notificationId, defaultMessage e resumeIntent.
O ForegroundServiceManager tem três métodos simples. Os dois primeiros são para mover um serviço para dentro e para fora do primeiro plano e geralmente são chamados de dentro do serviço que foi criado. O uso desses métodos garante que os serviços estejam associados à notificação persistente compartilhada. O método final, updateNotification(), sinaliza ao gerenciador que a notificação foi alterada e precisa ser renderizada novamente.
Se você precisar de controle total da notificação persistente compartilhada, a API vai fornecer uma interface NotificationContentProvider para definir um provedor de notificação, que contém um único método para receber uma notificação com o conteúdo atual. Ela também fornece uma classe de base, que você pode usar opcionalmente para ajudar a definir o provedor. Um dos principais propósitos da classe base é fornecer uma maneira de chamar updateNotification() sem a necessidade de acessar o ForegroundServiceManager. Se você usar uma instância do provedor de notificação para receber novas mensagens de notificação, poderá chamar esse método interno diretamente para renderizar a mensagem na notificação.
cenários de uso
Esta seção detalha os cenários de uso para notificações persistentes compartilhadas.
- Ocultar notificações persistentes de outros serviços em primeiro plano do app
- O cenário mais fácil é preservar o comportamento atual e usar apenas a notificação persistente para renderizar informações do SDK do Navigation. Outros serviços podem se ocultar por trás dessa notificação usando os métodos
startForeground()estopForeground()do gerenciador de serviços em primeiro plano. - Ocultar notificações persistentes de outros serviços em primeiro plano do app, mas definir texto padrão mostrado quando não estiver navegando
- O segundo cenário mais fácil é preservar o comportamento atual e usar apenas
a notificação persistente para renderizar informações do SDK do Navigation, exceto
quando o sistema não estiver navegando. Quando o sistema não está navegando, a
string fornecida para
initForegroundServiceManagerMessageAndIntent()é mostrada em vez da string padrão do SDK do Navigation que menciona "Google Maps". Você também pode usar essa chamada para definir o intent de retomada que é disparado quando a notificação é clicada. - Assumir o controle total da renderização da notificação persistente
- O cenário final exige a definição e a criação de um provedor de notificação
e a transmissão dele para o
ForegroundServiceManagerusandoinitForegroundServiceManagerProvider(). Essa opção oferece controle total do que é renderizado na notificação, mas também desconecta as informações de notificação do SDK do Navigation da notificação, removendo os avisos úteis de navegação guiada mostrados na notificação. O Google não oferece um meio simples de recuperar essas informações e inseri-las na notificação.
Exemplo de provedor de notificação
O exemplo de código a seguir demonstra como criar e retornar notificações usando um provedor de conteúdo de notificação simples.
public class NotificationContentProviderImpl
extends NotificationContentProviderBase
implements NotificationContentProvider {
private String channelId;
private Context context;
private String message;
/** Constructor */
public NotificationContentProviderImpl(Application application) {
super(application);
message = "-- uninitialized --";
channelId = null;
this.context = application;
}
/**
* Sets message to display in the notification. Calls updateNotification
* to display the message immediately.
*
* @param msg The message to display in the notification.
*/
public void setMessage(String msg) {
message = msg;
updateNotification();
}
/**
* Returns the notification as it should be rendered.
*/
@Override
public Notification getNotification() {
Notification notification;
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O) {
Spanned styledText = Html.fromHtml(message, FROM_HTML_MODE_LEGACY);
String channelId = getChannelId(context);
notification =
new Notification.Builder(context, channelId)
.setContentTitle("Notifications Demo")
.setStyle(new Notification.BigTextStyle()
.bigText(styledText))
.setSmallIcon(R.drawable.ic_navigation_white_24dp)
.setTicker("ticker text")
.build();
} else {
notification = new Notification.Builder(context)
.setContentTitle("Notification Demo")
.setContentText("testing non-O text")
.build();
}
return notification;
}
// Helper to set up a channel ID.
private String getChannelId(Context context) {
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O) {
if (channelId == null) {
NotificationManager notificationManager =
(NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
NotificationChannel channel = new NotificationChannel(
"default", "navigation", NotificationManager.IMPORTANCE_DEFAULT);
channel.setDescription("For navigation persistent notification.");
notificationManager.createNotificationChannel(channel);
channelId = channel.getId();
}
return channelId;
} else {
return "";
}
}
}
Depois de criar NotificationContentProviderImpl, conecte o SDK do Navigation a ele usando o seguinte código:
ForegroundServiceManager f = NavigationApi.getForegroundServiceManager(getApplication());
mNotification = new NotificationContentProviderImpl(getApplication());
NavigationApi.clearForegroundServiceManager();
NavigationApi.initForegroundServiceManagerProvider(getApplication(), null, mNotification);
Advertências e planos futuros
- Chame
initForegroundServiceManagerMessageAndIntent()ouinitForegroundServiceManagerProvider()no início para que o cenário de uso esperado seja bem definido. Você precisa chamar esse método antes de criar um novo navegador. - Capture exceções de chamadas para
initForegroundServiceManagerMessageAndIntent()ouinitForegroundServiceManagerProvider()caso o caminho do código seja inserido mais de uma vez. No SDK do Navigation v2.0, chamar esse método várias vezes gera uma exceção verificada em vez de uma exceção de execução. - O Google ainda pode ter trabalho a fazer para conseguir um estilo consistente durante a vida útil da notificação que corresponda ao estilo do cabeçalho.
- Ao definir um provedor de notificação, você pode controlar o comportamento de heads-up com a prioridade.
- O Google não oferece um meio simples de recuperar informações de navegação guiada que um provedor de notificação pode inserir na notificação.