在React Native裏有不少種方法來建立可滾動的list。好比,ScrollView和ListView。他們都各有優缺點。可是在React Native 0.43裏增長了兩種行的list view。一個是FlatList
, 一個是SectionList
。今天咱們就來詳細瞭解一下FlatList
。react
若是你熟悉RN以前的ListView的話你會發現FlatList的API更加的簡單,只須要給它一列數據,而後根據每一項數據繪製行就能夠。git
源代碼在github上。代碼中使用的是RN 0.49.5。github
基本上你只要給FlatList
的兩個props指定值就能夠了,一個是data
,一個是renderItem
。數據源通常就是一個數組,而renderItem
就是每一行的繪製方法。繪製行的時候只須要獲取當前的數據項就能夠。json
正式開始以前,咱們看下代碼是什麼樣子的。後端
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方法裏就能夠寫繪製的代碼了:react-native
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
組件。這個組件會根據傳入的數據呈現不一樣的內容。api
React Native爲了很快的達到重繪改變了的一組組件,規定要給這一組組件裏的每個都設置一個key。FlatList的每一行也都須要一個key。數組
咱們能夠直接設置一個key。好比,每一個元素的返回json裏都有一個id
屬性,正好就能夠用來做爲每一行的key值。FlatList還有另外的一個設置方式.使用keyExtractor
。網絡
render() { return ( <View style={styles.container}> <Text>Message</Text> <FlatList ... keyExtractor={item => item.id} /> </View> );
咱們的APP自己在顯示message的時候沒有明顯的分割線,而是用一塊一塊的方式顯示的。若是隻是簡單的一條線分割兩行,那麼只須要設置行組件的boderBottom
相關的屬性就能夠了。
如設置行組件的borderBottom:fetch
<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不置空。
這個很是的簡單,只要直接看代碼就能夠明白了。和使用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> ); }
但願這些對大家有用。