使用ReactNative开发狗狗说应用流程

效果图

开发流程

主界面Tab切换

1.先完成主界面的三个tab切换,我们这里使用TabBarIOS

来到官方的TabBarIOS文档直接将实例代码的拷贝到项目

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
var base64Icon = '';
var say = React.createClass( {
statics: {
title: '<TabBarIOS>',
description: 'Tab-based navigation.',
},
displayName: 'TabBarExample',
getInitialState: function() {
return {
selectedTab: 'redTab',
notifCount: 0,
presses: 0,
};
},
_renderContent: function(color: string, pageText: string, num?: number) {
return (
<View style={[styles.tabContent, {backgroundColor: color}]}>
<Text style={styles.tabText}>{pageText}</Text>
<Text style={styles.tabText}>{num} re-renders of the {pageText}</Text>
</View>
);
},
render: function() {
return (
<TabBarIOS
tintColor="#ee735c">
<TabBarIOS.Item
title="视频"
icon={{uri: base64Icon, scale: 3}}
selected={this.state.selectedTab === 'blueTab'}
onPress={() => {
this.setState({
selectedTab: 'blueTab',
});
}}>
{this._renderContent('#414A8C', 'Blue Tab')}
</TabBarIOS.Item>
<TabBarIOS.Item
systemIcon="history"
badge={this.state.notifCount > 0 ? this.state.notifCount : undefined}
selected={this.state.selectedTab === 'redTab'}
onPress={() => {
this.setState({
selectedTab: 'redTab',
notifCount: this.state.notifCount + 1,
});
}}>
{this._renderContent('#783E33', 'Red Tab', this.state.notifCount)}
</TabBarIOS.Item>
<TabBarIOS.Item
icon={require('./flux.png')}
title="More"
selected={this.state.selectedTab === 'greenTab'}
onPress={() => {
this.setState({
selectedTab: 'greenTab',
presses: this.state.presses + 1
});
}}>
{this._renderContent('#21551C', 'Green Tab', this.state.presses)}
</TabBarIOS.Item>
</TabBarIOS>
);
}
})

同时在项目根目录添加一个flux.png图片,因为上面的代码引用到了,添加他只是为了能够将项目跑起来,后面会替换掉他,现在运行就能看的效果了。

替换Tab图标

我们这里使用一个react-native-vector-icons库,它自带了很多的图标,安装

1
npm i react-native-vector-icons@2.0.2 --save

直接安装完还是不能直接使用的,作用就是将一些库连接到工程,包括字体和图标。这里使用rnpm

1
2
3
4
npm i rnpm@1.7.0 -g
//连接
rnpm link react-native-vector-icons

现在就可以来到http://ionicons.com选择喜欢的图标了,选择后单击图标就可以得到名称。然后在代码中引入

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
var Icon = require('react-native-vector-icons/Ionicons')
//将自带的tab替换为Icon.TabBarItem,他是react-native-vector-icons提供的,能更好的配合这个图标
<Icon.TabBarItem
title="Blue Tab"
iconName='ios-videocam-outline'
selectedIconName='ios-videocam'
selected={this.state.selectedTab === 'blueTab'}
onPress={() => {
this.setState({
selectedTab: 'blueTab',
});
}}>
{this._renderContent('#414A8C', 'Blue Tab')}
</Icon.TabBarItem>

特别注意icon变成了iconName,selected变为了selectedIconName,完整的tab代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
var say = React.createClass( {
getInitialState: function() {
return {
selectedTab: 'videoTab'
};
},
_renderContent: function(color: string, pageText: string, num?: number) {
return (
<View style={[styles.tabContent, {backgroundColor: color}]}>
<Text style={styles.tabText}>{pageText}</Text>
<Text style={styles.tabText}>{num} re-renders of the {pageText}</Text>
</View>
);
},
render: function() {
return (
<TabBarIOS
tintColor="#ee735c">
<Icon.TabBarItem
title="视频"
iconName='ios-videocam-outline'
selectedIconName='ios-videocam'
selected={this.state.selectedTab === 'videoTab'}
onPress={() => {
this.setState({
selectedTab: 'videoTab',
});
}}>
{this._renderContent('#414A8C', 'Blue Tab')}
</Icon.TabBarItem>
<Icon.TabBarItem
title="录制"
iconName='ios-recording-outline'
selectedIconName='ios-recording'
selected={this.state.selectedTab === 'recordTab'}
onPress={() => {
this.setState({
selectedTab: 'recordTab'
});
}}>
{this._renderContent('#783E33', 'Red Tab', this.state.notifCount)}
</Icon.TabBarItem>
<Icon.TabBarItem
title="更多"
iconName='ios-more-outline'
selectedIconName='ios-more'
selected={this.state.selectedTab === 'moreTab'}
onPress={() => {
this.setState({
selectedTab: 'moreTab'
});
}}>
{this._renderContent('#21551C', 'Green Tab', this.state.presses)}
</Icon.TabBarItem>
</TabBarIOS>
);
}
})

写tab对应的页面

接下来我们需要将内容更换为三个子组件,分别对应三个tab页面。

首先先写三个组件

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
var List=React.createClass({
render:function(){
return (
<View style={styles.container}>
<Text>视频列表</Text>
</View>
)
}
})
var Edit=React.createClass({
render:function(){
return (
<View style={styles.container}>
<Text>制作页面</Text>
</View>
)
}
})
var Account=React.createClass({
render:function(){
return (
<View style={styles.container}>
<Text>账户页面</Text>
</View>
)
}
})

然后在tab中分别加入三个组件

1
2
3
4
5
6
7
8
9
10
11
12
<Icon.TabBarItem
title="视频"
iconName='ios-videocam-outline'
selectedIconName='ios-videocam'
selected={this.state.selectedTab === 'list'}
onPress={() => {
this.setState({
selectedTab: 'list',
});
}}>
<List/> //在这里我们替换成了上面定义的组件
</Icon.TabBarItem>

将组件拿到对应的目录下

上面的组件还是放到一个文件中的,导致的问题是,多个页面逻辑都混到一个页面了,修改和删除代码都不是很方法,所以采用的方法时将每个组件都放到相应的目录,然后在入口文件导入他。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
var React = require('react-native')
var {
StyleSheet,
Text,
View,
} = React;
var Icon = require('react-native-vector-icons/Ionicons')
var List=React.createClass({
render:function(){
return (
<View style={styles.container}>
<Text>视频列表</Text>
</View>
)
}
})
var styles = StyleSheet.create({
container: {
flex: 1,
justifyContent: 'center',
alignItems: 'center',
backgroundColor: '#F5FCFF',
},
welcome: {
fontSize: 20,
textAlign: 'center',
margin: 10,
},
instructions: {
textAlign: 'center',
color: '#333333',
marginBottom: 5,
},
})
module.exports=List

在入口文件中导入

1
2
3
4
//自定义模块
var List=require('./app/list/index')
var Edit=require('./app/edit/index')
var Account=require('./app/account/index')

创建模拟数据

我们这里使用rap和mockjs。

先去rap官网创建一个项目。http://rap.taobao.org/org/index.do,配置完生成规则后,可以使用下面地址获取规则或真实数据

http://rap.taobao.org/mockjsdata/11015/api/video?accessToken=111

http://rap.taobao.org/mockjs/11015/api/video?accessToken=111

如果是规则,则需要使用mockjs解析,官网为:http://mockjs.com/

你可以在控制台下输入Mock.mock(mock规则数据试试)

1
2
3
Mock.mock({"data|10":[{"_id":"@ID","video":"http:\/\/v2.mukewang.com\/cef3e514-8203-4c67-877f-79f7763553cf\/L.mp4?auth_key=1480760293-0-0-a693983f9d989e19a2c1c19f52d1e169","thumb":"@IMG(1200x600,@color())"}],"success":true})
//会输出10条测试数据

我们也可以直接在该页面的控制台下输入

1
var d = Mock.mock({"data|10":[{"_id":"@ID","video":"http:\/\/v2.mukewang.com\/cef3e514-8203-4c67-877f-79f7763553cf\/L.mp4?auth_key=1480760293-0-0-a693983f9d989e19a2c1c19f52d1e169","thumb":"@IMG(1200x600,@color())"}],"success":true});d.data.forEach(function(item){$("#examples").append('<h3>'+item._id+'</h3><img src="'+item.thumb+'"/>')})

就可以看到该页面最后又很多的图片和id数组。

视频页面

添加视频页面标题

在List下面的index.js文件中添加

1
2
3
4
5
<View style={styles.container}>
<View style={styles.header}>
<Text style={styles.headerTitle}>列表页面</Text>
</View>
</View>

然后修改样式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
var styles = StyleSheet.create({
container: {
flex: 1,
backgroundColor: '#F5FCFF',
},
header:{
paddingTop:25,
paddingBottom:12,
backgroundColor:'#ee735c'
},
headerTitle:{
color:'white',
fontSize:16,
textAlign:'center',
fontWeight:'600'
}
})

添加视频列表

添加一个ListView,首先要导出

1
2
3
4
5
<ListView
dataSource={this.state.dataSource}
renderRow={this.renderRow}
enableEmptySections={true}
/>

在定义一个renderRow方法,用来渲染每一个条目,相当于Android中Adapter的getView

1
2
3
renderRow(row){
}

完整代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
var List=React.createClass({
getInitialState: function() {
var ds = new ListView.DataSource({rowHasChanged: (r1, r2) => r1 !== r2});
return {
dataSource: ds.cloneWithRows([]),
};
},
renderRow(row){
},
render:function(){
return (
<View style={styles.container}>
<View style={styles.header}>
<Text style={styles.headerTitle}>列表页面</Text>
</View>
<ListView
dataSource={this.state.dataSource}
renderRow={this.renderRow}
enableEmptySections={true}
/>
</View>
)
}
})

我们实现了一下renderRow方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
renderRow(row){
return(
<TouchableHighlight>
<View style={styles.item}>
{/* 标题 */}
<Text style={styles.title}>{row.title}</Text>
{/* 图片 */}
<Image
source={{uri:row.thumb}}
style={styles.thumb}
>
<Icon
name='ios-play'
size={28}
style={styles.play} />
</Image>
{/* 底部按钮 */}
<View style={styles.itemFooterContainer}>
<View style={styles.handleBox}>
<Icon
name='ios-heart-outline'
size={28}
style={styles.up} />
<Text style={styles.handleText}>喜欢</Text>
</View>
<View style={styles.handleBox}>
<Icon
name='ios-chatboxes-outline'
size={28}
style={styles.commonIcon} />
<Text style={styles.handleText}>评论</Text>
</View>
</View>
</View>
</TouchableHighlight>
)
}

在定义一些样式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
item:{
width:width,
marginBottom:10,
backgroundColor:'#fff'
},
thumb:{
width:width,
height:width*0.56,
resizeMode:'cover'
},
title:{
padding:10,
fontSize:18,
color:'#333'
},
itemFooterContainer:{
flexDirection:'row',
justifyContent:'space-between',
backgroundColor:'#eee'
},
handleBox:{
padding:10,
flexDirection:'row',
width:width/2-0.5,
justifyContent:'center',
backgroundColor:'#fff'
},
play:{
position:'absolute',
bottom:14,
right:14,
width:46,
height:46,
paddingTop:9,
paddingLeft:18,
backgroundColor:'transparent',
borderColor:'#fff',
borderWidth:1,
borderRadius:23,
color:'#ed7b66'
},
handleText:{
paddingLeft:12,
fontSize:18,
color:'#333'
},
up:{
fontSize:22,
color:'#333'
},
commonIcon:{
fontSize:22,
color:'#333'
}

样式中我们使用了屏幕的宽度,可以通过

1
2
3
4
//导入Dimensions
// 当前屏幕宽度
var width=Dimensions.get('window').width

然后初始化一些假数据,然列表能显示出来

1
2
3
4
5
6
7
8
9
10
11
12
13
14
getInitialState: function() {
var ds = new ListView.DataSource({rowHasChanged: (r1, r2) => r1 !== r2});
return {
dataSource: ds.cloneWithRows([
{
"_id":"350000197307022063","thumb":"http://dummyimage.com/1280x720/e5488e","title":"测试内容95hw","video":"http://v2.mukewang.com/cef3e514-8203-4c67-877f-79f7763553cf/L.mp4?auth_key=1480760293-0-0-a693983f9d989e19a2c1c19f52d1e169"
}
,
{
"_id":"330000201112047039","thumb":"http://dummyimage.com/1280x720/91d5da","title":"测试内容95hw","video":"http://v2.mukewang.com/cef3e514-8203-4c67-877f-79f7763553cf/L.mp4?auth_key=1480760293-0-0-a693983f9d989e19a2c1c19f52d1e169"
}
]),
};
}

获取网络数据

Fetch,WebSocket,XMLHttpRequest

官方网络文档:http://facebook.github.io/react-native/releases/0.22/docs/network.html#content

我们这里使用fetch

1
2
3
4
5
6
7
8
9
10
11
12
13
14
componentDidMount(){
this._fetchData()
},
_fetchData(){
fetch('http://rap.taobao.org/mockjs/11015/api/video')
.then((response) => response.json())
.then((responseText) => {
console.log(responseText);
})
.catch((error) => {
console.warn(error);
});
}

但是现在还是获取的mock规则,我们需要借助mockjs生成真是的数据

1
npm i mockjs --save

然后将生成的规则使用Mock解析,就可以得到真是数据。

1
var data=Mock.mock(response)

注意:这里有个小插曲就是要删除mock.js中的dataImage方法,我们用不到canvas,所以要删掉他。

获取完网络数据后还需要判断是否请求成功,然后将数据替换到datasource中

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
_fetchData(){
fetch('http://rap.taobao.org/mockjs/11015/api/video')
.then((response) => response.json())
.then((response) => {
var data=Mock.mock(response)
console.log(data)
if (data.success) {
this.setState({
dataSource:this.state.dataSource.cloneWithRows(data.data)
})
}
})
.catch((error) => {
console.warn(error);
});
}

重构网络模块

这样写不好是因为几乎我们每个页面都要用到网络请求,通常的情况下是将网络请求封装成一个类,所以这里我们将网络请求封装到request.js中

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
'use strict'
var queryString=require('query-string')
var _=require('lodash')
var config=require('./config')
var Mock=require('mockjs')
var request={}
request.get=function (url,params) {
if (params) {
url+='?'+queryString.stringify(params)
}
return fetch(url)
.then((response)=>response.json())
.then((response)=>Mock.mock(response))
}
request.post=function (url,body) {
var options = _.extend(config.header,{
body:JSON.stringify(body)
})
return fetch(url,options)
.then((response)=>response.json())
.then((response)=>Mock.mock(response))
}
module.exports=request

我暴露了get,post方法,其中config是一个配置文件,包括网络的根请求地址和一些接口地址

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
'use strict'
module.exports={
header: {
method: 'POST',
headers: {
'Accept': 'application/json',
'Content-Type': 'application/json',
}
},
api:{
base:'http://rap.taobao.org/mockjs/11015/',
// base:'http://rap.taobao.org/mockjsdata/11015/',
videoList:'api/video'
}
}

其中我们用到了query-string,lodash需要手动安装

1
2
npm i query-string --save
npm i lodash --save

这样我们首页的请求就可以改写为

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
_fetchData(){
request.get(config.api.base+config.api.videoList,{
accessToken:'abbbb'
})
.then((data) => {
console.log(data)
if (data.success) {
this.setState({
dataSource:this.state.dataSource.cloneWithRows(data.data)
})
}
})
.catch((error) => {
console.warn(error);
});
}

这样就方便快捷多了,代码变更地址为

上拉加载更多

我们要在listview中添加一个onEndReached方法监测是否滑到了底部

1
onEndReached={this._fetchMoreData}

然后实现_fetchMoreData方法

1
2
3
4
5
6
7
8
9
10
_fetchMoreData(){
if (!this._hasMore() || this.state.isLoading) {
//如果没有更多数据了,或者正在加载中,直接返回
return
}
var page = cacheReuslts.nextPage
this._fetchData(page)
}

当然回来的数据还是得和原来的数据追加

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
_fetchData(page){
var that = this
this.setState({
isLoading:true
})
request.get(config.api.base+config.api.videoList,{
accessToken:'abbbb',
page:page
})
.then((data) => {
console.log(data)
if (data.success) {
var items = cacheReuslts.items.slice()
items=items.concat(data.data)
cacheReuslts.items=items
cacheReuslts.total=data.total
console.log(cacheReuslts.items.length)
//这里是为了有显示loading
setTimeout(function () {
that.setState({
isLoading:false,
dataSource:that.state.dataSource.cloneWithRows(cacheReuslts.items)
})
},2000)
}
})
.catch((error) => {
this.setState({
isLoading:false
})
console.warn(error);
});
}

下拉刷新

使用refreshControl

1
2
3
4
5
6
7
8
9
refreshControl={
<RefreshControl
refreshing={this.state.isRefreshing}
onRefresh={this._onRefresh}
tintColor="#ff6600"
title="拼命加载中..."
colors={['#ff0000', '#00ff00', '#0000ff']}
progressBackgroundColor="#ffff00" />
}

然后实现_onRefresh方法

1
2
3
4
5
6
7
_onRefresh(){
if (this.state.isRefreshing) {
return
}
this._fetchData(0)
}

同时还得在fetchData方法里面通过判断page是否等于0来更改是刷新还是加载更多。

列表页点赞

这个功能分两部分,第一部分是在View中根据是否点赞显示不同的图标

1
2
3
4
5
6
7
8
<View style={styles.handleBox}>
<Icon
name={this.state.up ? 'ios-heart' : 'ios-heart-outline'}
size={28}
onPress={this._up}
style={[styles.up ,this.state.up ? null : styles.down ]} />
<Text style={styles.handleText} onPress={this._up}>喜欢</Text>
</View>

第二部分是当点击了按钮,发送点赞请求

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
_up(){
var that=this
var up = !this.state.up
var row=this.state.row
var url=config.api.base+config.api.up
var body={
id:row._id,
up:up,
accessToken:'ab'
}
console.log('up')
request.post(url,body)
.then(function (data) {
if (data && data.success) {
that.setState({
up:up
})
} else{
AlertIOS.alert('点赞失败,请稍后重试')
}
})
.catch(function (err) {
console.log(err)
AlertIOS.alert('点赞失败,请稍后重试')
})
}

请求回来了我们还得判断是否响应成功,然后改变状态,让界面刷新

添加进入视频详情页面

在这之前我们先要使用导航器包装

1
2
3
4
5
6
7
8
9
10
11
12
<Navigator
initialRoute={{
name:'list',
component:List
}}
configureScene={(route)=>{
return Navigator.SceneConfigs.FloatFromRight
}}
renderScene={(route,navigator)=>{
var Component=route.component
return <Component {...route.params} navigator={navigator} />
}} />

这样我们就可以在List组件里面通过navigator属性拿到外面的Navigator实例,从而进入一个页面。

然后在TouchableHighlight控件上添加一个onPress方法

1
2
3
4
5
6
_loadDetailPage(row){
this.props.navigator.push({
name:'detail',
component:Detail
})
}

这样我们就能跳转到详情页面了,现在需要实现返回到列表

1
2
3
_back(){
this.props.navigator.pop()
}

还要将外面的id传入详情页,我们就可以拿着id去请求详情页数据了

1
2
3
4
5
6
7
8
9
_loadDetailPage(row){
this.props.navigator.push({
name:'detail',
component:Detail,
params:{
_id:row._id
}
})
}

然后在详情界面就可以拿到这个id

1
2
3
4
5
6
7
8
9
render:function(){
var _id = this.props._id
return (
<View style={styles.container}>
<Text onPress={this._back}>详情页面{_id}</Text>
</View>
)
}

那这个params是怎么传递过来了的呢,起始就是在我们包装List的时候的使用了

1
<Component {...route.params} navigator={navigator} />

…的作用就是讲params中的所有属性展开依次传递给Component,所有我们在组件内部就能通过props拿到

详情页播放视频

这里使用一个第三方库

1
2
3
4
npm i react-native-video --save
//安装完后还需要链接,我这里链接又报错,所以只能手动链接
rnpm link react-native-video

链接完成后,在详情界面写上Video组件

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
<Video
style={styles.video}
ref='videoPlayer'
source={{uri:data.video}}
style={styles.video}
volume={3}
paused={false}
rate={this.state.rate}
muted={this.state.muted}
resizeMode={this.state.resizeMode}
repeat={this.state.repeat}
onLoadStart={this._onLoadStart}
onLoad={this._onLoad}
onProgress={this._onProgress}
onEnd={this._onEnd}
onError={this._onError} />

播放和暂停

只需要控制Video组件的paused属性就行了

添加返回和标题栏

1
2
3
4
5
6
7
8
9
<View style={styles.header}>
<TouchableOpacity style={styles.backBox} onPress={this._pop}>
<Icon name='ios-arrow-back' style={styles.backIcon} />
<Text style={styles.backText} >返回</Text>
</TouchableOpacity>
<Text style={styles.headerTitle} numberOflines={1}>
视频详情页
</Text>
</View>

对应的样式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
header:{
flexDirection:'row',
justifyContent:'center',
alignItems:'center',
width:width,
height:64,
paddingTop:20,
paddingLeft:10,
paddingRight:10,
borderBottomWidth:1,
borderColor:'rgba(0,0,0,0.1)',
backgroundColor:'#fff'
},
backBox:{
position:'absolute',
left:12,
top:32,
width:50,
flexDirection:'row',
alignItems:'center'
},
headerTitle:{
width:width-120,
textAlign:'center'
},
backIcon:{
color:'#999'
},
backText:{
color:'#999'
}
1
2
3
_pop(){
this.props.navigator.pop()
}

视频制作者信息

1
2
3
4
5
6
7
<View style={styles.infoBox} >
<Image style={styles.avatar} source={{uri:data.author.avatar}} />
<View style={styles.descBox}>
<Text style={styles.nickname} > {data.author.nickname} </Text>
<Text style={styles.title} > {data.title} </Text>
</View>
</View>

样式

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
infoBox:{
width:width,
flexDirection:'row',
justifyContent:'center',
marginTop:20
},
avatar:{
width:60,
height:60,
marginRight:10,
marginLeft:10,
borderRadius:30
},
descBox:{
flex:1
},
nickname:{
fontSize:18
},
title:{
marginTop:8,
fontSize:16,
color:'#666'
}

显示视频信息

1
2
3
4
5
6
7
<View style={styles.infoBox} >
<Image style={styles.avatar} source={{uri:data.author.avatar}} />
<View style={styles.descBox}>
<Text style={styles.nickname} > {data.author.nickname} </Text>
<Text style={styles.title} > {data.title} </Text>
</View>
</View>

显示评论内容

小插曲

1.像这种布局

1
2
3
4
5
6
7
<View key={row._id} style={styles.replyBox} >
<Image style={styles.replyAvatar} source={{uri:row.replyBy.avatar}} />
<View style={styles.reply}>
<Text style={styles.replyNickname} > {row.replyBy.nickname} </Text>
<Text style={styles.replyContent} > {row.content} </Text>
</View>
</View>

replyBox的样式一定要设置宽度,不然整个页面有可能一部分显示到了左边屏幕外面了

1
2
3
4
5
6
replyBox:{
width:width,
flexDirection:'row',
justifyContent:'flex-start',
marginTop:10
}

2.如果将ScrollView放到了VideoBox布局里面,将导致listview无法滚动,需要将他拿到和videoBox平级的布局中

优化评论布局

由于我们要显示视频信息和评论内容,所以用了ScrollView嵌套ListView

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
<ScrollView
showVerticalScrollIndicator={false}
enableEmptySections={true}
automaticallyAdjustContentInsets={false}
style={styles.scrollView} >
<View style={styles.infoBox} >
<Image style={styles.avatar} source={{uri:data.author.avatar}} />
<View style={styles.descBox}>
<Text style={styles.nickname} > {data.author.nickname} </Text>
<Text style={styles.title} > {data.title} </Text>
</View>
</View>
<ListView
dataSource={this.state.dataSource}
renderRow={this._renderRow}
onEndReached={this._fetchMoreData}
renderFooter={this._renderFooter}
onEndReachedThreshold={20}
showVerticalScrollIndicator={false}
enableEmptySections={true}
automaticallyAdjustContentInsets={false}
refreshControl={
<RefreshControl
refreshing={this.state.isRefreshing}
onRefresh={this._onRefresh}
tintColor="#ff6600"
title="拼命加载中..."
colors={['#ff0000', '#00ff00', '#0000ff']}
progressBackgroundColor="#ffff00" />
}
/>
</ScrollView>

显示需要去除ScrollView,这里使用renderHeader={this._renderHeader},用法和renderFooter差不多,先返回布局

1
2
3
4
5
6
7
8
9
10
11
12
13
_renderHeader(){
var data=this.state.data
return(
<View style={styles.infoBox} >
<Image style={styles.avatar} source={{uri:data.author.avatar}} />
<View style={styles.descBox}>
<Text style={styles.nickname} > {data.author.nickname} </Text>
<Text style={styles.title} > {data.title} </Text>
</View>
</View>
)
}

然后删除scrollview相关的代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
<ListView
dataSource={this.state.dataSource}
renderRow={this._renderRow}
onEndReached={this._fetchMoreData}
renderHeader={this._renderHeader}
renderFooter={this._renderFooter}
onEndReachedThreshold={20}
showVerticalScrollIndicator={false}
enableEmptySections={true}
automaticallyAdjustContentInsets={false}
refreshControl={
<RefreshControl
refreshing={this.state.isRefreshing}
onRefresh={this._onRefresh}
tintColor="#ff6600"
title="拼命加载中..."
colors={['#ff0000', '#00ff00', '#0000ff']}
progressBackgroundColor="#ffff00" />
}
/>

添加评论

我们在视频页面添加了一个引导用户评论的输入框,但这个框起始只是一个点击层而已,目的是用户点击弹出一个浮层,在浮层里面评论

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
_renderHeader(){
var data=this.state.data
return(
<View state={styles.listHeader}>
<View style={styles.infoBox} >
<Image style={styles.avatar} source={{uri:data.author.avatar}} />
<View style={styles.descBox}>
<Text style={styles.nickname} > {data.author.nickname} </Text>
<Text style={styles.title} > {data.title} </Text>
</View>
</View>
<View style={styles.commentBox}>
<View style={styles.comment}>
<TextInput
placeholder='敢不敢评论一个呀'
style={styles.content}
multiline={true}
onFocus={this._onFocus} />
</View>
</View>
<View style={styles.commentArea}>
<Text style={styles.commentTitle}>热门评论</Text>
</View>
</View>
)
}

提交组件

1
npm i react-native-button --save

不需要链接,先导入

1
2
3
//这种导入方式报错
//var Button=require('react-native-button')
import Button from 'react-native-button';

添加组件到布局

1
<Button style={styles.submit} onPress={this._submit} >评论</Button>

添加一个_submit方法,在这里处理评论逻辑

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
_submit(){
var that = this
if (!this.state.content) {
return AlertIOS.alert('请输入评论内容!')
}
if (this.state.isSending) {
return AlertIOS.alert('正在拼命评论中...')
}
//正在发送
this.setState({
isSending:true
},function () {
var body={
accessToken:'a',
videoId:'123',
content:this.state.content
}
var url = config.api.base+config.api.comment
request.post(url,body)
.then(function (data) {
if (data&&data.success) {
var items=cacheReuslts.items.slice()
var cotnent=that.state.content
items=[{
content:cotnent,
replyBy:{
avatar:'https://dummyimage.com/640x640/b108ca',
nickname:'你猜猜我是谁'
}
}].concat(items)
cacheReuslts.items=items
cacheReuslts.total=cacheReuslts.total+1
that.setState({
content:'',
isSending:false,
dataSource:that.state.dataSource.cloneWithRows(cacheReuslts.items)
})
that._setModalVisible(false)
}
})
.catch((err)=>{
console.log(err)
that.setState({
isSending:false
})
that._setModalVisible(false)
AlertIOS.alert('留言失败,稍后重试!')
})
})
}

验证码倒计时

1
npm i react-native-sk-countdown --save

我们在手机号旁边添加一个倒计时控件,先判断是否发送了验证码,在显示验证码输入框和倒计时按钮。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
{
this.state.codeSent
? <View style={styles.verifyCodeBox} >
<TextInput
placeholderTextColor='#444'
style={styles.inputField}
placeHolder='请输入验证码'
autoCaptialize={'none'}
autoCorrect={false}
keyboradType={'number-pad'}
onChangeText={(text)=>{
this.setState({
verifyCode:text
})
}} />
{
this.state.countingDone
? <Button
style={styles.countBtn}
onPress={this._sendVerifyCode}> 获取验证码</Button>
: <CountDown
style={styles.countBtn}
countType='seconds' // 计时类型:seconds / date
auto={true} // 自动开始
afterEnd={this._countingDone} // 结束回调
timeLeft={60} // 正向计时 时间起点为0秒
step={-1} // 计时步长,以秒为单位,正数则为正计时,负数为倒计时
startText='获取验证码' // 开始的文本
endText='获取验证码' // 结束的文本
intervalText={(sec) => sec + '秒重新获取'} />
}
</View>
: null
}

同时验证码倒计时完成了,在显示重新获取验证码按钮。

启动切换到登陆或首页逻辑

首先我们先在index.ios.js文件中获取是否有用户状态,如果有显示主界面,如果没有显示登陆界面

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
componentDidMount(){
this._getAppStatus()
},
_getAppStatus(){
var that = this
AsyncStorage.getItem('user')
.then((data)=>{
var user
var newStatus={}
if (data) {
newStatus.user=user
newStatus.logined=true
} else{
newStatus.logined=false
}
that.setState(newStatus)
})
}

然后我们判断是否登陆,并渲染相应的界面

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
render: function() {
if (!this.state.logined) {
return <Login afterLogin={this._afterLogin}/>
}
return (
<TabBarIOS
tintColor="#ee735c">
<Icon.TabBarItem
title="视频"
iconName='ios-videocam-outline'
selectedIconName='ios-videocam'
selected={this.state.selectedTab === 'list'}
onPress={() => {
this.setState({
selectedTab: 'list',
});
}}>
<Navigator
initialRoute={{
name:'list',
component:List
}}
configureScene={(route)=>{
return Navigator.SceneConfigs.FloatFromRight
}}
renderScene={(route,navigator)=>{
var Component=route.component
return <Component {...route.params} navigator={navigator} />
}} />
</Icon.TabBarItem>
<Icon.TabBarItem
title="录制"
iconName='ios-recording-outline'
selectedIconName='ios-recording'
selected={this.state.selectedTab === 'edit'}
onPress={() => {
this.setState({
selectedTab: 'edit'
});
}}>
<Edit/>
</Icon.TabBarItem>
<Icon.TabBarItem
title="更多"
iconName='ios-more-outline'
selectedIconName='ios-more'
selected={this.state.selectedTab === 'account'}
onPress={() => {
this.setState({
selectedTab: 'account'
});
}}>
<Account/>
</Icon.TabBarItem>
</TabBarIOS>
);
}

显示用户头像

首先我们判断用户是否有头像然后显示不同的布局

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
{
user.avatar
? <TouchableOpacity style={styles.avatarContainer}>
<Image source={{uri:user.avatar}} style={styles.avatarContainer}>
<View style={styles.avatarBox} >
<Image
source={{uri:user.avatar}}
style={styles.avatar} />
</View>
<Text style={styles.avatarTip}>添加头像</Text>
</Image>
</TouchableOpacity>
:
<View style={styles.avatarContainer}>
<Text style={styles.avatarTip}>添加头像</Text>
<TouchableOpacity style={styles.avatarBox} >
<Icon
name='ios-cloud-upload-outline'
style={styles.plusIcon} />
</TouchableOpacity>
</View>
}

选择图片

使用一个模块

1
npm i react-native-image-picker@0.20.0 --save

链接

1
npm link react-native-image-picker

如果你的系统是iOS10,需要添加权限:http://www.jianshu.com/p/c212cde86877

如果链接完不能用,请手动链接,并导入

1
var ImagePicker = require('react-native-image-picker');

然后配置一些设置

1
2
3
4
5
6
7
8
9
10
11
12
13
var options = {
title: '选择头像',
cancelButtonTitle:'取消',
takePhotoButtonTitle:'拍照',
chooseFromLibraryButtonTitle:'从相册选择',
quality:0.75,
allowsEditing:true,
noData:false,
storageOptions: {
skipBackup: true,
path: 'images'
}
}

然后就可以选择图片了

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
_pickPhoto(){
var that=this
console.log(ImagePicker)
ImagePicker.showImagePicker(options, (response) => {
console.log('Response = ', response);
if (response.didCancel) {
console.log('User cancelled image picker');
return
}
var avatarData='data:image/png;base64,'+response.data
var user=that.state.user
user.avatar=avatarData
that.setState({
user:user
})
// else if (response.error) {
// console.log('ImagePicker Error: ', response.error);
// }
// else if (response.customButton) {
// console.log('User tapped custom button: ', response.customButton);
// }
// else {
// // You can display the image using either data...
// const source = {uri: 'data:image/jpeg;base64,' + response.data, isStatic: true};
// // or a reference to the platform specific asset location
// if (Platform.OS === 'ios') {
// const source = {uri: response.uri.replace('file://', ''), isStatic: true};
// } else {
// const source = {uri: response.uri, isStatic: true};
// }
// this.setState({
// avatarSource: source
// });
// }
});
}

上传图片到图床

我们这里选用https://cloudinary.com,首先选择一个账号,然后将一些配置信息写到程序中

1
2
3
4
5
6
7
8
9
var CLOUDINARY = {
cloud_name: 'dawlcb8jj',
api_key: '6368311532331937597',
api_secret: 'N4rXJImMfxZq03454fSosbZFfOhfHR8',
base:'http://res.cloudinary.com/daw454lcb8jj',
image:'https://api.cloudinary.com/v1_1/daw454lcb8jj/image/upload',
video:'https://api.cloudinary.com/v1_1/daw454lcb8jj/video/upload',
audio: 'https://api.cloudinary.com/v1_1/daw454lcb8jj/raw/upload'
}

然后就是上传逻辑了,首先我们需要将一些参数传递到服务端签名(因为本地签名不安全)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
ImagePicker.showImagePicker(options, (response) => {
console.log('Response = ', response);
if (response.didCancel) {
console.log('User cancelled image picker');
return
}
var avatarData='data:image/png;base64,'+response.data
//先显示出头像
var user=that.state.user
user.avatar=avatarData
that.setState({
user:user
})
var timestamp=Date.now()
var tags='app,avatar'
var folder='avatar'
var signatureUrl=config.api.base+config.api.signature
var accessToken=this.state.user.accessToken
request.post(signatureUrl,{
accessToken:accessToken,
timestamp:timestamp,
folder:folder,
tags,tags,
type:'avatar'
})
.catch((e)=>{
console.log('upload image fail')
console.log(e)
})
.then((data)=>{
if (data&&data.success) {
var signature= 'folder='+folder+'&tags='+tags+'&timestamp='+timestamp+CLOUDINARY.api_secret
signature=sha1(signature)
var body=new FormData()
body.append('folder',folder)
body.append('signature',signature)
body.append('timestamp',timestamp)
body.append('tags',tags)
body.append('api_key',CLOUDINARY.api_key)
body.append('resource_type','image')
body.append('file',avatarData)
this._upload(body)
}
})
})

然后就上传图片

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
_upload(body){
console.log('_upload')
var that = this
var xhr=new XMLHttpRequest()
var url=CLOUDINARY.image
xhr.open('POST',url)
xhr.onload=()=>{
console.log('upload image onload')
if (xhr.status!==200) {
AlertIOS('请求失败')
console.log(xhr.responseText)
return
}
if (!xhr.responseText) {
AlertIOS('请求失败')
console.log(xhr.responseText)
return
}
var response
try{
response=JSON.parse(xhr.response)
}catch(e){
console.log(e)
console.log('parse fails')
}
console.log('upload success')
if (response&&response.public_id) {
var user=that.state.user
user.avatar=avatar(response.public_id,'image')
that.setState({
user:user
})
}
}
xhr.send(body)
}

上传进度

首先安装一个进度组件

1
npm i react-native-progress --save

录音

使用react-native-audio模块

1
2
npm i react-native-audio@1.2.1 --save
//这里安装1.3.0才不报错

然后链接

1
npm link react-native-audio

示例代码https://github.com/jsierles/react-native-audio/blob/master/AudioExample/AudioExample.js

app图标

使用appicontemplate

https://appicontemplate.com/(打开:https://applypixels.com/),然后点击ios app icon,下载app icon,然后打开app的设计稿,然后点击atn后缀的文件这样就导入了动作文件,然后打打开psd,然后将图标拖动到刚刚打开的psd,在调整好颜色,最后使用动作面板导出图标。

使用makeappcion

http://makeappicon.com/,使用网页端,上传一个图标,然后输入邮箱,下载附件,里面包括android,ios,watch

更改app名称

只需要在plist中添加key为CFBundleDisplayName,值为名称就行了,还可以将CFBundleDevelopmentRegion改为zh_CN

更改启动画面

直接修改LaunchScreen.xib

轮播图

使用一个第三方组件

1
npm i react-native-swiper --save

服务端开发过程

初始化项目

新建一个目录,然后输入npm init填写一些基本信息,就初始化完了一个项目,然后安装一些依赖

1
2
3
4
5
//初始化项目
npm init
//安装依赖
npm i koa koa-logger koa-session koa-bodyparser koa-router mongoose sha1 lodash uuid xss bluebird --save

speaksasy找不到了

升级到koa2,安装

1
2
3
4
5
6
7
8
npm install koa@next
koa-logger@2
//session还没支持2,所以要koa-convert
npm i koa-convert --save
app.use(convert(session(app)));
npm install koa-bodyparser@next --save

使用koa搭建一个小型服务器

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
'use strict'
var koa=require('koa')
var logger=require('koa-logger')
var session=require('koa-session')
var bodyParser=require('koa-bodyparser')
var app=koa()
app.keys=['say']
app.use(logger())
app.use(session(app))
app.use(bodyParser())
app.use(function *(next) {
console.log(this.href)
console.log(this.method)
this.body={
success:true
}
yield next
})
app.listen(1234)

现在访问localhost就可以返回一段成功的json

将路由写到单独的文件

首先我们在index.js中将引入路由文件

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
'use strict'
var koa=require('koa')
var logger=require('koa-logger')
var session=require('koa-session')
var bodyParser=require('koa-bodyparser')
var app=koa()
app.keys=['say']
app.use(logger())
app.use(session(app))
app.use(bodyParser())
//引入路由文件
var router=require('./config/routes')()
app
.use(router.routes())
.use(router.allowedMethods())
app.listen(1234)
console.log('listening on port 1234')

然后新建路由文件

config/routes.js

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
'use strict'
var Router=require('koa-router')
var User=require('../app/controllers/user')
var App=require('../app/controllers/app')
module.exports=function () {
var router=new Router({
//定义全局api的前缀
prefix:'/api/1'
})
//post请求使用User控制器上面的signup方法处理
//user
router.post('/u/signup',User.signup)
router.post('/u/verify',User.verify)
router.post('/u/update',User.update)
//app
router.post('/signature',App.signature)
return router
}

在路由文件中我们引入了几个控制器,将相应去请求指定不同的控制器方法调用

user.js

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
'use strict'
exports.signup=function *(next) {
this.body={
success:true
}
}
exports.verify=function *(next) {
this.body={
success:true
}
}
exports.update=function *(next) {
this.body={
success:true
}
}

app.js

1
2
3
4
5
6
7
'use strict'
exports.signature=function *(next) {
this.body={
success:true
}
}

现在我们可以随便访问一个api来测试接口是否正常工作

接入mongodb

首先新建一个model,在这里面定义user的模型

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
'use strict'
var mongoose=require('mongoose')
var UserSchema=new mongoose.Schema({
phoneNumber:{
unique:true,
type:String
},
areaCode:String,
verifyCode:String,
accessToken:String,
nickname:String,
gender:String,
breed:String,
age:String,
avatar:String,
meta:{
createAt:{
type:Date,
default:Date.now()
},
updateAt:{
type:Date,
default:Date.now()
}
}
})
UserSchema.pre('save',function (next) {
if (!this.isNew) {
this.meta.updateAt=Date.now()
}
next()
})
//暴露模型
module.exports=mongoose.model('User',UserSchema)

然后就可以在Controller里面操作数据库了

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
'use strict'
var xss=require('xss')
var mongoose=require('mongoose')
var User=mongoose.model('User')
exports.signup=function *(next) {
//拿到request里面的phoneNumber
// console.log(this.request.body)
// var phoneNumber=this.request.body.phoneNumber
var phoneNumber=this.query.phoneNumber
//根据手机号查找用户,调用exec后就返回的是一个Promise
var user=yield User.findOne({
phoneNumber:phoneNumber
}).exec()
console.log(user)
console.log(phoneNumber)
if (user) {
//更新用户的验证码
user.verifyCode='1212'
} else{
//不存在,则新建一个用户
user = new User({
phoneNumber:xss(phoneNumber)
})
}
try{
user=yield user.save()
}catch(e){
this.body={
success:false
}
return
}
this.body={
success:true
}
}
exports.verify=function *(next) {
this.body={
success:true
}
}
exports.update=function *(next) {
this.body={
success:true
}
}

接入注册短信

这里使用https://luosimao.com

通用验证

我们知道基本上每个请求都有accessToken,如果是post请求需要有body,像这样的逻辑,如果在每个接口都判断,可以说是很庞大的,这样我们将数据的校验写到app.js的功能中

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
exports.hasBody=function *(next) {
var body=this.request.body||{}
if (Object.keys(body).length===0) {
this.body={
success:false,
err:'没有传递body参数'
}
return next
}
yield next
}
//全局token判断
exports.hasToken=function *(next) {
//先获取url中的token
var accessToken=this.query.accessToken
if (!accessToken) {
accessToken=this.request.body.accessToken
}
if (!accessToken) {
this.body={
success:false,
err:'token不能为空!'
}
return next
}
//再查询用户,如果查询不到用户,说明token错误,或者用户没注册
var user=yield User.findOne({
accessToken:accessToken
}).exec()
if (!user) {
this.body={
success:false,
err:'token错误!'
}
return next
}
this.session=this.session||{}
this.session.user=user
yield next
}

然后在router中配置,中间件

1
2
3
4
router.post('/u/signup',App.hasBody,User.signup)
router.post('/u/verify',App.hasBody,User.verify)
//登陆才需要token
router.post('/u/update',App.hasBody,App.hasToken,User.update)

接入七牛云

先创建一个七牛空间,然后在程序里加入qiniu的配置

1
2
3
4
qiniu:{
'AK':'cMoZ5Ql3sJZ2ZReLyQDjNbIqiaNmAzn54QDzZE_J',
'SK':'8MCooB-Kb8s9lCAQbhLFwUukmJJiCEn-w483ainh'
}

卸载模块,并删除依赖

1
npm uninstall uuid --save

接入cloudinary

1
npm i cloudinary --save

将上传到七牛的视频同步到cloudinary

将视频上传到七牛是因为,访问七牛快,同步到cloudinary是因为他有将视频和音频合成一个视频的功能。

同步的逻辑很简单,就是客户端将视频上传到了七牛,然后将视频信息发送到我们的服务端,并保持,然后异步同步七牛的视频到cloudinary,同步完了,将他返回的信息在存储到我们的服务器

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
var body=this.request.body
var videoData=body.video
var user=this.session.user
console.log('create video')
console.log(videoData)
if (!videoData||!videoData.key) {
this.body={
success:false,
err:'视频没有上传成功'
}
return next
}
//用七牛key查询是否有视频存在
var video=yield Video.findOne({
qiniu_key:videoData.key
})
if (!video) {
//创建一个新视频
video=new Video({
author:user._id,
qiniu_key:videoData.key,
persistentId:videoData.persistentId
})
video=yield video.save()
console.log('new video save')
console.log(video)
}
//将生成的静音视频,同步到cloudinary,然后使用它的音频合成服务
var url=config.qiniu.video+video.qiniu_key
robot.uploadToCloudinary(url)
.then(function (data) {
if (data&&data.public_id) {
video.public_id=data.public_id
video.detail=data
video.save()
}
})
this.body={
success:true,
data:video._id
}

在这里就是用了cloudinary的sdk

1
2
3
4
5
6
7
8
9
10
11
12
13
14
exports.uploadToCloudinary=function (url) {
return new Promise(function (resolve,reject) {
cloudinary.uploader.upload(url,function (result) {
if (result&&result.public_id) {
resolve(result)
} else{
reject(result)
}
},{
resource_type:'video',
folder:'video'
})
})
}

屏幕单位

1英寸(inch)=2.54厘米(cm)

首先我们看iphone6

英寸

像素

缩放比ppi

总结

iphone屏幕尺寸分布

箭头函数

这里的箭头函数相当于java中的函数式编程

1
2
3
4
5
6
7
8
9
10
var numberArray=[1,2,3]
// var newNumberArray=numberArray.map(function(item){
// return item+1
// })
//等效于
var newNumberArray = numberArray.map(item=>item+1)
console.log(newNumberArray)

常见错误

RCTSRWebSocket.m在xcode8.1下报错

http://www.jianshu.com/p/175e820a1c51

CodeSign error: code signing is required for product type ‘Unit Test Bundle’ in SDK ‘iOS 8.1’

http://blog.csdn.net/skymingst/article/details/42489097

任苹蜻 wechat
欢迎您扫一扫上面的微信公众号,订阅我的博客!
坚持原创技术分享,您的支持将鼓励我继续创作!

热评文章