经典动画

话不多说,首先来个经典的动画函数:

function animate(element, name, from, to, time) {     time = time || 800; //默认0.8秒      var style = element.style,         latency = 60, // 每60ms一次变化         count = time / latency, //变化的次数         step = Math.round((to - from) / count), //每一步的变化量         now = from;      function go() {         count--;         now = count ? now + step : to;         style[name] = now + 'px';          if (count) {             setTimeout(go, latency);         }     }      style[name] = from + 'px';     setTimeout(go, latency); } 

姑且不论这个函数的设计存在局限性,如只能对以px为单位的样式进行修改。仅从函数的实现上来看,这可以是一个非常经典的动画理念,其基本逻辑由以下部分组成:

  1. 获取起点值from和终点值to,通过动画需要进行的时间time,以及每侦间隔latency的要求,计算出值的改变次数count和每次改变的量step
  2. 开启setTimeout(fn, latency);来步进到下一侦。
  3. 在下一侦中,设置属性步进一次,如果动画还没结束,再回到第2步继续下一侦。

这个函数工作得很好,服务了千千万万的站点和系统,事实上jQuery的animate函数的核心也无非是setInterval函数。

但是,随着现在系统复杂度的稳步上升,动画效果也越来越多,同时对动画的流畅度也有了更多的重视,这导致上面的函数会出现一些问题。例如同时打开 100个动画效果,根据上面的函数,很明显会有100个定时器在同时运行,这些定时器之间的调度会对性能有轻微的影响。虽然在正常的环境中,这些许的影响 并不会有什么关系,但是在动画这种对流畅度有很高要求的环境下,任何细微的影响都可能产生出不好的用户体验。

在这样的情况下,有一些开发者就发明了一种基于统一帧管理的动画框架,他使用一个定时器触发动画帧,不同的动画来注册这些帧,在每一帧上处理多个动 画的属性变化。这样的好处是减少了定时器调度的开销,但是对于动画框架的开发者来说,统一帧管理、提供监听帧的API等,都是需要开发和维护的。

浏览器的直接支持

最终,浏览器厂商们发现这件事其实可以由他们来做,并且基于浏览器层面,还可以有更多的优化,比如:

  • 对于一个侦中对DOM的所有操作,只进行一次Layout和Paint。
  • 如果发生动画的元素被隐藏了,那么就不再去Paint。

于是,浏览器开始推出一个API,叫做requestAnimationFrame,关于这个函数,MDC的相关页面有比较详细的介绍,简单来说,这个函数有2种使用方法:

  1. 调用requestAnimationFrame函数,传递一个callback参数,则在下一个动画帧时,会调用callback。
  2. 不传递参数地直接调用该函数,启动动画帧,下一个帧触发时,会同时触发window.onmozbeforepaint事件,可以通过注册该事件来进行动画。

第2种方法由于依赖于Firefox自己的事件,且beforepaint事件还没进入到标准中,所以不推荐使用,还是使用第1种方式比较好。此时,我们的动画逻辑可以变成这样:

  1. 记录当前时间startTime,作为动画开始的时间。
  2. 请求下一帧,带上回调函数。
  3. 下一帧触发时,回调函数的第一个参数为当前的时间,再与startTime进行比较,确定时间间隔ellapseTime
  4. 判断ellapseTime是否已经超过事先设定的动画时间time,如果超过,则结束动画。
  5. 计算动画属性变化的差值differ = to - from,再确定在ellapseTime的时候应该变化多少step = differ / time * ellapseTime
  6. 计算出现在应该变化到的位置Math.round(from + step),并重新对样式赋值。
  7. 继续请求下一帧。

新的动画函数

下面就是一个全新的动画函数:

function animate(element, name, from, to, time) {     time = time || 800; // 默认0.8秒     var style = element.style,         startTime = new Date;      function go(timestamp) {         var progress = timestamp - startTime;         if (progress >= duration) {             style[name] = to + 'px';             return;         }          var now = (to - from) * (progress / duration);         style[name] = now.toFixed() + 'px';         requestAnimationFrame(go);     }      style[name] = from + 'px';      requestAnimationFrame(go); } 

到这一步,还剩一个问题,那就是并不是每个浏览器都支持requestAnimationFrame函数的,所以再做一个简单的修正。

根据Firefox的特性来看,其mozRequestAnimationFrame提供的最高FPS为60,并且会根据每一帧的计算的耗时来进行调整,比如每一帧计算用了1s,那他只会提供1FPS的动画效果。

而Chrome的高版本同样也实现了这个函数,叫webkitRequestAnimationFrame,可以预见未来还会有Opera的oRequestAnimationFrame和IE的msRequestAnimationFrame,所以这里一并做一个简单的兼容处理:

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

转载于:https://blog.51cto.com/kuiba/830814

动画函数requestAnimationFrame相关推荐

  1. webvr动画函数requestAnimationFrame

    在学习three.js绘制动画的时候接触到了动画函数requestAnimationFrame. 在HTML5/css3的时代我们要在web上写动画选择其实已经很多种了: 1.css3的animatt ...

  2. 移动端 transition动画函数的封装(仿Zepto)以及 requestAnimationFrame动画函数封装(仿jQuery)...

    移动端 css3 transition 动画 ,requestAnimationFrame 动画  对于性能的要求,h5优先考虑: 移动端 单页有时候 制作只用到简单的css3动画即可,我们封装一下, ...

  3. JS原生封装动画函数

    封装一个简单的原生匀速动画函数 /*** 匀速动画函数* @param {位移的元素} elm * @param {位移的目标位置} target */ function moveAnimation( ...

  4. 简单的动画函数封装(2)

    <div></div><!-- <span></span> --><button class="btn1"> ...

  5. 最终的动画函数封装(2)

    <button>点击触发1</button><button>点击触发2</button><div></div> <styl ...

  6. 简单的动画函数封装(1)

    //创建简单的动画函数封装效果(目标对象,目标位置) function animate(obj,target){var id = setInterval(function(){if(obj.offse ...

  7. javascript中BOM介绍、屏幕尺寸、历史记录、URL解析、计算机信息获取、定时器、三大系列及兼容代码、封装动画函数、同步和异步

    BOM介绍: BOM指的是浏览器对象模型,是用来操作浏览器的,例如浏览器弹窗.地址栏.滚动条等,浏览器顶级对象:window:页面中的所有内容都是属于window的,window可以省略:confir ...

  8. js进阶 13 jquery动画函数有哪些

    js进阶 13 jquery动画函数有哪些 一.总结 一句话总结: 二.jquery动画函数有哪些 原生JavaScript编写动画效果代码比较复杂,而且还需要考虑兼容性.通过jQuery,我们使用简 ...

  9. 动画函数,为任意一个元素移动到指定的目标位置

    一.动画缓冲函数 /*** 动画函数* 任意一个元素移动到指定的目标位置* @param {*} element 任意一个元素* @param {*} target 目标位置(number类型)*/ ...

最新文章

  1. Git使用教程与基本原理和Sourcetree基本使用探微
  2. JAVA程序运行原理分析
  3. Java开发中遇到具有挑战的事_Java并发编程的挑战:遇到的问题及如何解决
  4. 设置了li(float:right),里面的li反过来显示 - 解决办法
  5. 【2016年第5期】基于深度学习的光学遥感机场与飞行器目标识别技术
  6. 输入一个三位正整数,输出百位数,十位数,个位数
  7. leetcode —— 206. 反转链表
  8. 转:extjs里的fieldset不居中的解决办法(记录)
  9. matlab输电线路模型,输电线路模型及其特性.ppt
  10. 考研政治(二):马克思主义基本原理
  11. 使用HTML制作网页
  12. python画图配色_python matplotlib包图像配色方案
  13. pure-ftpd安装与使用
  14. 【概率论基础进阶】随机事件和概率-古典概型与伯努利概型
  15. 洛谷P2678 Java解法
  16. matplotlib 给坐标轴上的数字加单位
  17. java求六位数以内所有自幂数
  18. 移动电源给路由器供电
  19. 8个优秀图片素材库,免费/商用/高分辨率。
  20. 基于GIS、、geosever插件实现当地旅游资源网格化管理系统的架构

热门文章

  1. 第一次摸服务器遇到的问题总结
  2. 54. Spiral Matrix (Matrix)
  3. [Linux]结合awk删除hdfs指定日期前的数据
  4. node-inspector使用方法
  5. rails 构建 API
  6. CSS实现导航条Tab切换的三种方法
  7. JBOSS通过Apache负载均衡方法一:使用mod_jk
  8. ASP.NET 如何catch存储过程中抛出的异常信息
  9. Arcgis desktop 9.3的破解方法_经验版
  10. PHP中操作MySQL的一些要注意的问题