Cuando haces una llamada a la API de Data Layer, puedes recibir el estado de la llamada cuando se completa. También puedes escuchar eventos de datos resultantes de modificaciones de datos que haga tu app en cualquier punto de la red de Wear OS by Google.
Para ver un ejemplo de cómo trabajar de manera efectiva con la API de Data Layer, consulta la app de ejemplo de DataLayer de Android.
Espera el estado de las llamadas a Data Layer
Las llamadas a la API de Data Layer, por ejemplo, una llamada que usa el método putDataItem
de la clase
DataClient
, a veces muestran un objeto
Task<ResultType>
. Apenas se crea el objeto Task
, se coloca la operación en cola en segundo plano. Si no realizas más acciones, la operación se completa en silencio.
Sin embargo, normalmente querrás hacer algo con el resultado después de que finalice la operación. El objeto Task
te permite esperar el estado del resultado, ya sea de forma síncrona o asíncrona.
Llamadas asíncronas
Si tu código se ejecuta en el subproceso de IU principal, no realices llamadas de bloqueo a la API de Data Layer. Ejecuta las llamadas de manera asíncrona agregando un método de devolución de llamada al objeto Task
, que se activa cuando la operación finaliza:
Kotlin
// Using Kotlin function references task.addOnSuccessListener(::handleDataItem) task.addOnFailureListener(::handleDataItemError) task.addOnCompleteListener(::handleTaskComplete) ... fun handleDataItem(dataItem: DataItem) { ... } fun handleDataItemError(exception: Exception) { ... } fun handleTaskComplete(task: Task<DataItem>) { ... }
Java
// Using Java 8 Lambdas. task.addOnSuccessListener(dataItem -> handleDataItem(dataItem)); task.addOnFailureListener(exception -> handleDataItemError(exception)); task.addOnCompleteListener(task -> handleTaskComplete(task));
Consulta la API de Task para ver otras opciones, como la de encadenar diferentes tareas.
Llamadas síncronas
Si se ejecuta tu código en un subproceso de controlador separado, en un servicio en segundo plano (como sucede con
WearableListenerService
), está bien que se bloqueen las llamadas. En este caso, puedes llamar a Tasks.await()
en el objeto Task
, que se bloquea hasta que se completa la solicitud y se muestra un objeto Result
. Esto se muestra en el siguiente ejemplo.
Nota: Asegúrate de no llamar a esta función cuando estés en el subproceso principal.
Kotlin
try { Tasks.await(dataItemTask).apply { Log.d(TAG, "Data item set: $uri") } } catch (e: ExecutionException) { ... } catch (e: InterruptedException) { ... }
Java
try { DataItem item = Tasks.await(dataItemTask); Log.d(TAG, "Data item set: " + item.getUri()); } catch (ExecutionException | InterruptedException e) { ... }
Escucha eventos de Data Layer
Como Data Layer se sincroniza y envía datos a través del dispositivo portátil y del wearable, generalmente necesitarás escuchar eventos importantes, como la creación de elementos de datos y la recepción de mensajes.
Para escuchar los eventos de Data Layer, tienes dos opciones:
- Crea un servicio que extienda
WearableListenerService
. - Crea una actividad o clase que implemente la interfaz
DataClient.OnDataChangedListener
.
Con estas dos opciones, anulas los métodos de devolución de llamada de eventos de datos para los eventos que te interesa manejar.
Nota: Ten en cuenta el uso de batería de la app a la hora de elegir una implementación del objeto de escucha. Un WearableListenerService
se registra en el manifiesto de la app y puede iniciarla si aún no se está ejecutando. Si solo necesitas escuchar eventos cuando la app ya se está ejecutando, lo que suele suceder con las aplicaciones interactivas, no uses un WearableListenerService
. En su lugar, registra un objeto de escucha activo.
Por ejemplo, usa el método addListener
de la clase DataClient
. De esta forma, se puede reducir la carga en el sistema y el uso de batería.
Usa un WearableListenerService
Por lo general, creas instancias de WearableListenerService
en tus apps para wearables y dispositivos portátiles. Sin embargo, si no te interesan los eventos de datos en una de estas apps, no es necesario que implementes el servicio en esa app.
Por ejemplo, puedes tener una app para dispositivos portátiles que establezca y obtenga objetos de elementos de datos, y una app para wearables que escuche estas actualizaciones para actualizar su IU. La app para wearables nunca actualiza los elementos de datos, por lo que la app para dispositivos portátiles no escuchará ningún evento de datos de la app para wearables.
Estos son algunos de los eventos que puedes escuchar mediante WearableListenerService
:
onDataChanged()
: Cada vez que se crea, se borra o se cambia un objeto de elemento de datos, el sistema activa esta devolución de llamada en todos los nodos conectados.onMessageReceived()
: Un mensaje enviado desde un nodo activa esta devolución de llamada en el nodo de destino.onCapabilityChanged()
: Cuando una función que anuncia una instancia de tu app está disponible en la red, ese evento activa esta devolución de llamada. Si buscas un nodo cercano, puedes enviar una consulta al métodoisNearby()
de los nodos que se proporcionan en la devolución de llamada.
También puedes escuchar eventos de
ChannelClient.ChannelCallback
, como onChannelOpened()
.
Todos los eventos anteriores se ejecutan en un subproceso en segundo plano, no en el subproceso principal.
Para crear un WearableListenerService
, sigue estos pasos:
- Crea una clase que extienda
WearableListenerService
. - Escucha los eventos que te interesen, como
onDataChanged()
. - Declara un filtro de intents en tu manifiesto de Android para informar al sistema sobre tu
WearableListenerService
. Esta declaración le permite al sistema vincular tu servicio según sea necesario.
En el siguiente ejemplo, se muestra cómo implementar un WearableListenerService
simple:
Kotlin
private const val TAG = "DataLayerSample" private const val START_ACTIVITY_PATH = "/start-activity" private const val DATA_ITEM_RECEIVED_PATH = "/data-item-received" class DataLayerListenerService : WearableListenerService() { override fun onDataChanged(dataEvents: DataEventBuffer) { if (Log.isLoggable(TAG, Log.DEBUG)) { Log.d(TAG, "onDataChanged: $dataEvents") } // Loop through the events and send a message // to the node that created the data item. dataEvents.map { it.dataItem.uri } .forEach { uri -> // Get the node ID from the host value of the URI. val nodeId: String = uri.host // Set the data of the message to be the bytes of the URI. val payload: ByteArray = uri.toString().toByteArray() // Send the RPC. Wearable.getMessageClient(this) .sendMessage(nodeId, DATA_ITEM_RECEIVED_PATH, payload) } } }
Java
public class DataLayerListenerService extends WearableListenerService { private static final String TAG = "DataLayerSample"; private static final String START_ACTIVITY_PATH = "/start-activity"; private static final String DATA_ITEM_RECEIVED_PATH = "/data-item-received"; @Override public void onDataChanged(DataEventBuffer dataEvents) { if (Log.isLoggable(TAG, Log.DEBUG)) { Log.d(TAG, "onDataChanged: " + dataEvents); } // Loop through the events and send a message // to the node that created the data item. for (DataEvent event : dataEvents) { Uri uri = event.getDataItem().getUri(); // Get the node ID from the host value of the URI. String nodeId = uri.getHost(); // Set the data of the message to be the bytes of the URI. byte[] payload = uri.toString().getBytes(); // Send the RPC. Wearable.getMessageClient(this).sendMessage( nodeId, DATA_ITEM_RECEIVED_PATH, payload); } } }
En la siguiente sección, se explica cómo usar un filtro de intents con este objeto de escucha.
Usa filtros con WearableListenerService
Un filtro de intents para el ejemplo de WearableListenerService
que se mostró en la sección previa puede verse de esta manera:
<service android:name=".DataLayerListenerService" android:exported="true" tools:ignore="ExportedService" > <intent-filter> <action android:name="com.google.android.gms.wearable.DATA_CHANGED" /> <data android:scheme="wear" android:host="*" android:path="/start-activity" /> </intent-filter> </service>
En este filtro, la acción DATA_CHANGED
reemplaza la acción BIND_LISTENER
que se recomendó anteriormente para que solo ciertos eventos activen o inicien tu app. Este cambio mejora la eficiencia del sistema y reduce el consumo de batería y otras sobrecargas asociadas con tu app. En este ejemplo, el reloj escucha el elemento de datos /start-activity
y el teléfono escucha la respuesta del mensaje /data-item-received
.
Se aplican las reglas de coincidencia de filtro estándar de Android. Puedes especificar varios servicios por manifiesto, varios filtros de intents por servicio, varias acciones por filtro y varias estrofas de datos por filtro. Los filtros pueden coincidir en un host comodín o en un host determinado. Para hacer que coincidan en un host comodín, usa host="*"
. Para hacer que coincidan en un host específico, especifica host=<node_id>
.
También puedes hacer coincidir una ruta de acceso literal o un prefijo de ruta. Para ello, debes especificar un comodín o un host determinado. De lo contrario, el sistema ignorará la ruta que especifiques.
Para obtener más información sobre los tipos de filtros compatibles con Wear OS, consulta la documentación de referencia de la API para WearableListenerService
.
Si deseas obtener más información sobre los filtros de datos y las reglas de coincidencias, consulta la documentación de referencia de la API para el elemento de manifiesto <data>
.
Cuando hagas coincidir filtros de intents, recuerda dos reglas importantes:
- Si no se especifica ningún esquema para el filtro de intents, el sistema ignora todos los demás atributos del URI.
- Si no se especifica ningún host para el filtro, el sistema ignora todos los atributos de la ruta de acceso.
Usa un objeto de escucha activo
Si tu app se centra en los eventos de Data Layer cuando el usuario está interactuando con ella, es posible que no necesites un servicio prolongado para manejar cada cambio de datos. En tal caso, puedes escuchar eventos en una actividad si implementas una o más de las siguientes interfaces:
DataClient.OnDataChangedListener
MessageClient.OnMessageReceivedListener
CapabilityClient.OnCapabilityChangedListener
ChannelClient.ChannelCallback
Para crear una actividad que escucha eventos de datos, haz lo siguiente:
- Implementa las interfaces deseadas.
- En el método
onCreate()
oonResume()
, llama aWearable.getDataClient(this).addListener()
,MessageClient.addListener()
,CapabilityClient.addListener()
oChannelClient.registerChannelCallback()
para informar a los Servicios de Google Play que tu actividad desea escuchar eventos de Data Layer. - En
onStop()
oonPause()
, anula el registro de los objetos de escucha conDataClient.removeListener()
,MessageClient.removeListener()
,CapabilityClient.removeListener()
oChannelClient.unregisterChannelCallback()
. - Si una actividad solo se interesa en eventos con un prefijo específico de ruta de acceso, puedes agregar un objeto de escucha con un filtro de prefijo adecuado y así recibir solo los datos relevantes para el estado actual de la app.
- Implementa
onDataChanged()
,onMessageReceived()
,onCapabilityChanged()
o los métodos deChannelClient.ChannelCallback
, según las interfaces que implementaste. Se llama a estos métodos en el subproceso principal, pero puedes especificar unLooper
personalizado usandoWearableOptions
.
En el siguiente ejemplo, se implementa DataClient.OnDataChangedListener
.
Kotlin
class MainActivity : Activity(), DataClient.OnDataChangedListener { public override fun onResume() { Wearable.getDataClient(this).addListener(this) } override fun onPause() { Wearable.getDataClient(this).removeListener(this) } override fun onDataChanged(dataEvents: DataEventBuffer) { dataEvents.forEach { event -> if (event.type == DataEvent.TYPE_DELETED) { Log.d(TAG, "DataItem deleted: " + event.dataItem.uri) } else if (event.type == DataEvent.TYPE_CHANGED) { Log.d(TAG, "DataItem changed: " + event.dataItem.uri) } } } }
Java
public class MainActivity extends Activity implements DataClient.OnDataChangedListener { @Override public void onResume() { Wearable.getDataClient(this).addListener(this); } @Override protected void onPause() { Wearable.getDataClient(this).removeListener(this); } @Override public void onDataChanged(DataEventBuffer dataEvents) { for (DataEvent event : dataEvents) { if (event.getType() == DataEvent.TYPE_DELETED) { Log.d(TAG, "DataItem deleted: " + event.getDataItem().getUri()); } else if (event.getType() == DataEvent.TYPE_CHANGED) { Log.d(TAG, "DataItem changed: " + event.getDataItem().getUri()); } } } }
Usa filtros con objetos de escucha activos
Como se mencionó anteriormente, del mismo modo en que puedes especificar filtros de intents para objetos WearableListenerService
basados en manifiestos, también puedes usar filtros de intents cuando registras un objeto de escucha activo a través de la API de Wearable. Las mismas reglas se aplican a los objetos de escucha activos basados en API y a los basados en manifiestos.
Un patrón común es registrar un objeto de escucha con una ruta de acceso específica o un prefijo de ruta en el método onResume()
de una actividad y quitar el objeto de escucha en el método onPause()
de la actividad.
Implementar objetos de escucha de este modo permite que tu app reciba eventos de manera más selectiva, lo que mejora su diseño y eficiencia.