@kingstinct/react-native-healthkit
Version:
React Native bindings for HealthKit
425 lines (399 loc) • 12.4 kB
text/typescript
import { Platform } from 'react-native'
// Import types for default values and function signatures.
// These types are expected to be available via the './types' export.
// You might need to adjust the import path or ensure these types are correctly exported from './types'
// This import is crucial for deriving the type of the default export `HealthkitModule`
// It assumes that index.ios.ts exports a default object matching the Healthkit native module structure.
import type ReactNativeHealthkit from './healthkit.ios'
import type { WorkoutProxy } from './specs/WorkoutProxy.nitro'
import { AuthorizationRequestStatus, AuthorizationStatus } from './types/Auth'
import type {
CategorySamplesWithAnchorResponseTyped,
CategorySampleTyped,
} from './types/CategoryType'
import type { CategoryTypeIdentifier } from './types/CategoryTypeIdentifier'
import {
BiologicalSex,
BloodType,
FitzpatrickSkinType,
WheelchairUse,
} from './types/Characteristics'
import type { QuantitySample } from './types/QuantitySample'
import type { QueryOptionsWithAnchor } from './types/QueryOptions'
export * from './types'
const notAvailableError = `[@kingstinct/react-native-healthkit] Platform "${Platform.OS}" not supported. HealthKit is only available on iOS.`
let hasWarned = false
// @ts-ignore
function UnavailableFnFromModule<
TKey extends keyof typeof ReactNativeHealthkit,
// @ts-ignore
// biome-ignore lint/complexity/noBannedTypes: it works
T extends Function = (typeof ReactNativeHealthkit)[TKey],
// @ts-ignore
>(_fn: TKey, defaultValue: ReturnType<T>): T {
// @ts-ignore
return () => {
if (Platform.OS !== 'ios' && !hasWarned) {
console.warn(notAvailableError)
hasWarned = true
}
return defaultValue
}
}
// --- Mock Implementations for exported functions ---
// CoreModule functions
export const authorizationStatusFor = UnavailableFnFromModule(
'authorizationStatusFor',
AuthorizationStatus.notDetermined,
)
export const disableAllBackgroundDelivery = UnavailableFnFromModule(
'disableAllBackgroundDelivery',
Promise.resolve(false),
)
export const disableBackgroundDelivery = UnavailableFnFromModule(
'disableBackgroundDelivery',
Promise.resolve(false),
)
export const enableBackgroundDelivery = UnavailableFnFromModule(
'enableBackgroundDelivery',
Promise.resolve(false),
)
export const getPreferredUnits = UnavailableFnFromModule(
'getPreferredUnits',
Promise.resolve([]),
)
export const getRequestStatusForAuthorization = UnavailableFnFromModule(
'getRequestStatusForAuthorization',
Promise.resolve(AuthorizationRequestStatus.unknown),
)
export const isHealthDataAvailable = UnavailableFnFromModule(
'isHealthDataAvailable',
false,
) // Original was synchronous
export const isHealthDataAvailableAsync = UnavailableFnFromModule(
'isHealthDataAvailableAsync',
Promise.resolve(false),
) // Added for consistency if needed
export const querySources = UnavailableFnFromModule(
'querySources',
Promise.resolve([]),
)
export const requestAuthorization = UnavailableFnFromModule(
'requestAuthorization',
Promise.resolve(false),
)
export const deleteObjects = UnavailableFnFromModule(
'deleteObjects',
Promise.resolve(0),
)
export const subscribeToChanges = UnavailableFnFromModule(
'subscribeToChanges',
'dummy-query-uuid',
) // Mocking the observer query UUID
export const isProtectedDataAvailable = UnavailableFnFromModule(
'isProtectedDataAvailable',
false,
)
export const isObjectTypeAvailable = UnavailableFnFromModule(
'isObjectTypeAvailable',
false,
)
export const isObjectTypeAvailableAsync = UnavailableFnFromModule(
'isObjectTypeAvailableAsync',
Promise.resolve(false),
)
export const areObjectTypesAvailable = UnavailableFnFromModule(
'areObjectTypesAvailable',
{},
)
export const areObjectTypesAvailableAsync = UnavailableFnFromModule(
'areObjectTypesAvailableAsync',
Promise.resolve({}),
)
// CharacteristicTypeModule functions
export const getBiologicalSex = UnavailableFnFromModule(
'getBiologicalSex',
BiologicalSex.notSet,
)
export const getBloodType = UnavailableFnFromModule(
'getBloodType',
BloodType.notSet,
)
export const getDateOfBirth = UnavailableFnFromModule(
'getDateOfBirth',
new Date(0),
) // Assuming string for date
export const getFitzpatrickSkinType = UnavailableFnFromModule(
'getFitzpatrickSkinType',
FitzpatrickSkinType.notSet,
)
export const getWheelchairUse = UnavailableFnFromModule(
'getWheelchairUse',
WheelchairUse.notSet,
)
// QuantityTypeModule functions
export const queryQuantitySamples = UnavailableFnFromModule(
'queryQuantitySamples',
Promise.resolve([]),
)
export const queryQuantitySamplesWithAnchor = UnavailableFnFromModule(
'queryQuantitySamplesWithAnchor',
Promise.resolve({
samples: [],
deletedSamples: [],
newAnchor: '',
}),
)
export const queryStatisticsForQuantity = UnavailableFnFromModule(
'queryStatisticsForQuantity',
Promise.resolve({}),
)
export const queryStatisticsCollectionForQuantity = UnavailableFnFromModule(
'queryStatisticsCollectionForQuantity',
Promise.resolve([]),
)
export const saveQuantitySample = UnavailableFnFromModule(
'saveQuantitySample',
Promise.resolve(false),
)
export const isQuantityCompatibleWithUnit = UnavailableFnFromModule(
'isQuantityCompatibleWithUnit',
false,
)
// CategoryTypeModule functions
export function queryCategorySamples<T extends CategoryTypeIdentifier>(
_categoryTypeIdentifier: T,
): Promise<CategorySampleTyped<T>[]> {
if (Platform.OS !== 'ios' && !hasWarned) {
console.warn(notAvailableError)
hasWarned = true
}
return Promise.resolve([])
}
export function queryCategorySamplesWithAnchor<
T extends CategoryTypeIdentifier,
>(
_categoryTypeIdentifier: T,
_options: QueryOptionsWithAnchor,
): Promise<CategorySamplesWithAnchorResponseTyped<T>> {
if (Platform.OS !== 'ios' && !hasWarned) {
console.warn(notAvailableError)
hasWarned = true
}
return Promise.resolve({
samples: [],
deletedSamples: [],
newAnchor: '',
})
}
export const saveCategorySample = UnavailableFnFromModule(
'saveCategorySample',
Promise.resolve(false),
)
// CorrelationTypeModule functions
export const queryCorrelationSamples = UnavailableFnFromModule(
'queryCorrelationSamples',
Promise.resolve([]),
)
export const saveCorrelationSample = UnavailableFnFromModule(
'saveCorrelationSample',
Promise.resolve(false),
)
// HeartbeatSeriesModule functions
export const queryHeartbeatSeriesSamples = UnavailableFnFromModule(
'queryHeartbeatSeriesSamples',
Promise.resolve([]),
)
export const queryHeartbeatSeriesSamplesWithAnchor = UnavailableFnFromModule(
'queryHeartbeatSeriesSamplesWithAnchor',
Promise.resolve({
samples: [],
deletedSamples: [],
newAnchor: '',
}),
)
// WorkoutsModule functions
export const queryWorkoutSamples = UnavailableFnFromModule(
'queryWorkoutSamples',
Promise.resolve([]),
)
export const queryWorkoutSamplesWithAnchor = UnavailableFnFromModule(
'queryWorkoutSamplesWithAnchor',
Promise.resolve({
workouts: [],
deletedSamples: [],
newAnchor: '',
}),
)
export const saveWorkoutSample = UnavailableFnFromModule(
'saveWorkoutSample',
Promise.resolve(''),
)
export const startWatchApp = UnavailableFnFromModule(
'startWatchApp',
Promise.resolve(false),
)
// StateOfMindModule functions
export const queryStateOfMindSamples = UnavailableFnFromModule(
'queryStateOfMindSamples',
Promise.resolve([]),
)
export const saveStateOfMindSample = UnavailableFnFromModule(
'saveStateOfMindSample',
Promise.resolve(false),
)
// Utility functions (from original export list)
export function getMostRecentCategorySample<T extends CategoryTypeIdentifier>(
_identifier: T,
): Promise<CategorySampleTyped<T> | undefined> {
if (Platform.OS !== 'ios' && !hasWarned) {
console.warn(notAvailableError)
hasWarned = true
}
return Promise.resolve(undefined)
}
export const getMostRecentQuantitySample = UnavailableFnFromModule(
'getMostRecentQuantitySample',
// biome-ignore lint/suspicious/noExplicitAny: it works
Promise.resolve(undefined as any as QuantitySample),
)
export const getMostRecentWorkout = UnavailableFnFromModule(
'getMostRecentWorkout',
// biome-ignore lint/suspicious/noExplicitAny: it works
Promise.resolve(undefined as any as WorkoutProxy),
)
export const getPreferredUnit = UnavailableFnFromModule(
'getPreferredUnit',
Promise.resolve('count'),
) // Defaulting to 'count'
// Hooks (from original export list)
export function useMostRecentCategorySample<T extends CategoryTypeIdentifier>(
_categoryTypeIdentifier: T,
): CategorySampleTyped<T> | undefined {
if (Platform.OS !== 'ios' && !hasWarned) {
console.warn(notAvailableError)
hasWarned = true
}
return undefined
}
export const useMostRecentQuantitySample = UnavailableFnFromModule(
'useMostRecentQuantitySample',
undefined,
)
export const useMostRecentWorkout = UnavailableFnFromModule(
'useMostRecentWorkout',
undefined,
)
export const useSubscribeToChanges = UnavailableFnFromModule(
'useSubscribeToChanges',
undefined,
) // Mocking callback structure
export const useHealthkitAuthorization = UnavailableFnFromModule(
'useHealthkitAuthorization',
[
AuthorizationRequestStatus.unknown,
() => Promise.resolve(AuthorizationRequestStatus.unknown),
] as const,
)
export const useIsHealthDataAvailable = UnavailableFnFromModule(
'useIsHealthDataAvailable',
false,
)
export const useSources = UnavailableFnFromModule('useSources', null)
export const useStatisticsForQuantity = UnavailableFnFromModule(
'useStatisticsForQuantity',
null,
)
export const getBiologicalSexAsync = UnavailableFnFromModule(
'getBiologicalSexAsync',
Promise.resolve(BiologicalSex.notSet),
)
export const getBloodTypeAsync = UnavailableFnFromModule(
'getBloodTypeAsync',
Promise.resolve(BloodType.notSet),
)
export const getDateOfBirthAsync = UnavailableFnFromModule(
'getDateOfBirthAsync',
Promise.resolve(new Date(0)),
) // Assuming string for date
export const getFitzpatrickSkinTypeAsync = UnavailableFnFromModule(
'getFitzpatrickSkinTypeAsync',
Promise.resolve(FitzpatrickSkinType.notSet),
)
export const getWheelchairUseAsync = UnavailableFnFromModule(
'getWheelchairUseAsync',
Promise.resolve(WheelchairUse.notSet),
)
export const unsubscribeQueries = UnavailableFnFromModule(
'unsubscribeQueries',
0,
)
// --- Default Export ---
// This attempts to match the structure of the default export from index.ios.ts
const HealthkitModule = {
// All named exports are also part of the default export object
authorizationStatusFor,
isObjectTypeAvailable,
unsubscribeQueries,
isObjectTypeAvailableAsync,
areObjectTypesAvailable,
areObjectTypesAvailableAsync,
isQuantityCompatibleWithUnit,
disableAllBackgroundDelivery,
disableBackgroundDelivery,
enableBackgroundDelivery,
getBiologicalSex,
getBloodType,
getDateOfBirth,
getFitzpatrickSkinType,
getMostRecentCategorySample,
getMostRecentQuantitySample,
getMostRecentWorkout,
getPreferredUnits,
getPreferredUnit,
getRequestStatusForAuthorization,
getWheelchairUse,
isHealthDataAvailable,
isHealthDataAvailableAsync,
queryCategorySamples,
queryCategorySamplesWithAnchor,
queryCorrelationSamples,
queryHeartbeatSeriesSamples,
queryHeartbeatSeriesSamplesWithAnchor,
queryQuantitySamples,
queryQuantitySamplesWithAnchor,
queryStatisticsForQuantity,
queryStatisticsCollectionForQuantity,
queryWorkoutSamples,
queryWorkoutSamplesWithAnchor,
querySources,
requestAuthorization,
deleteObjects,
saveCategorySample,
saveCorrelationSample,
saveQuantitySample,
saveWorkoutSample,
subscribeToChanges,
startWatchApp,
isProtectedDataAvailable,
queryStateOfMindSamples,
saveStateOfMindSample,
// Hooks
useMostRecentCategorySample,
useMostRecentQuantitySample,
useMostRecentWorkout,
useSubscribeToChanges,
useHealthkitAuthorization,
useIsHealthDataAvailable,
useSources,
useStatisticsForQuantity,
getBiologicalSexAsync,
getBloodTypeAsync,
getDateOfBirthAsync,
getFitzpatrickSkinTypeAsync,
getWheelchairUseAsync,
} as Omit<typeof ReactNativeHealthkit, 'default'>
export default {
...HealthkitModule,
default: HealthkitModule,
} as typeof ReactNativeHealthkit