npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

@awarns/wear-os

v1.0.0

Published

Reliable and concurrent access to a paired smartwatch's sensors and enables communication between both devices

Downloads

22

Readme

@awarns/wear-os

npm (scoped) npm

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, if startDetectingFastWatchAccelerometerChanges is executed after startDetectingWatchAccelerometerChanges 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.

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 the checkAreaOfInterestProximity 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 that WatchSensor 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