使用jQuery監聽DOM元素大小變化

原由

今天寫頁面的時候忽然有這麼個需求,因爲父元素(一個DIV)的height是由javascript計算出來的固定的值,而在其中增長了一個多說插件,在用戶評論後,子元素(DIV)的height屬性增長,致使子元素溢出。可是又不知道如何爲多說的評論按鈕增長回調函數,因而乎就想到了根據子元素的大小變化來從新計算父元素的height。javascript

onresize?

日常,都是在整個瀏覽器窗口變化時觸發一個修改佈局的回調函數。使用的是window對象的resize事件,利用:java

window.onresize = callback;

來綁定。但根據resize事件的target是defaultView (window),這裏詳見MDN的resize文檔,也就是說只有window對象有resize事件,因而乎就想到使用jQuery本身的事件機制來模擬一個普通元素上的resize事件jquery

使用jQuery事件的實現思路

能夠想到一種比較簡單的方式:
1. 在元素綁定resize對象時,記錄元素的width和height
2. 使用requestAnimationFrame、setTimeout、setInterval,每隔一段時間查詢其width和height,若是和記錄的width和height不同,運行回調函數並更新記錄中的width爲heightgit

jQuery插件

這個功能Ben Alman編寫了一個jQuery插件,這是傳送門
該插件的代碼(核心部分),詳細代碼請查看Ben Alman博客的內容:github

(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事件app

從setup鉤子能夠看到,在初始化整個事件處理時,建立一個元素隊列,隊列中的每隔元素都把width和height放在data中,而後每隔250ms啓動loopy函數,在loopy函數中判斷是否變化,若是有變,觸發回調函數並更新data中的width和height函數

從teardown鉤子能夠看到,在元素移除事件時,只須要將元素從元素隊列移除,並清除元素中的data數據。若是是元素隊列中的最後一個元素,則再也不繼續執行loopyoop

add鉤子中,對回調函數進行了包裝佈局

由此能夠看到一個簡單的jQuery自定義函數的實現機制

相關文章
相關標籤/搜索