懒加载

懒加载 lazyload

原理:先将img标签中的src属性设为同一个图片(空白图片),将真正图片的地址存储在img标签的自定义属性中(比如data-src)。当js监听到该图片元素进入可视窗口时,即将自定义属性中的地址存到src属性中,达到懒加载的目的。

实现

1.判断元素是否出现在窗口可视范围内。

屏幕可视大小

原生方法:标准浏览器及IE9+ || 标准浏览器及低版本IE标准模式 || 低版本混杂模式

window.innerHeight   document.documentElement.clientHeight   document.body.clientHeight

jQuery方法: $(window).height()

滚动条滚动的距离

原生方法:IE9+及标准浏览器 || 兼容ie低版本的标准模式 || 兼容混杂模式

window.pagYoffset   document.documentElement.scrollTop   document.body.scrollTop

jQuery方法:$(document).scrollTop();

获取元素的尺寸:

左边jquery方法,右边原生方法

  $(o).width() = o.style.width; 

  $(o).innerWidth() = o.style.width+o.style.padding;

  $(o).outerWidth() = o.offsetWidth = o.style.width+o.style.padding+o.style.border;

  $(o).outerWidth(true) = o.style.width+o.style.padding+o.style.border+o.style.margin;

  注意要使用原生的style.xxx方法获取属性这个元素必须已经有内嵌的样式<div style="...."></div>
  如果原先是通过外部或内部样式表定义css样式必须使用o.currentStyle[xxx] || document.defaultView.getComputedStyle(0)[xxx]来获取样式值

获取元素的位置信息:

1)返回元素相对于文档document顶部、左边的距离;

jQuery:$(o).offset().top元素距离文档顶的距离,$(o).offset().left元素距离文档左边缘的距离

原生:getoffsetTop();

顺便提一下返回元素相对于第一个以定位的父元素的偏移距离,注意与上面偏移距的区别;

jQuery:position()返回一个对象,$(o).position().left = style.left,$(o).position().top = style.top;

实现代码:

(function($) {
    $.fn.scrollLoading = function(options) {
        var defaults = {
            // 在html标签中存放的属性名称;
            attr: "data-url",
            // 父元素默认为window
            container: window,
            callback: $.noop
        };
        // 不管有没有传入参数,先合并再说;
        var params = $.extend({}, defaults, options || {});
        // 把父元素转为jquery对象;
        var container = $(params.container);
        // 新建一个数组,然后调用each方法,用于存储每个dom对象相关的数据;
        params.cache = [];
        $(this).each(function() {
            // 取出jquery对象中每个dom对象的节点类型,取出每个dom对象上设置的图片路径
            var node = this.nodeName.toLowerCase(), url = $(this).attr(params["attr"]);
            //重组,把每个dom对象上的属性存为一个对象;
            var data = {
                obj: $(this),
                tag: node,
                url: url
            };
            // 把这个对象加到一个数组中;
            params.cache.push(data);
        });

        var callback = function(call) {
            if ($.isFunction(params.callback)) {
                params.callback.call(call);
            }
        };
        
        //每次触发滚动事件时,对每个dom元素与container元素进行位置判断,如果满足条件,就把路径赋予这个dom元素!
        var loading = function() {
            // 获取父元素的高度
            var contHeight = container.outerHeight();
            var contWidth = container.outerWidth();

            // 获取父元素相对于文档页顶部的距离,这边要注意了,分为以下两种情况;
            if (container.get(0) === window) {
                // 第一种情况父元素为window,获取浏览器滚动条已滚动的距离;$(window)没有offset()方法;
                var contop = $(window).scrollTop();
                var conleft = $(window).scrollLeft();
            } else {
                // 第二种情况父元素为非window元素,获取它的滚动条滚动的距离;
                var contop = container.offset().top;
                var conleft = container.offset().left;
            }

            $.each(params.cache, function(i, data) {
                var o = data.obj, tag = data.tag, url = data.url, post, posb, posl, posr;
                if (o) {
                    //对象顶部与文档顶部之间的距离,如果它小于父元素底部与文档顶部的距离,则说明垂直方向上已经进入可视区域了;
                    post = o.offset().top - (contop + contHeight);
                    //对象底部与文档顶部之间的距离,如果它大于父元素顶部与文档顶部的距离,则说明垂直方向上已经进入可视区域了;
                    posb = o.offset().top + o.height() - contop;

                    // 水平方向上同理;
                    posl = o.offset().left - (conleft + contWidth);
                    posr = o.offset().left + o.width() - conleft;

                    // 只有当这个对象是可视的,并且这四个条件都满足时,才能给这个对象赋予图片路径;
                    if ( o.is(':visible') && (post < 0 && posb > 0) && (posl < 0 && posr > 0) ) {
                        if (url) {
                            //在浏览器窗口内
                            if (tag === "img") {
                                //设置图片src
                                callback(o.attr("src", url));
                            } else {
                                // 设置除img之外元素的背景url
                                callback(o.css("background-image", "url("+ url +")"));
                            }
                        } else {
                            // 无地址,直接触发回调
                            callback(o);
                        }
                        // 给对象设置完图片路径之后,把params.cache中的对象给清除掉;对象再进入可视区,就不再进行重复设置了;
                        data.obj = null;
                    }
                }
            });
        };
        //加载完毕即执行
        loading();
        //滚动执行
        container.bind("scroll", loading);
    };
})(jQuery);
var lazyload = (function($){
			var clock;

			function init(){
				$(window).on("scroll", function(){
					if (clock) {
						clearTimeout(clock);
					}
					clock = setTimeout(function(){
						checkShow();
					}, 200);
				})
				checkShow();
			}

			function checkShow(){
				$("img").each(function(){
					var $cur = $(this);
					if($cur.attr("isLoaded")){
						return;
					}
					if(shouldShow($cur)){
						showImg($cur);
					}
				});
			}

			function shouldShow($cur){
				var scrollH = $(window).scrollTop(),
					winH = $(window).height(),
					top = $cur.offset().top;

					if(top < scrollH + winH){
						return true;
					}else{
						return false;
					}
			}

			function showImg($cur){
				$cur.attr("src",$cur.attr("data-src"));
				$cur.attr("isLoaded",true);
			}

			return {init:init};
		})(jQuery);

		
		lazyload.init();

Table of Contents