235 lines
6.6 KiB
JavaScript
235 lines
6.6 KiB
JavaScript
/*! npm.im/object-fit-images 3.2.4 */
|
|
var objectFitImages = (function () {
|
|
'use strict';
|
|
|
|
var OFI = 'bfred-it:object-fit-images';
|
|
var propRegex = /(object-fit|object-position)\s*:\s*([-.\w\s%]+)/g;
|
|
var testImg = typeof Image === 'undefined' ? {style: {'object-position': 1}} : new Image();
|
|
var supportsObjectFit = 'object-fit' in testImg.style;
|
|
var supportsObjectPosition = 'object-position' in testImg.style;
|
|
var supportsOFI = 'background-size' in testImg.style;
|
|
var supportsCurrentSrc = typeof testImg.currentSrc === 'string';
|
|
var nativeGetAttribute = testImg.getAttribute;
|
|
var nativeSetAttribute = testImg.setAttribute;
|
|
var autoModeEnabled = false;
|
|
|
|
function createPlaceholder(w, h) {
|
|
return ("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='" + w + "' height='" + h + "'%3E%3C/svg%3E");
|
|
}
|
|
|
|
function polyfillCurrentSrc(el) {
|
|
if (el.srcset && !supportsCurrentSrc && window.picturefill) {
|
|
var pf = window.picturefill._;
|
|
// parse srcset with picturefill where currentSrc isn't available
|
|
if (!el[pf.ns] || !el[pf.ns].evaled) {
|
|
// force synchronous srcset parsing
|
|
pf.fillImg(el, {reselect: true});
|
|
}
|
|
|
|
if (!el[pf.ns].curSrc) {
|
|
// force picturefill to parse srcset
|
|
el[pf.ns].supported = false;
|
|
pf.fillImg(el, {reselect: true});
|
|
}
|
|
|
|
// retrieve parsed currentSrc, if any
|
|
el.currentSrc = el[pf.ns].curSrc || el.src;
|
|
}
|
|
}
|
|
|
|
function getStyle(el) {
|
|
var style = getComputedStyle(el).fontFamily;
|
|
var parsed;
|
|
var props = {};
|
|
while ((parsed = propRegex.exec(style)) !== null) {
|
|
props[parsed[1]] = parsed[2];
|
|
}
|
|
return props;
|
|
}
|
|
|
|
function setPlaceholder(img, width, height) {
|
|
// Default: fill width, no height
|
|
var placeholder = createPlaceholder(width || 1, height || 0);
|
|
|
|
// Only set placeholder if it's different
|
|
if (nativeGetAttribute.call(img, 'src') !== placeholder) {
|
|
nativeSetAttribute.call(img, 'src', placeholder);
|
|
}
|
|
}
|
|
|
|
function onImageReady(img, callback) {
|
|
// naturalWidth is only available when the image headers are loaded,
|
|
// this loop will poll it every 100ms.
|
|
if (img.naturalWidth) {
|
|
callback(img);
|
|
} else {
|
|
setTimeout(onImageReady, 100, img, callback);
|
|
}
|
|
}
|
|
|
|
function fixOne(el) {
|
|
var style = getStyle(el);
|
|
var ofi = el[OFI];
|
|
style['object-fit'] = style['object-fit'] || 'fill'; // default value
|
|
|
|
// Avoid running where unnecessary, unless OFI had already done its deed
|
|
if (!ofi.img) {
|
|
// fill is the default behavior so no action is necessary
|
|
if (style['object-fit'] === 'fill') {
|
|
return;
|
|
}
|
|
|
|
// Where object-fit is supported and object-position isn't (Safari < 10)
|
|
if (
|
|
!ofi.skipTest && // unless user wants to apply regardless of browser support
|
|
supportsObjectFit && // if browser already supports object-fit
|
|
!style['object-position'] // unless object-position is used
|
|
) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
// keep a clone in memory while resetting the original to a blank
|
|
if (!ofi.img) {
|
|
ofi.img = new Image(el.width, el.height);
|
|
ofi.img.srcset = nativeGetAttribute.call(el, "data-ofi-srcset") || el.srcset;
|
|
ofi.img.src = nativeGetAttribute.call(el, "data-ofi-src") || el.src;
|
|
|
|
// preserve for any future cloneNode calls
|
|
// https://github.com/bfred-it/object-fit-images/issues/53
|
|
nativeSetAttribute.call(el, "data-ofi-src", el.src);
|
|
if (el.srcset) {
|
|
nativeSetAttribute.call(el, "data-ofi-srcset", el.srcset);
|
|
}
|
|
|
|
setPlaceholder(el, el.naturalWidth || el.width, el.naturalHeight || el.height);
|
|
|
|
// remove srcset because it overrides src
|
|
if (el.srcset) {
|
|
el.srcset = '';
|
|
}
|
|
try {
|
|
keepSrcUsable(el);
|
|
} catch (err) {
|
|
if (window.console) {
|
|
console.warn('https://bit.ly/ofi-old-browser');
|
|
}
|
|
}
|
|
}
|
|
|
|
polyfillCurrentSrc(ofi.img);
|
|
|
|
el.style.backgroundImage = "url(\"" + ((ofi.img.currentSrc || ofi.img.src).replace(/"/g, '\\"')) + "\")";
|
|
el.style.backgroundPosition = style['object-position'] || 'center';
|
|
el.style.backgroundRepeat = 'no-repeat';
|
|
el.style.backgroundOrigin = 'content-box';
|
|
|
|
if (/scale-down/.test(style['object-fit'])) {
|
|
onImageReady(ofi.img, function () {
|
|
if (ofi.img.naturalWidth > el.width || ofi.img.naturalHeight > el.height) {
|
|
el.style.backgroundSize = 'contain';
|
|
} else {
|
|
el.style.backgroundSize = 'auto';
|
|
}
|
|
});
|
|
} else {
|
|
el.style.backgroundSize = style['object-fit'].replace('none', 'auto').replace('fill', '100% 100%');
|
|
}
|
|
|
|
onImageReady(ofi.img, function (img) {
|
|
setPlaceholder(el, img.naturalWidth, img.naturalHeight);
|
|
});
|
|
}
|
|
|
|
function keepSrcUsable(el) {
|
|
var descriptors = {
|
|
get: function get(prop) {
|
|
return el[OFI].img[prop ? prop : 'src'];
|
|
},
|
|
set: function set(value, prop) {
|
|
el[OFI].img[prop ? prop : 'src'] = value;
|
|
nativeSetAttribute.call(el, ("data-ofi-" + prop), value); // preserve for any future cloneNode
|
|
fixOne(el);
|
|
return value;
|
|
}
|
|
};
|
|
Object.defineProperty(el, 'src', descriptors);
|
|
Object.defineProperty(el, 'currentSrc', {
|
|
get: function () { return descriptors.get('currentSrc'); }
|
|
});
|
|
Object.defineProperty(el, 'srcset', {
|
|
get: function () { return descriptors.get('srcset'); },
|
|
set: function (ss) { return descriptors.set(ss, 'srcset'); }
|
|
});
|
|
}
|
|
|
|
function hijackAttributes() {
|
|
function getOfiImageMaybe(el, name) {
|
|
return el[OFI] && el[OFI].img && (name === 'src' || name === 'srcset') ? el[OFI].img : el;
|
|
}
|
|
if (!supportsObjectPosition) {
|
|
HTMLImageElement.prototype.getAttribute = function (name) {
|
|
return nativeGetAttribute.call(getOfiImageMaybe(this, name), name);
|
|
};
|
|
|
|
HTMLImageElement.prototype.setAttribute = function (name, value) {
|
|
return nativeSetAttribute.call(getOfiImageMaybe(this, name), name, String(value));
|
|
};
|
|
}
|
|
}
|
|
|
|
function fix(imgs, opts) {
|
|
var startAutoMode = !autoModeEnabled && !imgs;
|
|
opts = opts || {};
|
|
imgs = imgs || 'img';
|
|
|
|
if ((supportsObjectPosition && !opts.skipTest) || !supportsOFI) {
|
|
return false;
|
|
}
|
|
|
|
// use imgs as a selector or just select all images
|
|
if (imgs === 'img') {
|
|
imgs = document.getElementsByTagName('img');
|
|
} else if (typeof imgs === 'string') {
|
|
imgs = document.querySelectorAll(imgs);
|
|
} else if (!('length' in imgs)) {
|
|
imgs = [imgs];
|
|
}
|
|
|
|
// apply fix to all
|
|
for (var i = 0; i < imgs.length; i++) {
|
|
imgs[i][OFI] = imgs[i][OFI] || {
|
|
skipTest: opts.skipTest
|
|
};
|
|
fixOne(imgs[i]);
|
|
}
|
|
|
|
if (startAutoMode) {
|
|
document.body.addEventListener('load', function (e) {
|
|
if (e.target.tagName === 'IMG') {
|
|
fix(e.target, {
|
|
skipTest: opts.skipTest
|
|
});
|
|
}
|
|
}, true);
|
|
autoModeEnabled = true;
|
|
imgs = 'img'; // reset to a generic selector for watchMQ
|
|
}
|
|
|
|
// if requested, watch media queries for object-fit change
|
|
if (opts.watchMQ) {
|
|
window.addEventListener('resize', fix.bind(null, imgs, {
|
|
skipTest: opts.skipTest
|
|
}));
|
|
}
|
|
}
|
|
|
|
fix.supportsObjectFit = supportsObjectFit;
|
|
fix.supportsObjectPosition = supportsObjectPosition;
|
|
|
|
hijackAttributes();
|
|
|
|
return fix;
|
|
|
|
}());
|