jQuery的animate函数学习记录


Posted in Javascript onAugust 08, 2014

很久之前就对jQuery animate的实现非常感兴趣,不过前段时间很忙,直到前几天端午假期才有时间去研究。

jQuery.animate的每种动画过渡效果都是通过easing函数实现的。jQuery1.4.2中就预置了两个这样的函数:

easing: {
linear: function( p, n, firstNum, diff ) {
return firstNum + diff * p;
},
swing: function( p, n, firstNum, diff ) {
return ((-Math.cos(p*Math.PI)/2) + 0.5) * diff + firstNum;
}
}

从参数名隐约可以推测出firstNum是初始值。要是你的数学学得比较好,你可以发现linear函数是直线方程;要是你的物理学得比较好,你可以发现它是匀速运动的位移方程(我数学和物理都没学好,是别人提醒我的……)。那么diff和p就是速度和时间了。

再看看jQuery.animate的原型:

animate: function( prop, speed, easing, callback )

各参数的说明如下:

prop:一组包含作为动画属性和终值的样式属性和及其值的集合。
speed:动画时长。
easing:要使用的擦除效果的名称。
callback:动画完成时执行的函数。

元素的当前样式值(firstNum)可以获取,动画时长(p)就是duration,最终样式值是prop。理论上说,动画速度(diff)是可以算出来的。但是这又必然需要另一个函数进行运算。这样做明显是不明智的。再看看调用easing函数的相关代码(位于jQuery.fx.prototype.step中):

var t = now();
...
var n = t - this.startTime;
this.state = n / this.options.duration;
...
this.pos = jQuery.easing[specialEasing || defaultEasing](this.state, n, 0, 1, this.options.duration);

可以发现,p参数的值也就是this.state的值,从上下文得知它实际上是动画的时间进度。而firstNum和diff的参数值都是写死的,分别是0和1。这下easing函数的秘密完全被解开,p、firstNum、diff都是百分率而非实际数值,easing函数的返回值也就是位移的进度。diff的值是1,也就是以1倍的速度运行动画。算出位移进度后,通过“初始值+(最终值-初始值)×进度”就可以算出当前位移值:

this.now = this.start + ((this.end - this.start) * this.pos);

通过setInterval每隔一定时间(jQuery中是13ms)进行一次位移运算,直到当前时间与初始时间的差值大于动画时长,这就是jQuery.animate的执行过程。

按照常规思路,动画的实现方式是这样的:通过setInterval每隔一定时间给某个值增加特定数值,直到这个值达到限制值。这样做的主要问题是,不同浏览器的运行速度不同,从而导致动画速度有差异,一般是IE下比较慢,Firefox下比较快。而jQuery.animate是以当前时间来决定位移值,某个时刻的位移值总是固定的,因而动画速度不会有差异。

Javascript 相关文章推荐
从阿里妈妈发现的几个不错的表单验证函数
Sep 21 Javascript
jquery图片上下tab切换效果
Mar 18 Javascript
浅析Js(Jquery)中,字符串与JSON格式互相转换的示例(直接运行实例)
Jul 09 Javascript
jQuery easyui的validatebox校验规则扩展及easyui校验框validatebox用法
Jan 18 Javascript
jQuery遍历DOM元素与节点方法详解
Apr 14 Javascript
webpack+vue中使用别名路径引用静态图片地址
Nov 20 Javascript
bootstrapTable+ajax加载数据 refresh更新数据
Aug 31 Javascript
vue.js自定义组件directives的实例代码
Nov 09 Javascript
Vue动态修改网页标题的方法及遇到问题
Jun 09 Javascript
vue-cli脚手架引入弹出层layer插件的几种方法
Jun 24 Javascript
Electron vue的使用教程图文详解
Jul 05 Javascript
云服务器部署Node.js项目的方法步骤(小白系列)
Mar 23 Javascript
jQuery中get和post方法传值测试及注意事项
Aug 08 #Javascript
JSON.stringify转换JSON时日期时间不准确的解决方法
Aug 08 #Javascript
js事件监听机制(事件捕获)总结
Aug 08 #Javascript
使用jquery.qrcode生成彩色二维码实例
Aug 08 #Javascript
兼容最新firefox、chrome和IE的javascript图片预览实现代码
Aug 08 #Javascript
js调试工具console.log()方法查看js代码的执行情况
Aug 08 #Javascript
JS创建类和对象的两种不同方式
Aug 08 #Javascript
You might like
比特率,大家看看这个就不用收音机音质去比MP3音质了
2021/03/01 无线电
中国收音机工业发展史
2021/03/02 无线电
php 开发中加密的几种方法总结
2017/03/22 PHP
PHP实现文件下载【实例分享】
2017/04/28 PHP
Laravel5.1 框架Request请求操作常见用法实例分析
2020/01/04 PHP
php使用pthreads v3多线程实现抓取新浪新闻信息操作示例
2020/02/21 PHP
学习ExtJS Panel常用方法
2009/10/07 Javascript
Javascript继承机制的设计思想分享
2011/08/28 Javascript
jQuery实现可收缩展开的级联菜单实例代码
2013/11/27 Javascript
js利用数组length属性清空和截短数组的小例子
2014/01/15 Javascript
jQuery后代选择器用法实例
2014/12/23 Javascript
js库Modernizr的介绍和使用
2015/05/07 Javascript
解决JS无法调用Controller问题的方法
2015/12/31 Javascript
JS+CSS实现鼠标经过弹出一个DIV框完整实例(带缓冲动画渐变效果)
2016/03/25 Javascript
echarts学习笔记之箱线图的分析与绘制详解
2017/11/22 Javascript
NodeJS 中Stream 的基本使用
2018/07/30 NodeJs
微信小程序拼接图片链接无底洞深入探究
2019/09/03 Javascript
在react中使用vue的状态管理的方法示例
2020/05/02 Javascript
[44:50]2018DOTA2亚洲邀请赛 4.1 小组赛 A组 TNC vs VG
2018/04/02 DOTA
python实现斐波那契递归函数的方法
2014/09/08 Python
简介Django中内置的一些中间件
2015/07/24 Python
详解Python最长公共子串和最长公共子序列的实现
2018/07/07 Python
对Python的交互模式和直接运行.py文件的区别详解
2019/06/29 Python
Python实现名片管理系统
2020/02/14 Python
Python类和实例的属性机制原理详解
2020/03/21 Python
Python错误的处理方法
2020/06/23 Python
学python需要去培训机构吗
2020/07/01 Python
深入探究HTML5的History API
2015/07/09 HTML / CSS
Sunglasses Shop德国站:欧洲排名第一的太阳镜网站
2017/08/01 全球购物
匡威西班牙官网:Converse西班牙
2019/10/01 全球购物
社区志愿者心得体会
2014/01/03 职场文书
职位说明书范文
2014/05/07 职场文书
岗位职责说明书模板
2014/07/30 职场文书
男方婚前保证书
2015/02/28 职场文书
2015年工会工作总结
2015/03/30 职场文书
《假如》教学反思
2016/02/17 职场文书