- /*!
- * jQuery & Zepto Lazy - v1.7.6
- * http://jquery.eisbehr.de/lazy/
- *
- * Copyright 2012 - 2017, Daniel 'Eisbehr' Kern
- *
- * Dual licensed under the MIT and GPL-2.0 licenses:
- * http://www.opensource.org/licenses/mit-license.php
- * http://www.gnu.org/licenses/gpl-2.0.html
- *
- * $("img.lazy").lazy();
- */
-
- ;(function(window, undefined) {
- "use strict";
-
- // noinspection JSUnresolvedVariable
- /**
- * library instance - here and not in construct to be shorter in minimization
- * @return void
- */
- var $ = window.jQuery || window.Zepto,
-
- /**
- * unique plugin instance id counter
- * @type {number}
- */
- lazyInstanceId = 0,
-
- /**
- * helper to register window load for jQuery 3
- * @type {boolean}
- */
- windowLoaded = false;
-
- /**
- * make lazy available to jquery - and make it a bit more case-insensitive :)
- * @access public
- * @type {function}
- * @param {object} settings
- * @return void
- */
- $.fn.Lazy = $.fn.lazy = function(settings) {
- return new LazyPlugin(this, settings);
- };
-
- /**
- * helper to add plugins to lazy prototype configuration
- * @access public
- * @type {function}
- * @param {string|Array} names
- * @param {string|Array} [elements]
- * @param {function} loader
- * @return void
- */
- $.Lazy = $.lazy = function(names, elements, loader) {
- // make second parameter optional
- if( $.isFunction(elements) ) {
- loader = elements;
- elements = [];
- }
-
- // exit here if parameter is not a callable function
- if( !$.isFunction(loader) ) {
- return;
- }
-
- // make parameters an array of names to be sure
- names = $.isArray(names) ? names : [names];
- elements = $.isArray(elements) ? elements : [elements];
-
- var config = LazyPlugin.prototype.config,
- forced = config._f || (config._f = {});
-
- // add the loader plugin for every name
- for( var i = 0, l = names.length; i < l; i++ ) {
- if( config[names[i]] === undefined || $.isFunction(config[names[i]]) ) {
- config[names[i]] = loader;
- }
- }
-
- // add forced elements loader
- for( var c = 0, a = elements.length; c < a; c++ ) {
- forced[elements[c]] = names[0];
- }
- };
-
- /**
- * contains all logic and the whole element handling
- * is packed in a private function outside class to reduce memory usage, because it will not be created on every plugin instance
- * @access private
- * @type {function}
- * @param {LazyPlugin} instance
- * @param {object} config
- * @param {object|Array} items
- * @param {object} events
- * @param {string} namespace
- * @return void
- */
- function _executeLazy(instance, config, items, events, namespace) {
- /**
- * a helper to trigger the 'onFinishedAll' callback after all other events
- * @access private
- * @type {number}
- */
- var _awaitingAfterLoad = 0,
-
- /**
- * visible content width
- * @access private
- * @type {number}
- */
- _actualWidth = -1,
-
- /**
- * visible content height
- * @access private
- * @type {number}
- */
- _actualHeight = -1,
-
- /**
- * determine possibly detected high pixel density
- * @access private
- * @type {boolean}
- */
- _isRetinaDisplay = false,
-
- /**
- * dictionary entry for better minimization
- * @access private
- * @type {string}
- */
- _afterLoad = "afterLoad",
-
- /**
- * dictionary entry for better minimization
- * @access private
- * @type {string}
- */
- _load = "load",
-
- /**
- * dictionary entry for better minimization
- * @access private
- * @type {string}
- */
- _error = "error",
-
- /**
- * dictionary entry for better minimization
- * @access private
- * @type {string}
- */
- _img = "img",
-
- /**
- * dictionary entry for better minimization
- * @access private
- * @type {string}
- */
- _src = "src",
-
- /**
- * dictionary entry for better minimization
- * @access private
- * @type {string}
- */
- _srcset = "srcset",
-
- /**
- * dictionary entry for better minimization
- * @access private
- * @type {string}
- */
- _sizes = "sizes",
-
- /**
- * dictionary entry for better minimization
- * @access private
- * @type {string}
- */
- _backgroundImage = "background-image";
-
- /**
- * initialize plugin
- * bind loading to events or set delay time to load all items at once
- * @access private
- * @return void
- */
- function _initialize() {
- // detect actual device pixel ratio
- // noinspection JSUnresolvedVariable
- _isRetinaDisplay = window.devicePixelRatio > 1;
-
- // prepare all initial items
- items = _prepareItems(items);
-
- // if delay time is set load all items at once after delay time
- if( config.delay >= 0 ) {
- setTimeout(function() {
- _lazyLoadItems(true);
- }, config.delay);
- }
-
- // if no delay is set or combine usage is active bind events
- if( config.delay < 0 || config.combined ) {
- // create unique event function
- events.e = _throttle(config.throttle, function(event) {
- // reset detected window size on resize event
- if( event.type === "resize" ) {
- _actualWidth = _actualHeight = -1;
- }
-
- // execute 'lazy magic'
- _lazyLoadItems(event.all);
- });
-
- // create function to add new items to instance
- events.a = function(additionalItems) {
- additionalItems = _prepareItems(additionalItems);
- items.push.apply(items, additionalItems);
- };
-
- // create function to get all instance items left
- events.g = function() {
- // filter loaded items before return in case internal filter was not running until now
- return (items = $(items).filter(function() {
- return !$(this).data(config.loadedName);
- }));
- };
-
- // create function to force loading elements
- events.f = function(forcedItems) {
- for( var i = 0; i < forcedItems.length; i++ ) {
- // only handle item if available in current instance
- // use a compare function, because Zepto can't handle object parameter for filter
- // var item = items.filter(forcedItems[i]);
- /* jshint loopfunc: true */
- var item = items.filter(function() {
- return this === forcedItems[i];
- });
-
- if( item.length ) {
- _lazyLoadItems(false, item);
- }
- }
- };
-
- // load initial items
- _lazyLoadItems();
-
- // bind lazy load functions to scroll and resize event
- // noinspection JSUnresolvedVariable
- $(config.appendScroll).on("scroll." + namespace + " resize." + namespace, events.e);
- }
- }
-
- /**
- * prepare items before handle them
- * @access private
- * @param {Array|object|jQuery} items
- * @return {Array|object|jQuery}
- */
- function _prepareItems(items) {
- // fetch used configurations before loops
- var defaultImage = config.defaultImage,
- placeholder = config.placeholder,
- imageBase = config.imageBase,
- srcsetAttribute = config.srcsetAttribute,
- loaderAttribute = config.loaderAttribute,
- forcedTags = config._f || {};
-
- // filter items and only add those who not handled yet and got needed attributes available
- items = $(items).filter(function() {
- var element = $(this),
- tag = _getElementTagName(this);
-
- return !element.data(config.handledName) &&
- (element.attr(config.attribute) || element.attr(srcsetAttribute) || element.attr(loaderAttribute) || forcedTags[tag] !== undefined);
- })
-
- // append plugin instance to all elements
- .data("plugin_" + config.name, instance);
-
- for( var i = 0, l = items.length; i < l; i++ ) {
- var element = $(items[i]),
- tag = _getElementTagName(items[i]),
- elementImageBase = element.attr(config.imageBaseAttribute) || imageBase;
-
- // generate and update source set if an image base is set
- if( tag === _img && elementImageBase && element.attr(srcsetAttribute) ) {
- element.attr(srcsetAttribute, _getCorrectedSrcSet(element.attr(srcsetAttribute), elementImageBase));
- }
-
- // add loader to forced element types
- if( forcedTags[tag] !== undefined && !element.attr(loaderAttribute) ) {
- element.attr(loaderAttribute, forcedTags[tag]);
- }
-
- // set default image on every element without source
- if( tag === _img && defaultImage && !element.attr(_src) ) {
- element.attr(_src, defaultImage);
- }
-
- // set placeholder on every element without background image
- else if( tag !== _img && placeholder && (!element.css(_backgroundImage) || element.css(_backgroundImage) === "none") ) {
- element.css(_backgroundImage, "url('" + placeholder + "')");
- }
- }
-
- return items;
- }
-
- /**
- * the 'lazy magic' - check all items
- * @access private
- * @param {boolean} [allItems]
- * @param {object} [forced]
- * @return void
- */
- function _lazyLoadItems(allItems, forced) {
- // skip if no items where left
- if( !items.length ) {
- // destroy instance if option is enabled
- if( config.autoDestroy ) {
- // noinspection JSUnresolvedFunction
- instance.destroy();
- }
-
- return;
- }
-
- var elements = forced || items,
- loadTriggered = false,
- imageBase = config.imageBase || "",
- srcsetAttribute = config.srcsetAttribute,
- handledName = config.handledName;
-
- // loop all available items
- for( var i = 0; i < elements.length; i++ ) {
- // item is at least in loadable area
- if( allItems || forced || _isInLoadableArea(elements[i]) ) {
- var element = $(elements[i]),
- tag = _getElementTagName(elements[i]),
- attribute = element.attr(config.attribute),
- elementImageBase = element.attr(config.imageBaseAttribute) || imageBase,
- customLoader = element.attr(config.loaderAttribute);
-
- // is not already handled
- if( !element.data(handledName) &&
- // and is visible or visibility doesn't matter
- (!config.visibleOnly || element.is(":visible")) && (
- // and image source or source set attribute is available
- (attribute || element.attr(srcsetAttribute)) && (
- // and is image tag where attribute is not equal source or source set
- (tag === _img && (elementImageBase + attribute !== element.attr(_src) || element.attr(srcsetAttribute) !== element.attr(_srcset))) ||
- // or is non image tag where attribute is not equal background
- (tag !== _img && elementImageBase + attribute !== element.css(_backgroundImage))
- ) ||
- // or custom loader is available
- customLoader ))
- {
- // mark element always as handled as this point to prevent double handling
- loadTriggered = true;
- element.data(handledName, true);
-
- // load item
- _handleItem(element, tag, elementImageBase, customLoader);
- }
- }
- }
-
- // when something was loaded remove them from remaining items
- if( loadTriggered ) {
- items = $(items).filter(function() {
- return !$(this).data(handledName);
- });
- }
- }
-
- /**
- * load the given element the lazy way
- * @access private
- * @param {object} element
- * @param {string} tag
- * @param {string} imageBase
- * @param {function} [customLoader]
- * @return void
- */
- function _handleItem(element, tag, imageBase, customLoader) {
- // increment count of items waiting for after load
- ++_awaitingAfterLoad;
-
- // extended error callback for correct 'onFinishedAll' handling
- var errorCallback = function() {
- _triggerCallback("onError", element);
- _reduceAwaiting();
-
- // prevent further callback calls
- errorCallback = $.noop;
- };
-
- // trigger function before loading image
- _triggerCallback("beforeLoad", element);
-
- // fetch all double used data here for better code minimization
- var srcAttribute = config.attribute,
- srcsetAttribute = config.srcsetAttribute,
- sizesAttribute = config.sizesAttribute,
- retinaAttribute = config.retinaAttribute,
- removeAttribute = config.removeAttribute,
- loadedName = config.loadedName,
- elementRetina = element.attr(retinaAttribute);
-
- // handle custom loader
- if( customLoader ) {
- // on load callback
- var loadCallback = function() {
- // remove attribute from element
- if( removeAttribute ) {
- element.removeAttr(config.loaderAttribute);
- }
-
- // mark element as loaded
- element.data(loadedName, true);
-
- // call after load event
- _triggerCallback(_afterLoad, element);
-
- // remove item from waiting queue and possibly trigger finished event
- // it's needed to be asynchronous to run after filter was in _lazyLoadItems
- setTimeout(_reduceAwaiting, 1);
-
- // prevent further callback calls
- loadCallback = $.noop;
- };
-
- // bind error event to trigger callback and reduce waiting amount
- element.off(_error).one(_error, errorCallback)
-
- // bind after load callback to element
- .one(_load, loadCallback);
-
- // trigger custom loader and handle response
- if( !_triggerCallback(customLoader, element, function(response) {
- if( response ) {
- element.off(_load);
- loadCallback();
- }
- else {
- element.off(_error);
- errorCallback();
- }
- })) element.trigger(_error);
- }
-
- // handle images
- else {
- // create image object
- var imageObj = $(new Image());
-
- // bind error event to trigger callback and reduce waiting amount
- imageObj.one(_error, errorCallback)
-
- // bind after load callback to image
- .one(_load, function() {
- // remove element from view
- element.hide();
-
- // set image back to element
- // do it as single 'attr' calls, to be sure 'src' is set after 'srcset'
- if( tag === _img ) {
- element.attr(_sizes, imageObj.attr(_sizes))
- .attr(_srcset, imageObj.attr(_srcset))
- .attr(_src, imageObj.attr(_src));
- }
- else {
- element.css(_backgroundImage, "url('" + imageObj.attr(_src) + "')");
- }
-
- // bring it back with some effect!
- element[config.effect](config.effectTime);
-
- // remove attribute from element
- if( removeAttribute ) {
- element.removeAttr(srcAttribute + " " + srcsetAttribute + " " + retinaAttribute + " " + config.imageBaseAttribute);
-
- // only remove 'sizes' attribute, if it was a custom one
- if( sizesAttribute !== _sizes ) {
- element.removeAttr(sizesAttribute);
- }
- }
-
- // mark element as loaded
- element.data(loadedName, true);
-
- // call after load event
- _triggerCallback(_afterLoad, element);
-
- // cleanup image object
- imageObj.remove();
-
- // remove item from waiting queue and possibly trigger finished event
- _reduceAwaiting();
- });
-
- // set sources
- // do it as single 'attr' calls, to be sure 'src' is set after 'srcset'
- var imageSrc = (_isRetinaDisplay && elementRetina ? elementRetina : element.attr(srcAttribute)) || "";
- imageObj.attr(_sizes, element.attr(sizesAttribute))
- .attr(_srcset, element.attr(srcsetAttribute))
- .attr(_src, imageSrc ? imageBase + imageSrc : null);
-
- // call after load even on cached image
- imageObj.complete && imageObj.trigger(_load); // jshint ignore : line
- }
- }
-
- /**
- * check if the given element is inside the current viewport or threshold
- * @access private
- * @param {object} element
- * @return {boolean}
- */
- function _isInLoadableArea(element) {
- var elementBound = element.getBoundingClientRect(),
- direction = config.scrollDirection,
- threshold = config.threshold,
- vertical = // check if element is in loadable area from top
- ((_getActualHeight() + threshold) > elementBound.top) &&
- // check if element is even in loadable are from bottom
- (-threshold < elementBound.bottom),
- horizontal = // check if element is in loadable area from left
- ((_getActualWidth() + threshold) > elementBound.left) &&
- // check if element is even in loadable area from right
- (-threshold < elementBound.right);
-
- if( direction === "vertical" ) {
- return vertical;
- }
- else if( direction === "horizontal" ) {
- return horizontal;
- }
-
- return vertical && horizontal;
- }
-
- /**
- * receive the current viewed width of the browser
- * @access private
- * @return {number}
- */
- function _getActualWidth() {
- return _actualWidth >= 0 ? _actualWidth : (_actualWidth = $(window).width());
- }
-
- /**
- * receive the current viewed height of the browser
- * @access private
- * @return {number}
- */
- function _getActualHeight() {
- return _actualHeight >= 0 ? _actualHeight : (_actualHeight = $(window).height());
- }
-
- /**
- * get lowercase tag name of an element
- * @access private
- * @param {object} element
- * @returns {string}
- */
- function _getElementTagName(element) {
- return element.tagName.toLowerCase();
- }
-
- /**
- * prepend image base to all srcset entries
- * @access private
- * @param {string} srcset
- * @param {string} imageBase
- * @returns {string}
- */
- function _getCorrectedSrcSet(srcset, imageBase) {
- if( imageBase ) {
- // trim, remove unnecessary spaces and split entries
- var entries = srcset.split(",");
- srcset = "";
-
- for( var i = 0, l = entries.length; i < l; i++ ) {
- srcset += imageBase + entries[i].trim() + (i !== l - 1 ? "," : "");
- }
- }
-
- return srcset;
- }
-
- /**
- * helper function to throttle down event triggering
- * @access private
- * @param {number} delay
- * @param {function} callback
- * @return {function}
- */
- function _throttle(delay, callback) {
- var timeout,
- lastExecute = 0;
-
- return function(event, ignoreThrottle) {
- var elapsed = +new Date() - lastExecute;
-
- function run() {
- lastExecute = +new Date();
- // noinspection JSUnresolvedFunction
- callback.call(instance, event);
- }
-
- timeout && clearTimeout(timeout); // jshint ignore : line
-
- if( elapsed > delay || !config.enableThrottle || ignoreThrottle ) {
- run();
- }
- else {
- timeout = setTimeout(run, delay - elapsed);
- }
- };
- }
-
- /**
- * reduce count of awaiting elements to 'afterLoad' event and fire 'onFinishedAll' if reached zero
- * @access private
- * @return void
- */
- function _reduceAwaiting() {
- --_awaitingAfterLoad;
-
- // if no items were left trigger finished event
- if( !items.length && !_awaitingAfterLoad ) {
- _triggerCallback("onFinishedAll");
- }
- }
-
- /**
- * single implementation to handle callbacks, pass element and set 'this' to current instance
- * @access private
- * @param {string|function} callback
- * @param {object} [element]
- * @param {*} [args]
- * @return {boolean}
- */
- function _triggerCallback(callback, element, args) {
- if( (callback = config[callback]) ) {
- // jQuery's internal '$(arguments).slice(1)' are causing problems at least on old iPads
- // below is shorthand of 'Array.prototype.slice.call(arguments, 1)'
- callback.apply(instance, [].slice.call(arguments, 1));
- return true;
- }
-
- return false;
- }
-
- // if event driven or window is already loaded don't wait for page loading
- if( config.bind === "event" || windowLoaded ) {
- _initialize();
- }
-
- // otherwise load initial items and start lazy after page load
- else {
- // noinspection JSUnresolvedVariable
- $(window).on(_load + "." + namespace, _initialize);
- }
- }
-
- /**
- * lazy plugin class constructor
- * @constructor
- * @access private
- * @param {object} elements
- * @param {object} settings
- * @return {object|LazyPlugin}
- */
- function LazyPlugin(elements, settings) {
- /**
- * this lazy plugin instance
- * @access private
- * @type {object|LazyPlugin|LazyPlugin.prototype}
- */
- var _instance = this,
-
- /**
- * this lazy plugin instance configuration
- * @access private
- * @type {object}
- */
- _config = $.extend({}, _instance.config, settings),
-
- /**
- * instance generated event executed on container scroll or resize
- * packed in an object to be referenceable and short named because properties will not be minified
- * @access private
- * @type {object}
- */
- _events = {},
-
- /**
- * unique namespace for instance related events
- * @access private
- * @type {string}
- */
- _namespace = _config.name + "-" + (++lazyInstanceId);
-
- // noinspection JSUndefinedPropertyAssignment
- /**
- * wrapper to get or set an entry from plugin instance configuration
- * much smaller on minify as direct access
- * @access public
- * @type {function}
- * @param {string} entryName
- * @param {*} [value]
- * @return {LazyPlugin|*}
- */
- _instance.config = function(entryName, value) {
- if( value === undefined ) {
- return _config[entryName];
- }
-
- _config[entryName] = value;
- return _instance;
- };
-
- // noinspection JSUndefinedPropertyAssignment
- /**
- * add additional items to current instance
- * @access public
- * @param {Array|object|string} items
- * @return {LazyPlugin}
- */
- _instance.addItems = function(items) {
- _events.a && _events.a($.type(items) === "string" ? $(items) : items); // jshint ignore : line
- return _instance;
- };
-
- // noinspection JSUndefinedPropertyAssignment
- /**
- * get all left items of this instance
- * @access public
- * @returns {object}
- */
- _instance.getItems = function() {
- return _events.g ? _events.g() : {};
- };
-
- // noinspection JSUndefinedPropertyAssignment
- /**
- * force lazy to load all items in loadable area right now
- * by default without throttle
- * @access public
- * @type {function}
- * @param {boolean} [useThrottle]
- * @return {LazyPlugin}
- */
- _instance.update = function(useThrottle) {
- _events.e && _events.e({}, !useThrottle); // jshint ignore : line
- return _instance;
- };
-
- // noinspection JSUndefinedPropertyAssignment
- /**
- * force element(s) to load directly, ignoring the viewport
- * @access public
- * @param {Array|object|string} items
- * @return {LazyPlugin}
- */
- _instance.force = function(items) {
- _events.f && _events.f($.type(items) === "string" ? $(items) : items); // jshint ignore : line
- return _instance;
- };
-
- // noinspection JSUndefinedPropertyAssignment
- /**
- * force lazy to load all available items right now
- * this call ignores throttling
- * @access public
- * @type {function}
- * @return {LazyPlugin}
- */
- _instance.loadAll = function() {
- _events.e && _events.e({all: true}, true); // jshint ignore : line
- return _instance;
- };
-
- // noinspection JSUndefinedPropertyAssignment
- /**
- * destroy this plugin instance
- * @access public
- * @type {function}
- * @return undefined
- */
- _instance.destroy = function() {
- // unbind instance generated events
- // noinspection JSUnresolvedFunction, JSUnresolvedVariable
- $(_config.appendScroll).off("." + _namespace, _events.e);
- // noinspection JSUnresolvedVariable
- $(window).off("." + _namespace);
-
- // clear events
- _events = {};
-
- return undefined;
- };
-
- // start using lazy and return all elements to be chainable or instance for further use
- // noinspection JSUnresolvedVariable
- _executeLazy(_instance, _config, elements, _events, _namespace);
- return _config.chainable ? elements : _instance;
- }
-
- /**
- * settings and configuration data
- * @access public
- * @type {object}
- */
- LazyPlugin.prototype.config = {
- // general
- name : "lazy",
- chainable : true,
- autoDestroy : true,
- bind : "load",
- threshold : 500,
- visibleOnly : false,
- appendScroll : window,
- scrollDirection : "both",
- imageBase : null,
- defaultImage : "",
- placeholder : null,
- delay : -1,
- combined : false,
-
- // attributes
- attribute : "data-src",
- srcsetAttribute : "data-srcset",
- sizesAttribute : "data-sizes",
- retinaAttribute : "data-retina",
- loaderAttribute : "data-loader",
- imageBaseAttribute : "data-imagebase",
- removeAttribute : true,
- handledName : "handled",
- loadedName : "loaded",
-
- // effect
- effect : "show",
- effectTime : 0,
-
- // throttle
- enableThrottle : true,
- throttle : 250,
-
- // callbacks
- beforeLoad : undefined,
- afterLoad : undefined,
- onError : undefined,
- onFinishedAll : undefined
- };
-
- // register window load event globally to prevent not loading elements
- // since jQuery 3.X ready state is fully async and may be executed after 'load'
- $(window).on("load", function() {
- windowLoaded = true;
- });
- })(window);