react-native--07Navigator

这一章我参考的资源很多,官网的navigator、navigation、UIExplorer的navigatorExample,因此会涉及到非常多的es6语法,尽量一一记录下来。

最小化代码
import React, {Component} from 'react';
import SplashScreen from 'react-native-splash-screen';
import {AppRegistry, Text, Navigator} from 'react-native';
const sleep = require('es6-sleep').promise;

export default class NavAllDay extends Component {
    async sleep(milliansecond) {
        await sleep(milliansecond);
    }
    async componentDidMount() {
        // do anything while splash screen keeps, use await to wait for an async task.
        await sleep(2000);
        await SplashScreen.hide();
    }
    render() {
        return (<Navigator initialRoute={{ title: 'Awesome Scene', index: 0 }}
                           renderScene={(route, navigator) => <Text>Hello {route.title}!</Text> }
                           style={{padding: 100}}/> );
    }
}
AppRegistry.registerComponent('rn_practice', () => NavAllDay);

Navigator参数:
initialRoute: 类似MVC框架中的urls这一层,表明当前这个Navigator有多少个路径,每个路径表示一个可切换的场景。
renderScene: 类似MVC框架中的template这一层,每次触发场景切换时将定义的变量传递过来进行重新渲染。

备注: 上面这段代码不会有任何很直观的效果,因为只有一个route,没有切换,因此感觉不出来它的特性,下面逐量去增加代码。

两条路由
import React, {Component} from 'react';
import SplashScreen from 'react-native-splash-screen';
import {AppRegistry, Text, Navigator, TouchableHighlight} from 'react-native';
const sleep = require('es6-sleep').promise;
export default class NavAllDay extends Component {
    async sleep(milliansecond) {
        await sleep(milliansecond);
    }

    async componentDidMount() {
        // do anything while splash screen keeps, use await to wait for an async task.
        await sleep(2000);
        await SplashScreen.hide();
    }

    render() {
        const routes = [
            {title: 'First Scene', index: 0}, 
            {title: 'Second Scene', index: 1}
        ];
        return (<Navigator
                    initialRoute={routes[0]}
                    renderScene={
                        (route, navigator) =>
                        <Text onPress={() => {
                            if (route.index == 0) {
                                navigator.push(routes[route.index+1])
                            } else {
                                navigator.pop()
                            }}
                        }>
                        Hello {route.title} !
                        </Text>
                    }
                 />
        );
    }
}
AppRegistry.registerComponent('rn_practice', () => NavAllDay);

备注: 这里利用了Text组件的onPress来触发状态的切换,也就是说当我点击这个文本内容时,就会切换到另外一个文本。

多条路由
import React, {Component, PropTypes} from 'react';
import SplashScreen from 'react-native-splash-screen';
import {View, Text, TouchableHighlight, AppRegistry, Navigator} from 'react-native';
const sleep = require('es6-sleep').promise;
export default class MyScene extends Component {
    render() {
        return (
            <View>
                <Text>Current Scene: {this.props.title}</Text>
                <Text onPress={this.props.onForward}>Tap me to load the next scene</Text>
                <Text onPress={this.props.onBack}>Tap me to go back</Text>
            </View> )
    }
}
MyScene.propTypes = {
    title: PropTypes.string.isRequired,
    onForward: PropTypes.func.isRequired,
    onBack: PropTypes.func.isRequired,
};

class SimpleNavigationApp extends Component {
    async sleep(milliansecond) {
        await sleep(milliansecond);
    }

    async componentDidMount() {
        // do anything while splash screen keeps, use await to wait for an async task.
        await sleep(2000);
        await SplashScreen.hide();
    }
    render() {
        return (
            <Navigator
                initialRoute={{ title: 'My Initial Scene', index: 0 }}
                renderScene={(route, navigator) =>
                    <MyScene title={route.title}
                              onForward={() => {
                                  const nextIndex = route.index + 1;
                                  navigator.push({ title: 'Scene ' + nextIndex, index: nextIndex, }); }}
                              onBack={() => {
                                  if (route.index > 0) { navigator.pop(); }
                              }}
                     />
                }
            />
        )
    }
}
AppRegistry.registerComponent('rn_practice', () => SimpleNavigationApp);

propTypes是用来声明当前对象所需要的参数类型和是否必填,一般用于开发调试(注释掉不影像任何功能),线上产品不要使用propTypes因为它会影像性能和提出警告,官方详细说明
MyScene 这个对象内容很简单,直接就返回一个View里面包裹这三个Text,在react中,它会将MyScene当作一个组件对象,可以直接用在JSX中,例如<MyScene />。
备注: 在这个例子中,并没有提前定义总共有多少个routes,只定义了initialRoute的数据格式,然后所有的操作都在renderScene里面去动态的控制,也就是说通过这种方式可以无限点击和无限操作。

装饰(宽和高)--多条路由
import React, {Component, PropTypes} from 'react';
import SplashScreen from 'react-native-splash-screen';
import {View, Text, TouchableHighlight, AppRegistry, Navigator, StyleSheet} from 'react-native';
const sleep = require('es6-sleep').promise;
export default class MyScene extends Component {
    render() {
        return (
            <View>
                <View style={styles.row_title}>
                    <Text>Current Scene: {this.props.title}</Text>
                </View>
                <View style={styles.row_items}>
                    <Text onPress={this.props.onForward}>Tap me to load the next scene</Text>
                </View>
                <View style={styles.row_items}>
                    <Text onPress={this.props.onBack}>Tap me to go back</Text>
                </View>
            </View> )
    }
}

class SimpleNavigationApp extends Component {
    async sleep(milliansecond) {
        await sleep(milliansecond);
    }

    async componentDidMount() {
        // do anything while splash screen keeps, use await to wait for an async task.
        await sleep(2000);
        await SplashScreen.hide();
    }
    render() {
        return (
            <Navigator
                initialRoute={{ title: 'My Initial Scene', index: 0 }}
                renderScene={(route, navigator) =>
                    <MyScene title={route.title}
                              onForward={() => {
                                  const nextIndex = route.index + 1;
                                  navigator.push({ title: 'Scene ' + nextIndex, index: nextIndex, }); }}
                              onBack={() => {
                                  if (route.index > 0) { navigator.pop(); }
                              }}
                     />
                }
                configureScene={(route, navigator) => Navigator.SceneConfigs.HorizontalSwipeJump}
            />
        )
    }
}

const styles = StyleSheet.create({
    row_items: {
        padding: 20,
        borderBottomWidth: 1 / 3,
        borderBottomColor:'#CDCDCD',
    },

    row_title: {
        height: 50,
        borderBottomWidth: 1 / 3,
        borderBottomColor:'#CDCDCD',
        justifyContent: 'center',
        alignItems: 'center'
    }
});
AppRegistry.registerComponent('rn_practice', () => SimpleNavigationApp);

StyleSheet 加点样式跟没有样式,确实不一样哈,好看多了。
configureScene 这是Navigator组件其中的一个参数,它的用途是配置切换的效果,这里列出了可配置范围。

  • Navigator.SceneConfigs.PushFromRight (default)
  • Navigator.SceneConfigs.FloatFromRight
  • Navigator.SceneConfigs.FloatFromLeft
  • Navigator.SceneConfigs.FloatFromBottom
  • Navigator.SceneConfigs.FloatFromBottomAndroid
  • Navigator.SceneConfigs.FadeAndroid
  • Navigator.SceneConfigs.SwipeFromLeft
  • Navigator.SceneConfigs.==HorizontalSwipeJump== # 这个牛逼, 采用这个配置之后,在屏幕任意地方左划或又划分别标识前进和后退,但是前提是要存在多个routes,否则划不过去。
  • Navigator.SceneConfigs.HorizontalSwipeJumpFromRight
  • Navigator.SceneConfigs.HorizontalSwipeJumpFromLeft
  • Navigator.SceneConfigs.VerticalUpSwipeJump
  • Navigator.SceneConfigs.VerticalDownSwipeJump
装饰(material circle)--多条路由
import React, {Component, PropTypes} from 'react';
import SplashScreen from 'react-native-splash-screen';
import {View, Text, TouchableHighlight, AppRegistry, Navigator, StyleSheet, TouchableNativeFeedback} from 'react-native';
const sleep = require('es6-sleep').promise;

export default class MyScene extends Component {
    render() {
        return (
            <View>
                <View style={styles.row_title}>
                    <Text>Current Scene: {this.props.title}</Text>
                </View>
                <TouchableNativeFeedback
                    background={TouchableNativeFeedback.SelectableBackground()}
                    onPress={this.props.onForward}>
                    <View style={styles.row_items}>
                        <Text>
                            Enabled TouchableNativeFeedback
                        </Text>
                    </View>
                </TouchableNativeFeedback>

                <TouchableNativeFeedback
                    background={TouchableNativeFeedback.SelectableBackground()}
                    onPress={this.props.onBack}>
                    <View style={styles.row_items}>
                        <Text>Tap me to go back</Text>
                    </View>
                </TouchableNativeFeedback>
            </View> )
    }
}

class SimpleNavigationApp extends Component {
    async sleep(milliansecond) {
        await sleep(milliansecond);
    }

    async componentDidMount() {
        // do anything while splash screen keeps, use await to wait for an async task.
        await sleep(2000);
        await SplashScreen.hide();
    }
    render() {
        return (
            <Navigator
                initialRoute={{ title: 'My Initial Scene', index: 0 }}
                renderScene={(route, navigator) =>
                    <MyScene title={route.title}
                              onForward={() => {
                                  const nextIndex = route.index + 1;
                                  navigator.push({ title: 'Scene ' + nextIndex, index: nextIndex, }); }}
                              onBack={() => {
                                  if (route.index > 0) { navigator.pop(); }
                              }}
                     />
                }
                configureScene={(route, navigator) => Navigator.SceneConfigs.HorizontalSwipeJump}
            />
        )
    }
}

const styles = StyleSheet.create({
    row_items: {
        padding: 20,
        borderBottomWidth: 1 / 3,
        borderBottomColor:'#CDCDCD',
    },

    row_title: {
        height: 50,
        borderBottomWidth: 1 / 3,
        borderBottomColor:'#CDCDCD',
        justifyContent: 'center',
        alignItems: 'center'
    },
});
AppRegistry.registerComponent('rn_practice', () => SimpleNavigationApp);

腾讯新闻APP中的新闻列表,当我长按某一条新闻列表是,会出现一个底色为灰色的圆圈慢慢扩大一直灰色到把整行给覆盖掉,这个特性采用的是TouchableNativeFeedback,是google的material设计风格。
注意: 我在调试这个功能,代码已经写好了但是一直都没有出现NativeFeedback这个特效,反复折腾了好久才发现原来是我在Text组件中用了onPress,导致与TouchableNativeFeedback相互冲突了,所以我把Text中的onPress移除掉,将相关的状态切换代码写到TouchableNativeFeedback组件的onPress中。

两个navigator

一个是导航,另外一个是列表

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,293评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,604评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,958评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,729评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,719评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,630评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,000评论 3 397
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,665评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,909评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,646评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,726评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,400评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,986评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,959评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,996评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,481评论 2 342

推荐阅读更多精彩内容