jQuery实现瀑布流布局详解(PC和移动端)


Posted in Javascript onSeptember 01, 2020

瀑布流布局已成为当今非常普遍的图片展示方式,无论是PC还是手机等移动设备上。这种布局图片的样式大概分为三种:等高等宽、等宽不等高、等高不等宽,接下来我们就最为普遍的等宽不等高形式来作为示例。

我们用百度图片作为范例:

jQuery实现瀑布流布局详解(PC和移动端)

这就是PC端比较常见的瀑布流布局方式,接下来我们审查下元素看看百度图片里面是如何布局:

jQuery实现瀑布流布局详解(PC和移动端)

可以看到,它里面实际是若干个等宽的列容器,通过计算将图片push到不同的容器里。而本文介绍的展示方法是通过定位的方式,虽然最后布局展示的方式不同,但之前的算法都比较类似。

动手

首先我们将如下样式的若干个单元写进body中,并将“box”向左浮动:

<div class="box">
 <img class="img" src="./resource/images/1.jpg" />
 <div class="desc">Description</div>
</div>
<div class="box">
 <img class="img" src="./resource/images/2.jpg" />
 <div class="desc">Description</div>
</div>
<div class="box">
 <img class="img" src="./resource/images/3.jpg" />
 <div class="desc">Description</div>
</div>

得到如下效果:

jQuery实现瀑布流布局详解(PC和移动端)

接下来:

var boxArr = $('.box'),
  num = Math.floor(document.body.clientWidth / boxArr.eq(0).outerWidth(true)),
 columnHeightArr = [];
columnHeightArr.length = num;
boxArr.each(function(index, item) {
 if (index < num) {
 columnHeightArr[index] = $(item).outerHeight(true);
 } else {
 var minHeight = Math.min.apply(null, columnHeightArr),
 minHeightIndex = $.inArray(minHeight,columnHeightArr);

 $(item).css({
 position: 'absolute',
 top: minHeight,
 left: boxArr.eq(minHeightIndex).position().left
 });
 }
});

以上代码大意为:

1. 首先计算出在浏览器中一行能容纳多少图片 (num) ,注意这里用了outerWidth,当传入true时会返回元素包括margin、padding、border全部盒模型属性的尺寸;

2. 创建一个存储每一列高度的数组 (columnHeightArr) ,该数组的长度即为num值;

3. 遍历所有图片,将第一行的图片高度分别存入列高数组中 (columnHeightArr) ,从第二行开始,首先计算出所有列中最小的高度 (minHeight) 以及最小高度所在的列 (minHeightIndex)。之后将第二行开始的图片定位在高度最小列的下面,效果如下:

jQuery实现瀑布流布局详解(PC和移动端)

可以看到虽然摆对了地方但是所有的图片都放在同一个地方了,这是因为我们需要在摆放一张图片后就要增加该列的高度:

var boxArr = $('.box'),
 num = Math.floor(document.body.clientWidth / boxArr.eq(0).outerWidth(true)),
 columnHeightArr = [];
columnHeightArr.length = num;

boxArr.each(function(index, item) {
if (index < num) {


columnHeightArr[index] = $(item).outerHeight(true);

} else {


var minHeight = Math.min.apply(null, columnHeightArr),
 minHeightIndex = $.inArray(minHeight, columnHeightArr);



$(item).css({
 
position: 'absolute',
 
top: minHeight,
 
left: boxArr.eq(minHeightIndex).position().left
 });



columnHeightArr[minHeightIndex] += $(item).outerHeight(true);

}
});

 结果正确: 

jQuery实现瀑布流布局详解(PC和移动端)

注意:上面的代码需要运行于window.onload方法里,因为只有当页面中的图片资源全部加载完毕后,其每张图片的高度才会有效。

因此会有一些很严重的问题,当网络不好的时候图片没有完全加载完成就会出现图片展示不全高度缺失的情况,这点在移动端很明显。而且当我们加载更多时,更难判断新追加的图片是否加载完成。

在实际生产中更不会有一开始就将图片写死在HTML中的情况,所以我们通常用以下的方式来做:

首先我们在获得图片地址时同时也需要获取图片的宽和高 ,这点对服务端后台来说并不是什么难事,可以拜托后台兄弟将图片的宽高数据拼进JSON,传递给你;

*接下来介绍小技巧,非常实用,它能保证一个元素无论大小如何变化,比例始终保持一致。这个技巧尤其适用于移动端,因为元素为了响应式通常使用百分比的形式。

假如手机页面中有一张图片,其宽度要为屏幕的一半,高宽比为2:1,需要在任何分辨的手机上保持比例不变。如何做?给元素设置如下属性:

.box {
 width: 50%;
 height: 0;
 padding-bottom: 100%;
}

不设置高度,而是用padding“挤”出元素高度,而padding的百分比值都是基于父级容器的宽度。padding需要挤多少呢?就是宽度乘以高宽比(width和padding值均为百分比值),这就是我们为什么需要获得图片尺寸的原因。

效果:

jQuery实现瀑布流布局详解(PC和移动端)

可以看到在chrome手机模拟器中ipone4和肾6Plus的显示效果是完全一样的。在手机页面中宽是固定的,而高会随着页面内容的多少而变化,这个技巧利用元素padding百分比的值其实是基于其父级容器的宽,将高的值巧妙的转化成与宽相关。

说到现在可能有人终于忍不住要问了,讲了这么多和瀑布流有什么关系!简单就是一句话,我们要抛弃 img 标签,而采用背景图的方式。为了使用背景图,就得保持元素的比例永远与图片保持一致。

通过这种方式,可以不用判断图片都加载完毕,直接产生一些与图片同比例的div,再为其设置背景图,如下:

jQuery实现瀑布流布局详解(PC和移动端)

这里比如最外层的box宽度为220px,里面的img元素宽度就可以为100%,高度就可以通过padding挤出了。 

懒加载

使用背景图的方式还有好处那就是可以比较方便的实现懒加载。那什么是懒加载呢?就是当元素在我们的视野中时才展示图片,滚动时屏幕下方的图片并不展示,这可以很好的增加加载速度提升体验。

首先我们给最外层的box增加一个box-item类名(之后有用),将图片url并不设置给backgroundImage属性,而是赋给一个自定义属性:data-src

<div class="box box-item">
 <div class="img" data-src="./resource/images/1.jpg"></div>
 <div class="desc">Description</div>
</div>

接下来我们编写懒加载函数:

function lazyLoad() {
var boxArr = $('.box-item');



boxArr.each(function(index, item) {


var viewTop = $(item).offset().top - $(window).scrollTop(),
 imgObj = $(item).find('.img');

 if ((viewTop < $(window).height()) && (($(item).offset().top + $(item).outerHeight(true)) > $(window).scrollTop())) {
 
imgObj.css('backgroundImage','url('+imgObj.attr("data-src")+')').removeClass('data-src');
 
$(item).removeClass('box-item');
 }
 })
}

首先我们获取所有拥有 .box-item 类名的元素,遍历。viewTop 为图片相对于浏览器窗口的相对高度,类似于position:fixed感觉。

通过条件进行判断,只有当该图片在浏览器窗口内(之上或之下都不行)时,将需要设置背景图元素的 data-src 值展示出来,并删除该属性。

之后将最外层元素的 box-item 删除,因为已经展示出来的图片不需要再进行这些判断,删除了该类名下一次滚动时就不会获取到已经展示过的元素,需要遍历的次数就会越来越少,这样能起到一个优化的作用。

该函数需要在你的元素已经append进页面时调用,以及在滚动时调用:

lazyLoad();
$(window).scroll(lazyLoad);

滚动加载

说完了懒加载,再说说滚动加载。所谓滚动加载就是当页面滚动到底部附近时加载新的图片。我这里选择的是滚动到高度最小的列底部时加载新的数据,你也可以根据自己的喜好来做判断。

function scrollLoad() {
var viewHeight = $(window).scrollTop() + $(window).height(),
 minHeight = Math.min.apply(null, columnHeightArr);


if (viewHeight >= minHeight) {
 
//loadMore...

}
}

滚动加载也是在window的滚动事件中进行监听,可以与懒加载一起进行:

$(window).scroll(function() {
 scrollLoad();
 lazyLoad(); 
});

说完了PC端,我们来说下手机端。其实原理是一样的,只是从多列变成固定的两列了。

var boxArr = $('.box'),
 columnHeightArr = [];
columnHeightArr.length = 2;

boxArr.each(function(index, item) {
if (index < 2) {


columnHeightArr[index] = $(item).outerHeight(true);

} else {


var minHeight = Math.min.apply(null, columnHeightArr),
 minHeightIndex = $.inArray(minHeight, columnHeightArr);



$(item).css({
 
position: 'absolute',
 
top: minHeight,
 
left: boxArr.eq(minHeightIndex).position().left
 });



columnHeightArr[minHeightIndex] += $(item).outerHeight(true);

}
});

不同的是为了适应不同屏幕的手机,最外层的box容器宽度和边距要设置成百分比的形式。

jQuery实现瀑布流布局详解(PC和移动端)

 最后有一点要注意,因为我们没有像百度一样用一个个列盒子去装,而是用定位的方式。导致的问题是图片元素的父级没法自适应高度,如果你有相关的需求我们可以计算出所有列中最长的长度,并将这个值赋值给父容器的min-height属性:

$('body').css('minHeight',Math.max.apply(null, columnHeightArr)); 

整理下完整的代码,瀑布流的全套服务就到这了

var dataArr = [
 {picUrl:'./resource/images/1.jpg',width:522,height:783},
 {picUrl:'./resource/images/2.jpg',width:550,height:786},
 {picUrl:'./resource/images/3.jpg',width:535,height:800},
 {picUrl:'./resource/images/4.jpg',width:578,height:504},
 {picUrl:'./resource/images/5.jpg',width:1440,height:900}
 ];

 $.each(dataArr, function(index, item) {
 $("body").append('<div class="box box-item">' +
 '<div class="img" style="height:0;padding-bottom:'+cRate(item) * 100 + "%"+'" data-src="'+item.picUrl+'"></div>' +
 '<div class="desc">Description</div>' +
 '</div>');
 });

 var boxArr = $('.box'),
 num = Math.floor(document.body.clientWidth / boxArr.eq(0).outerWidth(true)),
 columnHeightArr = [];
 columnHeightArr.length = num;
 
 arrangement();

$('body').css('minHeight',Math.max.apply(null, columnHeightArr));

 lazyLoad();

 function arrangement() {
 boxArr.each(function(index, item) {
 if (index < num) {
 columnHeightArr[index] = $(item).outerHeight(true);
 } else {
 var minHeight = Math.min.apply(null, columnHeightArr),
 minHeightIndex = $.inArray(minHeight, columnHeightArr);
 $(item).css({
 position: 'absolute',
 top: minHeight,
 left: boxArr.eq(minHeightIndex).position().left
 });
 columnHeightArr[minHeightIndex] += $(item).outerHeight(true);
 }
 });
 }

 function lazyLoad() {
 var boxArr = $('.box-item');
 boxArr.each(function(index, item) {
 var viewTop = $(item).offset().top - $(window).scrollTop(),
 imgObj = $(item).find('.img');
 if ((viewTop < $(window).height()) && ($(item).offset().top + $(item).outerHeight(true) > $(window).scrollTop())) {
// console.log($(item).attr('data-src'));
 imgObj.css('backgroundImage','url('+imgObj.attr("data-src")+')').removeClass('data-src');
 $(item).removeClass('box-item');
 }
 })
 }

 function cRate(obj) {
 return obj.height / obj.width;
 }

 function scrollLoad() {
 var viewHeight = $(window).scrollTop() + $(window).height(),
 minHeight = Math.min.apply(null, columnHeightArr);
 if (viewHeight >= minHeight) {
 //loadMore...
 }
 }

 $(window).scroll(function() {
 lazyLoad();
 scrollLoad();
 });

以上就是为大家分享的关于jQuery瀑布流布局,内容很丰富,需要大家一点点的理解消化,真正的做到学以致用,希望能够帮助到大家。

Javascript 相关文章推荐
jQuery实现回车键(Enter)切换文本框焦点的代码实例
May 05 Javascript
Javascript基于AJAX回调函数传递参数实例分析
Dec 15 Javascript
基于AngularJs + Bootstrap + AngularStrap相结合实现省市区联动代码
May 30 Javascript
JS实现的多张图片轮流播放幻灯片效果
Jul 22 Javascript
解决BootStrap Fileinput手机图片上传显示旋转问题
Jun 01 Javascript
微信小程序slider组件使用详解
Jan 31 Javascript
JS设计模式之访问者模式定义与用法分析
Feb 05 Javascript
layui实现数据表格点击搜索功能
Mar 26 Javascript
layui lay-verify form表单自定义验证规则详解
Sep 18 Javascript
小程序最新获取用户昵称和头像的方法总结
Sep 23 Javascript
微信小程序自定义头部导航栏(组件化)
Nov 15 Javascript
手把手教你实现 Promise的使用方法
Sep 02 Javascript
jQuery实现切换页面过渡动画效果
Oct 29 #Javascript
js实现跨域的4种实用方法原理分析
Oct 29 #Javascript
异步JS框架的作用以及实现方法
Oct 29 #Javascript
图解JavaScript中的this关键字
May 28 #Javascript
jquery validate demo 基础
Oct 29 #Javascript
jQuery实现彩带延伸效果的网页加载条loading动画
Oct 29 #Javascript
jquery实现的伪分页效果代码
Oct 29 #Javascript
You might like
php标签云的实现代码
2012/10/10 PHP
phpize的深入理解
2013/06/03 PHP
PHP中unset,array_splice删除数组中元素的区别
2014/07/28 PHP
PHP获取昨天、今天及明天日期的方法
2016/02/03 PHP
js 判断脚本加载完毕的代码
2011/07/13 Javascript
javascript删除数组元素并且数组长度减小的简单实例
2014/02/14 Javascript
使用GruntJS构建Web程序之安装篇
2014/06/04 Javascript
node.js使用require()函数加载模块
2014/11/26 Javascript
创建你的第一个AngularJS应用的方法
2015/06/16 Javascript
JS控制表单提交的方法
2015/07/09 Javascript
js原型链与继承解析(初体验)
2016/05/09 Javascript
老司机带你解读jQuery插件开发流程
2016/05/16 Javascript
Javascript实现跑马灯效果的简单实例
2016/05/31 Javascript
javascript 使用正则test( )第一次是 true,第二次是false
2017/02/22 Javascript
浅谈js使用in和hasOwnProperty获取对象属性的区别
2017/04/27 Javascript
JavaScript的继承实现小结
2017/05/07 Javascript
angularjs中$http异步上传Excel文件方法
2018/02/23 Javascript
Bootstrap开发中Tab标签页切换图表显示问题的解决方法
2018/07/13 Javascript
JavaScript实现连连看连线算法
2019/01/05 Javascript
详解Vuex下Store的模块化拆分实践
2019/07/31 Javascript
[07:08]2014DOTA2西雅图国际邀请赛 小组赛7月11日TOPPLAY
2014/07/11 DOTA
[00:35]2016完美“圣”典风云人物:冷冷宣传片
2016/12/08 DOTA
Django 实现下载文件功能的示例
2018/03/06 Python
Python实现多条件筛选目标数据功能【测试可用】
2018/06/13 Python
Python爬虫 urllib2的使用方法详解
2019/09/23 Python
python实现淘宝购物系统
2019/10/25 Python
python基于socket模拟实现ssh远程执行命令
2020/12/05 Python
CSS中垂直居中的简单实现方法
2015/07/06 HTML / CSS
HTML5 FileReader对象的具体使用方法
2020/05/22 HTML / CSS
迪斯尼商品官方网站:ShopDisney
2016/08/01 全球购物
50道外企软件测试面试题
2014/08/18 面试题
庆祝新中国成立65周年“向国旗敬礼”网上签名寄语
2014/09/27 职场文书
会计人员岗位职责
2015/02/03 职场文书
五四青年节活动总结
2015/02/10 职场文书
用python删除文件夹中的重复图片(图片去重)
2021/05/12 Python
总结三种用 Python 作为小程序后端的方式
2022/05/02 Python