/* eslint-disable react-hooks/exhaustive-deps */ import React, { forwardRef } from 'react'; import { Animated } from 'react-native'; import type { ISupportedTransitions, ITransitionConfig, ITransitionProps, } 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: any) => ( initial: ISupportedTransitions, to: ISupportedTransitions ) => { const styles: any = { transform: [], }; for (let key in initial) { if (key === 'transition') { continue; } 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: ITransitionConfig = { type: 'timing', useNativeDriver: true, duration: 250, delay: 0, }; export const Transition = forwardRef( ( { children, onTransitionComplete, visible = false, initial, animate, exit, style, as, ...rest }: ITransitionProps, ref: any ) => { 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([ // @ts-ignore - delay is present in defaultTransitionConfig Animated.delay(transition.delay), 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'); } // if (animationState === 'entering') { // // // } // eslint-disable-next-line react-hooks/exhaustive-deps }, [animationState, onTransitionComplete]); React.useEffect(() => { // if (!visible) { if (prevVisible.current !== visible && !visible) { setAnimationState('exiting'); } if (visible) { setAnimationState('entering'); } prevVisible.current = visible; // } }, [visible]); // If exit animation is present and state is exiting, we replace 'initial' with 'exit' animation // const initialState = { ...defaultStyles, ...initial }; const initialState = animationState === 'exited' && exit ? { ...defaultStyles, ...exit } : { ...defaultStyles, ...initial }; // const initialState = { ...defaultStyles, ...initial }; // initial = // animationState === 'exited' // ? { ...defaultStyles, ...exit } // : { ...defaultStyles, ...initial }; const animateState = { ...defaultStyles, ...animate }; // const [initialState, setInitialState] = React.useState({ // ...defaultStyles, // ...initial, // }); // console.log('Initial state ', initial); // const [animateState] = React.useState({ ...defaultStyles, ...animate }); const styles = React.useMemo(() => { // console.log('display state here', initial); return [ getAnimatedStyles(animateValue)( initialState as ISupportedTransitions, animateState as ISupportedTransitions ), style, ]; }, [animateValue, initial, animate, style]); return ( {children} ); } );