Um Intent
é um objeto de mensagens que pode ser usado para solicitar uma ação
de outro componente do app.
Embora as intents facilitem a comunicação entre componentes de várias maneiras, há três
casos de uso fundamentais:
- Como iniciar uma atividade
Um
Activity
representa uma única tela em um app. É possível iniciar uma nova instância de umActivity
transmitindo umIntent
parastartActivity()
. OIntent
descreve a atividade a ser iniciada e carrega todos os dados necessários.Se você quiser receber um resultado da atividade quando ela for concluída, chame
startActivityForResult()
. Sua atividade recebe o resultado como um objetoIntent
separado no callbackonActivityResult()
da atividade. Para mais informações, consulte o guia Atividades. - Como iniciar um serviço
Um
Service
é um componente que executa operações em segundo plano sem uma interface do usuário. No Android 5.0 (nível 21 da API) e versões mais recentes, é possível iniciar um serviço comJobScheduler
. Para mais informações sobreJobScheduler
, consulte aAPI-reference documentation
.Para versões anteriores ao Android 5.0 (nível 21 da API), é possível iniciar um serviço usando métodos da classe
Service
. É possível iniciar um serviço para realizar uma operação única (como fazer o download de um arquivo) transmitindo umIntent
parastartService()
. OIntent
descreve o serviço a ser iniciado e carrega todos os dados necessários.Se o serviço for projetado com uma interface cliente-servidor, você poderá se vincular a ele de outro componente transmitindo um
Intent
parabindService()
. Para mais informações, consulte o guia Serviços. - Como transmitir um programa
Transmissão é uma mensagem que qualquer aplicativo pode receber. O sistema envia várias transmissões para eventos do sistema, como quando o sistema é inicializado ou o dispositivo começa a carregar. É possível transmitir para outros apps transmitindo um
Intent
parasendBroadcast()
ousendOrderedBroadcast()
.
O restante desta página explica como os intents funcionam e como usá-los. Para informações relacionadas, consulte Como interagir com outros apps e Como compartilhar conteúdo.
Tipos de intents
Há dois tipos de intents:
- As intents explícitas especificam qual componente de qual app vai atender à intent, especificando um
ComponentName
completo. Normalmente, você vai usar uma intent explícita para iniciar um componente no seu app, porque você sabe o nome da classe da atividade ou do serviço que quer iniciar. Por exemplo, você pode iniciar uma nova atividade no app em resposta a uma ação do usuário ou iniciar um serviço para fazer o download de um arquivo em segundo plano. - As intents implícitas não nomeiam um componente específico, mas declaram uma ação geral a ser realizada, o que permite que um componente de outro app a processe. Por exemplo, se você quiser mostrar um local no mapa para o usuário, use uma intent implícita para solicitar que outro app mostre um local especificado no mapa.
A figura 1 mostra como um intent é usado ao iniciar uma atividade. Quando o
objeto Intent
nomeia um componente de atividade específico explicitamente, o sistema
imediatamente inicia esse componente.

Figura 1. Como uma intent implícita é
enviada pelo sistema para iniciar outra atividade: [1] A atividade A cria um
Intent
com uma descrição de ação e o transmite para startActivity()
. [2] O sistema Android procura um filtro de intent que corresponda à intent em todos
os apps. Quando uma correspondência é encontrada, [3] o sistema
inicia a atividade correspondente (Atividade B) invocando o método onCreate()
e transmitindo o Intent
.
Quando você usa uma intent implícita, o sistema Android encontra o componente adequado para iniciar
comparando o conteúdo da intent aos filtros de intent declarados no arquivo de manifesto de outros apps no
dispositivo. Se a intent corresponder a um filtro de intent, o sistema vai iniciar esse componente e entregar
o objeto Intent
. Se vários filtros de intent forem compatíveis, o sistema
vai mostrar uma caixa de diálogo para que o usuário escolha qual app usar.
Um filtro de intent é uma expressão no arquivo de manifesto de um app que especifica o tipo de intents que o componente quer receber. Por exemplo, ao declarar um filtro de intent para uma atividade, você permite que outros apps iniciem sua atividade diretamente com um determinado tipo de intent. Da mesma forma, se você não declarar nenhum filtro de intent para uma atividade, ela só poderá ser iniciada com uma intent explícita.
Cuidado:para garantir que o app esteja protegido, sempre
use uma intent
explícita ao iniciar um Service
e não
declare filtros de intent para os serviços. O uso de uma intent implícita para iniciar um serviço representa um
risco de segurança, porque não é possível determinar qual serviço responderá à intent,
e o usuário não poderá ver qual serviço será iniciado. No Android 5.0 (nível 21 da API) e versões mais recentes, o sistema
gera uma exceção se você chamar bindService()
com uma intent implícita.
Criação de um intent
Um objeto Intent
carrega informações que o sistema Android usa
para determinar qual componente iniciar (como o nome exato do componente ou a categoria
do componente que deve receber a intent), além de informações que o componente receptor usa para
realizar a ação corretamente (como a ação a ser realizada e os dados a serem usados).
As informações principais de um Intent
são as seguintes:
- Nome do componente
- O nome do componente a ser iniciado.
Isso é opcional, mas é a informação essencial que torna uma intent explícita, o que significa que ela precisa ser transmitida apenas ao componente do app definido pelo nome do componente. Sem um nome de componente, a intent é implícita, e o sistema decide qual componente vai receber a intent com base nas outras informações de intent (como a ação, os dados e a categoria, descritos abaixo). Se você precisar iniciar um componente específico no app, especifique o nome dele.
Observação:ao iniciar um
Service
, especifique sempre o nome do componente. Caso contrário, não será possível determinar qual serviço vai responder à intent, e o usuário não poderá conferir qual serviço será iniciado.Esse campo do
Intent
é um objetoComponentName
, que pode ser especificado usando um nome de classe totalmente qualificado do componente de destino, incluindo o nome do pacote do app, por exemplo,com.example.ExampleActivity
. É possível definir o nome do componente comsetComponent()
,setClass()
,setClassName()
ou com o construtorIntent
. - Ação
- Uma string que especifica a ação genérica a ser realizada, como view ou pick.
No caso de um intent de transmissão, essa é a ação que entrou em vigor e que está sendo relatada. A ação determina em grande parte como o restante da intent é estruturado, principalmente as informações contidas nos dados e extras.
Você pode especificar suas próprias ações para uso por intents no app (ou para uso por outros apps para invocar componentes no app), mas geralmente especifica constantes de ação definidas pela classe
Intent
ou outras classes do framework. Confira algumas ações comuns para iniciar uma atividade:ACTION_VIEW
- Use essa ação em uma intent com
startActivity()
quando você tiver algumas informações que uma atividade pode mostrar ao usuário, como uma foto para visualizar em um app de galeria ou um endereço para visualizar em um app de mapa. ACTION_SEND
- Também conhecida como a intent share, ela precisa ser usada em uma intent com
startActivity()
quando você tem dados que o usuário pode compartilhar por outro app, como um app de e-mail ou de compartilhamento em redes sociais.
Consulte a referência da classe
Intent
para mais constantes que definem ações genéricas. Outras ações são definidas em outro lugar do framework do Android, como emSettings
para ações que abrem telas específicas no app Configurações do sistema.É possível especificar a ação de uma intent com
setAction()
ou com um construtorIntent
.Se você definir suas próprias ações, inclua o nome do pacote do app como prefixo, conforme mostrado no exemplo abaixo:
Kotlin
const val ACTION_TIMETRAVEL = "com.example.action.TIMETRAVEL"
Java
static final String ACTION_TIMETRAVEL = "com.example.action.TIMETRAVEL";
- Dados
- O URI (um objeto
Uri
) que faz referência aos dados a serem processados e/ou ao tipo MIME desses dados. O tipo de dados fornecidos geralmente é determinado pela ação da intent. Por exemplo, se a ação forACTION_EDIT
, os dados precisarão conter o URI do documento a ser editado.Ao criar uma intent, é importante especificar o tipo de dados (tipo MIME) além do URI. Por exemplo, uma atividade que pode mostrar imagens provavelmente não poderá reproduzir um arquivo de áudio, mesmo que os formatos de URI sejam semelhantes. Especificar o tipo MIME dos dados ajuda o sistema Android a encontrar o melhor componente para receber a intent. No entanto, o tipo MIME às vezes pode ser inferido do URI, principalmente quando os dados são um URI
content:
. Um URIcontent:
indica que os dados estão localizados no dispositivo e controlados por umContentProvider
, o que torna o tipo MIME dos dados visível para o sistema.Para definir apenas o URI de dados, chame
setData()
. Para definir apenas o tipo MIME, chamesetType()
. Se necessário, você pode definir os dois explicitamente comsetDataAndType()
.Atenção:se você quiser definir o URI e o tipo MIME, não chame
setData()
esetType()
, porque eles anulam o valor um do outro. Sempre usesetDataAndType()
para definir o URI e o tipo MIME. - Categorias
- Uma string que contém informações adicionais sobre o tipo de componente
que processa a intent. Qualquer número de descrições de categoria pode ser
colocado em uma intent, mas a maioria delas não exige uma categoria.
Confira algumas categorias comuns:
CATEGORY_BROWSABLE
- A atividade de destino permite ser iniciada por um navegador da Web para mostrar dados referenciados por um link, como uma imagem ou uma mensagem de e-mail.
CATEGORY_LAUNCHER
- A atividade é a atividade inicial de uma tarefa e é listada na tela de início de aplicativos do sistema.
Consulte a descrição da classe
Intent
para conferir a lista completa de categorias.É possível especificar uma categoria com
addCategory()
.
Essas propriedades listadas acima (nome do componente, ação, dados e categoria) representam as características definidoras de uma intent. Ao ler essas propriedades, o sistema Android é capaz de resolver qual componente do app ele deve iniciar. No entanto, uma intent pode carregar informações adicionais que não afetam como ela é resolvida para um componente do app. Os intents também podem fornecer o seguinte:
- Extras
- Pares de chave-valor que contêm informações adicionais necessárias para realizar
a ação solicitada.
Assim como algumas ações usam determinados tipos de URIs de dados, outras também usam determinados extras.
É possível adicionar dados extras com vários métodos
putExtra()
, cada um aceitando dois parâmetros: o nome da chave e o valor. Você também pode criar um objetoBundle
com todos os dados extras e inserir oBundle
noIntent
computExtras()
.Por exemplo, ao criar uma intent para enviar um e-mail com
ACTION_SEND
, você pode especificar o destinatário to com a chaveEXTRA_EMAIL
e o assunto com a chaveEXTRA_SUBJECT
.A classe
Intent
especifica muitas constantesEXTRA_*
para tipos de dados padronizados. Se você precisar declarar suas próprias chaves extras (para intents que o app recebe), inclua o nome do pacote do app como um prefixo, conforme mostrado no exemplo abaixo:Kotlin
const val EXTRA_GIGAWATTS = "com.example.EXTRA_GIGAWATTS"
Java
static final String EXTRA_GIGAWATTS = "com.example.EXTRA_GIGAWATTS";
Cuidado: não use dados
Parcelable
ouSerializable
ao enviar uma intent que você espera que outro app receba. Se um app tentar acessar dados em um objetoBundle
, mas não tiver acesso à classe parcelada ou serializada, o sistema vai gerar umaRuntimeException
. - Flags As flags
- são definidas na classe
Intent
que funcionam como metadados para a intent. As flags podem instruir o sistema Android sobre como iniciar uma atividade (por exemplo, a tarefa a que a atividade pertence) e como tratá-la depois que ela for iniciada (por exemplo, se ela pertence à lista de atividades recentes).Para mais informações, consulte o método
setFlags()
.
Exemplo de intent explícito
Uma intent explícita é aquela que você usa para iniciar um componente específico do app, como
uma atividade ou serviço específico no app. Para criar uma intent explícita, defina
o nome do componente para o objeto Intent
. Todas
as outras propriedades de intent são opcionais.
Por exemplo, se você criou um serviço no app chamado DownloadService
,
projetado para fazer o download de um arquivo da Web, é possível iniciá-lo com este código:
Kotlin
// Executed in an Activity, so 'this' is theContext
// The fileUrl is a string URL, such as "http://www.example.com/image.png" val downloadIntent = Intent(this, DownloadService::class.java).apply { data =Uri.parse
(fileUrl) } startService(downloadIntent)
Java
// Executed in an Activity, so 'this' is theContext
// The fileUrl is a string URL, such as "http://www.example.com/image.png" Intent downloadIntent = new Intent(this, DownloadService.class); downloadIntent.setData(Uri.parse
(fileUrl)); startService(downloadIntent);
O construtor Intent(Context, Class)
fornece o Context
do app e o
componente um objeto Class
. Assim,
essa intent inicia explicitamente a classe DownloadService
no app.
Para mais informações sobre como criar e iniciar um serviço, consulte o guia Serviços.
Exemplo de intent implícito
Uma intent implícita especifica uma ação que pode invocar qualquer app no dispositivo capaz de realizar a ação. O uso de uma intent implícita é útil quando o app não pode realizar a ação, mas outros apps provavelmente podem e você quer que o usuário escolha qual app usar.
Por exemplo, se você tiver conteúdo que quer que o usuário compartilhe com outras pessoas,
crie uma intent
com a ação ACTION_SEND
e adicione extras que especifiquem o conteúdo a ser compartilhado. Quando você chama
startActivity()
com essa intent, o usuário pode
escolher um app para compartilhar o conteúdo.
Kotlin
// Create the text message with a string. val sendIntent = Intent().apply { action = Intent.ACTION_SEND putExtra(Intent.EXTRA_TEXT, textMessage) type = "text/plain" } // Try to invoke the intent. try { startActivity(sendIntent) } catch (e: ActivityNotFoundException) { // Define what your app should do if no activity can handle the intent. }
Java
// Create the text message with a string. Intent sendIntent = new Intent(); sendIntent.setAction(Intent.ACTION_SEND); sendIntent.putExtra(Intent.EXTRA_TEXT, textMessage); sendIntent.setType("text/plain"); // Try to invoke the intent. try { startActivity(sendIntent); } catch (ActivityNotFoundException e) { // Define what your app should do if no activity can handle the intent. }
Quando startActivity()
é chamado, o sistema
examina todos os apps instalados para determinar quais deles podem processar esse tipo de intent (uma
intent com a ação ACTION_SEND
e que carrega dados "text/plain"). Se houver apenas um app que possa processá-la, ele será aberto imediatamente e receberá a
intent. Se nenhum outro app puder processá-la, seu app poderá capturar a
ActivityNotFoundException
que ocorre. Se várias atividades aceitarem a intent, o sistema
vai mostrar uma caixa de diálogo como a mostrada na Figura 2 para que o usuário escolha qual app usar.
Mais informações sobre como iniciar outros apps também são fornecidas no guia sobre enviar o usuário para outro app.

Figura 2. Caixa de diálogo seletora.
Como forçar um seletor de aplicativo
Quando há mais de um app que responde à intent implícita, o usuário pode selecionar qual app usar e fazer com que ele seja a escolha padrão para a ação. A capacidade de selecionar um padrão é útil ao realizar uma ação para a qual o usuário provavelmente quer usar o mesmo app todas as vezes, como ao abrir uma página da Web (os usuários geralmente preferem apenas um navegador da Web).
No entanto, se vários apps puderem responder à intent e o usuário quiser usar um app
diferente a cada vez, mostre explicitamente uma caixa de diálogo de escolha. A caixa de diálogo seletora pede ao
usuário para selecionar qual app usar para a ação. Não é possível selecionar um app padrão para
a ação. Por exemplo, quando o app realiza o "compartilhamento" com a ação ACTION_SEND
, os usuários podem querer compartilhar usando um app diferente, dependendo
da situação atual. Portanto, sempre use a caixa de diálogo de escolha, conforme mostrado na Figura 2.
Para mostrar o seletor, crie uma Intent
usando createChooser()
e transmita-o para startActivity()
, conforme mostrado no exemplo a seguir.
Este exemplo mostra uma caixa de diálogo com uma lista de apps que respondem à intent transmitida ao método createChooser()
e usa o texto fornecido como título da
caixa de diálogo.
Kotlin
val sendIntent = Intent(Intent.ACTION_SEND) ... // Always use string resources for UI text. // This says something like "Share this photo with" val title: String = resources.getString(R.string.chooser_title) // Create intent to show the chooser dialog val chooser: Intent = Intent.createChooser(sendIntent, title) // Verify the original intent will resolve to at least one activity if (sendIntent.resolveActivity(packageManager) != null) { startActivity(chooser) }
Java
Intent sendIntent = new Intent(Intent.ACTION_SEND); ... // Always use string resources for UI text. // This says something like "Share this photo with" String title = getResources().getString(R.string.chooser_title); // Create intent to show the chooser dialog Intent chooser = Intent.createChooser(sendIntent, title); // Verify the original intent will resolve to at least one activity if (sendIntent.resolveActivity(getPackageManager()) != null) { startActivity(chooser); }
Detectar inicializações de intents não seguras
O app pode iniciar intents para navegar entre componentes dentro do app ou para realizar uma ação em nome de outro app. Para melhorar a segurança da plataforma, o Android 12 (nível 31 da API) e versões mais recentes oferecem um recurso de depuração que avisa se o app realiza uma inicialização não segura de uma intent. Por exemplo, o app pode executar uma inicialização não segura de uma intent aninhada, que é transmitida como um extra em outra intent.
Se o app realizar as duas ações a seguir, o sistema vai detectar uma inicialização de intent não segura e ocorrerá uma violação de StrictMode:
- O app separa as intents aninhadas das outras intents enviadas.
- O app imediatamente inicia um componente
do app usando essa intent aninhada,
como ao transmitir a intent para
startActivity()
,startService()
oubindService()
.
Para mais detalhes sobre como identificar essa situação e fazer mudanças no app, leia a postagem do blog sobre Android Nesting Intents no Medium.
Verificar lançamentos de intents não seguras
Para verificar se há inicializações de intents não seguras no app, chame
detectUnsafeIntentLaunch()
ao configurar o VmPolicy
, conforme mostrado no snippet de código a seguir. Caso
o app detecte uma violação de StrictMode, é recomendado interromper a execução
para proteger informações possivelmente confidenciais.
Kotlin
fun onCreate() { StrictMode.setVmPolicy(VmPolicy.Builder() // Other StrictMode checks that you've previously added. // ... .detectUnsafeIntentLaunch() .penaltyLog() // Consider also adding penaltyDeath() .build()) }
Java
protected void onCreate() { StrictMode.setVmPolicy(new VmPolicy.Builder() // Other StrictMode checks that you've previously added. // ... .detectUnsafeIntentLaunch() .penaltyLog() // Consider also adding penaltyDeath() .build()); }
Usar intents de forma mais responsável
Para minimizar a chance de um lançamento de intent não seguro e uma violação do StrictMode, siga estas práticas recomendadas.
Copie apenas os extras essenciais nas intents e execute todas as limpezas
e validações necessárias. O app pode copiar os extras de uma intent para
outra usada para iniciar um novo componente. Isso ocorre quando o
app chama
putExtras(Intent)
ou
putExtras(Bundle)
.
Se o app executar uma dessas operações, copie apenas os extras esperados
pelo componente de recebimento. Se a outra intent (que recebe a cópia)
iniciar um componente que não foi
exportado, limpe e
valide os extras antes de copiá-los para a intent que inicia
o componente.
Não exporte os componentes do app sem necessidade. Por exemplo, se você
pretender iniciar um componente do app usando uma intent interna aninhada, defina o
atributo android:exported
do componente como false
.
Use uma PendingIntent
em vez de uma
intent aninhada. Dessa forma, quando outro app desempacotar a PendingIntent
da
Intent
que o contém, ele poderá iniciar a PendingIntent
usando a
identidade do seu app. Essa configuração permite que o outro app inicie com segurança
qualquer componente, incluindo um componente não exportado, no seu app.
O diagrama na Figura 2 mostra como o sistema transmite o controle do seu app (cliente) para outro app (serviço) e vice-versa:
- O app cria uma intent que invoca uma atividade em outro app. Dentro
dessa intent, você adiciona um objeto
PendingIntent
como um extra. Essa intent pendente invocou um componente no seu app, que não foi exportado. - Ao receber a intent do app, o outro app extrai o objeto
PendingIntent
aninhado. - O outro app invoca o método
send()
no objetoPendingIntent
. - Depois de transmitir o controle de volta ao app, o sistema invoca a intent pendente usando o contexto do app.
Figura 2. Diagrama da comunicação entre apps ao usar uma intent pendente anexada.
Como receber um intent implícito
Para anunciar quais intents implícitas seu app pode receber, declare um ou mais filtros de intent para
cada um dos componentes do app com um elemento <intent-filter>
no arquivo de manifesto.
Cada filtro de intent especifica o tipo de intent aceito com base na ação,
nos dados e na categoria da intent. O sistema vai enviar uma intent implícita para o componente do app somente se ela
puder passar por um dos filtros de intent.
Observação:uma intent explícita é sempre enviada para o destino, independente dos filtros de intent declarados pelo componente.
Um componente do app precisa declarar filtros separados para cada tarefa única que ele pode executar.
Por exemplo, uma atividade em um app de galeria de imagens pode ter dois filtros: um para
visualizar uma imagem e outro para editá-la. Quando a atividade é iniciada,
ela inspeciona o Intent
e decide como se comportar com base nas informações
no Intent
(como mostrar ou não os controles do editor).
Cada filtro de intent é definido por um elemento <intent-filter>
no arquivo de manifesto do app, aninhado no componente correspondente (por exemplo,
um elemento
<activity>
).
Em cada componente do app que inclui um elemento <intent-filter>
,
defina explicitamente um valor para
android:exported
.
Esse atributo indica se o componente do app é acessível a outros apps. Em algumas
situações, como atividades cujos filtros de intent incluem a
categoria LAUNCHER
, é útil definir esse atributo como true
. Caso contrário,
é mais seguro definir esse atributo como false
.
Aviso:caso uma atividade, um serviço ou um broadcast
receiver no app use filtros de intent e não defina explicitamente o valor
para android:exported
, não será possível instalar o app em um dispositivo
com o Android 12 ou mais recente.
Dentro do <intent-filter>
,
é possível especificar o tipo de intents a serem aceitas usando um ou mais
destes três elementos:
<action>
- Declara a ação de intent aceita no atributo
name
. O valor precisa ser o valor literal da string de uma ação, não a constante de classe. <data>
- Declara o tipo de dados aceito, usando um ou mais atributos que especificam vários
aspectos do URI de dados (
scheme
,host
,port
,path
) e o tipo MIME. <category>
- Declara a categoria de intent aceita no atributo
name
. O valor precisa ser o valor literal de string de uma ação, não a constante de classe.Observação:para receber intents implícitas, inclua a categoria
CATEGORY_DEFAULT
no filtro de intents. Os métodosstartActivity()
estartActivityForResult()
tratam todas as intents como se elas declarassem a categoriaCATEGORY_DEFAULT
. Se você não declarar essa categoria no filtro de intent, nenhuma intent implícita será resolvida para a atividade.
Por exemplo, este é um exemplo de declaração de atividade com um filtro de intent para receber uma
intent ACTION_SEND
quando o tipo de dados é texto:
<activity android:name="ShareActivity" android:exported="false"> <intent-filter> <action android:name="android.intent.action.SEND"/> <category android:name="android.intent.category.DEFAULT"/> <data android:mimeType="text/plain"/> </intent-filter> </activity>
É possível criar um filtro que inclua mais de uma instância de
<action>
,
<data>
ou
<category>
.
Se você fizer isso, precisa ter certeza de que o componente pode processar todas as
combinações desses elementos de filtro.
Quando você quer processar vários tipos de intent, mas apenas em combinações específicas de ação, dados e tipo de categoria, é necessário criar vários filtros de intent.
Uma intent implícita é testada em um filtro comparando a intent com cada um dos três elementos. Para ser entregue ao componente, o intent deve passar por todos os três testes. Se ele não corresponder a pelo menos uma delas, o sistema Android não vai transmitir a intent para o componente. No entanto, como um componente pode ter vários filtros de intent, uma intent que não passa por um dos filtros de um componente pode passar por outro. Mais informações sobre como o sistema resolve intents estão disponíveis na seção abaixo sobre resolução de intent.
Cuidado : usar um filtro de intent não é uma maneira segura de impedir que outros apps iniciem
seus componentes. Embora os filtros de intent restrinjam um componente a responder apenas a
determinados tipos de intents implícitas, outro app pode iniciar o componente do seu app
usando uma intent explícita se o desenvolvedor determinar os nomes dos componentes.
Se for importante que apenas seu app possa iniciar um dos seus componentes,
não declare filtros de intent no manifesto. Em vez disso, defina o atributo
exported
como "false"
para esse componente.
Da mesma forma, para evitar a execução acidental de um
Service
de um app diferente, sempre use uma intent explícita para iniciar seu próprio serviço.
Observação:para todas as atividades, é necessário declarar os filtros de intent no arquivo de manifesto.
No entanto, os filtros para broadcast receivers podem ser registrados dinamicamente chamando
registerReceiver()
. Em seguida, você pode cancelar o registro do broadcast receiver com unregisterReceiver()
. Isso permite que seu app
detecte transmissões específicas durante apenas um período especificado enquanto o app
está em execução.
Exemplos de filtros
Para demonstrar alguns dos comportamentos do filtro de intent, confira um exemplo do arquivo de manifesto de um app de compartilhamento social:
<activity android:name="MainActivity" android:exported="true"> <!-- This activity is the main entry, should appear in app launcher --> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> <activity android:name="ShareActivity" android:exported="false"> <!-- This activity handles "SEND" actions with text data --> <intent-filter> <action android:name="android.intent.action.SEND"/> <category android:name="android.intent.category.DEFAULT"/> <data android:mimeType="text/plain"/> </intent-filter> <!-- This activity also handles "SEND" and "SEND_MULTIPLE" with media data --> <intent-filter> <action android:name="android.intent.action.SEND"/> <action android:name="android.intent.action.SEND_MULTIPLE"/> <category android:name="android.intent.category.DEFAULT"/> <data android:mimeType="application/vnd.google.panorama360+jpg"/> <data android:mimeType="image/*"/> <data android:mimeType="video/*"/> </intent-filter> </activity>
A primeira atividade, MainActivity
, é o ponto de entrada principal do app, ou seja, a atividade que
é aberta quando o usuário inicia o app pela primeira vez com o ícone de acesso rápido:
- A ação
ACTION_MAIN
indica que esse é o ponto de entrada principal e não espera dados de intent. - A categoria
CATEGORY_LAUNCHER
indica que o ícone dessa atividade precisa ser colocado na tela de início do app do sistema. Se o elemento<activity>
não especificar um ícone comicon
, o sistema vai usar o ícone do elemento<application>
.
Esses dois devem ser pareados para que a atividade apareça no inicializador do aplicativo.
A segunda atividade, ShareActivity
, tem como objetivo facilitar o compartilhamento de conteúdo de texto e
mídia. Embora os usuários possam acessar essa atividade navegando até ela a partir de MainActivity
,
eles também podem acessar ShareActivity
diretamente de outro app que emite uma intent implícita
que corresponde a um dos dois filtros de intent.
Observação:o tipo MIME,
application/vnd.google.panorama360+jpg
, é um tipo de dados especial que especifica
fotos panorâmicas, que podem ser processadas com as APIs Google
panorama.
Corresponder intents a filtros de outros apps
Se outro app for direcionado ao Android 13 (nível 33 da API) ou mais recente, ele só poderá processar a
intent do seu app se ela corresponder às ações e categorias de um
elemento <intent-filter>
nesse outro app. Se o sistema não encontrar uma
correspondência, ele vai gerar uma
ActivityNotFoundException
.
O app de envio precisa processar
essa exceção.
Da mesma forma, se você atualizar o app para que ele seja direcionado ao Android 13
ou mais recente, todas as intents originadas de apps externos serão enviadas a um
componente exportado do app somente se a intent corresponder às ações e
categorias de um elemento <intent-filter>
declarado pelo app. Esse comportamento
ocorre independentemente da versão do SDK de destino do app de envio.
Nos casos a seguir, a correspondência de intent não é aplicada:
- Intents entregues a componentes que não declaram filtros de intent.
- Intents originadas de um mesmo app.
- Intents do sistema, ou seja, as intents enviadas do
UID do sistema (uid=1000). Apps do sistema incluem
system_server
e apps que definemandroid:sharedUserId
comoandroid.uid.system
. - Intents originadas de uma raiz.
Saiba mais sobre a correspondência de intenção.
Uso de um intent pendente
Um objeto PendingIntent
é um wrapper em torno de um objeto Intent
. O objetivo principal de uma PendingIntent
é conceder permissão a um aplicativo externo
para usar o Intent
contido como se ele fosse executado pelo
próprio processo do app.
Os principais casos de uso de um intent pendente são os seguintes:
- Declarar uma intent a ser executada quando o usuário realizar uma ação com sua notificação.
O
NotificationManager
do sistema Android executa oIntent
. - Declarar uma intent a ser executada quando o usuário realizar uma ação com o
widget do app
(o app da tela inicial executa o
Intent
). - Declarar uma intent a ser executada em um horário futuro especificado (o
AlarmManager
do sistema Android executa oIntent
).
Assim como cada objeto Intent
é projetado para ser processado por um tipo
específico de componente do app (Activity
, Service
ou
BroadcastReceiver
), um PendingIntent
também precisa ser
criado com a mesma consideração. Ao usar uma intent pendente, o app não
executa a intent com uma chamada, como startActivity()
. Em vez disso, declare o tipo de componente pretendido ao criar o
PendingIntent
chamando o método de criação correspondente:
PendingIntent.getActivity()
para umaIntent
que inicia umaActivity
.PendingIntent.getService()
para umaIntent
que inicia umaService
.PendingIntent.getBroadcast()
para umaIntent
que inicia umaBroadcastReceiver
.
A menos que seu app esteja recebendo intents pendentes de outros apps,
os métodos acima para criar uma PendingIntent
provavelmente são os únicos
PendingIntent
de que você vai precisar.
Cada método recebe a Context
do app atual, o
Intent
que você quer agrupar e uma ou mais flags que especificam
como a intent precisa ser usada, como se ela pode ser usada mais de uma vez.
Para mais informações sobre o uso de intents pendentes, consulte a documentação de cada dos respectivos casos de uso, como nos guias da API Notificações e Widgets de apps.
Especificar mutabilidade
Caso o app seja destinado ao Android 12 ou mais recente, especifique a
mutabilidade de cada objeto PendingIntent
criado pelo app. Para declarar que
um determinado objeto PendingIntent
é mutável ou imutável, use a flag
PendingIntent.FLAG_MUTABLE
ou
PendingIntent.FLAG_IMMUTABLE
, respectivamente.
Se o app tentar criar um objeto PendingIntent
sem definir uma flag de mutabilidade, o sistema gerará uma
IllegalArgumentException
e
a mensagem a seguir será exibida no Logcat:
PACKAGE_NAME: Targeting S+ (version 31 and above) requires that one of \
FLAG_IMMUTABLE or FLAG_MUTABLE be specified when creating a PendingIntent.
Strongly consider using FLAG_IMMUTABLE, only use FLAG_MUTABLE if \
some functionality depends on the PendingIntent being mutable, e.g. if \
it needs to be used with inline replies or bubbles.
Criar intents pendentes imutáveis sempre que possível
Na maioria dos casos, o app precisa criar objetos PendingIntent
imutáveis, conforme
mostrado no snippet de código a seguir. Se um objeto PendingIntent
for imutável,
outros apps não poderão modificar a intent para ajustar o resultado da invocação da
intent.
Kotlin
val pendingIntent = PendingIntent.getActivity(applicationContext, REQUEST_CODE, intent, /* flags */ PendingIntent.FLAG_IMMUTABLE)
Java
PendingIntent pendingIntent = PendingIntent.getActivity(getApplicationContext(), REQUEST_CODE, intent, /* flags */ PendingIntent.FLAG_IMMUTABLE);
No entanto, alguns casos de uso exigem objetos PendingIntent
mutáveis:
- Suporte a ações de resposta direta em
notificações. A
resposta direta exige uma mudança nos dados de clipe no objeto PendingIntent
associado à resposta. Normalmente, essa mudança é solicitada transmitindo
FILL_IN_CLIP_DATA
como uma flag para o métodofillIn()
. - Associar notificações ao framework do Android Auto usando instâncias de
CarAppExtender
. - Colocar conversas em balões usando instâncias
de
PendingIntent
. Um objetoPendingIntent
mutável permite que o sistema aplique as flags corretas, comoFLAG_ACTIVITY_MULTIPLE_TASK
eFLAG_ACTIVITY_NEW_DOCUMENT
. - Solicitar informações de localização do dispositivo chamando
requestLocationUpdates()
ou APIs semelhantes. O objetoPendingIntent
mutável permite que o sistema adicione extras de intent que representam eventos de ciclo de vida de local. Esses eventos incluem uma mudança de local e a disponibilidade de um provedor. - Programação de alarmes usando
AlarmManager
. O objeto mutávelPendingIntent
permite que o sistema adicione o extra de intentEXTRA_ALARM_COUNT
. Esse extra representa o número de vezes que um alarme repetitivo foi acionado. Ao conter esse extra, a intent pode notificar um app com precisão se um alarme recorrente foi acionado várias vezes, por exemplo, quando o dispositivo estava em suspensão.
Se o app criar um objeto PendingIntent
mutável, é altamente recomendável
que você use uma intent explícita e preencha o
ComponentName
. Dessa forma, todas as vezes que
outro app invocar PendingIntent
e devolver o controle para o app, o
mesmo componente será iniciado.
Usar intents explícitas em intents pendentes
Para definir melhor como outros apps podem usar as intents pendentes do seu app, sempre envolva uma intent pendente em uma intent explícita. Para ajudar a seguir essa prática recomendada, faça o seguinte:
- Verifique se os campos de ação, pacote e componente da intent de base estão definidos.
-
Use
FLAG_IMMUTABLE
, adicionado no Android 6.0 (nível 23 da API), para criar intents pendentes. Essa flag impede que os apps que recebem umaPendingIntent
preencham propriedades não preenchidas. Se ominSdkVersion
do app for22
ou mais recente, você poderá oferecer segurança e compatibilidade usando o seguinte código:if (Build.VERSION.SDK_INT >= 23) { // Create a PendingIntent using FLAG_IMMUTABLE. } else { // Existing code that creates a PendingIntent. }
Resolução de intents
Quando o sistema recebe uma intent implícita para iniciar uma atividade, ele procura a melhor atividade para a intent comparando-a com filtros de intent com base em três aspectos:
- Ação
- Dados (URI e tipo de dados)
- Categoria
As seções a seguir descrevem como as intents são associadas aos componentes apropriados de acordo com a declaração do filtro de intent no arquivo de manifesto de um app.
Teste de ação
Para especificar as ações de intent aceitas, um filtro de intent pode declarar zero ou mais
elementos <action>
, conforme mostrado no exemplo a seguir:
<intent-filter> <action android:name="android.intent.action.EDIT" /> <action android:name="android.intent.action.VIEW" /> ... </intent-filter>
Para passar por esse filtro, a ação especificada em Intent
precisa corresponder a uma das ações listadas no filtro.
Se o filtro não listar nenhuma ação, não haverá nada para uma
intent corresponder, então todas as intents vão falhar no teste. No entanto, se um Intent
não especificar uma ação, ele vai passar no teste, desde que o filtro
tenha pelo menos uma ação.
Teste de categoria
Para especificar as categorias de intent aceitas, um filtro de intent pode declarar zero ou mais
elementos <category>
, conforme mostrado no exemplo a seguir:
<intent-filter> <category android:name="android.intent.category.DEFAULT" /> <category android:name="android.intent.category.BROWSABLE" /> ... </intent-filter>
Para que uma intent seja aprovada no teste de categoria, todas as categorias no Intent
precisam corresponder a uma categoria no filtro. O inverso não é necessário. O filtro de intent pode
declarar mais categorias do que as especificadas no Intent
, e o
Intent
ainda é transmitido. Portanto, uma intent sem categorias
sempre passa nesse teste, independentemente das categorias declaradas no filtro.
Observação:o Android aplica automaticamente a categoria CATEGORY_DEFAULT
a todas as intents implícitas transmitidas para startActivity()
e startActivityForResult()
.
Se você quiser que a atividade receba intents implícitas, ela precisará
incluir uma categoria para "android.intent.category.DEFAULT"
nos filtros de intent, como
mostrado no exemplo de <intent-filter>
anterior.
Teste de dados
Para especificar os dados de intent aceitos, um filtro de intent pode declarar zero ou mais
elementos <data>
, conforme mostrado no exemplo a seguir:
<intent-filter> <data android:mimeType="video/mpeg" android:scheme="http" ... /> <data android:mimeType="audio/mpeg" android:scheme="http" ... /> ... </intent-filter>
Cada elemento <data>
pode especificar uma estrutura de URI e um tipo de dados (tipo de mídia MIME).
Cada parte do URI é um atributo
separado: scheme
, host
, port
e path
:
<scheme>://<host>:<port>/<path>
O exemplo abaixo mostra possíveis valores para esses atributos:
content://com.example.project:200/folder/subfolder/etc
Neste URI, o esquema é content
, o host é com.example.project
,
a porta é 200
e o caminho é folder/subfolder/etc
.
Cada um desses atributos é opcional em um elemento <data>
,
mas há dependências lineares:
- Se não houver esquema especificado, o host será ignorado.
- Se não houver host especificado, a porta será ignorada.
- Se não houver esquema nem host especificado, o caminho será ignorado.
Quando o URI em uma intent é comparado a uma especificação de URI em um filtro, ele é comparado apenas às partes do URI incluídas no filtro. Exemplo:
- Se um filtro especificar apenas um esquema, todos os URIs com esse esquema vão corresponder ao filtro.
- Se um filtro especificar um esquema e uma autoridade, mas não um caminho, todos os URIs com o mesmo esquema e autoridade vão passar pelo filtro, independentemente dos caminhos.
- Se um filtro especificar um esquema, uma autoridade e um caminho, apenas URIs com o mesmo esquema, autoridade e caminho vão passar pelo filtro.
Observação:uma especificação de caminho pode conter um asterisco curinga (*) para exigir apenas uma correspondência parcial do nome do caminho.
O teste de dados compara o URI e o tipo MIME na intent com um URI e um tipo MIME especificado no filtro. As regras são as seguintes:
- Uma intent que não contém um URI nem um tipo MIME só vai passar no teste se o filtro não especificar nenhum URI ou tipo MIME.
- Uma intent que contém um URI, mas nenhum tipo MIME (nem explícito nem inferido do URI), só vai passar no teste se o URI corresponder ao formato de URI do filtro e o filtro também não especificar um tipo MIME.
- Uma intent que contém um tipo MIME, mas não um URI, só vai passar no teste se o filtro listar o mesmo tipo MIME e não especificar um formato de URI.
- Uma intent que contém um URI e um tipo MIME (explícito ou inferido do
URI) só vai passar pela parte de tipo MIME do teste se esse
tipo corresponder a um tipo listado no filtro. Ele passa a parte do URI do teste
se o URI corresponder a um URI no filtro ou se tiver um URI
content:
oufile:
e o filtro não especificar um URI. Em outras palavras, um componente é considerado compatível com dadoscontent:
efile:
se o filtro dele listar apenas um tipo MIME.
Observação:se uma intent especificar um URI ou tipo MIME, o teste de dados vai falhar se não houver elementos <data>
na <intent-filter>
.
Essa última regra, regra (d), reflete a expectativa
de que os componentes possam receber dados locais de um arquivo ou provedor de conteúdo.
Portanto, os filtros podem listar apenas um tipo de dados e não precisam nomear explicitamente os esquemas content:
e file:
.
O exemplo a seguir mostra um caso típico em que um elemento <data>
informa ao Android que o componente pode receber e exibir dados de imagem de um
provedor de conteúdo:
<intent-filter> <data android:mimeType="image/*" /> ... </intent-filter>
Os filtros que especificam um tipo de dados, mas não um URI, são talvez os mais comuns, porque a maioria dos dados disponíveis é distribuída pelos provedores de conteúdo.
Outra configuração comum é de filtros com um esquema e um tipo de dados. Por
exemplo, um elemento <data>
como o seguinte informa ao Android que
o componente pode extrair dados de vídeo da rede para realizar a ação:
<intent-filter> <data android:scheme="http" android:mimeType="video/*" /> ... </intent-filter>
Correspondência de intents
As intents são comparadas com filtros de intent não apenas para descobrir um componente
alvo a ser ativado, mas também para descobrir algo sobre o conjunto de
componentes no dispositivo. Por exemplo, o app Home preenche a tela de início
encontrando todas as atividades com filtros de intent que especificam a
ação ACTION_MAIN
e a
categoria CATEGORY_LAUNCHER
.
Uma correspondência só é bem-sucedida se as ações e categorias na intent corresponderem
ao filtro, conforme descrito na documentação da classe
IntentFilter
.
O aplicativo pode usar a correspondência de intents de modo semelhante ao feito pelo aplicativo Home.
O PackageManager
tem um conjunto de métodos query...()
que retornam todos os componentes que podem aceitar uma intent específica e
uma série semelhante de métodos resolve...()
que determinam o melhor
componente para responder a uma intent. Por exemplo,
queryIntentActivities()
retorna uma lista de todas as atividades que podem executar
a intent transmitida como um argumento, e queryIntentServices()
retorna uma lista semelhante de serviços.
Nenhum dos métodos ativa os componentes. Eles apenas listam aqueles que
podem responder. Há um método semelhante,
queryBroadcastReceivers()
, para broadcast receivers.