Sync data

This document describes how to synchronize data between a Wear OS device and a phone. See the overview guidance for when to use the Data Layer API and when to use your infrastructure.

Send and sync data directly from the network

Build Wear OS apps to communicate directly with the network. Use the same APIs that you use for mobile development, but keep some Wear-OS-specific differences in mind.

Synchronize data using the Wear OS Data Layer API

A DataClient exposes an API for components to read or write to a DataItem or Asset.

It's possible to set data items and assets while not connected to any devices. They're synchronized when the devices establish a network connection. This data is private to your app and is only accessible to your app on other devices.

  • A DataItem is synchronized across all devices in a Wear OS network. They're generally small in size.

  • Use an Asset to transfer a larger object, such as an image. The system keeps track of which assets have already been transferred and performs deduplication automatically.

Listen for events in services

Extend the WearableListenerService class. The system manages the lifecycle of the base WearableListenerService, binding to the service when it needs to send data items or messages and unbinding the service when no work is needed.

Listen for events in activities

Implement the OnDataChangedListener interface. Use this interface instead of a WearableListenerService when you want to listen for changes only when the user is actively using your app.

description: Transfer large binary objects, such as images, between Android phones and Wear OS watches using Assets in the Data Layer API. keywords_public: Wear OS, Data Layer API, Assets, Bluetooth data transfer, data synchronization, DataMap, PutDataRequest

Synchronize data

To share large binary objects over the Bluetooth transport, such as a voice recording from another device, you can attach an Asset to a data item and then put the data item into the replicated datastore. However, if the exchange is a one-off exchange between two connected devices, consider whether a simpler direct transfer is more appropriate.

Note: The Data Layer API can send messages and synchronize data only with phones that run Android or Wear OS watches. If a Wear OS device is paired with an iOS device, the Data Layer API won't work.

For this reason, don't use the Data Layer API as the primary way to communicate with a network. Instead, follow the same pattern in your Wear OS app as in a phone app—with some minor differences, as described in Network access and sync on Wear OS.

Assets automatically handle caching of data to prevent retransmission and to conserve Bluetooth bandwidth. A common pattern is for a phone app to download an image, shrink it to an appropriate size for display on the watch, and share it to the watch app as an asset. The following examples demonstrate this pattern.

Transfer an asset

Create the asset using one of the create...() methods in the Asset class. Convert a bitmap to a byte array and then call createFromBytes() to create the asset, as shown in the following sample.

private fun createAssetFromBitmap(bitmap: Bitmap): Asset =
    ByteArrayOutputStream().let { byteStream ->
        bitmap.compress(Bitmap.CompressFormat.PNG, 100, byteStream)
        Asset.createFromBytes(byteStream.toByteArray())
    }

Next, attach the asset to a data item with the putAsset() method in DataMap or PutDataRequest. Then put the data item into the datastore using the putDataItem() method, as shown in the following samples.

The following sample uses PutDataRequest:

private fun Context.sendImagePutDataRequest(): Task<DataItem> {

    val asset: Asset = createAssetFromBitmap(BitmapFactory.decodeResource(resources, R.drawable.ic_walk))
    val request: PutDataRequest = PutDataRequest.create("/image").apply {
        putAsset("profileImage", asset)
    }
    val putTask: Task<DataItem> = Wearable.getDataClient(this).putDataItem(request)

    return putTask
}

The following sample uses PutDataMapRequest:

private fun Context.sendImagePutDataMapRequest(): Task<DataItem> {

    val asset: Asset = createAssetFromBitmap(BitmapFactory.decodeResource(resources, R.drawable.ic_walk))
    val request: PutDataRequest = PutDataMapRequest.create("/image").run {
        dataMap.putAsset("profileImage", asset)
        asPutDataRequest()
    }
    val putTask: Task<DataItem> = Wearable.getDataClient(this).putDataItem(request)

    return putTask
}

Receive assets

After you create an asset, you typically read and extract it on the other side of the connection. The following example shows how to implement the callback to detect an asset change and extract the asset:

override fun onDataChanged(dataEvents: DataEventBuffer) {
    dataEvents
        .filter { it.type == DataEvent.TYPE_CHANGED && it.dataItem.uri.path == "/image" }
        .forEach { event ->
            val asset = DataMapItem.fromDataItem(event.dataItem)
                .dataMap.getAsset("profileImage")

            asset?.let { safeAsset ->
                lifecycleScope.launch {
                    val bitmap = loadBitmapFromAsset(safeAsset)
                    // Do something with the bitmap
                }
            }
        }
}

private suspend fun loadBitmapFromAsset(asset: Asset): Bitmap? = withContext(Dispatchers.IO) {
    try {
        val assetResult = Wearable.getDataClient(this@DataLayerActivity2)
            .getFdForAsset(asset)
            .await()

        assetResult?.inputStream?.use { inputStream ->
            BitmapFactory.decodeStream(inputStream)
        }
    } catch (e: Exception) {
        e.printStackTrace()
        null
    }
}

For more information, see the DataLayer sample project on GitHub.