热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

原生JS完成一个瀑布流插件

瀑布流规划中的图片有一个中心特性——等宽不定等高,瀑布流规划在国内网网站都有肯定范围的应用,比方pinterest、花瓣网等等。那末接下来就基于这个特性最先瀑布流探索之旅。基础功用

《原生 JS 完成一个瀑布流插件》

瀑布流规划中的图片有一个中心特性 —— 等宽不定等高,瀑布流规划在国内网网站都有肯定范围的应用,比方pinterest、花瓣网等等。那末接下来就基于这个特性最先瀑布流探索之旅。

基础功用完成

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











...

《原生 JS 完成一个瀑布流插件》

由于未知的 css 知识点,丝袜最长的妹子把下面的空间都占用掉了。。。

接着正文,如果如上图,每排有 5 列,那第 6 张图片应当涌现前 5 张图片哪张的下面呢?当然是相对定位到前 5 张图片高度最小的图片下方。

那第 7 张图片呢?这时刻把第 6 张图片和在它上面的图片看成是一个团体后,思绪和上述是一致的。代码完成以下:

Waterfall.prototype.init = function () {
...
const perNum = this.getPerNum() // 猎取每排图片数
const perList = [] // 存储第一列的各图片的高度
for (let i = 0; i perList.push(imgList[i].offsetHeight)
}
let pointer = this.getMinPointer(perList) // 求出当前最小高度的数组下标
for (let i = perNum; 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;
}

至此完成了瀑布流的基础规划,结果图以下:

《原生 JS 完成一个瀑布流插件》

scroll、resize 事宜监听的完成

完成了初始化函数 init 今后,下一步就要完成对 scroll 转动事宜举行监听,从而完成当滚到父节点的底部有络绎不绝的图片被加载出来的结果。这时刻要斟酌一个点,是转动到什么位置时触发加载函数呢?这个因人而异,我的做法是当满足 父容器高度 + 转动间隔 > 末了一张图片的 offsetTop 这个前提,即橙色线条 + 紫色线条 > 蓝色线条时触发加载函数,代码以下:

《原生 JS 完成一个瀑布流插件》

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 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()
})

项目地点

项目地点

此插件在 React 项目中的应用

项目大略,不足之处在所难免,迎接留下你们珍贵的看法。


推荐阅读
  • Python 伦理黑客技术:深入探讨后门攻击(第三部分)
    在《Python 伦理黑客技术:深入探讨后门攻击(第三部分)》中,作者详细分析了后门攻击中的Socket问题。由于TCP协议基于流,难以确定消息批次的结束点,这给后门攻击的实现带来了挑战。为了解决这一问题,文章提出了一系列有效的技术方案,包括使用特定的分隔符和长度前缀,以确保数据包的准确传输和解析。这些方法不仅提高了攻击的隐蔽性和可靠性,还为安全研究人员提供了宝贵的参考。 ... [详细]
  • 本文深入解析了JDK 8中HashMap的源代码,重点探讨了put方法的工作机制及其内部参数的设定原理。HashMap允许键和值为null,但键为null的情况只能出现一次,因为null键在内部通过索引0进行存储。文章详细分析了capacity(容量)、size(大小)、loadFactor(加载因子)以及红黑树转换阈值的设定原则,帮助读者更好地理解HashMap的高效实现和性能优化策略。 ... [详细]
  • 为了确保iOS应用能够安全地访问网站数据,本文介绍了如何在Nginx服务器上轻松配置CertBot以实现SSL证书的自动化管理。通过这一过程,可以确保应用始终使用HTTPS协议,从而提升数据传输的安全性和可靠性。文章详细阐述了配置步骤和常见问题的解决方法,帮助读者快速上手并成功部署SSL证书。 ... [详细]
  • 在本文中,我们将详细介绍如何构建一个用于自动回复消息的XML类。当微信服务器接收到用户消息时,该类将生成相应的自动回复消息。以下是具体的代码实现:```phpclass We_Xml { // 代码内容}```通过这个类,开发者可以轻松地处理各种消息类型,并实现高效的自动回复功能。我们将深入探讨类的各个方法和属性,帮助读者更好地理解和应用这一技术。 ... [详细]
  • C++ 异步编程中获取线程执行结果的方法与技巧及其在前端开发中的应用探讨
    本文探讨了C++异步编程中获取线程执行结果的方法与技巧,并深入分析了这些技术在前端开发中的应用。通过对比不同的异步编程模型,本文详细介绍了如何高效地处理多线程任务,确保程序的稳定性和性能。同时,文章还结合实际案例,展示了这些方法在前端异步编程中的具体实现和优化策略。 ... [详细]
  • POJ 2482 星空中的星星:利用线段树与扫描线算法解决
    在《POJ 2482 星空中的星星》问题中,通过运用线段树和扫描线算法,可以高效地解决星星在窗口内的计数问题。该方法不仅能够快速处理大规模数据,还能确保时间复杂度的最优性,适用于各种复杂的星空模拟场景。 ... [详细]
  • 依然最钟爱《People Have the Power》,强烈推荐大家聆听这首经典之作
    尽管今日情绪低落,我在音乐库中反复筛选,最终还是选择了《People Have the Power》来激励自己。这首歌不仅旋律动听,歌词也充满力量,能够带给人正能量。强烈建议大家找来聆听,体验其独特的魅力。《People Have the Power》虽然不是出自专辑《Horses》,但同样是一首不可多得的经典之作。 ... [详细]
  • NFT市场热度持续攀升,波卡能否抓住机遇迎来NFT夏季热潮?
    NFT市场热度持续攀升,波卡能否抓住机遇迎来NFT夏季热潮? ... [详细]
  • 深入解析Java虚拟机的内存分区与管理机制
    Java虚拟机的内存分区与管理机制复杂且精细。其中,某些内存区域在虚拟机启动时即创建并持续存在,而另一些则随用户线程的生命周期动态创建和销毁。例如,每个线程都拥有一个独立的程序计数器,确保线程切换后能够准确恢复到之前的执行位置。这种设计不仅提高了多线程环境下的执行效率,还增强了系统的稳定性和可靠性。 ... [详细]
  • 深入解析Android 4.4中的Fence机制及其应用
    在Android 4.4中,Fence机制是处理缓冲区交换和同步问题的关键技术。该机制广泛应用于生产者-消费者模式中,确保了不同组件之间高效、安全的数据传输。通过深入解析Fence机制的工作原理和应用场景,本文探讨了其在系统性能优化和资源管理中的重要作用。 ... [详细]
  • 在Cisco IOS XR系统中,存在提供服务的服务器和使用这些服务的客户端。本文深入探讨了进程与线程状态转换机制,分析了其在系统性能优化中的关键作用,并提出了改进措施,以提高系统的响应速度和资源利用率。通过详细研究状态转换的各个环节,本文为开发人员和系统管理员提供了实用的指导,旨在提升整体系统效率和稳定性。 ... [详细]
  • 在优化Nginx与PHP的高效配置过程中,许多教程提供的配置方法存在诸多问题或不良实践。本文将深入探讨这些常见错误,并详细介绍如何正确配置Nginx和PHP,以实现更高的性能和稳定性。我们将从Nginx配置文件的基本指令入手,逐步解析每个关键参数的最优设置,帮助读者理解其背后的原理和实际应用效果。 ... [详细]
  • 使用 ListView 浏览安卓系统中的回收站文件 ... [详细]
  • Nginx 反向代理配置与应用指南
    本文详细介绍了 Nginx 反向代理的配置与应用方法。首先,用户可以从官方下载页面(http://nginx.org/en/download.html)获取最新稳定版 Nginx,推荐使用 1.14.2 版本。下载并解压后,通过双击 `nginx.exe` 文件启动 Nginx 服务。文章进一步探讨了反向代理的基本原理及其在实际应用场景中的配置技巧,包括负载均衡、缓存管理和安全设置等,为用户提供了一套全面的实践指南。 ... [详细]
  • 本文详细介绍了如何利用CSS技术对链接下划线进行个性化定制和美化,涵盖了多种实用技巧和方法。通过对CSS属性的灵活运用,可以实现不同风格的下划线效果,提升网页的视觉体验。文中不仅提供了基础的代码示例,还结合实际案例进行了深入解析,帮助读者更好地理解和应用这些技巧。此外,文章还引用了《CSS2.0中文手册》中的相关内容,增加了技术的权威性和实用性。 ... [详细]
author-avatar
处女冰水洗尘
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有