图片一直是网络资源占用大户,对于一个前端有几百张图片的网站来讲,若是首屏即加载全部图片(不管这些图片有没有被用户看到),那无疑是既浪费网络资源,又伤害用户体验的事。所以,图片懒加载,是提升前端性能的刚需所在。javascript
目前,淘宝网、知乎等大流量网站都已经使用了图片滚动懒加载的方案——仅当图片滚入视窗,被用户看到的时候,才会去真正加载。前端
图片滚动懒加载的原理很是简单:基于<img>
标签,在初次加载时,不把图片url放在src属性中,而是自定义一个属性,例如data-src。而后检测"scroll","resize"等窗体事件,判断图片是否进入了可视范围。若是进入,则将data-src的字段替换到src,此时浏览器会自动去加载对应图片资源。vue
首先是不添加src的img标签,新增data-src用于放置图片url:java
<img class="lazyImg" data-src="http://xxx.xxx.com"/>复制代码
而后,咱们须要新增一个数组队列,来储存全部未加载的img节点:git
var lazyImg=[].slice.call(document.querySelectorAll(".lazyImg"));复制代码
为了方便,这里直接用querySelectorAll来获取全部img节点。注意由于NodeList是只读数组,所以须要将其转化为数组,方便以后的增删。在真实环境中,还需给每一个成员添加其最近的可滚动祖先节点的引用,即el.parentNode。github
最关键的部分来了,如何判断图片是否进入了可视区域,以及实现加载呢?数组
// 参数传入lazyImg数组
function loadImage(images){
let scrollParent,src,el;
for(let i = 0;i < images.length;i++){
scrollParent=images[i].scrollParent; //img所属的最近的可滚动祖先节点
el=images[i].el;
//offset为预留的预加载距离
if(checkInView(el,scrollParent,this.options.offset)){
src=el.dataset.src;
el.setAttribute("src",src);
images.splice(i--,1); //将该img元素移除
}
}
}复制代码
上面提到的scrollParent是带有scroll特性的祖先节点,具体实现:可以使用getComputedStyle检查父节点是否设置了overflow(overflow-x,overflow-y)为"auto"或"scroll",不断循环直到找到知足条件的祖先节点。浏览器
下面封装了判断是否在可视区域的函数:网络
const checkInView=(el,scrollParent,offset)=>{
let scrollTop,scrollLeft,clientH,clientW;
if(scrollParent === window) {
scrollTop=document.documentElement.scrollTop||document.body.scrollTop;
scrollLeft=document.documentElement.scrollLeft||document.body.scrollLeft;
clientH=document.documentElement.clientHeight||document.body.clientHeight;
clientW=document.documentElement.clientWidth||document.body.clientWidth;
}
else {
scrollTop = scrollParent.scrollTop;
scrollLeft=scrollParent.scrollLeft;
clientH = scrollParent.clientHeight;
clientW=scrollParent.clientWidth;
}
while(el!=scrollParent && el!=null){
let borderWidth=parseInt(getStyle(el,"border-width"));
offsetTop+=el.offsetTop+borderWidth;
offsetLeft+=el.offsetLeft+borderWidth;
el=el.offsetParent;
}
//在这里判断是否滚入可视区域。offset为预留的预加载距离
if(scrollTop+clientH>el.offsetTop-offset && scrollLeft+clientW>el.offsetLeft-offset){
return true;
}
else return false;
}复制代码
最后再让各自的scrollParent监听"scroll","resize"等事件便可:前端性能
function initListener(el){
let scrollParent=getScrollParent(el);
if(this.scrollParent.indexOf(scrollParent)<0){
position = getStyle(scrollParent, "position"); //若为window则返回null
if (position==="" || position === "static") scrollParent.style.position = "relative"; //确保能检测到正确的offsetTop和offsetLeft
this.scrollParent.push(scrollParent); //数组用于保存已经监听的可滚动祖先节点
this.eventsList.forEach((event)=>{
scrollParent.addEventListener(event,this.loadImage.bind(this));
})
}
}复制代码
以上即是实现图片懒加载的关键代码。
若是有动态添加的img标签,该怎么办呢?其实很简单,只须要将新增的img元素push进这个lazyImg数组队列中,而后调用InitListener便可。
利用以上原理,我实现了一个基于vue2.x的图片懒加载的插件,完整代码可参考这里vue-lazyload-images。