Este documento describe las tareas básicas de NFC que realizas en Android. Se explica cómo enviar y recibir datos NFC en forma de mensajes NDEF y se describen las APIs del framework de Android que admiten estas funciones. Para obtener información sobre temas más avanzados, incluida una discusión sobre cómo trabajar con datos que no son NDEF, consulta conceptos avanzados de NFC.
La lectura de los datos NDEF de una etiqueta NFC se realiza con el sistema de envío de etiquetas, que analiza las etiquetas NFC detectadas, categoriza los datos de forma apropiada y, luego, inicia una aplicación que está interesada en los datos categorizados. Una aplicación que desea controlar la etiqueta NFC escaneada puede declarar un filtro de intent y solicitar el manejo de los datos.
Sistema de envío de etiquetas
Por lo general, los dispositivos con Android buscan etiquetas NFC cuando la pantalla está desbloqueada, a menos que el NFC esté inhabilitado en el menú de configuración del dispositivo. Cuando un dispositivo con Android descubre una etiqueta NFC, el comportamiento deseado es hacer que la actividad más adecuada maneje el intent sin preguntarle al usuario qué aplicación usar. Como los dispositivos escanean las etiquetas NFC en un rango muy corto, es probable que hacer que los usuarios seleccionen manualmente una actividad los obligue a alejar el dispositivo de la etiqueta y romper la conexión. Debes desarrollar tu actividad para manejar solo las etiquetas NFC que le interesan a tu actividad para evitar que aparezca el Selector de actividad.
Para ayudarte con este objetivo, Android proporciona un sistema especial de envío de etiquetas que analiza las etiquetas NFC escaneadas y trata de ubicar las aplicaciones que están interesadas en los datos escaneados. Para ello, sigue estos pasos:
- Analizando la etiqueta NFC y averiguando el tipo MIME o un URI que identifica la carga útil de datos en la etiqueta
- Encapsulando el tipo MIME o URI y la carga útil en un intent. Estos dos primeros pasos se describen en Cómo se asignan las etiquetas NFC a los tipos MIME y los URI.
- Iniciando una actividad basada en el intent. Esto se describe en Cómo se envían las etiquetas NFC a las aplicaciones.
Cómo se asignan las etiquetas NFC a los tipos MIME y URI
Antes de comenzar a escribir tus aplicaciones NFC, es importante comprender los diferentes tipos de etiquetas NFC, cómo el sistema de envío de etiquetas analiza las etiquetas NFC y el trabajo especial que hace el sistema de envío de etiquetas cuando detecta un mensaje NDEF. Las etiquetas NFC vienen en una amplia gama de tecnologías y también pueden tener datos escritos de muchas maneras diferentes. Android tiene la mayor compatibilidad con el estándar NDEF, que se define en el Foro NFC.
Los datos NDEF se encapsulan dentro de un mensaje (NdefMessage
) que contiene uno o más registros (NdefRecord
). Cada registro NDEF debe tener el formato indicado por la especificación del tipo de registro que deseas crear. Android también admite otros tipos de etiquetas que no contienen datos NDEF, con los que puedes trabajar utilizando las clases incluidas en el paquete android.nfc.tech
. Para obtener más información sobre estas tecnologías, consulta el tema Conceptos avanzados de NFC. Trabajar con estos otros tipos de etiquetas implica escribir tu propia pila de protocolos para poder comunicarte con ellas, por lo que te recomendamos usar NDEF siempre que sea posible, ya que de esa forma facilitas el desarrollo y la compatibilidad máxima para los dispositivos con Android.
Nota: Para descargar las especificaciones completas de NDEF, ve al sitio de especificaciones del foro de NFC y documentos de la aplicación y consulta Cómo crear tipos comunes de registros de NDEF para ver ejemplos de cómo crear registros de NDEF.
Ahora que tienes más información sobre las etiquetas NFC, en las siguientes secciones se describe en más detalle cómo Android maneja las etiquetas con formato NDEF. Cuando un dispositivo con Android analiza una etiqueta NFC que contiene datos con formato NDEF, analiza el mensaje y trata de determinar el tipo MIME o URI de identificación de los datos. Para ello, el sistema lee el primer NdefRecord
dentro del NdefMessage
para determinar cómo interpretar todo el mensaje NDEF (un mensaje NDEF puede tener varios registros NDEF). En un mensaje NDEF bien formado, el primer NdefRecord
contiene los siguientes campos:
- TNF (Formato de nombre de tipo) de 3 bits
- Indica cómo interpretar el campo de tipo de longitud variable. Los valores válidos se describen en la tabla 1.
- Tipo de longitud variable
- Describe el tipo de registro. Si usas
TNF_WELL_KNOWN
, usa este campo para especificar la definición del tipo de registro (RTD). Los valores válidos de RTD se describen en la tabla 2. - ID de longitud variable
- Un identificador único para el registro. Este campo no se usa con frecuencia, pero si necesitas identificar una etiqueta de forma única, puedes crear un ID para ella.
- Carga útil de longitud variable
- La carga útil de datos reales que deseas leer o escribir. Un mensaje NDEF puede contener varios registros NDEF, así que no asumas que la carga útil completa se encuentra en el primer registro NDEF del mensaje NDEF.
El sistema de envío de etiquetas utiliza los campos TNF y de tipo para intentar asignar un tipo MIME o URI al mensaje NDEF. Si tiene éxito, encapsula esa información dentro de un intent ACTION_NDEF_DISCOVERED
junto con la carga útil real. Sin embargo, hay casos en que el sistema de envío de etiquetas no puede determinar el tipo de datos en función del primer registro NDEF. Esto sucede cuando los datos NDEF no pueden asignarse a un tipo MIME o URI, o cuando la etiqueta NFC no contiene datos NDEF. En esos casos, se encapsula un objeto Tag
que tenga información sobre las tecnologías de la etiqueta y la carga útil dentro de un intent ACTION_TECH_DISCOVERED
.
La tabla 1 describe cómo el sistema de envío de etiquetas asigna TNF y campos de tipo a tipos MIME o URI. También describe qué TNF no se puede asignar a un tipo MIME o URI.
En estos casos, el sistema de envío de etiquetas recurre a ACTION_TECH_DISCOVERED
.
Por ejemplo, si el sistema de envío de etiquetas encuentra un registro de tipo TNF_ABSOLUTE_URI
, asigna el campo de tipo de longitud variable de ese registro a un URI. El sistema de envío de etiquetas encapsula ese URI en el campo de datos de un intent ACTION_NDEF_DISCOVERED
junto con otra información sobre la etiqueta, como la carga útil. Por otro lado, si encuentra un registro de tipo TNF_UNKNOWN
, crea un intent que encapsula las tecnologías de la etiqueta.
Formato de nombre de tipo (TNF) | Mapeo |
---|---|
TNF_ABSOLUTE_URI |
Este URI se basa en el campo de tipo. |
TNF_EMPTY |
Recurre a ACTION_TECH_DISCOVERED . |
TNF_EXTERNAL_TYPE |
Este URI se basa en el URN en el campo de tipo. El URN está codificado en el campo de tipo NDEF en un formulario abreviado: <domain_name>:<service_name> .
Android lo asigna a un URI en el formulario: vnd.android.nfc://ext/<domain_name>:<service_name> . |
TNF_MIME_MEDIA |
Este tipo de MIME se basa en el campo de tipo. |
TNF_UNCHANGED |
No es válido en el primer registro, por lo que recurre a ACTION_TECH_DISCOVERED . |
TNF_UNKNOWN |
Recurre a ACTION_TECH_DISCOVERED . |
TNF_WELL_KNOWN |
Tipo MIME o URI según la definición de tipo de registro (RTD) que establezcas en el campo de tipo. Consulta la tabla 2 para obtener más información sobre las RTD disponibles y sus asignaciones. |
Definición de tipo de registro (RTD) | Mapeo |
---|---|
RTD_ALTERNATIVE_CARRIER |
Recurre a ACTION_TECH_DISCOVERED . |
RTD_HANDOVER_CARRIER |
Recurre a ACTION_TECH_DISCOVERED . |
RTD_HANDOVER_REQUEST |
Recurre a ACTION_TECH_DISCOVERED . |
RTD_HANDOVER_SELECT |
Recurre a ACTION_TECH_DISCOVERED . |
RTD_SMART_POSTER |
Este URI se basa en el análisis de la carga útil. |
RTD_TEXT |
Es el tipo MIME de text/plain . |
RTD_URI |
Este tipo de URI se basa en la carga útil. |
Cómo se envían las etiquetas NFC a las aplicaciones
Cuando el sistema de envío de etiquetas termina de crear un intent que encapsula la etiqueta NFC y su información de identificación, envía el intent a una aplicación interesada que lo filtra. Si más de una aplicación puede manejar el intent, se muestra el Selector de actividad para que el usuario pueda seleccionar la actividad. El sistema de envío de etiquetas define tres intents, que se enumeran en orden de mayor a menor prioridad:
-
ACTION_NDEF_DISCOVERED
: Este intent se usa para iniciar una actividad cuando se analiza una etiqueta que contiene una carga útil NDEF y es de un tipo reconocido. Este es el intent de mayor prioridad, y el sistema de envío de etiquetas intenta iniciar una actividad con este intent antes que cualquier otro, siempre que sea posible. ACTION_TECH_DISCOVERED
: Si no se registra ninguna actividad para controlar el intentACTION_NDEF_DISCOVERED
, el sistema de envío de etiquetas intenta iniciar una aplicación con este intent. Este intent también se inicia directamente (sin iniciarACTION_NDEF_DISCOVERED
primero) si la etiqueta que se escanea contiene datos NDEF que no se pueden asignar a un tipo MIME o URI, o si la etiqueta no contiene datos NDEF, pero contiene una tecnología de etiqueta conocida.ACTION_TAG_DISCOVERED
: Se inicia este intent si ninguna actividad maneja los intentsACTION_NDEF_DISCOVERED
oACTION_TECH_DISCOVERED
.
La forma básica en que funciona el sistema de envíos de etiquetas es la siguiente:
- Intenta iniciar una actividad con el intent creado por el sistema de envío de etiquetas analizando la etiqueta NFC (ya sea
ACTION_NDEF_DISCOVERED
oACTION_TECH_DISCOVERED
). - Si ninguna actividad filtra ese intent, intenta iniciar una actividad con la siguiente prioridad más baja (
ACTION_TECH_DISCOVERED
oACTION_TAG_DISCOVERED
) hasta que una aplicación filtre por el intent o hasta que el sistema de envío de etiquetas pruebe todos los intents posibles. - Si no se filtran aplicaciones para ninguno de los intents, no hagas nada.
Siempre que sea posible, trabaja con mensajes NDEF y el intent ACTION_NDEF_DISCOVERED
, porque es el más específico de los tres. Este intent te permite iniciar tu aplicación en un momento más apropiado que los otros dos intents, lo que brinda al usuario una mejor experiencia.
Cómo solicitar acceso NFC en el manifiesto de Android
Para poder acceder al hardware NFC de un dispositivo y manejar adecuadamente los intents NFC, declara estos elementos en el archivo AndroidManifest.xml
:
- El elemento NFC
<uses-permission>
para acceder al hardware NFC:<uses-permission android:name="android.permission.NFC" />
- La versión mínima de SDK con la que es compatible tu aplicación. El nivel de API 9 solo es compatible con el envío limitado de etiquetas a través de
ACTION_TAG_DISCOVERED
, y da acceso únicamente a los mensajes NDEF a través deEXTRA_NDEF_MESSAGES
extra. No se puede acceder a ninguna otra propiedad de etiqueta ni a operaciones de E/S. La API nivel 10 incluye soporte integral de lectura/escritura, así como la implementación de cambios NDEF en primer plano, y la API nivel 14 proporciona métodos de conveniencia adicionales para crear registros NDEF.<uses-sdk android:minSdkVersion="10"/>
- El elemento
uses-feature
para que tu aplicación aparezca en Google Play solo para dispositivos que tienen hardware NFC:<uses-feature android:name="android.hardware.nfc" android:required="true" />
Si tu aplicación usa la funcionalidad NFC, pero esa funcionalidad no es crucial para tu aplicación, puedes omitir el elemento
uses-feature
y verificar la disponibilidad de NFC en el tiempo de ejecución verificando sigetDefaultAdapter()
esnull
.
Cómo filtrar por intents de NFC
Para iniciar tu aplicación cuando se escanea una etiqueta NFC que deseas manejar, tu aplicación puede filtrar uno, dos o los tres intents de NFC en el manifiesto de Android. Sin embargo, por lo general, deseas filtrar el intent ACTION_NDEF_DISCOVERED
para obtener el mayor control de cuándo se inicia tu aplicación. El intent ACTION_TECH_DISCOVERED
es una alternativa para ACTION_NDEF_DISCOVERED
cuando no se filtran aplicaciones para ACTION_NDEF_DISCOVERED
o cuando la carga útil no es NDEF. Por lo general, filtrar por ACTION_TAG_DISCOVERED
es demasiado general para filtrar una categoría. Muchas aplicaciones filtrarán por ACTION_NDEF_DISCOVERED
o ACTION_TECH_DISCOVERED
antes de ACTION_TAG_DISCOVERED
, por lo que tu aplicación tiene una baja probabilidad de iniciarse. ACTION_TAG_DISCOVERED
solo está disponible como último recurso para filtrar aplicaciones en los casos en que no se instalan otras aplicaciones para controlar el intent ACTION_NDEF_DISCOVERED
o ACTION_TECH_DISCOVERED
.
Debido a que las implementaciones de etiquetas NFC varían y muchas veces no están bajo tu control, esto no siempre es posible, por lo que puedes recurrir a los otros dos intents cuando sea necesario. Cuando tienes control sobre los tipos de etiquetas y datos escritos, te recomendamos usar NDEF para dar formato a tus etiquetas. Las siguientes secciones describen cómo filtrar para cada tipo de intent.
ACTION_NDEF_DISCOVERED
Para filtrar por intents ACTION_NDEF_DISCOVERED
, declara el filtro de intents junto con el tipo de datos que deseas filtrar. El siguiente ejemplo filtra para intents ACTION_NDEF_DISCOVERED
con un tipo MIME de text/plain
:
<intent-filter> <action android:name="android.nfc.action.NDEF_DISCOVERED"/> <category android:name="android.intent.category.DEFAULT"/> <data android:mimeType="text/plain" /> </intent-filter>
En el siguiente ejemplo, se filtra un URI en el formato de https://developer.android.com/index.html
.
<intent-filter> <action android:name="android.nfc.action.NDEF_DISCOVERED"/> <category android:name="android.intent.category.DEFAULT"/> <data android:scheme="https" android:host="developer.android.com" android:pathPrefix="/index.html" /> </intent-filter>
ACTION_TECH_DISCOVERED
Si tu actividad filtra el intent ACTION_TECH_DISCOVERED
, debes crear un archivo de recursos XML que especifique las tecnologías que admite tu actividad dentro de un conjunto tech-list
. Tu actividad se considera una coincidencia si un conjunto tech-list
es un subconjunto de las tecnologías admitidas por la etiqueta, que puedes obtener llamando getTechList()
.
Por ejemplo, si la etiqueta que se analiza admite MifareClassic, NdefFormatable y NfcA, tu conjunto de tech-list
debe especificar las tres, dos o una de las tecnologías (y nada más) para que tu actividad coincida.
Los siguientes ejemplos definen todas las tecnologías. Debes quitar las que no sean compatibles con tu etiqueta NFC. Guarda este archivo (puedes asignarle el nombre que desees) en la carpeta <project-root>/res/xml
.
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2"> <tech-list> <tech>android.nfc.tech.IsoDep</tech> <tech>android.nfc.tech.NfcA</tech> <tech>android.nfc.tech.NfcB</tech> <tech>android.nfc.tech.NfcF</tech> <tech>android.nfc.tech.NfcV</tech> <tech>android.nfc.tech.Ndef</tech> <tech>android.nfc.tech.NdefFormatable</tech> <tech>android.nfc.tech.MifareClassic</tech> <tech>android.nfc.tech.MifareUltralight</tech> </tech-list> </resources>
También puedes especificar múltiples conjuntos tech-list
. Cada uno de los conjuntos tech-list
se considera de forma independiente, y tu actividad se considera una coincidencia si cualquier conjunto tech-list
es un subconjunto de las tecnologías que muestra getTechList()
. Esto proporciona una semántica AND
y OR
para las tecnologías de concordancia. El siguiente ejemplo coincide con las etiquetas que son compatibles con las tecnologías NfcA y Ndef, o que pueden ser compatibles con las tecnologías NfcB y Ndef:
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2"> <tech-list> <tech>android.nfc.tech.NfcA</tech> <tech>android.nfc.tech.Ndef</tech> </tech-list> <tech-list> <tech>android.nfc.tech.NfcB</tech> <tech>android.nfc.tech.Ndef</tech> </tech-list> </resources>
En tu archivo AndroidManifest.xml
, especifica el archivo de recursos que acabas de crear en el elemento <meta-data>
dentro del elemento <activity>
, como en el siguiente ejemplo:
<activity> ... <intent-filter> <action android:name="android.nfc.action.TECH_DISCOVERED"/> </intent-filter> <meta-data android:name="android.nfc.action.TECH_DISCOVERED" android:resource="@xml/nfc_tech_filter" /> ... </activity>
Para obtener más información sobre cómo trabajar con tecnologías de etiquetas y el intent ACTION_TECH_DISCOVERED
, consulta Cómo trabajar con tecnologías de etiquetas compatibles en el documento de conceptos avanzados de NFC.
ACTION_TAG_DISCOVERED
Para filtrar por ACTION_TAG_DISCOVERED
, usa el siguiente filtro de intent:
<intent-filter> <action android:name="android.nfc.action.TAG_DISCOVERED"/> </intent-filter>
Cómo obtener información de intents
Si se inicia una actividad debido a un intent NFC, puedes usar el intent para obtener información sobre la etiqueta NFC escaneada. Los intents pueden contener los siguientes servicios adicionales según la etiqueta escaneada:
EXTRA_TAG
(obligatorio): Es un objetoTag
que representa la etiqueta analizada.EXTRA_NDEF_MESSAGES
(opcional): Es un array de mensajes NDEF analizados desde la etiqueta. Este adicional es obligatorio para intentsACTION_NDEF_DISCOVERED
.EXTRA_ID
(opcional): Es el ID de bajo nivel de la etiqueta.
Para obtener estos elementos adicionales, comprueba si tu actividad se inició con uno de los intents de NFC para asegurarte de que se escaneó una etiqueta y, luego, obtén los elementos adicionales del intent. En el siguiente ejemplo, se verifica el intent ACTION_NDEF_DISCOVERED
y se obtienen los mensajes NDEF de un intent adicional.
Kotlin
override fun onNewIntent(intent: Intent) { super.onNewIntent(intent) ... if (NfcAdapter.ACTION_NDEF_DISCOVERED == intent.action) { intent.getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES)?.also { rawMessages -> val messages: List<NdefMessage> = rawMessages.map { it as NdefMessage } // Process the messages array. ... } } }
Java
@Override protected void onNewIntent(Intent intent) { super.onNewIntent(intent); ... if (NfcAdapter.ACTION_NDEF_DISCOVERED.equals(intent.getAction())) { Parcelable[] rawMessages = intent.getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES); if (rawMessages != null) { NdefMessage[] messages = new NdefMessage[rawMessages.length]; for (int i = 0; i < rawMessages.length; i++) { messages[i] = (NdefMessage) rawMessages[i]; } // Process the messages array. ... } } }
Como alternativa, puedes obtener un objeto Tag
a partir del intent, que contendrá la carga útil y te permitirá enumerar las tecnologías de la etiqueta:
Kotlin
val tag: Tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG)
Java
Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
Cómo crear tipos comunes de registros NDEF
En esta sección, se describe cómo crear tipos comunes de registros NDEF para ayudarte a escribir en etiquetas NFC. A partir de Android 4.0 (nivel de API 14), se lanzó el método createUri()
para ayudarte a crear registros de URI automáticamente. A partir de Android 4.1 (nivel de API 16), createExternal()
y createMime()
están disponibles para ayudarte a crear registros MIME y de tipo externo NDEF. Usa estos métodos auxiliares siempre que sea posible para evitar errores cuando crees registros NDEF de forma manual.
Esta sección también describe cómo crear el filtro de intents correspondiente para el registro. Todos estos ejemplos de registros NDEF deben estar en el primer registro NDEF del mensaje NDEF que estás escribiendo en una etiqueta.
TNF_ABSOLUTE_URI
Nota: Te recomendamos que utilices el tipo RTD_URI
en lugar de TNF_ABSOLUTE_URI
, ya que es más eficiente.
Puedes crear un registro NDEF TNF_ABSOLUTE_URI
de la siguiente manera:
Kotlin
val uriRecord = ByteArray(0).let { emptyByteArray -> NdefRecord( TNF_ABSOLUTE_URI, "https://developer.android.com/index.html".toByteArray(Charset.forName("US-ASCII")), emptyByteArray, emptyByteArray ) }
Java
NdefRecord uriRecord = new NdefRecord( NdefRecord.TNF_ABSOLUTE_URI , "https://developer.android.com/index.html".getBytes(Charset.forName("US-ASCII")), new byte[0], new byte[0]);
El filtro de intents para el registro NDEF anterior se vería así:
<intent-filter> <action android:name="android.nfc.action.NDEF_DISCOVERED" /> <category android:name="android.intent.category.DEFAULT" /> <data android:scheme="https" android:host="developer.android.com" android:pathPrefix="/index.html" /> </intent-filter>
TNF_MIME_MEDIA
Puedes crear un registro NDEF TNF_MIME_MEDIA
de las siguientes maneras:
Con el método createMime()
:
Kotlin
val mimeRecord = NdefRecord.createMime( "application/vnd.com.example.android.beam", "Beam me up, Android".toByteArray(Charset.forName("US-ASCII")) )
Java
NdefRecord mimeRecord = NdefRecord.createMime("application/vnd.com.example.android.beam", "Beam me up, Android".getBytes(Charset.forName("US-ASCII")));
Creando el NdefRecord
de forma manual:
Kotlin
val mimeRecord = Charset.forName("US-ASCII").let { usAscii -> NdefRecord( NdefRecord.TNF_MIME_MEDIA, "application/vnd.com.example.android.beam".toByteArray(usAscii), ByteArray(0), "Beam me up, Android!".toByteArray(usAscii) ) }
Java
NdefRecord mimeRecord = new NdefRecord( NdefRecord.TNF_MIME_MEDIA , "application/vnd.com.example.android.beam".getBytes(Charset.forName("US-ASCII")), new byte[0], "Beam me up, Android!".getBytes(Charset.forName("US-ASCII")));
El filtro de intents para el registro NDEF anterior se vería así:
<intent-filter> <action android:name="android.nfc.action.NDEF_DISCOVERED" /> <category android:name="android.intent.category.DEFAULT" /> <data android:mimeType="application/vnd.com.example.android.beam" /> </intent-filter>
TNF_WELL_KNOWN con RTD_TEXT
Puedes crear un registro NDEF TNF_WELL_KNOWN
de la siguiente manera:
Kotlin
fun createTextRecord(payload: String, locale: Locale, encodeInUtf8: Boolean): NdefRecord { val langBytes = locale.language.toByteArray(Charset.forName("US-ASCII")) val utfEncoding = if (encodeInUtf8) Charset.forName("UTF-8") else Charset.forName("UTF-16") val textBytes = payload.toByteArray(utfEncoding) val utfBit: Int = if (encodeInUtf8) 0 else 1 shl 7 val status = (utfBit + langBytes.size).toChar() val data = ByteArray(1 + langBytes.size + textBytes.size) data[0] = status.toByte() System.arraycopy(langBytes, 0, data, 1, langBytes.size) System.arraycopy(textBytes, 0, data, 1 + langBytes.size, textBytes.size) return NdefRecord(NdefRecord.TNF_WELL_KNOWN, NdefRecord.RTD_TEXT, ByteArray(0), data) }
Java
public NdefRecord createTextRecord(String payload, Locale locale, boolean encodeInUtf8) { byte[] langBytes = locale.getLanguage().getBytes(Charset.forName("US-ASCII")); Charset utfEncoding = encodeInUtf8 ? Charset.forName("UTF-8") : Charset.forName("UTF-16"); byte[] textBytes = payload.getBytes(utfEncoding); int utfBit = encodeInUtf8 ? 0 : (1 << 7); char status = (char) (utfBit + langBytes.length); byte[] data = new byte[1 + langBytes.length + textBytes.length]; data[0] = (byte) status; System.arraycopy(langBytes, 0, data, 1, langBytes.length); System.arraycopy(textBytes, 0, data, 1 + langBytes.length, textBytes.length); NdefRecord record = new NdefRecord(NdefRecord.TNF_WELL_KNOWN, NdefRecord.RTD_TEXT, new byte[0], data); return record; }
El filtro de intents para el registro NDEF anterior se vería así:
<intent-filter> <action android:name="android.nfc.action.NDEF_DISCOVERED" /> <category android:name="android.intent.category.DEFAULT" /> <data android:mimeType="text/plain" /> </intent-filter>
TNF_WELL_KNOWN con RTD_URI
Puedes crear un registro NDEF TNF_WELL_KNOWN
de las siguientes maneras:
Con el método createUri(String)
:
Kotlin
val rtdUriRecord1 = NdefRecord.createUri("https://example.com")
Java
NdefRecord rtdUriRecord1 = NdefRecord.createUri("https://example.com");
Con el método createUri(Uri)
:
Kotlin
val rtdUriRecord2 = Uri.parse("https://example.com").let { uri -> NdefRecord.createUri(uri) }
Java
Uri uri = Uri.parse("https://example.com"); NdefRecord rtdUriRecord2 = NdefRecord.createUri(uri);
Creando el NdefRecord
de forma manual:
Kotlin
val uriField = "example.com".toByteArray(Charset.forName("US-ASCII")) val payload = ByteArray(uriField.size + 1) //add 1 for the URI Prefix payload [0] = 0x01 //prefixes https://www. to the URI System.arraycopy(uriField, 0, payload, 1, uriField.size) //appends URI to payload val rtdUriRecord = NdefRecord(NdefRecord.TNF_WELL_KNOWN, NdefRecord.RTD_URI, ByteArray(0), payload)
Java
byte[] uriField = "example.com".getBytes(Charset.forName("US-ASCII")); byte[] payload = new byte[uriField.length + 1]; //add 1 for the URI Prefix payload[0] = 0x01; //prefixes https://www. to the URI System.arraycopy(uriField, 0, payload, 1, uriField.length); //appends URI to payload NdefRecord rtdUriRecord = new NdefRecord( NdefRecord.TNF_WELL_KNOWN, NdefRecord.RTD_URI, new byte[0], payload);
El filtro de intents para el registro NDEF anterior se vería así:
<intent-filter> <action android:name="android.nfc.action.NDEF_DISCOVERED" /> <category android:name="android.intent.category.DEFAULT" /> <data android:scheme="https" android:host="example.com" android:pathPrefix="" /> </intent-filter>
TNF_EXTERNAL_TYPE
Puedes crear un registro NDEF TNF_EXTERNAL_TYPE
de las siguientes maneras:
Con el método createExternal()
:
Kotlin
var payload: ByteArray //assign to your data val domain = "com.example" //usually your app's package name val type = "externalType" val extRecord = NdefRecord.createExternal(domain, type, payload)
Java
byte[] payload; //assign to your data String domain = "com.example"; //usually your app's package name String type = "externalType"; NdefRecord extRecord = NdefRecord.createExternal(domain, type, payload);
Creando el NdefRecord
de forma manual:
Kotlin
var payload: ByteArray ... val extRecord = NdefRecord( NdefRecord.TNF_EXTERNAL_TYPE, "com.example:externalType".toByteArray(Charset.forName("US-ASCII")), ByteArray(0), payload )
Java
byte[] payload; ... NdefRecord extRecord = new NdefRecord( NdefRecord.TNF_EXTERNAL_TYPE, "com.example:externalType".getBytes(Charset.forName("US-ASCII")), new byte[0], payload);
El filtro de intents para el registro NDEF anterior se vería así:
<intent-filter> <action android:name="android.nfc.action.NDEF_DISCOVERED" /> <category android:name="android.intent.category.DEFAULT" /> <data android:scheme="vnd.android.nfc" android:host="ext" android:pathPrefix="/com.example:externalType"/> </intent-filter>
Usa TNF_EXTERNAL_TYPE
para implementaciones de etiquetas NFC más genéricas a fin de admitir mejor los dispositivos con tecnología de Android y sin ella.
Nota: Los URN para TNF_EXTERNAL_TYPE
tienen un formato canónico de urn:nfc:ext:example.com:externalType
. Sin embargo, la especificación RTF de NFC Forum declara que debe omitirse del registro NDEF la porción urn:nfc:ext:
del URN. Entonces, todo lo que necesitas proporcionar es el dominio (example.com
en el ejemplo) y el tipo (externalType
en el ejemplo) separados por dos puntos.
Cuando se envía TNF_EXTERNAL_TYPE
, Android convierte el URN urn:nfc:ext:example.com:externalType
en un URI vnd.android.nfc://ext/example.com:externalType
, que es lo que declara el filtro de intents en el ejemplo.
Registros de aplicaciones de Android
Un registro de aplicación para Android (AAR), presentado en Android 4.0 (nivel de API 14), proporciona una certeza más sólida de que tu aplicación se inicia cuando se escanea una etiqueta NFC. Un AAR tiene el nombre del paquete de una aplicación incorporada dentro de un registro NDEF. Puedes agregar un AAR a cualquier registro NDEF de tu mensaje NDEF, porque Android busca los AAR en todo el mensaje NDEF. Si encuentra un AAR, inicia la aplicación según el nombre del paquete dentro del AAR. Si la aplicación no está presente en el dispositivo, se inicia Google Play para descargarla.
Los AAR son útiles si deseas evitar que se filtren otras aplicaciones con el mismo intent y que, posiblemente, manejen etiquetas específicas que hayas implementado. Los AAR solo se admiten a nivel de la aplicación, debido a la restricción del nombre del paquete, y no a nivel de la Actividad, como ocurre con el filtro de intents. Si deseas manejar un intent al nivel de la Actividad, usa filtros de intents.
Si una etiqueta contiene un AAR, se despacha el sistema de envío de etiquetas de la siguiente manera:
- Intenta iniciar una Actividad usando un filtro de intents como de costumbre. Si la actividad que coincide con el intent también coincide con el AAR, iníciala.
- Si la actividad que filtra el intent no coincide con las AAR, si varias actividades pueden manejar el intent o si ninguna actividad lo hace, inicia la aplicación especificada por el AAR.
- Si no se inicia ninguna aplicación con el AAR, ve a Google Play para descargar la aplicación basada en el AAR.
Nota: Puedes anular los AAR y el sistema de envío de intents con el sistema de envío en primer plano, que permite que una actividad en primer plano tenga prioridad cuando se descubre una etiqueta NFC. Con este método, la actividad debe estar en primer plano para anular los AAR y el sistema de envío de intents.
Si aún deseas filtrar las etiquetas escaneadas que no contienen un AAR, puedes declarar los filtros de intents como normales. Esto es útil si tu aplicación está interesada en otras etiquetas que no contienen un AAR. Por ejemplo, tal vez desees garantizar que tu aplicación controle las etiquetas de propiedad que implementas así como también las etiquetas generales implementadas por terceros. Ten en cuenta que los AAR son específicos para dispositivos Android 4.0 o versiones posteriores, por lo que, cuando implementas etiquetas, lo más recomendable es utilizar una combinación de AAR y tipos/URI MIME para lograr ser compatible con la mayor cantidad de dispositivos posibles. Además, cuando implementes etiquetas NFC, piensa en cómo deseas escribirlas para habilitar la compatibilidad con la mayoría de los dispositivos (con Android y otros dispositivos). Para ello, define un URI o tipo de MIME relativamente único que facilite la distinción entre las aplicaciones.
Android proporciona una API simple para crear un AAR: createApplicationRecord()
. Todo lo que necesitas hacer es incluir el AAR en cualquier parte de tu NdefMessage
. No es aconsejable utilizar el primer registro de tu NdefMessage
, a menos que el AAR sea el único registro en el NdefMessage
. Esto se debe a que el sistema Android verifica el primer registro de un NdefMessage
para determinar el tipo MIME o URI de la etiqueta, que se utiliza para crear un intent para que las aplicaciones filtren. En el siguiente código, se muestra cómo crear un AAR:
Kotlin
val msg = NdefMessage( arrayOf( ..., NdefRecord.createApplicationRecord("com.example.android.beam") ) )
Java
NdefMessage msg = new NdefMessage( new NdefRecord[] { ..., NdefRecord.createApplicationRecord("com.example.android.beam")} ); )