1

我试图结合使用 React Native Navbar 和 React Native Router Flux 来简化我的 RN 应用程序的制作。在这样做的过程中,我已经到了通过React Native Router Flux切换路由的地步,直到我到达主页之后,我需要再次使用 Navigator。

原因是我认为用 React Native Router Flux创建这样的东西是不可能的。我想要做的是创建一个从登录/注册页面到主页提要页面的简单过渡,这些页面之间通过侧边菜单导航。当然,您不想在每个组件上创建不同的侧边菜单,所以我首先模仿了 React Native Starter App Repo 中所做的事情。但是当我尝试从登录屏幕转换到主屏幕(本质上是 React Native Starter App 代码)时,我收到以下错误:

在此处输入图像描述

我的主页代码如下(同样来自React Native Starter App):

//user home feed with personalized news based
//on onboarding experience 
'use strict';

//component variables 
var React = require('react-native');
var { Component, StyleSheet, Navigator, Text, View } = React;
var EventEmitter = require('EventEmitter');
var Subscribable = require('Subscribable');
//libraries 
var NavigationBar = require('react-native-navbar');
var SideMenu = require('react-native-side-menu');
// App Globals
var AppStyles = require('../styles/styles');
// Components
var Icons = require('../components/icons');
var Menu = require('../components/menu');
// Screens / Pages
var Index = require('../screens/soon');


/* ==============================
  Main Navigator with Sidemenu
  =============================== */

module.exports = React.createClass({
    mixins: [Subscribable.Mixin],
    //before application load
    getInitialState: function() {
      return {
        touchToClose: true,
        disableGestures: false,
      };
    },
    //on application load
    componentWillMount: function() {
      this.eventEmitter = new EventEmitter();
    },
    //When Back Button from NavBar is Clicked
    onLeftBackButtonPress: function(navigator) {
      this.refs.rootNavigator.pop();
    },
    //When Hamburger from NavBar is Clicked
    onLeftButtonPress: function() {
      this.eventEmitter.emit('toggleMenu');
    },
    //Navigates to page from menu
    navigate: function(title, link) {
      this.refs.rootSidebarMenu.closeMenu();

      this.refs.rootNavigator.replace({
        title: title,
        component: link,
      });
    },
    //Generate Custom Navbar
    renderScene: function(route, navigator) {
      var Component = route.component;
      var navBar = route.navigationBar;

      // Icons
      var MenuIcon = Icons.MenuIcon;
      var BackIcon = Icons.BackIcon;

      // Navbar Setup
      if (navBar) {
        navBar = React.addons.cloneWithProps(navBar, {
          navigator: navigator,
          route: route
        });
      }


      // Determine which Icon component - hamburger or back?
      var customPrev = <MenuIcon leftButtonPress={this.onLeftButtonPress} />;
      if (route.index > 0){
        var customPrev = <BackIcon leftButtonPress={this.onLeftBackButtonPress} />;
      }

      // Done
      return (
        <View style={AppStyles.container}>
          <NavigationBar
            style={AppStyles.navbar}
            customPrev={customPrev} />

          <Component navigator={navigator} route={route} />
        </View>
      );
    },

    /**
      * RENDER
      */
    render: function() {
      return (
        <SideMenu
          ref="rootSidebarMenu"
          menu={<Menu events={this.eventEmitter} navigate={this.navigate} />}
          touchToClose={this.state.touchToClose}
          disableGestures={this.state.disableGestures}>

          <Navigator
            ref="rootNavigator"
            style={[AppStyles.container, AppStyles.appContainer]}
            renderScene={this.renderScene}
            initialRoute={{
              component: Index,
              index: 0,
            }} />

        </SideMenu>
      );
    }
  });

/* ==============================
  Styles
  =============================== */
  var styles = StyleSheet.create({
  });

而推送到首页路由的代码是这样的:

//component that opens up app to login screen
var React = require('react-native');
var {
    View, 
    Text, 
    StyleSheet,
    Image,  
    TextInput,
} = React;

//additional libraries
var Parse = require('parse/react-native');
var ParseReact = require('parse-react/react-native');
var FBLoginManager = require('NativeModules').FBLoginManager;
var FBSDKCore = require('react-native-fbsdkcore');
var Actions = require('react-native-router-flux').Actions;
var { FBSDKGraphRequest, FBSDKGraphRequestManager } = FBSDKCore;

//dimensions
var Dimensions = require('Dimensions');
var window = Dimensions.get('window');

//dynamic variable components
var ImageButton = require('../components/imageButton');

module.exports = React.createClass({
    componentWillMount: function(){

    },
    getInitialState: function() {
        return {
            username: '', 
            password: '', 
            errorMessage: '',
            loadingCurrentUser: true, 
        };
    },
    render: function() {
        return (
            <View style={[styles.container]}>
                <Image 
                    style={styles.bg} 
                    source={require('../img/login_bg1_3x.png')}>
                    <View style={[styles.header, this.border('red')]} >
                        <View style={styles.headerWrapper} >
                            <Image 
                                style={[styles.login_brand]}
                                resizeMode={"contain"}
                                source={require('../img/login_brand_2.png')} />
                            <ImageButton
                                style={[styles.fb_btn]}
                                resizeMode={'contain'}
                                onPress={this.onFbLoginPress}
                                source={require('../img/fb_login.png')} />
                            <Image 
                                style={[styles.loginBar]}
                                style={[styles.loginBar]} 
                                resizeMode={'contain'}
                                source={require('../img/login_bar_3x.png')} />
                        </View>
                    </View>
                    <View style={[styles.footer, this.border('blue')]} >
                        <View style={styles.footerWrapper} >
                        <Text style={styles.error}>{this.state.errorMessage}</Text>
                            <TextInput 
                                placeholder={'Email'}
                                style={styles.input} 
                                value={this.state.username}
                                onChangeText={(text) => this.setState({username: text})} />
                            <TextInput 
                                placeholder={'Password'}
                                style={styles.input} 
                                secureTextEntry={true} 
                                value={this.state.password}
                                onChangeText={(text) => this.setState({password: text})} />
                            <ImageButton
                                style={[styles.email_btn]}
                                resizeMode={'contain'}
                                onPress={this.onEmailLoginPress}
                                source={require('../img/email_login_btn.png')} />
                            <ImageButton
                                style={[styles.email_btn]}
                                resizeMode={'contain'}
                                onPress={Actions.pop}
                                source={require('../img/create_acct_btn.png')} />
                        </View>
                    </View>
                </Image>
            </View>
        );
    }, 
    onFbLoginPress: function() {

    },
    onEmailLoginPress: function() {
        //log the user on, get eror if login information doesn't exist 
        //we need to show the user that the error occured
        Parse.User.logIn(this.state.username, this.state.password, {
              success: (user) => { 
                Actions.home();
                console.log("Successful Login!"); 
              },
              error: (data, error) => { this.setState({ errorMessage: error.message }); }
        });
    },
     //function that helps with laying out flexbox itmes 
     //takes a color argument to construct border, this is an additional 
     //style because we dont want to mess up our real styling 
     border: function(color) {
        return {
          //borderColor: color, 
          //borderWidth: 4,
        } 
     },
});

var styles = StyleSheet.create({
    container: {
        flex: 1,
        alignItems: 'stretch',
    },
    bg: {
        flex: 1,
        width: window.width, 
        height: window.height, 
    },
    header: {
        flex: 2,
    }, 
    headerWrapper: {
        flex: 1, 
        flexDirection: 'column', 
        alignItems: 'center',
        justifyContent:'space-around',
        marginTop: window.height/35,
    }, 
    footerWrapper: {
        flexDirection: 'column', 
        alignItems: 'center',
        justifyContent:'space-around',
        marginTop: 15, 
    }, 
    footer: {
        flex: 4, 
    }, 
    loginBar: {
        width: (window.width/1.3)/1.8, 
        height: (70/553)*((window.width/1.3)/1.8),
    }, 
    fb_btn: {
        width: window.width/1.3,
        height: (147/1095)*window.width/1.3,
        margin: 10, 
    }, 
    login_brand: {
        width: window.width/6,
        height: (268/273)*window.width/6,
        margin: 6,
    },
    input: {
        padding: 4, //gives us offset to border 
        height: window.height/20, 
        backgroundColor: 'rgba(255,255,255, 0.4)', 
        borderColor: 'gray', 
        borderWidth: 1, 
        borderRadius: 2, //round input box
        margin: 2, 
        width: window.width/1.3,
        alignSelf: 'center', //center yourself on form when you have fixed widths 
    }, 
    email_btn: {
        width: window.width/1.3,
        height: (147/1095)*window.width/1.3,
        margin: 3, 
    }, 
    error: {
        alignItems: 'center', 
        alignSelf:'center',
        fontFamily: 'Bebas Neue', 
        fontSize: 15,
        color:'red',
    },
});

最后,我的应用在 main 中的路由结构如下所示:

//routing component to connect
//android and ios to same build
'use strict';

//component variables 
var React = require('react-native');
var { Text, View, StyleSheet, Navigator} = React;
var Launch = require('./src/screens/launch');
var Signup = require('./src/screens/signup');
var Signin = require('./src/screens/signin');
var Introduction = require('./src/screens/introduction');
var Home = require('./src/screens/home');
//dynamic libraries 
var SplashScreen = require('@remobile/react-native-splashscreen');
var Parse = require('parse/react-native');
var {Router, Route, Schema, Animations, TabBar} = require('react-native-router-flux');

module.exports = React.createClass({
  componentWillMount: function() {
    //executed when component shows on screen
    //tells app to initialize parse and facebook js sdk
    Parse.initialize("???", "???");
  }, 
  componentDidMount: function() {
    SplashScreen.hide();
  },
  render: function() {
    return (
      <Router hideNavBar={true} initialRoutes={['launch']}>
          <Schema name="bottom" sceneConfig={Navigator.SceneConfigs.FloatFromBottom}/>
          <Schema name="right" sceneConfig={Navigator.SceneConfigs.FloatFromRight}/>
          <Schema name="withoutAnimation"/>

          <Route name="launch" component={Launch} wrapRouter={true} title="Launch" hideNavBar={true} schema="right"/>
          <Route name="signin" component={Signin} title="Signin" schema="right"/>
          <Route name="signup" component={Signup} title="Signup" schema="right"/>

          <Route name="introduction" component={Introduction} title="Introduction" schema="right"/>

          <Route name="home" component={Home} title="Home" schema="right"/>


      </Router>
    );
  }
});
4

0 回答 0