在React Native里有很多種方法來創建可滾動的list。比如,ScrollView和ListView。他們都各有優缺點。但是在React Native 0.43里增加了兩種行的list view。一個是FlatList
, 一個是SectionList
。今天我們就來詳細了解一下FlatList
。
如果你熟悉RN之前的ListView的話你會發現FlatList的API更加的簡單,只需要給它一列數據,然后根據每一項數據繪制行就可以。
源代碼在github上。代碼中使用的是RN 0.49.5。
基本使用方法
基本上你只要給FlatList
的兩個props指定值就可以了,一個是data
,一個是renderItem
。數據源一般就是一個數組,而renderItem
就是每一行的繪制方法。繪制行的時候只需要獲取當前的數據項就可以。
正式開始之前,我們看下代碼是什么樣子的。
import React from 'react';
import {
View,
Text,
FlatList,
Dimensions,
} from 'react-native';
import MessageCell from './MessageCell';
const { width, height } = Dimensions.get('window');
const SCREEN_WIDTH = width;
export default class MessageContainer extends React.Component {
constructor(props) {
super(props);
this.state = {
error: false,
page: 1,
refreshing: false,
loading: false,
data: {},
};
}
componentDidMount() {
this.requestData();
}
requestData = () => {
const url = 'Some rest api url address';
fetch(url).then(res => {
return res.json()
}).then(res => {
this.setState({
data: [...this.state.data, ...res],
});
}).catch(err => {
this.setState({ error: err, loading: false, refreshing: false});
});
};
render() {
return (
<View style={{ flex: 1, justifyContent: 'center', alignItems: 'stretch', backgroundColor: 'white' }}>
<Text>Message</Text>
<FlatList
data={[{ key: 'a' }, { key: 'b' }, { key: 'c' }, { key: 'd' }]}
renderItem={({ item }) => (
<MessageCell item={item} />
)} />
</View>
);
}
}
首先import必要的組件:import { FlatList } from 'react-native';
。當然還有其他的一些組件。render方法里就可以寫繪制的代碼了:
render() {
return (
<View style={...}>
<Text>Message</Text>
<FlatList
data={this.state.data}
renderItem={({ item }) => (
<MessageCell item={item} />
)} />
</View>
);
}
```=
`data`是從github的API上請求來的數據,json數據被解析之后填充到了`this.state.data`里。就像這樣:`data: [...this.state.data, ...res]`。每個元素是一個只有一個key鍵值的對象。`renderItem`方法里會根據每一個item返回一個`MessageCell`組件。這個組件會根據傳入的數據呈現不同的內容。
## 每行需要一個key
React Native為了很快的達到重繪改變了的一組組件,規定要給這一組組件里的每一個都設置一個**key**。FlatList的每一行也都需要一個key。
我們可以直接設置一個key。比如,每個元素的返回json里都有一個`id`屬性,正好就可以用來作為每一行的key值。FlatList還有另外的一個設置方式.使用`keyExtractor`。
```js
render() {
return (
<View style={styles.container}>
<Text>Message</Text>
<FlatList
...
keyExtractor={item => item.id} />
</View>
);
分割線 - seperator
我們的APP本身在顯示message的時候沒有明顯的分割線,而是用一塊一塊的方式顯示的。如果只是簡單的一條線分割兩行,那么只需要設置行組件的boderBottom
相關的屬性就可以了。
如設置行組件的borderBottom:
<View style={ borderTopWidth: 0, borderBottomWidth: 1, borderBottomColor: 'grey' }>
// content...
</View>
如果你一定要一個分割線的話可以使用FlatList的ItemSeperatorComponent
prop。如:
renderSeparator = () => {
return (
<View
style={{
height: 1,
width: "86%",
backgroundColor: "#CED0CE",
marginLeft: "14%"
}}
/>
);
};
使用seperator:
render() {
return (
<List containerStyle={{ borderTopWidth: 0, borderBottomWidth: 0 }}>
<FlatList
...
ItemSeparatorComponent={this.renderSeparator}
/>
</List>
);
}
在FlatList里使用prop ItemSeparatorComponent
就可以。
注意:list的頂部和底部的分割組件是不繪制的。
下拉刷新和上拉加載更多
自從這兩個交互的方式自從發明出來之后就基本上是每一個應用里list的標配了。我們來看看FlatList如何添加這兩個功能的。
render() {
return (
<View style={styles.container}>
<Text>Message</Text>
<FlatList
...
refreshing={this.state.refreshing}
onRefresh={this.handleRefresh}
onEndReached={this.handleLoadMore}
onEndReachedThreshold={0} />
</View>
);
}
FlatList的幾個props:
refreshing:表明list是否在refresh的狀態。
onRefresh:開始refresh的事件。在這個方法里開始設置refresh的時候組件的state,並在setState
方法的回調里開始請求后端的數據。
onEndReached: 上拉加載跟個多的事件。在這里設置加載更多對應的組件狀態,並在setState
方法的回調里請求后端數據。
onEndReachedThreshold:這個值是觸發onEndReached
方法的閾值。值是RN的邏輯像素。
下面看一下下拉刷新的方法。上拉加載更多基本類似,各位可以參考代碼。
handleRefresh = () => {
this.setState({
page: 1,
refreshing: true,
loading: false,
data: [],
}, () => {
this.requestData();
});
}
請求github的API的方法是:
requestData = () => {
const url = 'https://api.github.com/users/[your github name]/repos';
fetch(url).then(res => {
console.log('started fetch');
return res.json()
}).then(res => {
this.setState({
data: [...this.state.data, ...res],
error: res.error || null,
laoding: false,
refreshing: false,
});
}).catch(err => {
console.log('==> fetch error', err);
this.setState({ error: err, loading: false, refreshing: false});
});
}
在下拉刷新開始請求后端的數據的時候首先設置組件狀態。給組件的state設置初始值。
下拉刷新的話,每次都會清空已經存在的數據,並在之后給他設置為獲得的第一頁(或者)最新的數據,所以page:1
。接下來要開始刷新,那么表示刷新的小菊花就需要轉起來,所以refreshing的值設為true。loading在這個時候是不存在的,所以為false。
在setState
方法的回調里開始請求后端的數據。數據返回之后,下拉刷新或者加載更多的狀態都不存在。如果請求數據的時候有錯,那么我們要處理錯誤。所以秦秋網絡數據的方法為:
requestData = () => {
const url = 'https://api.github.com/users/futurechallenger/repos';
fetch(url).then(res => {
console.log('started fetch');
return res.json()
}).then(res => {
this.setState({
data: [...this.state.data, ...res],
error: res.error || null,
laoding: false,
refreshing: false,
});
}).catch(err => {
console.log('==> fetch error', err);
this.setState({ error: err, loading: false, refreshing: false});
});
}
在返回的數據轉化為json格式之后,合成data。這個時候refreshing和loading都已經完成,值都設置為false。數據是累加的:data: [...this.state.data, ...res],
,所以每次在下拉刷新的時候this.setState({data: []})
,在上拉加載更多的時候可以留着data不置空。
List的header和footer
這個非常的簡單,只要直接看代碼就可以明白了。和使用prop renderItem
一樣的,header和footer都有對應的prop來繪制。
// Header
renderHeader = () => {
return <SearchBar placeholder="Type Here..." lightTheme round />;
};
// Footer
renderFooter = () => {
if (!this.state.loading) return null;
return (
<View
style={{
paddingVertical: 20,
borderTopWidth: 1,
borderColor: "#CED0CE"
}}
>
<ActivityIndicator animating size="large" />
</View>
);
};
然后這么用:
render() {
return (
<List containerStyle={{ borderTopWidth: 0, borderBottomWidth: 0 }}>
<FlatList
...
ListHeaderComponent={this.renderHeader}
ListFooterComponent={this.renderFooter}
/>
</List>
);
}
希望這些對你們有用。