UNPKG

@hangtime/grip-connect-react-native

Version:
110 lines (90 loc) 3.47 kB
import { BleManager, Device } from "react-native-ble-plx" import { WHC06 as WHC06Base } from "@hangtime/grip-connect" import { Buffer } from "buffer" /** * Represents a Weiheng - WH-C06 (or MAT Muscle Meter) device. * To use this device enable: `chrome://flags/#enable-experimental-web-platform-features`. * {@link https://googlechrome.github.io/samples/web-bluetooth/scan.html| Web Bluetooth} * {@link https://weihengmanufacturer.com} */ export class WHC06 extends WHC06Base { manager: BleManager device?: Device constructor() { super() this.manager = new BleManager() } private base64ToHex(base64: string): string { const binary = Buffer.from(base64, "base64").toString("binary") return Array.from(binary) .map((char) => char.charCodeAt(0).toString(16).padStart(2, "0")) .join("") } private parseWeightData(manufacturerData: string | null): number { if (!manufacturerData) return 0 try { const hexData = this.base64ToHex(manufacturerData) const weightHex = hexData.substring(24, 28) return parseInt(weightHex, 16) / 100 } catch (error) { console.error("Error parsing weight data:", error) return 0 } } override connect = async ( onSuccess: () => void = () => console.log("Connected successfully"), onError: (error: Error) => void = (error) => console.error(error), ): Promise<void> => { try { this.manager.startDeviceScan(null, { scanMode: 2, callbackType: 1 }, (error, scannedDevice) => { if (error) { onError(error) return } if (scannedDevice && (scannedDevice.localName === "IF_B7" || scannedDevice.name === "IF_B7")) { // Update timestamp this.updateTimestamp() // Device has no services / characteristics, so we directly call onSuccess onSuccess() const manufacturerData = scannedDevice.manufacturerData // Handle recieved data const weight = this.parseWeightData(manufacturerData) // Update massMax const receivedTime: number = Date.now() const receivedData = weight // Tare correction const numericData = receivedData - this.applyTare(receivedData) * -1 // Add data to downloadable Array this.downloadPackets.push({ received: receivedTime, sampleNum: this.dataPointCount, battRaw: 0, samples: [numericData], masses: [numericData], }) // Update massMax this.massMax = Math.max(Number(this.massMax), numericData).toFixed(1) // Update running sum and count const currentMassTotal = Math.max(-1000, numericData) this.massTotalSum += currentMassTotal this.dataPointCount++ // Calculate the average dynamically this.massAverage = (this.massTotalSum / this.dataPointCount).toFixed(1) // Check if device is being used this.activityCheck(numericData) // Notify with weight data this.notifyCallback({ massMax: this.massMax, massAverage: this.massAverage, massTotal: Math.max(-1000, numericData).toFixed(1), }) } }) } catch (error) { onError(error as Error) } } override download = async (): Promise<void> => { throw new Error("Download is not supported on React Native") } }