CSS3动画那么强,requestAnimationFrame还有毛线用?

这篇文章发布于 2013年09月30日,星期一,19:12,归类于 Web综合。 阅读 433964 次, 今日 5 次 113 条评论

 

一、哟,requestAnimationFrame, 新同学,先自我介绍下

大家好
Hello, 大家好,我就是风姿卓越,万种迷人的requestAnimationFrame,呵呵呵呵。很高兴和大家见面,请多指教!

平静的众人
。。。。

小丸子无奈
咳咳,大家不要一副不屑的样子嘛。跟你讲,我可是很有用的。所谓人如其名,看我名字这么长,表意为“请求动画帧”,明摆着一副很屌的样子!

平静的众人
。。。。按照这种说法,“樱桃小丸子”就是樱桃做的丸子咯,恩,看脑袋确实蛮像的~

樱桃小丸子汗
。。。。

想到明天就是国庆大假,今天我就小人不记大人过。给你们来副震精的图:

帧丢失示意

相当一部分的浏览器的显示频率是16.7ms, 就是上图第一行的节奏,表现就是“我和你一步两步三步四步往前走……”。如果我们火力搞猛一点,例如搞个10ms setTimeout,就会是下面一行的模样——每第三个图形都无法绘制(红色箭头指示),表现就是“我和你一步两步 坑 四步往前走……”。

囊爹(なんでよ)?

国庆北京高速,最多每16.7s通过一辆车,结果,突然插入一批setTimeout的军车,强行要10s通过。显然,这是超负荷的,要想顺利进行,只能让第三辆车直接消失(正如显示绘制第三帧的丢失)。然,这是不现实的,于是就有了会堵车!

同样的,显示器16.7ms刷新间隔之前发生了其他绘制请求(setTimeout),导致所有第三帧丢失,继而导致动画断续显示(堵车的感觉),这就是过度绘制带来的问题。不仅如此,这种计时器频率的降低也会对电池使用寿命造成负面影响,并会降低其他应用的性能。

这也是为何setTimeout的定时器值推荐最小使用16.7ms的原因(16.7 = 1000 / 60, 即每秒60帧)。

而我requestAnimationFrame就是为了这个而出现的。我所做的事情很简单,跟着浏览器的绘制走,如果浏览设备绘制间隔是16.7ms,那我就这个间隔绘制;如果浏览设备绘制间隔是10ms, 我就10ms绘制。这样就不会存在过度绘制的问题,动画不会掉帧,自然流畅的说~~

内部是这么运作的:
浏览器(如页面)每次要洗澡(重绘),就会通知我(requestAnimationFrame):小丸子,我要洗澡了,你可以跟我一起洗哦!

这是资源非常高效的一种利用方式。怎么讲呢?

  1. 就算很多个小丸子要一起洗澡,浏览器只要通知一次就可以了。而setTimeout貌似是多个独立绘制。
  2. 页面最小化了,或者被Tab切换关灯了。页面是不会洗澡的,自然,小丸子也不会洗澡的(没通知啊)。页面绘制全部停止,资源高效利用。

肿么样?requestAnimationFrame桑麻我果然是万千迷人的吧!!

不错哦,四基佬表情
耶!果然有料,不是看上去的平板电脑。

花痴 小丸子
唷~夸得人家都不好意思了!

平静的众人
。。。。那你的兼容性如何

我啊 小丸子
我的兼容性啊~~ 孬,见下面~

requestAnimationFrame的兼容性表 张鑫旭-鑫空间-鑫生活

平静的众人
Android设备压根就不支持嘛!其他设备基本上跟CSS3动画的支持一模一样嘛。

我说小美女,据我所知,CSS3 transitionanimation动画也是走的跟你一样的绘制原理(补充于2013-10-09:根据自己后来的测试,发现,CSS3动画在Tab切换回来的时候,动画表现并不暂停;通过Chrome frames工具测试发现,Tab切换之后,计算渲染绘制都停止,Tab切换回来时似乎通过内置JS计算了动画位置实现重绘,造成动画不暂停的感觉)。但是人家的实现轻松很多啊,而且相当强大,那你还有个毛线用!你该想想你一直鲜有人问津的原因了!

二、反击吧,requestAnimationFrame同学

反击
基佬们,你们的眼界太狭隘了,观点太肤浅的。首先从哲学宏观讲,事物存在必有其道理。因此,本大人肯定是有价值的。

平静的众人
那你到说说你有纳尼价值~~~

反击
准备好了木有,待会儿说出来吓死你们。听好了,足足有3大点:
1. 统一的向下兼容策略
虽说CSS3实现动画即高效又方便,但是对于PC浏览器,IE8, IE9之流,你想兼容实现某些动画效果,比方说淡入淡出,敢问,你怎么实现?

看你们那副呆若木鸡的表情就知道了,IE10+ CSS3实现,IE9-之流JS setTimeout实现,我说累不累啊,两套完全不同的style. 你改下动画时间是不是要改两处?但是我requestAnimationFramesetTimeout非常类似,都是单回调,用法也类似。

var handle = setTimeout(renderLoop, PERIOD);
var handle = requestAnimationFrame(renderLoop);

requestAnimationFrame调用一次只会重绘一次动画,因此,如果想要实现联系动画,就使用renderLoop反复蹂躏我吧~

受虐

So,如果想要简单的兼容,可以这样子:

window.requestAnimFrame = (function(){
  return  window.requestAnimationFrame       ||
          window.webkitRequestAnimationFrame ||
          window.mozRequestAnimationFrame    ||
          function( callback ){
            window.setTimeout(callback, 1000 / 60);
          };
})();

但是呢,并不是所有设备的绘制时间间隔是1000/60 ms, 以及上面并木有cancel相关方法,所以,就有下面这份更全面的兼容方法:

(function() {
    var lastTime = 0;
    var vendors = ['webkit', 'moz'];
    for(var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) {
        window.requestAnimationFrame = window[vendors[x] + 'RequestAnimationFrame'];
        window.cancelAnimationFrame = window[vendors[x] + 'CancelAnimationFrame'] ||    // Webkit中此取消方法的名字变了
                                      window[vendors[x] + 'CancelRequestAnimationFrame'];
    }

    if (!window.requestAnimationFrame) {
        window.requestAnimationFrame = function(callback, element) {
            var currTime = new Date().getTime();
            var timeToCall = Math.max(0, 16.7 - (currTime - lastTime));
            var id = window.setTimeout(function() {
                callback(currTime + timeToCall);
            }, timeToCall);
            lastTime = currTime + timeToCall;
            return id;
        };
    }
    if (!window.cancelAnimationFrame) {
        window.cancelAnimationFrame = function(id) {
            clearTimeout(id);
        };
    }
}());

上JS可点击这里下载。

然后,我们就可以以使用setTimeout的调调使用requestAnimationFrame方法啦,IE6也能支持哦!

不错哦,四基佬表情
耶耶耶!……

承蒙赏识 小丸子
多谢捧场!(*^__^*) 嘻嘻……

2. CSS3动画不能应用所有属性
使用CSS3动画可以改变高宽,方位,角度,透明度等等。但是,就像六道带土也有弱点一样,CSS3动画也有属性鞭长莫及。比方说scrollTop值。如果我们希望返回顶部是个平滑滚动效果,就目前而言,CSS3似乎是无能为力的。此时,还是要JS出马,势必,我requestAnimationFrame大人就可以大放异彩,万众瞩目啦,哈哈哈哈哈哈哈~~

哈哈哈哈~~

比方说点下面这个按钮,滚上去再滚下来,哈哈~


不错哦,四基佬表情
哟哟哟,不错不错,刮目相看啊,丸子~~

承蒙赏识 小丸子
人家不禁夸的啦~~

3. CSS3支持的动画效果有限
由于CSS3动画的贝塞尔曲线是一个标准3次方曲线(详见:贝塞尔曲线与CSS3动画、SVG和canvas的基情),因此,只能是:Linear, Sine, Quad, Cubic, Expo等,但对于Back, Bounce等缓动则只可观望而不可亵玩焉。

下面这张图瞅瞅,那些波澜壮阔的曲线都是CSS3木有的~~

动画曲线大全 张鑫旭-鑫空间-鑫生活

咋办,咋办?只能是JS实现啦,于是,本大人我requestAnimationFrame可以再一次大放异彩啦,啊哈哈哈!

平静的众人
得意的太早了吧,这些动画曲线看上去很复杂,偶们显然驾驭不了了。

了不起 丸子
就知道你们这些基佬中看不中用。先给大家普及下缓动(Tween)知识吧:

  • Linear:无缓动效果
  • Quadratic:二次方的缓动(t^2)
  • Cubic:三次方的缓动(t^3)
  • Quartic:四次方的缓动(t^4)
  • Quintic:五次方的缓动(t^5)
  • Sinusoidal:正弦曲线的缓动(sin(t))
  • Exponential:指数曲线的缓动(2^t)
  • Circular:圆形曲线的缓动(sqrt(1-t^2))
  • Elastic:指数衰减的正弦曲线缓动
  • 超过范围的三次方缓动((s+1)*t^3 – s*t^2)
  • 指数衰减的反弹缓动

每个效果都分三个缓动方式,分别是(可采用后面的邪恶记忆法帮助记忆):

  • easeIn:从0开始加速的缓动,想象OOXX进去,探路要花时间,因此肯定是先慢后快的;
  • easeOut:减速到0的缓动,想象OOXX出来,肯定定先快后慢的,以防掉出来;
  • easeInOut:前半段从0开始加速,后半段减速到0的缓动,想象OOXX进进出出,先慢后快然后再慢。

每周动画效果都有其自身的算法。我们都知道jQuery UI中就有缓动,As脚本也内置了缓动,其中的运动算法都是一致的。我特意弄了一份,哦呵呵呵~~(因为较高,滚动显示),或GitHub访问

/*
 * Tween.js
 * t: current time(当前时间)
 * b: beginning value(初始值)
 * c: change in value(变化量)
 * d: duration(持续时间)
*/
var Tween = {
    Linear: function(t, b, c, d) { return c*t/d + b; },
    Quad: {
        easeIn: function(t, b, c, d) {
            return c * (t /= d) * t + b;
        },
        easeOut: function(t, b, c, d) {
            return -c *(t /= d)*(t-2) + b;
        },
        easeInOut: function(t, b, c, d) {
            if ((t /= d / 2) < 1) return c / 2 * t * t + b;
            return -c / 2 * ((--t) * (t-2) - 1) + b;
        }
    },
    Cubic: {
        easeIn: function(t, b, c, d) {
            return c * (t /= d) * t * t + b;
        },
        easeOut: function(t, b, c, d) {
            return c * ((t = t/d - 1) * t * t + 1) + b;
        },
        easeInOut: function(t, b, c, d) {
            if ((t /= d / 2) < 1) return c / 2 * t * t*t + b;
            return c / 2*((t -= 2) * t * t + 2) + b;
        }
    },
    Quart: {
        easeIn: function(t, b, c, d) {
            return c * (t /= d) * t * t*t + b;
        },
        easeOut: function(t, b, c, d) {
            return -c * ((t = t/d - 1) * t * t*t - 1) + b;
        },
        easeInOut: function(t, b, c, d) {
            if ((t /= d / 2) < 1) return c / 2 * t * t * t * t + b;
            return -c / 2 * ((t -= 2) * t * t*t - 2) + b;
        }
    },
    Quint: {
        easeIn: function(t, b, c, d) {
            return c * (t /= d) * t * t * t * t + b;
        },
        easeOut: function(t, b, c, d) {
            return c * ((t = t/d - 1) * t * t * t * t + 1) + b;
        },
        easeInOut: function(t, b, c, d) {
            if ((t /= d / 2) < 1) return c / 2 * t * t * t * t * t + b;
            return c / 2*((t -= 2) * t * t * t * t + 2) + b;
        }
    },
    Sine: {
        easeIn: function(t, b, c, d) {
            return -c * Math.cos(t/d * (Math.PI/2)) + c + b;
        },
        easeOut: function(t, b, c, d) {
            return c * Math.sin(t/d * (Math.PI/2)) + b;
        },
        easeInOut: function(t, b, c, d) {
            return -c / 2 * (Math.cos(Math.PI * t/d) - 1) + b;
        }
    },
    Expo: {
        easeIn: function(t, b, c, d) {
            return (t==0) ? b : c * Math.pow(2, 10 * (t/d - 1)) + b;
        },
        easeOut: function(t, b, c, d) {
            return (t==d) ? b + c : c * (-Math.pow(2, -10 * t/d) + 1) + b;
        },
        easeInOut: function(t, b, c, d) {
            if (t==0) return b;
            if (t==d) return b+c;
            if ((t /= d / 2) < 1) return c / 2 * Math.pow(2, 10 * (t - 1)) + b;
            return c / 2 * (-Math.pow(2, -10 * --t) + 2) + b;
        }
    },
    Circ: {
        easeIn: function(t, b, c, d) {
            return -c * (Math.sqrt(1 - (t /= d) * t) - 1) + b;
        },
        easeOut: function(t, b, c, d) {
            return c * Math.sqrt(1 - (t = t/d - 1) * t) + b;
        },
        easeInOut: function(t, b, c, d) {
            if ((t /= d / 2) < 1) return -c / 2 * (Math.sqrt(1 - t * t) - 1) + b;
            return c / 2 * (Math.sqrt(1 - (t -= 2) * t) + 1) + b;
        }
    },
    Elastic: {
        easeIn: function(t, b, c, d, a, p) {
            var s;
            if (t==0) return b;
            if ((t /= d) == 1) return b + c;
            if (typeof p == "undefined") p = d * .3;
            if (!a || a < Math.abs(c)) {
                s = p / 4;
                a = c;
            } else {
                s = p / (2 * Math.PI) * Math.asin(c / a);
            }
            return -(a * Math.pow(2, 10 * (t -= 1)) * Math.sin((t * d - s) * (2 * Math.PI) / p)) + b;
        },
        easeOut: function(t, b, c, d, a, p) {
            var s;
            if (t==0) return b;
            if ((t /= d) == 1) return b + c;
            if (typeof p == "undefined") p = d * .3;
            if (!a || a < Math.abs(c)) {
                a = c; 
                s = p / 4;
            } else {
                s = p/(2*Math.PI) * Math.asin(c/a);
            }
            return (a * Math.pow(2, -10 * t) * Math.sin((t * d - s) * (2 * Math.PI) / p) + c + b);
        },
        easeInOut: function(t, b, c, d, a, p) {
            var s;
            if (t==0) return b;
            if ((t /= d / 2) == 2) return b+c;
            if (typeof p == "undefined") p = d * (.3 * 1.5);
            if (!a || a < Math.abs(c)) {
                a = c; 
                s = p / 4;
            } else {
                s = p / (2  *Math.PI) * Math.asin(c / a);
            }
            if (t < 1) return -.5 * (a * Math.pow(2, 10* (t -=1 )) * Math.sin((t * d - s) * (2 * Math.PI) / p)) + b;
            return a * Math.pow(2, -10 * (t -= 1)) * Math.sin((t * d - s) * (2 * Math.PI) / p ) * .5 + c + b;
        }
    },
    Back: {
        easeIn: function(t, b, c, d, s) {
            if (typeof s == "undefined") s = 1.70158;
            return c * (t /= d) * t * ((s + 1) * t - s) + b;
        },
        easeOut: function(t, b, c, d, s) {
            if (typeof s == "undefined") s = 1.70158;
            return c * ((t = t/d - 1) * t * ((s + 1) * t + s) + 1) + b;
        },
        easeInOut: function(t, b, c, d, s) {
            if (typeof s == "undefined") s = 1.70158; 
            if ((t /= d / 2) < 1) return c / 2 * (t * t * (((s *= (1.525)) + 1) * t - s)) + b;
            return c / 2*((t -= 2) * t * (((s *= (1.525)) + 1) * t + s) + 2) + b;
        }
    },
    Bounce: {
        easeIn: function(t, b, c, d) {
            return c - Tween.Bounce.easeOut(d-t, 0, c, d) + b;
        },
        easeOut: function(t, b, c, d) {
            if ((t /= d) < (1 / 2.75)) {
                return c * (7.5625 * t * t) + b;
            } else if (t < (2 / 2.75)) {
                return c * (7.5625 * (t -= (1.5 / 2.75)) * t + .75) + b;
            } else if (t < (2.5 / 2.75)) {
                return c * (7.5625 * (t -= (2.25 / 2.75)) * t + .9375) + b;
            } else {
                return c * (7.5625 * (t -= (2.625 / 2.75)) * t + .984375) + b;
            }
        },
        easeInOut: function(t, b, c, d) {
            if (t < d / 2) {
                return Tween.Bounce.easeIn(t * 2, 0, c, d) * .5 + b;
            } else {
                return Tween.Bounce.easeOut(t * 2 - d, 0, c, d) * .5 + c * .5 + b;
            }
        }
    }
}
Math.tween = Tween;

于是,借助大人我requestAnimationFrame以及上面的动画算法,各种动画效果所向披靡了,哈哈哈哈!!

不错哦,四基佬表情
哇哦哇哦,大赞!可不可以给我们这些鼠辈开开眼界?

了不起 丸子
明天放假,老娘心情好,就给你们露一手,实现个铁球落下的效果。

您可以狠狠地点击这里:requestAnimationFrame+Tween缓动小球落地效果

截图如下:
小球截图 张鑫旭-鑫空间-鑫生活

相关源代码可以参见demo页面源代码——直接右键即可以。核心动画部分的脚本是:

funFall = function() {
    var start = 0, during = 100;
    var _run = function() {
         start++;
         var top = Tween.Bounce.easeOut(start, objBall.top, 500 - objBall.top, during);
         ball.css("top", top);
         shadowWithBall(top);    // 投影跟随小球的动
         if (start < during) requestAnimationFrame(_run);
    };
    _run();
};

太棒了 基佬们
我靠!太劲爆了!太带感了!丸子殿下,你太牛逼啦!!

耶 小丸子
嘻嘻,小意思啦,诸位!最后,预祝大家节日快乐!!

补充于2014-02-08
新年伊始,根据这篇翻译文章一些测试说法,FireFox/Chrome浏览器对setInterval, setTimeout做了优化,页面处于闲置状态的时候,如果定时间隔小于1秒钟(1000ms),则停止了定时器。与requestAnimationFrame有类似行为。但如果时间间隔大于或等于1000ms,定时器依然执行,即使页面最小化或非激活状态。

参见下表:

setInterval requestAnimationFrame
IE 无影响 暂停
Safari 无影响 暂停
Firefox >=1s 1s - 3s
Chrome >=1s 暂停
Opera 无影响 暂停

相关链接

  • http://easings.net/zh-cn
  • http://www.jiawin.com/requestanimationframe-animation-windmill/
  • http://msdn.microsoft.com/zh-cn/library/ie/hh920765(v=vs.85).aspx
  • http://www.paulirish.com/2011/requestanimationframe-for-smart-animating/
  • http://caniuse.com/#search=requestAnimationFrame
  • https://github.com/zhangxinxu/Tween/blob/master/tween.js
  • http://www.cnblogs.com/cloudgamer/archive/2009/01/06/tween.html
  • http://blog.segmentfault.com/humphry/1190000000386368

(本篇完)

分享到:


发表评论(目前113 条评论)

  1. naruto说道:

    我遇到了requestAnimationFrame()在安卓手机上面卡顿的问题,请问大神有什么什么好的解决办法,是我的用法问题还是requestAnimationFrame()本身不兼容安卓的问题

    • naruto说道:

      另外,我是用的touchstart事件触发的requestAnimationFrame(),不知道是不是因为这个问题

  2. 今天刚好用到这个,学习一下。

  3. assface说道:

    妹子好萌

  4. ivorzk说道:

    直接说重点不就好了,扯半天,想看小说一样,还带图……..

  5. jasonxxp说道:

    你好~~请问你知道Elastic 函数的 a 和 p 表示什么意思吗?我找了很久也没找到~~谢谢~~

  6. 飞常快乐说道:

    居然自称老娘,以为是个污污的女生,结果唉。。,有点小失望,不过还是要膜拜下大牛

  7. 小苏说道:

    html5的一切性能都是硬伤。
    在此,送给html5一首歌:
    千万里我追寻着你 可是你却并不在意 你不象是在我梦里 在梦里你是我的唯一 time and time again you ask me 问我到底爱不爱你 time and time again i ask my self 问自己是否离的开你 我今生看来注定要独行 热情已被你耗尽 我已经变的不再是我 可是你却依然是你 问我到底恨不恨你 问自己你到底好在哪里 好在那里

  8. 小苏说道:

    看着花哨。测过性能以后,你们就哭了

  9. kk说道:

    jq3已经用requestAnimationFrame了,看3.0新特性搜到这里,没想到大师13年就研究过此物了。。。另拜服tween.js

  10. 方贤正说道:

    很好的内容,就是太污了,哈哈哈

  11. darr说道:

    浏览器(如页面)每次要洗澡(重绘),就会通知我(requestAnimationFrame)。我想请教下,这里的浏览器每次重绘是指什么触发的重绘呢?

  12. 不死粗人说道:

    广告实在是太多了,另外改版也太难看了些

  13. jxyy说道:

    哈哈哈哈哈哈哈哈,好喜欢博主的风格

  14. kad0108说道:

    scroll那个在火狐浏览器里好使,需要使用document.documentElement.scrollTop

  15. bnm114说道:

    不错,解决了我的问题
    谢谢博主

  16. jone说道:

    好多废话,读起来很烦,博主写文章就不能言简意赅,一语中的?

  17. kenzi说道:

    haha,看到那么多小丸子,我还想是不是女生,结果偶然看到了你的照片。突然发现很面熟,你就是慕课网上的深入理解系列的。。。讲师。艾玛,看你的东西都笑死了,你说巧不巧。

  18. 前端路人说道:

    里面掺杂了那么多废话,读起来好痛苦。。。

  19. sikaco说道:

    CSS3有个cubic-bezier属性,可以详细自定义ease function,这里该改一下啦。

  20. Mask说道:

    XXOO那段太邪恶了~
    不过我喜欢。

  21. 淡然说道:

    大神,你太邪恶了

  22. 小生佩服说道:

    楼主厉害,喜欢这种风格

  23. Reeoo说道:

    我说博主,你写博的时候,脑子里是不是有两个小人在对话的啊?

  24. 祁磊说道:

    闲话真多,技术人员看着蛋疼的很。

  25. 彭尼玛说道:

    楼主患有严重的精神分裂症,鉴定完毕

  26. superchangme说道:

    张哥,你的缓动函数我试了一下, 对于初始值大于终点值得,结论是错的,为什么呢
    #################console##########
    Math.tween.Linear(0,0,10,10);
    0
    Math.tween.Linear(1,0,10,10);
    1
    Math.tween.Linear(2,0,10,10);
    2
    Math.tween.Sine.easeOut(1, 100,50,10);
    107.82172325201154
    Math.tween.Sine.easeOut(2, 100,50,10);
    115.45084971874736

  27. linger说道:

    您好,求助一下。请问ie的msRequestAnimationFrame有没有对应的停止方法?类似cancelAnimationFrame这样的方法。因为需要可控制动画开始和停止。谢谢

  28. 幸运猴子说道:

    仔细研究了一下,requestAnimationFrame这个方式应该是说高速动效才应该选择吧,普通setTimeout其实应该够用了。

  29. 小菜说道:

    做手机端动画还是用css3好是吗

  30. 水中月明说道:

    window.mozCancelAnimationFrame 这个现在火狐也是支持的,所以感觉
    var vendors = [‘webkit’, ‘moz’];
    for(var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) {
    window.requestAnimationFrame = window[vendors[x] + 'RequestAnimationFrame'];
    window.cancelAnimationFrame = window[vendors[x] + 'CancelAnimationFrame'] ||
    window[vendors[x] + 'CancelRequestAnimationFrame'];
    }

    可以改成

    var vendors = ['webkit', 'moz', 'moz'];
    for(var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) {
    window.requestAnimationFrame = window[vendors[x] + 'RequestAnimationFrame'];
    window.cancelAnimationFrame = window[vendors[x] + 'CancelAnimationFrame'] ||
    window[vendors[x] + 'CancelAnimationFrame'] ||
    window[vendors[x] + 'CancelRequestAnimationFrame'];
    }

  31. 深蓝说道:

    关于那段兼容处理的代码requestAnimationFrame传入element参数 没有被用到,是不是多余了?而且callBack中传入下运行时间戳也没有意义啊?这是我的困惑,请博主解答!

  32. wisodmoon说道:

    helle,可以加一下好友么?有个问题需要请教

  33. qping说道:

    有的部分好没有节操。。。不过写的很棒!!

  34. AJ说道:

    厉害,简简单单一篇文章,信息量很大!!

  35. 李荣平说道:

    楼主,你太有才了

  36. 佳佳君说道:

    博主的篇篇文章都充满了内涵。。。

  37. lync说道:

    我爸看报纸都会画出错别字哈:
    “…
    1. 统一的向下兼容策略

    因此,如果想要实现联系动画,就使用renderLoop…”

  38. humphry说道:

    关于requestAnimationFrame是否暂停,见我翻译的文章:http://blog.segmentfault.com/humphry/1190000000386368

  39. bean说道:

    楼主好像对css3动画曲线有些误解,上面贴的那张“css3木有的波澜壮阔的曲线图”里面除了最下面一排是多阶bezier外,上面都是三阶或者更低阶的,现有css3 transition的timeing-function完全可以做到,只是需要自定义两个控制点。

    • 张 鑫旭说道:

      @bean 难道我不是这个意思吗?[\抠鼻]

      • bean说道:

        额,好吧, 是我理解错了 🙂
        话说回来,css3只提供了3阶bezier估计也是考虑到3阶足够web端的需求了,而没必要像工业设计那样追求精密曲线,像楼主设计的铁球落下这样的场景的确是比较适合用requestAnimationFrame。

        另外,由于RAF使用的是js执行线程,因此在动画过程中的中断(比如播放动画时手指触摸屏幕或者鼠标点击停止播放)很方便,而如果是css3,由于不是同一个线程,执行中断比较麻烦。

  40. macroth说道:

    噢噢噢噢超直观这个!!
    setTimeout做大范围滑动在chrome似乎会卡顿 用requestAnimationFrame就没问题的样子!!

  41. zgyunfan说道:

    问题:FireFox24,那个滚上去再滚下来的按钮无效,chrome测试有效
    还有,为什么提交评论没有Ctrl+Enter快捷键呢?

  42. zgyunfan说道:

    问了不相干的问题:博主日语过几级?

  43. 博主好厉害。文章生动有趣,真正的寓教于乐啊

  44. chjund说道:

    那个链接现在报405 Not Allowed
    百度不靠谱啊

  45. Herrington Darkholme说道:

    还有一些用处
    比方,渐现一个display:none的对话框,使用transition: opacity。
    display:none -> display:block这个过程需要浏览器渲染,将对话框重新放回dom流。
    由于一开始元素不在dom内,改动opacity实际上是无效的。这时transition的动画效果就显现不出来了(@keyframe可以用delay,不知道能否实现,anyway就改一个opacity还要加一堆@keyframe实在不值)
    这时用raf就能侦听到display:block这个渲染事件,在放回dom,渲染得到opacity:0之后,再改opacity:1。

    还有在重绘非常重的事件中,raf也会有用。比方onscroll事件。如果直接将改变dom的行为绑定在scroll上,会使得dom改变非常频繁,重绘回流也多。那么就可以设置一个状态变量,在特定条件下用raf改变dom(比方,脱下网页200px的那个点)

    扯得更远一点……其实requestAnimationFrame和node里面的process.nextTick有相近之处。前者是在浏览器改变画面时候触发,后者是在服务器发生IO流时触发…

  46. light说道:

    requestAnimationFrame+Tween缓动小球落地效果
    链接忘加http://

  47. Bob说道:

    O(∩_∩)O,节日快乐。
    放假也不忘来看你的博客。
    最后那个DEMO地址忘记加http://了吧,变成相对地址了。

  48. XXX说道:

    鑫神,requestAnimationFrame+Tween缓动小球落地效果页面404了~

  49. woween说道:

    学习了,好文,赞一个

  50. hheedat说道:

    大哥啊,那个外链404了 = =!