React Native填坑之旅 -- FlatList
在React Native里有很多种方法来创建可滚动的list。比如,ScrollView和ListView。他们都各有优缺点。但是在React Native 0.43里增加了两种行的list view。一个是 如果你熟悉RN之前的ListView的话你会发现FlatList的API更加的简单,只需要给它一列数据,然后根据每一项数据绘制行就可以。 源代码在github上。代码中使用的是RN 0.49.5。 基本使用方法基本上你只要给 正式开始之前,我们看下代码是什么样子的。 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,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必要的组件: render() { return ( <View style={...}> <Text>Message</Text> <FlatList data={this.state.data} renderItem={({ item }) => ( <MessageCell item={item} /> )} /> </View> ); }
每行需要一个keyReact Native为了很快的达到重绘改变了的一组组件,规定要给这一组组件里的每一个都设置一个key。FlatList的每一行也都需要一个key。 我们可以直接设置一个key。比如,每个元素的返回json里都有一个 render() { return ( <View style={styles.container}> <Text>Message</Text> <FlatList ... keyExtractor={item => item.id} /> </View> ); 分割线 - seperator我们的APP本身在显示message的时候没有明显的分割线,而是用一块一块的方式显示的。如果只是简单的一条线分割两行,那么只需要设置行组件的 <View style={ borderTopWidth: 0,borderBottomWidth: 1,borderBottomColor: 'grey' }> // content... </View> 如果你一定要一个分割线的话可以使用FlatList的 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 注意: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: 下面看一下下拉刷新的方法。上拉加载更多基本类似,各位可以参考代码。 handleRefresh = () => { this.setState({ page: 1,refreshing: true,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,error: res.error || null,laoding: false,}); }).catch(err => { console.log('==> fetch error',err); this.setState({ error: err,refreshing: false}); }); } 在下拉刷新开始请求后端的数据的时候首先设置组件状态。给组件的state设置初始值。 下拉刷新的话,每次都会清空已经存在的数据,并在之后给他设置为获得的第一页(或者)最新的数据,所以 在 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,refreshing: false}); }); } 在返回的数据转化为json格式之后,合成data。这个时候refreshing和loading都已经完成,值都设置为false。数据是累加的: List的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> ); } 希望这些对你们有用。 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |