国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 編程 > JavaScript > 正文

使用jQuery監(jiān)聽DOM元素大小變化

2019-11-20 10:31:40
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

起因

今天寫頁(yè)面的時(shí)候突然有這么個(gè)需求,由于父元素(一個(gè)DIV)的height是由javascript計(jì)算出來(lái)的固定的值,而在其中增加了一個(gè)多說(shuō)插件,在用戶評(píng)論后,子元素(DIV)的height屬性增加,導(dǎo)致子元素溢出。但是又不知道如何為多說(shuō)的評(píng)論按鈕增加回調(diào)函數(shù),于是乎就想到了根據(jù)子元素的大小變化來(lái)重新計(jì)算父元素的height。

onresize?

平常,都是在整個(gè)瀏覽器窗口變化時(shí)觸發(fā)一個(gè)修改布局的回調(diào)函數(shù)。使用的是window對(duì)象的resize事件,利用:

window.onresize = callback;

來(lái)綁定。但根據(jù)resize事件的target是defaultView (window),這里詳見(jiàn)MDN的resize文檔,也就是說(shuō)只有window對(duì)象有resize事件,于是乎就想到使用jQuery自己的事件機(jī)制來(lái)模擬一個(gè)普通元素上的resize事件

使用JQUERY事件的實(shí)現(xiàn)思路

可以想到一種比較簡(jiǎn)單的方式:

1. 在元素綁定resize對(duì)象時(shí),記錄元素的width和height
2. 使用requestAnimationFrame、setTimeout、setInterval,每隔一段時(shí)間查詢其width和height,如果和記錄的width和height不一樣,運(yùn)行回調(diào)函數(shù)并更新記錄中的width為height

JQUERY插件

這個(gè)功能Ben Alman編寫了一個(gè)jQuery插件,傳送門
該插件的代碼(核心部分),詳細(xì)代碼請(qǐng)查看Ben Alman博客的內(nèi)容:

(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插件的開發(fā)者提供了添加自定義事件的接口,詳細(xì)可以參考jQuery官方文檔,這里就是典型的jQuery自定義事件添加方式,其中有三個(gè)鉤子:

1. setup:The setup hook is called the first time an event of a particular type is attached to an element.首次綁定時(shí)執(zhí)行,如果返回 false,使用默認(rèn)方式綁定事件
2. teardown:The teardown hook is called when the final event of a particular type is removed from an element.若指定該方法,其在移除事件處理程序(removeEventListener)前執(zhí)行,如果返回 false,移除默認(rèn)綁定事件
3. add:Each time an event handler is added to an element through an API such as .on(), jQuery calls this hook.每一次給元素綁定事件,都會(huì)執(zhí)行這個(gè)方法

setup、teardown和add三個(gè)鉤子,每個(gè)鉤子最先做的事都是檢測(cè)是否該對(duì)象為window對(duì)象,然后根據(jù)window對(duì)象特殊處理,因?yàn)閣indow對(duì)象本身有resize事件

從setup鉤子可以看到,在初始化整個(gè)事件處理時(shí),創(chuàng)建一個(gè)元素隊(duì)列,隊(duì)列中的每隔元素都把width和height放在data中,然后每隔250ms啟動(dòng)loopy函數(shù),在loopy函數(shù)中判斷是否變化,如果有變,觸發(fā)回調(diào)函數(shù)并更新data中的width和height

從teardown鉤子可以看到,在元素移除事件時(shí),只需要將元素從元素隊(duì)列移除,并清除元素中的data數(shù)據(jù)。如果是元素隊(duì)列中的最后一個(gè)元素,則不再繼續(xù)執(zhí)行l(wèi)oopy

add鉤子中,對(duì)回調(diào)函數(shù)進(jìn)行了包裝

由此可以看到一個(gè)簡(jiǎn)單的jQuery自定義函數(shù)的實(shí)現(xiàn)機(jī)制

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 临海市| 沐川县| 桦川县| 黑河市| 望奎县| 土默特右旗| 庆阳市| 客服| 江永县| 邛崃市| 东城区| 建昌县| 宜川县| 沧源| 疏勒县| 阳曲县| 繁峙县| 通化县| 乌拉特前旗| 同心县| 新晃| 修武县| 曲沃县| 建始县| 凤山市| 南昌县| 永平县| 伊金霍洛旗| 惠安县| 惠东县| 改则县| 漾濞| 镇宁| 囊谦县| 镇江市| 稷山县| 肇东市| 南京市| 德格县| 同心县| 黔南|