cubic-bezier 模拟实现

cubic-bezier 曲线css3 动画的一个重要基石。另一个为 steps (ease 等都是 cubic-bezier 的特殊形式),css3 中的 cubic_bezier 曲线限制了首尾两控制点的位置,通过调整中间两控制点的位置可以灵活得到常用的动画效果,同时 canvas 也进行了相应的支持,也存在相应的工具可以根据想要的曲线得到对应 cubic bezier 曲线的控制点参数。

 

而 ie(6-9) 却没有相应的支持,为了能在各个平台得到一致的动画效果,则不可避免要在 ie 上通过定时器沿着指定控制点参数的 cubic bezier 曲线来手动更新动画对象的数值.

模拟实现

公式

cubic-bezier 公式不是简单的 y= x 公式,而是引入了第三个变量 t,由于动画中关键在于计算比例,即在总时间的某个时间点百分比得到相应的值相对于最终值的比例,那么只需要得到 0,1 区间的曲线即可。 而 [x,y] -> [0,1] 内的曲线则是通过 t 在 0,1 内连续变化而得到:

 

 

 

其中 P0, P1 ,P2, P3 都为两维 xy 向量

 

将向量拆开表示即为:

 

y= (1-t)^3*p0y + 3*(1-t)^2*t*p1y + 3*(1-t)*t^2p2y + t^3p3y


x= (1-t)^3*p0x + 3*(1-t)^2*t*p1x + 3*(1-t)*t^2p2x + t^3p3x
  

而 css3 所用的 cubic bezier 已经限定死 p0 = (0,0) , p3= (1,1) ,因此公式可简化为

 

var ax = 3 * p1x - 3 * p2x + 1,
      bx = 3 * p2x - 6 * p1x,
      cx = 3 * p1x;

var ay = 3 * p1y - 3 * p2y + 1,
      by = 3 * p2y - 6 * p1y,
      cy = 3 * p1y;

y= ((ay * t + by) * t + cy ) * t

x= ((ax * t + bx) * t + cx ) * t
  

为了提高效率以及减少计算精度丢失公式进一步经过了 Horner 's method 变化。

计算

css3 中某个限定了特定控制参数的 cubic -bezier 曲线如下所示:

 

 

动画所做的事情就是把 x 轴当做时间比例,根据曲线得到 y 轴对应的值,并更新到动画对象中去.

即转化为以下问题:如何根据上述公式在已知 x 的情况下如何得到 y.

求 t

首先需要根据公式

 

var ax = 3 * p1x - 3 * p2x + 1,
      bx = 3 * p2x - 6 * p1x,
      cx = 3 * p1x;

x= ((ax * t + bx) * t + cx ) * t
  

在已知 x 的情况下求 t,即经典的多项式求参问题,首先可以通过 newton method 尝试求出 t 的值,若不行(可能性很小)则可通过可靠但慢速的二分法求值.

求 y

上步得到 t 后则可以带入另一个 y 公式求得最终值 y

 

var ay = 3 * p1y - 3 * p2y + 1,
      by = 3 * p2y - 6 * p1y,
      cy = 3 * p1y;

y= ((ay * t + by) * t + cy ) * t
  

上述解法也是源自 webkit webcore c++ 原生实现.

使用对比

在传入动画的 easing 设置时,可以传入 css3 cubic-bezier 的语法格式或者直接传入特定的曲线设置(ease-in ease-out).

 

$('#xx').animate({
   left:500
},{
  duration: 2,
  easing: 'cubic-bezier(1,0.22,0,0.84)' // 'ease-in'
});

 

效果对比:

 

cubic-bezier in kissy

 

通过对比即可发现,ease-out 和以前 js 实现的简单二次曲线 easeOut 还是有明显的不同,并且 js 实现和 css3 原生几乎效果完全一样(效率可能稍微低了些),更多自带曲线对比可见:

 

easing for kissy

 

 

 

 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值