react-native-reanimated
Version:
More powerful alternative to Animated library for React Native.
69 lines (61 loc) • 2.53 kB
text/typescript
import { useRef } from 'react';
import { WorkletEventHandler } from '../WorkletEventHandler';
import type { IWorkletEventHandler, ReanimatedEvent } from './commonTypes';
/** Worklet to provide as an argument to `useEvent` hook. */
export type EventHandler<
Event extends object,
Context extends Record<string, unknown> = never,
> = (event: ReanimatedEvent<Event>, context?: Context) => void;
export type EventHandlerProcessed<
Event extends object,
Context extends Record<string, unknown> = never,
> = (event: Event, context?: Context) => void;
export type EventHandlerInternal<Event extends object> = {
workletEventHandler: IWorkletEventHandler<Event>;
};
/**
* Lets you run a function whenever a specified native event occurs.
*
* @param handler - A function that receives an event object with event data -
* {@link EventHandler}.
* @param eventNames - An array of event names the `handler` callback will react
* to.
* @param rebuild - Whether the event handler should be rebuilt. Defaults to
* `false`.
* @returns A function that will be called when the event occurs -
* {@link EventHandlerProcessed}.
* @see https://docs.swmansion.com/react-native-reanimated/docs/advanced/useEvent
*/
// @ts-expect-error This overload is required by our API.
// We don't know which properites of a component that is made into
// an AnimatedComponent are event handlers and we don't want to force the user to define it.
// Therefore we disguise `useEvent` return type as a simple function and we handle
// it being a React Ref in `createAnimatedComponent`.
export function useEvent<
Event extends object,
Context extends Record<string, unknown> = never,
>(
handler: EventHandler<Event, Context>,
eventNames?: string[],
rebuild?: boolean
): EventHandlerProcessed<Event, Context>;
export function useEvent<Event extends object, Context = never>(
handler: (event: ReanimatedEvent<Event>, context?: Context) => void,
eventNames: string[] = [],
rebuild = false
): EventHandlerInternal<Event> {
const initRef = useRef<EventHandlerInternal<Event>>(null!);
if (initRef.current === null) {
const workletEventHandler = new WorkletEventHandler<Event>(
handler,
eventNames
);
initRef.current = { workletEventHandler };
} else if (rebuild) {
const workletEventHandler = initRef.current.workletEventHandler;
workletEventHandler.updateEventHandler(handler, eventNames);
initRef.current = { workletEventHandler };
}
return initRef.current;
}
;