ReactNative之FlatList-列表入场动画

FlatList-列表入场动画

前言

网上查找资料:RN FlatList入场动画,竟然一个都没搜索到,而且还是百度+google双大法。最后没办法,只能自己仔细看Animated库怎么使用,各种属性都慢慢试了,才大致弄清楚动画的实现。先看实现的效果图:

效果图1

merge.gif

效果图2

mergeX.gif

效果图3

x.gif

<figcaption style="display: block; text-align: center; font-size: 1rem; line-height: 1.6; color: rgb(144, 144, 144); margin-top: 2px;"></figcaption>

项目中的效果图

project.gif

gif有点卡,真机运行是很流畅的。

动画基本介绍

ReactNative的view动画效果主要使用Animated库来进行制作,通过start和stop来进行控制动画的启动。animate封装了四个可以动画化的组件:View,Text,Image,ScrollView。也可以使用Animated.createAnimatedComponent()封装自己的动画组件。

简单的使用方式(中文网上的代码),淡入动画效果的视图:

import React from 'react';
import { Animated, Text, View } from 'react-native';

class FadeInView extends React.Component {
  state = {
    fadeAnim: new Animated.Value(0),  // 透明度初始值设为0
  }

  componentDidMount() {
    Animated.timing(                  // 随时间变化而执行动画
      this.state.fadeAnim,            // 动画中的变量值
      {
        toValue: 1,                   // 透明度最终变为1,即完全不透明
        duration: 10000,              // 让动画持续一段时间
      }
    ).start();                        // 开始执行动画
  }

  render() {
    let { fadeAnim } = this.state;

    return (
      <Animated.View                 // 使用专门的可动画化的View组件
        style={{
          ...this.props.style,
          opacity: fadeAnim,         // 将透明度指定为动画变量值
        }}
      >
        {this.props.children}
      </Animated.View>
    );
  }
}

// 然后你就可以在组件中像使用`View`那样去使用`FadeInView`了
export default class App extends React.Component {
  render() {
    return (
      <View style={{flex: 1, alignItems: 'center', justifyContent: 'center'}}>
        <FadeInView style={{width: 250, height: 50, backgroundColor: 'powderblue'}}>
          <Text style={{fontSize: 28, textAlign: 'center', margin: 10}}>Fading in</Text>
        </FadeInView>
      </View>
    )
  }
}
复制代码

Animated的动画基本配置方法有timing,spring,decay;

1、timing

它是线性动画,在设定时间内移动到终点,中间的动画可以设置

toValue:线性改变的数值

duration: 动画持续时间 默认500.

easing:默认的实现Easing.in和Easing.out和Easing.inOut。

delay: 动画延迟开始时间。
复制代码

简单的使用方式:

 Animated.timing(
        this.state.animatePress, {
            toValue: 0.8,
            duration: 200
        }
    ).start();
复制代码

2、spring

类似弹簧的实现,动画结束时会有个渐弱的摆动,它的属性在源码中可以看到有很多:

interface SpringAnimationConfig extends AnimationConfig {
    toValue: number | AnimatedValue | { x: number; y: number } | AnimatedValueXY;
    overshootClamping?: boolean;
    restDisplacementThreshold?: number;
    restSpeedThreshold?: number;
    velocity?: number | { x: number; y: number };
    bounciness?: number;
    speed?: number;
    tension?: number;
    friction?: number;
    stiffness?: number;
    mass?: number;
    damping?: number;
    delay?: number;
}
复制代码

常用的有: toValue:线性改变的数值 friction:弹簧界限值,越小弹的越大,默认7 tension:弹簧张力,默认40,越大进入的速度越快

简单的使用方式:

 Animated.spring(this.state.animateItem, {
                        toValue: 1,
                        velocity: 0.1,
                        bounciness: 5,
                        friction:5
                    }),
复制代码

3、decay

动画的速度逐渐变慢,最后停止,类似上面的转场动画

Animated的动画组合方式有:parallel,sequence,stagger,

parallel:并行运行动画

sequence:依次运行动画,如果一个动画被停止了,那么下一个动画就不会被执行。

stagger:每个动画延迟一段时间执行

这三个动画的基本入参是动画数组,stagger有额外的time(延时)参数。而且数组内仍然可以传入parallel,sequence,stagger等各种类型,安装规则进行动画播放。

栗子如下:

     //动画并行parallel
    Animated.parallel(
        [   
            //动画顺序执行
            Animated.sequence(
                [
                    Animated.delay(delay),
                    // Animated.spring(this.state.animateItem, {
                    //     toValue: 1,
                    //     velocity: 0.1,
                    //     bounciness: 5,
                    // }),
                    Animated.timing(this.state.animateItem, {
                        toValue: 1,
                        duration: 500,
                        // delay: delay
                    })
                ]
            ),
            Animated.sequence(
                [
                    Animated.delay(delay),
                    Animated.timing(this.state.animateOpacity, {
                        toValue: 1,
                        duration: 1000
                    })
                ]
            )

        ]
    ).start()
复制代码

效果图的实现

动画的基本介绍完成后,在看看效果图,其实就是动画简单的组合,使用了transform参数,另外加了个插值器。

核心代码:

 <Animated.View style={{
                backgroundColor: "transparent",
                margin: 5,
                transform: [
                    {
                        scale: this.state.animatePress
                    },
                    {
                        translateX: this.state.animateItem.interpolate({
                            inputRange: [0, 1],
                            outputRange: [700, 1]
                        })
                    }
                ]
            }}
复制代码

对FlatList的子Item进行封装,全部代码如下(其中一个组件的文件):

import React, { Component } from 'react';
import {
    Text,
    View,
    StyleSheet,
    Image,
    TouchableWithoutFeedback,
    Animated
} from 'react-native';

export default class ListItem extends Component {
    state = {
        animatePress: new Animated.Value(1),
        animateItem: new Animated.Value(0)
    }

    animateIn() {
        Animated.timing(
            this.state.animatePress, {
                toValue: 0.8,
                duration: 200
            }
        ).start();
    }

    animateOut() {
        Animated.timing(this.state.animatePress, {
            toValue: 1,
            duration: 200
        }).start();
    }

    componentWillMount() {
        const { index } = this.props;
        const delay = index * 300
        Animated.timing(this.state.animateItem, {
            toValue: 1,
            duration: 1000,
            delay: delay
        }).start();
    }

    render() {
        const { text,index } = this.props;

        return (
            <TouchableWithoutFeedback
                onPressIn={() => this.animateIn()}
                onPressOut={() => this.animateOut()}
                style={{
                    backgroundColor: "transparent",

                }}
            >
                <Animated.View style={{
                    backgroundColor: "transparent",
                    margin: 5,
                    transform: [
                        {
                            scale: this.state.animatePress
                        },
                        {
                            translateX: this.state.animateItem.interpolate({
                                inputRange: [0, 1],
                                outputRange: [700, 1]
                            })
                        }
                    ]
                }}
                >
                    <Text
                        style={{ width: 150, height: 100, backgroundColor: 'yellow', }}
                    >
                        {text}{index}
                    </Text>
                </Animated.View>

            </TouchableWithoutFeedback>
        );

    }

}
复制代码

Animated使用中遇到的问题

FlatList横向排列,使用translateX参数,进行子item的入参动画,从屏幕外进入到相应的位置。开发完成后,发现在ios表现正常,但是在android中存在问题,子item不会从屏幕外进入,而是在item的原先位置进行展示,看图:

IOS正常表现:

g3vhq.gif

<figcaption style="display: block; text-align: center; font-size: 1rem; line-height: 1.6; color: rgb(144, 144, 144); margin-top: 2px;"></figcaption>

android异常表现:

l8tic.gif

<figcaption style="display: block; text-align: center; font-size: 1rem; line-height: 1.6; color: rgb(144, 144, 144); margin-top: 2px;"></figcaption>

这个问题被折磨了2天时间,网上各种查找资料都没有,后来在startoverflow上发现一个和我遇到同样问题的人,但是并没有得到解决,地址如下:StackOverFlow上有人遇到同样的错误

后来突然想到FlatList有个优化属性:getItemLayou,当我加入后,android端就表现正常了,对FlatList添加的代码如下:

          getItemLayout={(data, index) => (
                { length: 200, offset: 200 * index, index }
          )}
复制代码

getItemLayou的作用是为了不让FlatList在渲染过程中对内容尺寸做动态计算,在该方法中给予他设定的高度,但是使用它的前提是知道Item的宽高,如果item的宽高是固定的,使用这个方法就会大大优化列表性能,如果数据少,可能不明显,但是数据量大的时候就会表现比较明显。

加入这个方法能够解决这个问题,但是我不知道是什么原因导致的这样,而且这个方法我这样写动画也能渲染正常:

getItemLayout={(data, index) => (
    { length: 1, offset: 1 , index }
  )}
复制代码

所以这个地方很纠结,不过这个bug在最新的版本0.59.5中已近解决了,不需要使用getItemLayout也能正常播放动画。

    入场动画bug版本:
    0.59.4版本及之前的版本都存在此bug,不过在0.59.5版本中修复了该问题。在package.json中查看版本号:

            "react-native": "0.59.4",
            //该版本已修复入场动画的bug
            "react-native": "0.59.5"
复制代码
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352