アプリをウェアラブルに拡張して、健康とフィットネスの機能を強化できます Wear OS 搭載のデバイス
Wear OS モジュールを追加する
Android Studio には、Wear OS モジュールをアプリに追加するための便利なウィザードが用意されています。イン [ファイル >[New Module] メニューで、下に示すように [Wear OS] を選択します。 image:
<ph type="x-smartling-placeholder">[Minimum SDK] は API 30 以上にする必要があります。 最新バージョンのヘルスサービスを使用できるようにします。ヘルスサービス 正常性を設定することで、指標の追跡とデータの記録が簡単になります。 センサーを自動的に切り替えます。
ウィザードが完了したら、プロジェクトを同期します。その後の実行 表示されます。
<ph type="x-smartling-placeholder">これにより、ウェアラブル デバイスで Wear OS モジュールを実行できます。次の 2 通りの方法があります。
構成を実行すると、アプリが Wear OS エミュレータにデプロイされるか、 「Hello World」と表示されますが、体験できますこれは基本的な UI 設定です。 Wear OS 向け Compose を使用してアプリの作成を開始しましょう。
ヘルスサービスと Hilt を追加する
次のライブラリを Wear OS モジュールに統合します。
ヘルスサービス マネージャーを作成する
ヘルスサービスを少し使いやすくし、より小規模なサービスを よりスムーズな API を実現するには、次のようにラッパーを作成します。
private const val TAG = "WATCHMAIN"
class HealthServicesManager(context: Context) {
private val measureClient = HealthServices.getClient(context).measureClient
suspend fun hasHeartRateCapability() = runCatching {
val capabilities = measureClient.getCapabilities()
(DataType.HEART_RATE_BPM in capabilities.supportedDataTypesMeasure)
}.getOrDefault(false)
/**
* Returns a cold flow. When activated, the flow will register a callback for heart rate data
* and start to emit messages. When the consuming coroutine is canceled, the measure callback
* is unregistered.
*
* [callbackFlow] creates a bridge between a callback-based API and Kotlin flows.
*/
@ExperimentalCoroutinesApi
fun heartRateMeasureFlow(): Flow<MeasureMessage> = callbackFlow {
val callback = object : MeasureCallback {
override fun onAvailabilityChanged(dataType: DeltaDataType<*, *>, availability: Availability) {
// Only send back DataTypeAvailability (not LocationAvailability)
if (availability is DataTypeAvailability) {
trySendBlocking(MeasureMessage.MeasureAvailability(availability))
}
}
override fun onDataReceived(data: DataPointContainer) {
val heartRateBpm = data.getData(DataType.HEART_RATE_BPM)
Log.d(TAG, "💓 Received heart rate: ${heartRateBpm.first().value}")
trySendBlocking(MeasureMessage.MeasureData(heartRateBpm))
}
}
Log.d(TAG, "⌛ Registering for data...")
measureClient.registerMeasureCallback(DataType.HEART_RATE_BPM, callback)
awaitClose {
Log.d(TAG, "👋 Unregistering for data")
runBlocking {
measureClient.unregisterMeasureCallback(DataType.HEART_RATE_BPM, callback)
}
}
}
}
sealed class MeasureMessage {
class MeasureAvailability(val availability: DataTypeAvailability) : MeasureMessage()
class MeasureData(val data: List<SampleDataPoint<Double>>) : MeasureMessage()
}
管理する Hilt モジュールを作成したら、次のスニペットを使用します。
@Module
@InstallIn(SingletonComponent::class)
internal object DataModule {
@Provides
@Singleton
fun provideHealthServices(@ApplicationContext context: Context): HealthServicesManager = HealthServicesManager(context)
}
HealthServicesManager
を他の Hilt 依存関係として注入できます。
新しい HealthServicesManager
には、次の処理を行う heartRateMeasureFlow()
メソッドが用意されています。
心臓モニターのリスナーを登録し、受信したデータを出力します。
ウェアラブル デバイスでデータの更新を有効にする
フィットネス関連のデータを更新するには、BODY_SENSORS
権限が必要です。もし
まだ行っていない場合は、BODY_SENSORS
権限を
アプリのマニフェスト ファイルに追加します。次に、以下のスニペットに示すように、権限をリクエストします。
val permissionState = rememberPermissionState(
permission = Manifest.permission.BODY_SENSORS,
onPermissionResult = { granted -> /* do something */ }
)
[...]
if (permissionState.status.isGranted) {
// do something
} else {
permissionState.launchPermissionRequest()
}
実機でアプリをテストすると、データの更新が開始されます。
Wear OS 4 以降のエミュレータでは、テストデータも自動的に表示されます。以前 センサーからのデータ ストリームをシミュレートできます。ターミナル 次の ADB コマンドを実行します。
adb shell am broadcast \
-a "whs.USE_SYNTHETIC_PROVIDERS" \
com.google.android.wearable.healthservices
異なる心拍数値を確認するには、さまざまなエクササイズをシミュレートしてみてください。 このコマンドは歩行をシミュレートします。
adb shell am broadcast \
-a "whs.synthetic.user.START_WALKING" \
com.google.android.wearable.healthservices
このコマンドは、以下をシミュレートします。
adb shell am broadcast \
-a "whs.synthetic.user.START_RUNNING" \
com.google.android.wearable.healthservices
データのシミュレーションを停止するには、次のコマンドを実行します。
adb shell am broadcast -a \
"whs.USE_SENSOR_PROVIDERS" \
com.google.android.wearable.healthservices
心拍数データの読み取り
BODY_SENSORS
権限が付与されると、ユーザーの心拍数を読み取ることができます。
(heartRateMeasureFlow()
)(HealthServicesManager
)。Wear OS アプリの
画面上にセンサーによって測定されている現在の心拍数値が表示され、
ウェアラブル デバイスが必要です。
ViewModel
で、心拍数フロー オブジェクトを使用してデータの収集を開始します。
次のスニペットのようになります。
val hr: MutableState<Double> = mutableStateOf(0.0)
[...]
healthServicesManager
.heartRateMeasureFlow()
.takeWhile { enabled.value }
.collect { measureMessage ->
when (measureMessage) {
is MeasureData -> {
val latestHeartRateValue = measureMessage.data.last().value
hr.value = latestHeartRateValue
}
is MeasureAvailability -> availability.value =
measureMessage.availability
}
}
次のようなコンポーズ可能なオブジェクトを使用して、ライブデータを 作成してみましょう。
val heartRate by viewModel.hr
Text(
text = "Heart Rate: $heartRate",
style = MaterialTheme.typography.display1
)
ハンドヘルド デバイスにデータを送信する
健康とフィットネスのデータをハンドヘルド デバイスに送信するには、DataClient
を使用します
クラスです。次のコード スニペットは、ハートを送信する方法を示しています。
アプリが以前に収集したレートデータ:
class HealthServicesManager(context: Context) {
private val dataClient by lazy { Wearable.getDataClient(context) }
[...]
suspend fun sendToHandheldDevice(heartRate: Int) {
try {
val result = dataClient
.putDataItem(PutDataMapRequest
.create("/heartrate")
.apply { dataMap.putInt("heartrate", heartRate) }
.asPutDataRequest()
.setUrgent())
.await()
Log.d(TAG, "DataItem saved: $result")
} catch (cancellationException: CancellationException) {
throw cancellationException
} catch (exception: Exception) {
Log.d(TAG, "Saving DataItem failed: $exception")
}
}
}
スマートフォンでデータを受信する
スマートフォンでデータを受信するには、
WearableListenerService
:
@AndroidEntryPoint
class DataLayerListenerService : WearableListenerService() {
@Inject
lateinit var heartRateMonitor: HeartRateMonitor
override fun onDataChanged(dataEvents: DataEventBuffer) {
dataEvents.forEach { event ->
when (event.type) {
DataEvent.TYPE_CHANGED -> {
event.dataItem.run {
if (uri.path?.compareTo("/heartrate") == 0) {
val heartRate = DataMapItem.fromDataItem(this)
.dataMap.getInt(HR_KEY)
Log.d("DataLayerListenerService",
"New heart rate value received: $heartRate")
heartRateMonitor.send(heartRate)
}
}
}
DataEvent.TYPE_DELETED -> {
// DataItem deleted
}
}
}
}
}
このステップが完了すると、いくつかの興味深い詳細が表示されます。
@AndroidEntryPoint
アノテーションにより、このクラスで Hilt を使用できます。@Inject lateinit var heartRateMonitor: HeartRateMonitor
は実際に このクラスに依存関係を注入する- このクラスは
onDataChanged()
を実装し、イベントのコレクションを受け取ります。 解析して使用できる
次の HeartRateMonitor
ロジックを使用すると、受信した心拍数を送信できます。
値をアプリのコードベースの別の部分にマッピングできます。
class HeartRateMonitor {
private val datapoints = MutableSharedFlow<Int>(extraBufferCapacity = 10)
fun receive(): SharedFlow<Int> = datapoints.asSharedFlow()
fun send(hr: Int) {
datapoints.tryEmit(hr)
}
}
データバスは onDataChanged()
メソッドからイベントを受け取り、
データ オブザーバーが SharedFlow
を使って使用できるようにします。
最後の部分は、スマートフォン アプリでの Service
の宣言です。
AndroidManifest.xml
:
<service
android:name=".DataLayerListenerService"
android:exported="true">
<intent-filter>
<!-- listeners receive events that match the action and data filters -->
<action android:name="com.google.android.gms.wearable.DATA_CHANGED" />
<data
android:host="*"
android:pathPrefix="/heartrate"
android:scheme="wear" />
</intent-filter>
</service>
ハンドヘルド デバイスでリアルタイム データを表示する
ハンドヘルド デバイスで動作するアプリの部分に、
HeartRateMonitor
をビューモデルのコンストラクタに追加します。このHeartRateMonitor
オブジェクトは心拍数データを監視し、必要に応じて UI の更新を出力します。