@awarns/wear-os
v1.0.0
Published
Reliable and concurrent access to a paired smartwatch's sensors and enables communication between both devices
Downloads
10
Readme
@awarns/wear-os
This module allows to collect data from the sensors of an Android WearOS smartwatch, and enables the communication between both devices (i.e., smartphone and smartwatch).
This plugin acts as a wrapper on top of the nativescript-wearos-sensors plugin, which enables the communication and the data collection from a paired WearOS smartwatch. In order to use these features, a WearOS smartwatch must be paired with the smartphone and have installed a counterpart application, developed using the WearOSSensors WearOS library.
Note: check the requirements of both libraries for more detailed information:
Install the plugin using the following command line instruction:
ns plugin add @awarns/wear-os
Usage
Setup
This plugin requires you to register its loader during the framework initialization, like this:
// ... platform imports
import { awarns } from '@awarns/core';
import { demoTasks } from '../tasks';
import { demoTaskGraph } from '../graph';
import { registerWearOSPlugin } from '@awarns/wear-os';
awarns
.init(
demoTasks,
demoTaskGraph,
[
registerWearOSPlugin(config), // Parameter config is optional
]
)
// ... handle initialization promise
Plugin loader parameter (WearOSPluginConfig
):
| Parameter | Type | Description |
|-------------------------|-----------------|------------------------------------------------------------------------------------------------------------|
| sensors?
| WatchSensor[]
| Enable the specified sensors. By default, all sensors are enabled. |
| enablePlainMessaging?
| boolean
| Enable the plain messaging communication mechanism. Default: false
. |
| enableWearCommands?
| boolean
| Enable the command mechanism that allows to start the data collection on the watch side. Default: false
. |
In addition, you also have to specify if you want to use these plugin features and which watch you want to use. This offers to possibility to use or not these plugin features without modifying the task graph definition. For example, you can disable the features if there isn't a paired watch available. Here is an example:
import { getConnectedWatches, setWatchFeaturesState, useWatch } from '@awarns/wear-os';
export async function setupWatchToUse(): Promise<void> {
const watches = await getConnectedWatches();
if (watches.length === 0) {
console.log('No WearOS watches connected! Disabling wear-os plugin features...');
setWatchFeaturesState(false);
return;
}
const watch = watches[0];
console.log(`Setup wear-os plugin to use ${watch.name} watch!`);
setWatchFeaturesState(true);
useWatch(watch);
}
Tasks
| Task name | Description |
|-----------------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| startDetecting{prefix}Watch{sensor}Changes
| Allows to start the data collection for a sensor
with a specific configuration (see below). The prefix
can be used to distinguish among different configurations. |
| stopDetectingWatch{sensor}Changes
| The complement to the previous task. Allows to stop collecting data from sensor
. |
| sendPlainMessageToWatch
| Allows to send a string-based message to the paired smartwatch. An example of use could be to send information for updating the UI. |
| sendPlainMessageToWatchAndAwaitResponse
| Allows to send a string-based message to the paired smartwatch and to wait for a response from it. |
Start data collection for a sensor with specific configuration
To register these tasks for their use, you just need to import them and call their generator functions inside your application's task list:
import { Task } from '@awarns/core/tasks';
import {
startDetectingWatchSensorChangesTask,
WatchSensor,
WatchSensorDelay,
} from '@awarns/wear-os';
export const demoTasks: Array<Task> = [
startDetectingWatchSensorChangesTask(WatchSensor.ACCELEROMETER, { sensorDelay: WatchSensorDelay.NORMAL, batchSize: 50 }),
// startDetectingWatchAccelerometerChanges
startDetectingWatchSensorChangesTask(WatchSensor.ACCELEROMETER, { sensorDelay: WatchSensorDelay.FASTEST, batchSize: 50 }, 'Fast'),
// startDetectingFastWatchAccelerometerChanges
startDetectingWatchSensorChangesTask(WatchSensor.HEART_RATE, { sensorDelay: WatchSensorDelay.NORMAL, batchSize: 5 }),
// startDetectingWatchHeartRateChanges
startDetectingWatchSensorChangesTask(WatchSensor.GEOLOCATION, { sensorDelay: 5000, batchSize: 5 }),
// startDetectingWatchGeolocationChanges
]
Warning: the data collection for a
WatchSensor
can only be started once, ifstartDetectingFastWatchAccelerometerChanges
is executed afterstartDetectingWatchAccelerometerChanges
and while the collection is in progress,startDetectingFastWatchAccelerometerChanges
will be ignored.Therefore, if you want to dynamically change the collection's configuration while the collection is in progress, you will have to stop the collection and start it again with the new desired configuration. However, due to the underlying communication APIs (i.e., Bluetooth), the order of the tasks is not guaranteed. This means that if the stop and start tasks are executed very close in time, the start task could be executed before the stop task in the smartwatch. If you really need to change the collection's configuration, you should wait a certain amount of time (e.g., 1 second) after the execution of the stop task before executing the start task.
Task generator parameters:
| Parameter name | Type | Description |
|-------------------------|-------------------------|--------------------------------------------------------------------------------------------------------------------------------|
| sensor
| WatchSensor
| Sensor to collect data from. See below. |
| providerConfiguration
| ProviderConfiguration
| Collection's configuration of the task. See below. |
| prefix
(Optional) | string
| Adds the prefix to the name of the task. Useful to create multiple tasks for the same sensor but with multiple configurations. |
- WatchSensor
| Value | Description |
|-----------------|----------------------------------------------|
| ACCELEROMETER
| Represents the watch's accelerometer sensor. |
| GYROSCOPE
| Represents the watch's gyroscope sensor. |
| MAGNETOMETER
| Represents the watch's magnetometer sensor. |
| HEART_RATE
| Represents the watch's heart rate monitor. |
| GEOLOCATION
| Represents the watch's GPS system. |
- ProviderConfiguration
| Property | Type | Description |
|---------------|---------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| sensorDelay
| WatchSensorDelay | number | Indicates the time between two consecutive samples. It can be a WatchSensorDelay
(i.e., UI
, NORMAL
, GAME
or FASTEST
) or a value in milliseconds. It is highly recommended to use values in milliseconds (and higher than 1000 ms) with WatchSensor.GEOLOCATION
due to WatchSensorDelay
it's oriented for high sampling rate sensors. |
| batchSize
| number
| Indicates the amount of samples to be delivered in each record. |
Note: the
sensorDelay
is taken as a suggestion by the Android OS. Samples could be delivered at a smaller or higher rate.
Task output events:
These tasks don't produce significant events after they complete their execution aside from the regular
{task-name}Finished
events.However, once the start task has finished running, relevant events will be emitted by the internal listeners while the data collection is active. These are listed below.
watchAccelerometerSamplesAcquired
watchGyroscopeSamplesAcquired
watchMagnetometerSamplesAcquired
watchHeartRateSamplesAcquired
watchGeolocationAcquired
Example usage in the application task graph:
on('startEvent', run('startDetectingWatchAccelerometerChanges')); on('startEvent', run('startDetectingWatchGyroscopeChanges')); on('startEvent', run('startDetectingWatchMagnetometerChanges')); on('startEvent', run('startDetectingWatchHeartRateChanges')); on('startEvent', run('startDetectingWatchGeolocationChanges')); on('watchAccelerometerSamplesAcquired', run('writeRecords')); on('watchGyroscopeSamplesAcquired', run('writeRecords')); on('watchMagnetometerSamplesAcquired', run('writeRecords')); on('watchHeartRateSamplesAcquired', run('writeRecords')); on('watchGeolocationAcquired', run('writeRecords')); on('watchGeolocationAcquired', run('checkAreaOfInterestProximity', { nearbyRange: 100, offset: 15 }));
Note: To use the
writeRecords
or thecheckAreaOfInterestProximity
task, the persistence or geofencing package must be installed and configured, respectively. See persistence and geofencing package docs.
Stop data collection for a sensor
To register these tasks for their use, you just need to import them and call their generator functions inside your application's task list:
import { Task } from '@awarns/core/tasks';
import {
stopDetectingWatchSensorChangesTask,
WatchSensor,
} from '@awarns/wear-os';
export const demoTasks: Array<Task> = [
stopDetectingWatchSensorChangesTask(WatchSensor.ACCELEROMETER), // stopDetectingWatchAccelerometerChanges
stopDetectingWatchSensorChangesTask(WatchSensor.GYROSCOPE), // stopDetectingWatchGyroscopeChanges
stopDetectingWatchSensorChangesTask(WatchSensor.MAGNETOMETER), // stopDetectingWatchMagnetometerChanges
stopDetectingWatchSensorChangesTask(WatchSensor.HEART_RATE), // stopDetectingWatchHeartRateChanges
stopDetectingWatchSensorChangesTask(WatchSensor.GEOLOCATION), // stopDetectingWatchGeolocationChanges
];
Note: a stop task of a specific
WatchSensor
can be used to stop the collection started by any start task for thatWatchSensor
no matter the specific configuration.
Task generator parameters:
| Parameter name | Type | Description |
|-------------------------|-------------------------|------------------------------------------|
| sensor
| WatchSensor
| Sensor to stop the data collection from. |
Task output events:
These tasks don't produce significant events after they complete their execution aside from the regular
{task-name}Finished
events.Example usage in the application task graph:
on('startEvent', run('startDetectingWatchAccelerometerChanges').every(1, 'minute')); on('startEvent', run('startDetectingWatchGyroscopeChanges').every(1, 'minute')); on('startEvent', run('startDetectingWatchMagnetometerChanges').every(1, 'minute')); on('startEvent', run('startDetectingWatchHeartRateChanges').every(1, 'minute')); on('startEvent', run('startDetectingWatchGeolocationChanges').every(1, 'minute')); on('watchAccelerometerSamplesAcquired', run('stopDetectingWatchAccelerometerChanges')); on('watchGyroscopeSamplesAcquired', run('stopDetectingWatchGyroscopeChanges')); on('watchMagnetometerSamplesAcquired', run('stopDetectingWatchMagnetometerChanges')); on('watchHeartRateSamplesAcquired', run('stopDetectingWatchHeartRateChanges')); on('watchGeolocationAcquired', run('stopDetectingWatchGeolocationChanges'));
Note: it makes no sense to use these tasks without using before their complementary tasks to start the data collection.
Send a message to the paired watch
Note: to be able to use this feature, the messaging feature must be enabled.
To register these tasks for their use, you just need to import them and call their generator functions inside your application's task list:
import { Task } from '@awarns/core/tasks';
import {
sendPlainMessageToWatchTask
} from '@awarns/wear-os';
export const demoTasks: Array<Task> = [
sendPlainMessageToWatchTask() // sendPlainMessageToWatch
];
Task generator parameters:
This task generators take no parameters
Task output events:
Example usage in the application task graph:
on('startEvent', run('sendPlainMessageToWatch', { plainMessage: { message: 'Hi from the smartphone!!' } }).every(1, 'minute')); on('plainMessageSent', run('writeRecords'));
Note: To use the
writeRecords
task, the persistence package must be installed and configured. See persistence package docs.
Send a message to the paired watch and wait for a response
Note: to be able to use this feature, the messaging feature must be enabled.
To register these tasks for their use, you just need to import them and call their generator functions inside your application's task list:
import { Task } from '@awarns/core/tasks';
import {
sendPlainMessageToWatchAndAwaitResponseTask
} from '@awarns/wear-os';
export const demoTasks: Array<Task> = [
sendPlainMessageToWatchAndAwaitResponseTask() // sendPlainMessageToWatchAndAwaitResponse
];
Task generator parameters:
This task generators take no parameters
Task output events:
Example usage in the application task graph:
on('startEvent', run('sendPlainMessageToWatchAndAwaitResponse', { plainMessage: { message: 'Tell me something ;)' }, timeout: 3000 }).every(1, 'minute')); on('plainMessageSentAndResponseReceived', run('writeRecords'));
Note: To use the
writeRecords
task, the persistence package must be installed and configured. See persistence package docs.
Send a message from an event's data
Note: to be able to use this feature, the messaging feature must be enabled.
You can also invoke these tasks by injecting the message in the event that triggers their execution. This allows to send messages in a more flexible way (i.e., no need to specify the message in the task graph).
Example usage:
import { awarns } from '@awarns/core'; import { PlainMessage } from '@awarns/wear-os'; export function sendMessage(message: PlainMessage) { awarns.emitEvent('sendMessage', { data: message }); }
Then, in the task graph:
on('sendMessage', run('sendPlainMessageToWatch'));
Receive watch-triggered message
Note: to be able to use this feature, the messaging feature must be enabled.
The watch can also send message to the smartphone by its own (i.e., no need to receive a message from the smartphone first to then reply). When those messages are received
by the smartphone, the plainMessageReceivedEvent
is emitted.
Example usage in the application task graph:
on('plainMessageReceived', run('writeRecords'));
Note: To use the
writeRecords
task, the persistence package must be installed and configured. See persistence package docs.
Events
| Name | Payload | Description |
|---------------------------------------|---------------------------------------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| watchAccelerometerSamplesAcquired
| TriAxial
| Contains a list of samples with the x
, y
, and z
values of an accelerometer sensor. |
| watchGyroscopeSamplesAcquired
| TriAxial
| Contains a list of samples with the x
, y
, and z
values of a gyroscope sensor. |
| watchMagnetometerSamplesAcquired
| TriAxial
| Contains a list of samples with the x
, y
, and z
values of a magnetometer sensor. |
| watchHeartRateSamplesAcquired
| HeartRate
| Contains a list with the values of a heart rate sensor. |
| watchGeolocationAcquired
| Geolocation | Array<Geolocation> | Contains one or more GNSS locations. If the batchSize
is set to 1, the payload will be a Geolocation
record. Otherwise, the payload will be a Geolocation
array. |
| plainMessageSent
| MessageSent
| Contains the content of the message sent to the watch. |
| plainMessageSentAndResponseReceived
| MessageReceived
| Contains the content of the message sent to the watch and the response from it. |
| plainMessageReceived
| MessageReceived
| Contains the content of a message received from the watch. |
Records
TriAxial
| Property | Type | Description |
|-------------|---------------------------------------|--------------------------------------------------------------------------------------------------------|
| id
| string
| Record's unique id. |
| type
| string
| One of the following values: watch-accelerometer
, watch-gyroscope
, or watch-magnetometer
. |
| change
| Change
| Always NONE
. |
| timestamp
| Date
| The local time when the data was collected. It is equal to the time of the first sample in the record. |
| samples
| TriAxialSample[]
| List with the collected samples. |
TriAxialSample
| Property | Type | Description |
|-------------|----------|----------------------------------------------------------------|
| x
| number
| Value x
of the sensor. |
| y
| number
| Value y
of the sensor. |
| z
| number
| Value z
of the sensor. |
| timestamp
| number
| The local time (UNIX timestamp) when the sample was collected. |
HeartRate
| Property | Type | Description |
|-------------|-----------------------------------------|--------------------------------------------------------------------------------------------------------|
| id
| string
| Record's unique id. |
| type
| string
| Always watch-heart-rate
. |
| change
| Change
| Always NONE
. |
| timestamp
| Date
| The local time when the data was collected. It is equal to the time of the first sample in the record. |
| samples
| HeartRateSample[]
| List with the collected samples. |
HeartRateSample
| Property | Type | Description |
|-------------|----------|----------------------------------------------------------------|
| value
| number
| Heart rate value reported by the sensor. |
| timestamp
| number
| The local time (UNIX timestamp) when the sample was collected. |
Geolocation
| Property | Type | Description |
|----------------------|----------|--------------------------------------------------------------------------------------------------------|
| id
| string
| Record's unique id. |
| type
| string
| Always watch-geolocation
. |
| change
| Change
| Always NONE
. |
| timestamp
| Date
| The local time when the data was collected. It is equal to the time of the first sample in the record. |
| latitude
| number
| Latitude reported by the GPS. |
| longitude
| number
| Longitude reported by the GPS. |
| altitude
| number
| Altitude reported by the GPS. |
| verticalAccuracy
| number
| The estimated error in the latitude. |
| horizontalAccuracy
| number
| The estimated error in the longitude. |
| speed
| number
| The estimated speed of the device when the location was acquired. |
| direction
| number
| The estimated direction of the device when the location was acquired. |
MessageSent
| Property | Type | Description |
|-------------|--------------------------------|-------------------------------------------|
| id
| string
| Record's unique id. |
| type
| string
| Always plain-message-sent
. |
| change
| Change
| Always NONE
. |
| timestamp
| Date
| The local time when the message was sent. |
| content
| PlaiMessage
| Content of the message sent. |
PlainMessage
| Property | Type | Description |
|-----------------|----------------|----------------------------------------------------------------------------------------------------------------|
| message
| string
| The content of the message. |
| inResponseTo?
| PlainMessage
| Can contain a PlainMessage
to indicate that the current message is a response to the inResponseTo
message. |
MessageReceived
| Property | Type | Description |
|-------------|---------------------------------------|-------------------------------------------|
| id
| string
| Record's unique id. |
| type
| string
| Always plain-message-received
. |
| change
| Change
| Always NONE
. |
| timestamp
| Date
| The local time when the message was sent. |
| content
| ReceivedMessage
| Content of the received message. |
ReceivedMessage
| Property | Type | Description |
|----------------|----------------|----------------------------------------|
| senderNodeId
| string
| Id of the watch that sent the message. |
| plainMessage
| PlainMessage
| Message received. |
License
Apache License Version 2.0