js 写插件入门(图片懒加载)
图片懒加载的插件有很多,其原理基本就是将图片链接先隐藏于data属性中,等滚动到特点位置后再将图片链接替换到 img src中
在不添加任何优化和修饰的情况下,核心代码逻辑非常简单,只做两件事即可
HTML:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, minimum-scale=1, user-scalable=no" />
<meta name="referrer" content="no-referrer">
<title>图片懒加载demo | PHPer | Web前端 | 编程爱好者 | 郑凯文的个人网站</title>
<style>
img {
max-width: 100%;
}
</style>
</head>
<body style="padding: 20px;margin: 0;">
<div style="width: 640px;word-wrap: break-word;margin: 0 auto;">
<p><img src="img/loading-yellow.gif" data-original="https://pic4.zhimg.com/v2-098ae36dfa6d61ae3918092eefa699d1_r.jpg?source=1940ef5c"/></p>
<p><img src="img/loading-yellow.gif" data-original="https://pic4.zhimg.com/v2-098ae36dfa6d61ae3918092eefa699d1_r.jpg?source=1940ef5c"/></p>
<p><img src="img/loading-yellow.gif" data-original="https://pic4.zhimg.com/v2-098ae36dfa6d61ae3918092eefa699d1_r.jpg?source=1940ef5c"/></p>
<p><img src="img/loading-yellow.gif" data-original="https://pic4.zhimg.com/v2-098ae36dfa6d61ae3918092eefa699d1_r.jpg?source=1940ef5c"/></p>
<p><img src="img/loading-yellow.gif" data-original="https://pic4.zhimg.com/v2-098ae36dfa6d61ae3918092eefa699d1_r.jpg?source=1940ef5c"/></p>
<p><img src="img/loading-yellow.gif" data-original="https://pic4.zhimg.com/v2-098ae36dfa6d61ae3918092eefa699d1_r.jpg?source=1940ef5c"/></p>
<p><img src="img/loading-yellow.gif" data-original="https://pic4.zhimg.com/v2-098ae36dfa6d61ae3918092eefa699d1_r.jpg?source=1940ef5c"/></p>
<p><img src="img/loading-yellow.gif" data-original="https://pic4.zhimg.com/v2-098ae36dfa6d61ae3918092eefa699d1_r.jpg?source=1940ef5c"/></p>
<p><img src="img/loading-yellow.gif" data-original="https://pic4.zhimg.com/v2-098ae36dfa6d61ae3918092eefa699d1_r.jpg?source=1940ef5c"/></p>
<p><img src="img/loading-yellow.gif" data-original="https://pic4.zhimg.com/v2-098ae36dfa6d61ae3918092eefa699d1_r.jpg?source=1940ef5c"/></p>
<p><img src="img/loading-yellow.gif" data-original="https://pic4.zhimg.com/v2-098ae36dfa6d61ae3918092eefa699d1_r.jpg?source=1940ef5c"/></p>
</div>
</body>
</html>
其实核心JS:
就这些就可以实现图片懒加载了
/*
* 监听函数
* obj 监听对象
* type 监听事件
* handle 执行函数
*/
function addEvent(obj,type,handle){
try{
obj.addEventListener(type,handle,false);
}catch(e){
try{
obj.attachEvent('on'+type,handle);
}
catch(e){
obj['on' + type]=handle;//早期浏览器
}
}
}
var render = function() {
// 所有需要懒加载的元素
var nodes = document.querySelectorAll('[data-original]');
// 打印所有图片,可以看到各元素对象有一些属性,可以自己调试看一下
console.log(nodes)
// 获取滚动距离
var scrollHeight = window.pageYOffset + document.documentElement.clientHeight
// 遍历需要懒加载的元素
for (var key in nodes) {
if (nodes.hasOwnProperty(key)) {
var item = nodes[key]
// 滚动到图片位置时,替换src
// 这里的item.offsetTop就是图片元素距离顶部的距离
if(scrollHeight >= item.offsetTop) {
// 从data属性中获取original图片链接
item.setAttribute('src', item.dataset.original)
}
}
}
}
// 监听两个事件,第一,当页面滚动到某个图片元素位置时,加载图片
addEvent(document ,'scroll', render)
// 当页面及资源完全加载后
addEvent(window ,'load', render)
到这里就可以完成“图片懒加载”的功能了。
那在实际使用情况下,每次滚动都会遍历并执行懒加载的操作,显然这样是很不合理的,下面进行一些优化
待优化的点:
1.已经加载好的图片,不再重新再被document.querySelectorAll('[data-original]')查询到,也就是后面不再重新设置src
我的一开始的思路是,直接把 data-original 删除即可,用removeAttribute,即:item.removeAttribute('data-original')
2.第一步做完发现,由于监听了滚动事件,每次滚动,函数可能被执行n多次,可以再优化一下,加入函数节流
3.如果待加载的图片地址404了,即“图裂了”,我想优化一下,换成一张好看点的404占位图
优化思路是:判断图片链接响应状态,如果不是200,则填充本地404图片地址
4.现在的代码是在刚滚动到图片位置时,就把src改了,比如我们作为开发人员,我可以通过控制台调试,查看实际代码是否工作,但是作为老板,他刚进来看到就是已经替换好的图片,老板都没都根本看不清是不是懒加载,他估计会想:“这程序员到底有没有做这个功能”,遇到这种情况怎么办呢?我们需要在滚动到图片位置的条件上加点偏移量,即需要让老板看到滚动到图片时,图片还是Loading的情况,让他看清情况后,我再把图片地址改成真实图片,甚至可以加点延时和渐显的效果。
5.将某些设置做成可配置项
一步步来,我先完善下代码,把代码插件化,然后把优化点1完成:
优化功能点1
;(function(undefined) {
"use strict"
var _global
// 合并对象(深拷贝)
function extend(defaults, n) {
var n = n || {};
for (var x in defaults) {
// 对于使用时,没有设置的参数;用默认参数代替
if (typeof n[x] === 'undefined') {
n[x] = defaults[x];
}
}
return n
}
/*
* 监听函数
* obj 监听对象
* type 监听事件
* handle 执行函数
*/
function addEvent(obj,type,handle){
try{
obj.addEventListener(type,handle,false);
}catch(e){
try{
obj.attachEvent('on'+type,handle);
}
catch(e){
obj['on' + type]=handle;//早期浏览器
}
}
}
// 插件函数
function Klayz(opt){
// 未使用new关键词时
if (!(this instanceof Klayz)){
// 等于再new一个
return new Klayz(opt)
}
this._init(opt)
}
Klayz.prototype = {
constructor: this,
_init: function (opt) {
// 监听两个事件,第一,当页面滚动到某个图片元素位置时,加载图片
addEvent(document ,'scroll', this._render.bind(this))
// 当页面及资源完全加载后
addEvent(window ,'load', this._render.bind(this))
},
_render: function () {
var nodes = document.querySelectorAll('[data-original]')
console.log(nodes)
var scrollHeight = window.pageYOffset + document.documentElement.clientHeight
console.log(scrollHeight)
for (var key in nodes) {
if (nodes.hasOwnProperty(key)) {
var item = nodes[key]
if(scrollHeight >= item.offsetTop) {
// 设置真实图片链接
item.setAttribute('src', item.dataset.original)
// 移除data-original后,下次不再遍历该原素
item.removeAttribute('data-original')
}
}
}
}
}
// 最后将插件对象暴露给全局对象
_global = (function(){ return this || (0, eval)('this'); }());
if (typeof module !== "undefined" && module.exports) {
module.exports = Klayz;
} else if (typeof define === "function" && define.amd) {
define(function(){return Klayz;});
} else {
!('Klayz' in _global) && (_global.Klayz = Klayz);
}
}());
优化功能点2
/** 节流
* 应用场景:用户进行高频事件触发(滚动),但在限制在n秒内只会执行一次。
* 实现原理: 每次触发时间的时候,判断当前是否存在等待执行的延时函数
* @params fun 传入的防抖函数(callback) delay 等待时间
* */
function throttle(fun, delay){
delay = delay || 1000
let flag = true
var f = fun
return function (...args) {
if (!flag) return;
flag = false
setTimeout(function () {
f.apply(this, args)
flag = true
}, delay)
}
}
优化功能点3
// 验证图片是否404状态
function validateImage(url)
{
var xmlHttp
if (window.ActiveXObject)
{
xmlHttp = new ActiveXObject("Microsoft.XMLHTTP");
}
else if (window.XMLHttpRequest)
{
xmlHttp = new XMLHttpRequest();
}
xmlHttp.open('Get', url, false);
xmlHttp.send();
if(xmlHttp.status==404){
return false;
}else{
return true;
}
}
优化功能点4
主要是淡入的效果
// 设置元素透明度,透明度值按IE规则计,即0~100
function SetOpacity(el, v){
el.filters ? el.style.filter = 'alpha(opacity=' + v + ')' : el.style.opacity = v / 100;
}
/*
* 淡入效果(含淡入到指定透明度)
* el 需要淡入的元素
* speed 淡入速度,正整数(可选)
* opacity 淡入到指定的透明度,0~100(可选)
*/
function fadeIn(el, speed, opacity){
speed = speed || 20;
opacity = opacity || 100;
//显示元素,并将元素值为0透明度(不可见)
el.style.display = 'block';
SetOpacity(el, 0);
//初始化透明度变化值为0
var val = 0;
//循环将透明值以5递增,即淡入效果
var timer = setInterval(function() {
if (val <= opacity) {
SetOpacity(el, val)
val += 5
} else {
clearInterval(timer)
}
}, speed);
}
优化功能点5(最终版)
也就是最终的完成版,至少是v1可进行测试的版本了
;(function(undefined) {
"use strict"
var _global
// 合并对象(深拷贝)
function extend(defaults, n) {
var n = n || {};
for (var x in defaults) {
// 对于使用时,没有设置的参数;用默认参数代替
if (typeof n[x] === 'undefined') {
n[x] = defaults[x];
}
}
return n
}
/*
* 监听函数
* obj 监听对象
* type 监听事件
* handle 执行函数
*/
function addEvent(obj,type,handle){
try{
obj.addEventListener(type,handle,false);
}catch(e){
try{
obj.attachEvent('on'+type,handle);
}
catch(e){
obj['on' + type]=handle;//早期浏览器
}
}
}
// 验证图片是否404状态
function validateImage(url)
{
var xmlHttp
if (window.ActiveXObject)
{
xmlHttp = new ActiveXObject("Microsoft.XMLHTTP");
}
else if (window.XMLHttpRequest)
{
xmlHttp = new XMLHttpRequest();
}
xmlHttp.open('Get', url, false);
xmlHttp.send();
if(xmlHttp.status==404){
return false;
}else{
return true;
}
}
// 设置元素透明度,透明度值按IE规则计,即0~100
function SetOpacity(el, v){
el.filters ? el.style.filter = 'alpha(opacity=' + v + ')' : el.style.opacity = v / 100;
}
/*
* 淡入效果(含淡入到指定透明度)
* el 需要淡入的元素
* speed 淡入速度,正整数(可选)
* opacity 淡入到指定的透明度,0~100(可选)
*/
function fadeIn(el, speed, opacity){
speed = speed || 20;
opacity = opacity || 100;
//显示元素,并将元素值为0透明度(不可见)
el.style.display = 'block';
SetOpacity(el, 0);
//初始化透明度变化值为0
var val = 0;
//循环将透明值以5递增,即淡入效果
var timer = setInterval(function() {
if (val <= opacity) {
SetOpacity(el, val)
val += 5
} else {
clearInterval(timer)
}
}, speed);
}
/** 节流
* 应用场景:用户进行高频事件触发(滚动),但在限制在n秒内只会执行一次。
* 实现原理: 每次触发时间的时候,判断当前是否存在等待执行的延时函数
* @params fun 传入的防抖函数(callback) delay 等待时间
* */
function throttle(fn, delay){
delay = delay || 1000
var flag = true
var f = fn
return function () {
if (!flag) return;
flag = false
setTimeout(function () {
f.apply(this)
flag = true
}, delay)
}
}
// 插件函数
function Klayz(opt){
// 未使用new关键词时
if (!(this instanceof Klayz)){
// 等于再new一个
return new Klayz(opt)
}
this._init(opt)
}
Klayz.prototype = {
constructor: this,
_init: function (opt) {
var defaults = {
defaultImg: '', // 默认的占位图
errorImg: '', // 图裂的占位图
offset: 0, // 向上的偏移量,单位px
}
// 没有设置的参数,使用默认值
this.opt = extend(defaults, opt) // 得到的this.opt, 在方法中调用;
// 监听两个事件,第一,当页面滚动到某个图片元素位置时,加载图片
addEvent(document ,'scroll', throttle(this._render.bind(this)))
// 当页面及资源完全加载后
addEvent(window ,'load', throttle(this._render.bind(this)))
},
_render: function () {
var nodes = document.querySelectorAll('[data-original]')
console.log(nodes)
var scrollHeight = window.pageYOffset + document.documentElement.clientHeight
console.log(scrollHeight)
for (var key in nodes) {
if (nodes.hasOwnProperty(key)) {
var item = nodes[key]
// 如果src是空的,未设置的,则填充默认占位图
if(!item.getAttribute('src')) {
item.setAttribute('src', this.opt.defaultImg)
}
// 计算时加入偏移量
if((scrollHeight - this.opt.offset) >= item.offsetTop) {
// 设置真实图片链接
var imgUrl = item.dataset.original
// 图片响应状态
var imgStatus = validateImage(imgUrl)
if(imgStatus) {
// 正常情况
item.setAttribute('src', item.dataset.original)
} else {
// 404图片设置404占位
item.setAttribute('src', this.opt.errorImg)
}
// 移除data-original后,下次不再遍历该原素
item.removeAttribute('data-original')
// 图片淡入
fadeIn(item)
}
}
}
}
}
// 最后将插件对象暴露给全局对象
_global = (function(){ return this || (0, eval)('this'); }());
if (typeof module !== "undefined" && module.exports) {
module.exports = Klayz;
} else if (typeof define === "function" && define.amd) {
define(function(){return Klayz;});
} else {
!('Klayz' in _global) && (_global.Klayz = Klayz);
}
}());
上面代码看着很多,其中有很长一段图片的base64代码,因为不想额外引用固定的图片地址,所以就转成base64引用了,如果去掉base64的图片内容,代码就清晰很多了
调用插件:
Klayz({
defaultImg: './img/loading-yellow.gif',
errorImg: './img/404.png',
offset: 100
})
demo:http://returnc.com/demo/layz.html
又重复造了个轮子。。。