React Native支持任意组件实现下拉刷新功能,并且可以自定义下拉刷新头部

1.背景

无论是 Androi 还是 ios,下拉刷新都是一个很有必要也很重要的功能。那么在 RN(以下用 RN 表示 React Native )之中,我们该如何实现下拉刷新功能呢?RN 官方提供了一个用于 ScrollView , ListView 等带有滑动功能组件的下拉刷新组件 RefreshControl。查看 RefreshControl 相关源码可以发现,其实它是对原生下拉刷新组件的一个封装,好处是使用方便快捷。但缺点也很明显,就是它不可以进行自定义下拉刷新头部,并且只能使用与 ScrollView,ListView 这种带有滚动功能的组件之中。那么我们该如何去解决这两个问题呢?
先看下最终实现的效果,这里借助了 ScrollableTabView

React Native支持任意组件实现下拉刷新功能,并且可以自定义下拉刷新头部_第1张图片
ios.gif

android.gif

2.实现原理分析

对于下拉刷新功能,其实它的原理很简单。就是对要操作的组件进行 y 轴方向的位置进行判断。当滚动到顶部的时候,此时如果下拉的话,那么就进行下拉刷新的操作,如果上拉的话,那么就进行原本组件的滚动操作。基于这个原理,找了一些第三方实现的框架,基本上实现方式都是通过 ScrollView,ListView 等的 onScroll 方法进行监听回调。然后设置 Enable 属性来控制其是否可以滚动。但在使用的过程中有两个问题,一个是 onScroll 回调的频率不够,很多时候在滚动到了顶部的时候不能正确回调数值。另外一个问题就是 Enable 属性的问题,当在修改 Enable 数值的时候,当前的手势操作会停止。具体反映到 UI 上的效果就是,完成一次下拉刷新之后,第一次向上滚动的效果不能触发。那么,能不能有其他的方式去实现 RN 上的下拉刷新呢?

3.实现过程

3.1 判断组件的滚动位置

在上面的原理分析中,一个重点就是判断要操作的组件的滚动位置,那么改如何去判断呢?在这里我们对 RN 的 View,ScrollView,ListView,FlatList 进行了相关的判断,不过要注意的是,FlatList 是 RN0.43 版本之后才出现的,所以如果你使用的 RN 版本小于 0.43 的话,那么你就要删除掉该下拉刷新框架关于 FlatList 的部分。
我们来看下如何进行相关的判断。

 onShouldSetPanResponder = (e, gesture) => {
        let y = 0
        if (this.scroll instanceof ListView) { //ListView下的判断
            y = this.scroll.scrollProperties.offset;
        } else if (this.scroll instanceof FlatList) {//FlatList下的判断
            y = this.scroll.getScrollMetrics().offset  //这个方法需要自己去源码里面添加
        }
        //根据y的值来判断是否到达顶部
        this.state.atTop = (y <= 0)
        if (this.state.atTop && index.isDownGesture(gesture.dx, gesture.dy) && this.props.refreshable) {
            this.lastY = this.state.pullPan.y._value;
            return true;
        }
        return false;
    }

首先对于普通的 View,由于它没有滚动属性,所以它默认处于顶部。而对于 ListView 来说,通过查找它的源码,发现它有个 scrollProperties 属性,里面包含了一些滚动的属性值,而 scrollProperties.offset 就是表示横向或者纵向的滚动值。而对于 FlatList 而言,它并没相关的属性。但是发现 VirtualizedList 中存在如下属性,而 FlatList 是对 VirtualizedList 的一个封装

 _scrollMetrics = {
        visibleLength: 0, contentLength: 0, offset: 0, dt: 10, velocity: 0, timestamp: 0,
    };

那么很容易想到自己添加方法去获取。那么在
FlatList(node_modules/react-native/Libraries/Lists/FlatList.js) 添加如下方法

getScrollMetrics = () => {
    return this._listRef.getScrollMetrics()
}

同时在 VirtualizedList(node_modules/react-native/Libraries/Lists/VirtualizedList.js) 添加如下方法

getScrollMetrics = () => {
    return this._scrollMetrics
 }

另外,对于 ScrollView 而言,并没有找到相关滚动位置的属性,所以在这里用 ListView 配合 ScrollView 来使用,将 ScrollView 作为
ListView 的一个子控件

//ScrollView 暂时没有找到比较好的方法去判断时候滚动到顶部,
//所以这里用ListView配合ScrollView进行使用
export default  class PullScrollView extends Pullable {
    getScrollable=()=> {
        return (
             {this.scroll = c;}}
                renderRow={this.renderRow}
                dataSource={new ListView.DataSource({rowHasChanged: (r1, r2) => r1 !== r2}).cloneWithRows([])}
                enableEmptySections={true}
                renderHeader={this._renderHeader}/>
        );
    }

    renderRow = (rowData, sectionID, rowID, highlightRow) => {
        return 
    }

    _renderHeader = () => {
        return (
            
                {this.props.children}
            
        )
    }
}

那么当要操作的组件滚动到顶部的时候,此时下拉就是下拉刷新操作,而上拉就实现原本的操作逻辑

3.2 组件位置的布局控制

下拉刷新的滚动方式一般有两种,一种是内容跟随下拉头部一起下拉滚动,一种是内容固定不动,只有下拉头部在滚动。在这里用isContentScroll属性来进行选择判断

render() {
        return (
            
                {this.props.isContentScroll ?
                    
                        
                            {this.renderTopIndicator()}
                             {this.scrollContainer = c;}}
                                  style={{width: this.state.width, height: this.state.height}}>
                                {this.getScrollable()}
                            
                        
                     :
                    
                         {this.scrollContainer = c;}}
                              style={{width: this.state.width, height: this.state.height}}>
                            {this.getScrollable()}
                        
                        
                            
                                {this.renderTopIndicator()}
                            
                        
                    }
            
        );
    }

从里面可以看到一个方法 this.getScrollable() , 这个就是我们要进行下拉刷新的内容,这个方法类似我们在 java 中的抽象方法,是一定要实现的,并且操作的内容的要指定 ref 为 this.scroll,举个例子

export default class PullView extends Pullable {

    getScrollable = () => {
        return (
             {this.scroll = c;}}
                {...this.props}>
                {this.props.children}
            
        );
    }
}

3.3 添加默认刷新头部

这里我们添加个默认的下拉刷新头部,用于当不添加下拉刷新头部时候的默认的显示

defaultTopIndicatorRender = () => {
        return (
            
                
                 {
                    this.txtPulling = c;
                }} style={styles.hide}>{index.pulling}
                 {
                    this.txtPullok = c;
                }} style={styles.hide}>{index.pullok}
                 {
                    this.txtPullrelease = c;
                }} style={styles.hide}>{index.pullrelease}
            
        );
    }

效果就是上面的 gif 中除了 View 的 tab 的展示效果,同时需要根据下拉的状态来进行头部效果的切换

 if (this.pullSatte == "pulling") {
                this.txtPulling && this.txtPulling.setNativeProps({style: styles.show});
                this.txtPullok && this.txtPullok.setNativeProps({style: styles.hide});
                this.txtPullrelease && this.txtPullrelease.setNativeProps({style: styles.hide});
            } else if (this.pullSatte == "pullok") {
                this.txtPulling && this.txtPulling.setNativeProps({style: styles.hide});
                this.txtPullok && this.txtPullok.setNativeProps({style: styles.show});
                this.txtPullrelease && this.txtPullrelease.setNativeProps({style: styles.hide});
            } else if (this.pullSatte == "pullrelease") {
                this.txtPulling && this.txtPulling.setNativeProps({style: styles.hide});
                this.txtPullok && this.txtPullok.setNativeProps({style: styles.hide});
                this.txtPullrelease && this.txtPullrelease.setNativeProps({style: styles.show});
            }
const styles = StyleSheet.create({
    wrap: {
        flex: 1,
        flexGrow: 1,
        zIndex: -999,
    },
    hide: {
        position: 'absolute',
        left: 10000,
        backgroundColor: 'transparent'
    },
    show: {
        position: 'relative',
        left: 0,
        backgroundColor: 'transparent'
    }
});

这里借助 setNativeProps 方法来代替 setStat e的使用,减少 render 的次数

3.4 下拉刷新手势控制

在下拉刷新之中,手势的控制是必不可少的一环,至于如何为组件添加手势,大家可以看下 RN 官网上的介绍

this.panResponder = PanResponder.create({
            onStartShouldSetPanResponder: this.onShouldSetPanResponder,
            onStartShouldSetPanResponderCapture: this.onShouldSetPanResponder,
            onMoveShouldSetPanResponder: this.onShouldSetPanResponder,
            onMoveShouldSetPanResponderCapture: this.onShouldSetPanResponder,
            onPanResponderTerminationRequest: (evt, gestureState) => false, //这个很重要,这边不放权
            onPanResponderMove: this.onPanResponderMove,
            onPanResponderRelease: this.onPanResponderRelease,
            onPanResponderTerminate: this.onPanResponderRelease,
        });

这里比较重要的一点就是 onPanResponderTerminationRequest (有其他组件请求使用手势),这个时候不能将手势控制交出去

onShouldSetPanResponder = (e, gesture) => {
        let y = 0
        if (this.scroll instanceof ListView) { //ListView下的判断
            y = this.scroll.scrollProperties.offset;
        } else if (this.scroll instanceof FlatList) {//FlatList下的判断
            y = this.scroll.getScrollMetrics().offset  //这个方法需要自己去源码里面添加
        }
        //根据y的值来判断是否到达顶部
        this.state.atTop = (y <= 0)
        if (this.state.atTop && index.isDownGesture(gesture.dx, gesture.dy) && this.props.refreshable) {
            this.lastY = this.state.pullPan.y._value;
            return true;
        }
        return false;
    }

onShouldSetPanResponder方法主要是对当前是否进行下拉操作进行判断。下拉的前提是内容滚动到顶部,下拉手势并且该内容需要下拉刷新操作( refreshable 属性)

onPanResponderMove = (e, gesture) => {
        if (index.isDownGesture(gesture.dx, gesture.dy) && this.props.refreshable) { //下拉
            this.state.pullPan.setValue({x: this.defaultXY.x, y: this.lastY + gesture.dy / 2});
            this.onPullStateChange(gesture.dy)
        }
    }
 //下拉的时候根据高度进行对应的操作
    onPullStateChange = (moveHeight) => {
        //因为返回的moveHeight单位是px,所以要将this.topIndicatorHeight转化为px进行计算
        let topHeight = index.dip2px(this.topIndicatorHeight)
        if (moveHeight > 0 && moveHeight < topHeight) { //此时是下拉没有到位的状态
            this.pullSatte = "pulling"
        } else if (moveHeight >= topHeight) { //下拉刷新到位
            this.pullSatte = "pullok"
        } else { //下拉刷新释放,此时返回的值为-1
            this.pullSatte = "pullrelease"
        }

        if (this.props.topIndicatorRender == null) { //没有就自己来
            if (this.pullSatte == "pulling") {
                this.txtPulling && this.txtPulling.setNativeProps({style: styles.show});
                this.txtPullok && this.txtPullok.setNativeProps({style: styles.hide});
                this.txtPullrelease && this.txtPullrelease.setNativeProps({style: styles.hide});
            } else if (this.pullSatte == "pullok") {
                this.txtPulling && this.txtPulling.setNativeProps({style: styles.hide});
                this.txtPullok && this.txtPullok.setNativeProps({style: styles.show});
                this.txtPullrelease && this.txtPullrelease.setNativeProps({style: styles.hide});
            } else if (this.pullSatte == "pullrelease") {
                this.txtPulling && this.txtPulling.setNativeProps({style: styles.hide});
                this.txtPullok && this.txtPullok.setNativeProps({style: styles.hide});
                this.txtPullrelease && this.txtPullrelease.setNativeProps({style: styles.show});
            }
        }
        //告诉外界是否要锁住
        this.props.onPushing && this.props.onPushing(this.pullSatte != "pullrelease")
        //进行状态和下拉距离的回调
        this.props.onPullStateChangeHeight && this.props.onPullStateChangeHeight(
            this.pullSatte == "pulling", this.pullSatte == "pullok",
            this.pullSatte == "pullrelease", moveHeight)
    }

onPanResponderMove 方法中主要是对下拉时候头部组件 UI 进行判断,这里有三个状态的判断以及下拉距离的回调

 onPanResponderRelease = (e, gesture) => {
        if (this.pullSatte == 'pulling') { //没有下拉到位
            this.resetDefaultXYHandler(); //重置状态
        } else if (this.pullSatte == 'pullok') { //已经下拉到位了
            //传入-1,表示此时进行的是释放刷新的操作
            this.onPullStateChange(-1)
            //进行下拉刷新的回调
            this.props.onPullRelease && this.props.onPullRelease();
            //重置刷新的头部到初始位置
            Animated.timing(this.state.pullPan, {
                toValue: {x: 0, y: 0},
                easing: Easing.linear,
                duration: this.duration
            }).start();
        }
    }
 //重置刷新的操作
    resetDefaultXYHandler = () => {
        Animated.timing(this.state.pullPan, {
            toValue: this.defaultXY,
            easing: Easing.linear,
            duration: this.duration
        }).start(() => {
            //ui要进行刷新
            this.onPullStateChange(-1)
        });
    }

onPanResponderRelease 方法中主要是下拉刷新完成或者下拉刷新中断时候对头部 UI 的一个重置,并且有相关的回调操作

4.属性和方法介绍

4.1 属性

Porp Type Optional Default Description
refreshable bool yes true 是否需要下拉刷新功能
isContentScroll bool yes false 在下拉的时候内容时候要一起跟着滚动
onPullRelease func yes 刷新的回调
topIndicatorRender func yes 下拉刷新头部的样式,当它为空的时候就使用默认的
topIndicatorHeight number yes 下拉刷新头部的高度,当topIndicatorRender不为空的时候要设置正确的topIndicatorHeight
onPullStateChangeHeight func yes 下拉时候的回调,主要是刷新的状态的下拉的距离
onPushing func yes 下拉时候的回调,告诉外界此时是否在下拉刷新

4.2 方法

startRefresh() : 手动调用下拉刷新功能
finishRefresh() : 结束下拉刷新

5.最后

该组件已经发布到 npm 仓库,使用的时候只需要 npm install react-native-rk-pull-to-refresh --save 就可以了,同时需要 react-native link react-native-rk-pull-to-refresh,它的使用Demo已经上传Github了:https://github.com/hzl123456/react-native-rk-pull-to-refresh
另外:在使用过程中不要设置内容组件 Bounce 相关的属性为 false ,例如:ScrollView 的 bounces 属性( ios 特有)

6.更新与2018年1月9日

在使用的过程中,发现在 Android 中使用的过程中经常会出现下拉无法触发下拉刷新的问题,所以 Android 的下拉刷新采用原生组件封装的形式。对 android-Ultra-Pull-To-Refresh 进行封装。调用主要如下

'use strict';
import React from 'react';
import RefreshLayout from '../view/RefreshLayout'
import RefreshHeader from '../view/RefreshHeader'
import PullRoot from './PullRoot'
import * as index from './info';

export default class Pullable extends PullRoot {

    constructor(props) {
        super(props);
        this.pullState = 'pulling'; //pulling,pullok,pullrelease
        this.topIndicatorHeight = this.props.topIndicatorHeight ? this.props.topIndicatorHeight : index.defaultTopIndicatorHeight;
    }

    render() {
        return (
             this.refresh = c}>

                 this.onPushingState(e)}>
                    {this.renderTopIndicator()}
                

                {this.getScrollable()}
            
        )
    }


    onPushingState = (event) => {
        let moveHeight = event.nativeEvent.moveHeight
        let state = event.nativeEvent.state
        //因为返回的moveHeight单位是px,所以要将this.topIndicatorHeight转化为px进行计算
        let topHeight = index.dip2px(this.topIndicatorHeight)
        if (moveHeight > 0 && moveHeight < topHeight) { //此时是下拉没有到位的状态
            this.pullState = "pulling"
        } else if (moveHeight >= topHeight) { //下拉刷新到位
            this.pullState = "pullok"
        } else { //下拉刷新释放,此时返回的值为-1
            this.pullState = "pullrelease"
        }
        //此时处于刷新中的状态
        if (state == 3) {
            this.pullState = "pullrelease"
        }
        //默认的设置
        this.defaultTopSetting()
        //告诉外界是否要锁住
        this.props.onPushing && this.props.onPushing(this.pullState != "pullrelease")
        //进行状态和下拉距离的回调
        this.props.onPullStateChangeHeight && this.props.onPullStateChangeHeight(this.pullState, moveHeight)
    }

    finishRefresh = () => {
        this.refresh && this.refresh.finishRefresh()
    }

    startRefresh = () => {
        this.refresh && this.refresh.startRefresh()
    }
}

同时修改了主动调用下拉刷新的的方法为 startRefresh() , 结束刷新的方法为 finishRefresh() , 其他的使用方式和方法没有修改

7.更新于2018年5月14日

由于 React Native 版本的更新,移除了 React.PropTypes ,更新了 PropTypes 的引入方式,改动如下(基于 RN 0.55.4 版本):
1.使用 import PropTypes from 'prop-types' 引入 PropTypes
2.修改 FlatList 滑动距离的判断,这样你就不需要再修改源码了

let y = 0
if (this.scroll instanceof ListView) { //ListView下的判断
     y = this.scroll.scrollProperties.offset;
} else if (this.scroll instanceof FlatList) {//FlatList下的判断
    y = this.scroll._listRef._getScrollMetrics().offset
}

8.更新于2019年2月15日

最近升级了 React Native 到 0.58.1 版本,发现 android 的下拉刷新头部无法隐藏,一直显示在最顶端,排查 RN 的源码发现。

  public ReactViewGroup(Context context) {
    super(context);
    setClipChildren(false);
    mDrawingOrderHelper = new ViewGroupDrawingOrderHelper(this);
  }

ReactViewGroup 默认调用了setClipChildren(false)方法,这样子 View 将可以超出父 View 的布局范围,也就导致了我们的下拉刷新头部无法隐藏的问题。修改如下:

//设置所有的parent的clip属性为true,为了兼容RN的view默认为false的bug
        setViewClipChildren(getParent());
private void setViewClipChildren(ViewParent rootView) {
        if (rootView != null && rootView instanceof ViewGroup) {
            ViewGroup viewGroup = ((ViewGroup) rootView);
            viewGroup.setClipChildren(true);
            setViewClipChildren(viewGroup.getParent());
        }
    }

在 onFinishInflate() 的最后调用 setViewClipChildren(getParent()) 方法,修改下拉刷新控件的所有父 View 的 clipChildren 属性为 true,可以解决这个 bug。

你可能感兴趣的:(React Native支持任意组件实现下拉刷新功能,并且可以自定义下拉刷新头部)