瀑布流javascript
又称瀑布流式布局,是比较流行的一种网站页面布局方式。即多行等宽元素排列,后面的元素依次添加到其后,等宽不等高,根据图片原比例缩放直至宽度达到咱们的要求,依次按照规则放入指定位置。css
为何使用瀑布流html
瀑布流布局在咱们如今的前端页面中常常会用的到,它能够有效的下降页面的复杂度,节省不少的空间,对于整个页面不须要太多的操做,只须要下拉就能够浏览用户须要看到的数据;而且,在当前这个APP至上的时代,瀑布流能够提供很好的用户体验,经过结合下拉刷新,上拉加载进行数据的懒加载等操做,对于用户的体验感来讲是接近于满分的!前端
瀑布流的特色java
其实瀑布流的特色就是良莠不齐的排列方式,以及流式布局的扩展性,能够经过界面展现给用户多条数据,而且让用户能够有向下浏览的冲动。web
瀑布流的代码实现算法
1.纯 css 瀑布流:( multi-columns 方法 )数组
// 这里是第一次接触到 column-columns 这个属性,这是一个能够设置将div元素中的文本分红几列浏览器
//默认值是:autoapp
//写法:
column-count:3;
-moz-column-count:3; / Firefox /
-webkit-column-count:3; / Safari and Chrome /
/ 注意:IE9及更早 IE 版本浏览器不支持 column-count 属性 /
//这里还会用到另外一个属性 column-gap,用来调整边距,实现瀑布流布局
html结构代码以下:
学习Q-q-u-n: 784783012 ,分享学习的方法和须要注意的小细节,不停更新最新的教程和学习技巧 (从零基础开始到前端项目实战教程,学习工具,全栈开发学习路线以及规划) <div class="demo-1"> <div class="item"> <div class="item_content content-lar" style="height:100px;" > 1</div> </div> <div class="item"> <div class="item_content content-sma"style="height:150px;" > 2</div> </div> <div class="item"> <div class="item_content content-mid"style="height:50px;" > 3</div> </div> <div class="item"> <div class="item_content content-sma" style="height:200px;" > 4</div> </div> <div class="item"> <div class="item_content content-mid"style="height:60px;" > 5 </div> </div> <div class="item"> <div class="item_content content-lar"style="height:90px;" > 6</div> </div> <div class="item"> <div class="item_content content-sma"> 7</div> </div> <div class="item"> <div class="item_content content-lar"style="height:120px;" > 8</div> </div> <div class="item"> <div class="item_content content-lar"> 9</div> </div> <div class="item"> <div class="item_content content-sma" style="height:100px;" > 10 </div> </div> <div class="item"> <div class="item_content content-mid"> 11 </div> </div> <div class="item"> <div class="item_content content-mid"style="height:100px;" > 12</div> </div> <!-- more items --> </div>
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: 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代码实现
实现思路:
代码实现
<script type="text/javascript"> // 定义瀑布流算法函数 function fall() { const minGap = 20; // 最小间距,让每一列的最小空隙能够自定义,避免太过拥挤的状况发生。可是,会经过计算获得真实的间距。 const itemWidth = 300; // 每一项的宽度,即当前每个图片容器的宽度。保证每一列都是等宽不等高的。 const scrollBarWidth = getScrollbarWidth(); // 获取滚动条的宽度 const pageWidth = window.innerWidth - 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() { 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 - oDiv.clientWidth;// 使最大宽度和可视宽度相减,得到到滚动条宽度。 oDiv.remove();//移除建立的div return scrollbarWidth;//返回滚动条宽度 } for (let i = 0; i < items.length; i++) { // 遍历全部的外层容器 const height = items[i].offsetHeight; // 若是当前处在第一行 if (i < column) { // 直接设置元素距离上部的位置和距离左边的距离。 items[i].style.cssText = `top: ${gap}px;left: ${(itemWidth + gap) * i + gap}px`; // 保存当前元素的高度。 heightArr.push(height); } else { // 不是第一行的话,就进行比对。 let minHeight = heightArr[0]; // 先保存第一项的高度 let minIndex = 0; // 保存第一项的索引值 for (let j = 0; j < heightArr.length; j++) { // 经过循环遍历比对,拿到最小值和最小值的索引。 if (minHeight > heightArr[j]) { minHeight = heightArr[j]; minIndex = j; } } // 经过最小值为当前元素设置top值,经过索引为当前元素设置left值。 items[i].style.cssText = `top: ${minHeight + gap *2}px; left: ${(itemWidth + gap) * minIndex + gap}px`; // 并修改当前索引的高度为当前元素的高度 heightArr[minIndex] = minHeight + gap + height; } } } // 页面加载完成调用一次。 window.onload = fall; // 页面尺寸发生改变再次调用。 window.onresize = fall; </script> 学习Q-q-u-n: 784783012 ,分享学习的方法和须要注意的小细节,不停更新最新的教程和学习技巧 (从零基础开始到前端项目实战教程,学习工具,全栈开发学习路线以及规划)
最终效果图:
总结瀑布流布局原理
引伸知识点