Image Host Helper

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

Mint 2020.10.13.. Lásd a legutóbbi verzió

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name         Image Host Helper
// @namespace    https://greatest.deepsurf.us/users/321857-anakunda
// @version      1.083
// @description  Directly upload local / rehost remote images or galleries to whatever supported image host by dropping/pasting them to target field
// @icon         
// @author       Anakunda
// @copyright    2020, Anakunda (https://greatest.deepsurf.us/cs/users/321857-anakunda)
// @license      GPL-3.0-or-later
// @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://greatest.deepsurf.us/scripts/408084-xhrlib/code/xhrLib.js
// @require      https://greatest.deepsurf.us/scripts/404516-progressbars/code/progressBars.js
// @require      https://greatest.deepsurf.us/scripts/401726-imagehostuploader/code/imageHostUploader.js
// ==/UserScript==

'use strict';

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

Array.prototype.flatten = function() {
  return this.reduce(function(flat, toFlatten) {
	return flat.concat(Array.isArray(toFlatten) ? toFlatten.flatten() : toFlatten);
  }, []);
};

var cheveretoCustomHosts = GM_getValue('chevereto_custom_hosts');
if (cheveretoCustomHosts !== undefined) try {
  JSON.parse(cheveretoCustomHosts).forEach(function(siteDef) {
	if (!siteDef.host_name || !siteDef.alias) {
	  console.warn('Incomplete Chevereto custom site definition:', siteDef);
	  return;
	}
	imageHostHandlers[siteDef.alias.replace(nonWordStripper, '').toLowerCase()] = new Chevereto(
	  siteDef.host_name,
	  siteDef.alias,
	  siteDef.types,
	  siteDef.size_limit, {
		sizeLimitAnonymous: siteDef.size_limit_anonymous,
		configPrefix: siteDef.config_prefix,
		apiEndpoint: siteDef.api_endpoint,
		apiFieldName: siteDef.api_field_name,
		apiResultKey: siteDef.api_result_key,
		jsonEndpoint: siteDef.json_endpoint,
	});
  });
} catch (e) { console.warn(e) } else GM_setValue('chevereto_custom_hosts', '[]');
console.log('Image host handlers:', imageHostHandlers);

['upload_hosts', 'rehost_hosts'].forEach(propName => { if (!GM_getValue(propName)) GM_setValue(propName, [
  'PTPimg', 'ImgBB', 'PixHost', 'ImgBox', 'Jerking', 'Abload', 'VgyMe', 'Slowpoke', 'FunkyIMG',
  'Gifyu', 'PostImage', 'GeekPic', 'LightShot', 'ImgURL', 'Radikal', 'Z4A', 'PicaBox', 'PimpAndHost',
  'Imgur', 'ImageBan', 'SVGshare', 'UuploadIr', 'Catbox', 'ImageVenue', 'FastPic',
].join(', ')) });
[
  ['passthepopcorn.me', [
	'PTPimg', 'PixHost', 'ImgBB', 'Jerking', 'Gifyu', 'Slowpoke', 'ImgBox', 'Abload', 'VgyMe', 'FunkyIMG', 'GeekPic',
	'LightShot', 'ImgURL', 'Radikal', 'Z4A', 'PicaBox', 'PimpAndHost', 'ImageBan', 'UuploadIr', 'Catbox', 'ImageVenue',
  ]],
  ['notwhat.cd', ['NWCD']],
].forEach(hostDefaults => { if (!GM_getValue(hostDefaults[0])) GM_setValue(hostDefaults[0], hostDefaults[1].join(', ')) });

var imageHosts = new ImageHostManager(logFail,
  GM_getValue(document.domain) || GM_getValue('upload_hosts'),
  GM_getValue(document.domain) || GM_getValue('rehost_hosts'));

imageHostUploaderInit(inputDataHandler, textAreaDropHandler, textAreaPasteHandler, imageUrlResolver);

// Set single input UI handlers
document.querySelectorAll([
  'image', 'picture', 'cover', 'photo', 'avatar', 'poster', 'screen',
].map(pattern => ['id', 'name'].map(attr => 'input[type="text"][' + attr + '*="' + pattern + '"]')).join(','))
  .forEach(setInputHandlers);
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);
  });
}
// Set multiple inputs UI handlers
for (let textArea of document.getElementsByTagName('textarea')) {
  if (textArea.className != 'ua-input') setTextAreahandlers(textArea);
}

// site-specific extensions
switch (document.domain) {
  case 'passthepopcorn.me':
	// Auto-fill missing/invalid images from IMDB
	if (/\/artist\.php\?action=edit&artistid=(\d+)\b/i.test(document.URL)) {
	  let artistId = parseInt(RegExp.$1), input = document.querySelector('input[name="image"]');
	  if (input != null) verifyImageUrl(input.value).catch(() => localXHR('/artist.php?id=' + artistId).then(function(dom) {
		let imdb = dom.querySelector('div#artistinfo > div.panel__body > ul.list > li > a');
		if (imdb != null) imageUrlResolver(imdb.href).then(setCover.bind(input), reason => { logFail('No IMDB photo of this artist') });
	  }));
	} else if (/\/torrents\.php??action=editgroup&groupid=(\d+)\b/i.test(document.URL)) {
	  let groupId = parseInt(RegExp.$1), input = document.querySelector('input[name="image"]');
	  if (input != null) verifyImageUrl(input.value).catch(() => localXHR('/torrents.php?id=' + groupId).then(function(dom) {
		let imdb = dom.querySelector('a#imdb-title-link');
		if (imdb != null) imageUrlResolver(imdb.href).then(setCover.bind(input), reason => { logFail('No IMDB poster for this movie') });
	  }));
	}
	// HJ Toolkit patch
	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);
	break;
  case 'tracker.czech-server.com':
	document.querySelectorAll('input[name="urlobr"]').forEach(setInputHandlers);
	break;
}

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

// Old versions adjustment
['image_size_warning', 'image_size_reduce_threshold'].forEach(itemProp => {
  var val = GM_getValue('itemProp');
  if (val < 8) GM_setValue('itemProp', val * 2**10);
});
var opti_PNG = GM_getValue('optipng', false);

function coverPreview(input, imgUrl, size) {
  var child = document.getElementById('cover-preview');
  if (child == null) {
	return; // not implemented so far
	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(size => {
		div.textContent = this.naturalWidth + '×' + this.naturalHeight + ' (' + formattedSize(size) + ')';
	  }, 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 writeInfo() {
  var input = document.querySelector('input[name="summary"]');
  if (input != null && !input.disabled && !input.value) input.value = 'Image update/rehost';
}

function setCover(url) {
  return verifyImageUrl(url).then(imageUrl => {
	this.value = imageUrl;
	writeInfo();
	coverPreview(this, imageUrl);
	checkImageSize(imageUrl, this).then(imageUrl => {
	  this.disabled = true;
	  return imageHosts.rehostImages([imageUrl]).then(singleImageGetter).then(imageUrl => {
		if (imageUrl == null) throw 'invalid image';
		this.value = imageUrl;
	  }).catch(reason => {
		this.value = imageUrl;
		logFail(reason + ' (not rehosted)');
	  }).then(() => {
		this.disabled = false;
		return imageUrl;
	  });
	});
	return imageUrl;
  });
}

function inputDataHandler(evt, data) {
  function rehoster(imageUrl) {
	evt.target.disabled = true;
	return imageHosts.rehostImages([imageUrl]).then(singleImageGetter).then(function(imageUrl) {
	  if (!imageUrl) throw 'invalid image';
	  evt.target.value = imageUrl;
	  writeInfo();
	}).catch(function(reason) {
	  evt.target.value = imageUrl;
	  Promise.resolve(reason).then(msg => { alert(msg + ' (not rehosted)') });
	}).then(() => { evt.target.disabled = false });
  }

  if (!data) return true;
  if (data.files.length > 0) {
	if (data.files[0].type && !data.files[0].type.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 lastPct, lastUpdate, current;
	function progressHandler(worker, param = null) {
	  if (param && typeof param == 'object') {
		if (param.readyState > 1 || current != undefined && worker !== current || Date.now() < lastUpdate + 100) return;
		let pct = Math.floor(Math.min(param.done * 100 / param.total, 100));
		if (pct <= lastPct) return;
		evt.target.value = 'Uploading... [' + (lastPct = pct) + '%]';
		lastUpdate = Date.now();
	  } else if (param == null) {
		lastPct = lastUpdate = undefined;
		current = worker;
		evt.target.value = 'Uploading...';
	  }
	}
	let size = data.files[0].size;
	checkImageSize(data.files[0], evt.target, size, progressHandler).catch(reason => data.files[0]).then(function(result) {
	  (result instanceof File ? imageHosts.uploadImages([result], progressHandler).then(singleImageGetter).then(function(imgUrl) {
		evt.target.value = imgUrl;
		coverPreview(evt.target, imgUrl, size);
		writeInfo();
	  }) : rehoster(result)).then(function() {
		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);
	  }, function(reason) {
		imageClear(evt);
		evt.target.style.backgroundColor = null;
		evt.target.style.color = null;
		Promise.resolve(reason).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);
	  checkImageSize(imageUrl, evt.target).then(rehoster);
	}).catch(function(e) {
	  console.error(e);
	  alert(e);
	});
	return false;
  }
  return true;
}

function rehoster(promises, resultsHandler, target = null) {
  if (!Array.isArray(promises)) throw 'invalid parameter';
  return Promise.all(promises).then(function(resolved) {
	var resolvedUrls = resolved.flatten();
	if (target instanceof HTMLElement) {
	  target.disabled = true;
	  if (resolvedUrls.length > 1 && !['notwhat.cd'].some(hostname => document.domain == hostname))
		var progressBar = new RHProgressBar(target, resolvedUrls.length);
	}
	return (opti_PNG && target instanceof HTMLElement ?
		Promise.all(resolvedUrls.map(resolvedUrl => optiPNG(resolvedUrl).catch(reason => Promise.resolve(resolvedUrl))))
			: Promise.resolve(resolvedUrls))
	.then(srcUrls => imageHosts.rehostImages(srcUrls, RHProgressBar.prototype.update.bind(progressBar)).catch(function(reason) {
	  Promise.resolve(reason).then(msg => { alert(msg + ' (not rehosted)') });
	  RHProgressBar.prototype.update.call(progressBar, -1, false);
	  return verifyImageUrls(srcUrls);
	}).then(function(results) {
	  resultsHandler(results, arrayGrouping(resolved).flatten());
	  RHProgressBar.prototype.cleanUp.call(progressBar);
	  if (target instanceof HTMLElement) target.disabled = false;
	}));
  });
}

function textAreaDropHandler(evt) {
  if (!evt.dataTransfer || 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));
	(function() {
	  if (!opti_PNG || !images.every(image => image.type == 'image/png')) return Promise.reject('!optiPNG');
	  ULProgressBar.prototype.update.call(progressBar, -1);
	  return rehoster([Promise.all(images.map((image, index) => optiPNG(image, (param = null) =>
		ULProgressBar.prototype.update.call(progressBar, -1, param, index))))], resultsHandler);
	})().catch(reason => imageHosts.uploadImages(images, ULProgressBar.prototype.update.bind(progressBar))
	.then(resultsHandler, reason => { Promise.resolve(reason).then(msg => { alert(msg) }) })).then(function() {
	  ULProgressBar.prototype.cleanUp.call(progressBar);
	  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;
	rehoster(content.map(imageUrlResolver), resultsHandler, evt.target);
	evt.stopPropagation();
	return false;
  }
  return true;

  function resultsHandler(results, groups = undefined) {
	if (results.length <= 0) return;
	if (evt.altKey && !evt.target.noBBCode) {
	  let modal = document.createElement('div');
	  modal.id = 'ihh-template-selector-background';
	  modal.style = 'position: fixed; left: 0; top: 0; width: 100%; height: 100%; background-color: #0008;' +
		'opacity: 0; transition: opacity 0.15s linear;';
	  modal.innerHTML = `
<div id="ihh-template-selector" style="background-color: darkslategray; position: absolute; top: 30%; left: 35%; border-radius: 0.5em; padding: 20px 30px;">
  <div style="color: white;margin-bottom: 20px;">Insert as:</div>
  <input id="btn-insert" type="button" value="Insert" style="margin-top: 30px"/>
  <input id="btn-cancel" type="button" value="Cancel" style="margin-top: 30px"/>
</div>
`;
	  document.body.append(modal);
	  let form = document.getElementById('ihh-template-selector'),
		  btnInsert = document.querySelector('div#ihh-template-selector input#btn-insert'),
		  btnCancel = document.querySelector('div#ihh-template-selector input#btn-cancel');
	  if (form == null || btnInsert == null || btnCancel == null) {
		console.warn('Dialog creation error');
		insertResults();
		return;
	  }
	  [
		['BBcode: original size', 1],
		['BBcode: thumbnails with link to original', 2],
		['BBcode: thumbnails with link to share page', 3],
		['BBcode: screenshot comparison (PTP)', 4],
		['BBcode: screenshot comparison + encode images (PTP)', 5],
		['Markdown: original size', 9],
		['HTML: original size', 6],
		['HTML: thumbnails with link to original', 7],
		['HTML: thumbnails with link to share page', 8],
		['Raw links', 0],
	  ].forEach(function(item) {
		var radio = document.createElement('input');
		radio.type = 'radio';
		radio.name = 'template';
		radio.value = item[1];
		radio.style = 'margin: 5px 15px 5px 0px; cursor: pointer;';
		var label = document.createElement('label');
		label.style = 'color: white; cursor: pointer; -webkit-user-select: none; ' +
		  '-moz-user-select: none; -ms-user-select: none; user-select: none;';
		label.append(radio);
		label.append(item[0]);
		form.insertBefore(label, btnInsert);
		var br = document.createElement('br');
		form.insertBefore(br, btnInsert);
	  });
	  if (!results.some(result => typeof result == 'object'
			&& urlParser.test(result.original) && urlParser.test(result.thumb))) disableItem(2, 7);
	  if (!results.some(result => typeof result == 'object'
			&& urlParser.test(result.original) && urlParser.test(result.share))) disableItem(3, 8);
	  if (results.length % 2 != 0) disableItem(4, 5);
	  form.onclick = evt => { evt.stopPropagation() };
	  btnInsert.onclick = function(evt) {
		var template = document.querySelector('div#ihh-template-selector input[name="template"]:checked');
		if (template != null) template = parseInt(template.value);
		modal.remove();
		insertResults(template);
	  };
	  modal.onclick = btnCancel.onclick = evt => { modal.remove() };
	  window.setTimeout(() => { modal.style.opacity = 1 }, 0);

	  function disableItem(...n) {
		n.forEach(function(n) {
		  var radio = document.querySelector('div#ihh-template-selector input[type="radio"][value="' + n + '"]');
		  if (radio == null) return;
		  radio.parentNode.style.opacity = 0.5;
		  radio.disabled = true;
		});
	  }
	} else insertResults();

	function insertResults(template = 1) {
	  if (evt.target.noBBCode) template = 0;
	  if (typeof template != 'number' || isNaN(template)) return;
	  var code = '', nl = [6, 7, 8].includes(template) ? '<br>\n' : '\n', _template;
	  results.forEach(function(result, index) {
		if (_template == 1 && /\[img\]\[\/img\]/i.test(evt.target.value)) {
		  evt.target.value = RegExp.leftContext + '[img]' + getImgUrl(result) + '[/img]' + RegExp.rightContext;
		  return;
		}
		_template = template;
		if (template == 2 && (typeof result != 'object' || !urlParser.test(result.original) || !urlParser.test(result.thumb))
			|| template == 3 && (typeof result != 'object' || !urlParser.test(result.share) || !urlParser.test(result.thumb)))
		  _template = 1;
		else if (template == 7 && (typeof result != 'object' || !urlParser.test(result.original) || !urlParser.test(result.thumb))
			|| template == 8 && (typeof result != 'object' || !urlParser.test(result.share) || !urlParser.test(result.thumb)))
		  _template = 6;
		else _template = template;
		if (index > 0) {
		  let thumb = [2, 3, 7, 8].includes(_template);
		  code += isGroupBoundary(groups, index) ? thumb ? nl : nl + nl : thumb ? ' ' : nl;
		}
		switch (_template) {
		  case 0: case 4: case 5: code += getImgUrl(result); break;
		  case 1: code += '[img]' + getImgUrl(result) + '[/img]'; break;
		  case 2: code += '[url=' + getImgUrl(result) + '][img]' + result.thumb + '[/img][/url]'; break;
		  case 3: code += '[url=' + result.share + '][img]' + result.thumb + '[/img][/url]'; break;
		  case 6: code += '<img src="' + getImgUrl(result) + '">'; break;
		  case 7: code += '<a href="' + getImgUrl(result) + '" target="_blank"><img src="' + result.thumb + '"></a>'; break;
		  case 8: code += '<a href="' + result.share + '" target="_blank"><img src="' + result.thumb + '"></a>'; break;
		  case 9: code += '![](' + getImgUrl(result) + ')'; break;
		}
	  });
	  if ([4, 5].includes(template)) {
		code = '[comparison=Source, Encode]' + code + '[/comparison]';
		if (template == 5) {
		  code += nl;
		  results.forEach((result, index) => { if (index % 2 != 0) code += nl + '[img]' + getImgUrl(result) + '[/img]' });
		}
	  }
	  if (evt.target.value.trimRight().length <= 0) evt.target.value = code; else if (evt.ctrlKey) {
		evt.target.value = evt.target.value.slice(0, evt.rangeOffset) + code + evt.target.value.slice(evt.rangeOffset);
	  } else evt.target.value = evt.target.value.trimRight() + nl + nl + code;

	  function getImgUrl(result) {
		if (typeof result == 'object' && urlParser.test(result.original)) return result.original;
		if (typeof result == 'string' && urlParser.test(result)) return result;
		throw 'Invalid result format';
	  }
	}
  }
}

function textAreaPasteHandler(evt) {
  if (!evt.clipboardData) 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));
	(function() {
	  if (!opti_PNG || !images.every(image => image.type == 'image/png')) return Promise.reject('!optiPNG');
	  ULProgressBar.prototype.update.call(progressBar, -1);
	  return rehoster([Promise.all(images.map((image, index) => optiPNG(image, (param = null) =>
		ULProgressBar.prototype.update.call(progressBar, -1, param, index))))], resultsHandler);
	})().catch(reason => imageHosts.uploadImages(images, ULProgressBar.prototype.update.bind(progressBar))
		.then(resultsHandler, reason => { Promise.resolve(reason).then(msg => { alert(msg) }) }))
	.then(function() { // __finally
	  ULProgressBar.prototype.cleanUp.call(progressBar);
	  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(function(resolved) {
// 	  evt.target.disabled = true;
// 	  var resolvedUrls = resolved.flatten();
// 	  if (resolvedUrls.length > 1 && !['notwhat.cd'].some(hostname => document.domain == hostname))
// 	  	progressBar = new RHProgressBar(evt.target, resolvedUrls.length);
// 	  imageHosts.rehostImages(resolvedUrls, RHProgressBar.prototype.update.bind(progressBar)).catch(function(reason) {
// 		Promise.resolve(reason).then(msg => { alert(msg + ' (not rehosted)') });
// 		RHProgressBar.prototype.update.call(progressBar, -1, false);
// 		return verifyImageUrls(resolvedUrls);
// 	  }).then(function(results) {
// 		resultsHandler(results, arrayGrouping(resolved).flatten());
// 		progressBar.cleanUp.call(progressBar);
// 		evt.target.disabled = false;
// 	  });
// 	});
// 	evt.stopPropagation();
// 	return false;
  }
  return true;

  function resultsHandler(results, groups = undefined) {
	var selStart = evt.target.selectionStart, phpBB = '';
	results.forEach(function(result, index) {
	  var thumb = evt.altKey && !evt.target.noBBCode && typeof result == 'object'
		&& urlParser.test(result.originasl) && urlParser.test(result.thumb);
	  if (index > 0) phpBB += isGroupBoundary(groups, index) ? thumb ? '\n' : '\n\n' : thumb ? ' ' : '\n';
	  if (typeof result == 'object' && result.original) var imgUrl = result.original;
	  	else if (typeof result == 'string') imgUrl = result;
	  		else throw 'Invalid result format';
	  phpBB += evt.target.noBBCode ? phpBB += imgUrl : !thumb ? '[img]' + imgUrl + '[/img]'
		: '[url=' + imgUrl + '][img]' + result.thumb + '[/img][/url]';
	});
	if (phpBB.length <= 0) return;
	evt.target.value = evt.target.value.slice(0, selStart) + phpBB + evt.target.value.slice(evt.target.selectionEnd);
	evt.target.setSelectionRange(selStart + phpBB.length, selStart + phpBB.length);
  }
}

function arrayGrouping(arr) {
  return Array.isArray(arr) ? arr.map(function(elem) {
	if (!Array.isArray(elem)) return 1;
	return elem.every(elem => !Array.isArray(elem)) ? elem.length : arrayGrouping(elem);
  }) : null;
}

function isGroupBoundary(groups, index) {
  return index > 0 && Array.isArray(groups)
  	&& groups.some((len, ndx, arr) => index == arr.slice(0, ndx).reduce((acc, len) => acc + len, 0));
}

function checkImageSize(imageUrl, node, size, progressHandler) {
  var imageSizeReduce = GM_getValue('image_size_reduce_threshold');
  if (!(imageSizeReduce > 0)) return Promise.resolve(imageUrl);
  if (!(node instanceof Node)) node = null;
  if (node != null) node.disabled = true;
  return (size > 0 ? Promise.resolve(size) : getRemoteFileSize(imageUrl)).then(function(size) {
	if (size <= imageSizeReduce * 2**10) return Promise.resolve(imageUrl);
	return reduceImageSize(imageUrl, GM_getValue('image_reduce_maxheight', 2160),
		GM_getValue('image_reduce_jpegquality', 90), progressHandler).then(function(output) {
	  if (node != null) node.value = output.uri;
	  Promise.resolve(output.size).then(reducedSize => {
		console.log('cover size reduced by ' + Math.round((size - reducedSize) * 100 / size) +
			'% (' + Math.ceil(size / 2**10) + ' → ' + Math.ceil(reducedSize / 2**10) + ' KiB)');
	  });
	  return output.uri;
	});
  }).catch(function(reason) {
	logFail('failed to get remote image size or optimize the image: ' + reason + ' (size reduction was not performed)');
	return imageUrl;
  }).then(function(finalUrl) {
	if (node != null) node.disabled = false;
	return finalUrl;
  });
}

function imageUrlResolver(url) {
  return urlResolver(url).then(url => verifyImageUrl(url).catch(function(reason) {
	const notFound = Promise.reject('No title image for this URL');
	function getFromMeta(root) {
	  var meta = root instanceof Document || root instanceof Element ? [
		'meta[property="og:image:secure_url"][content]',
		'meta[property="og:image"][content]',
		'meta[name="og:image"][content]',
		'meta[itemprop="og:image"][content]',
		'meta[itemprop="image"][content]',
	  ].reduce((elem, selector) => elem || root.querySelector(selector), null) : null;
	  return meta != null && urlParser.test(meta.content) ? meta.content : undefined;
	}

	try { url = new URL(url) } catch(e) { return Promise.reject(e) }
	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 globalXHR(url).then(function(response) {
	  var ref = response.document.querySelector('div#tralbumArt > a.popupImage');
	  ref = ref != null ? ref.href : getFromMeta(response.document);
	  return ref ? Promise.resolve(ref.replace(/_\d+(?=\.\w+$)/, '_0')) : notFound;
	}); else if (url.hostname.endsWith('7digital.com') && url.pathname.startsWith('/artist/'))
	  return globalXHR(url).then(function(response) {
		var img = response.document.querySelector('img[itemprop="image"]');
		return img != null ? img.src : notFound;
	  });
	else if (url.hostname.endsWith('geekpic.net')) return globalXHR(url).then(function(response) {
	  var a = response.document.querySelector('div.img-upload > a.mb');
	  return a != null ? a.href : notFound;
	}); else if (url.hostname.endsWith('qq.com') && url.pathname.includes('/album/')) return globalXHR(url).then(function(response) {
	  var img = response.document.querySelector('img#albumImg');
	  return img != null ? img.src.replace(/(?:_\d+)?(\.\w+)(?:\?.*)?$/, '$1').replace(/R\d+x\d+/, '') : notFound;
	}); else switch (url.hostname) {
	  // general image hostings
	  case 'www.imgur.com': case 'imgur.com':
		if (url.pathname.startsWith('/a/')) return globalXHR(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/' + image.hash + image.ext);
		  } catch(e) { debug.warn(e) }
		  return notFound;
		});
		return globalXHR(url).then(response => response.document.querySelector('link[rel="image_src"]').href);
	  case 'pixhost.to':
		if (url.pathname.startsWith('/gallery/')) return globalXHR(url).then(response =>
			Promise.all(Array.from(response.document.querySelectorAll('div.images > a')).map(a => imageUrlResolver(a.href))));
		if (url.pathname.startsWith('/show/')) return globalXHR(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 globalXHR(url).then(response =>
			Promise.all(Array.from(response.document.querySelectorAll('div#gallery-view-content > a'))
				.map(a => imageUrlResolver('https://imgbox.com' + 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 globalXHR(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 globalXHR(url).then(response => response.document.querySelector('a#share-dl').href);
	  case 'www.flickr.com': case 'flickr.com':
		if (!url.pathname.startsWith('/photos/')) break;
		return globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(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 imageUrlResolver(resolved);
	  case 'www.pimpandhost.com': case 'pimpandhost.com':
		if (!url.pathname.startsWith('/image/')) break;
		return globalXHR(url).then(function(response) {
		  var elem = resopnse.document.querySelector('div.main-image-wrapper');
		  if (elem != null && elem.dataset.src) return 'https:'.concat(elem.dataset.src);
		  elem = resopnse.document.querySelector('div.img-wrapper > a > img');
		  return elem != null ? 'https:'.concat(elem.src) : notFound;
		});
	  case 'www.screencast.com': case 'screencast.com':
		return globalXHR(url).then(function(response) {
		  var ref = response.document.querySelectorAll('ul#containerContent > li a.media-link');
		  if (ref.length <= 0) return getFromMeta(response.document) || notFound;
		  return Promise.all(Array.from(ref).map(a => imageUrlResolver('https://www.screencast.com' + a.href)));
		});
	  case 'abload.de':
		if (!url.pathname.startsWith('/image.php')) break;
		return globalXHR(url).then(function(response) {
		  var elem = response.document.querySelector('img#image');
		  if (elem == null) return notFound;
		  var src = new URL(elem.src);
		  return imageHostHandlers.abload.origin + src.pathname + src.search;
		});
	  case 'fastpic.ru':
		if (url.pathname.startsWith('/view/'))
		  return globalXHR(url).then(response => imageUrlResolver(response.document.querySelector('a.img-a').href));
		if (url.pathname.startsWith('/fullview/')) return globalXHR(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 globalXHR(url).then(response => response.document.querySelector('div.mainBlock img').src);
	  case 'imageban.ru': case 'ibn.im':
		return globalXHR(url).then(response => response.document.querySelector('a[download]').href);
	  case 'svgshare.com':
		return globalXHR(url).then(function(response) {
		  var link;
			response.document.querySelectorAll('ul#shares > li > input[type="text"]')
			  .forEach(input => { if (!link && /^(?:https?:\/\/.+\.svg)$/.test(input.value)) link = input.value; });
		  return link || notFound;
		});
	  case 'slow.pics':
		if (!url.pathname.startsWith('/c/')) break;
		return globalXHR(url).then(function(response) {
		  var nodes = response.document.querySelectorAll('img.card-img-top');
		  if (nodes.length > 1) return Array.from(nodes).map(img => img.src);
		  	else if (nodes.length > 0) return nodes[0].src;
		  nodes = response.document.querySelectorAll('a#comparisons + div.dropdown-menu > a.dropdown-item');
		  if (nodes.length > 0) return Promise.all(Array.from(nodes).map(a => globalXHR(url.origin + a.pathname).then(response =>
			Array.from(response.document.querySelectorAll('div#preload-images > img')).map(img => img.src))));
		  return notFound;
		});
	  case 'www.amazon.com': case 'amazon.com':
		return globalXHR(url).then(function(response) {
		  const rx = /\._\w+?(\d+)_(?=\.)/,
				getImgOrigin = colorImage => (colorImage.hiRes || colorImage.large || colorImage.thumb).replace(rx, '');
		  let obj = /^\s*(?:var\s+obj\s*=\s*jQuery\.parseJSON)\('(\{.+\})'\);/m.exec(response.responseText);
		  if (obj != null) {
			try { obj = JSON.parse(obj[1]) } catch(e) { try { obj = eval('(' + obj[1] + ')') } catch(e) { obj = { } } }
			let variants = Object.keys(obj.colorImages);
			if (variants.length > 0) return variants.map(key => obj.colorImages[key].map(getImgOrigin));
		  }
		  let colorImages = /^\s*'colorImages':\s*(\{.+\}),?$/m.exec(response.responseText);
		  if (colorImages != null) {
			try { colorImages = JSON.parse(colorImages[1].replace(/'/g, '"')) }
				catch(e) { try { colorImages = eval('(' + colorImages[1] + ')') } catch(e) { colorImages = { } } }
			if (Array.isArray(colorImages.initial) && colorImages.initial.length > 0)
			  return colorImages.initial.map(getImgOrigin);
		  }
		  let img = response.document.querySelector('div#ppd-left img');
		  return img != null && urlParser.test(img.src) ? img.src.replace(rx, '') : notFound;
		});
	  // music-related
	  case 'www.musicbrainz.org': case 'musicbrainz.org':
		if (!['release', 'release-group'].some(branch => url.pathname.includes('/' + branch + '/'))) break;
		return globalXHR(url).then(function(response) {
		  var node = response.document.querySelector('p.subheader > span.small > a');
		  return (node != null ? imageUrlResolver('https://musicbrainz.org' + node.pathname) : Promise.reject('no release group')).catch(function(reason) {
			return (node = response.document.querySelector('a.artwork-image')) != null ? node.href
				: (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 globalXHR(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 globalXHR(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 globalXHR(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+$)/, '_org'))
			.catch(reason => verifyImageUrl(img.src.replace(/_\d{3}(?=\.\w+$)/, '_max')))
			.catch(reason => img.src);
		});
	  case 'www.discogs.com': case 'discogs.com':
		return globalXHR(url).then(function(response) {
		  const dcOrigin = 'https://www.discogs.com';
		  let master = response.document.getElementById('all-versions-link');
		  return (master != null ?
			globalXHR(dcOrigin + master.pathname).then(response => getFromMeta(response.document) || notFound)
		  		: Promise.reject('no master')).catch(reason => getFromMeta(response.document) || notFound)
		  	.then(imgUrl => /^(?:https?):\/\/(?:img\.discogs\.com)\/.+\/(\S+?\.\w+)\b/i.test(imgUrl) ?
				dcOrigin + '/image/' + RegExp.$1 : imgUrl);
		});
	  case 'www.boomkat.com': case 'boomkat.com':
		if (!url.pathname.startsWith('/products/')) break;
		return globalXHR(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 globalXHR(url).then(function(response) {
		  var meta = getFromMeta(response.document);
		  return meta ? verifyImageUrl(meta.replace(/\/r\/[a-z]\//i, '/r/')).catch(reason => meta) : notFound;
		});
	  case 'www.soundcloud.com': case 'soundcloud.com':
		return globalXHR(url).then(function(response) {
		  var meta = getFromMeta(response.document);
		  return meta ? verifyImageUrl(meta.replace(/\bt\d+x\d+(?=\.\w+$)/, 'original')).catch(reason => meta) : notFound;
		});
	  case 'www.prestomusic.com': case 'prestomusic.com':
		if (!url.pathname.includes('/products/')) break;
		return globalXHR(url)
		  .then(response => verifyImageUrl(response.document.querySelector('div.c-product-block__aside > a').href.replace(/\?\d+$/)));
	  case 'www.bontonland.cz':case 'bontonland.cz':
		return globalXHR(url).then(response => response.document.querySelector('a.detailzoom').href);
	  case 'www.nativedsd.com':case 'nativedsd.com':
		if (!url.pathname.includes('/albums/')) break;
		return globalXHR(url).then(response => response.document.querySelector('a#album-cover').href);
	  case 'www.prostudiomasters.com': case 'prostudiomasters.com':
		if (!url.pathname.includes('/album/')) break;
		return globalXHR(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 globalXHR(url).then(function(response) {
		  var meta = getFromMeta(response.document);
		  return meta ? meta.replace(/\/s\d+\//, '/s0/') : notFound;
		})
	  case 'store.acousticsounds.com':
		return globalXHR(url).then(function(response) {
		  var link = response.document.querySelector('div#detail > link[rel="image_src"]');
		  return verifyImageUrl(link.href.replace(/\/medium\//i, '/xlarge/')).catch(reason => link.href);
		});
	  case 'www.indies.eu': case 'indies.eu':
		if (!url.pathname.includes('/alba/')) break;
		return globalXHR(url).then(response => verifyImageUrl)(response.document.querySelector('div.obrazekDetail > img').src);
	  case 'www.beatport.com': case 'classic.beatport.com': case 'beatport.com':
		if (!url.pathname.includes('/release/')) break;
		return globalXHR(url).then(function(response) {
		  var elem = getFromMeta(response.document);
		  return elem || ((elem = response.document.querySelector('div.artwork')) != null ?
			'https:' + elem.dataset.modalArtwork : notFound);
		}).then(imgUrl => imgUrl.replace(/\/image_size\/\d+x\d+\//i, '/image/'));
	  case 'www.supraphonline.cz': case 'supraphonline.cz':
		if (!url.pathname.includes('/album/')) break;
		return globalXHR(url).then(response => verifyImageUrl(response.document.querySelector('meta[itemprop="image"]')
			.content.replace(/\?.*$/, '')).catch(reason => notFound));
	  case 'vgmdb.net':
		if (!url.pathname.includes('/album/')) break;
		return globalXHR(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 globalXHR(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 globalXHR(url).then(function(response) {
		  var script = response.document.querySelector('script.light-data');
		  return verifyImageUrl(JSON.parse(script.text).image).catch(reason => notFound);
		});
	  case 'www.pias.com': case 'store.pias.com': case 'pias.com':
		return globalXHR(url).then(function(response) {
		  var node = getFromMeta(response.document);
		  if (node) return verifyImage(node.replace(/\/[sbl]\//i, '/')).catch(reason => node);
		  node = response.document.querySelector('img[itemprop="image"]');
		  return node != null ? verifyImage(node.src.replace(/\/[sbl]\//i, '/')).catch(reason => node.src) : notFound;
		});
	  case 'www.eclassical.com': case 'eclassical.com':
		return globalXHR(url).then(function(response) {
		  var a = response.document.querySelector('div#articleImage > a');
		  return a != null ? a.href : notFound;
		});
	  case 'www.hdtracks.com': case 'hdtracks.com':
		if (!/\/album\/(\w+)\b/.test(url)) break;
		return fetch('https://hdtracks.azurewebsites.net/api/v1/album/' + RegExp.$1).then(response => response.json())
		  .then(result => result.status.toLowerCase() == 'ok' ? result.cover : Promise.reject(result.status));
	  case 'www.muziekweb.nl': case 'muziekweb.nl':
		if (!/\/Link\/(\w+)\b/i.test(url)) break;
		return globalXHR(url).then(function(response) {
		  let meta = getFromMeta(response.document)
		  return meta ? meta.replace(/\/COVER\/\w+\b/i, '/COVER/SUPERLARGE') : notFound;
		});
	  case 'www.deejay.de': case 'deejay.de':
		return globalXHR(url).then(function(response) {
		  var elem = response.document.querySelector('div#gallery > a') || response.document.querySelector('div.cover a');
		  if (elem != null) return 'https://www.deejay.de' + elem.pathname;
		  return (elem = getFromMeta(response.document)) ? elem : notFound;
		}).then(imgUrl => verifyImageUrl(imgUrl.replace(/\/images\/\w+\//i, '/images/xxl/')).catch(() => imgUrl));
	  case 'music.163.com':
		if (!/\/album.*\b(?:id)=(\d+)\b/i.test(url.href)) break;
		return globalXHR('https://music.163.com/api/album/' + RegExp.$1, { responseType: 'json' })
			.then(response => response.response.album.picUrl ?
				response.response.album.picUrl.replace(/\b(?:p[123])(?=\.music\.\d+\.net\b)/i, 'p4') : notFound);
	  case 'www.tidal.com': case 'tidal.com':
		if (!(/\/album\/(\d+)(?:\/|$)/i.test(url.pathname) && !/\b(?:albumId)=(\d+)\b/i.test(url.search))) break;
		return globalXHR('https://api.tidal.com/v1/albums/' + RegExp.$1 + '?countrycode=US&token=_DSTon1kC8pABnTw', {
		  responseType: 'json',
		}).then(response => response.response.cover ? 'https://resources.tidal.com/images/' + response.response.cover.replace(/-/g, '/') + '/1280x1280.jpg' : notFound);
	  // movie-related
	  case 'www.imdb.com': case 'imdb.com':
		if (!['title/tt', 'name/nm'].some(cat => url.pathname.startsWith('/' + cat))) break;
		return globalXHR(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 ? globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(url).then(response => verifyImageUrl(response.document.querySelector('div.mojo-primary-image img').src));
	  case 'www.metacritic.com': case 'metacritic.com':
		return globalXHR(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 globalXHR(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 = [], 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 globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(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 globalXHR(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);
	  return getFromMeta(response.document) || 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);
}