Image Host Helper

Directly upload local / rehost remote images or galleries to whatever supported image host by dropping/pasting them to target field

Від 16.06.2020. Дивіться остання версія.

// ==UserScript==
// @name         Image Host Helper
// @namespace    https://greasyfork.org/users/321857-anakunda
// @version      1.072
// @description  Directly upload local / rehost remote images or galleries to whatever supported image host by dropping/pasting them to target field
// @icon         data:image/png;base64,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
// @author       Anakunda
// @match        https://passthepopcorn.me/*
// @match        https://redacted.ch/*
// @match        https://orpheus.network/*
// @match        https://broadcasthe.net/*
// @match        https://notwhat.cd/*
// @match        https://dicmusic.club/*
// @match        https://*/torrents.php?id=*
// @match        https://*/artist.php?id=*
// @match        https://*/artist.php?action=edit&artistid=*
// @match        https://*/reportsv2.php?action=report&id=*
// @match        https://*/forums.php?action=new*
// @match        https://*/forums.php?*action=viewthread*
// @match        https://*/requests.php?action=view*
// @match        https://*/collages.php?id=*
// @match        https://*/collages.php?action=edit&collageid=*
// @match        https://*/collages.php?action=comments&collageid=*
// @match        https://*/collages.php?action=new
// @match        http*://tracker.czech-server.com/*
// @connect      *
// @grant        GM_xmlhttpRequest
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_deleteValue
// @require      https://greasyfork.org/scripts/404642-js-xhr/code/js-xhr.js
// @require      https://greasyfork.org/scripts/404516-progressbars/code/progressBars.js
// @require      https://greasyfork.org/scripts/401726-imagehostuploader/code/imageHostUploader.js
// ==/UserScript==

'use strict';

if (document.getElementById('upload assistant') != null) return; // don't clash with Upload Assistant

[
  'ptpimg_api_key',
  'malzo_uid', 'malzo_password',
  'imgbb_uid', 'imgbb_password', 'imgbb_api_key',
  'catbox_userhash',
  'imgbox_uid', 'imgbox_password',
  'jerking_uid', 'jerking_password',
  //'picload_uid', 'picload_password',
  '24a_uid', '24a_password',
  'postimg_uid', 'postimg_password',
  'imagevenue_uid', 'imagevenue_password',
  'radikal_uid', 'radikal_password',
  'chevereto_custom_hosts',
].forEach(propName => { if (!GM_getValue(propName)) GM_setValue(propName, '') });
['upload_hosts', 'rehost_hosts'].forEach(propName => { if (!GM_getValue(propName)) GM_setValue(propName, [
  'PTPimg', 'NWCD', 'PixHost', /*'Malzo', */'ImgBB', 'ImgBox', 'Catbox', 'Jerking',
  'Abload', 'FastPic', 'Radikal', 'PicLoad', '24A', 'PostImage', 'Imgur', 'ImageVenue',
].join(', ')) });

var cheveretoCustomHosts = GM_getValue('chevereto_custom_hosts');
if (cheveretoCustomHosts) try {
  JSON.parse(cheveretoCustomHosts).forEach(function(siteDef) {
	if (!siteDef.host_name || !siteDef.alias)
	  console.warn('Incomplete Chevereto custom site definition:', siteDef);
	else imageHostHandlers[siteDef.alias.replace(nonWordStripper, '').toLowerCase()] =
	  new Chevereto(siteDef.host_name, siteDef.alias, siteDef.types, siteDef.limit_anon,
		siteDef.limit_authorized, siteDef.config_prefix);
  });
} catch (e) { console.warn(e) }
console.log('Image host handlers:', imageHostHandlers);

buildUploadChain(GM_getValue(document.domain) || GM_getValue('upload_hosts'));
buildRehostChain(GM_getValue(document.domain) || GM_getValue('rehost_hosts'));

document.querySelectorAll('input[id*="image"], input[name*="image"], input[id*="avatar"], input[name*="avatar"]')
  .forEach(setInputHandlers);
if (document.domain == 'tracker.czech-server.com') document.querySelectorAll('input[name="urlobr"]').forEach(setInputHandlers);
Array.from(document.getElementsByTagName('textarea')).forEach(setTextAreahandlers);
if (document.URL.includes('/torrents.php?id=')) {
  let a = document.querySelector('span.additional_add_artists > a');
  if (a != null) a.addEventListener('click', function() {
	document.querySelectorAll('input[name="image[]"]').forEach(setInputHandlers);
  });
}
if (document.URL.includes('/reportsv2.php')) {
  setReportHandlers();
  var reportTypeSelect = document.querySelector('select#type');
  if (reportTypeSelect != null) reportTypeSelect.addEventListener('change', setReportHandlers);

  function setReportHandlers(evt) {
	setTimeout(function() {
	  document.querySelectorAll('input[id*="image"]').forEach(setInputHandlers);
	  document.querySelectorAll('textarea').forEach(setTextAreahandlers);
	}, 2000);
  }
}

// HJ Toolkit helper
if (document.domain.endsWith('passthepopcorn.me')) setTimeout(function() {
  if (document.querySelector('div.HJ-toolkit-badge') != null) {
	let hjtkTimer = setInterval(function() {
	  document.querySelectorAll('textarea[id^="HJMA"], textarea.form-control[name="screen"], textarea.form-control[name="comp"]')
		.forEach(setTextAreahandlers);
	}, 1000);
  }
}, 1000);

function uploadImages(files, progressHandler) {
  if (!Array.isArray(ulHostChain) || ulHostChain.length <= 0) return Promise.reject('No hosts where to upload');
  if (typeof files != 'object') return Promise.reject('Invalid argument');
  if (!Array.isArray(files)) files = Array.from(files);
  //if (files.length > 1) files.push(files.shift());
  files = files.filter(file => file instanceof File && file.size > 0 && file.type && file.type.startsWith('image/'));
  if (files.length <= 0) return Promise.reject('Nothing to upload');
  return Promise.all(files.map(file => new Promise(function(resolve, reject) {
	var reader = new FileReader();
	reader.onload = function() {
	  if (reader.result.length != file.size) console.warn('FileReader: binary string read length mismatch (',
		reader.result.length, '≠',  file.size, ')');
	  resolve({ name: file.name, type: file.type, size: file.size, data: reader.result });
	};
	reader.onerror = reader.ontimeout = function() { reject('FileReader error (' + file.name + ')') };
	reader.readAsBinaryString(file);
  }))).then(function(images) {
	return uploadInternal();

	function uploadInternal(hostIndex = 0) {
	  return hostIndex >= 0 && hostIndex < ulHostChain.length ? (function() {
		if (Array.isArray(ulHostChain[hostIndex].types)
			&& !files.every(file => ulHostChain[hostIndex].types.some(type => file.type == 'image/'.concat(type.toLowerCase()))))
		  return Promise.reject('MIME type not accepted for all files');
		if (ulHostChain[hostIndex].sizeLimit && files.some(file => file.size > ulHostChain[hostIndex].sizeLimit * 2**20))
		  return Promise.reject('one or more images exceed size limit (' + ulHostChain[hostIndex].sizeLimit + 'MB)');
// 		if (ulHostChain[hostIndex].batchLimit && files.length > ulHostChain[hostIndex].batchLimit)
// 		  return Promise.reject('batch limit exceeded (' + ulHostChain[hostIndex].batchLimit + ')');
		if (typeof progressHandler == 'function') {
		  progressHandler(hostIndex, null);
		  var _progressHandler = (param = null, index = undefined) => progressHandler(hostIndex, param, index);
		}
		return ulHostChain[hostIndex].upload(ulHostChain[hostIndex].upload.acceptFiles ? files : images, _progressHandler);
	  })().catch(function(reason) {
		console.warn('Upload to', ulHostChain[hostIndex].alias, 'failed:', reason);
		var msg = 'Upload to '.concat(ulHostChain[hostIndex].alias, ' failed (', reason, ')');
		if (++hostIndex < ulHostChain.length) {
		  logFail(msg.concat(', falling back to ', ulHostChain[hostIndex].alias));
		  return uploadInternal(hostIndex);
		}
		logFail(msg);
		return Promise.reject('Upload failed to all hosts');
	  }) : Promise.reject('host index out of bounds ('.concat(hostIndex, ')'));
	}
  });
}

function rehostImages(urls, progressHandler) {
  if (!Array.isArray(urls)) return Promise.reject('Invalid argument');
  if (urls.length <= 0) return Promise.reject('Nothing to rehost');
  urls = urls.filter(url => urlParser.test(url));
  return Array.isArray(rhHostChain) && rhHostChain.length > 0 ? rehostInternal() : Promise.resolve(urls);

  function rehostInternal(hostIndex = 0) {
	if (hostIndex < 0 || hostIndex >= rhHostChain.length)
	  return Promise.reject('Host index out of bounds ('.concat(hostIndex, ')'));
// 	if (rhHostChain[hostIndex].batchLimit && urls.length > rhHostChain[hostIndex].batchLimit)
// 	  return Promise.reject('batch limit exceeded (' + rhHostChain[hostIndex].batchLimit + ')');
	if (typeof progressHandler == 'function') {
	  progressHandler(hostIndex, false);
	  var _progressHandler = (param = true) => progressHandler(hostIndex, param);
	}
	return rhHostChain[hostIndex].rehost(urls, _progressHandler).catch(function(reason) {
	  console.warn('Rehost to', rhHostChain[hostIndex].alias, 'failed:', reason);
	  var msg = 'Rehost to '.concat(rhHostChain[hostIndex].alias, ' failed (', reason, ')');
	  if (++hostIndex < rhHostChain.length) {
		logFail(msg.concat(', falling back to ', rhHostChain[hostIndex].alias));
		return rehostInternal(hostIndex);
	  }
	  logFail(msg);
	  return Promise.reject('Rehost failed to all hosts');
	});
  }
}

function coverPreview(input, imgUrl, size) {
  var child = document.getElementById('cover-preview');
  if (child == null) {
	return;
	if (!(input instanceof HTMLElement) || input.parentNode.previousElementSibling == null) return;
	var elem = document.createElement('div');
	elem.style = 'padding-top: 10px; float: right; width: 90%;';
	child = document.createElement('img');
	child.id = 'cover-preview';
	elem.append(child);
	var div = document.createElement('div');
	div.id = 'cover-size';
	elem.append(div);
	input.parentNode.previousElementSibling.append(document.createElement('br'));
	input.parentNode.previousElementSibling.append(elem);
  }
  if ((div = div || document.getElementById('cover-size')) == null) return;
  if (urlParser.test(imgUrl)) {
	child.onload = function(evt) {
	  this.onload = null;
	  if (!this.naturalWidth || !this.naturalHeight) return; // invalid image
	  (size > 0 ? Promise.resolve(size) : getRemoteFileSize(imgUrl)).then(function(size) {
		div.textContent = this.naturalWidth + '×' + this.naturalHeight + ' (' + formattedSize(size) + ')';
	  }.bind(this)).catch(reason => { div.textContent = this.naturalWidth + '×' + this.naturalHeight });
	};
	child.onerror = function(evt) {
	  this.onerror = null;
	  div.textContent = this.src = '';
	  console.warn('Image source cannot be updated:', evt, imgUrl);
	};
	child.src = imgUrl;
  } else div.textContent = child.src = '';
}

function imageDataHandler(evt, data) {
  if (!data) return true;
  if (data.files.length > 0) {
	if (!data.files[0].type || !data.files[0].type.toLowerCase().startsWith('image/')) return true;
	evt.target.disabled = true;
	if (evt.target.hTimer) {
	  clearTimeout(evt.target.hTimer);
	  delete evt.target.hTimer;
	}
	evt.target.style.color = 'white';
	evt.target.style.backgroundColor = 'darkred';
	let size = data.files[0].size, lastPct, lastUpdate, current;
	uploadImages([data.files[0]], function(worker, param = null) {
	  if (param && typeof param == 'object') {
		if (param.readyState > 1 || current != undefined && worker !== current) return;
		if (Date.now() - 100 < lastUpdate) return;
		let pct = Math.floor(Math.min(param.done * 100 / param.total, 100));
		if (pct <= lastPct) return;
		lastPct = pct;
		evt.target.value = 'Uploading... [' + pct + '%]';
		lastUpdate = Date.now();
	  } else if (typeof param != 'number') {
		lastPct = lastUpdate = undefined;
		current = worker;
		evt.target.value = 'Uploading...';
	  }
	}).then(function(urls) {
	  evt.target.value = urls[0];
	  evt.target.style.backgroundColor = '#008000';
	  evt.target.hTimer = setTimeout(function() {
		evt.target.style.backgroundColor = null;
		evt.target.style.color = null;
		delete evt.target.hTimer;
	  }, 10000);
	  coverPreview(evt.target, urls[0], size);
	}).catch(function(error) {
	  imageClear(evt);
	  evt.target.style.backgroundColor = null;
	  evt.target.style.color = null;
	  Promise.resolve(error).then(msg => { alert(msg) });
	}).then(() => { evt.target.disabled = false });
	return false;
  } else if (data.items.length > 0) {
	let links = data.getData('text/uri-list');
	if (links) links = links.split(/\r?\n/); else {
	  links = data.getData('text/x-moz-url');
	  if (links) links = links.split(/\r?\n/).filter((item, ndx) => ndx % 2 == 0);
	  	else if (links = data.getData('text/plain')) links = links.split(/\r?\n/);
	}
	if (Array.isArray(links) && links.length > 0) imageUrlResolver(links[0]).then(verifyImageUrl).then(function(imageUrl) {
	  evt.target.value = imageUrl;
	  coverPreview(evt.target, imageUrl);
	  evt.target.disabled = true;
	  rehostImages([imageUrl])
		.then(urls => { if (urls.length > 0 && urls[0] != evt.target.value) evt.target.value = urls[0] })
		.catch(reason => { Promise.resolve(reason).then(msg => { alert(msg + ' (not rehosted)') }) })
		.then(() => { evt.target.disabled = false });
	}).catch(function(e) {
	  console.error(e);
	  alert(e);
	});
	return false;
  }
  return true;
}

function descDropHandler(evt) {
  if (evt.dataTransfer == null || evt.shiftKey) return true;
  if (evt.dataTransfer.files.length > 0) {
	let images = Array.from(evt.dataTransfer.files).filter(file => file.type && file.type.startsWith('image/'));
	if (images.length <= 0) return true;
	evt.target.disabled = true;
	if (!['notwhat.cd'].some(hostname => document.domain == hostname))
	  var progressBar = new ULProgressBar(evt.target, images.map(image => image.size));
	uploadImages(images, ULProgressBar.prototype.update.bind(progressBar)).then(urlHandler)
	  .catch(reason => { Promise.resolve(reason).then(msg => { alert(msg) }) })
	  .then(function() {
		if (progressBar) progressBar.cleanUp();
		evt.target.disabled = false;
	  });
	evt.stopPropagation();
	return false;
  } else if (evt.dataTransfer.items.length > 0) {
	let content = evt.dataTransfer.getData('text/uri-list');
	if (content) content = content.split(/(?:\r?\n)+/); else {
	  content = evt.dataTransfer.getData('text/x-moz-url');
	  if (content) content = content.split(/(?:\r?\n)+/).filter((item, ndx) => ndx % 2 == 0);
	};
	if (!Array.isArray(content) || content.length <= 0) return true;
	Promise.all(content.map(imageUrlResolver)).then(resolvedUrls => resolvedUrls.flatten()).then(function(resolvedUrls) {
	  evt.target.disabled = true;
	  if (resolvedUrls.length > 1 && !['notwhat.cd'].some(hostname => document.domain == hostname))
		progressBar = new RHProgressBar(evt.target, resolvedUrls.length);
	  rehostImages(resolvedUrls, RHProgressBar.prototype.update.bind(progressBar)).catch(function(reason) {
		Promise.resolve(reason).then(msg => { alert(msg + ' (not rehosted)') });
		if (progressBar) progressBar.update(-1, false);
		return verifyImageUrls(resolvedUrls);
	  }).then(function(imgUrls) {
		urlHandler(imgUrls, shouldCouple(content));
		if (progressBar) progressBar.cleanUp();
		evt.target.disabled = false;
	  });
	});
	evt.stopPropagation();
	return false;
  }
  return true;

  function urlHandler(urls, coupled = false) {
	urls.forEach(function(url, ndx) {
	  if (url.length <= 0 || !urlParser.test(url)) return;
	  var phpBB = evt.target.nophpbb ? url : '[img]'.concat(url, '[/img]');
	  if (evt.target.value.trimRight().length <= 0) evt.target.value = phpBB; else if (evt.ctrlKey) {
		evt.target.value = evt.target.value.slice(0, evt.rangeOffset) + phpBB + evt.target.value.slice(evt.rangeOffset);
	  } else if (!evt.target.nophpbb && /\[img\]\[\/img\]/i.test(evt.target.value)) {
		evt.target.value = RegExp.leftContext + phpBB + RegExp.rightContext;
	  } else evt.target.value = evt.target.value.trimRight()
		.concat(ndx <= 0 || coupled && ndx % 2 == 0 ? '\n\n' : '\n', phpBB);
	});
  }
}

function descPasteHandler(evt) {
  if (evt.clipboardData == null) return true;
  if (evt.clipboardData.files.length > 0) {
	let images = Array.from(evt.clipboardData.files).filter(file => file.type && file.type.startsWith('image/'));
	if (images.length <= 0) return true;
	evt.target.disabled = true;
	if (!['notwhat.cd'].some(hostname => document.domain == hostname))
	  var progressBar = new ULProgressBar(evt.target, images.map(image => image.size));
	uploadImages(images, ULProgressBar.prototype.update.bind(progressBar)).then(insert,
		reason => { Promise.resolve(reason).then(msg => { alert(msg) }) })
	  .then(function() {
		if (progressBar) progressBar.cleanUp();
		evt.target.disabled = false;
	  });
	evt.stopPropagation();
	return false;
  } else if (evt.clipboardData.items.length > 0) {
	let urls = evt.clipboardData.getData('text/plain').split(/(?:\r?\n)+/);
	if (urls.length <= 0 || !urls.every(RegExp.prototype.test.bind(urlParser))) return true;
// 	Promise.all(urls.map(imageUrlResolver)).then(resolvedUrls => resolvedUrls.flatten()).then(function(resolvedUrls) {
// 	  evt.target.disabled = true;
// 	  if (resolvedUrls.length > 1 && !['notwhat.cd'].some(hostname => document.domain == hostname))
// 	  	progressBar = new RHProgressBar(evt.target, resolvedUrls.length);
// 	  rehostImages(resolvedUrls, RHProgressBar.prototype.update.bind(progressBar)).catch(function(reason) {
// 		Promise.resolve(reason).then(msg => { alert(msg + ' (not rehosted)') });
// 		if (progressBar) progressBar.update(-1, false);
// 		return verifyImageUrls(resolvedUrls);
// 	  }).then(function(imgUrls) {
// 		insert(imgUrls, shouldCouple(urls));
// 		if (progressBar) progressBar.cleanUp();
// 		evt.target.disabled = false;
// 	  });
// 	});
// 	evt.stopPropagation();
// 	return false;
  }
  return true;

  function insert(imgUrls, coupled = false) {
	var selStart = evt.target.selectionStart, phpBB = imgUrls.map(function(imgUrl, ndx) {
	  return (ndx <= 0 ? '' : coupled && ndx % 2 == 0 ? '\n\n' : '\n').concat('[img]', imgUrl, '[/img]');
	}).join('');
	evt.target.value = evt.target.value.slice(0, evt.target.selectionStart)
	  .concat(phpBB, evt.target.value.slice(evt.target.selectionEnd));
	evt.target.setSelectionRange(selStart + phpBB.length, selStart + phpBB.length);
  }
}

function shouldCouple(urlList) {
  return urlList.length == 1 && [
	'caps-a-holic.com/',
	'screenshotcomparison.com/',
	//'dvdbeaver.com/',
  ].some(pattern => urlList[0].includes(pattern));
}

function imageUrlResolver(url) {
  return urlResolver(url).then(url => verifyImageUrl(url).catch(function(reason) {
	try { url = new URL(url) } catch(e) { return Promise.reject(e) }
	const notFound = Promise.reject('No title image for this URL');
	if (url.hostname.endsWith('pinterest.com'))
	  return pinterestResolver(url);
	else if (url.hostname.endsWith('free-picload.com')) {
	  if (url.pathname.startsWith('/album/')) return imageHostHandlers['picload'].galleryResolver(url);
	} else if (url.hostname.endsWith('bandcamp.com')) return globalFetch(url).then(function(response) {
	  var ref = response.document.querySelector('div#tralbumArt > a.popupImage');
	  if (ref != null) ref = ref.href;
	  else if ((ref = response.document.querySelector('meta[property="og:image"]')) != null) ref = ref.content;
	  return ref ? ref.replace(/_\d+(?=\.\w+$)/, '_0') : notFound;
	}); else switch (url.hostname) {
	  // general image hostings
	  case 'www.imgur.com': case 'imgur.com':
		if (url.pathname.startsWith('/a/')) return globalFetch(url, { responseType: 'text' }).then(function(response) {
		  if (/^\s*(?:image)\s*:\s*(\{.+\}),\s*$/m.test(response.responseText)) try {
		  	return JSON.parse(RegExp.$1).album_images.images.map(image => 'https://i.imgur.com/'.concat(image.hash, image.ext));
		  } catch(e) { debug.warn(e) }
		  return notFound;
		});
		return globalFetch(url).then(response => response.document.querySelector('link[rel="image_src"]').href);
	  case 'pixhost.to':
		if (url.pathname.startsWith('/gallery/')) return globalFetch(url).then(response =>
			Promise.all(Array.from(response.document.querySelectorAll('div.images > a')).map(a => imageUrlResolver(a.href))));
		if (url.pathname.startsWith('/show/')) return globalFetch(url)
		  .then(response => response.document.querySelector('img#image').src);
		break;
	  case 'malzo.com':
		if (url.pathname.startsWith('/al/')) return imageHostHandlers['malzo'].galleryResolver(url);
		break;
	  case 'imgbb.com': case 'ibb.co':
		if (url.pathname.startsWith('/album/')) return imageHostHandlers['imgbb'].galleryResolver(url);
		break;
	  case 'jerking.empornium.ph':
		if (url.pathname.startsWith('/album/')) return imageHostHandlers['jerking'].galleryResolver(url);
		break;
	  case 'imgbox.com':
		if (url.pathname.startsWith('/g/')) return globalFetch(url).then(response =>
			Promise.all(Array.from(response.document.querySelectorAll('div#gallery-view-content > a'))
				.map(a => imageUrlResolver('https://imgbox.com'.concat(a.pathname)))));
		break;
	  case 'postimage.org': case 'postimg.cc':
		if (!url.pathname.startsWith('/gallery/')) break;
		return PostImage.galleryResolver(url);
	  case 'www.imagevenue.com': case 'imagevenue.com':
		return globalFetch(url, { headers: { Referer: 'http://www.imagevenue.com/' } }).then(function(response) {
		  var images = Array.from(response.document.querySelectorAll('div.card img')).map(function(img) {
			return img.src.includes('://cdn-images') ? Promise.resolve(img.src) : imageUrlResolver(img.parentNode.href);
		  });
		  return images.length > 1 ? Promise.all(images) : images.length == 1 ? images[0] : notFound;
		});
	  case 'www.imageshack.us': case 'imageshack.us':
		return globalFetch(url).then(response => response.document.querySelector('a#share-dl').href);
	  case 'www.flickr.com': case 'flickr.com':
		if (!url.pathname.startsWith('/photos/')) break;
		return globalFetch(url).then(function(response) {
		  if (/\b(?:modelExport)\s*:\s*(\{.+\}),/.test(response.responseText)) try {
			var urls = JSON.parse(RegExp.$1).main['photo-models'].map(function(photoModel) {
			  var sizes = Object.keys(photoModel.sizes).sort((a, b) => photoModel.sizes[b].width * photoModel.sizes[b].height
					- photoModel.sizes[a].width * photoModel.sizes[a].height);
			  return sizes.length > 0 ? 'https:'.concat(photoModel.sizes[sizes[0]].url) : null;
			});
			if (urls.length == 1) return urls[0]; else if (urls.length > 1) return urls;
		  } catch(e) { console.warn(e) }
		  return notFound;
		});
	  case 'photos.google.com':
		return googlePhotosResolver(url);
	  case 'www.500px.com': case 'web.500px.com': case '500px.com':
		if (/^\/photo\/(\d+)\b/i.test(url.pathname))
		  return _500pxUrlHandler('photos?ids='.concat(RegExp.$1));
		else if (/\/galleries\/([\w\-]+)/i.test(url.pathname)) {
		  let galleryId = RegExp.$1;
		  return globalFetch(url, { rsponseType: 'text' }).then(function(response) {
			if (!/\b(?:App\.CuratorId)\s*=\s*"(\d+)"/.test(response.responseText)) return Promise.reject('Unexpected page structure');
			return _500pxUrlHandler('users/' + RegExp.$1 + '/galleries/' + galleryId + '/items?sort=position&sort_direction=asc&rpp=999');
		  });
		}
		break;
	  case 'www.pxhere.com': case 'pxhere.com':
		if (url.pathname.includes('/photo/')) return globalFetch(url).then(response =>
			JSON.parse(response.document.querySelector('div.hub-media-content > script[type="application/ld+json"]').text).contentUrl);
		else if (url.pathname.includes('/collection/')) return pxhereCollectionResolver(url);
		break;
	  case 'www.unsplash.com': case 'unsplash.com':
		if (url.pathname.startsWith('/photos/')) return globalFetch(url.origin + url.pathname + '/download', { method: 'HEAD' })
		  .then(response => response.finalUrl.replace(/\?.*$/, ''));
		else if (url.pathname.includes('/collections/')) return unsplashCollectionResolver(url);
		break;
	  case 'www.pexels.com': case 'pexels.com':
		if (url.pathname.startsWith('/photo/')) return globalFetch(url)
		  .then(response => response.document.querySelector('meta[property="og:image"][content]').content.replace(/\?.*$/, ''));
		else if (url.pathname.startsWith('/collections/')) return pexelsCollectionResolver(url);
		break;
	  case 'www.piwigo.org': case 'piwigo.org':
		/*if (url.pathname.includes('/picture/')) */return globalFetch(url, { responseType: 'text' }).then(function(response) {
		  if (/^(?:RVAS)\s*=\s*(\{[\S\s]+?\})$/m.test(response.responseText)) try {
			var derivatives = eval('(' + RegExp.$1 + ')').derivatives.sort((a, b) => b.w * b.h - a.w * a.h);
			return derivatives.length > 0 ? 'https://piwigo.org/demo/'.concat(derivatives[0].url) : notFound;
		  } catch(e) { console.warn(e) }
		  return Promise.reject('Unexpected page structure');
		});
		break;
	  case 'www.freeimages.com': case 'freeimages.com':
		if (url.pathname.startsWith('/photo/')) return globalFetch(url).then(function(response) {
		  var types = Array.from(response.document.querySelectorAll('ul.download-type > li > span.reso'))
		  	.sort((a, b) => eval(b.textContent.replace('x', '*')) - eval(a.textContent.replace('x', '*')));
		  return types.length > 0 ? url.origin.concat(types[0].parentNode.querySelector('a').pathname) : notFound;
		});
		break;
	  case 'redacted.ch':
		if (url.pathname != '/image.php') break;
		return globalFetch(url, { method: 'HEAD' }).then(response => response.finalUrl);
	  case 'demo.cloudimg.io':
		if (!/\b(https?:\/\/\S+)$/.test(url.pathname.concat(url.search, url.hash))) break;
		var resolved = RegExp.$1;
		if (/\b(?:https?):\/\/(?:\w+\.)*discogs\.com\//i.test(resolved)) break;
		return imageResolver(resolved);
	  case 'fastpic.ru':
		if (url.pathname.startsWith('/view/'))
		  return globalFetch(url).then(response => imageUrlResolver(response.document.querySelector('a.img-a').href));
		if (url.pathname.startsWith('/fullview/')) return globalFetch(url).then(function(response) {
		  var node = response.document.getElementById('image');
		  if (node != null) return node.src;
		  return /\bvar\s+loading_img\s*=\s*'(\S+?)';/.test(response.responseText) ? RegExp.$1 : notFound;
		});
		break;
	  case 'www.radikal.ru': case 'radikal.ru': case 'a.radikal.ru':
		return globalFetch(url).then(response => response.document.querySelector('div.mainBlock img').src);
	  case 'imageban.ru': case 'ibn.im':
		return globalFetch(url).then(response => response.document.getElementById('img_main').src /* dataset.original */);
	  // music-related
	  case 'www.musicbrainz.org': case 'musicbrainz.org':
		if (!['release', 'release-group'].some(branch => url.pathname.includes('/'.concat(branch, '/')))) break;
		return globalFetch(url).then(function(response) {
		  var node = response.document.querySelector('a.artwork-image');
		  if (node != null) return node.href;
		  return (node = response.document.querySelector('div.cover-art > img')) != null ? node.src : notFound;
		});
	  case 'music.apple.com':
		if (!/^https?:\/\/(?:\w+\.)*apple\.com\/.*\/(\d+)(?=$|\?)/i.test(url)) break;
		return globalFetch(url).then(function(response) {
		  var meta = response.document.querySelector('meta[property="og:image"][content]');
		  if (meta == null || !meta.content) return notFound;
		  return verifyImageUrl(meta.content.replace(/\/\d+x\d+\w*(?=\.\w+$)/, '/100000x100000-999')).catch(reason => meta.content);
		});
	  case 'www.deezer.com': case 'deezer.com':
		return globalFetch(url).then(function(response) {
		  var meta = response.document.querySelector('meta[property="og:image"][content]');
		  if (meta == null || !meta.content) return notFound;
		  return verifyImageUrl(meta.content.replace(/\/\d+x\d+\w*(?=\.\w+$)/, '/1400x1400-000000-100-0-0'))
			.catch(reason => meta.content);
		});
	  case 'www.qobuz.com': case 'qobuz.com':
		if (!url.pathname.includes('/album/')) break;
		return globalFetch(url).then(function(response) {
		  var img = response.document.querySelector('div.album-cover > img');
		  if (img == null) return notFound;
		  return verifyImageUrl(img.src.replace(/_\d{3}(?=\.\w+$)/, '_max')).catch(reason => img.src);
		});
	  case 'www.boomkat.com': case 'boomkat.com':
		if (!url.pathname.startsWith('/products/')) break;
		return globalFetch(url).then(function(response) {
		  var img = response.document.querySelector('img[itemprop="image"]');
		  if (img == null) return notFound;
		  return verifyImageUrl(img.src.replace(/\/large\//i, '/original/')).catch(reason => img.src);
		});
	  case 'www.bleep.com': case 'bleep.com':
		if (!url.pathname.startsWith('/release/')) break;
		return globalFetch(url).then(function(response) {
		  var meta = response.document.querySelector('meta[property="og:image"][content]');
		  if (meta == null) return notFound;
		  return verifyImageUrl(meta.content.replace(/\/r\/[a-z]\//i, '/r/')).catch(reason => meta.content);
		});
	  case 'www.soundcloud.com': case 'soundcloud.com':
		return globalFetch(url).then(function(response) {
		  var meta = response.document.querySelector('meta[property="og:image"][content]');
		  if (meta == null) return notFound;
		  return verifyImageUrl(meta.content.replace(/\bt\d+x\d+(?=\.\w+$)/, 'original')).catch(reason => meta.content);
		});
	  case 'www.prestomusic.com': case 'prestomusic.com':
		if (!url.pathname.includes('/products/')) break;
		return globalFetch(url)
		  .then(response => verifyImageUrl(response.document.querySelector('div.c-product-block__aside > a').href.replace(/\?\d+$/)));
	  case 'www.bontonland.cz':case 'bontonland.cz':
		return globalFetch(url).then(response => response.document.querySelector('a.detailzoom').href);
	  case 'www.nativedsd.com':case 'nativedsd.com':
		if (!url.pathname.includes('/albums/')) break;
		return globalFetch(url).then(response => response.document.querySelector('a#album-cover').href);
	  case 'www.prostudiomasters.com': case 'prostudiomasters.com':
		if (!url.pathname.includes('/album/')) break;
		return globalFetch(url).then(function(response) {
		  var a = response.document.querySelector('img.album-art');
		  return verifyImageUrl(a.currentSrc).catch(reason => a.src);
		});
	  case 'www.e-onkyo.com': case 'e-onkyo.com':
		if (!url.pathname.includes('/album/')) break;
		return globalFetch(url).then(function(response) {
		  var a = response.document.querySelector('figure > a.colorbox');
		  return new URL(response.finalUrl).origin.concat(a.pathname);
		})
	  case 'store.acousticsounds.com':
		return globalFetch(url).then(function(response) {
		  var link = response.document.querySelector('div#detail > link[rel="image_src"]');
		  return verifyImageUrl(link.href.replace(/\/medium\//i, '/large/')).catch(reason => link.href);
		});
	  case 'www.indies.eu': case 'indies.eu':
		if (!url.pathname.includes('/alba/')) break;
		return globalFetch(url).then(response => verifyImageUrl)(response.document.querySelector('div.obrazekDetail > img').src);
	  case 'www.beatport.com': case 'beatport.com':
		if (!url.pathname.includes('/release/')) break;
		return globalFetch(url).then(response =>
			verifyImageUrl(response.document.querySelector('div > img.interior-release-chart-artwork').src));
	  case 'www.supraphonline.cz': case 'supraphonline.cz':
		if (!url.pathname.includes('/album/')) break;
		return globalFetch(url).then(function(response) {
		  verifyImageUrl(response.document.querySelector('meta[itemprop="image"]').content.replace(/\?.*$/, '')).catch(reason => notFound);
		});
	  case 'vgmdb.net':
		if (!url.pathname.includes('/album/')) break;
		return globalFetch(url).then(function(response) {
		  var div = response.document.querySelector('div#coverart');
		  return verifyImageUrl(/\b(?:url)\s*\(\"(.*)"\)/i.test(div.style['background-image']) && RegExp.$1).catch(reason => notFound);
		});
	  case 'www.ototoy.jp': case 'ototoy.jp':
		return globalFetch(url).then(function(response) {
		  var img = response.document.querySelector('div#tralbumArt > a.popupImage');
		  return verifyImageUrl(img.dataset.src).catch(reason => img.src);
		});
	  case 'music.yandex.ru':
		if (!url.pathname.includes('/album/')) break;
		return globalFetch(url).then(function(response) {
		  var script = response.document.querySelector('script.light-data');
		  return verifyImageUrl(JSON.parse(script.text).image).catch(reason => notFound);
		});
	  // movie-related
	  case 'www.imdb.com': case 'imdb.com':
		if (!['title/tt', 'name/nm'].some(cat => url.pathname.startsWith('/' + cat))) break;
		return globalFetch(url).then(function(response) {
		  const galleryDetector = /\/mediaindex(?:[\/\?].*)?$/i, imgStripper = /\._V\d+_[\w\,]*(?=\.)/;
		  if (!galleryDetector.test(response.finalUrl)) {
			let node = response.document.head.querySelector(':scope > script[type="application/ld+json"]');
			if (node != null) try {
			  let image = JSON.parse(node.text).image;
			  if (typeof image == 'string') return verifyImageUrl(image.replace(imgStripper, '')).catch(reason => notFound);
			} catch(e) { console.warn(e) }
			node = response.document.querySelector('meta[property="og:image"][content]');
			return node != null && !/\/imdb\w*_logo\./i.test(node.content) ?
			  node.content.replace(imgStripper, '') : notFound;
		  }
		  var titleId = /\/title\/(tt\d+)\//i.test(response.finalUrl) && RegExp.$1;
		  return titleId ? globalFetch(response.finalUrl.replace(galleryDetector, '/mediaviewer'), { responseType: 'text' }).then(function(response) {
			if (/\b(?:window\.IMDbMediaViewerInitialState)\s*=\s*(\{.*\});/.test(response.responseText)) try {
			  let allImages = eval('(' + RegExp.$1 + ')').mediaviewer.galleries[titleId].allImages;
			  if (allImages.length > 0) return allImages.map(image => image.src.replace(imgStripper, ''));
			} catch(e) { console.warn(e) }
			return notFound;
		  }) : Promise.reject('title id not found');
		});
	  case 'www.themoviedb.org': case 'themoviedb.org':
		if (!['movie', 'person'].some(cat => url.pathname.startsWith('/' + cat + '/'))) break;
		return globalFetch(url).then(function(response) {
		  var node = response.document.querySelector('meta[property="og:image"][content]');
		  return verifyImageUrl(node.content.replace(/\/p\/\w+\//i, '/p/original/')).catch(function(reason) {
			node = response.document.querySelector('div.image_content > img');
			return verifyImageUrl(node.dataset.src.replace(/\/p\/\w+\//i, '/p/original/'))
			  .catch(reason => verifyImageUrl(node.src.replace(/\/p\/\w+\//i, '/p/original/')))
			  .catch(reason => verifyImageUrl(dataset.src)).catch(reason => node.src);
		  }).catch(reason => notFound);
		});
	  case 'www.omdb.org': case 'omdb.org':
		if (!['movie', 'person'].some(cat => url.pathname.startsWith('/' + cat + '/'))) break;
		return globalFetch(url).then(function(response) {
		  var node = response.document.querySelector('meta[property="og:image"][content]');
		  return node != null ? verifyImageUrl(node.content) : notFound;
		});
	  case 'www.thetvdb.com': case 'thetvdb.com':
		if (!['movies', 'series', 'people'].some(cat => url.pathname.startsWith('/' + cat + '/'))) break;
		return globalFetch(url).then(response => verifyImageUrl(response.document.querySelector('img.img-responsive').src));
	  case 'www.rottentomatoes.com': case 'rottentomatoes.com':
		if (!['m', 'celebrity', 'tv'].some(cat => url.pathname.startsWith('/' + cat + '/'))) break;
		return globalFetch(url).then(function(response) {
// 		  if (/\b(?:context\.shell)\s*=\s*(\{.+?});/.test(response.responseText)) try {
// 			return JSON.parse(RegExp.$1).header.certifiedMedia.certifiedFreshMovieInTheater4.media.posterImg;
// 		  } catch(e) { console.warn(e) }
		  return verifyImageUrl(response.document.querySelector('meta[property="og:image"]').content);
		});
	  case 'www.bcdb.com': case 'bcdb.com':
		if (!['cartoon'].some(cat => url.pathname.startsWith('/' + cat + '/'))) break;
		return globalFetch(url).then(response =>
			verifyImageUrl(document.location.protocol.concat(response.document.querySelector('meta[property="og:image"]').content)));
	  case 'www.boxofficemojo.com': case 'boxofficemojo.com':
		if (!['releasegroup'].some(cat => url.pathname.startsWith('/' + cat + '/'))) break;
		return globalFetch(url).then(response => verifyImageUrl(response.document.querySelector('div.mojo-primary-image img').src));
	  case 'www.metacritic.com': case 'metacritic.com':
		return globalFetch(url).then(function(response) {
		  var image = response.document.querySelector('meta[property="og:image"]').content;
		  return verifyImageUrl(image.replace(/-\d+h(?=(?:\.\w+)?$)/, '')).catch(reason => image);
		});
	  case 'www.csfd.cz': case 'csfd.cz':
		if (!['film', 'tvurce'].some(cat => url.pathname.startsWith('/' + cat + '/'))) break;
		return globalFetch(url).then(function(response) {
		  const gallerySel = 'div.ct-general.photos > div.content > ul > li > div.photo';
		  if (response.document.querySelectorAll(gallerySel).length > 0) return new Promise(function(resolve, reject) {
			var urls = [], domParser = new DOMParser, origin = new URL(response.finalUrl).origin;
			loadPage(response.finalUrl.replace(/\/strana-\d+(?=$|\/|\?)/, ''));

			function loadPage(url) {
			  GM_xmlhttpRequest({ method: 'GET', url: url,
				onload: function(response) {
				  if (response.status < 200 || response.status >= 400) return reject(defaultErrorHandler(response));
				  var dom = domParser.parseFromString(response.responseText, 'text/html');
				  Array.prototype.push.apply(urls, Array.from(dom.querySelectorAll(gallerySel))
					.map(div => /^(?:url)\s*\("?(.+?)"?\)$/i.test(div.style.backgroundImage) ?
						 'https:'.concat(RegExp.$1).replace(/\?.*$/, '') : null));
				  var nextPage = dom.querySelector('div.paginator > a.next[href]');
				  if (nextPage != null) loadPage(origin.concat(nextPage.pathname, nextPage.search)); else resolve(urls);
				},
				onerror: response => { reject(defaultErrorHandler(response)) },
				ontimeout: response => { reject(defaultTimeoutHandler(response)) },
			  });
			}
		  });
		  var img = ['img.film-poster', 'img.creator-photo', 'div.image > img']
		  	.reduce((acc, selector) => acc || response.document.querySelector(selector), null);
		  return img != null ? verifyImageUrl(img.src.replace(/\?.*$/, '')) : notFound;
		});
	  case 'www.fdb.cz': case 'fdb.cz':
		//if (!url.pathname.startsWith('/film/')) break;
		return globalFetch(url).then(function(response) {
		  var a = response.document.querySelector('a.boxPlakaty');
		  if (a == null) return Promise.reject('Invalid page structure');
		  a.hostname = 'www.fdb.cz';
		  return globalFetch(a.href).then(function(response) {
			var imgs = response.document.querySelectorAll('span#popup_plakaty > img');
			return imgs.length > 0 ? verifyImageUrl(imgs[0].src) : notFound;
		  });
		});
	  case 'www.caps-a-holic.com': case 'caps-a-holic.com':
		if (url.pathname != '/c.php') break;
		return globalFetch(url).then(function(response) {
		  function heightExtractor(n) {
			var node = response.document.querySelector('div.main > div.c_table > div[style]:nth-of-type(' + n + ')');
			if (node != null && /\b(\d{3,})\s?[x×]\s?(\d{3,})\b/.test(node.textContent)) return parseInt(RegExp.$2);
			console.warn(response.finalUrl, 'failed to get resolution (' + n + ')', node);
			return null;
		  }
		  const baseUrl = 'https://caps-a-holic.com/c_image.php?a=0&x=0&y=0&l=1';
		  return Array.from(response.document.querySelectorAll('div.main > div[style] > a > img.thumb')).map(function(img) {
			var query = new URLSearchParams(new URL(img.parentNode.href).search);
			return [
			  `${baseUrl}&s=${parseInt(query.get('s1'))}&max_height=${heightExtractor(2)}`,
			  `${baseUrl}&s=${parseInt(query.get('s2'))}&max_height=${heightExtractor(3)}`,
			];
		  });
		});
	  case 'www.screenshotcomparison.com': case 'screenshotcomparison.com':
		if (url.pathname.startsWith('/comparison/')) return globalFetch(url).then(function(response) {
		  const origin = new URL(response.finalUrl).origin;
		  return Array.from(response.document.querySelectorAll('div#img_nav li > a')).map(function(a) {
			return globalFetch(origin.concat(a.pathname), { responseType: 'text' }).then(response => [
			  /\b(?:images)\[1\]='(\S+?)'/.test(response.responseText) && RegExp.$1,
			  /\b(?:images)\[0\]='(\S+?)'/.test(response.responseText) && RegExp.$1,
			].map(src => origin.concat(src)));
		  });
		});
		break;
	  case 'www.dvdbeaver.com': case 'dvdbeaver.com':
		if (url.pathname.startsWith('/film')) return globalFetch(url).then(function(response) {
		  const origin = new URL(response.finalUrl).origin;
		  return Array.from(response.document.querySelectorAll('div[align="center"] > table > tbody > tr > td > a[target="_blank"] > img'))
		  	.map(img => origin.concat(img.parentNode.pathname));
		});
		break;
	}
	return globalFetch(url, { headers: { 'Referer': url.origin } }).then(function(response) {
	  if (url.pathname.startsWith('/album/')
		  && response.document.querySelector('div#tabbed-content-group > div.content-listing > div.pad-content-listing') != null)
		return new Chevereto(url.hostname).galleryResolver(url);
	  var meta = [
		'head > meta[property="og:image"][content]', 'head > meta[itemprop="image"][content]', 'head > meta[name="og:image"][content]',
	  ].reduce((acc, selector) => acc || response.document.querySelector(selector), null);
	  return meta != null && meta.content ? meta.content : notFound;
	});
  }));
}

function logFail(message) {
  var log = document.getElementById('ihh-console');
  if (log == null) {
	log = document.createElement('div');
	log.id = 'ihh-console';
	log.style = 'position: fixed; bottom: 20px; right: 20px; width: 64em; border: solid lightsalmon 4px;' +
	  ' background-color: antiquewhite; padding: 10px; opacity: 1;' +
	  ' transition: opacity 1000ms linear; -webkit-transition: opacity 1000ms linear;';
	document.body.append(log);
  } else if (log.hTimer) {
	clearTimeout(log.hTimer);
	log.style.opacity = 1;
  }
  var div = document.createElement('div');
  div.style = 'font: 600 9pt Verdana, sans-serif; color: red;';
  div.textContent = message;
  log.append(div);
  log.hTimer = setTimeout(function(node) {
	node.style.opacity = 0;
	node.hTimer = setTimeout(function(node) { node.remove() }, 1000, node);
  }, 30000, log);
}