技术心得记录:瀑布流的布局原理分析(纯CSS瀑布流与JS瀑布流)

简介: 技术心得记录:瀑布流的布局原理分析(纯CSS瀑布流与JS瀑布流)

瀑布流


又称瀑布流式布局,是比较流行的一种网站页面布局方式。即多行等宽元素排列,后面的元素依次添加到其后,等宽不等高,根据图片原比例缩放直至宽度达到我们的要求,依次按照规则放入指定位置。


为什么使用瀑布流


瀑布流布局在我们现在的前端页面中经常会用的到,它可以有效的降低页面的复杂度,节省很多的空间,对于整个页面不需要太多的操作,只需要下拉就可以浏览用户需要看到的数据;并且,在当前这个APP至上的时代,瀑布流可以提供很好的用户体验,通过结合下拉刷新,上拉加载进行数据的懒加载等操作,对于用户的体验感来说是接近于满分的!


瀑布流的特点


其实瀑布流的特点就是参差不齐的排列方式,以及流式布局的扩展性,可以通过界面展示给用户多条数据,并且让用户可以有向下浏览的冲动。


瀑布流的代码实现


1.纯 css 瀑布流:( multi-columns 方法 )


// 这里是第一次接触到 column-columns 这个属性,这是一个可以设置将div元素中的文本分成几列


//默认值是:auto


//写法:


column-count:3;


-moz-column-count:3; / Firefox /


-webkit-column-count:3; / Safari and Chrome /


/ 注意:IE9及更早 IE 版本浏览器不支持 column-count 属性 /


//这里还会用到另一个属性 column-gap,用来调整边距,实现瀑布流布局


html结构代码如下:


[span class="hljs-name">div class="demo-1"

[span class="hljs-name">div class="item"

[span class="hljs-name">div class="item_content content-lar" style="height:100px;" > 1


[span class="hljs-name">div class="item"

[span class="hljs-name">div class="item_content content-sma"style="height:150px;" > 2


[span class="hljs-name">div class="item"

[span class="hljs-name">div class="item_content content-mid"style="height:50px;" > 3


//代码效果参考:http://www.jhylw.com.cn/373138457.html

[span class="hljs-name">div class="item"

[span class="hljs-name">div class="item_content content-sma" style="height:200px;" > 4


[span class="hljs-name">div class="item"

[span class="hljs-name">div class="item_content content-mid"style="height:60px;" > 5


[span class="hljs-name">div class="item"//代码效果参考:http://www.jhylw.com.cn/383238661.html<p

/span]

[span class="hljs-name">div class="item_content content-lar"style="height:90px;" > 6


[span class="hljs-name">div class="item"

[span class="hljs-name">div class="item_content content-sma"


[span class="hljs-name">div class="item"

[span class="hljs-name">div class="item_content content-lar"style="height:120px;" > 8


[span class="hljs-name">div class="item"

[span class="hljs-name">div class="item_content content-lar"


[span class="hljs-name">div class="item"

[span class="hljs-name">div class="item_content content-sma" style="height:100px;" > 10


[span class="hljs-name">div class="item"

[span class="hljs-name">div class="item_content content-mid"


[span class="hljs-name">div class="item"

[span class="hljs-name">div class="item_content content-mid"style="height:100px;" > 12



css代码如下:


.demo-1{


-moz-column-count:3; / Firefox /


-webkit-column-count:3; / Safari 和 Chrome /


column-count:3;


-moz-column-gap: 1em;


-webkit-column-gap: 1em;


column-gap: 1em;


Width</span>: 80%;


margin:0 auto;


}


.item {


padding: 2em;


margin-bottom: 2em;


-webkit-column-break-inside: avoid;


break-inside: avoid; /防止断点/


background: #ccc;


text-align: center;


}


效果图:


这里有个弊端,这并不符合瀑布流的原理,如果使用纯css写瀑布流,则每一块都是从上往下排列,不能做到从左到右排列,并且不会识别哪一块图片放在哪个地方合适,若是再配合动态加载,效果会特别不好,所以只能通过js来实现瀑布流。


那么这里用图片来分析一下我们想要的瀑布流是什么样的。


瀑布流的位置分析图解


如下方图片。假设一排放5张图片。当第一排排满足够多的等宽图片时,显示的是这样的。那么假如我们要放第6张图片的时候,应该放在什么位置呢?


如果按照我们的正常逻辑来想,应该是放在第一张图片下面,依次水平排列过去(如下图)


但现实并非如此!在瀑布流中,从第2行开始,接下去的每一张图片都会放在上行中高度最低的那一列图片下方。(如下图)


为什么呢?因为放置它之前,这一列的高度为所有列中最小,所以会放置在这个地方。


那么如果再继续放置下去,第七张图片应该放在第三列图片下方,以此类推。


所以每次加载图片时,会需要判断哪一列的图片累计的高度最小,那么下一张图片就放在哪一列,即瀑布流算法去判断图片的确定位置。


js代码实现


实现思路:


  设定每一列图片的宽度和间距


  获取当前窗口的总宽度,从而根据图片宽度去旁段分成几列


  获取所有图片元素,定义一个空数组来保存高度


  遍历所有容器,开始判断  当页面加载完成,或页面宽度发生变化时,调用函数。


如果当前处于第一行时: 直接设置图片位置【 即 top为间距的大小,left为(当前图片的宽度+间距) 当前图片的值+间距大小 】,并保存当前元素高度。


如果当前不处于第一行时:进行高度对比,通过遍历循环,拿到最小高度和相对应的索引,设置图片位置【 即 top为最小高度值+间距2,left为 (当前图片的宽度+间距) 索引 值+间距大小)】,并修改当前索引的高度为当前元素高度。


  当页面加载完成,或页面宽度发生变化时,调用函数。


广州设计公司 我的007办公资源网站


代码实现


[span class="hljs-name">script type="text/JavaScript"

// 定义瀑布流算法函数


function fall() {


const minGap = 20; // 最小间距,让每一列的最小空隙可以自定义,避免太过拥挤的情况发生。但是,会通过计算得到真实的间距。


const itemWidth = 300; // 每一项的宽度,即当前每一个图片容器的宽度。保证每一列都是等宽不等高的。


const scrollBarWidth = getScrollbarWidth</span>(); // 获取滚动条的宽度


const pageWidth = window.innerWidth</span> - scrollBarWidth; // 获取当前页面的宽度 = window.innerWidth - 滚动条的宽度


const column = Math.floor(pageWidth / (itemWidth + minGap)); // 实际列数=页面宽度/(图片宽度+最小间距)


const gap = (pageWidth - itemWidth column) / column/2; // 计算真实间距 = (页面宽度- 图片宽度实际列数)/实际列数/2


const items = document.querySelectorAll('img'); // 获取所有的外层元素


const heightArr = 【】; // 定义一个空数组,保存最低高度。


// 获取滚动条的宽度


function getScrollbarWidth</span>() {


const oDiv = document.createElement('div');//创建一个div


// 给div设置样式。随便定义宽高,只要能获取到滚动条就可以


oDiv.style.cssText = width: 50px;height: 50px;overflowY: scroll;


document.body.appendChild(oDiv);//把div添加到body中


const scrollbarWidth = oDiv.offsetWidth</span> - oDiv.clientWidth</span>;// 使最大宽度和可视宽度相减,获得到滚动条宽度。


oDiv.remove();//移除创建的div


return scrollbarWidth;//返回滚动条宽度


}


for (let i = 0; i < items.length; i++) {


// 遍历所有的外层容器


const height = items【i】.offsetHeight</span>;


// 如果当前处在第一行


if (i < column) {


// 直接设置元素距离上部的位置和距离左边的距离。


items【i】.style.cssText = `top: ${gap}px;left: ${(itemWidth + gap) i + gap}px;</p> <p> // 保存当前元素的高度。</p> <p> heightArr.push(height);</p> <p> } else {</p> <p> // 不是第一行的话,就进行比对。</p> <p> let minHeight = heightArr【0】; // 先保存第一项的高度</p> <p> let minIndex = 0; // 保存第一项的索引值</p> <p> for (let j = 0; j < heightArr.length; j++) {</p> <p> // 通过循环遍历比对,拿到最小值和最小值的索引。</p> <p> if (minHeight > heightArr【j】) {</p> <p> minHeight = heightArr【j】;</p> <p> minIndex = j;</p> <p> }</p> <p> }</p> <p> // 通过最小值为当前元素设置top值,通过索引为当前元素设置left值。</p> <p> items【i】.style.cssText =top: ${minHeight + gap 2}px; left: ${(itemWidth + gap) minIndex + gap}px;</p> <p> // 并修改当前索引的高度为当前元素的高度</p> <p> heightArr【minIndex】 = minHeight + gap + height;</p> <p> }</p> <p> }</p> <p> }</p> <p> // 页面加载完成调用一次。</p> <p> window.onload = fall;</p> <p> // 页面尺寸发生改变再次调用。</p> <p> window.onresize = fall;</p> <p></script</span]</p> <p>最终效果图:</p> <p>总结瀑布流布局原理</p> <p>设置图片宽度一致</p> <p>根据浏览器宽度以及每列宽度计算出列表个数,列表默认0</p> <p>当图片加载完成,所有图片依次放置在最小的列数下面</p> <p>父容器高度取列表数组的最大值</p> <p>引申知识点</p> <p>let,const以及var三者的区别</p> <p>滚动加载图片(懒加载原理)</p> <p>反撇号()基础知识


style和style.cssTest 的区别

相关文章
|
16天前
|
机器学习/深度学习 JavaScript 前端开发
JS进阶教程:递归函数原理与篇例解析
通过对这些代码示例的学习,我们已经了解了递归的原理以及递归在JS中的应用方法。递归虽然有着理论升华,但弄清它的核心思想并不难。举个随手可见的例子,火影鸣人做的影分身,你看到的都是同一个鸣人,但他们的行为却能在全局产生影响,这不就是递归吗?雾里看花,透过其间你或许已经深入了递归的魅力之中。
66 19
|
1月前
|
监控 算法 JavaScript
基于 JavaScript 图算法的局域网网络访问控制模型构建及局域网禁止上网软件的技术实现路径研究
本文探讨局域网网络访问控制软件的技术框架,将其核心功能映射为图论模型,通过节点与边表示终端设备及访问关系。以JavaScript实现DFS算法,模拟访问权限判断,优化动态策略更新与多层级访问控制。结合流量监控数据,提升网络安全响应能力,为企业自主研发提供理论支持,推动智能化演进,助力数字化管理。
51 4
|
3月前
|
数据采集 前端开发 JavaScript
金融数据分析:解析JavaScript渲染的隐藏表格
本文详解了如何使用Python与Selenium结合代理IP技术,从金融网站(如东方财富网)抓取由JavaScript渲染的隐藏表格数据。内容涵盖环境搭建、代理配置、模拟用户行为、数据解析与分析等关键步骤。通过设置Cookie和User-Agent,突破反爬机制;借助Selenium等待页面渲染,精准定位动态数据。同时,提供了常见错误解决方案及延伸练习,帮助读者掌握金融数据采集的核心技能,为投资决策提供支持。注意规避动态加载、代理验证及元素定位等潜在陷阱,确保数据抓取高效稳定。
98 17
|
4月前
|
数据采集 JavaScript Android开发
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
138 7
【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
|
3月前
|
JavaScript 前端开发 Java
深入理解 JavaScript 中的 Array.find() 方法:原理、性能优势与实用案例详解
Array.find() 是 JavaScript 数组方法中一个非常实用和强大的工具。它不仅提供了简洁的查找操作,还具有性能上的独特优势:返回的引用能够直接影响原数组的数据内容,使得数据更新更加高效。通过各种场景的展示,我们可以看到 Array.find() 在更新、条件查找和嵌套结构查找等场景中的广泛应用。 在实际开发中,掌握 Array.find() 的特性和使用技巧,可以让代码更加简洁高效,特别是在需要直接修改原数据内容的情形。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一
|
3月前
|
监控 JavaScript 前端开发
MutationObserver详解+案例——深入理解 JavaScript 中的 MutationObserver:原理与实战案例
MutationObserver 是一个非常强大的 API,提供了一种高效、灵活的方式来监听和响应 DOM 变化。它解决了传统 DOM 事件监听器的诸多局限性,通过异步、批量的方式处理 DOM 变化,大大提高了性能和效率。在实际开发中,合理使用 MutationObserver 可以帮助我们更好地控制 DOM 操作,提高代码的健壮性和可维护性。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
MutationObserver详解+案例——深入理解 JavaScript 中的 MutationObserver:原理与实战案例
|
4月前
|
前端开发
【2025优雅草开源计划进行中01】-针对web前端开发初学者使用-优雅草科技官网-纯静态页面html+css+JavaScript可直接下载使用-开源-首页为优雅草吴银满工程师原创-优雅草卓伊凡发布
【2025优雅草开源计划进行中01】-针对web前端开发初学者使用-优雅草科技官网-纯静态页面html+css+JavaScript可直接下载使用-开源-首页为优雅草吴银满工程师原创-优雅草卓伊凡发布
115 1
【2025优雅草开源计划进行中01】-针对web前端开发初学者使用-优雅草科技官网-纯静态页面html+css+JavaScript可直接下载使用-开源-首页为优雅草吴银满工程师原创-优雅草卓伊凡发布
|
5月前
|
人工智能 程序员 UED
【01】完成新年倒计时页面-蛇年新年快乐倒计时领取礼物放烟花html代码优雅草科技央千澈写采用html5+div+CSS+JavaScript-优雅草卓伊凡-做一条关于新年的代码分享给你们-为了C站的分拼一下子
【01】完成新年倒计时页面-蛇年新年快乐倒计时领取礼物放烟花html代码优雅草科技央千澈写采用html5+div+CSS+JavaScript-优雅草卓伊凡-做一条关于新年的代码分享给你们-为了C站的分拼一下子
224 21
【01】完成新年倒计时页面-蛇年新年快乐倒计时领取礼物放烟花html代码优雅草科技央千澈写采用html5+div+CSS+JavaScript-优雅草卓伊凡-做一条关于新年的代码分享给你们-为了C站的分拼一下子
|
4月前
|
监控 网络协议 算法
基于问题“如何监控局域网内的电脑”——Node.js 的 ARP 扫描算法实现局域网内计算机监控的技术探究
在网络管理与安全领域,监控局域网内计算机至关重要。本文探讨基于Node.js的ARP扫描算法,通过获取IP和MAC地址实现有效监控。使用`arp`库安装(`npm install arp`)并编写代码,可定期扫描并对比设备列表,判断设备上线和下线状态。此技术适用于企业网络管理和家庭网络安全防护,未来有望进一步提升效率与准确性。
105 8
|
5月前
|
前端开发 JavaScript
【02】v1.0.1更新增加倒计时完成后的放烟花页面-优化播放器-优化结构目录-蛇年新年快乐倒计时领取礼物放烟花html代码优雅草科技央千澈写采用html5+div+CSS+JavaScript-优雅草卓伊凡-做一条关于新年的代码分享给你们-为了C站的分拼一下子
【02】v1.0.1更新增加倒计时完成后的放烟花页面-优化播放器-优化结构目录-蛇年新年快乐倒计时领取礼物放烟花html代码优雅草科技央千澈写采用html5+div+CSS+JavaScript-优雅草卓伊凡-做一条关于新年的代码分享给你们-为了C站的分拼一下子
120 14
【02】v1.0.1更新增加倒计时完成后的放烟花页面-优化播放器-优化结构目录-蛇年新年快乐倒计时领取礼物放烟花html代码优雅草科技央千澈写采用html5+div+CSS+JavaScript-优雅草卓伊凡-做一条关于新年的代码分享给你们-为了C站的分拼一下子