详解javascript实现瀑布流绝对式布局


Posted in Javascript onJanuary 29, 2016

瀑布流也应该算是流行几年了吧。首先是由Pinterest掀起的浪潮,然后国内设计如雨后春笋般,冒出很多瀑布流的例子,比如,蘑菇街,Mark之(不过最近涉黄,好像被喝茶了),还有淘宝的 “哇哦”. 这些都是很棒的例子, 今天我们就聊一聊瀑布流。
一、绝对式布局:

详解javascript实现瀑布流绝对式布局

JS实现原理

其实瀑布式主要的难点就在于,如果将图片整齐的排列在对应的列下,以及什么时候开始刷新加载图片。
而图片整齐的排列的主要逻辑和算法即,先获取容器内可以放多少列,然后,通过计算,存放第一列的高度,再遍历剩下(除第一列的元素)的高度,分别放入,高度最小的那一列。 逐个添加,最后结束遍历。
开始刷新的设置就很简单了,瀑布流刷新只和一个事件有关,即,window.onscroll. 主要的算法即,当页面滑动到最低高度的时候开始加载节点并且添加,当然,节点添加的个数是不固定的。
先上代码吧,我这里分3部分讲解,一个是图片的排列,一个是设置加载的位置.另外再补充一个响应式加载。

1、图片排列

var $ = function() {
  return document.querySelectorAll.apply(document, arguments);
 }
 var arrHeight = []; //得到分列的高度
 var columns = function() { //计算页面最多可以放多少列
  var containerW = $("#main")[0].clientWidth,
   pinW = $(".pin")[0].offsetWidth;
  return Math.floor(containerW / pinW);
 }
 var getIndex = function(arr) { //获得最小高度的index
   var minHeight = Math.min.apply(null, arr); //获得最小高度
   for (var i in arr) {
    if (arr[i] === minHeight) {
     return i;
    }
   }
  }
  //根据列数确定第一排img的高度并放入数组当中。
 var setCenter = (function() { //通过列数设置宽度
  var main = $('#main')[0]; //获得罩层
  var getPadding = function() { //设置padding
   var col = columns(); //获得最后一列
   var padding = main.clientWidth - col * $('.pin')[0].offsetWidth;
   return padding / 2;
  }
  var getComputedStyle = function(ele) { //兼容IE的支持情况
   if (window.getComputedStyle) {
    return window.getComputedStyle(ele);
   } else {
    return ele.currentStyle;
   }
  }
  var getPinPad = function() { //获得pin的padding值
   var pin = $(".pin")[0];
   return parseInt(getComputedStyle(pin).paddingLeft);
  }
  return function() { //设置宽度
   main.style.padding = `0 ${getPadding()}px 0 ${getPadding()-getPinPad()}px`;
  }
 })();
 var overLoad = function(ele) {
  var index = getIndex(arrHeight),
   minHeight = Math.min.apply(null, arrHeight), //获取最小高度
   pins = $('.pin'),
   style = ele.style;
  style.position = "absolute";
  style.top = minHeight + "px"; //设置当前元素的高度
  style.left = pins[index].offsetLeft + "px";
  arrHeight[index] += ele.offsetHeight;
 }
 //初始化时执行函数
 var init = function() {
   var pins = $(".pin"),
    col = columns();
   setCenter(); //设置包裹容器的宽度
   for (var i = 0, pin; pin = pins[i]; i++) {
    if (i < col) { //存储第一排的高度
     arrHeight.push(pin.offsetHeight);
    } else {
     overLoad(pin); //将元素的位置重排
    }
   }
  }

里面一共有7个函数(大函数),一个变量。 说一下思路吧。 首先,页面onload之后 执行的函数是 init. 要知道,一个js程序一定有他的入口,关键看你怎么找了。 然后我们深入init函数体,观察。init里面执行的业务逻辑就是 存储第一排元素的高度,然后重排剩下的元素。 通过columns函数来获得当前窗口最多可以放下多少列,然后设置容器的居中(通过padding设置即可). 接下来,遍历pin的单元框,将第一排的元素高度存放在arrHeight数组里,将剩下的元素进行重排。 其他的函数觉得没什么讲解的必要了。 我们着重来看一下overLoad函数.
2、overLoad

var overLoad = function(ele) {
  var index = getIndex(arrHeight),
   minHeight = Math.min.apply(null, arrHeight), //获取最小高度
   pins = $('.pin'),
   style = ele.style;
  style.position = "absolute";
  style.top = minHeight + "px"; //设置当前元素的高度
  style.left = pins[index].offsetLeft + "px";
  arrHeight[index] += ele.offsetHeight;
 }

在overLoad里面有getIndex函数,获取最小高度的index,然后就可以设置传入进来的ele元素的位置(绝对定位), top为数组中最小高度的px值,left就为第一排制定的Index元素的左边距位置。 最后更新一下高度,ok!!! that's enough.
3、设置加载位置

var dataInt = [{
  'src': '1.jpg'
 }, {
  'src': '2.jpg'
 }, {
  'src': '3.jpg'
 }, {
  'src': '4.jpg'
 }, {
  'src': '1.jpg'
 }, {
  'src': '2.jpg'
 }, {
  'src': '3.jpg'
 }, {
  'src': '4.jpg'
 }];

 function isLoad() { //是否可以进行加载
  var scrollTop = document.documentElement.scrollTop || document.body.scrollTop,
   wholeHeight = document.documentElement.clientHeight || document.body.clientHeight,
   point = scrollTop + wholeHeight; //页面底部距离header的距离
  var arr = $('.pin');
  var lastHei = arr[arr.length - 1].getBoundingClientRect().top;
  return (lastHei < point) ? true : false;
 }
 //处理滑动
 var dealScroll = (function() {
  var main = $('#main')[0],
   flag = true;
  return function() {
   if (isLoad() && flag) {
    for (var i = 0, data; data = dataInt[i++];) {
     var div = document.createElement('div');
     div.innerHTML = temp(data.src);
     div.className = "pin";
     main.appendChild(div);
     overLoad(div); //和上面的overload有耦合性质
    }
    flag = false;
    setTimeout(function() { //控制滑行手速,时间越长对速度的滑动时间影响越大。
     flag = true;
    }, 200);
   }
  }
 })();

 function temp(src) {
  return `
  <div class="box">
   <img src="http://cued.xunlei.com/demos/publ/img/P_00${src}"/>
  </div>
 `;
 }

其实,精华就在上一部分,这个只是作为一个加载数据的手段,当然,你可以点击加载(手动触发),或者其他的加载方法。 当然,怎么设置完全是取决于你的。 所以,随大流,依然是通过下滑滚动加载。 继续,找入口函数->dealScroll. 该函数执行的任务就是,通过isload函数,判断是否可以进行加载判断。 我们看一下isload函数,这个就是滚动加载的关键点.

function isLoad() { //是否可以进行加载
  var scrollTop = document.documentElement.scrollTop || document.body.scrollTop,
   wholeHeight = document.documentElement.clientHeight || document.body.clientHeight,
   point = scrollTop + wholeHeight; //页面底部距离header的距离
  var arr = $('.pin');
  var lastHei = arr[arr.length - 1].getBoundingClientRect().top;
  return (lastHei < point) ? true : false;
 }

通过计算得出,页面底部距视口的位置(工具条下部) 与 最后一个元素的绝对位置比较,如果 滑动距离超过,则启用加载。
yeah~ That's over.
back to dealScroll
接下来就是看加载的部分了,这个部分其实也没什么说的,就是创建一个div节点,然后将他放到容器的最后,并且通过overLoad函数来处理该节点的位置。 另外在该函数的末尾,我设置了一个控制滑动速度的trick,通过对函数的节流,防止有时候,请求过慢,用户重复发送请求,造成资源浪费。
然后,这一部分也可以告一段落了。
4、响应式
最后一部分就是响应式了, 这部分,也超级简单,只要你封装性做的好,其实这一部分就是添加一个resize事件就over了。我们继续找入口函数。

var resize = (function() {
   var flag;
   return function(fn) {
    clearTimeout(flag);
    flag = setTimeout(function() { //函数的节流,防止用户过度移动
     fn();
     console.log("ok")
    }, 500);
   }
})();

同样,这里使用到了函数节流的思想,要知道,作为一个程序员,永远不要以为 用户 干不出什么 傻事. 比如,没事的时候拖着浏览器窗口玩,放大,缩小,再放大... 其实,这事我经常干,因为没有女朋友,写代码写累了,就拖浏览器玩。 所以,考虑到我们单身狗的需求,使用函数节流是非常有必要的。 感兴趣的童鞋,可以参考我前面的文章,进行学习。 说明一下,这里的回调函数指的就是init函数,但是 需要对init做些改动。详见。

var update = function(ele) { //当resize的时候,重新设置
  ele.style.position = "initial";
 }
 //初始化时执行函数
 var init = function() {
   var pins = $(".pin"),
    col = columns();
   arrHeight = []; //清空高度
   setCenter(); //设置包裹容器的宽度
   for (var i = 0, pin; pin = pins[i]; i++) {
    if (i < col) { //存储第一排的高度
     arrHeight.push(pin.offsetHeight);
     update(pin);
    } else {
     overLoad(pin); //将元素的位置重排
    }
   }
  }

上面需要加入update,对新的第一排元素进行更新。
然后就可以直接搬过来使用即可。
这就是绝对是布局的大部分内容了,关于javascript瀑布流另一种布局方式请参考下一篇文章《详解javascript实现瀑布流列式布局》

Javascript 相关文章推荐
写自已的js类库需要的核心代码
Jul 16 Javascript
textarea焦点的用法实现获取焦点清空失去焦点提示效果
May 19 Javascript
jquery动态改变form属性提交表单
Jun 03 Javascript
JS按回车键实现登录的方法
Aug 25 Javascript
JQuery中使用on方法绑定hover事件实例
Dec 09 Javascript
一篇文章掌握RequireJS常用知识
Jan 26 Javascript
基于js对象,操作属性、方法详解
Aug 11 Javascript
jquery实现点击页面回到顶部
Nov 23 Javascript
用纯Node.JS弹出Windows系统消息提示框实例(MessageBox)
May 17 Javascript
一步步教你利用webpack如何搭一个vue脚手架(超详细讲解和注释)
Jan 08 Javascript
Bootstrap Table 双击、单击行获取该行及全表内容
Aug 31 Javascript
vue 指令和过滤器的基本使用(品牌管理案例)
Nov 04 Javascript
理解Javascript文件动态加载
Jan 29 #Javascript
JavaScript操作select元素和option的实例代码
Jan 29 #Javascript
JavaScript学习总结之JS、AJAX应用
Jan 29 #Javascript
Angularjs中UI Router全攻略
Jan 29 #Javascript
JavaScript数据结构与算法之集合(Set)
Jan 29 #Javascript
AngularJS 使用 UI Router 实现表单向导
Jan 29 #Javascript
JavaScript数据结构与算法之链表
Jan 29 #Javascript
You might like
简单介绍下 PHP5 中引入的 MYSQLI的用途
2007/03/19 PHP
PHP实现的方程求解示例分析
2016/11/11 PHP
WAF的正确bypass
2017/01/05 PHP
JavaScript.The.Good.Parts阅读笔记(二)作用域&amp;闭包&amp;减缓全局空间污染
2010/11/16 Javascript
DOM_window对象属性之--clipboardData对象操作代码
2011/02/03 Javascript
JS继承--原型链继承和类式继承
2013/04/08 Javascript
jquery图片放大功能简单实现
2013/08/01 Javascript
jQuery中is()方法用法实例
2015/01/06 Javascript
使用AOP改善javascript代码
2015/05/01 Javascript
js代码验证手机号码和电话号码是否合法
2015/07/30 Javascript
基于JS代码实现实时显示系统时间
2016/06/16 Javascript
vue.js入门教程之基础语法小结
2016/09/01 Javascript
angularjs定时任务的设置与清除示例
2017/06/02 Javascript
react 父子组件之间通讯props
2018/09/08 Javascript
JavaScript实现数字前补“0”的五种方法示例
2019/01/03 Javascript
jQuery实现文本显示一段时间后隐藏的方法分析
2019/06/20 jQuery
微信小程序使用车牌号输入法的示例代码
2019/08/20 Javascript
JS轮播图的实现方法
2020/08/24 Javascript
Python实现多并发访问网站功能示例
2017/06/19 Python
matplotlib简介,安装和简单实例代码
2017/12/26 Python
python安装教程
2018/02/28 Python
python实现列表中由数值查到索引的方法
2018/06/27 Python
使用Pyhton 分析酒店针孔摄像头
2020/03/04 Python
Python 统计位数为偶数的数字代码详解
2020/03/15 Python
python中操作文件的模块的方法总结
2021/02/04 Python
numpy实现RNN原理实现
2021/03/02 Python
澳大利亚家具和家居用品在线:BROSA
2017/11/02 全球购物
FirstCry阿联酋儿童和婴儿产品网上购物:FirstCry.ae
2021/02/22 全球购物
C++的几个面试题附答案
2016/08/03 面试题
办公室主任主任岗位责任制
2014/02/11 职场文书
市场拓展计划书
2014/05/03 职场文书
公司活动总结范文
2014/07/01 职场文书
《花钟》教学反思
2016/02/17 职场文书
python如何读取.mtx文件
2021/04/22 Python
python如何正确使用yield
2021/05/21 Python
Pandas 稀疏数据结构的实现
2021/07/25 Python