js 写插件入门(图片懒加载)

阅读 (317)
原生JS写插件,本例重点在于window滚动位置的判断

图片懒加载的插件有很多,其原理基本就是将图片链接先隐藏于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

又重复造了个轮子。。。

更新于:2020-07-22 14:17:11
返回顶部