基于JavaScript实现瀑布流布局


Posted in Javascript onAugust 15, 2018

本文实例为大家分享了js实现瀑布流布局的具体代码,供大家参考,具体内容如下

1、html+css+js代码:

<!DOCTYPE html>
<html>
<head>
  <meta http-equiv="Content-Type" content="text/html" charset="utf-8" />
  <title>hhh</title>
</head>
<body>
  <style type="text/css">
    *{
      padding: 0;
      margin: 0;
    }
    #main{
      position: relative;
    }
    .pin{
      float: left;
      padding: 15px 0 0 15px;
    }
    .box{
      border-radius: 5px;
      box-shadow: 0 0 6px #ccc;
      border:1px solid #ccc;
      padding: 10px;

    }
    .box img{
      width: 162px;
      height:auto;
    }
  </style>

<script type="text/javascript">
  window.onload = function(){

    waterfall("main","pin");

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

    window.onscroll = function(){
      if (checkscrollside()) {
        var oparent = document.getElementById('main');
        for (var i = 0; i < dataint.data.length; i++) {
          var opin = document.createElement('div');
          opin.className = 'pin';
          oparent.appendChild(opin);
          var obox = document.createElement('div');
          obox.className = 'box';
          opin.appendChild(obox);
          var oimg = document.createElement('img');
          oimg.src = './images/' +dataint.data[i].src;
          obox.appendChild(oimg);
        }
        waterfall('main','pin');
      };
    }


  }

  //parent为父元素的id,pin为子元素id
  function waterfall(parent,pin){
    var oparent = document.getElementById(parent);
    var apin = getclassobj(oparent,pin);//获取id为oparent的类名为pin的元素
    var ipinw = apin[0].offsetWidth;
    var num = Math.floor(document.documentElement.clientWidth/ipinw);
    oparent.style.cssText = 'width:' + ipinw*num + 'px;margin:0 auto;';

    var pinharr = [];
    for( var i = 0;i < apin.length; i++)
    {
      var pinh = apin[i].offsetHeight;
      if (i < num) {
        pinharr[i] = pinh;
      }
      else{
        var minh = Math.min.apply(null,pinharr);
        var minhindex = getminhindex(pinharr,minh);
        apin[i].style.position = 'absolute';
        apin[i].style.top = minh +'px';
        apin[i].style.left = apin[minhindex].offsetLeft + 'px';
        pinharr[minhindex] += apin[i].offsetHeight; 
      }

    }
  }

  //获取id为parent的类名为classname的元素
  function getclassobj(parent,classname){

    var obj = parent.getElementsByTagName('*');
    var pins = [];
    for (var i = 0; i < obj.length; i++) {
      if (obj[i].className == classname) {
        pins.push(obj[i]);
      }
    };
    return pins;
  }

  function getminhindex(arr,minh){
    for(var i in arr){
      if (arr[i] == minh) {
        return i;
      }
    }
  }

  function checkscrollside(){
    var oparent = document.getElementById('main');
    var apin = getclassobj(oparent,'pin');
    var lastpinh = apin[apin.length - 1].offsetTop + Math.floor(apin[apin.length - 1].offsetHeight/2);
    var scrollTop = document.documentElement.scrollTop||document.body.scrollTop;
    var documenth = document.documentElement.clientHeight;
    return(lastpinh<scrollTop + documenth)?true:false;
  }
</script>
</body>
  <div id="main">
    <div class="pin">
      <div class="box">
        <img src="images/0.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/1.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/2.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/3.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/4.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/5.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/6.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/7.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/8.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/9.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/10.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/11.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/12.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/13.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/14.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/15.jpg">
      </div>
    </div>

    <div class="pin">
      <div class="box">
        <img src="images/16.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/17.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/18.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/19.jpg">
      </div>
    </div>

    <div class="pin">
      <div class="box">
        <img src="images/20.jpg">
      </div>
    </div>
    <div class="pin">
      <div class="box">
        <img src="images/21.jpg">
      </div>
    </div>
  </div>
</html>

2、思路分析

首先做出静态布局来。先计算出一行放多少个元素,然后再让下一个元素放入第二行中,然后我们要计算出,放入第二行的第一个元素要放在哪个位置,故我们应该有一个数组用来存放每列的高度,当静态的这些元素都放进去之后,可以更加完善,比如当拖动滚动条的时候,页面刷新,更多元素填充,这里要用到json。

3、实现过程

1.初始的静态页面通过css来实现

2.静态的瀑布流布局,先要获取到父级对象main下面的所有类为pin的元素,然后计算一行中有几个块,此时用当前屏幕的宽度去除一个块的宽度,得到num。然后用一个数组,用来存储一行中每列的高度,通过循环,找出最小的高度,以及最小高度的下标值,然后用绝对定位设置高度。

3.当鼠标滚动的时候,通过一个函数来检查是否需要加载新的图片元素,这里用一个变量lastpinh计算出最后一个元素距离顶部的高度和自身高度的一半之和,当页面的高度与滚动出去的高度之和大于最后一个的高度时,触发事件,添加新的元素,以及调整布局。

4、需要掌握知识点:

json的数据存储

window.onscroll();
document.createElement();
obj.className;
obj.appendChild(obj1);
obj.offsetWidth/offsetHeight/offsetLeft/offsetTop;
document.documentElement.clientWidth/clientHeight;
obj.style.cssText
Math.min.apply();
Math.floor();
obj.push();
document.documentElement.scrollTop
document.body.scrollTop;

注:这些都是我所不熟练的知识点。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JavaScript 利用Cookie记录用户登录信息
Dec 08 Javascript
JQuery控制div外点击隐藏而div内点击不会隐藏的方法
Jan 13 Javascript
用js判断是否为360浏览器的实现代码
Jan 15 Javascript
Javascript typeof与instanceof的区别
Oct 18 Javascript
jQuery插件FusionCharts实现的2D饼状图效果【附demo源码下载】
Mar 03 Javascript
JavaScript瀑布流布局实现代码
May 06 Javascript
解决在vue项目中webpack打包后字体不生效的问题
Sep 01 Javascript
小程序实现多选框功能
Oct 30 Javascript
原生js实现公告滚动效果
Jan 10 Javascript
Vue实现回到顶部和底部动画效果
Jul 31 Javascript
ligerUI的ligerDialog关闭刷新的方法
Sep 27 Javascript
js仿京东放大镜效果
Aug 09 Javascript
解决Layui 表单提交数据为空的问题
Aug 15 #Javascript
jQuery轮播图实例详解
Aug 15 #jQuery
JS实现获取毫秒值及转换成年月日时分秒的方法
Aug 15 #Javascript
对layui中表单元素的使用详解
Aug 15 #Javascript
layui中使用jquery控制radio选中事件的示例代码
Aug 15 #jQuery
vue生命周期实例小结
Aug 15 #Javascript
layui获取多选框中的值方法
Aug 15 #Javascript
You might like
php-cli简介(不会Shell语言一样用Shell)
2013/06/03 PHP
php读取富文本的时p标签会出现红线是怎么回事
2014/05/13 PHP
php生成4位数字验证码的实现代码
2015/11/23 PHP
PHP array_shift()用法实例分析
2019/01/07 PHP
laravel框架中路由设置,路由参数和路由命名实例分析
2019/11/23 PHP
HTML页面登录时的JS验证方法
2014/05/28 Javascript
js实现遮罩层弹出框的方法
2015/01/15 Javascript
详解javascript的变量与标识符
2016/01/04 Javascript
js基于setTimeout与setInterval实现多线程
2016/06/17 Javascript
webstorm中vue语法的支持详解
2018/05/09 Javascript
vue使用自定义icon图标的方法
2018/05/14 Javascript
swiper 自动图片无限轮播实现代码
2018/05/21 Javascript
vue + el-form 实现的多层循环表单验证
2020/11/25 Vue.js
[13:25]VP vs VICI (BO3)
2018/06/07 DOTA
[51:29]Alliance vs TNC 2019国际邀请赛小组赛 BO2 第二场 8.16
2019/08/18 DOTA
python实现自动发送报警监控邮件
2018/06/21 Python
Numpy中矩阵matrix读取一列的方法及数组和矩阵的相互转换实例
2018/07/02 Python
学习python可以干什么
2019/02/26 Python
Python实现堡垒机模式下远程命令执行操作示例
2019/05/09 Python
对Django项目中的ORM映射与模糊查询的使用详解
2019/07/18 Python
Python实现点云投影到平面显示
2020/01/18 Python
logging level级别介绍
2020/02/21 Python
python针对Oracle常见查询操作实例分析
2020/04/30 Python
ffmpeg+Python实现B站MP4格式音频与视频的合并示例代码
2020/10/21 Python
CSS3+Sprite实现僵尸行走动画特效源码
2016/01/27 HTML / CSS
HTML5高仿微信聊天、微信聊天表情|对话框|编辑器功能
2018/04/23 HTML / CSS
伊莱克斯阿根廷网上商店:Tienda Electrolux
2021/03/08 全球购物
linux面试题参考答案(7)
2014/07/24 面试题
文秘专业自荐信
2013/10/14 职场文书
2014年维修电工工作总结
2014/11/20 职场文书
心理健康教育培训研修感言
2015/11/18 职场文书
2016十一国庆节感言
2015/12/09 职场文书
八年级作文之友情
2019/11/25 职场文书
解决mysql问题:由于找不到MSVCR120.dll,无法继续执行代码
2021/06/26 MySQL
如何利用Python实现一个论文降重工具
2021/07/09 Python
Oracle使用别名的好处
2022/04/19 Oracle