JS奇技之利用scroll來監(jiān)聽resize詳解

前言

創(chuàng)新互聯(lián)專注為客戶提供全方位的互聯(lián)網(wǎng)綜合服務,包含不限于成都網(wǎng)站設計、成都做網(wǎng)站、靈璧網(wǎng)絡推廣、小程序開發(fā)、靈璧網(wǎng)絡營銷、靈璧企業(yè)策劃、靈璧品牌公關、搜索引擎seo、人物專訪、企業(yè)宣傳片、企業(yè)代運營等,從售前售中售后,我們都將竭誠為您服務,您的肯定,是我們最大的嘉獎;創(chuàng)新互聯(lián)為所有大學生創(chuàng)業(yè)者提供靈璧建站搭建服務,24小時服務熱線:13518219792,官方網(wǎng)址:bm7419.com

大家都知道知道原生的 resize 事件只能作用于 defaultView 即 window 上,那么我們應該通過什么樣的方式來監(jiān)聽其他元素的大小改變呢?筆者最近學習發(fā)現(xiàn)了一種神奇的方法,通過 scroll 事件來間接實現(xiàn) resize 事件的監(jiān)聽,本文將對這種方式進行原理的剖析與代碼實現(xiàn)。

原理

首先,我們先來看一下 scroll 事件是干嘛的。

The scroll event is fired when the document view or an element has been scrolled.

當文檔視圖或者元素滾動的時候會觸發(fā) scroll 事件。

也就是說元素滾動的時候會觸發(fā)這個事件,那么什么時候元素會滾動?當元素大于其父級元素,且父級元素允許其滾動的時候,該元素可以進行滾動。換句話說,元素可以滾動意味著父子元素大小不一致,這是這個方法的核心。

那么我們需要讓元素大小發(fā)生改變時,使得 scrollLeft 或者 scrollTop 發(fā)生改變,從而觸發(fā) scroll 事件,進一步得知其大小發(fā)生了改變。

監(jiān)聽元素變大

元素變大的時候,我們可以看到更多,其內(nèi)部可滾動區(qū)域?qū)⒙郎p小,但這并不會造成滾動條位置的改變,但當元素大到讓滾動條消失的時候會讓 scrollLeft 或者 scrollTop 變成 0,這樣我們就知道了元素變大了,因此我們其實只需要 1px 來判斷,其圖示如下:

JS奇技之利用scroll來監(jiān)聽resize詳解

監(jiān)聽元素變小

當元素變小的時候,可滾動區(qū)域會變大,滾動條的位置其實并不會進行改變,這里采取的做法是,讓可滾動區(qū)域和父元素成一定的比例一起縮小,讓父元素來擠壓滾動區(qū)域,從而間接改變滾動條 scrollLeft 或者 scrollTop 的大小,文字描述可能不是很清楚,我們看下圖:

JS奇技之利用scroll來監(jiān)聽resize詳解

通過以上兩種方式,我們可以就可以獲得 resize 事件。

實現(xiàn)

首先,為了不影響原有的元素,我們應當創(chuàng)建一個和要監(jiān)聽元素等大的元素,并對其進行相關操作,然后我們需要兩個子元素來分別監(jiān)聽元素變大和元素變小兩個情況。因此構造如下的 HTML 結構:

<div class="resize-triggers">
 <div class="expand-trigger">
 <div></div>
 </div>
 <div class="contract-trigger"></div>
</div>

他們對應的 CSS 如下:

.resize-triggers {
 visibility: hidden;
 opacity: 0;
}

.resize-triggers,
.resize-triggers > div,
.contract-trigger:before {
 content: " ";
 display: block;
 position: absolute;
 top: 0;
 left: 0;
 height: 100%;
 width: 100%;
 overflow: hidden;
}

.resize-triggers > div {
 overflow: auto;
}

.contract-triggers:before {
 width: 200%;
 height: 200%;
}

其中 .expand-triggers 的子元素寬高應當保持大于父元素 1px,且兩個觸發(fā)器都應當保持在最右下角的狀態(tài),因此我們可以實現(xiàn)如下的狀態(tài)重置函數(shù),并在初始化和每次滾動事件的時候調(diào)用:

/**
 * 重置觸發(fā)器
 * @param element 要處理的元素
 */
const resetTrigger = function(element) {
 const trigger = element.__resizeTrigger__; // 要重置的觸發(fā)器
 const expand = trigger.firstElementChild; // 第一個子元素,用來監(jiān)聽變大
 const contract = trigger.lastElementChild; // 最后一個子元素,用來監(jiān)聽變小
 const expandChild = expand.firstElementChild; // 第一個子元素的第一個子元素,用來監(jiān)聽變大

 contract.scrollLeft = contract.scrollWidth; // 滾動到最右
 contract.scrollTop = contract.scrollHeight; // 滾動到最下
 expandChild.style.width = expand.offsetWidth + 1 + 'px'; // 保持寬度多1px
 expandChild.style.height = expand.offsetHeight + 1 + 'px'; // 保持高度多1px
 expand.scrollLeft = expand.scrollWidth; // 滾動到最右
 expand.scrollTop = expand.scrollHeight; // 滾動到最右
};

我們可以用如下函數(shù)檢測元素大小是否發(fā)生了改變:

/**
 * 檢測觸發(fā)器狀態(tài)
 * @param element 要檢查的元素
 * @returns {boolean} 是否改變了大小
 */
const checkTriggers = function(element) {
 // 寬度或高度不一致就返回true
 return element.offsetWidth !== element.__resizeLast__.width || element.offsetHeight !== element.__resizeLast__.height;
};

最終,我們可以實現(xiàn)簡單的事件監(jiān)聽的添加:

/**
 * 添加大小更改監(jiān)聽
 * @param element 要監(jiān)聽的元素
 * @param fn 回調(diào)函數(shù)
 */
export const addResizeListener = function(element, fn) {
if (isServer) return; // 服務器端直接返回
 if (attachEvent) { // 處理低版本ie
 element.attachEvent('onresize', fn);
 } else {
if (!element.__resizeTrigger__) { // 如果沒有觸發(fā)器
  if (getComputedStyle(element).position === 'static') {
  element.style.position = 'relative'; // 將static改為relative
  }
createStyles();
  element.__resizeLast__ = {}; // 初始化觸發(fā)器最后的狀態(tài)
  element.__resizeListeners__ = []; // 初始化觸發(fā)器的監(jiān)聽器

  const resizeTrigger = element.__resizeTrigger__ = document.createElement('div'); // 創(chuàng)建觸發(fā)器
  resizeTrigger.className = 'resize-triggers';
  resizeTrigger.innerHTML = '<div class="expand-trigger"><div></div></div><div class="contract-trigger"></div>';
  element.appendChild(resizeTrigger); // 添加觸發(fā)器

  resetTrigger(element); // 重置觸發(fā)器
  element.addEventListener('scroll', scrollListener, true); // 監(jiān)聽滾動事件

  /* Listen for a css animation to detect element display/re-attach */
  // 監(jiān)聽CSS動畫來檢測元素顯示或者重新添加
  if (animationStartEvent) { // 動畫開始
  resizeTrigger.addEventListener(animationStartEvent, function(event) { // 增加動畫開始的事件監(jiān)聽
   if (event.animationName === RESIZE_ANIMATION_NAME) { // 如果是大小改變事件
   resetTrigger(element); // 重置觸發(fā)器
   }
  });
  }
 }
 element.__resizeListeners__.push(fn); // 加入該回調(diào)
 }
};

以及如下的函數(shù)來移除事件監(jiān)聽:

/**
 * 移除大小改變的監(jiān)聽
 * @param element 被監(jiān)聽的元素
 * @param fn 對應的回調(diào)函數(shù)
 */
export const removeResizeListener = function(element, fn) {
if (attachEvent) { // 處理ie
 element.detachEvent('onresize', fn);
 } else {
 element.__resizeListeners__.splice(element.__resizeListeners__.indexOf(fn), 1); // 移除對應的回調(diào)函數(shù)
 if (!element.__resizeListeners__.length) { // 如果全部時間被移除
  element.removeEventListener('scroll', scrollListener); // 移除滾動監(jiān)聽
  element.__resizeTrigger__ = !element.removeChild(element.__resizeTrigger__); // 移除對應的觸發(fā)器,但保存下來
 }
 }
};

其他

其中有部分內(nèi)容是用來優(yōu)化的,并不影響基礎功能,如對服務器渲染、客戶端渲染的區(qū)分,對 IE 的特殊處理,以及通過 opacity 的動畫來解決 chrome 上的bug。

好了,以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作能帶來一定的幫助,如有疑問大家可以留言交流,謝謝大家對創(chuàng)新互聯(lián)的支持。

網(wǎng)站欄目:JS奇技之利用scroll來監(jiān)聽resize詳解
本文鏈接:http://bm7419.com/article48/ijhchp.html

成都網(wǎng)站建設公司_創(chuàng)新互聯(lián),為您提供移動網(wǎng)站建設、網(wǎng)站建設網(wǎng)站導航、響應式網(wǎng)站企業(yè)建站、App開發(fā)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

網(wǎng)站建設網(wǎng)站維護公司