این سند نحوه همگام سازی داده ها را بین یک دستگاه Wear OS و یک دستگاه دستی شرح می دهد.
ارسال و همگام سازی داده ها به طور مستقیم از شبکه
برنامههای Wear OS را بسازید تا مستقیماً با شبکه ارتباط برقرار کنید . از همان API هایی استفاده کنید که برای توسعه موبایل استفاده می کنید، اما برخی از تفاوت های خاص Wear-OS را در نظر داشته باشید.
داده ها را با استفاده از Wear OS Data Layer API همگام سازی کنید
یک DataClient
یک API برای خواندن یا نوشتن اجزا در یک DataItem
یا Asset
نمایش می دهد.
تنظیم اقلام و دارایی های داده در حالی که به هیچ دستگاهی متصل نیستید امکان پذیر است. هنگامی که دستگاه ها اتصال شبکه برقرار می کنند، همگام می شوند. این داده ها برای برنامه شما خصوصی هستند و فقط برای برنامه شما در دستگاه های دیگر قابل دسترسی هستند.
یک
DataItem
در تمام دستگاههای یک شبکه Wear OS همگامسازی میشود. آنها معمولاً اندازه کوچکی دارند.از یک
Asset
برای انتقال یک شی بزرگتر مانند یک تصویر استفاده کنید. این سیستم پیگیری می کند که کدام دارایی قبلاً منتقل شده است و به طور خودکار حذف مجدد را انجام می دهد.
به رویدادهای خدمات گوش دهید
کلاس WearableListenerService
را گسترش دهید. این سیستم چرخه حیات پایگاه WearableListenerService
را مدیریت میکند، در صورت نیاز به ارسال اقلام داده یا پیام، به سرویس متصل میشود و زمانی که نیازی به کاری نیست، سرویس را لغو پیوند میکند.
به رویدادها در فعالیت ها گوش دهید
رابط OnDataChangedListener
را پیاده سازی کنید. زمانی که می خواهید تغییرات را تنها زمانی که کاربر به طور فعال از برنامه شما استفاده می کند گوش دهید، از این رابط به جای WearableListenerService
استفاده کنید.
انتقال داده ها
برای ارسال اشیاء بزرگ باینری از طریق انتقال بلوتوث، مانند ضبط صدا از دستگاه دیگر، میتوانید یک Asset
به یک مورد داده متصل کنید و سپس مورد داده را در ذخیرهگاه داده تکراری قرار دهید.
دارایی ها به طور خودکار ذخیره داده ها را برای جلوگیری از ارسال مجدد و حفظ پهنای باند بلوتوث مدیریت می کنند. یک الگوی رایج این است که یک برنامه دستی یک تصویر را بارگیری می کند، آن را به اندازه مناسب برای نمایش در پوشیدنی کوچک می کند و آن را به عنوان دارایی به برنامه پوشیدنی منتقل می کند. مثال های زیر این الگو را نشان می دهند.
توجه: اگرچه اندازه اقلام داده از نظر تئوری به 100 کیلوبایت محدود شده است، اما در عمل می توان از اقلام داده بزرگتر استفاده کرد. برای اقلام داده بزرگتر، داده ها را بر اساس مسیرهای منحصر به فرد جدا کنید و از استفاده از یک مسیر واحد برای همه داده ها خودداری کنید. انتقال داراییهای بزرگ در بسیاری از موارد بر تجربه کاربر تأثیر میگذارد، بنابراین برنامههای خود را آزمایش کنید تا مطمئن شوید که هنگام انتقال داراییهای بزرگ عملکرد خوبی دارند.
انتقال یک دارایی
دارایی را با استفاده از یکی از متدهای create...()
در کلاس Asset
ایجاد کنید. همانطور که در نمونه زیر نشان داده شده است، یک بیت مپ را به یک جریان بایت تبدیل کنید و سپس createFromBytes()
برای ایجاد دارایی فراخوانی کنید.
private fun createAssetFromBitmap(bitmap: Bitmap): Asset =
ByteArrayOutputStream().let { byteStream ->
bitmap.compress(Bitmap.CompressFormat.PNG, 100, byteStream)
Asset.createFromBytes(byteStream.toByteArray())
}
private static Asset createAssetFromBitmap(Bitmap bitmap) {
final ByteArrayOutputStream byteStream = new ByteArrayOutputStream();
bitmap.compress(Bitmap.CompressFormat.PNG, 100, byteStream);
return Asset.createFromBytes(byteStream.toByteArray());
}
سپس، دارایی را با متد putAsset()
در DataMap
یا PutDataRequest
به یک آیتم داده متصل کنید. سپس با استفاده از روش putDataItem()
مانند نمونه های زیر، آیتم داده را در datastore قرار دهید.
نمونه زیر از PutDataRequest
استفاده می کند:
val asset: Asset = BitmapFactory.decodeResource(resources, R.drawable.image).let { bitmap ->
createAssetFromBitmap(bitmap)
}
val request: PutDataRequest = PutDataRequest.create("/image").apply {
putAsset("profileImage", asset)
}
val putTask: Task<DataItem> = Wearable.getDataClient(context).putDataItem(request)
Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.image);
Asset asset = createAssetFromBitmap(bitmap);
PutDataRequest request = PutDataRequest.create("/image");
request.putAsset("profileImage", asset);
Task<DataItem> putTask = Wearable.getDataClient(context).putDataItem(request);
نمونه زیر از PutDataMapRequest
استفاده می کند:
val asset: Asset = BitmapFactory.decodeResource(resources, R.drawable.image).let { bitmap ->
createAssetFromBitmap(bitmap)
}
val request: PutDataRequest = PutDataMapRequest.create("/image").run {
dataMap.putAsset("profileImage", asset)
asPutDataRequest()
}
val putTask: Task<DataItem> = Wearable.getDataClient(context).putDataItem(request)
Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.image);
Asset asset = createAssetFromBitmap(bitmap);
PutDataMapRequest dataMap = PutDataMapRequest.create("/image");
dataMap.getDataMap().putAsset("profileImage", asset);
PutDataRequest request = dataMap.asPutDataRequest();
Task<DataItem> putTask = Wearable.getDataClient(context).putDataItem(request);
دریافت دارایی
هنگامی که یک دارایی ایجاد می شود، احتمالاً می خواهید آن را در طرف دیگر اتصال بخوانید و استخراج کنید. در اینجا مثالی از نحوه اجرای callback برای تشخیص تغییر دارایی و استخراج دارایی آورده شده است:
override fun onDataChanged(dataEvents: DataEventBuffer) {
dataEvents
.filter { it.type == DataEvent.TYPE_CHANGED && it.dataItem.uri.path == "/image" }
.forEach { event ->
val bitmap: Bitmap? = DataMapItem.fromDataItem(event.dataItem)
.dataMap.getAsset("profileImage")
.let { asset -> loadBitmapFromAsset(asset) }
// Do something with the bitmap
}
}
fun loadBitmapFromAsset(asset: Asset): Bitmap? {
// Convert asset into a file descriptor and block until it's ready
val assetInputStream: InputStream? =
Tasks.await(Wearable.getDataClient(context).getFdForAsset(asset))
?.inputStream
return assetInputStream?.let { inputStream ->
// Decode the stream into a bitmap
BitmapFactory.decodeStream(inputStream)
} ?: run {
Log.w(TAG, "Requested an unknown Asset.")
null
}
}
@Override
public void onDataChanged(DataEventBuffer dataEvents) {
for (DataEvent event : dataEvents) {
if (event.getType() == DataEvent.TYPE_CHANGED &&
event.getDataItem().getUri().getPath().equals("/image")) {
DataMapItem dataMapItem = DataMapItem.fromDataItem(event.getDataItem());
Asset profileAsset = dataMapItem.getDataMap().getAsset("profileImage");
Bitmap bitmap = loadBitmapFromAsset(profileAsset);
// Do something with the bitmap
}
}
}
public Bitmap loadBitmapFromAsset(Asset asset) {
if (asset == null) {
throw new IllegalArgumentException("Asset must be non-null");
}
// Convert asset into a file descriptor and block until it's ready
InputStream assetInputStream =
Tasks.await(Wearable.getDataClient(context).getFdForAsset(asset))
.getInputStream();
if (assetInputStream == null) {
Log.w(TAG, "Requested an unknown Asset.");
return null;
}
// Decode the stream into a bitmap
return BitmapFactory.decodeStream(assetInputStream);
}
برای اطلاعات بیشتر، پروژه نمونه DataLayer در GitHub را ببینید.