当数学遇上动画(2)

671 查看

当数学遇上动画:讲述 ValueAnimator、TypeEvaluator和TimeInterpolator之间的恩恩怨怨(2)

上一节的结论是,ValueAnimator就是由TimeInterpolatorTypeEvaluator这两个简单函数组合而成的一个复合函数。如下图所示:

上一节我们还将TimeInterpolatorTypeEvaluator看作是工厂流水线上的两个小员工,那么ValueAnimator就是车间主管,TimeInterpolator这个小员工面对的是产品的半成品,他负责控制半成品输出到下一个生产线的速度,而下一个生产线上的小员工TypeEvaluator的任务就是打磨半成品得到成品,最后将成品输出。

本小节进一步深究TimeInterpolatorTypeEvaluator在动画实现过程中承担的作用以及它们之间的联系与差异。

还是先说结论,借助TimeInterpolator或者TypeEvaluator“单独”来控制动画所产生的动画效果殊途同归!

1 两种特殊情况下的ValueAnimator

(1)上一节提到过,假设TimeInterpolatorLinearInterpolator(线性插值器,f(t)=t),也就是说时间比率不被“篡改”的话,那么ValueAnimator对应的函数其实就简化成了TypeEvaluator函数(F=g(x,a,b)=g(f(t),a,b)=g(t,a,b)),即动画实际上只由TypeEvaluator来控制。

这里可以理解为,TimeInterpolator这个员工请假了,但是工厂为了不停止生产安排了一个自动机器人代替他的工作,它只会匀速地将半成品输入到下一个生产线。

(2)同理,我们假设TypeEvaluator“LinearTypeEvaluator”(线性估值器,并没有这个说法,所以加上引号,计算方式就是g(x,a,b)=a+x*(b-a))的话,那么ValueAnimator对应的函数也可以简化,F=g(x,a,b)=g(f(t),a,b)=a+f(t)*(b-a),即动画实际上只由TimeInterpolator来控制。

同样的,这里可以理解为,TypeEvaluator这个员工请假了,默认也有个自动机器人采用默认的操作将半成品加工成最终成品输出。

(3)综上所述,我们来思考上一节留下的问题,即TimeInterpolatorTypeEvaluator到底啥关系?
其实TimeInterpolator是用来控制动画速度的,而TypeEvaluator是用来控制动画中值的变化曲线的。
虽然它们本质的作用是不同的,但是它们两个既可以联手来控制动画,也可以"单独"来控制动画(并非真的单独,而是另一方有个默认操作)。

单独控制动画的典型例子就是上一节提到的EaseInterpolatorAnimationEasingFunctions,这两个项目对于制作动画起到殊途同归作用。

为什么说TimeInterpolatorTypeEvaluator对于制作动画有着殊途同归的作用呢?

不难想象,在某些定制的情况下,上面两种特殊情况下的构造出来的ValueAnimator所产生的动画效果是一样的!那如何来验证我们的这个结论呢?我们可以通过构造两个不同的特殊情况下的ValueAnimator来验证。

下面的代码显示了两个ValueAnimator,都是在1s中内将float类型的数值从0变化到1。第一个ValueAnimator使用的是LinearInterpolator和自定义的TypeEvaluator,第二个ValueAnimator使用的是自定义的TimeInterpolator"LinearTypeEvaluator"。打印输出的是两个ValueAnimator每次值变化的时候的大小。

打印输出的结果如下图所示,从图中可以看出,两个ValueAnimator的在真实时间序列中的输出结果是一样的,也就说明如果将它们作用在同一个View组件的某个属性上的话,那么产生的动画效果是完全一样的。例如,可以将两个ValueAnimator改成ObjectAnimator,并将其作用在两个不同的TextView的translationY属性上,你可以看到一样的动画效果。所以说,在特殊的单独控制动画的情况下,TimeInterpolatorTypeEvaluator对于制作动画有着殊途同归的作用。(注意结论的前提,那就是在我们理解了ValueAnimator内部动画原理之后自己定制的一些特殊情况,它们并非总是能够产生一样的动画效果)

2 简单动画实例分析:弹跳!

经过前面的分析,我们差不多理解了ValueAnimator是怎么借助TimeInterpolatorTypeEvaluator来实现动画的了。在实现动画的时候,为了简便,我们常常可以选择将TimeInterpolator设置为LinearInterpolator或者将TypeEvaluator设置为"LinearTypeEvaluator"这两种特殊的方式。

举个栗子!假设我们要来实现弹跳的动画效果。首先我们要确定一个弹跳效果的函数曲线,自己想不太好想,我们先来看看项目EaseInterpolator中的EaseBounceOutInterpolator内部表示的函数曲线的形态。如下图所示,它是一个分段函数,每个段内都是一个简单的二次曲线。如果将这个曲线作用在View组件的translationY属性上,那么组件将在垂直方向上来回地跳动从而就形成了弹跳的效果。

我们先看下EaseBounceOutInterpolator的核心方法getInterpolation的实现,它其实就是刻画了上面的函数曲线。