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

nativescript-wearos-sensors

v1.3.0

Published

Plugin to develop smartphone apps that receive sensors' data from a paired WearOS smartwatch.

Downloads

19

Readme

nativescript-wearos-sensors

npm npm

The nativescript-wearos-sensors is a plugin developed with the NativeScript framework. It allows to develop smartphone applications for collecting data from the IMU sensors (i.e., accelerometer and gyroscope), the magnetometer, the heart rate and the GPS of a paired Android WearOS smartwatch (if the corresponding sensor is available in the device).

The Android WearOS smartwatch must run a counterpart application built using the native WearOS Sensors library. Then, the smartphone application can request the smartwatch to start/stop the collection of the specified sensor, and the smartwatch will send the collected samples to the smartphone application.

[!WARNING] An application using this plugin is completely useless if there is not a counterpart application build with the WearOS Sensors library installed in the paired smartwatch. In other words, the smartphone can not work by itself alone. It requires a smartwatch to work.

The data collection can be started both from the smartphone and from the paired smartwatch. In addition. the plugin offers a way to communicate with the smartwatch by sending messages.

Installation

Run the following command in your project's root folder.

ns plugin add nativescript-wearos-sensors

Requirements

This plugin is only supported by Android smartphones. To use it to build an application, the following requirements apply:

  • An Android smartphone running Android 6 (API level 23) or higher.

[!CAUTION] The targetSdkVersion of the built application must be lower or equal than 31 (Android 12). Apps built targeting >=31 will not work due to an issue with Dynamic File Loading.

  • In addition, the smartphone must be paired with a smartwatch with the counterpart application installed. To link a smartwatch to the smartphone, you must also install the Smartwatch WearOS by Google or the specific application provided by the smartwatch manufacturer (e.g., Mobvoi Health, Samsung Wearable, etc.) and follow the procedure to link both devices.

[!IMPORTANT] Both applications (smartphone and smartwatch apps) must have the same application id. If that's not the case, the applications will not be able to interact. You can change the application id of a NativeScript application in the nativescript.config.ts.

Usage

The plugin offers two features:

  • Sensor data collection from the paired smartwatch: it can be started/stopped from the smartphone and from the smartwatch. The smartwatch is able to start and stop the data collection thanks to the WearCommands feature.
  • Plain Messaging: it allows to send and receive simple messages between both devices.

In first place, you need to initialize the plugin with a WearosSensorsConfig in your app.ts (TypeScript app) or main.ts (Angular app) file:

// TypeScript App:
import { Application } from "@nativescript/core";
// or Angular App:
import { platformNativeScriptDynamic } from "nativescript-angular/platform";
import { AppModule } from "./app/app.module";

// WearOSSensors import
import { wearosSensors, allSensors } from "nativescript-wearos-sensors";

wearosSensors.init({
    sensors: allSensors,
    disablePlainMessaging: false,
    disableWearCommands: false
});

// TypeScript App:
Application.run({ moduleName: "app-root" });
// Angular App:
platformNativeScriptDynamic().bootstrapModule(AppModule);

The initialization parameter is optional, and it allows specifying which sensors are enabled (sensors), and if PlainMessaging (disablePlainMessaging) and WearCommands (disableWearCommands) features are enabled. The default configuration is the one shown in the example above: all sensors and features enabled.

[!NOTE] The configuration allows to conditionally wire up native components with the core of the plugin. This allows to reduce the memory used by the application when some features are not going to be used.

Sensor data collection

As stated above, the data collection can be started/stopped by both devices, but only the smartphone has access to the collected data.

To receive the collected data, the smartphone has to register a listener (at least one) via the CollectorManager. A listener can be set up for a specific node/s (i.e., smartwatch) and a specific sensor/s. This means that there can be several listeners registered, all of them listening for different nodes or sensors. This behaviour can be achieved using ListenerFilters. Here's an example of registering different listeners:

import { getCollectorManager } from "nativescript-wearos-sensors/collection";
import { SensorRecord } from "nativescript-wearos-sensors/sensors/records";
import { Node } from "nativescript-wearos-sensors/node";
import { ListenerFilter } from "nativescript-wearos-sensors/listeners";
import { SensorType } from "nativescript-wearos-sensors/sensors";


function registerGlobalListener() {
    // Register a listener witout filters --> receives records from all sources
    getCollectorManager().addSensorListener((sensorRecord: SensorRecord<any>) => {
        console.log(deviceId, JSON.stringify(sensorRecord));
    });
}

function registerListenerForNode(node: Node) {
    // Register a listener filtering per node --> receives all kind of records from that node
    const filter: ListenerFilter = {
        nodes: [node]
    }
    getCollectorManager().addSensorListener((sensorRecord: SensorRecord<any>) => {
        console.log(deviceId, JSON.stringify(sensorRecord));
    }, filter);
}

function registerListenerForSensor(sensor: SensorType) {
    // Register a listener filtering per sensor --> receives records of that sensor from any node
    const filter: ListenerFilter = {
        sensors: [sensor]
    }
    getCollectorManager().addSensorListener((sensorRecord: SensorRecord<any>) => {
        console.log(deviceId, JSON.stringify(sensorRecord));
    }, filter);
}

function registerListenerForNodeAndSensors(node: Node, sensors: SensorType[]) {
    // Register a listener filtering per node and sensor --> receives records of that sensors from that node
    const filter: ListenerFilter = {
        nodes: [node],
        sensors: sensors
    }
    getCollectorManager().addSensorListener((sensorRecord: SensorRecord<any>) => {
        console.log(deviceId, JSON.stringify(sensorRecord));
    }, filter);
}

Start/stop data collection from smartphone

In order to start the data collection for node, first you have to get the connected nodes using the NodeDiscoverer. Then, once you have the connected node, you have to follow some steps to start the data collection:

  1. Check if a specific sensor on the node is ready to be collected from.
  2. If it is not ready:
    1. It is because the sensor is not present in the device.
    2. The sensor is in the device, but the smartwatch app has no permissions to collect from that sensor.
  3. If we lack of permissions, we can just ask the user to grant them.
    1. If permissions are rejected, it's the end of the way...
    2. If permissions are granted, we can start the collection!!

To start the data collection, we also should specify a CollectionConfiguration, where we can indicate the desired time between consecutive samples (sensorDelay), and the amount of samples to deliver each time (batchSize). The configuration is optional, if no configuration is provided then default values apply.

[!NOTE] Due to the smartwatch has to send the collected data via Bluetooth, we can't send individual samples when working with a high sampling rate. That would saturate the connection. To solve this problem, we send the samples in batches.

Here is an example of this collection procedure:

import { getNodeDiscoverer } from "nativescript-wearos-sensors/node";
import { getCollectorManager, PrepareError, CollectionConfiguration } from "nativescript-wearos-sensors/collection";
import { Node } from "nativescript-wearos-sensors/node";
import { SensorType } from "nativescript-wearos-sensors/sensors";

async function getNodes(): Promise<Node[]> {
    await nodesDiscovered = nodeDiscoverer.getConnectedNodes();
    const nodes = []
    nodesDiscovered.forEach((nodeDiscovered) => {
        if (nodeDiscovered.error) {
            this.logger.logResult(nodeDiscovered.error);
            return;
        }
        nodes.push(nodeDiscovered.node);
    });
    return nodes;
}

async function collectFrom(node: Node, sensor: SensorType, config: CollectionConfiguration) {
    const collectorManager = getCollectorManager();
    
    const isReady = await collectorManager.isReady(node, sensor);
    if (!isReady) {
        const prepareError: PrepareError = await collectorManager.prepare(node, sensor);
        if (prepareError) {
            console.log(prepareError.message);
            return;
        }
    }
    
    await collectorManager.startCollecting(node, sensor, config);
}

async function stopCollecingFrom(node: Node, sensor: SensorType) {
    await collectorManager.stopCollecting(node, sensor);
}

Start/stop data collection from smartwatch

The plugin fully handles this for you. You only have to make sure to register at least a listener to receive the collected data.

[!IMPORTANT] The WearCommands feature must be enabled at plugin initialization.

PlainMessaging

With a system composed by several devices, it is important to have a way to communicate. We provide the PlainMessageClient, which allows to send and receive string based messages. There are two types of received messages: the ones which require a response and the ones which don't. Here's an example on how to use the messaging feature:

import { getPlainMessageClient } from "src/internal/communication/plain-message";

function registerListener(): void {
    // Register a listener to receive messages from the smartwatch
    getPlainMessageClient().registerListener((receivedMessage) => {
        console.log(`received single message ${JSON.stringify(receivedMessage)}`);
    });
}

async function sendMessage(node: Node, message: string): void {
    // Send a message to the smartwatch
    const plainMessage = {message: "You don't have to reply :)"};
    await getPlainMessageClient().send(node, plainMessage);
}

async function sendMessageAndWaitResponse(node: Node, message: string): void {
    // Send a message to the smartwatch and wait for a response
    const plainMessage = {message: "PING!"};
    const receivedMessage = await getPlainMessageClient().sendExpectingResponse(node, plainMessage);
    console.log(`response received: ${JSON.stringify(receivedMessage)}`);
}

[!IMPORTANT] The PlainMessaging feature must be enabled at plugin initialization.

API

wearosSensors - Methods

| Name | Return type | Description | |--------------------------------------|-----------------|---------------------------------------------------------------------------------------------------------------------------------------| | init(config?: WearosSensorsConfig) | Promise<void> | Initializes the native components depending on the provided configuration. If no configuration provided, defaults to defaultConfig. |

WearosSensorsConfig

| Property | Type | Description | |--------------------------|----------------|----------------------------------------------------------| | sensors? | SensorType[] | Sensors that are going to be used. Default: all sensors. | | disablePlainMessaging? | boolean | Disable plain messaging feature. Default: false. | | disableWearCommands? | boolean | Disable wear commands feature. Default: false. |

defaultConfig
export const defaultConfig = {
    sensors: allSensors, // Constant containing all the sensors
    disablePlainMessaging: false,
    disableWearCommands: false
};

NodeDiscoverer

| Function | Return type | Description | |---------------------------------------------|----------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------| | getLocalNode() | Promise<Node> | Get a reference to the local node (smartphone). | | areConnectedNodes() | Promise<boolean> | Returns true if there are connected nodes. | | getConnectedNodes(timeout: number = 5000) | Promise<NodeDiscovered[] | Get the currently connected nodes and their available sensors. Timeout indicates the maximum wait time for the connected nodes to communicate with the smartphone. |

Node

| Field | Type | Description | |----------------|----------------|-------------------------------------------| | name | string | Name of the device. | | id | string | Id number of the device. | | capabilities | SensorType[] | Sensors that are available on the device. |

NodeDiscovered

| Field | Type | Description | |----------|--------|-----------------------------------------------------------------------------------------------------------------| | node | Node | Reference to a Node. | | error? | any | An error message. Present if the Node was not able to communicate with the smartphone in the specified timeout. |

SensorType

| Value | Description | |-----------------|--------------------------------------| | ACCELEROMETER | Represents the accelerometer sensor. | | GYROSCOPE | Represents the gyroscope sensor. | | MAGNETOMETER | Represents the magnetometer sensor. | | HEART_RATE | Represents the heart rate sensor. | | LOCATION | Represents the GPS sensor. |

CollectorManager

| Method | Return type | Description | |------------------------------------------------------------------------------------|-------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | isEnabled(sensor: SensorType) | boolean | Returns true if the sensor type is enabled in the initial configuration. | | isReady(node: Node, sensor: SensorType) | Promise<boolean> | Return true if the sensor is ready to collect data from. | | prepare(node: Node, sensor: SensorType) | Promise<PrepareError> | Returns a PrepareError if anything failed in the preparation (e.g., sensor unavailable, no permissions, etc). Returns undefined if the preparation was successful. | | startCollecting(node: Node, sensor: SensorType, config?: CollectionConfiguration | Promise<void> | Starts the data collection of a sensor in a node with the specified configuration. | | stopCollecting(node: Node, sensor: SensorType) | Promise<void> | Stops the data collection of a sensor in a node. | | addSensorListener(listener: SensorListener, filters?: ListenerFilter) | number | Adds a listener with the specified filters and returns a listener identifier. | | removeSensorListener(listenerId?: number) | void | Removes the listener specified by the listenerId. If not provided, removes all listeners. |

PrepareError

| Property | Type | Description | |-----------|----------|----------------------------------------------------------| | node | Node | Reference to the Node where the PrepareError comes from. | | message | string | Message describing the error. |

CollectionConfiguration

| Property | Type | Description | |------------------|------------------|---------------------------------------------------------------------------------------------------| | sensorInterval | SensorInterval | Time between each consecutive sample. Can be a NativeSensorInterval or a value in milliseconds. | | batchSize | number | Amount of samples to be sent in each record. |

SensorListener

(sensorRecord: SensorRecord<any>) => void

ListenerFilter

| Property | Type | Description | |------------|----------------|-------------------------------------------------| | nodes? | Node[] | For which nodes the related listener applies. | | sensors? | SensorType[] | For which sensors the related listener applies. |

[!TIP] Filter works as follows:

{ 
  nodes: [node1, /* OR */ node2]
  // AND
  sensors: [SensorType.ACCELEROMETER, /* OR */ SensorType.GYROSCOPE]
}

SensorRecord

| Property | Type | Description | |------------|--------------|--------------------------------------------------------------------------------------------------------| | type | SensorType | Type of the collected data. | | deviceId | string | Id of the device where the collected data comes from. | | samples | T[] | List of samples, where T is TriAxialSensorSample, HeartRateSensorSample, or LocationSensorSample |

TriAxialSensorSample

| Property | Type | Description | |-----------|-----------|----------------| | x | number | Component x. | | y | number | Component y. | | z | number | Component z. |

HeartRateSensorSample

| Property | Type | Description | |----------|-----------|-------------------| | value | number | Heart rate value. |

LocationSensorSample

| Property | Type | Description | |----------------------|----------|--------------------------------------------------------------| | latitude | number | Latitude coordinate component. | | longitude | number | Longitude coordinate component. | | altitude | number | Altitude coordinate component. | | verticalAccuracy | number | Estimated error in the latitude. | | horizontalAccuracy | number | Estimated error in the longitude. | | speed | number | Estimated device's speed when the location was acquired. | | direction | number | Estimated device's direction when the location was acquired. |

PlainMessageClient

| Function | Return type | Description | |-----------------------------------------------------------------------------------|----------------------------|------------------------------------------------------------------------------------| | enabled() | boolean | Returns true if the plain message feature is enabled in the initial configuration. | | registerListener(listener: PlainMessageListener) | void | Registers the listener for the feature. | | unregisterListener() | void | Unregisters the listener for the feature. | | send(node: Node, plainMessage: PlainMessage) | Promise<void> | Sends a message to the specified Node. | | sendExpectingResponse(node: Node, plainMessage: PlainMessage, timeout?: number) | Promise<ReceivedMessage> | Sends a message to the specified Node and wait timeout ms for a response. |

PlainMessage

| Property | Type | Description | |-----------------|----------------|-----------------------------------------------------------------------------------------------------------------------------------------| | message | string | Content of the message. | | inResponseTo? | PlainMessage | Contains the message at which the current message is responding. undefined means that the message is not responding to other message. |

ReceivedMessage

| Property | Type | Description | |----------------|----------------|--------------------------------------| | senderNodeId | string | Id of the node who sent the message. | | PlainMessage | PlainMessage | Message received. |

PlainMessageListener

(receivedMessage: ReceivedMessage) => void

License

Apache License 2.0

See LICENSE.

Author

Acknowledgements

The development of this library has been possible thanks to the Spanish Ministry of Universities (grant FPU19/05352).