import * as React from 'react'; import { Animated, Platform, processColor, StyleSheet } from 'react-native'; import createNativeWrapper from '../handlers/createNativeWrapper'; import GestureHandlerButton from './GestureHandlerButton'; import { State } from '../State'; import { GestureEvent, HandlerStateChangeEvent, } from '../handlers/gestureHandlerCommon'; import type { NativeViewGestureHandlerPayload } from '../handlers/GestureHandlerEventPayload'; import type { BaseButtonWithRefProps, BaseButtonProps, RectButtonWithRefProps, RectButtonProps, BorderlessButtonWithRefProps, BorderlessButtonProps, } from './GestureButtonsProps'; import { isFabric } from '../utils'; export const RawButton = createNativeWrapper(GestureHandlerButton, { shouldCancelWhenOutside: false, shouldActivateOnStart: false, }); let IS_FABRIC: null | boolean = null; class InnerBaseButton extends React.Component { static defaultProps = { delayLongPress: 600, }; private lastActive: boolean; private longPressTimeout: ReturnType | undefined; private longPressDetected: boolean; constructor(props: BaseButtonWithRefProps) { super(props); this.lastActive = false; this.longPressDetected = false; } private handleEvent = ({ nativeEvent, }: HandlerStateChangeEvent) => { const { state, oldState, pointerInside } = nativeEvent; const active = pointerInside && state === State.ACTIVE; if (active !== this.lastActive && this.props.onActiveStateChange) { this.props.onActiveStateChange(active); } if ( !this.longPressDetected && oldState === State.ACTIVE && state !== State.CANCELLED && this.lastActive && this.props.onPress ) { this.props.onPress(pointerInside); } if ( !this.lastActive && // NativeViewGestureHandler sends different events based on platform state === (Platform.OS !== 'android' ? State.ACTIVE : State.BEGAN) && pointerInside ) { this.longPressDetected = false; if (this.props.onLongPress) { this.longPressTimeout = setTimeout( this.onLongPress, this.props.delayLongPress ); } } else if ( // Cancel longpress timeout if it's set and the finger moved out of the view state === State.ACTIVE && !pointerInside && this.longPressTimeout !== undefined ) { clearTimeout(this.longPressTimeout); this.longPressTimeout = undefined; } else if ( // Cancel longpress timeout if it's set and the gesture has finished this.longPressTimeout !== undefined && (state === State.END || state === State.CANCELLED || state === State.FAILED) ) { clearTimeout(this.longPressTimeout); this.longPressTimeout = undefined; } this.lastActive = active; }; private onLongPress = () => { this.longPressDetected = true; this.props.onLongPress?.(); }; // Normally, the parent would execute it's handler first, then forward the // event to listeners. However, here our handler is virtually only forwarding // events to listeners, so we reverse the order to keep the proper order of // the callbacks (from "raw" ones to "processed"). private onHandlerStateChange = ( e: HandlerStateChangeEvent ) => { this.props.onHandlerStateChange?.(e); this.handleEvent(e); }; private onGestureEvent = ( e: GestureEvent ) => { this.props.onGestureEvent?.(e); this.handleEvent( e as HandlerStateChangeEvent ); // TODO: maybe it is not correct }; render() { const { rippleColor: unprocessedRippleColor, style, ...rest } = this.props; if (IS_FABRIC === null) { IS_FABRIC = isFabric(); } const rippleColor = IS_FABRIC ? unprocessedRippleColor : processColor(unprocessedRippleColor ?? undefined); return ( ); } } const AnimatedInnerBaseButton = Animated.createAnimatedComponent(InnerBaseButton); export const BaseButton = React.forwardRef< React.ComponentType, Omit >((props, ref) => ); const AnimatedBaseButton = React.forwardRef< React.ComponentType, Animated.AnimatedProps >((props, ref) => ); const btnStyles = StyleSheet.create({ underlay: { position: 'absolute', left: 0, right: 0, bottom: 0, top: 0, }, }); class InnerRectButton extends React.Component { static defaultProps = { activeOpacity: 0.105, underlayColor: 'black', }; private opacity: Animated.Value; constructor(props: RectButtonWithRefProps) { super(props); this.opacity = new Animated.Value(0); } private onActiveStateChange = (active: boolean) => { if (Platform.OS !== 'android') { this.opacity.setValue(active ? this.props.activeOpacity! : 0); } this.props.onActiveStateChange?.(active); }; render() { const { children, style, ...rest } = this.props; const resolvedStyle = StyleSheet.flatten(style) ?? {}; return ( {children} ); } } export const RectButton = React.forwardRef< React.ComponentType, Omit >((props, ref) => ); class InnerBorderlessButton extends React.Component { static defaultProps = { activeOpacity: 0.3, borderless: true, }; private opacity: Animated.Value; constructor(props: BorderlessButtonWithRefProps) { super(props); this.opacity = new Animated.Value(1); } private onActiveStateChange = (active: boolean) => { if (Platform.OS !== 'android') { this.opacity.setValue(active ? this.props.activeOpacity! : 1); } this.props.onActiveStateChange?.(active); }; render() { const { children, style, innerRef, ...rest } = this.props; return ( {children} ); } } export const BorderlessButton = React.forwardRef< React.ComponentType, Omit >((props, ref) => ); export { default as PureNativeButton } from './GestureHandlerButton';