自己用jQuery写一个瀑布流
前言
这个月一直在忙工作。一直没有机会学习新的知识。前两天,突然想写一个瀑布流代码。倒不是找不到瀑布流代码。而是我想自己练练脑子。
首先,先考虑思路。所有的图片,全部采用相对父目录定位的方式。然后用jQuery
来找出它应该排在什么位置。最终,达成瀑布流的效果。
想再多没用。开干。
构建html构架
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>瀑布流jquery版本测试</title>
<link rel="stylesheet" href="style/style.css">
<script src="js/jquery/jquery.js"></script>
<script src="js/FengWaterFall.beta2.js"></script>
</head>
<body>
<h1 class="tc cf">瀑布流jquery版本测试</h1>
<section id="waterfall">
<ul class="piclist">
<li><img src="image/1.jpg" alt=""><span>瀑布流测试1</span></li>
<li><img src="image/2.jpg" alt=""><span>瀑布流测试2</span></li>
<li><img src="image/3.jpg" alt=""><span>瀑布流测试3</span></li>
<li><img src="image/4.jpg" alt=""><span>瀑布流测试4</span></li>
<li><img src="image/5.jpg" alt=""><span>瀑布流测试5</span></li>
<li><img src="image/6.jpg" alt=""><span>瀑布流测试6</span></li>
<li><img src="image/7.jpg" alt=""><span>瀑布流测试7</span></li>
<li><img src="image/8.jpg" alt=""><span>瀑布流测试8</span></li>
<li><img src="image/9.jpg" alt=""><span>瀑布流测试9</span></li>
<li><img src="image/10.jpg" alt=""><span>瀑布流测试10</span></li>
<li><img src="image/1.jpg" alt=""><span>瀑布流测试1</span></li>
<li><img src="image/2.jpg" alt=""><span>瀑布流测试2</span></li>
<li><img src="image/3.jpg" alt=""><span>瀑布流测试3</span></li>
<li><img src="image/4.jpg" alt=""><span>瀑布流测试4</span></li>
<li><img src="image/5.jpg" alt=""><span>瀑布流测试5</span></li>
<li><img src="image/6.jpg" alt=""><span>瀑布流测试6</span></li>
<li><img src="image/7.jpg" alt=""><span>瀑布流测试7</span></li>
<li><img src="image/8.jpg" alt=""><span>瀑布流测试8</span></li>
<li><img src="image/9.jpg" alt=""><span>瀑布流测试9</span></li>
<li><img src="image/10.jpg" alt=""><span>瀑布流测试10</span></li>
</ul>
</section>
</body>
</html>
SCSS代码
@charset "UTF-8";
@import "reset.scss";
.piclist {
margin: auto;position: relative;
li {width: 180px;padding: 10px;border-radius: 5px;box-shadow: 0 0 4px #ddd;position: absolute;}
img {display: block;width: 180px;}
span {display: block;text-align: center;height: 26px;overflow: hidden;line-height: 26px;}
}
关于 reser.scss 初始文件。其实里面没啥,只是清除了默认样式而已。大家有兴趣,可以看一下我之前写的博文:
页面宽度设置为100%
$WinWitdh:100%;
jquery 实战 第一回合
/*
FengWaterFall.beta1.js
这个版本基本实现了瀑布流的效果。但是其算法不是很优秀。
当顺序操作到图片时,它不是找空白最大的一列,所以某些情况看上去会很怪。
2015年10月15日 15:07:03
*/
$(function(){
var Obj = $("#waterfall"),
Ul = Obj.children('ul'),
Li = Ul.children('li');
waterfall();
$(window).resize(function(event) {
waterfall();
});
function waterfall(){
var WinW = $(window).width();
var Blank = 20, // 每个图片之间的间隔留白
LiW = 200+Blank, // 一个图片距离上一个图片的宽度距离
LiCol = parseInt(WinW/LiW), // 计算在当前窗口下,有几列
UlW = LiCol*LiW-Blank; // 根据有几列,设定总的宽度(减掉最后一个留白)
Ul.width(UlW);
Li.each(function(i, e){
var T = $(this),
TCol = i%LiCol, // 用求余数的方法获知当前图片为一行中的第几个
TRow = parseInt(i/LiCol); // 当前索引除以列数,并取整,得知为第几行
/*
取得位于当前图片上侧的图片元素
获取这个元素本身的高度,和父元素顶部的距离,加上空格留白,得出当前图片距离顶部的距离。
*/
var PrevLi = Li.eq(i-LiCol),
PrevLiSize = PrevLi.innerHeight()+PrevLi.position().top+Blank;
TRow==0 ? Ttop = 0 : Ttop = PrevLiSize+'px';
var Tleft = TCol*LiW+'px';
T.css({
top: Ttop,
left: Tleft
});
});
}
})
实现了第一个方法之后,通过预览效果,感觉算法不是很好。所以,开始勾勒第二版。
jquery 实战 第二回合
/*
FengWaterFall.beta2.js
完美实现了瀑布流效果。其中使用的是for循环来查找索引。
*/
$(function(){
var Obj = $("#waterfall"),
Ul = Obj.children('ul'),
Li = Ul.children('li');
waterfall();
$(window).resize(function(event) {
waterfall();
});
function waterfall(){
var WinW = $(window).width();
var Blank = 20, // 每个图片之间的间隔留白
LiW = 200+Blank, // 一个图片距离上一个图片的宽度距离
LiCol = parseInt(WinW/LiW), // 计算在当前窗口下,有几列
UlW = LiCol*LiW-Blank; // 根据有几列,设定总的宽度(减掉最后一个留白)
Ul.width(UlW);
var AllLi = []; // 建立一个空数组变量
Li.each(function(index, e){
var T = $(this);
if (index<LiCol) { // 第一行的处理
AllLi[index] = T.outerHeight(); // 给数组添加内容,为当前元素的高度值
T.css({
top: 0,
left: LiW*index+'px'
});
} else{
var MinH = Math.min.apply(null,AllLi); // 找到数组中,最小的那个值(也就是留白最大的)
// 通过 for 循环数组 找到最小的这个数字所在的索引值
for (var i = 0; i < AllLi.length; i++) {
if (AllLi[i] == MinH) {
var MinI = i;
};
};
var ThisH = T.outerHeight()+Blank; // 自身的高度加上留白
AllLi[MinI] = parseInt(MinH+ThisH); // 将被占位的数组重新赋值
T.css({
top: MinH+Blank+'px',
left: LiW*MinI+'px'
});
};
});
// console.log(AllLi)
}
})
第二个方法的逻辑是完全正确了。实现效果也是完美的。
不过for循环比较怪。jquery嘛,应该用jquery的方法来实现。
jquery 实战 第三回合
/*
FengWaterFall.beta3.js
完美实现了瀑布流效果。jquery的each方法循环数组。但是被迫用了全局变量。
*/
$(function(){
var Obj = $("#waterfall"),
Ul = Obj.children('ul'),
Li = Ul.children('li');
waterfall();
$(window).resize(function(event) {
waterfall();
});
function waterfall(){
var WinW = $(window).width();
var Blank = 20, // 每个图片之间的间隔留白
LiW = 200+Blank, // 一个图片距离上一个图片的宽度距离
LiCol = parseInt(WinW/LiW), // 计算在当前窗口下,有几列
UlW = LiCol*LiW-Blank; // 根据有几列,设定总的宽度(减掉最后一个留白)
Ul.width(UlW);
var AllLi = []; // 建立一个空数组变量
Li.each(function(index, e){
var T = $(this);
if (index<LiCol) { // 第一行的处理
AllLi[index] = T.outerHeight(); // 给数组添加内容,为当前元素的高度值
T.css({
top: 0,
left: LiW*index+'px'
});
} else{
var MinH = Math.min.apply(null,AllLi); // 找到数组中,最小的那个值(也就是留白最大的)
// 通过each 循环数组的方式,找到索引
$.each(AllLi,function(index,value){
if (value == MinH) {
MinI = index; // 因为 var 局部变量不能被用到下面去,所以用了全局变量
};
});
var ThisH = T.outerHeight()+Blank; // 自身的高度加上留白
AllLi[MinI] = parseInt(MinH+ThisH); // 将被占位的数组重新赋值
T.css({
top: MinH+Blank+'px',
left: LiW*MinI+'px'
});
};
});
// console.log(AllLi)
}
})
总结
这时候,我去看了一下别人是怎么写的。结果发现整体思路如出一辙。但是其查找索引的用法比我用for或者each循环要简单可靠得多。不过我还没有弄懂到底是怎么运作的。所以,就不做搬运工了。