/** * Copyright 2004-present Facebook. All Rights Reserved. * * @providesModule ViewPagerAndroid * @flow */ 'use strict'; var NativeMethodsMixin = require('NativeMethodsMixin'); var NativeModules = require('NativeModules'); var React = require('React'); var ReactElement = require('ReactElement'); var ReactNativeViewAttributes = require('ReactNativeViewAttributes'); var ReactPropTypes = require('ReactPropTypes'); var View = require('View'); var RCTUIManager = NativeModules.UIManager; var dismissKeyboard = require('dismissKeyboard'); var requireNativeComponent = require('requireNativeComponent'); var VIEWPAGER_REF = 'viewPager'; /** * Container that allows to flip left and right between child views. Each * child view of the `ViewPagerAndroid` will be treated as a separate page * and will be streched to fill the `ViewPagerAndroid`. * * It is important all children are ``s and not composite components. * You can set style properties like `padding` or `backgroundColor` for each * child. * * Example: * * ``` * render: function() { * return ( * * * First page * * * Second page * * * ); * } * * ... * * var styles = { * ... * pageStyle: { * alignItems: 'center', * padding: 20, * } * } * ``` */ var ViewPagerAndroid = React.createClass({ propTypes: { ...View.propTypes, /** * Index of initial page that should be selected. Use `setPage` method to * update the page, and `onPageSelected` to monitor page changes */ initialPage: ReactPropTypes.number, /** * Executed when transitioning between pages (ether because of animation for * the requested page change or when user is swiping/dragging between pages) * The `event.nativeEvent` object for this callback will carry following data: * - position - index of first page from the left that is currently visible * - offset - value from range [0,1) describing stage between page transitions. * Value x means that (1 - x) fraction of the page at "position" index is * visible, and x fraction of the next page is visible. */ onPageScroll: ReactPropTypes.func, /** * This callback will be caleld once ViewPager finish navigating to selected page * (when user swipes between pages). The `event.nativeEvent` object passed to this * callback will have following fields: * - position - index of page that has been selected */ onPageSelected: ReactPropTypes.func, /** * Determines whether the keyboard gets dismissed in response to a drag. * - 'none' (the default), drags do not dismiss the keyboard. * - 'on-drag', the keyboard is dismissed when a drag begins. */ keyboardDismissMode: ReactPropTypes.oneOf([ 'none', // default 'on-drag', ]), }, componentDidMount: function() { if (this.props.initialPage) { this.setPageWithoutAnimation(this.props.initialPage); } }, getInnerViewNode: function(): ReactComponent { return this.refs[VIEWPAGER_REF].getInnerViewNode(); }, _childrenWithOverridenStyle: function(): Array { // Override styles so that each page will fill the parent. Native component // will handle positioning of elements, so it's not important to offset // them correctly. return React.Children.map(this.props.children, function(child) { var newProps = { ...child.props, style: [child.props.style, { position: 'absolute', left: 0, top: 0, right: 0, bottom: 0, width: undefined, height: undefined, }], collapsable: false, }; if (child.type && child.type.displayName && (child.type.displayName !== 'RCTView') && (child.type.displayName !== 'View')) { console.warn('Each ViewPager child must be a . Was ' + child.type.displayName); } return ReactElement.createElement(child.type, newProps); }); }, _onPageScroll: function(e: Event) { if (this.props.onPageScroll) { this.props.onPageScroll(e); } if (this.props.keyboardDismissMode === 'on-drag') { dismissKeyboard(); } }, _onPageSelected: function(e: Event) { if (this.props.onPageSelected) { this.props.onPageSelected(e); } }, /** * A helper function to scroll to a specific page in the ViewPager. * The transition between pages will be animated. */ setPage: function(selectedPage: number) { RCTUIManager.dispatchViewManagerCommand( React.findNodeHandle(this), RCTUIManager.AndroidViewPager.Commands.setPage, [selectedPage], ); }, /** * A helper function to scroll to a specific page in the ViewPager. * The transition between pages will be *not* be animated. */ setPageWithoutAnimation: function(selectedPage: number) { RCTUIManager.dispatchViewManagerCommand( React.findNodeHandle(this), RCTUIManager.AndroidViewPager.Commands.setPageWithoutAnimation, [selectedPage], ); }, render: function() { return ( ); }, }); var NativeAndroidViewPager = requireNativeComponent('AndroidViewPager', ViewPagerAndroid); module.exports = ViewPagerAndroid;