UNPKG

create-expo-cljs-app

Version:

Create a react native application with Expo and Shadow-CLJS!

159 lines (137 loc) 3.93 kB
import React from 'react'; import { View, findNodeHandle } from 'react-native'; import ReanimatedModule from '../ReanimatedModule'; const TransitioningContext = React.createContext(); function configFromProps(type, props) { const config = { type }; if ('durationMs' in props) { config.durationMs = props.durationMs; } if ('interpolation' in props) { config.interpolation = props.interpolation; } if ('type' in props) { config.animation = props.type; } if ('delayMs' in props) { config.delayMs = props.delayMs; } if ('propagation' in props) { config.propagation = props.propagation; } return config; } /** * The below wrapper is used to support legacy context API with Context.Consumer * render prop. We need it as we want to access `context` from within * `componentDidMount` callback. If we decided to drop support for older * react native we could rewrite it using hooks or `static contextType` API. */ function wrapTransitioningContext(Comp) { return (props) => { return ( <TransitioningContext.Consumer> {(context) => <Comp context={context} {...props} />} </TransitioningContext.Consumer> ); }; } class In extends React.Component { componentDidMount() { this.props.context.push(configFromProps('in', this.props)); } render() { return this.props.children || null; } } class Change extends React.Component { componentDidMount() { this.props.context.push(configFromProps('change', this.props)); } render() { return this.props.children || null; } } class Out extends React.Component { componentDidMount() { this.props.context.push(configFromProps('out', this.props)); } render() { return this.props.children || null; } } class Together extends React.Component { transitions = []; componentDidMount() { const config = configFromProps('group', this.props); config.transitions = this.transitions; this.props.context.push(config); } render() { return ( <TransitioningContext.Provider value={this.transitions}> {this.props.children} </TransitioningContext.Provider> ); } } class Sequence extends React.Component { transitions = []; componentDidMount() { const config = configFromProps('group', this.props); config.sequence = true; config.transitions = this.transitions; this.props.context.push(config); } render() { return ( <TransitioningContext.Provider value={this.transitions}> {this.props.children} </TransitioningContext.Provider> ); } } function createTransitioningComponent(Component) { class Wrapped extends React.Component { propTypes = Component.propTypes; transitions = []; viewRef = React.createRef(); componentDidMount() { if (this.props.animateMount) { this.animateNextTransition(); } } setNativeProps(props) { this.viewRef.current.setNativeProps(props); } animateNextTransition() { const viewTag = findNodeHandle(this.viewRef.current); ReanimatedModule.animateNextTransition(viewTag, { transitions: this.transitions, }); } render() { const { transition, ...rest } = this.props; return ( <React.Fragment> <TransitioningContext.Provider value={this.transitions}> {transition} </TransitioningContext.Provider> <Component {...rest} ref={this.viewRef} collapsable={false} /> </React.Fragment> ); } } return Wrapped; } const Transitioning = { View: createTransitioningComponent(View), }; const Transition = { Sequence: wrapTransitioningContext(Sequence), Together: wrapTransitioningContext(Together), In: wrapTransitioningContext(In), Out: wrapTransitioningContext(Out), Change: wrapTransitioningContext(Change), }; export { Transitioning, Transition, createTransitioningComponent };