老鳥:怎樣去監聽 DOM
元素的高度變化呢?
菜鳥:哈哈哈哈哈,這都不知道哦,用 onresize
事件鴨!
老鳥扶了扶眼睛,空氣安靜幾秒鐘,菜鳥才晃過神來。對鴨,普通 DOM
元素沒有 onresize
事件,只有在 window
對象下有此事件,該死,又雙叒叕糗大了。css
哈哈哈哈,以上純屬虛構,不過在最近項目中還真遇到過對容器監聽高(寬)變化:在使用 iscroll
或 better-scroll
滾動插件,若是容器內部元素有高度變化要去及時更新外部包裹容器,即調用 refresh()
方法。否則就會形成滾動偏差(滾動不到底部或滾動脫離底部)。html
可能咱們通常處理思路:css3
DOM
節點有更新(刪除或插入)後就去調用 refresh()
,更新外部容器。onload
監聽每次加載完成,再去調用 refresh()
,更新外部容器。這樣咱們會發現,若是容器內部元素比較複雜,調用會愈來愈繁瑣,甚至還要考慮到用戶使用的每個操做均可能致使內部元素寬高變化,進而要去調整外部容器,調用 refresh()
。git
實際上,不論是對元素的哪一種操做,都會形成它的屬性、子孫節點、文本節點發生了變化,若是能能監聽獲得這種變化,這時只需比較容器寬高變化,便可實現對容器寬高的監聽,而無需關係它外部行爲。DOM3 Events
規範爲咱們提供了 MutationObserver
接口監視對 DOM
樹所作更改的能力。github
Mutation Observer API
用來監視 DOM
變更。DOM
的任何變更,好比節點的增減、屬性的變更、文本內容的變更,這個 API
均可以獲得通知。web
PS Mutation Observer API
已經有很很少的瀏覽器兼容性,若是對IE10及如下沒有要求的話。npm
DOM
發生變更都會觸發 Mutation Observer
事件。可是,它跟事件仍是有不用點:事件是同步觸發,DOM
變化當即觸發相應事件;Mutation Observer
是異步觸發,DOM
變化不會立刻觸發,而是等當前全部 DOM
操做都結束後才觸發。總的來講,特色以下:數組
DOM
變更記錄封裝成一個數組進行處理,而不是一條條個別處理 DOM
變更。DOM
的全部類型變更,也能夠指定只觀察某一類變更。MutationObserver
構造函數的實例傳的是一個回調函數,該函數接受兩個參數,第一個是變更的數組,第二個是觀察器是實例。瀏覽器
var observer = new MutationObserver(function (mutations, observer){ mutations.forEach(function (mutaion) { console.log(mutation); }) })
observe
方法用來執行監聽,接受兩個參數:數據結構
DOM
節點;var $tar = document.getElementById('tar'); var option = { childList: true, // 子節點的變更(新增、刪除或者更改) attributes: true, // 屬性的變更 characterData: true, // 節點內容或節點文本的變更 subtree: true, // 是否將觀察器應用於該節點的全部後代節點 attributeFilter: ['class', 'style'], // 觀察特定屬性 attributeOldValue: true, // 觀察 attributes 變更時,是否須要記錄變更前的屬性值 characterDataOldValue: true // 觀察 characterData 變更,是否須要記錄變更前的值 } mutationObserver.observe($tar, option);
option
中,必須有 childList
、attributes
和characterData
中一種或多種,不然會報錯。其中各個屬性意思以下:
childList
布爾值,表示是否應用到子節點的變更(新增、刪除或者更改);attributes
布爾值,表示是否應用到屬性的變更;characterData
布爾值,表示是否應用到節點內容或節點文本的變更;subtree
布爾值,表示是否應用到是否將觀察器應用於該節點的全部後代節點;attributeFilter
數組,表示觀察特定屬性;attributeOldValue
布爾值,表示觀察 attributes
變更時,是否須要記錄變更前的屬性值;characterDataOldValue
布爾值,表示觀察 characterData
變更,是否須要記錄變更前的值;childList
屬性表示是否應用到子節點的變更(新增、刪除或者更改),監聽不到子節點後代節點變更。
var mutationObserver = new MutationObserver(function (mutations) { console.log(mutations); }) mutationObserver.observe($tar, { childList: true, // 子節點的變更(新增、刪除或者更改) }) var $div1 = document.createElement('div'); $div1.innerText = 'div1'; // 新增子節點 $tar.appendChild($div1); // 能監聽到 // 刪除子節點 $tar.childNodes[0].remove(); // 能監聽到 var $div2 = document.createElement('div'); $div2.innerText = 'div2'; var $div3 = document.createElement('div'); $div3.innerText = 'div3'; // 新增子節點 $tar.appendChild($div2); // 能監聽到 // 替換子節點 $tar.replaceChild($div3, $div2); // 能監聽到 // 新增孫節點 $tar.childNodes[0].appendChild(document.createTextNode('新增孫文本節點')); // 監聽不到
attributes
屬性表示是否應用到 DOM
節點屬性的值變更的監聽。而 attributeFilter
屬性是用來過濾要監聽的屬性 key
。
// ... mutationObserver.observe($tar, { attributes: true, // 屬性的變更 attributeFilter: ['class', 'style'], // 觀察特定屬性 }) // ... // 改變 style 屬性 $tar.style.height = '100px'; // 能監聽到 // 改變 className $tar.className = 'tar'; // 能監聽到 // 改變 dataset $tar.dataset = 'abc'; // 監聽不到
subtree
屬性characterData
屬性表示是否應用到節點內容或節點文本的變更。subtree
是否將觀察器應用於該節點的全部後代節點。爲了更好觀察節點文本變化,將二者結合應用到富文本監聽上是不錯的選擇。
簡單的富文本,好比
<div id="tar" contentEditable>A simple editor</div>
var $tar = document.getElementById('tar'); var MutationObserver = window.MutationObserver || window.webkitMutationObserver || window.MozMutationObserver; var mutationObserver = new MutationObserver(function (mutations) { console.log(mutations); }) mutationObserver.observe($tar, { characterData: true, // 節點內容或節點文本的變更 subtree: true, // 是否將觀察器應用於該節點的全部後代節點 })
MutationObserver
實例上還有兩個方法,takeRecords()
用來清空記錄隊列並返回變更記錄的數組。disconnect()
用來中止觀察。調用該方法後,DOM
再發生變更,也不會觸發觀察器。
var $text5 = document.createTextNode('新增文本節點5'); var $text6 = document.createTextNode('新增文本節點6'); // 新增文本節點 $tar.appendChild($text5); var record = mutationObserver.takeRecords(); console.log('record: ', record); // 返回 記錄新增文本節點操做,並清空監聽隊列 // 替換文本節點 $tar.replaceChild($text6, $text5); mutationObserver.disconnect(); // 此處之後的再也不監聽 // 刪除文本節點 $tar.removeChild($text6); // 監聽不到
前面還有兩個屬性 attributeOldValue
和 characterDataOldValue
沒有說,實際上是影響 takeRecords()
方法返回 MutationRecord
實例。若是設置了這兩個屬性,就會對應返回對象中 oldValue
爲記錄以前舊的 attribute
和 data
值。
好比將原來的 className
的值 aaa
替換成 tar
,oldValue
記錄爲 aaa
。
record: [{ addedNodes: NodeList [] attributeName: "class" attributeNamespace: null nextSibling: null oldValue: "aaa" previousSibling: null removedNodes: NodeList [] target: div#tar.tar type: "attributes" }]
一個容器自己以及內部元素的屬性變化,節點變化和文本變化是影響該容器高寬的重要因素(固然還有其餘因素),以上了解了 MutationObserver
API 的一些細節,能夠實現監聽容器寬高的變化。
var $tar = document.getElementById('tar'); var MutationObserver = window.MutationObserver || window.webkitMutationObserver || window.MozMutationObserver; var recordHeight = 0; var mutationObserver = new MutationObserver(function (mutations) { console.log(mutations); let height = window.getComputedStyle($tar).getPropertyValue('height'); if (height === recordHeight) { return; } recordHeight = height; console.log('高度變化了'); // 以後更新外部容器等操做 }) mutationObserver.observe($tar, { childList: true, // 子節點的變更(新增、刪除或者更改) attributes: true, // 屬性的變更 characterData: true, // 節點內容或節點文本的變更 subtree: true // 是否將觀察器應用於該節點的全部後代節點 })
除了容器內部元素節點、屬性變化,還有 css3 動畫會影響容器高寬,因爲動畫並不會形成元素屬性的變化,因此 MutationObserver
API 是監聽不到的。
將 #tar
容器加入如下 css
動畫
@keyframes changeHeight { to { height: 300px; } } #tar { background-color: aqua; border: 1px solid #ccc; animation: changeHeight 2s ease-in 1s; }
能夠看出,沒有打印輸出,是監聽不到動畫改變高寬的。因此,在這還需對這條「漏網之魚」進行處理。處理很簡單,只需在動畫(transitionend
、animationend
)中止事件觸發時監聽高寬變化便可。在這裏用 Vue
自定義指令處理以下:
/** * 監聽元素高度變化,更新滾動容器 */ Vue.directive('observe-element-height', { insert (el, binding) { const MutationObserver = window.MutationObserver || window.webkitMutationObserver || window.MozMutationObserver let recordHeight = 0 const onHeightChange = _.throttle(function () { // _.throttle 節流函數 let height = window.getComputedStyle(el).getPropertyValue('height'); if (height === recordHeight) { return } recordHeight = height console.log('高度變化了') // 以後更新外部容器等操做 }, 500) el.__onHeightChange__ = onHeightChange el.addEventListener('animationend', onHeightChange) el.addEventListener('transitionend', onHeightChange) el.__observer__ = new MutationObserver((mutations) => { onHeightChange() }); el.__observer__.observe(el, { childList: true, subtree: true, characterData: true, attributes: true }) }, unbind (el) { if (el.__observer__) { el.__observer__.disconnect() el.__observer__ = null } el.removeEventListener('animationend', el.__onHeightChange__) el.removeEventListener('transitionend', el.__onHeightChange__) el.__onHeightChange__ = null } })
既然對容器區域寬高監聽有硬性需求,那麼是否有相關規範呢?答案是有的,ResizeObserver
接口能夠監聽到 Element
的內容區域或 SVGElement
的邊界框改變。內容區域則須要減去內邊距 padding
。目前仍是實驗性的一個接口,各大瀏覽器對ResizeObserver兼容性不夠,實際應用需謹慎。
實驗性的 API
不足,總有 Polyfill
來彌補。
ResizeObserver Polyfill
利用事件冒泡,在頂層 document
上監聽動畫 transitionend
;window
的 resize
事件;MutationObserver
監聽 document
元素;DOMSubtreeModified
監聽 document
元素。利用MapShim
(相似ES6中 Map
) 數據結構,key
爲被監聽元素,value
爲 ResizeObserver
實例,映射監聽關係,頂層 document
或 window
監聽到觸發事件,經過綁定元素便可監聽元素尺寸變化。部分源碼以下:
/** * Initializes DOM listeners. * * @private * @returns {void} */ ResizeObserverController.prototype.connect_ = function () { // Do nothing if running in a non-browser environment or if listeners // have been already added. if (!isBrowser || this.connected_) { return; } // Subscription to the "Transitionend" event is used as a workaround for // delayed transitions. This way it's possible to capture at least the // final state of an element. document.addEventListener('transitionend', this.onTransitionEnd_); window.addEventListener('resize', this.refresh); if (mutationObserverSupported) { this.mutationsObserver_ = new MutationObserver(this.refresh); this.mutationsObserver_.observe(document, { attributes: true, childList: true, characterData: true, subtree: true }); } else { document.addEventListener('DOMSubtreeModified', this.refresh); this.mutationEventsAdded_ = true; } this.connected_ = true; };
PS:不過,這裏貌似做者沒有對 animation
作處理,也就是 animation
改變元素尺寸仍是監聽不到。不知道是否是我沒有全面的考慮,這點已向做者提了issue。
window
的 resize
沒有兼容性問題,按照這個思路,能夠用隱藏的 iframe
模擬 window
撐滿要監聽得容器元素,當容器尺寸變化時,天然會 iframe
尺寸也會改變,經過contentWindow.onresize()
就能監聽獲得。
function observeResize(element, handler) { let frame = document.createElement('iframe'); const CSS = 'position:absolute;left:0;top:-100%;width:100%;height:100%;margin:1px 0 0;border:none;opacity:0;visibility:hidden;pointer-events:none;'; frame.style.cssText = CSS; frame.onload = () => { frame.contentWindow.onresize = () => { handler(element); }; }; element.appendChild(frame); return frame; } let element = document.getElementById('main'); // listen for resize observeResize(element, () => { console.log('new size: ', { width: element.clientWidth, height: element.clientHeight }); });
採用這種方案經常使用插件有 iframe-resizer、resize-sensor等。不過這種方案不是特別優雅,須要插入 iframe
元素,還需將父元素定位,可能在頁面上會有其餘意想不到的問題,僅做爲供參考方案吧。
最後,要優雅地監聽元素的寬高變化,不要去根據交互行爲而是從元素自己去監聽,瞭解 MutationObserver
接口是重點,其次要考慮到元素動畫可能形成寬高變化,兼容IE11如下,經過 DOMSubtreeModified
監聽。用 iframe 模擬 window 的 resize
屬於一種供參考方案。作的功課有點少,歡迎指正,完~