原生JS实现照片瀑布流与懒加载

什么是瀑布流和懒加载

瀑布流是目前比较流行的一种网站页面布局,会在网页上呈现参差不齐的多栏布局,页面向下滚动,网页就会不断加载数据块并附加至当前页面尾部。
它的好处就是按需加载:
根据我们每行图片的最小高度来动态的加载图片
1.首屏加载的是减少向http请求次数
2.减少浏览器以及服务器的内存负荷

效果图是这样滴

demo2.gif

首先看一下布局:

// html部分  img标签部分发布的时候就变成了![]了,,不知道怎么改
<div id="container">
    <div class="box">
        <div class="box-img">
            ![](images/0.jpg)
        </div>
    </div>
    <div class="box">
        <div class="box-img">
            ![](images/1.jpg)
        </div>
    </div>
    <div class="box">
        <div class="box-img">
            ![](images/2.jpg)
        </div>
    </div>
    <div class="box">
        <div class="box-img">
            ![](images/3.jpg)
        </div>
    </div>
    <div class="box">
        <div class="box-img">
            ![](images/4.jpg)
        </div>
    </div>
    <div class="box">
        <div class="box-img">
            ![](images/5.jpg)
        </div>
    </div>
    <div class="box">
        <div class="box-img">
            ![](images/6.jpg)
        </div>
    </div>
    <div class="box">
        <div class="box-img">
            ![](images/7.jpg)
        </div>
    </div>
    <div class="box">
        <div class="box-img">
            ![](images/8.jpg)
        </div>
    </div>
    <div class="box">
        <div class="box-img">
            ![](images/9.jpg)
        </div>
    </div>
    <div class="box">
        <div class="box-img">
            ![](images/10.jpg)
        </div>
    </div>
    <div class="box">
        <div class="box-img">
            ![](images/11.jpg)
        </div>
    </div>
</div>

// CSS部分
    *{
        margin: 0;
        padding: 0;
    }
    #container{
        position: relative;
    }
    .box{
        float: left;
        padding: 5px;
    }
    .box-img{
        padding: 5px;
        border: 1px solid #ccc;
        box-shadow: 0 0 5px #ccc;
        border-radius: 5px;
    }
    .box-img img{
        width: 230px;
        height: auto;
    }

JS部分的逻辑

首先我们得秉承着多人协作的思想来写,因为代码不一定只是给一个人看到,所以为了避免全局污染,要用函数模块化封装的的思想。如果直接写在全局,一个项目可能有多个模块函数,命名变量名的时候并不知道其他人的命名,可能会导致bug。所以全局部分只写怎么来用这个封装好了的函数。

window.onload = function () {
    imgLocation('container', 'box');

    // 模仿数据
    var imgData = {"data": [{"src":"24.jpg"},{"src":"25.jpg"},{"src":"26.jpg"},{"src":"27.jpg"},{"src":"28.jpg"}]};
    window.onscroll = function () {
        if (checkFlag()) {
            var cparent = document.getElementById("container");
            for (var i =0; i<imgData.data.length; i++) {
                var ccontent = document.createElement('div');
                ccontent.className = 'box';
                cparent.appendChild(ccontent);
                var boxImg = document.createElement('div');
                boxImg.className = 'box-img';
                ccontent.appendChild(boxImg);
                var img = document.createElement("img");
                img.style.cssText = 'opacity: 0; transform:scale(0)';
                img.src = "images/" + imgData.data[i].src + "";
                boxImg.appendChild(img);
                (function(img){  // 自执行程序闭包
                    setTimeout(function(){
                        img.style.cssText="opacity:1;transform:scale(1)";
                    },1000); // 这里的时间自定,我是为了测试才写的1000
                })(img);
            }
            imgLocation('container', 'box');
        }
    }
};

首先我们得定位图片的位置

function imgLocation(parent, content) {
    // 将parent下面的所有content全部取出
    var cparent = document.getElementById(parent);
    var ccontent = getChildElement(cparent, content);

    // 完善图片布局
    var imgWidth = ccontent[0].offsetWidth; // 图片的宽度
    var num = Math.floor(document.documentElement.clientWidth / imgWidth); // 横排的显示个数
    cparent.style.cssText = "width:" + imgWidth * num + "px;margin: 0 auto"; // 给父级添加宽度

    // 计算图片的高度
    var boxHeightArr = [];
    for (var i=0; i<ccontent.length; i++) {
        if(i < num) {
            boxHeightArr[i] = ccontent[i].offsetHeight;
            console.log(boxHeightArr);
        } else {
            var minHeight =  getMin(boxHeightArr); //最小的高度
            var minIndex = getMinheightLocation(boxHeightArr, minHeight);
            ccontent[i].style.position = 'absolute';
            ccontent[i].style.top = minHeight + 'px';
            ccontent[i].style.left = ccontent[minIndex].offsetLeft + 'px';
            boxHeightArr[minIndex] = boxHeightArr[minIndex] + ccontent[i].offsetHeight; // 更新最小高度
        }
    }
}

这里面有三个函数,一个是得到图片最小高度的函数getMin(arr),为什么要得到最小高度?因为我们计算第一排最小高度的图片之后,第二排的图片排序是这样的,第一张排在第一排的最小高度图片下,第二张排在第一排倒数第二高度图片下,依次排完第二排,后面的图片排序都是按照这样来排。

function getMin(arr) {  // 得到图片的最小高度
    for(var i=0, ret=arr[0]; i<arr.length; i++) {
        ret = Math.min(ret, arr[i]);  // 依次将最小值赋值给ret,ret始终最小
    }
    return ret;
}

还有一个函数是得到最小高度图片索引getMinheightLocation:

function getMinheightLocation(boxHeightArr, minHeight) { // 得到图片最小高度的序列号
    for (var i in boxHeightArr) {
        if ( boxHeightArr[i] === minHeight) {
            return i;
        }
    }
}

要得到索引位置,才能将图片排在这个索引位置下。
另外一个是得到子集空间的函数,一开始就强调要用模块化的思想来做,我们要做的是函数之外,全局部分才用html里的id值,函数只通过传参来得到里面的东西。

function getChildElement(cparent, content) { // 得到子集空间
    var contentArr = [];
    var allcontent = cparent.getElementsByTagName('*'); // 获取到所有的元素
    for (var i=0; i<allcontent.length; i++ ) {
        if (allcontent[i].className === content) {
            contentArr.push(allcontent[i]);
        }
    }
    return contentArr;
}

关于懒加载

懒加载的思想是 页面的高度 + 滚动的高度 > 最后一张图片距离浏览器顶部的高度,然后就开始加载图片。

function checkFlag() {
    var cparent = document.getElementById('container');
    var ccontent = getChildElement(cparent, "box");

    // 数组最后一个元素的高度距离顶部的距离
    var lastContentHeight = ccontent[ccontent.length-1].offsetTop;
    var scrollTop = document.documentElement.scrollTop || document.body.scrollTop;
    var pageHeight = document.documentElement.clientHeight || document.body.scrollHeight;
    if(lastContentHeight < scrollTop + pageHeight){
        return true;
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容