import React, { forwardRef } from 'react'; import { Animated } from 'react-native'; import type { SupportedTransitions, TransitionConfig, TransitionProps } from './types'; const transformStylesMap = { translateY: true, translateX: true, scale: true, scaleX: true, scaleY: true, rotate: true, }; const defaultStyles = { opacity: 1, translateY: 0, translateX: 0, scale: 1, scaleX: 1, scaleY: 1, rotate: '0deg', }; const getAnimatedStyles = (animateValue: Animated.Value) => (initial: SupportedTransitions, to: SupportedTransitions) => { const styles: any = { transform: [], }; const keys = Object.keys(initial); keys.forEach(key => { if (key !== 'transition') { if (key in transformStylesMap) { styles.transform?.push({ [key]: animateValue.interpolate({ inputRange: [0, 1], outputRange: [(initial as any)[key], (to as any)[key]], }), } as any); } else { styles[key] = animateValue.interpolate({ inputRange: [0, 1], outputRange: [(initial as any)[key], (to as any)[key]], }); } } }); return styles; }; const defaultTransitionConfig: TransitionConfig = { type: 'timing', useNativeDriver: true, duration: 250, delay: 0, }; const Transition = forwardRef((props, ref) => { const { children, onTransitionComplete, visible = false, initial, animate, exit, style, as, ...rest } = props; const animateValue = React.useRef(new Animated.Value(0)).current; const Component = React.useMemo(() => { if (as) { return Animated.createAnimatedComponent(as); } return Animated.View; }, [as]); const [animationState, setAnimationState] = React.useState(''); const prevVisible = React.useRef(visible); React.useEffect(() => { if (animationState === 'entering' || animationState === 'exiting') { const entryTransition = { ...defaultTransitionConfig, ...animate?.transition, }; const exitTransition = { ...defaultTransitionConfig, ...exit?.transition, }; const startAnimation = animationState === 'entering' ? 1 : 0; const transition = startAnimation ? entryTransition : exitTransition; Animated.sequence([ Animated.delay(transition.delay ?? 0), Animated[transition.type ?? 'timing'](animateValue, { toValue: startAnimation, useNativeDriver: true, ...transition, }), ]).start(() => { if (animationState === 'entering') { setAnimationState('entered'); } else if (animationState === 'exiting') { setAnimationState('exited'); } }); } if (animationState === 'exited') { onTransitionComplete && onTransitionComplete('exited'); } else if (animationState === 'entered') { onTransitionComplete && onTransitionComplete('entered'); } }, [animationState, onTransitionComplete]); React.useEffect(() => { if (prevVisible.current !== visible && !visible) { setAnimationState('exiting'); } if (visible) { setAnimationState('entering'); } prevVisible.current = visible; }, [visible]); const initialState = animationState === 'exited' && exit ? { ...defaultStyles, ...exit } : { ...defaultStyles, ...initial }; const animateState = { ...defaultStyles, ...animate }; const styles = React.useMemo(() => { return [ getAnimatedStyles(animateValue)( initialState as SupportedTransitions, animateState as SupportedTransitions ), style, ]; }, [animateValue, initial, animate, style]); return ( {children} ); }); export default Transition;