import { NativeEventEmitter, NativeModules, Platform } from 'react-native'; import type { EmitterSubscription as Subscription } from 'react-native'; import type { StepCountData, Spec } from './NativeBackgroundStepCounter'; import { eventName, VERSION, NAME } from './NativeBackgroundStepCounter'; /* A way to check if the module is linked. */ const LINKING_ERROR = "The package 'react-native-background-step-counter' doesn't seem to be linked. Make sure: \n\n" + Platform.select({ ios: '- You have run `pod install` in the `ios` directory and then clean, rebuild and re-run the app. You may also need to re-open Xcode to get the new pods.\n', android: '- You have the Android development environment set up: `https://reactnative.dev/docs/environment-setup.`', default: '', }) + '- Use the "npx react-native clean" command to clean up the module\'s cache and select the ' + '"watchman", "yarn", "metro", "android", "npm" options with comma-separated. ' + 'Re-Install packages and re-build the app again .' + '- You rebuilt the app after installing the package\n' + '- You are not using Expo Go\n' + 'If none of these fix the issue, please open an issue on the Github repository: ' + 'https://github.com/Ugur-Atakan/react-native-background-step-counter`'; export interface ParsedStepCountData { steps: number; startDate: Date; endDate: Date; distance: number; } /** * We keep TurboModuleManager alive until the JS VM is deleted. * It is perfectly valid to only use/create TurboModules from JS. * In such a case, we shouldn't de-alloc TurboModuleManager if there * aren't any strong references to it in ObjC. Hence, we give * __turboModuleProxy a strong reference to TurboModuleManager. * @see https://github.com/facebook/react-native/blob/main/packages/react-native/Libraries/TurboModule/TurboModuleRegistry.js * @see https://github.com/facebook/react-native/blob/main/packages/react-native/ReactCommon/react/nativemodule/core/platform/ios/RCTTurboModuleManager.mm * @see https://github.com/facebook/react-native/blob/main/packages/react-native/ReactCommon/react/nativemodule/core/ReactCommon/TurboModuleBinding.cpp */ // @ts-ignore const isTurboModuleEnabled = global.__turboModuleProxy != null; /** * The `StepCounterModule` constant is used to import the native module `NativeStepCounter` if * TurboModules are enabled. If TurboModules are not enabled, it falls back to using the * `NativeModules.StepCounter` module. This allows the code to work with both TurboModules and * non-TurboModules environments. * https://github.com/Ugur-Atakan/react-native-step-counter-fixed/issues/29#issue-1857677086 */ const BackgroundStepCounterModule = isTurboModuleEnabled ? require('./NativeBackgroundStepCounter').default : NativeModules.BackgroundStepCounter; /** * A module that allows you to get the step count data. * `CMStepCounter` is deprecated in iOS 8.0. Used `CMPedometer` instead. * floorsAscended - The number of floors ascended during the time period. iOS Only. * floorsDescended - The number of floors descended during the time period. iOS Only. * counterType - The type of counter used to count the steps. * @throws {Error} LINKING_ERROR - Throws Error If global variable turboModuleProxy is undefined. * @example * import { StepCounter } from '@uguratakan/react-native-step-counter'; */ const BackgroundStepCounter = ( BackgroundStepCounterModule ? BackgroundStepCounterModule : new Proxy( {}, { get() { throw new Error(LINKING_ERROR); }, } ) ) as Spec; const StepEventEmitter = new NativeEventEmitter(BackgroundStepCounter); type StepCountUpdateCallback = (result: StepCountData) => void; export const isSensorWorking = StepEventEmitter.listenerCount(eventName) > 0; /** * Transform the step count data into a more readable format. * You can use it or directly use the `StepCountData` type. * @param {StepCountData} data - Step Counter Sensor Event Data. * @returns {ParsedStepCountData} - String Parsed Count Data. */ export function parseStepData(data: StepCountData): ParsedStepCountData { const { steps, startDate, endDate, distance } = data; const endDateTime = new Date(endDate); const startDateTime = new Date(startDate); const roundedDistance = Number(distance.toFixed(1)); return { steps, startDate: startDateTime, endDate: endDateTime, distance: roundedDistance, }; } /** * If you're using a method or property that's not available on the current platform, throw this error. * @param {string} moduleName The name of the module. * @param {string} propertyName The name of the property. * @returns {Error} The error. * @example * if (!StepCounter.startStepCounterUpdate) { * throw new UnavailabilityError(NativeModuleName, eventName); * } */ class UnavailabilityError extends Error { code: string; constructor(moduleName: string, propertyName: string) { super( `The method or property ${moduleName}.${propertyName} is not available on ${Platform.OS}, ` + "are you sure you've linked all the native dependencies properly?" ); this.code = 'ERR_UNAVAILABLE'; } } /** * Returns whether the stepCounter is enabled on the device. * iOS 8.0+ only. Android is available since KitKat (4.4 / API 19). * @see https://developer.android.com/about/versions/android-4.4.html * @see https://developer.apple.com/documentation/coremotion/cmpedometer/1613963-isstepcountingavailable * @returns {Promise>} A promise that resolves with an object containing the stepCounter availability. * supported - Whether the stepCounter is supported on device. * granted - Whether user granted the permission. */ export function startBackgroundService() { return BackgroundStepCounter.startBackgroundService(); } export function stopBackgroundService() { return BackgroundStepCounter.stopBackgroundService(); } export function isStepCountingSupported(): Promise> { return BackgroundStepCounter.isStepCountingSupported(); } /** * Start to subscribe stepCounter updates. * Only the past seven days worth of data is stored and available for you to retrieve. * Specifying a start date that is more than seven days in the past returns only the available data. * ### iOS * `CMStepCounter.startStepCountingUpdates` is deprecated since iOS 8.0. so used `CMPedometer.startUpdates` instead. * @see https://developer.apple.com/documentation/coremotion/cmpedometer/1613950-startupdates * @see https://developer.apple.com/documentation/coremotion/cmstepcounter/1616151-startstepcountingupdates * @param {Date} start A date indicating the start of the range over which to measure steps. * @param {StepCountUpdateCallback} callBack - This callback function makes it easy for app developers to receive sensor events. * @returns {Subscription} - Returns a Subscription that enables you to call. * When you would like to unsubscribe the listener, just use a method of subscriptions's `remove()`. * @example * const startDate = new Date(); * startStepCounterUpdate(startDate).then((response) => { * const data = parseStepCountData(response); * }) */ export function startStepCounterUpdate( callBack: StepCountUpdateCallback ): Subscription { if (!BackgroundStepCounter.startStepCounterUpdate) { throw new UnavailabilityError(NAME, eventName); } BackgroundStepCounter.startStepCounterUpdate(); return StepEventEmitter.addListener(eventName, callBack); } /** * Stop the step counter updates. * ### iOS * `CMStepCounter.stopStepCountingUpdates` is deprecated since iOS 8.0. so used `CMPedometer.stopUpdates` instead. * @see https://developer.apple.com/documentation/coremotion/cmpedometer/1613973-stopupdates * @see https://developer.apple.com/documentation/coremotion/cmstepcounter/1616157-stopstepcountingupdates */ export function stopStepCounterUpdate(): void { StepEventEmitter.removeAllListeners(eventName); BackgroundStepCounter.stopStepCounterUpdate(); } export { NAME, VERSION }; export default BackgroundStepCounter;