uweb/searchurl/bml/content/fullimg.js
2020-03-13 12:48:29 +08:00

497 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;
}
})();