react-native的学习

最近突然对react-native感兴趣. 主要被它在iOS和安卓平台通用性吸引, 就花了几天的课余时间学习了一下. 也稍微有点小收获, 想跟大家分享一下. 这门技术在2015年也是火的可以, facebook推出的, 声称是 learn once, write anywhere(感觉就跟java一样, 一次书写,到处运行).hhhh, 我也比较捣鼓新技术, 加上之前开发过iOS, 有点移动端开发基础,后台也熟悉, 所以感觉学起来还算是轻松.

. 具体的环境配置就就不跟大家讲, 网上的教程还挺多, 也很详细.

如.
react-native init DemoRN
npm install --save react-navigation
react-native run-ios
等等.

. 跟大家分享是以下一些知识点:(涉及到内容不多, 后续继续补充哈)

  1. es6中export和import的介绍.
  2. 以及一些基础的(props, state的区别)
  3. es6中fetch进行网络请求的简单介绍.
  4. react-navigation, 这里主要介绍StackNavigator和TabNavigator,类似iOS中UINavigationController和TabBarController.
  5. 正向传值和逆向回调传值
  6. ListView的使用(类似iOS中UITableView,不过还是有点区别)
  7. 如何书写一些公共模块(将一些常用的功能抽取到一个Util类中)
  8. 以及一些全局的常量如何归类.
  9. 关于style样式.
  10. 生命周期函数

按照惯例啦----先看效果图.
(本来已经录好了gif给大家看的, 发现上传到简书失败,估计后台有文件大小限制, 也尝试了压缩gif, 可是文件还是有点大, 就只能截图给大家看了,不开心啊), 大家将就点吧.

Simulator Screen Shot 2017年6月22日 23.36.57.png
Simulator Screen Shot 2017年6月22日 23.37.03.png
Simulator Screen Shot 2017年6月22日 23.37.09.png
Simulator Screen Shot 2017年6月22日 23.37.11.png
Simulator Screen Shot 2017年6月22日 23.37.21.png
Simulator Screen Shot 2017年6月22日 23.37.24.png

1. es6中的import和export的简单介绍.

Home.js文件
这里是导出默认的模块. Home,
注意一个js文件中只能有一个默认的导出的模块. 
export default class Home extends Component {
    
}

index.ios.js文件
注意, 这样在使用import的时候,是可以自定义导入的默认模块的名字的 
import Home from './Home';


可能你在一些文件中会看到这样的导入方式:
import React, {Component} from 'react';
这里我解释下哈
React模块是react.js文件默认导出的文件.
{Component}注意这里加了大括号, 表示react.js文件中非默认导出
(即没有default关键修饰导出的模块), 还有注意一点的是
    非默认模块在import的时候是不可以重命名的.

2. 关于props和state的简单介绍.

RN的组件其实就是一个状态机, 主要的两个参数就是props和state, 
最后返回一个虚拟DOM, 进行渲染.(这也是它的高效原因之一).

props 一般用于父组件向子组件通信,在组件之间进行传值使用.
state 一般用于组件内部的状态维护, 跟新组件内部数据, 状态, 重新进行虚拟DOM的渲染.
注意:
1. 不管是props还是state的改变, 都会引发render函数的重新调用.
2. 都可以由组件自身提供的函数进行初始化.
3. props是一个父组件传递给子组件的数据流 getDefaultProps
4. state只能在自身组件中setState   getInitalState
5. getDefaultProps和getInitalState在es6中推荐用constructor进行初始化.

如:
  constructor(props) {
    super(props);
    let ds = new ListView.DataSource({rowHasChanged: (r1, r2) => r1 !== r2});
    this.state = {
        isLoaded: false,
        dataSource: ds
    }
}

3.fetch进行网络请求的简单介绍.

一个最简单的请求就是这样:
 fetch(url).then((response) => {
        return response.json();
    }).then((responseData) => {
        callBack(responseData[objectName]);
    }).catch((error) => {
        alert(error);
    })

当然这里你也可以指定请求方式, 请求头, 以及请求参数, (由于示例,未详细介绍, 具体还请读者自行去查哈).

4 StackNavigator的使用.

注意要导入 
import {StackNavigator, TabNavigator} from 'react-navigation';

StackNavigator是facebook最近才推出的react-navigation框架中所带的一个模块, 超级好用, 比之前的那些navigatorIOS好用多了. gitHub上的star也猛增.

示例:
const Nav2 = StackNavigator(
{
    personCenter: {screen: PersonCenter},
    movieList:{screen:MovieList},
},
{
    navigationOptions:{
        headerStyle:{
            backgroundColor:"#eec",
        }
    }
}
);

import PersonCenter from './MyApp/PersonCenterModule/PersonCenter';
import MovieList from './MyApp/PersonCenterModule/MovieList';
这里PersonCenter和MovieList是导入的另外两个模块, 前面的名称是可以自定义的.
navigationOptions的就是全局的导航栏的设置.(会影响每一个导航栏). 当然也可以在
每个页面进行单页面自定义.
如:
export default class MovieList extends Component {

static navigationOptions = {
    tabBarVisible: false,
    title: "科技新闻"
}
}

讲导航栏肯定少不了页面跳转.
导航栏提供了一些函数和属性.这里介绍常用的.

this.props.navigation.navigate('需要跳转到的页面', {参数传递, 回调函数的定义})
this.props.navigation.goBack()返回上一级
this.props.navigation.state.params 获取传递参数的值.
或许读者会对这样的写法有疑问:
const {params} = this.props.navigation.state.params, 我解释下哈
这样是相当于等价先判断this.props.navigation.state中是否有params这个属性,
如何没有程序运行时会由警告.

如:
跳转至homeSecond页面,并且将传递titleValue值. 这里homeSecond是在导航控制器
    {screen: PersonCenter}
这里指定的.
<Button title="详情" onPress={() => {
            const {navigate} = navigation;
            navigate('homeSecond', {titleValue: "哈哈"})
        }}/>

5 正向传值和逆向回调传值

先看代码咯:
export default class HomeSecond extends Component {
static navigationOptions = {
title: "HomeSecond",
tabBarVisible:false
}

constructor(props) {
    super(props);
    const {params} = props.navigation.state;
    this.state = {
        titleValue: params.titleValue
    }
}

render() {
    const {navigate} = this.props.navigation;
    return (
        <View style={styles.container}>
            <Text style={styles.text}>{this.state.titleValue}</Text>
            <Button title="继续点击"
                    onPress={() => navigate('homeThird',
                        {
                            titleValue: this.state.titleValue,
                            username: "React-Native",
                            callBack: (data) => {
                                this.setState({
                                    titleValue: data
                                });
                            }
                        })
                    }/>
        </View>
    );
}
}

这里首先初始化state, 有一个属性为titleValue, 这里是作为接收下级界面回调传回的值.

navigate('homeThird',{
    titleValue: this.state.titleValue,
    username: "React-Native",
    callBack: (data) => {
        this.setState({
            titleValue: data
        });
}

由于该界面是在导航控制器(StackNavigator)的管理下, 
所以该界面的this.props会   自动有一个navigatioin参数, 
那么就可以使用这个navigation中的navigate进行界    面跳转和参数传递了啊.
const {navigate}  = this.props.navigation;
利用该参数先进行跳转到homeThird.js页面. 并且传递titleValue和username的值, 
以及一个callback的回调函数.



下面看homeThird.js页面如何接收的.
  
  这里接收titleValue和username.
  constructor(props) {
    super(props);
    this.state = {
        textInputValue: props.navigation.state.params.titleValue,
        username:props.navigation.state.params.username
    }
}

render() {
    const {params} = this.props.navigation.state;
    return (
        <View style={styles.container}>
            <Text style={styles.text}>{params.username}</Text>
            <TextInput style={styles.textInputViewStyle}
                       placeholder="请输入"
                       onChangeText={(text) => this.setState({textInputValue: text})}/>
            <Button title="确认并返回"
                    onPress={this.clickBackBtn.bind(this)}/>
        </View>
    );
}

 clickBackBtn() {
    const {goBack} = this.props.navigation;
    const {params} = this.props.navigation.state;
    params.callBack(this.state.textInputValue);
    goBack();
}

这里先获取到前一个界面传递过来的参数params, 
得到回调函数callBack进行回调传值. 将输入框中的值逆传给上一个界面.

6. TabNavigator的简单使用

注意要导入 
import {StackNavigator, TabNavigator} from 'react-navigation';

和StackNavigator非常相似.

const HelloRN = TabNavigator(
{
    first: {
        screen: Nav1,
        navigationOptions: {
            tabBarLabel: '主页',
            tabBarIcon: ({tintColor}) => (
                <Image style={{tintColor: tintColor}} source={require('./MyApp/images/tab_groups@2x.png')}/>)
        },
    },
    second: {
        screen: Nav2,
        navigationOptions: {
            tabBarLabel: "我的",
            tabBarIcon: ({tintColor}) => (
                <Image style={{tintColor: tintColor}} source={require('./MyApp/images/tab_settings@2x.png')}/>)
        }
    },
},

//这里设tabbar的一些全局属性, 会影响所有的页面中的tabbar.
{
    animationEnabled: false, // 切换页面时是否有动画效果
    tabBarPosition: 'bottom', // 显示在底端,android 默认是显示在页面顶端的
    swipeEnabled: false, // 是否可以左右滑动切换tab
    backBehavior: 'none', // 按 back 键是否跳转到第一个Tab(首页), none 为不跳转
    tabBarOptions: {
        activeTintColor: '#17e', // 文字和图片选中颜色
        inactiveTintColor: '#666', // 文字和图片未选中颜色
        showIcon: true, // android 默认不显示 icon, 需要设置为 true 才会显示
        indicatorStyle: {
            height: 0  // 如TabBar下面显示有一条线,可以设高度为0后隐藏
        },
        style: {
            backgroundColor: '#eee', // TabBar 背景色
            height: 44, //默认为44
        },
        labelStyle: {
            fontSize: 14, // 文字大小
        },
    },
}
);

7 ListView的使用

//创建一个DataSource数据源.
constructor(props) {
    super(props);
    //这里进行DataSource的创建, 指定渲染策略.
    let ds = new ListView.DataSource({rowHasChanged: (r1, r2) => r1 !== r2});
    this.state = {
        isLoaded: false,
        dataSource: ds
    }
}

//进行数据请求. 获取数据,更新state 进而重新渲染页面,
//注意只要state中的值发生变化就会进行render方法的调用.
componentDidMount() {
    Util.getRquest(Constants.IMOOC_API, (data) => {
        this.setState({
            dataSource: this.state.dataSource.cloneWithRows(data),
            isLoaded: true
        });
    }, "data");
}

页面渲染
render() {
    if (!this.state.isLoaded) {
        return this.renderLoadingView();
    }
    return (
        <View style={styles.container}>
            <ListView
                style={styles.listView}
                dataSource={this.state.dataSource}
                renderRow={this._renderRow.bind(this)}
                initialListSize={10}/>
        </View>
    );
}

渲染每一行
_renderRow(model) {
    return (
        <View style={styles.rowContainer}>
            <Image source={{uri: model.picSmall}} style={styles.cellImage}/>
            <View style={styles.textContainer}>
                <Text style={styles.title}>{model.name}</Text>
                <Text style={styles.subTitle}>{model.description}</Text>
            </View>
        </View>
    )
}

//加载等待的view
renderLoadingView() {
    return (
        <View style={{flex: 1, alignItems: "center", justifyContent: "center"}}>
            <Text style={{fontSize: 18, color: "#17e"}}>
                加载中...
            </Text>
        </View>
    );
}

8 公共模块和全局常量的书写.

Constants.js文件

/**
*定义APP需要要的全局常量文件.
* 存放全局的const, 如:公共的API, 以及主题等.
*/

const Constatnts = {
    /**
     * 新闻接口API
     */
    NEWS_API : 'http://c.m.163.com/nc/article/list/T1348649580692/0-20.html',
    /**
     * 慕课网API
     */
    IMOOC_API:"http://www.imooc.com/api/teacher?type=4&num=30"
};

export default Constatnts;

MyUtils文件如下. 常用的工具如设备屏幕的宽高, 设备的操作系统.以及get请求方法的封装.

const Dimensions = require('Dimensions');
const Platform = require('Platform');

export default class MyUtils {

static getScreenWidth() {
    return Dimensions.get('window').width;
}

static getScreenHeight() {
    return Dimensions.get('window').height;
}

static getPlatformOS() {
    return Platform.OS;
}

static getRquest(url, callBack, objectName) {
    fetch(url).then((response) => {
        return response.json();
    }).then((responseData) => {
        callBack(responseData[objectName]);
    }).catch((error) => {
        alert(error);
    })
}
}

9 关于style样式的简单介绍

你可以内联这样写, 注意这里要加两个大括号, 外层大括号是javascript的语法. 内层大括号是jsx的语法.

 <View style={{flex:1,alignItems:"center", justifyContent:"center"}}>
    <Text style={{ fontSize:18, color:"#17e"}}>
        加载中...
    </Text>
 </View>

当然, 更推荐这样的方式(外联).

const styles = StyleSheet.create({
ListView: {
    marginTop: 20,
    backgroundColor: "#F5FCFF"
},
rowContainer: {
    flexDirection: "row",
    flex: 1,
    paddingTop: 5,
    paddingBottom: 5,
    borderBottomWidth: 1,
    borderColor: "#666"
},
image: {
    width: 120,
    height: 100,
    marginLeft: 10
},
textContainer: {
    flex: 1,
    paddingLeft: 10,
    paddingRight: 10,
    justifyContent: "space-around"
},
title: {
    color: "black",
    fontSize: 15
},
subtitle: {
    color: "#666",
    fontSize: 13,
    textAlign: "right",
}
});

/**
 * justifyContent是垂直方式
 * alignItems是水平方式
 *
 * 具体的样式有:
 * 1. center 容器内居中对齐
 * 2. flex-start 从容器的头部开始排列
 * 3. flex-end 从容器的底部开始排列
 * 4. space-around 容器内各组件以及与容器之间间距相等
 * 5. space-between 容器内各组之间间距相等(不包含和容器之间)
 */

总结

好了, 今天就介绍到这里了, 一些常用的简单功能基本上都讲了哈. 可能由于我的水平有限, 讲的不是很清楚, 希望读者谅解. 待我再学些时日, 再来向大家献丑. hhhh(腹黑). 谢谢!

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

推荐阅读更多精彩内容