瀑布流和懶加載實例【轉載】

<meta charset="utf-8">

什么是瀑布流和懶加載

瀑布流是目前比較流行的一種網站頁面布局,會在網頁上呈現參差不齊的多欄布局,頁面向下滾動,網頁就會不斷加載數據塊并附加至當前頁面尾部。
它的好處就是按需加載:
根據我們每行圖片的最小高度來動態的加載圖片
1.首屏加載的是減少向http請求次數
2.減少瀏覽器以及服務器的內存負荷

效果圖是這樣滴


瀑布流懶加載基礎演示2.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;
    }
}

本文轉載自簡書作者:八寶君
原文鏈接:http://www.lxweimin.com/p/18e0234365ff

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,156評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,401評論 3 415
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,069評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,873評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,635評論 6 408
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,128評論 1 323
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,203評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,365評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,881評論 1 334
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,733評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,935評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,475評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,172評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,582評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,821評論 1 282
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,595評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,908評論 2 372