React Native填坑之旅 -- FlatList

在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,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

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,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设置初始值。

下拉刷新的话,每次都会清空已经存在的数据,并在之后给他设置为获得的第一页(或者)最新的数据,所以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,refreshing: false});
    });
  }

在返回的数据转化为json格式之后,合成data。这个时候refreshing和loading都已经完成,值都设置为false。数据是累加的:data: [...this.state.data,,所以每次在下拉刷新的时候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>
  );
}

希望这些对你们有用。

相关文章

react 中的高阶组件主要是对于 hooks 之前的类组件来说的,如...
我们上一节了解了组件的更新机制,但是只是停留在表层上,例...
我们上一节了解了 react 的虚拟 dom 的格式,如何把虚拟 dom...
react 本身提供了克隆组件的方法,但是平时开发中可能很少使...
mobx 是一个简单可扩展的状态管理库,中文官网链接。小编在接...
我们在平常的开发中不可避免的会有很多列表渲染逻辑,在 pc ...