mirror of
https://github.com/torappinfo/uweb.git
synced 2025-01-15 16:40:57 +01:00
498 lines
14 KiB
JavaScript
498 lines
14 KiB
JavaScript
/**
|
|
* Load the full-size versions of resized images based on their "src"
|
|
* attribute, or their containing link's "href" attribute. Also, make IFRAMEs
|
|
* take up the entire width of their offset parent (useful for embedded videos
|
|
* and whatnot). Same goes for the VIDEO elements.
|
|
*
|
|
* @title Load full images
|
|
*/
|
|
(function fullimg() {
|
|
/* Create a new IFRAME to get a "clean" Window object, so we can use its
|
|
* console. Sometimes sites (e.g. Twitter) override console.log and even
|
|
* the entire console object. "delete console.log" or "delete console"
|
|
* does not always work, and messing with the prototype seemed more
|
|
* brittle than this. */
|
|
var console = (function () {
|
|
var iframe = document.getElementById('xxxJanConsole');
|
|
if (!iframe) {
|
|
iframe = document.createElementNS('http://www.w3.org/1999/xhtml', 'iframe');
|
|
iframe.id = 'xxxJanConsole';
|
|
iframe.style.display = 'none';
|
|
|
|
(document.body || document.documentElement).appendChild(iframe);
|
|
}
|
|
|
|
return iframe && iframe.contentWindow && iframe.contentWindow.console || {
|
|
log: function () {}
|
|
};
|
|
})();
|
|
|
|
/* Get rid of "width=", "height=" etc. followed by numbers or number pairs
|
|
* in IMG@src query strings. */
|
|
var parameterNames = [
|
|
'width',
|
|
'Width',
|
|
|
|
'height',
|
|
'Height',
|
|
|
|
'maxwidth',
|
|
'maxWidth',
|
|
'MaxWidth',
|
|
|
|
'maxheight',
|
|
'maxHeight',
|
|
'MaxHeight',
|
|
|
|
'w',
|
|
'W',
|
|
|
|
'h',
|
|
'H',
|
|
|
|
'fit',
|
|
'Fit',
|
|
|
|
'resize',
|
|
'reSize',
|
|
'Resize',
|
|
|
|
'size',
|
|
'Size'
|
|
];
|
|
|
|
parameterNames.forEach(function (parameterName) {
|
|
var selector = 'img[src*="?' + parameterName + '="]'
|
|
+ ', img[src*="?"][src*="&' + parameterName + '="]';
|
|
|
|
/* Match query string parameters (?[…&]name=value[&…]) where the value is
|
|
* a number (e.g. "width=1200") or a pair of numbers (e.g. * "resize=640x480"). */
|
|
var parameterReplacementRegexp = new RegExp('(\\?[^#]*&)?' + parameterName + '=[1-9][0-9]+(?:(?:[xX,*:]|%2[CcAa]|%3[Aa])[1-9][0-9]+)?([^&#]*)');
|
|
|
|
[].forEach.call(document.querySelectorAll(selector), function (img) {
|
|
var newSrc = img.src
|
|
/* Remove the parameter "name=value" pair from the query string. */
|
|
.replace(parameterReplacementRegexp, '$1$2')
|
|
|
|
/* Remove trailing "&" from the query string. */
|
|
.replace(/(\?[^#]*)&(#.*)?$/, '$1$2')
|
|
|
|
/* Remove empty query strings ("?" not followed by
|
|
* anything) from the URL. */
|
|
.replace(/\?(#.*)?$/, '$1')
|
|
|
|
/* Remove empty fragment identifiers from the URL. */
|
|
.replace(/#$/, '')
|
|
;
|
|
|
|
changeSrc(img, newSrc, 'found image with parameter "' + parameterName + '" in query string');
|
|
});
|
|
});
|
|
|
|
/* Show the original image for Polopoly CMS "generated derivatives".
|
|
*
|
|
* Example:
|
|
* https://sporza.be/polopoly_fs/1.2671026!image/1706320883.jpg_gen/derivatives/landscape670/1706320883.jpg
|
|
* https://sporza.be/polopoly_fs/1.2671026!image/1706320883.jpg
|
|
*/
|
|
[].forEach.call(
|
|
document.querySelectorAll('img[src*="_gen/derivatives/"]'),
|
|
function (img) {
|
|
var matches = img.src.match(/(.*\.(jpe?g|png|gif))_gen.*\.\2(\?.*)?$/);
|
|
if (matches && matches[1]) {
|
|
changeSrc(img, matches[1], 'found image with Polopoly CMS "generated derivative" URL');
|
|
}
|
|
}
|
|
);
|
|
|
|
/* Try to load the originals for images whose source URLs look like
|
|
* thumbnail/resized versions with dimensions.
|
|
*/
|
|
[].forEach.call(
|
|
document.images,
|
|
function (img) {
|
|
var oldSrc = img.src;
|
|
/* Example:
|
|
* https://www.cycling-challenge.com/wp-content/uploads/2014/08/IMG_6197-150x150.jpg
|
|
* https://www.cycling-challenge.com/wp-content/uploads/2014/08/IMG_6197.jpg
|
|
*/
|
|
var matches = oldSrc.match(/(.*)[-_.@]\d+x\d+(\.[^\/.]+)/);
|
|
if (matches && matches[1] && matches[2]) {
|
|
var newSrc = matches[1] + matches[2];
|
|
|
|
return changeSrc(img, newSrc, 'found image whose URL looks like a thumbnail/resized version');
|
|
}
|
|
|
|
/* Example:
|
|
* https://upload.wikimedia.org/wikipedia/commons/thumb/8/83/Kowloon-Walled-City-1898.jpg/220px-Kowloon-Walled-City-1898.jpg
|
|
* https://upload.wikimedia.org/wikipedia/commons/8/83/Kowloon-Walled-City-1898.jpg
|
|
*/
|
|
matches = oldSrc.match(/(.*\/)thumb\/(.*)\/[^\/]+$/);
|
|
if (matches) {
|
|
var newSrc = matches[1] + matches[2];
|
|
|
|
return changeSrc(img, newSrc, 'found image whose URL looks like a MediaWiki thumbnail/resized version');
|
|
}
|
|
|
|
}
|
|
);
|
|
|
|
/* Try to load the originals for images whose source URLs look like
|
|
* thumbnail/resized versions with a text label.
|
|
*
|
|
* Example:
|
|
* https://www.crazyguyonabike.com/pics/docs/00/01/27/84/small/DSCF3555.JPG
|
|
* https://www.crazyguyonabike.com/pics/docs/00/01/27/84/large/DSCF3555.JPG
|
|
*/
|
|
var thumbnailPathRegexp = /(.*[/.-])(small|thumb|thumbnail|resized|preview|medium)([/.-].*)/;
|
|
|
|
var fullSizePathParts = [
|
|
'large',
|
|
'original',
|
|
'source',
|
|
'normal',
|
|
'xlarge',
|
|
];
|
|
|
|
[].forEach.call(
|
|
document.images,
|
|
function (img) {
|
|
var oldSrc = img.src;
|
|
var matches = oldSrc.match(thumbnailPathRegexp);
|
|
if (matches) {
|
|
var newSources = [];
|
|
|
|
fullSizePathParts.forEach(function (part) {
|
|
newSources.push(matches[1] + part + matches[3]);
|
|
});
|
|
|
|
changeSrc(img, newSources, 'found image whose URL looks like a thumbnail/resized version');
|
|
}
|
|
}
|
|
);
|
|
|
|
/* Change the IMG@src of linked images to their link's A@href if they look
|
|
* similar, assuming that the linked version is larger. */
|
|
[].forEach.call(
|
|
document.querySelectorAll('a img'),
|
|
function (img) {
|
|
if (!img.src) {
|
|
return;
|
|
}
|
|
|
|
var a = img.parentNode;
|
|
while (a && a.tagName && a.tagName.toLowerCase() !== 'a') {
|
|
a = a.parentNode;
|
|
}
|
|
|
|
if (!a) {
|
|
return;
|
|
}
|
|
|
|
var aHref = a.href;
|
|
|
|
if (a.hostname.match(/\.blogspot\.com$/)) {
|
|
/* Get rid of Blogspot's links to useless HTML wrappers. */
|
|
aHref = aHref.replace(/\/(s\d+)-h\/([^\/]+)$/, '/$1/$2');
|
|
}
|
|
|
|
if (aHref === img.src) {
|
|
return;
|
|
}
|
|
|
|
/* Simplify a URL for similarity calculation. */
|
|
function simplifyUrl(url) {
|
|
return ('' + url)
|
|
.replace(/\d+/g, '0')
|
|
.replace(/^https?:/, '');
|
|
}
|
|
|
|
var similarity = getSimilarity(simplifyUrl(img.src), simplifyUrl(a.href));
|
|
|
|
if (similarity > 0.66) {
|
|
changeSrc(img, aHref, 'found linked image with ' + Math.round(similarity * 100) + '% similarity');
|
|
}
|
|
}
|
|
);
|
|
|
|
/* Change all Blogspot images that have not been changed yet. */
|
|
Array.from(
|
|
document.querySelectorAll('img[src*="bp.blogspot.com/"]')
|
|
).forEach(img => {
|
|
let matches;
|
|
if ((matches = img.src.match(/^(.*\/)s(\d+)(\/[^/]+)$/)) && matches[2] < 9999) {
|
|
let newSrc = matches[1] + 's9999' + matches[3];
|
|
changeSrc(img, newSrc, 'found Blogspot image with restricted size (' + matches[2] + ')');
|
|
}
|
|
});
|
|
|
|
/* Use larger YouTube thumbnails. */
|
|
Array.from(
|
|
document.querySelectorAll('img[src*="//yt"][src*=".ggpht.com"]')
|
|
).forEach(img => {
|
|
let matches;
|
|
if ((matches = img.src.match(/^(.*\/)s(\d+)([^/]+\/photo\.[^/.]+)$/)) && matches[2] < 1024) {
|
|
let newSrc = matches[1] + 's1024' + matches[3];
|
|
changeSrc(img, newSrc, 'found YouTube avatar with restricted size (' + matches[2] + ')');
|
|
}
|
|
});
|
|
|
|
/* Get rid of all IMG@srcset attributes that have not been removed in the
|
|
* previous steps.
|
|
*/
|
|
[].forEach.call(
|
|
document.querySelectorAll('img[srcset]'),
|
|
function (img) {
|
|
console.log('Load full images: removing srcset attribute: ', img);
|
|
|
|
img.originalSrcset = img.getAttribute('srcset');
|
|
img.removeAttribute('srcset');
|
|
}
|
|
);
|
|
|
|
/* Make native VIDEO elements and video IFRAMEs take up the entire width
|
|
* of their offset parent. */
|
|
var elementsToEnlargeSelectors = [
|
|
'video',
|
|
'iframe.twitter-tweet-rendered',
|
|
'iframe[src*="embed"]',
|
|
'iframe[src*="video"]',
|
|
'iframe[src*="syndication"]',
|
|
'iframe[class*="altura"]',
|
|
'iframe[id*="altura"]',
|
|
'iframe[src*="altura"]',
|
|
'iframe[src*="//e.infogr.am/"]',
|
|
'iframe[src*="//www.kickstarter.com/projects/"]',
|
|
'iframe[src*="//media-service.vara.nl/player.php"]',
|
|
'iframe[src*="//player.vimeo.com/video/"]'
|
|
];
|
|
|
|
[].forEach.call(
|
|
document.querySelectorAll(elementsToEnlargeSelectors.join(', ')),
|
|
function (element) {
|
|
var scale = element.offsetParent.offsetWidth / element.offsetWidth;
|
|
var newWidth = Math.round(element.offsetWidth * scale);
|
|
var newHeight = Math.round(element.offsetHeight * scale);
|
|
|
|
console.log(
|
|
'Load full images: resizing element ', element,
|
|
' from ' + element.offsetWidth + 'x' + element.offsetHeight
|
|
+ ' to ' + newWidth + 'x' + newHeight
|
|
);
|
|
|
|
element.xxxJanReadableAllowStyle = true;
|
|
element.style.width = newWidth + 'px';
|
|
element.style.height = newHeight + 'px';
|
|
}
|
|
);
|
|
|
|
/* Show controls on AUDIO and VIDEO elements. */
|
|
[].forEach.call(
|
|
document.querySelectorAll('audio, video'),
|
|
function (element) {
|
|
element.controls = true;
|
|
}
|
|
);
|
|
|
|
/* Show controls on YouTube embeds. */
|
|
[].forEach.call(
|
|
document.querySelectorAll('iframe[src^="https://www.youtube.com/embed/"][src*="?"][src*="=0"]'),
|
|
function (iframe) {
|
|
var beforeAndAfterHash = iframe.src.split('#');
|
|
var beforeAndAfterQuery = beforeAndAfterHash[0].split('?');
|
|
|
|
|
|
var newPrefix = beforeAndAfterQuery[0];
|
|
|
|
var newQueryString = '';
|
|
if (beforeAndAfterQuery.length > 1) {
|
|
beforeAndAfterQuery.shift();
|
|
|
|
var newQueryParts = beforeAndAfterQuery
|
|
.join('?')
|
|
.split('&')
|
|
.filter(function (keyValuePair) {
|
|
return !keyValuePair.match(/^(controls|showinfo|rel)=0$/);
|
|
}
|
|
);
|
|
|
|
if (newQueryParts.length) {
|
|
newQueryString = '?' + newQueryParts.join('&');
|
|
}
|
|
}
|
|
|
|
var newHash = '';
|
|
if (beforeAndAfterHash.length > 1) {
|
|
beforeAndAfterHash.shift();
|
|
newHash = '#' + beforeAndAfterHash.join('#');
|
|
}
|
|
|
|
var newSrc = newPrefix + newQueryString + newHash;
|
|
|
|
if (newSrc !== iframe.src) {
|
|
iframe.src = newSrc;
|
|
}
|
|
}
|
|
);
|
|
|
|
/**
|
|
* Crudely calculate the similarity between two strings. Taken from
|
|
* https://stackoverflow.com/a/10473855. An alternative would be the
|
|
* Levenshtein distance, implemented in JavaScript here:
|
|
* https://andrew.hedges.name/experiments/levenshtein/
|
|
*/
|
|
function getSimilarity(strA, strB) {
|
|
var result = 0;
|
|
|
|
var i = Math.min(strA.length, strB.length);
|
|
|
|
if (i === 0) {
|
|
return;
|
|
}
|
|
|
|
while (--i) {
|
|
if (strA[i] === strB[i]) {
|
|
continue;
|
|
}
|
|
|
|
if (strA[i].toLowerCase() === strB[i].toLowerCase()) {
|
|
result++;
|
|
} else {
|
|
result += 4;
|
|
}
|
|
}
|
|
|
|
return 1 - (result + 4 * Math.abs(strA.length - strB.length)) / (2 * (strA.length + strB.length));
|
|
}
|
|
|
|
/**
|
|
* Change the IMG@src and fall back to the original source if the new
|
|
* source triggers an error. You can specify an array of new sources that
|
|
* will be tried in order. When all of the new sources fail, the original
|
|
* source will be used.
|
|
*/
|
|
function changeSrc(img, newSrc, reason)
|
|
{
|
|
var basename = img.src.replace(/[?#].*/, '').replace(/.*?([^\/]*)\/*$/, '$1');
|
|
|
|
console.log('[' + basename + '] Load full images: ' + reason + ': ', img);
|
|
|
|
if (img.hasNewSource) {
|
|
console.log('[' + basename + '] Image already has a new source: ', img);
|
|
return;
|
|
}
|
|
|
|
var newSources = Array.isArray(newSrc)
|
|
? newSrc
|
|
: [ newSrc ];
|
|
|
|
while ((newSrc = newSources.shift())) {
|
|
if (newSrc && img.src !== newSrc) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!newSrc) {
|
|
return;
|
|
}
|
|
|
|
console.log('[' + basename + '] → Old img.src: ' + img.src);
|
|
console.log('[' + basename + '] → Try img.src: ' + newSrc);
|
|
|
|
/* Save the original source. */
|
|
if (!img.originalSrc) {
|
|
img.originalSrc = img.src;
|
|
}
|
|
|
|
if (!img.originalNaturalWidth) {
|
|
img.originalNaturalWidth = img.naturalWidth;
|
|
}
|
|
|
|
if (!img.originalNaturalHeight) {
|
|
img.originalNaturalHeight = img.naturalHeight;
|
|
}
|
|
|
|
/* Save and disable the srcset on the IMG element. */
|
|
if (img.hasAttribute('srcset')) {
|
|
img.originalSrcset = img.getAttribute('srcset');
|
|
img.removeAttribute('srcset');
|
|
}
|
|
|
|
/* Save and disable the srcset in the container PICTURE element's SOURCE descendants. */
|
|
if (img.parentNode.tagName.toLowerCase() === 'picture') {
|
|
[].forEach.call(
|
|
img.parentNode.querySelectorAll('source[srcset]'),
|
|
function (source) {
|
|
source.originalSrcset = source.getAttribute('srcset');
|
|
source.removeAttribute('srcset');
|
|
}
|
|
);
|
|
}
|
|
|
|
/* When the new source has failed to load, load the next one from the
|
|
* list of possible new sources. If there are no more left, revert to
|
|
* the original source. */
|
|
var errorHandler;
|
|
|
|
if (newSources.length) {
|
|
console.log('[' + basename + '] Setting errorHandler to loadNextNewSrc for ', img, '; newSources: "' + newSources.join('", "') + '"; reason:', reason);
|
|
errorHandler = function loadNextNewSrc() {
|
|
img.removeEventListener('error', loadNextNewSrc);
|
|
changeSrc(img, newSources, reason);
|
|
};
|
|
} else {
|
|
console.log('[' + basename + '] Setting errorHandler to restoreOriginalSrc for ', img, '; originalSrc: "' + img.originalSrc + '"; reason:', reason);
|
|
errorHandler = function restoreOriginalSrc() {
|
|
console.log('[' + basename + '] Load full images: error while loading new source for image: ', img);
|
|
console.log('[' + basename + '] → Unable to load new img.src: ' + newSrc);
|
|
console.log('[' + basename + '] → Resetting to original img.src: ' + img.originalSrc);
|
|
|
|
img.removeEventListener('error', restoreOriginalSrc);
|
|
|
|
/* Restore the original source. */
|
|
img.src = img.originalSrc;
|
|
|
|
/* Re-enable the original srcset on the IMG element. */
|
|
if (img.originalSrcset) {
|
|
img.setAttribute('srcset', img.originalSrcset);
|
|
delete img.originalSrcset;
|
|
}
|
|
|
|
/* Re-enable the original srcset in the container PICTURE element's SOURCE descendants. */
|
|
if (img.parentNode.tagName.toLowerCase() === 'picture') {
|
|
[].forEach.call(
|
|
img.parentNode.querySelectorAll('source'),
|
|
function (source) {
|
|
if (source.originalSrcset) {
|
|
source.setAttribute('srcset', source.originalSrcset);
|
|
delete source.originalSrcset;
|
|
}
|
|
}
|
|
);
|
|
}
|
|
|
|
};
|
|
}
|
|
|
|
img.addEventListener('error', errorHandler);
|
|
|
|
/* When the new source image is smaller than the original image,
|
|
* treat that as an error, too. */
|
|
img.addEventListener('load', function () {
|
|
if (img.naturalWidth * img.naturalHeight < img.originalNaturalWidth * img.originalNaturalHeight) {
|
|
console.log('[' + basename + '] Load full images: new image (', img.naturalWidth, 'x', img.naturalHeight, ') is smaller than old image (', img.originalNaturalWidth, 'x', img.originalNaturalHeight, '): ', img);
|
|
|
|
return errorHandler();
|
|
}
|
|
|
|
if (img.src !== img.originalSrc) {
|
|
console.log('[' + basename + '] → Success: ' + img.src);
|
|
img.hasNewSource = true;
|
|
}
|
|
});
|
|
|
|
/* Finally, actually try to load the image. */
|
|
img.src = newSrc;
|
|
}
|
|
})();
|