JS代码实现瀑布流插件

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

基础功能实现

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

   #waterfall {   position: relative;  }  .waterfall-box {   float: left;   width: 200px;  } 

  @@##@@  @@##@@  @@##@@  @@##@@  @@##@@  @@##@@  ... 

由于未知的 css 知识点,丝袜最长的妹子把下面的空间都占用掉了。。。接着正文,假如如上图,每排有 5 列,那第 6 张图片应该出现前 5 张图片哪张的下面呢?当然是绝对定位到前 5 张图片高度最小的图片下方。那第 7 张图片呢?这时候把第 6 张图片和在它上面的图片当作是一个整体后,思路和上述是一致的。代码实现如下:Waterfall.prototype.init = function () { ... const perNum = this.getPerNum() // 获取每排图片数 const perList = []       // 存储第一列的各图片的高度 for (let i = 0; i 

细心的朋友也许发现了代码中获取图片的高度用到了 offsetHeight 这个属性,这个属性的高度之和等于图片高度 + 内边距 + 边框,正因为此,我们用了 padding 而不是 margin 来设置图片与图片之间的距离。此外除了offsetHeight 属性,此外还要理解 offsetHeight、clientHeight、offsetTop、scrollTop 等属性的区别,才能比较好的理解这个项目。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 

因为父节点可能自定义节点,所以提供了对监听 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 

接着,要让 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 = falseconst 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()})

登录后复制

相关推荐:

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

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

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

JS代码实现瀑布流插件JS代码实现瀑布流插件JS代码实现瀑布流插件JS代码实现瀑布流插件JS代码实现瀑布流插件JS代码实现瀑布流插件

以上就是JS代码实现瀑布流插件的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/3125062.html

(0)
上一篇 2025年3月29日 19:38:22
下一篇 2025年3月29日 19:38:38

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • localStorage存储读取JSON怎样实现

    这次给大家带来localStorage存储读取JSON怎样实现,localStorage存储读取JSON的注意事项有哪些,下面就是实战案例,一起来看一下。 localStorage是HTML5提供的再注意事项本地存储的一种方法,但是loca…

    编程技术 2025年3月29日
    100
  • data-*与js的交互

    这次给大家带来data-*与js的交互,data-*与js的交互的注意事项有哪些,下面就是实战案例,一起来看一下。 HTML5注意事项注意事项data-* 书写实例 登录后复制 1. 定义: data-* 属性用于存储页面或应用程序的私有自…

    编程技术 2025年3月29日
    100
  • 操作Vue渲染与插件加载的顺序

    这次给大家带来Vue渲染与插件加载顺序,操作Vue渲染与插件加载顺序的注意事项有哪些,下面就是实战案例,一起来看一下。 可以通过Vue中的注意事项Tick来解决 Vue.nextTick(function() { //widget}); 登…

    编程技术 2025年3月29日
    100
  • JsChart的组件使用详解

    这次给大家带来JsChart的组件使用详解,JsChart组件使用的注意事项有哪些,下面就是实战案例,一起来看一下。 JsChart是什么? JSChart能够在网页上生成图标,常用于统计信息,十分好用的一个JS组件。 使用JsChart …

    编程技术 2025年3月29日
    100
  • 使用javascript的模块加载器

    这次给大家带来使用javascript的模块加载器,使用javascript模块加载器的使用javascript有哪些,下面就是实战案例,一起来看一下。 定义 var MyModules = (function Manager() { va…

    编程技术 2025年3月29日
    100
  • vue.js的select下拉框怎样绑定事件和取值

    这次给大家带来vue.js的select下拉框怎样绑定时间和取值,select下拉框怎样绑定事件和取值的事件有哪些,下面就是实战案例,一起来看一下。 最近在做mui+vue.js的移动项目,遇到了这个解决了,所以记录一下: 1、绑定sele…

    编程技术 2025年3月29日
    100
  • 在Vuejs里利用index对第一项添加class的方法

    这次给大家带来在Vuejs里利用index对第一项添加class的方法,在Vuejs里利用index对第一项添加class的class有哪些,下面就是实战案例,一起来看一下。 (1)在v-for中,利用index来对第一项添加class 在…

    编程技术 2025年3月29日
    100
  • vue.js-div怎么隐藏滚动条

    这次给大家带来vue.js-div怎么隐藏滚动条,vue.js-div隐藏滚动条的注意事项有哪些,下面就是实战案例,一起来看一下。 组件被包在一个高度固定的p mounted () { var bop = document.getEleme…

    编程技术 2025年3月29日
    100
  • angularjs的内存溢出怎么处理

    这次给大家带来angularjs的内存溢出怎么处理,处理angularjs内存溢出的angularjs有哪些,下面就是实战案例,一起来看一下。 这次给大家带来angularjs的内存溢出怎么处理,处理angularjs内存溢出的注意事项有哪…

    编程技术 2025年3月29日
    100
  • 怎样使用javascript Date Format方法

    这次给大家带来怎样使用javascript Date Format方法,使用javascript Date Format使用javascript有哪些,下面就是实战案例,一起来看一下。 方法一: // 对Date的扩展,将 Date 转化为…

    编程技术 2025年3月29日
    100

发表回复

登录后才能评论