/** * Copyright (c) 2013-present, Facebook, Inc. * All rights reserved. * * This source code is licensed under the BSD-style license found in the * LICENSE file in the root directory of this source tree. An additional grant * of patent rights can be found in the PATENTS file in the same directory. * * Facebook, Inc. ("Facebook") owns all right, title and interest, including * all intellectual property and other proprietary rights, in and to the React * Native CustomComponents software (the "Software"). Subject to your * compliance with these terms, you are hereby granted a non-exclusive, * worldwide, royalty-free copyright license to (1) use and copy the Software; * and (2) reproduce and distribute the Software as part of your own software * ("Your Software"). Facebook reserves all rights not expressly granted to * you in this license agreement. * * THE SOFTWARE AND DOCUMENTATION, IF ANY, ARE PROVIDED "AS IS" AND ANY EXPRESS * OR IMPLIED WARRANTIES (INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE) ARE DISCLAIMED. * IN NO EVENT SHALL FACEBOOK OR ITS AFFILIATES, OFFICERS, DIRECTORS OR * EMPLOYEES BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THE SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * * @providesModule NavigationCardStack * @flow */ 'use strict'; const NativeAnimatedModule = require('NativeModules').NativeAnimatedModule; const NavigationCard = require('NavigationCard'); const NavigationCardStackPanResponder = require('NavigationCardStackPanResponder'); const NavigationCardStackStyleInterpolator = require('NavigationCardStackStyleInterpolator'); const NavigationPropTypes = require('NavigationPropTypes'); const NavigationTransitioner = require('NavigationTransitioner'); const React = require('React'); const StyleSheet = require('StyleSheet'); const View = require('View'); const {PropTypes} = React; const {Directions} = NavigationCardStackPanResponder; import type { NavigationState, NavigationSceneRenderer, NavigationSceneRendererProps, NavigationTransitionProps, NavigationStyleInterpolator, } from 'NavigationTypeDefinition'; import type { NavigationGestureDirection, } from 'NavigationCardStackPanResponder'; type Props = { direction: NavigationGestureDirection, navigationState: NavigationState, onNavigateBack?: Function, renderHeader: ?NavigationSceneRenderer, renderScene: NavigationSceneRenderer, cardStyle?: any, style: any, gestureResponseDistance?: ?number, enableGestures: ?boolean, cardStyleInterpolator?: ?NavigationStyleInterpolator, scenesStyle?: any, }; type DefaultProps = { direction: NavigationGestureDirection, enableGestures: boolean, }; /** * A controlled navigation view that renders a stack of cards. * * ```html * +------------+ * +-| Header | * +-+ |------------| * | | | | * | | | Focused | * | | | Card | * | | | | * +-+ | | * +-+ | * +------------+ * ``` * * ## Example * * ```js * * class App extends React.Component { * constructor(props, context) { * this.state = { * navigation: { * index: 0, * routes: [ * {key: 'page 1'}, * }, * }, * }; * } * * render() { * return ( * * ); * } * * _renderScene: (props) => { * return ( * * {props.scene.route.key} * * ); * }; * ``` */ class NavigationCardStack extends React.Component { _render : NavigationSceneRenderer; _renderScene : NavigationSceneRenderer; static propTypes = { /** * Custom style applied to the card. */ cardStyle: PropTypes.any, /** * Direction of the cards movement. Value could be `horizontal` or * `vertical`. Default value is `horizontal`. */ direction: PropTypes.oneOf([Directions.HORIZONTAL, Directions.VERTICAL]), /** * The distance from the edge of the card which gesture response can start * for. Default value is `30`. */ gestureResponseDistance: PropTypes.number, /** * An interpolator function that is passed an object parameter of type * NavigationSceneRendererProps and should return a style object to apply to * the transitioning navigation card. * * Default interpolator transitions translateX, scale, and opacity. */ cardStyleInterpolator: PropTypes.func, /** * Enable gestures. Default value is true. * * When disabled, transition animations will be handled natively, which * improves performance of the animation. In future iterations, gestures * will also work with native-driven animation. */ enableGestures: PropTypes.bool, /** * The controlled navigation state. Typically, the navigation state * look like this: * * ```js * const navigationState = { * index: 0, // the index of the selected route. * routes: [ // A list of routes. * {key: 'page 1'}, // The 1st route. * {key: 'page 2'}, // The second route. * ], * }; * ``` */ navigationState: NavigationPropTypes.navigationState.isRequired, /** * Callback that is called when the "back" action is performed. * This happens when the back button is pressed or the back gesture is * performed. */ onNavigateBack: PropTypes.func, /** * Function that renders the header. */ renderHeader: PropTypes.func, /** * Function that renders the a scene for a route. */ renderScene: PropTypes.func.isRequired, /** * Custom style applied to the cards stack. */ style: View.propTypes.style, /** * Custom style applied to the scenes stack. */ scenesStyle: View.propTypes.style, }; static defaultProps: DefaultProps = { direction: Directions.HORIZONTAL, enableGestures: true, }; constructor(props: Props, context: any) { super(props, context); } componentWillMount(): void { this._render = this._render.bind(this); this._renderScene = this._renderScene.bind(this); } render(): React.Element { return ( ); } _configureTransition = () => { const isVertical = this.props.direction === 'vertical'; const animationConfig = {}; if ( !!NativeAnimatedModule // Gestures do not work with the current iteration of native animation // driving. When gestures are disabled, we can drive natively. && !this.props.enableGestures // Native animation support also depends on the transforms used: && NavigationCardStackStyleInterpolator.canUseNativeDriver(isVertical) ) { animationConfig.useNativeDriver = true; } return animationConfig; } _render(props: NavigationTransitionProps): React.Element { const { renderHeader, } = this.props; const header = renderHeader ? {renderHeader(props)} : null; const scenes = props.scenes.map( scene => this._renderScene({ ...props, scene, }) ); return ( {scenes} {header} ); } _renderScene(props: NavigationSceneRendererProps): React.Element { const isVertical = this.props.direction === 'vertical'; const interpolator = this.props.cardStyleInterpolator || (isVertical ? NavigationCardStackStyleInterpolator.forVertical : NavigationCardStackStyleInterpolator.forHorizontal); const style = interpolator(props); let panHandlers = null; if (this.props.enableGestures) { const panHandlersProps = { ...props, onNavigateBack: this.props.onNavigateBack, gestureResponseDistance: this.props.gestureResponseDistance, }; panHandlers = isVertical ? NavigationCardStackPanResponder.forVertical(panHandlersProps) : NavigationCardStackPanResponder.forHorizontal(panHandlersProps); } return ( ); } } const styles = StyleSheet.create({ container: { flex: 1, // Header is physically rendered after scenes so that Header won't be // covered by the shadows of the scenes. // That said, we'd have use `flexDirection: 'column-reverse'` to move // Header above the scenes. flexDirection: 'column-reverse', }, scenes: { flex: 1, }, }); module.exports = NavigationCardStack;