使用jQuery监听DOM元素大小变化
起因
今天写页面的时候突然有这么个需求,由于父元素(一个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插件
这个功能BenAlman编写了一个jQuery插件,传送门
该插件的代码(核心部分),详细代码请查看BenAlman博客的内容:
(function($,window,undefined){ varelems=$([]), 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]){ returnfalse; } varelem=$(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]){ returnfalse; } varelem=$(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]){ returnfalse; } varold_handler; functionnew_handler(e,w,h){ varelem=$(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; returnnew_handler; }else{ old_handler=handleObj.handler; handleObj.handler=new_handler; } } }; functionloopy(){ timeout_id=window[str_setTimeout](function(){ elems.each(function(){ varelem=$(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:Thesetuphookiscalledthefirsttimeaneventofaparticulartypeisattachedtoanelement.首次绑定时执行,如果返回false,使用默认方式绑定事件
2.teardown:Theteardownhookiscalledwhenthefinaleventofaparticulartypeisremovedfromanelement.若指定该方法,其在移除事件处理程序(removeEventListener)前执行,如果返回false,移除默认绑定事件
3.add:EachtimeaneventhandlerisaddedtoanelementthroughanAPIsuchas.on(),jQuerycallsthishook.每一次给元素绑定事件,都会执行这个方法
setup、teardown和add三个钩子,每个钩子最先做的事都是检测是否该对象为window对象,然后根据window对象特殊处理,因为window对象本身有resize事件
从setup钩子可以看到,在初始化整个事件处理时,创建一个元素队列,队列中的每隔元素都把width和height放在data中,然后每隔250ms启动loopy函数,在loopy函数中判断是否变化,如果有变,触发回调函数并更新data中的width和height
从teardown钩子可以看到,在元素移除事件时,只需要将元素从元素队列移除,并清除元素中的data数据。如果是元素队列中的最后一个元素,则不再继续执行loopy
add钩子中,对回调函数进行了包装
由此可以看到一个简单的jQuery自定义函数的实现机制