import React, { ComponentRef, Ref, useEffect, useMemo, useRef } from 'react'; import { Platform, Text as RNText, TextProps as RNTextProps, } from 'react-native'; import { GestureObjects as Gesture } from '../handlers/gestures/gestureObjects'; import { GestureDetector } from '../handlers/gestures/GestureDetector'; type TextProps = RNTextProps & { ref?: Ref | null>; }; type RNGHTextRef = Ref & { rngh?: boolean }; export const Text = (props: TextProps) => { const { onPress, onLongPress, ref, ...rest } = props; const textRef = useRef(null); const native = useMemo(() => Gesture.Native().runOnJS(true), []); const refHandler = useMemo(() => { const handler: RNGHTextRef = (node: RNText | null) => { textRef.current = node; if (!ref) { return; } if (typeof ref === 'function') { ref(node); } else { ref.current = node; } }; // This is a special case for `Text` component. After https://github.com/software-mansion/react-native-gesture-handler/pull/3379 we check for // `displayName` field. However, `Text` from RN has this field set to `Text`, but is also present in `RNSVGElements` set. // We don't want to treat our `Text` as the one from `SVG`, therefore we add special field to ref. handler.rngh = true; return handler; }, [ref]); useEffect(() => { if (Platform.OS !== 'web') { return; } // At this point we are sure that textElement is div in HTML tree (textRef.current as unknown as HTMLDivElement)?.setAttribute( 'rnghtext', 'true' ); }, []); return onPress || onLongPress ? ( ) : ( ); }; // eslint-disable-next-line @typescript-eslint/no-redeclare export type Text = typeof Text & RNText;