使用jQuery监听DOM元素大小变化


Posted in Javascript onFebruary 24, 2016

起因

今天写页面的时候突然有这么个需求,由于父元素(一个DIV)的height是由javascript计算出来的固定的值,而在其中增加了一个多说插件,在用户评论后,子元素(DIV)的height属性增加,导致子元素溢出。但是又不知道如何为多说的评论按钮增加回调函数,于是乎就想到了根据子元素的大小变化来重新计算父元素的height。

onresize?

平常,都是在整个浏览器窗口变化时触发一个修改布局的回调函数。使用的是window对象的resize事件,利用:

window.onresize = callback;

来绑定。但根据resize事件的target是defaultView (window),这里详见MDN的resize文档,也就是说只有window对象有resize事件,于是乎就想到使用jQuery自己的事件机制来模拟一个普通元素上的resize事件

使用JQUERY事件的实现思路

可以想到一种比较简单的方式:

1. 在元素绑定resize对象时,记录元素的width和height
2. 使用requestAnimationFrame、setTimeout、setInterval,每隔一段时间查询其width和height,如果和记录的width和height不一样,运行回调函数并更新记录中的width为height

JQUERY插件

这个功能Ben Alman编写了一个jQuery插件,传送门
该插件的代码(核心部分),详细代码请查看Ben Alman博客的内容:

(function($, window, undefined) {
 var elems = $([]),
  jq_resize = $.resize = $.extend($.resize, {}),
  timeout_id,
  str_setTimeout = 'setTimeout',
  str_resize = 'resize',
  str_data = str_resize + '-special-event',
  str_delay = 'delay',
  str_throttle = 'throttleWindow';
 jq_resize[str_delay] = 250;
 jq_resize[str_throttle] = true;
 $.event.special[str_resize] = {
  setup: function() {
   if (!jq_resize[str_throttle] && this[str_setTimeout]) {
    return false;
   }
   var elem = $(this);
   elems = elems.add(elem);
   $.data(this, str_data, {
    w: elem.width(),
    h: elem.height()
   });
   if (elems.length === 1) {
    loopy();
   }
  },
  teardown: function() {
   if (!jq_resize[str_throttle] && this[str_setTimeout]) {
    return false;
   }
   var elem = $(this);
   elems = elems.not(elem);
   elem.removeData(str_data);
   if (!elems.length) {
    clearTimeout(timeout_id);
   }
  },
  add: function(handleObj) {
   if (!jq_resize[str_throttle] && this[str_setTimeout]) {
    return false;
   }
   var old_handler;
   function new_handler(e, w, h) {
    var elem = $(this),
     data = $.data(this, str_data);
    data.w = w !== undefined ? w : elem.width();
    data.h = h !== undefined ? h : elem.height();
    old_handler.apply(this, arguments);
   }
   if ($.isFunction(handleObj)) {
    old_handler = handleObj;
    return new_handler;
   } else {
    old_handler = handleObj.handler;
    handleObj.handler = new_handler;
   }
  }
 };

 function loopy() {
  timeout_id = window[str_setTimeout](function() {
   elems.each(function() {
    var elem = $(this),
     width = elem.width(),
     height = elem.height(),
     data = $.data(this, str_data);
    if (width !== data.w || height !== data.h) {
     elem.trigger(str_resize, [data.w = width, data.h = height]);
    }
   });
   loopy();
  }, jq_resize[str_delay]);
 }
})(jQuery, this);

jQuery为jQuery插件的开发者提供了添加自定义事件的接口,详细可以参考jQuery官方文档,这里就是典型的jQuery自定义事件添加方式,其中有三个钩子:

1. setup:The setup hook is called the first time an event of a particular type is attached to an element.首次绑定时执行,如果返回 false,使用默认方式绑定事件
2. teardown:The teardown hook is called when the final event of a particular type is removed from an element.若指定该方法,其在移除事件处理程序(removeEventListener)前执行,如果返回 false,移除默认绑定事件
3. add:Each time an event handler is added to an element through an API such as .on(), jQuery calls this hook.每一次给元素绑定事件,都会执行这个方法

setup、teardown和add三个钩子,每个钩子最先做的事都是检测是否该对象为window对象,然后根据window对象特殊处理,因为window对象本身有resize事件

从setup钩子可以看到,在初始化整个事件处理时,创建一个元素队列,队列中的每隔元素都把width和height放在data中,然后每隔250ms启动loopy函数,在loopy函数中判断是否变化,如果有变,触发回调函数并更新data中的width和height

从teardown钩子可以看到,在元素移除事件时,只需要将元素从元素队列移除,并清除元素中的data数据。如果是元素队列中的最后一个元素,则不再继续执行loopy

add钩子中,对回调函数进行了包装

由此可以看到一个简单的jQuery自定义函数的实现机制

Javascript 相关文章推荐
JavaScript开发时的五个注意事项
Dec 08 Javascript
jquery mobile changepage的三种传参方法介绍
Sep 13 Javascript
jquery分页插件jpaginate在IE中不兼容问题
Apr 22 Javascript
基于Bootstrap+jQuery.validate实现表单验证
May 30 Javascript
无缝滚动的简单实现代码(推荐)
Jun 07 Javascript
使用jquery.qrcode.js生成二维码插件
Oct 17 Javascript
canvas绘制表盘时钟
Jan 23 Javascript
JavaScript装饰器函数(Decorator)实例详解
Mar 30 Javascript
webpack4.0+vue2.0利用批处理生成前端单页或多页应用的方法
Jun 28 Javascript
微信小程序实现树莓派(raspberry pi)小车控制
Feb 12 Javascript
基于JavaScript实现十五拼图代码实例
Apr 26 Javascript
Node.js利用Express实现用户注册登陆功能(推荐)
Oct 26 Javascript
JavaScript中的闭包
Feb 24 #Javascript
jQuery中判断对象是否存在的方法汇总
Feb 24 #Javascript
jquery中键盘事件小结
Feb 24 #Javascript
javascript实现九宫格相加数值相等
May 28 #Javascript
Javascript类型转换的规则实例解析
Feb 23 #Javascript
理解Javascript图片预加载
Feb 23 #Javascript
Bootstarp风格的toggle效果分享
Feb 23 #Javascript
You might like
第十五节--Zend引擎的发展
2006/11/16 PHP
用PHP实现的四则运算表达式计算实现代码
2011/08/02 PHP
Windows下的PHP安装文件线程安全和非线程安全的区别
2014/04/23 PHP
ECMall支持SSL连接邮件服务器的配置方法详解
2014/05/19 PHP
PHP实现的多彩标签效果代码分享
2014/08/21 PHP
destoon实现调用当前栏目分类及子分类和三级分类的方法
2014/08/21 PHP
自定义session存储机制避免会话保持问题
2014/10/08 PHP
thinkPHP利用ajax异步上传图片并显示、删除的示例
2018/09/26 PHP
Jquery带搜索框的下拉菜单
2013/05/06 Javascript
js改变img标签的src属性在IE下没反应的解决方法
2013/07/23 Javascript
JS获取各种浏览器窗口大小的方法
2014/01/14 Javascript
EasyUI中实现form表单提交的示例分享
2015/03/01 Javascript
js+css绘制颜色动态变化的圈中圈效果
2016/01/27 Javascript
用JavaScript和jQuery实现瀑布流
2017/03/19 Javascript
详解nodejs爬虫程序解决gbk等中文编码问题
2017/04/06 NodeJs
js canvas实现QQ拨打电话特效
2017/05/10 Javascript
使用JS组件实现带ToolTip验证框的实例代码
2017/08/23 Javascript
基于Bootstrap和JQuery实现动态打开和关闭tab页的实例代码
2019/06/10 jQuery
Vue发布项目实例讲解
2019/07/17 Javascript
[37:45]完美世界DOTA2联赛PWL S3 LBZS vs Phoenix 第二场 12.09
2020/12/11 DOTA
python实现简单聊天应用 python群聊和点对点均实现
2017/09/14 Python
Python如何获得百度统计API的数据并发送邮件示例代码
2019/01/27 Python
python3利用ctypes传入一个字符串类型的列表方法
2019/02/12 Python
Python中私有属性的定义方式
2020/03/05 Python
python 装饰器的实际作用有哪些
2020/09/07 Python
python获取命令行参数实例方法讲解
2020/11/02 Python
python pyg2plot的原理知识点总结
2021/02/28 Python
使用css3实现的tab选项卡代码分享
2014/12/09 HTML / CSS
CSS3自定义滚动条样式的示例代码
2017/08/21 HTML / CSS
什么是重载?CTS、CLS和CLR分别做何解释
2012/05/06 面试题
高中毕业生个人自我鉴定
2013/11/24 职场文书
实习生自我评价
2014/01/18 职场文书
党员教师群众路线个人整改措施
2014/10/28 职场文书
支行行长岗位职责
2015/02/15 职场文书
导游词之鲁迅祖居
2019/10/17 职场文书
Elasticsearch 基本查询和组合查询
2022/04/19 Python