JS代码实现瀑布流插件


本文摘自PHP中文网,作者小云云,侵删。

瀑布流布局中的图片有一个核心特点―等宽不定等高,瀑布流布局在国内网网站都有一定规模的使用,比如pinterest、花瓣网等等。本文主要和大家详细分析了一个原生JS实现瀑布流插件以及代码相关讲解,对此有兴趣的读者们参考学习下吧,希望能帮助到大家。

基础功能实现

首先我们定义好一个有 20 张图片的容器,


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

<body>

 <style>

  #waterfall {

   position: relative;

  }

  .waterfall-box {

   float: left;

   width: 200px;

  }

 </style>

</body>

<p 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">

  ...

 </p>

由于未知的 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 代码简单如下:


1

2

3

4

5

6

.waterfall-box {

 float: left;

 width: 200px;

 padding-left: 10px;

 padding-bottom: 10px;

}

scroll、resize 事件监听的实现

实现了初始化函数 init 以后,下一步就要实现对 scroll 滚动事件进行监听,从而实现当滚到父节点的底部有源源不断的图片被加载出来的效果。这时候要考虑一个点,是滚动到什么位置时触发加载函数呢?这个因人而异,我的做法是当满足 父容器高度 + 滚动距离 > 最后一张图片的 offsetTop 这个条件,即橙色线条 + 紫色线条 > 蓝色线条时触发加载函数,代码如下:


1

2

3

4

5

6

7

8

9

10

11

12

13

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 函数的封装,代码如下:


1

2

3

4

5

6

7

8

9

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 的使用方式),


1

2

3

4

const waterfall = new Waterfall({options})

waterfall.on("load", function () {

 // 此处进行 ajax 同步/异步添加图片

})

观察调用方式,不难联想到使用发布/订阅模式来实现它,关于发布/订阅模式,之前在 Node.js 异步异闻录 有介绍它。其核心思想即通过订阅函数将函数添加到缓存中,然后通过发布函数实现异步调用,下面给出其代码实现:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

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 函数,代码实现如下:


1

2

3

4

5

6

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 参数表示是否在加载中,并根据其布尔值决定是否加载,代码如下:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

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, 从而告知当前图片已经加载完毕,代码如下:


1

2

3

4

5

const waterfall = new Waterfall({})

waterfall.on("load", function () {

 // 异步/同步加载图片

 waterfall.done()

})

相关推荐:

纯原生JS的瀑布流插件Macy.js使用详解

Jquery瀑布流插件使用介绍_jquery

jQuery瀑布流插件Wookmark使用实例_jquery

以上就是JS代码实现瀑布流插件的详细内容,更多文章请关注木庄网络博客

相关阅读 >>

javascript是不是弱语言

node.js“多线程”如何处理高并发任务?

javascript中求最大值语句是什么

详解css和js动画底层原理及如何优化它们的性能

详解javascript中逻辑运算符

vue框架是什么

javascript可以获取input的值吗

轻松理解函数防抖和节流的使用

javascript和java区别有什么

javascript怎么显示隐藏div

更多相关阅读请进入《javascript》频道 >>




打赏

取消

感谢您的支持,我会继续努力的!

扫码支持
扫码打赏,您说多少就多少

打开支付宝扫一扫,即可进行扫码打赏哦

分享从这里开始,精彩与您同在

评论

管理员已关闭评论功能...