摘要:瀑布流布局中的圖片有一個(gè)核心特點(diǎn)等寬不定等高,瀑布流布局在國(guó)內(nèi)網(wǎng)網(wǎng)站都有一定規(guī)模的使用,比如花瓣網(wǎng)等等。那么接下來(lái)就基于這個(gè)特點(diǎn)開(kāi)始瀑布流探索之旅。
瀑布流布局中的圖片有一個(gè)核心特點(diǎn) —— 等寬不定等高,瀑布流布局在國(guó)內(nèi)網(wǎng)網(wǎng)站都有一定規(guī)模的使用,比如pinterest、花瓣網(wǎng)等等。那么接下來(lái)就基于這個(gè)特點(diǎn)開(kāi)始瀑布流探索之旅。
基礎(chǔ)功能實(shí)現(xiàn)首先我們定義好一個(gè)有 20 張圖片的容器,
...
由于未知的 css 知識(shí)點(diǎn),絲襪最長(zhǎng)的妹子把下面的空間都占用掉了。。。
接著正文,假如如上圖,每排有 5 列,那第 6 張圖片應(yīng)該出現(xiàn)前 5 張圖片哪張的下面呢?當(dāng)然是絕對(duì)定位到前 5 張圖片高度最小的圖片下方。
那第 7 張圖片呢?這時(shí)候把第 6 張圖片和在它上面的圖片當(dāng)作是一個(gè)整體后,思路和上述是一致的。代碼實(shí)現(xiàn)如下:
Waterfall.prototype.init = function () { ... const perNum = this.getPerNum() // 獲取每排圖片數(shù) const perList = [] // 存儲(chǔ)第一列的各圖片的高度 for (let i = 0; i < perNum; i++) { perList.push(imgList[i].offsetHeight) } let pointer = this.getMinPointer(perList) // 求出當(dāng)前最小高度的數(shù)組下標(biāo) for (let i = perNum; i < imgList.length; i++) { imgList[i].style.position = "absolute" // 核心語(yǔ)句 imgList[i].style.left = `${imgList[pointer].offsetLeft}px` imgList[i].style.top = `${perList[pointer]}px` perList[pointer] = perList[pointer] + imgList[i].offsetHeight // 數(shù)組最小的值加上相應(yīng)圖片的高度 pointer = this.getMinPointer(perList) } }
細(xì)心的朋友也許發(fā)現(xiàn)了代碼中獲取圖片的高度用到了 offsetHeight 這個(gè)屬性,這個(gè)屬性的高度之和等于圖片高度 + 內(nèi)邊距 + 邊框,正因?yàn)榇?,我們用?padding 而不是 margin 來(lái)設(shè)置圖片與圖片之間的距離。此外除了offsetHeight 屬性,此外還要理解 offsetHeight、clientHeight、offsetTop、scrollTop 等屬性的區(qū)別,才能比較好的理解這個(gè)項(xiàng)目。css 代碼簡(jiǎn)單如下:
.waterfall-box { float: left; width: 200px; padding-left: 10px; padding-bottom: 10px; }
至此完成了瀑布流的基本布局,效果圖如下:
scroll、resize 事件監(jiān)聽(tīng)的實(shí)現(xiàn)實(shí)現(xiàn)了初始化函數(shù) init 以后,下一步就要實(shí)現(xiàn)對(duì) scroll 滾動(dòng)事件進(jìn)行監(jiān)聽(tīng),從而實(shí)現(xiàn)當(dāng)滾到父節(jié)點(diǎn)的底部有源源不斷的圖片被加載出來(lái)的效果。這時(shí)候要考慮一個(gè)點(diǎn),是滾動(dòng)到什么位置時(shí)觸發(fā)加載函數(shù)呢?這個(gè)因人而異,我的做法是當(dāng)滿足 父容器高度 + 滾動(dòng)距離 > 最后一張圖片的 offsetTop 這個(gè)條件,即橙色線條 + 紫色線條 > 藍(lán)色線條時(shí)觸發(fā)加載函數(shù),代碼如下:
window.onscroll = function() { // ... if (scrollPX + bsHeight > imgList[imgList.length - 1].offsetTop) {// 瀏覽器高度 + 滾動(dòng)距離 > 最后一張圖片的 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) } }
因?yàn)楦腹?jié)點(diǎn)可能自定義節(jié)點(diǎn),所以提供了對(duì)監(jiān)聽(tīng) scroll 函數(shù)的封裝,代碼如下:
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 事件的監(jiān)聽(tīng)與 scroll 事件監(jiān)聽(tīng)大同小異,當(dāng)觸發(fā)了 resize 函數(shù),調(diào)用 init 函數(shù)進(jìn)行重置就行。
使用發(fā)布-訂閱模式和繼承實(shí)現(xiàn)監(jiān)聽(tīng)綁定既然以開(kāi)發(fā)插件為目標(biāo),不能僅僅滿足于功能的實(shí)現(xiàn),還要留出相應(yīng)的操作空間給開(kāi)發(fā)者自行處理。聯(lián)想到業(yè)務(wù)場(chǎng)景中瀑布流中下拉加載的圖片一般都來(lái)自 Ajax 異步獲取,那么加載的數(shù)據(jù)必然不能寫死在庫(kù)里,期望能實(shí)現(xiàn)如下調(diào)用(此處借鑒了 waterfall 的使用方式),
const waterfall = new Waterfall({options}) waterfall.on("load", function () { // 此處進(jìn)行 ajax 同步/異步添加圖片 })
觀察調(diào)用方式,不難聯(lián)想到使用發(fā)布/訂閱模式來(lái)實(shí)現(xiàn)它,關(guān)于發(fā)布/訂閱模式,之前在 Node.js 異步異聞錄 有介紹它。其核心思想即通過(guò)訂閱函數(shù)將函數(shù)添加到緩存中,然后通過(guò)發(fā)布函數(shù)實(shí)現(xiàn)異步調(diào)用,下面給出其代碼實(shí)現(xiàn):
function eventEmitter() { this.sub = {} } eventEmitter.prototype.on = function (eventName, func) { // 訂閱函數(shù) if (!this.sub[eventName]) { this.sub[eventName] = [] } this.sub[eventName].push(func) // 添加事件監(jiān)聽(tīng)器 } eventEmitter.prototype.emit = function (eventName) { // 發(fā)布函數(shù) 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) // 調(diào)用事件監(jiān)聽(tīng)器 } }
接著,要讓 Waterfall 能使用發(fā)布/訂閱模式,只需讓 Waterfall 繼承 eventEmitter 函數(shù),代碼實(shí)現(xiàn)如下:
function Waterfall(options = {}) { eventEmitter.call(this) this.init(options) // 這個(gè) this 是 new 的時(shí)候,綁上去的 } Waterfall.prototype = Object.create(eventEmitter.prototype) Waterfall.prototype.constructor = Waterfall
繼承方式的寫法吸收了基于構(gòu)造函數(shù)繼承和基于原型鏈繼承兩種寫法的優(yōu)點(diǎn),以及使用 Object.create 隔離了子類和父類,關(guān)于繼承更多方面的細(xì)節(jié),可以另寫一篇文章了,此處點(diǎn)到為止。
小優(yōu)化為了防止 scroll 事件觸發(fā)多次加載圖片,可以考慮用函數(shù)防抖與節(jié)流實(shí)現(xiàn)。在基于發(fā)布-訂閱模式的基礎(chǔ)上,定義了個(gè) isLoading 參數(shù)表示是否在加載中,并根據(jù)其布爾值決定是否加載,代碼如下:
let isLoading = false const scroll = function () { if (isLoading) return false // 避免一次觸發(fā)事件多次 if (scrollPX + bsHeight > imgList[imgList.length - 1].offsetTop) { // 瀏覽器高度 + 滾動(dòng)距離 > 最后一張圖片的 offsetTop isLoading = true this.emit("load") } } proto.done = function () { this.on("done", function () { isLoading = false ... }) this.emit("done") }
這時(shí)候需要在調(diào)用的地方加上 waterfall.done, 從而告知當(dāng)前圖片已經(jīng)加載完畢,代碼如下:
const waterfall = new Waterfall({}) waterfall.on("load", function () { // 異步/同步加載圖片 waterfall.done() })項(xiàng)目地址
項(xiàng)目地址
此插件在 React 項(xiàng)目中的運(yùn)用
項(xiàng)目簡(jiǎn)陋,不足之處在所難免,歡迎留下你們寶貴的意見(jiàn)。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/115601.html
摘要:瀑布流布局中的圖片有一個(gè)核心特點(diǎn)等寬不定等高,瀑布流布局在國(guó)內(nèi)網(wǎng)網(wǎng)站都有一定規(guī)模的使用,比如花瓣網(wǎng)等等。那么接下來(lái)就基于這個(gè)特點(diǎn)開(kāi)始瀑布流探索之旅。 showImg(https://segmentfault.com/img/remote/1460000013059759?w=640&h=280); 瀑布流布局中的圖片有一個(gè)核心特點(diǎn) —— 等寬不定等高,瀑布流布局在國(guó)內(nèi)網(wǎng)網(wǎng)站都有一定規(guī)模...
摘要:瀑布流布局中的圖片有一個(gè)核心特點(diǎn)等寬不定等高,瀑布流布局在國(guó)內(nèi)網(wǎng)網(wǎng)站都有一定規(guī)模的使用,比如花瓣網(wǎng)等等。那么接下來(lái)就基于這個(gè)特點(diǎn)開(kāi)始瀑布流探索之旅。 showImg(https://segmentfault.com/img/remote/1460000013059759?w=640&h=280); 瀑布流布局中的圖片有一個(gè)核心特點(diǎn) —— 等寬不定等高,瀑布流布局在國(guó)內(nèi)網(wǎng)網(wǎng)站都有一定規(guī)模...
摘要:為了保證在拿到圖片高度,也即圖片加載完成后再進(jìn)行排列,我根據(jù)的推薦,選用了這一款插件。另外,在做一些圖片加載效果的時(shí)候也可以用到,比如說(shuō)圖片未加載完成之前放個(gè)圖,加載失敗時(shí)放個(gè)錯(cuò)誤提示什么的都很方便呢。 慣例,首先貼上imagesLoaded的官方網(wǎng)址:http://imagesloaded.desandro.com/ 第一次知道imagesLoaded這個(gè)插件是在做瀑布流布局時(shí),當(dāng)時(shí)...
摘要:前言最近在整理基礎(chǔ)知識(shí),接觸到了幾個(gè)常用的頁(yè)面特效,其中覺(jué)得用原生實(shí)現(xiàn)瀑布流的案例十分有趣,于是與大家分享一下。瀑布流瀑布流,又稱瀑布流式布局。通過(guò)定位的方式是我們實(shí)現(xiàn)瀑布流的最基本的原理,只要我們動(dòng)態(tài)的設(shè)置它的值值,就能讓它排列。 showImg(https://segmentfault.com/img/remote/1460000012621941?w=1052&h=542); 前...
摘要:本位為官方文檔翻譯,原始鏈接安裝下載下載壓縮或未壓縮的壓縮未壓縮在直接飲用文件。排列未加載完成的圖片時(shí)會(huì)導(dǎo)致元素的重疊,可以解決這個(gè)問(wèn)題。布局組件尺寸尺寸配置項(xiàng)和可以可以設(shè)置組件的列寬和間距。增加移除控件在瀑布流末尾增加新控件并重排。 本位為Masonry官方文檔翻譯,原始鏈接 安裝Install 下載 下載壓縮或未壓縮的masonry masonry.pkgd.min.js (壓縮...
閱讀 3230·2023-04-26 02:27
閱讀 2145·2021-11-22 14:44
閱讀 4107·2021-10-22 09:54
閱讀 3203·2021-10-14 09:43
閱讀 759·2021-09-23 11:53
閱讀 12747·2021-09-22 15:33
閱讀 2715·2019-08-30 15:54
閱讀 2691·2019-08-30 14:04