RN实战项目页面导航和启动页面实现(三)

RN中处理页面切换可以使用导航器来实现,本项目中用的是Navigator,而官方已经不推荐使用了,而采用新的react-navigation库来替代。
直接看代码
/**
 * Sample React Native App
 * https://github.com/facebook/react-native
 * @flow
 */

import React, { Component } from 'react';
import {
  AppRegistry,
  Navigator,
} from 'react-native';
import Splash from './views/Splash';

export default class Index extends Component {

  constructor(props) {
    super(props);
  }

  render() {
    return (
      <Navigator
        initialRoute={{ name: '欢迎界面', component: Splash }}
        configureScene={() => Navigator.SceneConfigs.FloatFromBottom}
        renderScene={(route, navigator) => {
          //返回按钮
          navigator.goBack = function () {
            navigator.pop();
          }
          //退出整个app
          navigator.exitApp = function () {
            let routes = navigator.getCurrentRoutes();
            for (var i = 0; i < routes.length; i++) {
              navigator.pop();
            }
          }
          return (
            <route.component {...route} navigator={navigator}></route.component>
          );
        }}
      ></Navigator>
    );
  }
}

AppRegistry.registerComponent('react_test', () => Index);
initialRoute初始化一个route,我们看代码得知,初始化的时候给了一个对象作为参数,{ name: ‘欢迎界面’, component: Splash },Splash即为默认的第一个界面
configureScene配置信息,页面进入退出的动画效果等
renderScene渲染一个组件,这里需要返回一个组件
{…route} es6 写法,表示把当前routes身上的所有属性赋值给当前组件,第一次执行的时候 我们在Splash组件里面即可通过this.props.name获取到 ‘欢迎界面’
navigator={navigator} 在当前组件通过this.props.navigator 获取到navigator对象
//返回按钮 为navigator新增加一个方法 goBack,使用的时候通过this.props.navigator.goBack();即可返回到上一个界面
navigator.goBack = function () {
     navigator.pop();
}
//退出整个app, pop掉所有的界面
navigator.exitApp = function () {
   let routes = navigator.getCurrentRoutes();
   for (var i = 0; i < routes.length; i++) {
       navigator.pop();
   }
}
来到Splash
import React , { Component } from 'react';
import {
    View,
    Image, 
    BackAndroid,
    Platform,
} from 'react-native';
import Login from './Login';

/**
 * 启动界面
 */

let navigator;

export default class Splash extends Component{

    constructor(props){
        super(props);
        navigator = this.props.navigator;
    }


    componentWillMount() {
        if (Platform.OS === 'android') {
            BackAndroid.addEventListener('hardwareBackPress', this.onBackAndroid);
        }
    }
    componentWillUnMount() {
        if (Platform.OS === 'android') {
            BackAndroid.removeEventListener('hardwareBackPress', this.onBackAndroid);
        }
    }

    onBackAndroid() {
        if (navigator) {
            let routes = navigator.getCurrentRoutes();
            if (routes.length > 3) {
                navigator.pop();
                return true;
            }
        }
        return false;
    }

    componentDidMount(){
        setTimeout(() => {
            const navigator = this.props.navigator;
            if(navigator){
                navigator.push({
                    name:'登录界面',
                    component:Login,
                    params:{
                        data:'login test',
                    }
                });
            }
        },2000);
    }

    render(){
        return (
            <View style={{flex:1}}>
                <Image source={require('../img/ic_splash.png')}></Image>
            </View>
        );
    }
}
启动界面很简单,就是显示一个图片,然后在组件加载完成2秒之后中去到登录界面
重点说下BackAndroid,安卓手机中存在物理返回按键,而苹果手机中并不存在,所以这里需要我们针对安卓平台特殊处理,在组件将要挂载和将要卸载分别注册和注销监听器,BackAndroid.addEventListener(‘hardwareBackPress’, this.onBackAndroid);BackAndroid.removeEventListener(‘hardwareBackPress’, this.onBackAndroid);
onBackAndroid() { 
    //这里遇到的坑,刚开始navigator 是在这里获取的 也就是navigator = this.props.navigator;
    //但是这里获取不到,因为这里的this 指的不是这个组件 ,后来在构造器中获取到,并设置为全局变量使用即可
    if (navigator) {
    //获取到当前路由的长度,因为我的页面逻辑是这样的 启动页-登录夜-主页
        let routes = navigator.getCurrentRoutes();
        if (routes.length > 3) {
            navigator.pop();
            return true;
        }
    }
    //false 表示直接退出app
    return false;
}
END