Vanilla javascript Lazy-load
Rus
Eng
Vanilla javascript Lazy-load

Отличное решение lazy-load скрипта предложил автор сайта web3r.ru

Javascript

/**
 * Анонимная самовызывающаяся функция-обертка
 * @param {document} d - получает документ
 */
!function(d) {

  "use strict";

  /**
   * Полифилл для Object.assign()
   */
  Object.assign||Object.defineProperty(Object,"assign",{enumerable:!1,configurable:!0,writable:!0,value:function(e,r){"use strict";if(null==e)throw new TypeError("Cannot convert first argument to object");for(var t=Object(e),n=1;n= 0 && coords.left >= 0 && coords.top) <= (window.innerHeight || d.documentElement.clientHeight);
  };

  /**
   * Функция загрузки элемента
   * @param {Event|Object} e - событие получаемое при click/hover или элемент, если используется "view"
   * @param {Function|false} [callback] - функция, выполняемая после загрузки элемента
   */
  lazy.get = function(e, callback) {
    var item = e.target || e;
    var event = e.type || false;
    if (event) item.removeEventListener(event, lazy.get); // удаление обработчика события
    if (!item.src) { // проверка элемента на наличие атрибута src
      console.error("lazy-load не поддерживается для этого элемента (" + item.tagName + ")");
      return;
    }
    if (!item.hasAttribute("data-js-loaded")) {
      var src = item.getAttribute("data-js-source");
      if (src) { // если атрибут data-js-source присутствует
        item.src = src;
        item.setAttribute("data-js-loaded", "");
        item.removeAttribute("data-js-source");
        item.onerror = function() { // обработка ошибок загрузки
          console.error("загрузка источника не удалась (элемент: " + item.tagName + ", путь: " + item.src + ")");
        };
        setTimeout(function() { // удаление обработчика ошибок
          item.onerror = null;
        }, 3000);
      } else console.error("отсутствует оригинальный источник (элемент: " + item.tagName + ", путь: " + item.src + ")");
      if (callback) callback(item); // вызов callback, если необходимо
    }
  };




  /**
   * Функция инициализации lazy-load
   * @param {Object} settings - объект с предварительными настройками
   */
  lazy.init = function(settings) {

    // настройки по умолчанию
    settings = Object.assign({}, {
      items:    "[data-js-source]",
      on:       "click",
      callback: false,
    }, settings);

    // поиск всех lazy-элементов
    var els = d.querySelectorAll(settings.items);


    // обработчики событий
    if (settings.on === "view") { // для события с условным наимнованием "view"
      var onviewLoad = function() {
        Array.prototype.slice.call(els, 0).forEach(function(el) {
          if (lazy.isVisible(el)) lazy.get(el, settings.callback); // проверка на видимость во viewport, lazy.get() принимает элемент
        });
      };

      // обработка событий для "view"
      window.addEventListener("scroll", onviewLoad);
      window.addEventListener("resize", onviewLoad);
      if (screen.msOrientation || (screen.orientation || screen.mozOrientation || { type: false }).type) window.addEventListener("orientationchange", onviewLoad);
    } else { // обработка событий "hover" и "click"
      var eventName = (settings.on === "hover") ? "mouseover" : settings.on;
      Array.prototype.slice.call(els, 0).forEach(function(el) {
        el.addEventListener(eventName, function(e) {
          lazy.get(e, settings.callback);
        }); // lazy.get() принимает событие
      });
    }

  };

  window.lazy = lazy;

}(document);

CSS оформление

/* все загрузившиеся элементы */
[data-js-loaded] { }

/* все не загрузившиеся фреймы */
iframe[data-js-source] { }

/* все не загрузившиеся картинки */
img[data-js-source] { }

/* все загрузившиеся фреймы */
iframe[data-js-loaded] { }

/* все загрузившиеся картинки */
img[data-js-loaded] { }

Использование

// вызов функции
document.addEventListener("DOMContentLoaded", function() {

  // lazy-load
  lazy.init({
    on: "view", // при попадании во viewport
    callback: function(el) { // функция при загрузке 
      el.classList.add("loading");
      el.onload = function() {
        el.classList.remove("loading");
        el.onload = null;
      }
    }
  });

  // отключение кеширования оригинальных картинок
  var images = document.querySelectorAll("[data-source]");
  images.forEach(function(el) {
    var source = el.getAttribute("data-source");
    el.setAttribute("data-source", source + "?" + Math.random());
  })
});

html

<img src="placeholder.gif" alt="" data-js-source="path_to_img" />

Источник

Комментарии

Комментариев еще нет, Вы можете стать первым кто его оставит

Оставьте комментарий

На сайте используется система премодерирования комментариев, поэтому ваше сообщение будет опубликовано лишь после одобрения модератором

Вы отвечаете на комментарий пользователя

Отправить

ОБРАТНАЯ СВЯЗЬ

Напишите мне

Вы разрабатываете новый сервис, вносите доработки в существующий и хотите лучше чем у конкурентов? Вы обратились по адресу. Предлагаю вам комплексную разработку сайтов студийного уровня. У меня вы можете заказать дизайн, верстку, програмированние, разработку нетрадиционного функционала, реализацию связи между CMS, CRM и Data Analitics, а так же все остальное касаемое сайтов, кроме продвижения.

Обращайтесь, я всегда проконсультирую по всем вопросам и помогу подобрать наиболее эффективное решение для Вашего бизнеса. Я занимаюсь созданием сайтов в Новосибирске и в других регионах России, также работаю со странами СНГ. Вы останетесь довольны нашим сотрудничеством

Во время отправки произошла ошибка, пожалуйста попробуйте еще раз через некоторое время
Сообщение отправлено успешно

Телефоны

+7(993) 007-18-96

Email

info@tichiy.ru

Адрес

Россия, г. Москва

Отправляя форму Вы автоматически подтверждаете, что ознакомились и принимаете Политику конфиденциальности сайта

Написать мне
Отправить
Отправляя форму Вы автоматически подтверждаете, что ознакомились и принимаете Политику конфиденциальности сайта
Отправка успешна!
Thank you for your feedback. I will answer you within the next working hours
Отправка не удалась
Во время отправки запроса произошла ошибка. Пожалуйста, подождите и попробуйте снова через некоторое время или свяжитесь со мной