<span id="mktg5"></span>

<i id="mktg5"><meter id="mktg5"></meter></i>

        <label id="mktg5"><meter id="mktg5"></meter></label>
        最新文章專題視頻專題問答1問答10問答100問答1000問答2000關鍵字專題1關鍵字專題50關鍵字專題500關鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關鍵字專題關鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
        問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
        當前位置: 首頁 - 科技 - 知識百科 - 正文

        原生JS實現瀑布流插件

        來源:懂視網 責編:小采 時間:2020-11-27 22:19:56
        文檔

        原生JS實現瀑布流插件

        原生JS實現瀑布流插件:瀑布流布局中的圖片有一個核心特點—等寬不定等高,瀑布流布局在國內網網站都有一定規模的使用,比如pinterest、花瓣網等等。那么接下來就基于這個特點開始瀑布流探索之旅。 基礎功能實現 首先我們定義好一個有 20 張圖片的容器, <body> &
        推薦度:
        導讀原生JS實現瀑布流插件:瀑布流布局中的圖片有一個核心特點—等寬不定等高,瀑布流布局在國內網網站都有一定規模的使用,比如pinterest、花瓣網等等。那么接下來就基于這個特點開始瀑布流探索之旅。 基礎功能實現 首先我們定義好一個有 20 張圖片的容器, <body> &

        瀑布流布局中的圖片有一個核心特點—等寬不定等高,瀑布流布局在國內網網站都有一定規模的使用,比如pinterest、花瓣網等等。那么接下來就基于這個特點開始瀑布流探索之旅。

        基礎功能實現

        首先我們定義好一個有 20 張圖片的容器,

        <body>
         <style>
         #waterfall {
         position: relative;
         }
         .waterfall-box {
         float: left;
         width: 200px;
         }
         </style>
        </body>
        <div id="waterfall">
         <img src="images/1.png" class="waterfall-box">
         <img src="images/2.png" class="waterfall-box">
         <img src="images/3.png" class="waterfall-box">
         <img src="images/4.png" class="waterfall-box">
         <img src="images/5.png" class="waterfall-box">
         <img src="images/6.png" class="waterfall-box">
         ...
         </div>
        由于未知的 css 知識點,絲襪最長的妹子把下面的空間都占用掉了。。
        接著正文,假如如上圖,每排有 5 列,那第 6 張圖片應該出現前 5 張圖片哪張的下面呢?當然是絕對定位到前 5 張圖片高度最小的圖片下方。
        那第 7 張圖片呢?這時候把第 6 張圖片和在它上面的圖片當作是一個整體后,思路和上述是一致的。代碼實現如下:
        Waterfall.prototype.init = function () {
         ...
         const perNum = this.getPerNum() // 獲取每排圖片數
         const perList = [] // 存儲第一列的各圖片的高度
         for (let i = 0; i < perNum; i++) {
         perList.push(imgList[i].offsetHeight)
         }
         let pointer = this.getMinPointer(perList) // 求出當前最小高度的數組下標
         for (let i = perNum; i < imgList.length; i++) {
         imgList[i].style.position = 'absolute' // 核心語句
         imgList[i].style.left = `${imgList[pointer].offsetLeft}px`
         imgList[i].style.top = `${perList[pointer]}px`
        
         perList[pointer] = perList[pointer] + imgList[i].offsetHeight // 數組最小的值加上相應圖片的高度
         pointer = this.getMinPointer(perList)
         }
        }
        

        細心的朋友也許發現了代碼中獲取圖片的高度用到了 offsetHeight 這個屬性,這個屬性的高度之和等于圖片高度 + 內邊距 + 邊框,正因為此,我們用了 padding 而不是 margin 來設置圖片與圖片之間的距離。此外除了offsetHeight 屬性,此外還要理解 offsetHeightclientHeightoffsetTopscrollTop 等屬性的區別,才能比較好的理解這個項目。css 代碼簡單如下:

        .waterfall-box {
         float: left;
         width: 200px;
         padding-left: 10px;
         padding-bottom: 10px;
        }
        
        scroll、resize 事件監聽的實現

        實現了初始化函數 init 以后,下一步就要實現對 scroll 滾動事件進行監聽,從而實現當滾到父節點的底部有源源不斷的圖片被加載出來的效果。這時候要考慮一個點,是滾動到什么位置時觸發加載函數呢?這個因人而異,我的做法是當滿足 父容器高度 + 滾動距離 > 最后一張圖片的 offsetTop 這個條件,即橙色線條 + 紫色線條 > 藍色線條時觸發加載函數,代碼如下:

        window.onscroll = function() {
         // ...
         if (scrollPX + bsHeight > imgList[imgList.length - 1].offsetTop) {// 瀏覽器高度 + 滾動距離 > 最后一張圖片的 offsetTop
         const fragment = document.createDocumentFragment()
         for(let i = 0; i < 20; i++) {
         const img = document.createElement('img')
         img.setAttribute('src', `images/${i+1}.png`)
         img.setAttribute('class', 'waterfall-box')
         fragment.appendChild(img)
         }
         $waterfall.appendChild(fragment)
         }
        }

        因為父節點可能自定義節點,所以提供了對監聽 scroll 函數的封裝,代碼如下:

        proto.bind = function () {
         const bindScrollElem = document.getElementById(this.opts.scrollElem)
         util.addEventListener(bindScrollElem || window, 'scroll', scroll.bind(this))
         }
         const util = {
         addEventListener: function (elem, evName, func) {
         elem.addEventListener(evName, func, false)
         },
         }

        resize 事件的監聽與 scroll 事件監聽大同小異,當觸發了 resize 函數,調用 init 函數進行重置就行。

        使用發布-訂閱模式和繼承實現監聽綁定

        既然以開發插件為目標,不能僅僅滿足于功能的實現,還要留出相應的操作空間給開發者自行處理。聯想到業務場景中瀑布流中下拉加載的圖片一般都來自 Ajax 異步獲取,那么加載的數據必然不能寫死在庫里,期望能實現如下調用(此處借鑒了 waterfall 的使用方式),

        const waterfall = new Waterfall({options})
        waterfall.on("load", function () {
         // 此處進行 ajax 同步/異步添加圖片
        })

        觀察調用方式,不難聯想到使用發布/訂閱模式來實現它,關于發布/訂閱模式,之前在 Node.js 異步異聞錄 有介紹它。其核心思想即通過訂閱函數將函數添加到緩存中,然后通過發布函數實現異步調用,下面給出其代碼實現:

        function eventEmitter() {
         this.sub = {}
        }
        eventEmitter.prototype.on = function (eventName, func) { // 訂閱函數
         if (!this.sub[eventName]) {
         this.sub[eventName] = []
         }
         this.sub[eventName].push(func) // 添加事件監聽器
        }
        eventEmitter.prototype.emit = function (eventName) { // 發布函數
         const argsList = Array.prototype.slice.call(arguments, 1)
         for (let i = 0, length = this.sub[eventName].length; i < length; i++) {
         this.sub[eventName][i].apply(this, argsList) // 調用事件監聽器
         }
        }

        接著,要讓 Waterfall 能使用發布/訂閱模式,只需讓 Waterfall 繼承 eventEmitter 函數,代碼實現如下:

        function Waterfall(options = {}) {
         eventEmitter.call(this)
         this.init(options) // 這個 this 是 new 的時候,綁上去的
        }
        Waterfall.prototype = Object.create(eventEmitter.prototype)
        Waterfall.prototype.constructor = Waterfall
        
        繼承方式的寫法吸收了基于構造函數繼承和基于原型鏈繼承兩種寫法的優點,以及使用 Object.create 隔離了子類和父類,關于繼承更多方面的細節,可以另寫一篇文章了,此處點到為止。

        小優化

        為了防止 scroll 事件觸發多次加載圖片,可以考慮用函數防抖與節流實現。在基于發布-訂閱模式的基礎上,定義了個 isLoading 參數表示是否在加載中,并根據其布爾值決定是否加載,代碼如下:

        let isLoading = false
        const scroll = function () {
         if (isLoading) return false // 避免一次觸發事件多次
         if (scrollPX + bsHeight > imgList[imgList.length - 1].offsetTop) { // 瀏覽器高度 + 滾動距離 > 最后一張圖片的 offsetTop
         isLoading = true
         this.emit('load')
         }
        }
        proto.done = function () {
         this.on('done', function () {
         isLoading = false
         ...
         })
         this.emit('done')
        }
        
        這時候需要在調用的地方加上 waterfall.done, 從而告知當前圖片已經加載完畢,代碼如下:
        const waterfall = new Waterfall({})
        waterfall.on("load", function () {
         // 異步/同步加載圖片
         waterfall.done()
        })
        源碼地址:https://github.com/MuYunyun/waterfall

        項目簡陋,不足之處在所難免,歡迎留下你們寶貴的意見。

        聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

        文檔

        原生JS實現瀑布流插件

        原生JS實現瀑布流插件:瀑布流布局中的圖片有一個核心特點—等寬不定等高,瀑布流布局在國內網網站都有一定規模的使用,比如pinterest、花瓣網等等。那么接下來就基于這個特點開始瀑布流探索之旅。 基礎功能實現 首先我們定義好一個有 20 張圖片的容器, <body> &
        推薦度:
        標簽: 插件 js 瀑布流
        • 熱門焦點

        最新推薦

        猜你喜歡

        熱門推薦

        專題
        Top
        主站蜘蛛池模板: 亚洲欧洲自拍拍偷综合| 久久亚洲精品中文字幕三区| 亚洲欧洲日韩国产一区二区三区| 91老湿机福利免费体验| 亚洲色偷偷偷网站色偷一区| 亚洲高清免费视频| 美女视频黄a视频全免费网站一区 美女视频黄a视频全免费网站色 | 国产高潮流白浆喷水免费A片 | 亚洲色偷偷狠狠综合网| 国产免费一级高清淫曰本片| 亚洲不卡中文字幕无码| 亚洲黄色免费电影| 中文日韩亚洲欧美制服| 国产一区二区三区无码免费| 一级毛片免费不卡| 好爽又高潮了毛片免费下载| 亚洲第一第二第三第四第五第六| 人人狠狠综合久久亚洲高清 | 最新国产乱人伦偷精品免费网站| 亚洲成人在线网站| 免费AA片少妇人AA片直播| 国产成人亚洲精品青草天美| 一区二区免费视频| 亚洲精品无码专区在线| 亚洲国模精品一区| 精品无码人妻一区二区免费蜜桃 | 免费国产黄网站在线观看可以下载 | 亚洲一级视频在线观看| 国产网站免费观看| a在线观看免费视频| 亚洲性线免费观看视频成熟| 亚洲午夜精品久久久久久浪潮 | 国产网站在线免费观看| 水蜜桃视频在线观看免费播放高清| 久久久久久亚洲Av无码精品专口| 四虎影院在线免费播放| 两个人看www免费视频| 亚洲美国产亚洲AV| 亚洲av无码无在线观看红杏| 日韩在线看片免费人成视频播放| 暖暖日本免费中文字幕|