// ==UserScript== // @name Image Host Helper // @namespace https://greasyfork.org/users/321857-anakunda // @version 1.91.6 // @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-23, Anakunda (https://greasyfork.org/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://dicmusic.com/* // @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?page=*&id=* // @match https://*/collages.php?action=edit&collageid=* // @match https://*/collages.php?action=comments&collageid=* // @match https://*/collages.php?action=new // @match https://*/collage.php?id=* // @match https://*/collage.php?page=*&id=* // @match https://*/collage.php?action=edit&collageid=* // @match https://*/collage.php?action=comments&collageid=* // @match https://*/collage.php?action=new // @match http*://tracker.czech-server.com/* // @connect * // @grant GM_xmlhttpRequest // @grant GM_getValue // @grant GM_setValue // @grant GM_deleteValue // @grant GM_info // @grant GM_registerMenuCommand // @require https://openuserjs.org/src/libs/Anakunda/xhrLib.min.js // @require https://openuserjs.org/src/libs/Anakunda/libCtxtMenu.min.js // @require https://openuserjs.org/src/libs/Anakunda/progressBars.min.js // @require https://openuserjs.org/src/libs/Anakunda/imageHostUploader.min.js // @downloadURL none // ==/UserScript== { 'use strict'; const previewDelay = GM_getValue('preview_delay', 12); const amEntityParser = /^(?:https?):\/\/(?:[\w\%\-]+\.)*apple\.com\/(?:\S+\/)?(album|artist|playlist)\/(?:[\w\%\-]+\/)?(?:id)?(\d+)\b/i; const itunesImageMax = [/\/(\d+x\d+)\w*\.(\w+)$/, '/100000x100000-999.' + (GM_getValue('apple_get_png_cover', false) ? 'png' : '$2')]; const dzrEntityParser = /^(?:https?):\/\/(?:[\w\%\-]+\.)*deezer\.com\/(?:\S+\/)?(album|artist|track|comment|playlist|radio|user)\/(\d+)\b/i; const dzrImageMax = GM_getValue('deezer_get_png_cover', false) ? [/\/(\d+x\d+)(?:\-\d+)*\.\w+$/, '/1400x1400.png'] : [/\/(\d+x\d+)(?:\-\d+)*(?=\.\w+$)/, '/1400x1400-000000-' + (parseInt(GM_getValue('deezer_jpeg_quality')) || 100) + '-0-0']; const discogsOrigin = 'https://www.discogs.com'; const discogsKey = GM_getValue('discogsKey', 'LWiNvIWBobGMRhfSCAiC'); const discogsSecret = GM_getValue('discogsSecret', 'HAQUKFmebpCSLyRNwjmSgOMgbnxsVQcp'); const lfmApiKey = GM_getValue('lfmApiKey', '920db0d2f86108f2fbe1917b53d63858'); function getDiscogsImageMax(imageUrl) { if (!httpParser.test(imageUrl)) return Promise.reject('Image URL is not valid'); if (imageUrl.endsWith('/images/spacer.gif')) return Promise.reject('Dummy image (placeholder)'); const matches = [ /^(?:https?):\/\/(?:(?:img|i)\.discogs\.com)\/.+\/([\w\%\-]+\.\w+)\b(?:\.\w+)*$/i, ].map(rx => rx.exec(imageUrl)); if (matches[0] != null) return verifyImageUrl(discogsOrigin + '/image/' + matches[0][1]).catch(reason => imageUrl); return Promise.resolve(imageUrl); } function getDeezerImageMax(imageUrl) { if (!httpParser.test(imageUrl)) return Promise.reject('invalid image URL'); const dzrImgResParser = /\/(\d+x\d+)(?:\-\d+)*\.(\w+)$/; let ext = dzrImgResParser.exec(imageUrl); if (ext != null) ext = GM_getValue('deezer_get_png_cover', false) ? 'png' : ext[2]; else { console.warn('Unscalable Deezer image, returning unchanged:', imageUrl); return Promise.resolve(imageUrl); } const urlByResolution = resolution => imageUrl.replace(dzrImgResParser, '/' + resolution + 'x' + resolution) + (/^j(?:pe?g|fif)$/i.test(ext) ? `-000000-${parseInt(GM_getValue('deezer_jpeg_quality')) || 100}-0-0.${ext}` : '.' + ext); const deezerHighestResolution = Math.max(parseInt(GM_getValue('deezer_highest_resolution')) || 1500, 1200); const defaultMax = (res = deezerHighestResolution) => verifyImageUrl(urlByResolution(res)).catch(reason => imageUrl); const resolutions = [/*1200, */1400, 1425, 1440, 1500, 1600, 1800, 1920].filter(size => size <= deezerHighestResolution); return Math.max(...resolutions) > 1400 ? Promise.all(resolutions.map(res => new Promise(function(resolve, reject) { const img = document.createElement('img'); img.onload = load => { resolve(load.target.naturalWidth * load.target.naturalHeight) }; img.onerror = (message, source, lineno, colno, error) => { reject(message) }; img.src = imageUrl.replace(dzrImgResParser, '/' + res + 'x' + res + '.png'); }).catch(reason => -Infinity))).then(function(pixTotals) { let maxArea = Math.max(...pixTotals); if (maxArea <= 0) { console.warn('Deezer: no max variant returns valid image', pixTotals, imageUrl); return Promise.reject('all size variants failed to load'); //defaultMax() } return urlByResolution(resolutions[pixTotals.indexOf(maxArea)]); }) : defaultMax(deezerHighestResolution); } function logFail(message, timeout = 30) { if (!message) return; let console = document.getElementById('ihh-console'); if (console == null) { console = document.createElement('DIV'); console.id = 'ihh-console'; console.style = ` position: fixed; bottom: 20px; right: 20px; width: fit-content; max-width: 66%; max-height: 66%; z-index: 10000001; padding: 10px; overflow-y: auto; overscroll-behavior-y: none; scrollbar-gutter: stable; border: solid lightsalmon 4px; color: #c00; background-color: antiquewhite; opacity: 1; font: 600 10pt "Segoe UI", Verdana, sans-serif; text-align: left; line-height: 1em; transition: opacity 1000ms linear; -webkit-transition: opacity 1000ms linear; `; document.body.append(console); } else if ('hTimer' in console) { if (console.hTimer) clearTimeout(console.hTimer); console.style.opacity = 1; } const entry = document.createElement('DIV'); entry.class = 'ihh-console-entry'; entry.style = 'display: block;'; if (console.childElementCount > 0) entry.style.marginTop = '3pt'; entry.textContent = message; console.append(entry); console.scrollTop = console.scrollHeight; if (timeout > 0) console.hTimer = setTimeout(function(elem) { elem.style.opacity = 0; elem.hTimer = setTimeout(elem => { elem.remove() }, Math.min(timeout, 1) * 1000, elem); }, Math.max(timeout - 1, 1) * 1000, console); } let cheveretoCustomHosts = GM_getValue('chevereto_custom_hosts'); if (cheveretoCustomHosts) { if (!Array.isArray(cheveretoCustomHosts)) try { cheveretoCustomHosts = JSON.parse(cheveretoCustomHosts); if (Array.isArray(cheveretoCustomHosts)) GM_setValue('chevereto_custom_hosts', cheveretoCustomHosts) } catch(e) { console.warnd(e) } if (Array.isArray(cheveretoCustomHosts)) for (let siteDef of cheveretoCustomHosts) if (siteDef.host_name && siteDef.alias) { const key = siteDef.alias.replace(nonWordStripper, '').toLowerCase(); imageHostHandlers[key] = 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, }); } else console.warn('Incomplete Chevereto custom site definition (excluded from chaining):', siteDef); else { console.warn('chevereto_custom_hosts invalid format (', cheveretoCustomHosts, ')'); //GM_deleteValue('chevereto_custom_hosts'); } } console.log('Image host handlers:', imageHostHandlers); PTPimg.prototype.setSession = function() { return this.apiKey ? Promise.resolve(this.apiKey) : globalXHR(this.origin).then(({document}) => { var apiKey = document.getElementById('api_key'); if (apiKey == null) { let counter = GM_getValue('ptpimg_reminder_read', 0); if (counter < 3) { alert(` ${this.alias} API key could not be captured. Please login to ${this.origin}/ and redo the action. If you don\'t have PTPimg account or don\'t want to use it, consider to remove PTPimg from 'upload_hosts' and 'rehost_hosts' storage entries, and all sites' local hostlists where does it appear. `); GM_setValue('ptpimg_reminder_read', ++counter); } return Promise.reject('API key not configured'); } else if (!(this.apiKey = apiKey.value)) return Promise.reject('assertion failed: empty PTPimg API key'); GM_setValue('ptpimg_api_key', this.apiKey); Promise.resolve(this.apiKey) .then(apiKey => { alert(`Your PTPimg API key [${apiKey}] was successfully configured`) }); return this.apiKey; }); } const generalImgHosts = [ 'ImgBB', 'PixHost', 'ImgBox', 'Slowpoke', 'PostImage', 'Gifyu', 'Ra', 'Abload', 'VgyMe', 'GeekPic', 'LightShot', 'ImgURL', 'Radikal', 'Z4A', 'PicaBox', 'PimpAndHost', 'SMMS', 'PomfCat', 'CasImages', 'CubeUpload', 'GooPics', 'ImageBan', 'UuploadIr', 'LinkPicture', 'Imgur', 'Catbox', 'ImageVenue', 'GetaPic', 'FastPic', 'SVGshare', ]; ['upload_hosts', 'rehost_hosts'].forEach(propName => { if (!GM_getValue(propName)) GM_setValue(propName, ['PTPimg'].concat(generalImgHosts).join(', ')) }); [ ['passthepopcorn.me', [ 'PTPimg', 'ImgBB', 'PixHost', 'ImgBox', 'Slowpoke', 'Gifyu', 'Ra', 'Abload', 'VgyMe', 'GeekPic', 'LightShot', 'ImgURL', 'Radikal', 'Z4A', 'PicaBox', 'PimpAndHost', 'SMMS', 'PomfCat', 'CasImages', 'CubeUpload', 'GooPics', 'ImageBan', 'UuploadIr', 'Catbox', 'ImageVenue', 'GetaPic', ]], ['notwhat.cd', ['NWCD']], ['forum.mobilism.org', ['Mobilism'].concat(generalImgHosts)], ['forum.mobilism.me', ['Mobilism'].concat(generalImgHosts)], ].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')); const queryAppleAPI = (endPoint, params, market = 'us') => endPoint ? (function() { const configValidator = config => config && config.MEDIA_API && config.MEDIA_API.token && (!config.timeStamp || config.timeStamp + 7 * 24 * 60*60*1000 >= Date.now() + 30 * 1000); if ('appleMusicDesktopConfig' in localStorage) try { var config = JSON.parse(localStorage.getItem('appleMusicDesktopConfig')); if (!configValidator(config)) throw 'Expired or incomplete cached Apple Music desktop environment'; console.info('Re-using cached Apple Music desktop environment:', config); return Promise.resolve(config); } catch(e) { console.info(e, localStorage.appleMusicDesktopConfig); localStorage.removeItem('appleMusicDesktopConfig'); } const timeStamp = Date.now(); return globalXHR('https://music.apple.com/').then(function({document}) { if ((config = document.head.querySelector('meta[name="desktop-music-app/config/environment"][content]')) != null) try { (config = JSON.parse(decodeURIComponent(config.content))).timeStamp = timeStamp; if (configValidator(config)) return config; } catch(e) { console.warn('Invalid Apple Music desktop environment format:', e, config.content) } if ((config = document.head.querySelector('script[type="module"][src]')) != null) return globalXHR(new URL(config.getAttribute('src'), 'https://music.apple.com'), { responseType: 'text' }).then(({responseText}) => (config = /\b(?:const\s+kd\s*=\s*['"]([^\s'"]{64,}?)|\w+\s*=\s*['"]([^\s'"]{268}))['"]/.exec(responseText)) != null && configValidator(config = { MEDIA_API : { token: config[1] || config[2] }, timeStamp: timeStamp, }) ? config : Promise.reject('Missing Apple Music OAuth2 token')); return Promise.reject('Missing Apple Music OAuth2 token'); }).then(function(config) { console.info('Apple Music OAuth2 token successfully extracted:', config.MEDIA_API.token); localStorage.setItem('appleMusicDesktopConfig', JSON.stringify(config)); return config; }); })().then(function request(config) { if (!config.retryCounter) config.retryCounter = 0; let url = config.MUSIC && config.MUSIC.BASE_URL || 'https://amp-api.music.apple.com/v1'; url = new URL(`${url}/catalog/${market || 'us'}/${endPoint.replace(/^\/+|\/+$/g, '')}`); if (params) url.search = new URLSearchParams(params); url.searchParams.set('omit[resource]', 'relationships,views,meta,autos'); url.searchParams.set('l', config.i18n && config.i18n.defaultLocale || 'en-us'); url.searchParams.set('platform', 'web'); return globalXHR(url, { responseType: 'json', headers: { Referer: 'https://music.apple.com/', Origin: 'https://music.apple.com', Host: url.hostname, Authorization: 'Bearer ' + config.MEDIA_API.token, }, }).then(({response}) => response, function(reason) { let status = /^HTTP error (\d+)\b/.exec(reason); if (status != null) status = parseInt(status[1]); if ([400, 401, 403].includes(status)) { localStorage.removeItem('appleMusicDesktopConfig'); if (config.retryCounter++ <= 0) return request(config); alert('Apple Music request problem:\n' + reason + '\n(retry with new token)'); //return queryAppleAPI(endPoint, params); } return Promise.reject(reason); }); }) : Promise.reject('Endpoint is missing'); const tidalAccess = { apiBase: 'https://api.tidal.com/v1', clientId: GM_getValue('tidal_clientid', localStorage.getItem('tidalClientId') || 'zU4XHVVkc2tDPo4t' || '7m7Ap0JC9j1cOM3n'), clientSecret: GM_getValue('tidal_clientsecret', localStorage.getItem('tidalClientSecret') || 'VJKhDFqJPqvsPVNBV6ukXTJmwlvbttP7wlMlrc72se4=' || 'vRAdA108tlvkJpTsGZS8rGZ7xTlbJ0qaZ2K9saEzsgY='), auth: null, authorize: function() { const oAuth2base = 'https://auth.tidal.com/v1/oauth2', devAuthEndpoint = oAuth2base + '/device_authorization', tokenEndpoint = oAuth2base + '/token', scopes = ['r_usr', 'w_usr', 'w_sub'], oAuth2timeReserve = 30; // reserve this time (s) for upcoming authorized request const isTokenValid = accessToken => typeof accessToken == 'object' && accessToken.token_type && accessToken.access_token && accessToken.expires_at >= Date.now() + oAuth2timeReserve * 1000; const isSessionValid = session => session && typeof session == 'object' && session.userId > 0 && session.sessionId; const authMethods = { 'SessionId': () => Promise.reject('Method removed'), 'DeviceToken': () => Promise.reject('Method disabled'), //Promise.resolve([undefined, { 'token': this.clientId }]), 'OAuth2': () => (function() { if ('tidalAccessToken' in localStorage) try { var accessToken = JSON.parse(localStorage.getItem('tidalAccessToken')); if (isTokenValid(accessToken)) return Promise.resolve(accessToken); } catch(e) { localStorage.removeItem('tidalAccessToken') } if (!this.clientId || !this.clientSecret) return Promise.reject('Tidal credentials not configured (OAuth2-deviceFlow)'); let timeStamp; return (accessToken && accessToken.refresh_token ? (function() { timeStamp = Date.now(); return globalXHR(tokenEndpoint, { responseType: 'json' }, new URLSearchParams({ grant_type: 'refresh_token', refresh_token: accessToken.refresh_token, client_id: this.clientId, client_secret: this.clientSecret, })).then(({response}) => { if (!response.refresh_token) response.refresh_token = accessToken.refresh_token; return response; }); }).call(this) : Promise.reject('Cached token not available')).then(response => { if (typeof response != 'object') throw 'invalid response'; console.assert(timeStamp > 0, 'timeStamp > 0'); const tzOffset = new Date().getTimezoneOffset() * 60 * 1000; accessToken = response; if (!accessToken.timestamp) accessToken.timestamp = timeStamp; if (!accessToken.expires_at) accessToken.expires_at = accessToken.timestamp + (accessToken.expires_in_ms || accessToken.expires_in * 1000); if (!isTokenValid(accessToken)) { console.warn('Ivalid Tidal token received:', accessToken); return Promise.reject('invalid token received'); } localStorage.setItem('tidalAccessToken', JSON.stringify(accessToken)); return accessToken; }); }).call(this).then(accessToken => [{ Authorization: `${accessToken.token_type} ${accessToken.access_token}` }]), }; const authSequence = [/*'SessionId', 'DeviceToken'*/]; authSequence['tidalAccessToken' in localStorage ? 'unshift' : 'push']('OAuth2'); return (this.auth || (this.auth = (function tidalAuth(index = 0) { const method = authMethods[authSequence[index]]; if (typeof method == 'function') return method.call(this).catch(reason => { console.warn('Tidal ' + authSequence[index] + ' auth metod failed:', reason); return tidalAuth.call(this, index + 1); }); //this.auth = null; localStorage.setItem('tidalLoginSuccess', false); return Promise.reject('all auth methods failed'); }).call(this))); }, requestAPI: function(endPoint, params, countryCode = 'US') { if (!endPoint) return Promise.reject('No API endpoint'); const weakRequest = /^(?:search)\//i.test(endPoint); return (function apiCall() { return this.authorize().then(credentials => { if ('tidalLoginSuccess' in localStorage) localStorage.removeItem('tidalLoginSuccess'); setTimeout(() => { this.auth = null }, 5000); return globalXHR(this.apiBase + '/' + endPoint + '?' + new URLSearchParams(Object.assign({ }, params || { }, { deviceType: 'BROWSER', locale: 'en_US', countryCode: countryCode, }, credentials[1] || { })).toString(), { responseType: 'json', headers: credentials[0], }).then(({response}) => response, reason => { if (!/^(?:HTTP error (401))\b/i.test(reason) || !('tidalAccessToken' in localStorage)) return Promise.reject(reason); localStorage.removeItem('tidalAccessToken'); if (weakRequest) return Promise.reject(reason); this.auth = null; return apiCall.call(this); }); }); }).call(this); }, }; const mixcloudQuery = (query, variables) => ('mixcloudCsrfToken' in sessionStorage ? Promise.resolve(sessionStorage.getItem('mixcloudCsrfToken')) : globalXHR('https://www.mixcloud.com/', { method: 'HEAD' }).then(function(response) { let csrfToken = /^set-cookie:.*\b(?:csrftoken)\s*=\s*(\w+)\b/im.exec(response.responseHeaders); if (csrfToken != null) csrfToken = csrfToken[1]; else return Promise.reject('No CSRF token returned'); sessionStorage.setItem('mixcloudCsrfToken', csrfToken); return csrfToken; })).then(csrfToken => globalXHR('https://www.mixcloud.com/graphql', { responseType: 'json', headers: { 'X-CSRFToken': csrfToken }, }, { query: query || { }, variables: variables || { } })).then(({response}) => response.data); const getAmazonCfg = (url = 'https://www.amazon.com/') => globalXHR(url = new URL(url), { headers: { 'User-Agent': navigator.userAgent } }).then(function(response) { let preConnect = response.document.head.querySelector('link[rel="preconnect"]'); //preConnect = 'https://na.web.skill.music.a2z.com/' if (preConnect != null) preConnect = preConnect.href; else throw 'Assertion failed: preConnect != null'; for (var appConfig of response.document.head.getElementsByTagName('SCRIPT')) if ((appConfig = /^\s*(?:window\.amznMusic)\s*=\s*(\{[\S\s]+\});\s*$/.exec(appConfig.text)) != null) try { appConfig = eval('(' + appConfig[1] + ')').appConfig; break; } catch (e) { console.warn(e) } if (!appConfig) throw 'Assertion failed: amznMusic != null'; sessionStorage.setItem('amznAppConfig', JSON.stringify(appConfig)); return { urlBase: preConnect, headers: { //'User-Agent': UA, 'Referer': url.href, 'x-amzn-authentication': JSON.stringify({ interface: 'ClientAuthenticationInterface.v1_0.ClientTokenElement', accessToken: appConfig.accessToken, }), 'x-amzn-request-id': 'f4a75e51-e7ef-4080-986a-4041738b1198', 'x-amzn-session-id': appConfig.sessionId, 'x-amzn-timestamp': Date.now(), 'x-amzn-page-url': url.href, 'x-amzn-csrf': JSON.stringify({ interface: 'CSRFInterface.v1_0.CSRFHeaderElement', token: appConfig.csrf.token, timestamp: appConfig.csrf.ts, rndNonce: appConfig.csrf.rnd, }), 'x-amzn-application-version': appConfig.version, 'x-amzn-currency-of-preference': 'USD' || appConfig.currencyOfPreference, 'x-amzn-device-family': 'RetailWebPlayer.web', 'x-amzn-device-model': 'WEBPLAYER', 'x-amzn-device-type': appConfig.deviceType, 'x-amzn-device-id': appConfig.deviceId, 'x-amzn-device-language': 'en_US' || appConfig.displayLanguage, 'x-amzn-device-time-zone': 'Etc/UTC', 'x-amzn-os-version': '1.0', 'x-amzn-device-width': 1920, 'x-amzn-device-height': 1080, 'x-amzn-user-agent': navigator.userAgent, 'x-amzn-affiliate-tags': '', 'x-amzn-ref-marker': '', 'x-amzn-music-domain': url.hostname, 'x-amzn-referer': url.href, 'x-amzn-page-url': url.href, 'x-amzn-weblab-id-overrides': '', 'x-amzn-video-player-token': '', 'x-amzn-feature-flags': 'hd-supported', }, }; return Promise.reject('Config could not be extracted'); }); function imagePreview(imgUrl, size) { if (previewDelay <= 0) return; let div = document.getElementById('image-preview'); if (div != null) document.body.removeChild(div); if (!httpParser.test(imgUrl)) return; div = document.createElement('div'); div.id = 'image-preview'; div.style = 'position: fixed; bottom: 20px; right: 20px; border: thin solid silver; ' + 'background-color: #8888; padding: 10px; opacity: 0; transition: opacity 1s ease-in-out; z-index: 999999999;'; const cleanUp = function(div) { if (div.parentNode == null) return; div.style.opacity = 0; setTimeout(div => { document.body.removeChild(div) }, 1000, div); }; div.ondblclick = evt => { cleanUp(evt.currentTarget) }; let img = document.createElement('img'); img.style = 'width: 225px;'; img.onload = function(evt) { if (evt.currentTarget.parentNode.parentNode == null) document.body.append(evt.currentTarget.parentNode); setTimeout(div => { div.style.opacity = 1 }, 0, evt.currentTarget.parentNode); setTimeout(cleanUp, (previewDelay || 12) * 1000, evt.currentTarget.parentNode); if (!evt.currentTarget.naturalWidth || !evt.currentTarget.naturalHeight) return; // invalid image let info = document.createElement('div'); info.id = 'image-info'; info.style = 'text-align: center; background-color: #29434b; padding: 5px; color: white;' + 'font: 500 10pt "Segoe UI", Verdana, sans-serif;'; evt.currentTarget.parentNode.append(info); const resolution = evt.currentTarget.naturalWidth + '×' + evt.currentTarget.naturalHeight; (size > 0 ? Promise.resolve(size) : size instanceof Promise ? size : getRemoteFileSize(imgUrl)).then(function(size) { if (!(size >= 0)) throw 'invalid size'; let imageSizeLimit = GM_getValue('image_size_reduce_threshold'), html = resolution + ' ( 0 && size > imageSizeLimit * 2**10) html += ' style="color: red;"'; html += '>' + formattedSize(size) + ')'; info.innerHTML = html; }).catch(reason => { info.textContent = resolution }); }; img.onerror = evt => { console.warn('Image source couldnot be loaded:', evt, imgUrl) }; img.src = imgUrl; div.append(img); } function checkImageSize(image, elem = null, param) { let imageSizeLimit = GM_getValue('image_size_reduce_threshold'); if (!(imageSizeLimit > 0)) return Promise.resolve(image); if (!(elem instanceof HTMLElement)) elem = null; if (elem != null) elem.disabled = true; return (image instanceof File ? Promise.resolve(image.size) : param > 0 ? Promise.resolve(param) : param instanceof Promise ? param : getRemoteFileSize(image)).then(function(size) { if (size <= imageSizeLimit * 2**10) return image; const haveRhHosts = Array.isArray(imageHosts.rhHostChain) && imageHosts.rhHostChain.length > 0; if (!haveRhHosts && !GM_getValue('force_reduce', true)) return Promise.reject('no hosts to upload result'); return reduceImageSize(image, GM_getValue('image_reduce_maxheight', 2160), GM_getValue('image_reduce_jpegquality', 90), typeof param == 'function' ? param : null).then(function(output) { if (elem != null) { elem.value = output.uri; if (image instanceof File) imagePreview(output.uri, output.size); } 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 haveRhHosts ? output.uri : forcedRehost(output.uri); }); }).catch(function(reason) { logFail('failed to get remote image size or optimize the image: ' + reason + ' (size reduction was not performed)'); return image; }).then(function(finalResult) { if (elem != null) { if (httpParser.test(finalResult)) { if (finalResult != elem.value) elem.value = finalResult; } else elem.value = ''; elem.disabled = false; } return finalResult; }); } // Export public API unsafeWindow.imageHostHelper = { }; const defEndpoint = (publicName, localRef) => { unsafeWindow.imageHostHelper[publicName] = localRef }; defEndpoint('uploadFiles', function uploadImages(files, checkSize = true, preview = false) { if (files instanceof Blob) files = [files]; if (!Array.isArray(files)) return Promise.reject('Invalid parameter (files)'); if ((files = files.filter(file => file instanceof File && file.type.startsWith('image/'))).length <= 0) return Promise.reject('Invalid parameter (no valid images passed)'); console.time('Image uploader'); return checkSize || preview ? Promise.all(files.map(file => (checkSize ? checkImageSize(file).catch(function(reason) { logFail('Downsizing of source image not possible (' + reason + '), uploading original size'); return file; }) : Promise.resolve(file)).then(function(result) { const uploader = file => imageHosts.uploadImages([file]).then(singleImageGetter).then(function(imageUrl) { if (preview) imagePreview(imageUrl, file.size); return imageUrl; }); if (httpParser.test(result)) return imageHosts.rehostImages([result]).catch(function(reason) { logFail('Downsizing of source image failed (' + reason + '), uploading original size'); return uploader(file); }); if (result instanceof File) return uploader(result); console.warn('invalid checkImageSize(...) result:', result); return Promise.reject('invalid checkImageSize(...) result'); }))) : imageHosts.uploadImages(files); }); defEndpoint('rehostImageLinks', function rehostImageLinks(urls, checkSize = true, preview = false, enforceRehost = false, modifiers) { if (typeof urls == 'string' && httpParser.test(urls)) urls = [urls]; if (!Array.isArray(urls) || urls.length <= 0) return Promise.reject('Invalid parameter (urls)'); console.time('Image URL rehoster'); return Promise.all(urls.map(url => imageUrlResolver(url, { altKey: Boolean(typeof modifiers == 'object' && modifiers.altKey), ctrlKey: Boolean(typeof modifiers == 'object' && modifiers.ctrlKey), shiftKey: Boolean(typeof modifiers == 'object' && modifiers.shiftKey), }).then(verifyImageUrl).then(function(imageUrl) { if (!checkSize) return imageUrl; const size = getRemoteFileSize(imageUrl); if (preview) imagePreview(imageUrl, size); return checkImageSize(imageUrl, null, size); }))).then(imageUrls => imageHosts.rehostImages(imageUrls).then(function(rehostedImages) { console.timeEnd('Image URL rehoster'); return rehostedImages; }, reason => enforceRehost ? Promise.reject(reason) : Promise.resolve(imageUrls))); }); defEndpoint('imageHostHandlers', imageHostHandlers); defEndpoint('uploadImages', ImageHostManager.prototype.uploadImages.bind(imageHosts)); defEndpoint('rehostImages', ImageHostManager.prototype.rehostImages.bind(imageHosts)); defEndpoint('logFail', logFail); defEndpoint('getDeezerImageMax', getDeezerImageMax); defEndpoint('getDiscogsImageMax', getDiscogsImageMax); defEndpoint('dzrImageMax', dzrImageMax); defEndpoint('itunesImageMax', itunesImageMax); defEndpoint('urlResolver', urlResolver); defEndpoint('verifyImageUrl', verifyImageUrl); defEndpoint('getRemoteFileType', getRemoteFileType); defEndpoint('getRemoteFileSize', getRemoteFileSize); defEndpoint('imageUrlResolver', imageUrlResolver); defEndpoint('checkImageSize', checkImageSize); defEndpoint('reduceImageSize', reduceImageSize); defEndpoint('optiPNG', optiPNG); defEndpoint('directLinkGetter', directLinkGetter); defEndpoint('singleImageGetter', singleImageGetter); unsafeWindow.dispatchEvent(Object.assign(new Event('imageHostHelper'), { data: unsafeWindow.imageHostHelper })); // const meta = document.createElement('META'); // meta.name = 'ImageHostHelper'; // meta.content = 'All endpoints exported'; // meta.setAttribute('propertyname', 'imageHostHelper'); // document.head.append(meta); function imageUrlResolver(url, modifiers = { }) { return urlResolver(url).then(url => verifyImageUrl(url).catch(function(reason) { if (/^HTTP error (\d+)\b/.test(reason) && [ 401, 402, 404, 407, 408, 410, 451, 502, 503, 504, 511, ].includes(parseInt(RegExp.$1)) || /\b(?:timeout|timed out)\b/.test(reason)) return Promise.reject(reason); const notFound = Promise.reject('No title image for this URL'); function getFromMeta(root) { let 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 && httpParser.test(meta.content) ? meta.content : undefined; } try { url = new URL(url) } catch(e) { return Promise.reject(e) } let entryIds; 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({document}) { let ref = document.querySelector('div#tralbumArt > a.popupImage'); ref = ref != null ? ref.href : getFromMeta(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({document}) { let img = document.querySelector('img[itemprop="image"]'); return img != null ? img.src : notFound; }); else if (url.hostname.endsWith('geekpic.net')) return globalXHR(url).then(function({document}) { let a = document.querySelector('div.img-upload > a.mb'); return a != null ? a.href : notFound; }); else if (url.hostname.endsWith('qq.com') && /\/album(?:Detail)?\/(\w+)/i.test(url.pathname)) return globalXHR(url).then(function({document}) { for (let script of document.body.querySelectorAll(':scope > script')) if ((script = /\b__INITIAL_DATA__\s*=\s*({.+})/.exec(script.text)) != null) try { var initialData = eval('(' + script[1] + ')') } catch(e) { console.warn(e) } if (!initialData) throw 'Assertion failed: __INITIAL_DATA__ not triggered'; if (initialData = initialData.detail.picurl) { if (!httpParser.test(initialData)) initialData = url.protocol + initialData; return initialData.replace(/\/(T\d+)?(R\d+x\d+)?(M\w+?)(_\d+)?\.(\w+(?:\.\w+)*)(\?.*)?$/, '/$1$3.$5'); } else return notFound; }); else if (url.hostname.startsWith('books.google.') && url.pathname.startsWith('/books')) return globalXHR(url).then(function({document}) { let meta = getFromMeta(document); return meta != null ? meta.replace(/\b(?:zoom=1)\b/, 'zoom=0') : notFound; }); else if (/^(?:\w+\.)?amazon(?:\.\w+)+$/.test(url.hostname)) return getAmazonCfg(url).then(function(amazonCfg) { return globalXHR(amazonCfg.urlBase + 'api/showHome', { responseType: 'json', headers: amazonCfg.headers }, { deeplink: JSON.stringify({ interface: 'DeeplinkInterface.v1_0.DeeplinkClientInformation', deeplink: '/' + url.pathname.split('/').filter(Boolean).slice(-2).join('/'), }), }).then(function({response}) { const method = response.methods.find(method => method.interface.endsWith('.CreateAndBindTemplateMethod')); return method && method.template && method.template.headerImage || notFound; }); }).catch(reason => globalXHR(url).then(function(response) { const getFullImage = imageUrl => httpParser.test(imageUrl) && (imageUrl = imageUrl.replace(/\._\w+(?:_\w+)*_\./, '.'), !['31CTP6oiIBL.jpg', '31zMd62JpyL.jpg'] .some(path => imageUrl.endsWith('/images/I/' + path))) ? imageUrl : Promise.reject('Dummy image (placeholder)'); const getImgOrigin = colorImage => getFullImage(colorImage.hiRes || colorImage.large || colorImage.thumb); 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 Promise.all(variants.map(key => Promise.all(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 Promise.all(colorImages.initial.map(getImgOrigin)); } let img = ['div#ppd-left img', 'img#igImage', 'img#imgBlkFront'] .reduce((acc, sel) => acc || response.document.querySelector(sel), null); if (img == null) return notFound; if (img.dataset.aDynamicImage) try { let imgUrl = Object.keys(JSON.parse(img.dataset.aDynamicImage))[0]; if (httpParser.test(imgUrl)) return getFullImage(imgUrl); } catch(e) { } return getFullImage(img.src); })); else switch (url.hostname) { // general image hostings case 'www.imgur.com': case 'imgur.com': return (entryIds = /^\/(?:(a)\/)?(\w+)\b/.exec(url.pathname)) != null ? imageHostHandlers.imgur.setSession().then(clientId => globalXHR(`https://api.imgur.com/post/v1/${entryIds[1] == 'a' ? 'albums' : 'media'}/${entryIds[2]}?${new URLSearchParams({ client_id: clientId, include: 'media', }).toString()}`, { responseType: 'json' }).then(({response}) => response.media.map(media => media.url))).catch(reason => globalXHR(url, { responseType: 'text' }).then(function({responseText}) { let image = /^\s*(?:image)\s*:\s*(\{.+\}),\s*$/m.exec(responseText); if (image != null) try { return JSON.parse(image[1]).album_images.images.map(image => 'https://i.imgur.com/' + image.hash + image.ext); } catch(e) { console.warn(e) } return notFound; })) : globalXHR(url).then(function({document}) { let link = document.querySelector('link[rel="image_src"]'); return link != null ? link.href : notFound; }); case 'pixhost.to': if (url.pathname.startsWith('/gallery/')) return globalXHR(url).then(({document}) => Promise.all(Array.from(document.querySelectorAll('div.images > a')).map(a => imageUrlResolver(a.href, modifiers)))); if (url.pathname.startsWith('/show/')) return globalXHR(url) .then(({document}) => document.querySelector('img#image').src); break; case 'malzo.com': if (url.pathname.startsWith('/al/')) return imageHostHandlers.malzo.galleryResolver(url); else break; case 'imgbb.com': case 'ibb.co': if (url.pathname.startsWith('/album/')) return imageHostHandlers.imgbb.galleryResolver(url); else break; case 'jerking.empornium.ph': if (url.pathname.startsWith('/album/')) return imageHostHandlers.jerking.galleryResolver(url); else break; case 'imgbox.com': if (url.pathname.startsWith('/g/')) return globalXHR(url).then(({document}) => Promise.all(Array.from(document.querySelectorAll('div#gallery-view-content > a')) .map(a => imageUrlResolver('https://imgbox.com' + a.pathname, modifiers)))); break; case 'postimage.org': case 'postimg.cc': if (url.pathname.startsWith('/gallery/')) return PostImage.resultsHandler(url).then(results => results.map(result => result.original)); return globalXHR(url).then(function({document}) { const elem = document.body.querySelector('a#download'); return elem != null ? elem.href : getFromMeta(document.head) || notFound; }); case 'www.imagevenue.com': case 'imagevenue.com': return globalXHR(url, { headers: { Referer: 'http://www.imagevenue.com/' } }).then(function({document}) { let images = Array.from(document.querySelectorAll('div.card img')).map(function(img) { return img.src.includes('://cdn-images') ? Promise.resolve(img.src) : imageUrlResolver(img.parentNode.href, modifiers); }); return images.length > 1 ? Promise.all(images) : images.length == 1 ? images[0] : notFound; }); case 'www.imageshack.us': case 'imageshack.us': return globalXHR(url).then(({document}) => document.querySelector('a#share-dl').href); case 'www.flickr.com': case 'flickr.com': if (url.pathname.startsWith('/photos/')) return globalXHR(url, { responseType: 'text' }).then(function({responseText}) { if (/\b(?:modelExport)\s*:\s*(\{.+\}),/.test(responseText)) try { let urls = JSON.parse(RegExp.$1).main['photo-models'].map(function(photoModel) { let 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; }); else break; 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({responseText}) { if (!/\b(?:App\.CuratorId)\s*=\s*"(\d+)"/.test(responseText)) return Promise.reject('Unexpected page structure'); return _500pxUrlHandler('users/' + RegExp.$1 + '/galleries/' + galleryId + '/items?sort=position&sort_direction=asc&rpp=999'); }); } else break; case 'www.pxhere.com': case 'pxhere.com': if (url.pathname.includes('/photo/')) return globalXHR(url).then(({document}) => JSON.parse(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(({document}) => 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({responseText}) { if (/^(?:RVAS)\s*=\s*(\{[\S\s]+?\})$/m.test(responseText)) try { let 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'); }); case 'www.freeimages.com': case 'freeimages.com': if (url.pathname.startsWith('/photo/')) return globalXHR(url).then(function({document}) { let types = Array.from(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; }); else break; case 'redacted.ch': if (url.pathname == '/image.php') return globalXHR(url, { method: 'HEAD' }).then(response => response.finalUrl); else break; case 'demo.cloudimg.io': { if (!/\b(https?:\/\/\S+)$/.test(url.pathname.concat(url.search, url.hash))) break; let resolved = RegExp.$1; if (/\b(?:https?):\/\/(?:\w+\.)*discogs\.com\//i.test(resolved)) break; return imageUrlResolver(resolved, modifiers); } case 'www.pimpandhost.com': case 'pimpandhost.com': if (url.pathname.startsWith('/image/')) return globalXHR(url).then(function(response) { let 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; }); else break; case 'www.screencast.com': case 'screencast.com': return globalXHR(url).then(function({document}) { let ref = document.querySelectorAll('ul#containerContent > li a.media-link'); if (ref.length <= 0) return getFromMeta(document) || notFound; return Promise.all(Array.from(ref).map(a => imageUrlResolver('https://www.screencast.com' + a.href, modifiers))); }); case 'abload.de': if (url.pathname.startsWith('/image.php')) return globalXHR(url).then(function({document}) { let elem = document.querySelector('img#image'); if (elem == null) return notFound; let src = new URL(elem.src); return imageHostHandlers.abload.origin + src.pathname + src.search; }); else break; case 'fastpic.ru': if (url.pathname.startsWith('/view/')) return globalXHR(url).then(({document}) => imageUrlResolver(document.querySelector('a.img-a').href, modifiers)); else if (url.pathname.startsWith('/fullview/')) return globalXHR(url).then(function(response) { let 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; }); else break; case 'www.radikal.ru': case 'radikal.ru': case 'a.radikal.ru': return globalXHR(url).then(({document}) => document.querySelector('div.mainBlock img').src); case 'imageban.ru': case 'ibn.im': return globalXHR(url).then(({document}) => document.querySelector('a[download]').href); case 'svgshare.com': return globalXHR(url).then(function({document}) { let link; 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/')) return globalXHR(url).then(function({document}) { let nodes = 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 = 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(({document}) => Array.from(document.querySelectorAll('div#preload-images > img')).map(img => img.src)))); return notFound; }); else break; case 'www.casimages.com': case 'casimages.com': if (url.pathname.startsWith('/i/')) return globalXHR(url).then(function({document}) { let elem = document.querySelector('div.logo > a'); if (elem != null) return elem.href; elem = document.querySelector('div.logo img'); return elem != null ? elem.src : notFound; }); else break; case 'www.getapic.me': case 'getapic.me': return globalXHR(url, { responseType: 'json' }).then(function({response}) { if (!response.result.success) return Promise.reject(response.result.errors); if (Array.isArray(response.result.data.images)) return response.result.data.images.map(image => image.url); return response.result.data.image ? response.result.data.image.url : notFound; }); case 'sm.ms': if (url.pathname.startsWith('/image/')) return globalXHR(url).then(function({document}) { let img = document.querySelector('img.image'); return img != null ? img.src || img.parentElement.href : notFound; }); else break; case 'www.kizunaai.com': case 'kizunaai.com': //if (!url.pathname.includes('/music/')) break; return globalXHR(url).then(function({document}) { let img = document.querySelector('div.post-body span > img'); return img != null ? img.src.replace(/-\d+x\d+(?=\.\w+$)/, '') : notFound; }); case 'play.google.com': if (url.pathname.startsWith('/store/')) return globalXHR(url).then(function({document}) { let meta = getFromMeta(document); return meta != null ? meta.replace(/(?:=[swh]\d+.*)?$/, '=s0') : notFound; }); else break; // music-related case 'www.discogs.com': case 'discogs.com': return globalXHR(url).then(({document}) => (function() { if (url.pathname.includes('/master/')) return Promise.reject('This is master'); if (modifiers.ctrlKey) return Promise.reject('master release inquiry avoided (force release gallery)'); let master = document.body.querySelector('section#release-actions a.link_1ctor[href^="/master/"]'); if (master == null) return Promise.reject('no master release for this page'); return imageUrlResolver(discogsOrigin + master.pathname, modifiers); })().catch(function(reason) { let elem = document.querySelector('div.image_gallery, div.image_gallery_large'); if (elem != null) try { elem = JSON.parse(elem.dataset.images).map(image => image.full || image.thumb) .filter(RegExp.prototype.test.bind(httpParser)); if (elem.length <= 0) throw 'empty imagem list'; return Promise.all(elem.map(getDiscogsImageMax)).catch(function(reason) { console.error('One of getDiscogsImageMax workers rejected:', reason, elem); return elem; }); } catch(e) { console.warn('Invalid Discogs image gallery:', elem, '(' + e + ')') } else { console.warn('Missing Discogs image gallery record for', url.href); } const ids = /\/(artist|master|release|label|user)s?\/(?:view\/)?(\d+)\b/i.exec(url.pathname); if (ids == null) return Promise.reject('Unsupported entity'); let sha256Hashes; if ('discogsGraphqlHashes' in localStorage) try { sha256Hashes = JSON.parse(localStorage.getItem('discogsGraphqlHashes')); } catch(e) { console.warn(e) } if (!sha256Hashes || typeof sha256Hashes != 'object' || !(sha256Hashes.timeStamp > 0) || Date.now() >= sha256Hashes.timeStamp + 24 * 60 * 60 * 1000) sha256Hashes = null; return (sha256Hashes ? Promise.resolve(sha256Hashes) : (function updateHHashes() { const script = document.querySelector('script[data-chunk="main"][src^="https://catalog-assets.discogs.com/main."]'); return script != null ? globalXHR(script.src, { responseType: 'text' }).then(function({responseText}) { let hashes = /\bJSON\.parse\s*\(\s*'(\{\s*"\w+Data".+?)'\);/.exec(responseText); if (hashes != null) hashes = Object.assign(JSON.parse(hashes[1]), { timeStamp: Date.now() }); else throw 'Script pattern wasnot located'; localStorage.setItem('discogsGraphqlHashes', JSON.stringify(hashes)); return hashes; }) : Promise.reject('Unexpected document structure'); })()).then(function(sha256Hashes) { const reflectUrl = new URL(discogsOrigin + '/internal/release-page/api/graphql'); switch(ids[1].toLowerCase()) { case 'artist': reflectUrl.searchParams.set('operationName', 'ArtistAllImages'); break; case 'master': reflectUrl.searchParams.set('operationName', 'MasterReleaseAllImages'); break; case 'release': reflectUrl.searchParams.set('operationName', 'ReleaseAllImages'); break; } reflectUrl.searchParams.set('variables', JSON.stringify({ discogsId: parseInt(ids[2]) , count: 500 })); reflectUrl.searchParams.set('extensions', JSON.stringify({ persistedQuery: { version: 1, sha256Hash: sha256Hashes[reflectUrl.searchParams.get('operationName')], } })); return globalXHR(reflectUrl, { responseType: 'json' }).then(function({response}) { switch(ids[1].toLowerCase()) { case 'artist': var root = response.data.artist; break; case 'master': root = response.data.masterRelease.keyRelease; break; case 'release': root = response.data.release; break; } return root.images.totalCount > 0 ? root.images.edges.map(edge => edge.node.fullsize.sourceUrl) : notFound; }).catch(reason => (elem = getFromMeta(document)) ? getDiscogsImageMax(elem) : notFound); }); })); case 'www.musicbrainz.org': case 'beta.musicbrainz.org': case 'musicbrainz.org': if (url.pathname.startsWith('/release/')) { if (/^\/release\/([\w\-]+)(?=\/|$)/i.test(url.pathname)) url.pathname = '/release/' + RegExp.$1 + '/cover-art'; else console.warn('Unexpected MusicBrainz release url path:', url.pathname); } else if (!url.pathname.startsWith('/release-group/')) break; return globalXHR(url).then(({document}) => (function() { if (url.pathname.startsWith('/release-group/')) return Promise.reject('this is release group'); if (modifiers.ctrlKey) return Promise.reject('release group inquiry avoided (force release gallery)'); let releaseGroup = document.querySelector('p.subheader > span.small > a'); if (releaseGroup == null) return Promise.reject('no release group for this page'); return imageUrlResolver('https://musicbrainz.org' + releaseGroup.pathname, modifiers); })().catch(function(reason) { let elem = document.querySelector('head > script[type="application/ld+json"]'); if (elem != null) try { if (Array.isArray(elem = JSON.parse(elem.text).image)) { if (elem.length > 0) return elem.map(image => 'https:' + image.contentUrl); } else if (elem && elem.contentUrl) return 'https:' + elem.contentUrl; } catch(e) { console.warn('MusicBrainz: invalid meta record', elem) } elem = document.querySelectorAll('div#content > div.artwork-cont span.cover-art-image > img'); if (elem.length > 0) return Array.from(elem).map(img => img.src.replace(/-\d+(?=(?:\.\w+)+$)/, '')); return (elem = document.querySelector('a.artwork-image')) != null ? elem.href : (elem = document.querySelector('div.cover-art > img')) != null ? elem.src : notFound; })); case 'www.allmusic.com': case 'allmusic.com': return globalXHR(url).then(function({document}) { function imageResolver(document) { function imageMax(imageUrl) { if (imageUrl) try { imageUrl = new URL(imageUrl); imageUrl.searchParams.set('f', 0); return imageUrl.href; } catch(e) { console.warn(e) } } const galleryExtractor = /\b(?:imageGallery) *= *(\[.+\]);?\s*$/; let imageGallery = Array.prototype.find.call(document.body.getElementsByTagName('script'), script => galleryExtractor.test(script.text)); if (imageGallery) try { imageGallery = galleryExtractor.exec(imageGallery.text); console.assert(imageGallery != null); imageGallery = eval(imageGallery[1]).map(image => imageMax(image.url)); if (imageGallery.length > 0) return imageGallery; } catch(e) { console.warn(e) } return imageMax(getFromMeta(document)) || notFound; } const mainAlbum = document.querySelector('div#mainAlbumMeta a'); if (mainAlbum == null || !modifiers.ctrlKey) return imageResolver(document); return globalXHR(mainAlbum).then(({document}) => imageResolver(document)).catch(reason => imageResolver(document)); }); case 'music.apple.com': case 'itunes.apple.com': { if ((entryIds = amEntityParser.exec(url)) == null) break; const market = /\/([a-z]{2})\//.exec(url.pathname); return queryAppleAPI(`${entryIds[1]}s/${entryIds[2]}`, undefined, market != null ? market[1] : undefined).then(function(response) { const artwork = response.data[0].attributes.artwork; return artwork ? artwork.url.replace('{w}', artwork.width).replace('{h}', artwork.height) : notFound; }); } case 'www.deezer.com': case 'deezer.com': if ((entryIds = dzrEntityParser.exec(url)) != null) return verifyImageUrl(`https://api.deezer.com/${entryIds[1]}/${entryIds[2]}/image`).catch(function(reason) { console.warn('Deezer API image retrieval failed:', reason, url); return globalXHR(url).then(({document}) => getFromMeta(document) || notFound); }).then(imageUrl => !modifiers.ctrlKey ? getDeezerImageMax(imageUrl) : verifyImageUrl(imageUrl.replace(...dzrImageMax)).catch(reason => imageUrl)); else break; case 'www.qobuz.com': case 'qobuz.com': if (url.pathname.includes('/album/')) return globalXHR(url).then(function({document}) { let img = document.querySelector('div.album-cover > img'); if (img == null) return getFromMeta(document) || notFound; return verifyImageUrl(img.src.replace(/_\d{3}(?=\.\w+$)/, '_org')) .catch(reason => verifyImageUrl(img.src.replace(/_\d{3}(?=\.\w+$)/, '_max'))) .catch(reason => img.src); }); else if (url.pathname.includes('/interpreter/') || url.pathname.includes('/artist/')) return globalXHR(url).then(function({document}) { let img = document.querySelector('div.catalog-heading__picture') || document.querySelector('div.catalog-heading__background'); if (img != null) img = /\b(?:url)\(\"(.+)\"\)/i.exec(img.style.backgroundImage); if (img != null) img = img[1]; else return getFromMeta(document) || notFound; if (!httpParser.test(img)) img = 'https:' + img; return verifyImageUrl(img.replace(/\/small\//i, '/large/')).catch(reason => img); }); else break; case 'www.boomkat.com': case 'boomkat.com': if (url.pathname.startsWith('/products/')) return globalXHR(url).then(function({document}) { let img = document.querySelector('img[itemprop="image"]'); if (img == null) return notFound; return verifyImageUrl(img.src.replace(/\/large\//i, '/original/')).catch(reason => img.src); }); else break; case 'www.bleep.com': case 'bleep.com': if (url.pathname.startsWith('/release/')) return globalXHR(url).then(function({document}) { let image = getFromMeta(document); if (!image && (image = document.body.querySelector('a.main-product-image > img')) != null) image = image.src; return image ? verifyImageUrl(image.replace(/\/r\/[a-z]\//i, '/r/')).catch(reason => image) : notFound; }); else break; case 'www.soundcloud.com': case 'soundcloud.com': return globalXHR(url).then(function({document}) { const meta = getFromMeta(document); return meta ? verifyImageUrl(meta.replace(/-\w+(?=\.\w+$)/, '-original')).catch(reason => meta) : notFound; }); case 'www.prestomusic.com': case 'prestomusic.com': if (url.pathname.includes('/products/')) return globalXHR(url).then(({document}) => verifyImageUrl(document.querySelector('div.c-product-block__aside > a').href.replace(/\?\d+$/))); else break; case 'www.bontonland.cz':case 'bontonland.cz': return globalXHR(url).then(({document}) => document.querySelector('a.detailzoom').href); case 'www.prostudiomasters.com': case 'prostudiomasters.com': if (url.pathname.includes('/album/')) return globalXHR(url).then(function({document}) { let a = document.querySelector('img.album-art'); return verifyImageUrl(a.currentSrc).catch(reason => a.src); }); else break; case 'www.e-onkyo.com': case 'e-onkyo.com': if (url.pathname.includes('/album/')) return globalXHR(url).then(function({document}) { let meta = getFromMeta(document); return meta ? meta.replace(/\/s\d+\//, '/s0/') : notFound; }); else break; case 'store.acousticsounds.com': return globalXHR(url).then(function({document}) { let link = 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/')) return globalXHR(url) .then(({document}) => verifyImageUrl(document.querySelector('div.obrazekDetail > img').src)); else break; case 'www.beatport.com': case 'classic.beatport.com': case 'pro.beatport.com': case 'beatport.com': if (url.pathname.startsWith('/release/')) return globalXHR(url).then(function({document}) { let elem = getFromMeta(document); if (!elem && (elem = document.body.querySelector('div > img.interior-release-chart-artwork')) != null) elem = elem.src; if (!elem && (elem = document.body.querySelector('div.artwork')) != null && elem.dataset.modalArtwork) // BP Classic elem = 'https:' + elem.dataset.modalArtwork; return elem || notFound; }).then(imgUrl => verifyImageUrl(imgUrl.replace(/\/image_size\/\d+x\d+\//i, '/image/'))); else break; case 'www.beatsource.com': case 'beatsource.com': if (url.pathname.startsWith('/release/')) return globalXHR(url).then(function({document}) { let imgUrl = getFromMeta(document); return imgUrl ? imgUrl.replace(/\/image_size\/\d+x\d+\//i, '/') : notFound; }); else break; case 'www.supraphonline.cz': case 'supraphonline.cz': if (!url.pathname.includes('/album/')) break; return globalXHR(url).then(function({document}) { let imageUrl = document.querySelector('div.sidebar div.sexycover > div.btn-group > button:last-of-type'); if (imageUrl != null && /^(?:coverzoom):(\S+)\$$/.test(imageUrl.dataset.plugin) && (imageUrl = imageUrl.parentNode.querySelector('script[type="data-plugin/' + RegExp.$1 + '"]')) != null) return 'https://www.supraphonline.cz' + eval(imageUrl.text); return (imageUrl = getFromMeta(document)) ? imageUrl.replace(/\?.*$/, '') : notFound; }); case 'vgmdb.net': if (url.pathname.includes('/album/')) return globalXHR(url).then(function({document}) { let div = document.querySelector('div#coverart'); return verifyImageUrl(/\b(?:url)\s*\(\"(.*)"\)/i.test(div.style['background-image']) && RegExp.$1).catch(reason => notFound); }); else break; case 'www.ototoy.jp': case 'ototoy.jp': return globalXHR(url).then(function({document}) { let img = document.querySelector('div#jacket-full-wrapper > img'); // img[alt="album jacket"] return img != null ? img.dataset.src || img.src : notFound; }); case 'music.yandex.ru': if (url.pathname.includes('/album/')) return globalXHR(url).then(function({document}) { let script = document.querySelector('script.light-data'); return verifyImageUrl(JSON.parse(script.text).image).catch(reason => notFound); }); else break; case 'www.pias.com': case 'store.pias.com': case 'pias.com': return globalXHR(url).then(function({document}) { let node = getFromMeta(document); if (node) return verifyImage(node.replace(/\/[sbl]\//i, '/')).catch(reason => node); node = 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({document}) { let a = 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)) return globalXHR(url).then(function({document}) { let meta = getFromMeta(document); return meta ? meta.replace(/\/COVER\/\w+\b/i, '/COVER/SUPERLARGE') : notFound; }); else break; case 'www.deejay.de': case 'deejay.de': return globalXHR(url).then(function({document}) { let elem = document.querySelector('div#gallery > a') || document.querySelector('div.cover a'); if (elem != null) return 'https://www.deejay.de' + elem.pathname; return (elem = getFromMeta(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.album.picUrl ? response.album.picUrl.replace(/\?.*$/, '').replace(/\b(?:p[123])(?=\.music\.\d+\.net\b)/i, 'p4') : notFound); case 'www.tidal.com': case 'listen.tidal.com': case 'tidal.com': if (!(/\/album\/(\d+)(?:\/|$)/i.test(url.pathname) && !/\b(?:albumId)=(\d+)\b/i.test(url.search))) break; return tidalAccess.requestAPI('albums/' + RegExp.$1).then(album => album.cover ? 'https://resources.tidal.com/images/' + album.cover.replace(/-/g, '/') + '/1280x1280.jpg' : notFound); case 'www.extrememusic.com': case 'extrememusic.com': if (url.pathname.startsWith('/albums/')) return globalXHR(url).then(function({document}) { let meta = getFromMeta(document); return meta ? meta.replace(/\/album\/\w+\//i, '/album/600/') : notFound; }); else break; case 'www.recochoku.jp': case 'recochoku.jp': if (url.pathname.startsWith('/album/')) return globalXHR(url).then(function({document}) { let imgUrl = getFromMeta(document); if (!imgUrl) return notFound; imgUrl = new URL(imgUrl); let params = new URLSearchParams(imgUrl.search); params.set('FFw', 999999999); params.set('FFh', 999999999); params.delete('h'); params.delete('option'); imgUrl.search = params; return imgUrl; }); else break; case 'www.elusivedisc.com': case 'elusivedisc.com': return globalXHR(url).then(function({document}) { let img = document.querySelector('figure > img.zoomImg'); if (img != null) return img.src; img = document.querySelector('section.productView-images > figure'); return img != null && img.dataset.zoomImage || notFound; }); case 'music.youtube.com': return globalXHR(url).then(function({document}) { for (let script of document.querySelectorAll('body > script[nonce]')) { let data = /\b(?:initialData\.push)\s*\(\s*\{\s*(?:path):\s*('\\\/browse'),\s*(?:params):\s*(.+?)\s*,\s*(?:data):\s*('.+?')\s*\}\s*\);/.exec(script.text); if (data != null) try { const imgMax = [/(?:=[swh]\d+.*)?$/, '=s0']; data = JSON.parse(eval(data[3])); if ('frameworkUpdates' in data) try { data = data.frameworkUpdates.entityBatchUpdate.mutations .find(mutation => mutation.payload && 'musicAlbumRelease' in mutation.payload); if (data != undefined && 'thumbnailDetails' in data.payload.musicAlbumRelease) return data.payload.musicAlbumRelease.thumbnailDetails.thumbnails[0].url.replace(...imgMax); } catch(e) { console.warn(e) } if ('header' in data) try { data = data.header.musicImmersiveHeaderRenderer.thumbnail.musicThumbnailRenderer.thumbnail.thumbnails; if (data) return data[0].url.replace(...imgMax); } catch(e) { console.warn(e) } } catch(e) { console.warn(e) } } return notFound; }); case 'www.kuwo.cn': case 'kuwo.cn': if (url.pathname.startsWith('/album_detail/')) return globalXHR(url).then(function({document}) { for (let script of document.querySelectorAll('body > script')) { if (!/\b(?:__NUXT__)\b/.test(script.text)) continue; if (/\b(?:pic):"(.+?)"/.test(script.text)) return eval('"' + RegExp.$1 + '"').replace(/(\/albumcover)\/\d+\//i, '$1/0/'); } return notFound; }); else break; case 'www.melon.com': case 'melon.com': /*if (url.pathname.startsWith('/album/')) */return globalXHR(url).then(function({document}) { let imgUrl = getFromMeta(document); if (imgUrl) imgUrl = imgUrl.replace(/\?.*$/, ''); else return notFound; return verifyImageUrl(imgUrl.replace(/(?:_\d+)?(?=\.\w+$)/, '_1000')).catch(reason => imgUrl); });// else break; case 'music.bugs.co.kr': /*if (url.pathname.startsWith('/album/')) */return globalXHR(url).then(function({document}) { let imgUrl = getFromMeta(document); return imgUrl ? imgUrl.replace(/(\/album\/images)\/\w+\//i, '$1/original/') : notFound; }); //else break; case 'www.joox.com': case 'joox.com': if (/\/album\/([^\/\?\#]+)/i.test(url.pathname)) return globalXHR('https://api-jooxtt.sanook.com/page/albumDetail?' + new URLSearchParams({ id: RegExp.$1, lang: 'en', country: 'intl', device: 'desktop', }).toString(), { responseType: 'json' }).then(({response}) => response.albumTracks.images && response.albumTracks.images.reduceRight((acc, img) => img.url.replace(/\/(\d+)$/, '/0'), undefined) || notFound); case 'mixcloud.com': case 'www.mixcloud.com': { const folders = url.pathname.split('/').filter(Boolean); if (folders.length <= 0) break; const query = folders.length > 1 ? ` query cloudcastQuery($lookup: CloudcastLookup!) { cloudcast: cloudcastLookup(lookup: $lookup) { owner { ...CloudcastBaseSidebar_user } ...CloudcastHeadTags_cloudcast } } fragment CloudcastBaseSidebar_user on User { ...UserLiveCard_user } fragment CloudcastHeadTags_cloudcast on Cloudcast { picture { urlRoot } } fragment UserLiveCard_user on User { liveStream { streamStatus id } } ` : ` query userQuery($lookup: UserLookup! $bannerContentKey: String!) { user: userLookup(lookup: $lookup) { ...UserHeadTags_user } viewer { ...UserDashboardBanner_viewer_1HzGx id } } fragment UserDashboardBanner_viewer_1HzGx on Viewer { showHideableContent(contentKey: $bannerContentKey) } fragment UserHeadTags_user on User { picture { urlRoot } } `; return mixcloudQuery(query, { lookup: { username: folders[0], slug: folders[1] }, bannerContentKey: 'DASHBOARD_BANNER_PROFILE', }).then(function(data) { let imgUrl = 'cloudcast' in data ? data.cloudcast.picture.urlRoot : 'user' in data ? data.user.picture.urlRoot : null; return imgUrl ? 'https://thumbnailer.mixcloud.com/unsafe/' + imgUrl : notFound; }); } case 'www.metal-archives.com': case 'metal-archives.com': if (url.pathname.startsWith('/albums/')) return globalXHR(url).then(function({document}) { const cover = document.getElementById('cover'); return cover != null ? cover.href.replace(/\?\S*$/, '') : getFromMeta(document) || notFound; }); else break; case 'www.rateyourmusic.com': case 'rateyourmusic.com': if (url.pathname.startsWith('/release/')) return globalXHR(url).then(function({document}) { let cover = document.querySelector('div.page_release_art_frame img'); return cover != null ? cover.src : notFound; }); else break; // books-related case 'www.goodreads.com': case 'goodreads.com': if (url.pathname.includes('/show/')) return globalXHR(url).then(function({document}) { let img = ['div.BookCover__image img', 'div.editionCover > img', 'img#coverImage'] .reduce((elem, selector) => elem || document.querySelector(selector), null); img = img != null ? img.src : getFromMeta(document); return img && !['/nophoto/', '/books/1570622405l/50809027', '/images/no-cover.png'].some(pattern => img.includes(pattern)) ? img.replace(/\._\w+_\./g, '.').replace(/\?.*$/, '') : notFound; }); else break; case 'www.databazeknih.cz': case 'databazeknih.cz': if (url.pathname.startsWith('/knihy/')) return globalXHR(url).then(function({document}) { let elem = document.querySelector('div#icover_mid > a'); if (elem != null) return imageUrlResolver('https://www.databazeknih.cz' + elem.pathname, modifiers); const imageMax = imageUrl => httpParser.test(imageUrl) ? verifyImageUrl([ [/\/\d+\/([a-z]+)(?=_)/, 'big'], [/\?.*$/, ''], ].reduce((acc, def) => acc.replace(...def), imageUrl)).catch(reason => imageUrl) : Promise.reject('invalid url'); if ((elem = document.querySelector('div#lbImage')) != null && (elem = /\b(?:url)\("(.*)"\)/i.exec(elem.style.backgroundImage)) != null) return imageMax(elem[1]); return (elem = document.querySelector('img.kniha_img')) != null ? imageMax(elem.src) : notFound; }); else if (url.pathname.startsWith('/obalka-knihy/')) return globalXHR(url).then(function({document}) { let elem = document.querySelector('img.book_cover_big'); return elem != null ? elem.src.replace(/\?.*/, '') : notFound; }); else break; case 'www.alza.cz': case 'alza.cz': case 'www.alza.sk': case 'alza.sk': return globalXHR(url).then(function({document}) { const imageMax = imgSrc => imgSrc.replace(/([\?\&])fd=(?:f\d+)\b\&?/i, '$1'); let meta = document.querySelectorAll('div#galleryPreview a.lightBoxImage'); if (meta.length > 0) return Array.from(meta) .map(a => imageMax(a.dataset.original || a.href || a.dataset.bigimage)); meta = document.querySelector('div.detail-page > script[type="application/ld+json"]'); if (meta != null) try { meta = JSON.parse(meta.text) } catch(e) { meta = null } if (meta != null && httpParser.test(meta.image)) return imageMax(meta.image); return (meta = getFromMeta(document)) ? imageMax(meta) : 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({responseText}) { if (/\b(?:window\.IMDbMediaViewerInitialState)\s*=\s*(\{.*\});/.test(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({document}) { let node = document.querySelector('meta[property="og:image"][content]'); return verifyImageUrl(node.content.replace(/\/p\/\w+\//i, '/p/original/')).catch(function(reason) { node = 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({document}) { let node = 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(({document}) => verifyImageUrl(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({document}) { //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(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(({document}) => verifyImageUrl(document.location.protocol.concat(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(({document}) => verifyImageUrl(document.querySelector('div.mojo-primary-image img').src)); case 'www.metacritic.com': case 'metacritic.com': return globalXHR(url).then(function({document}) { let image = 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) { let 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)); let 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)); let 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)) }, }); } }); let 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({document}) { let a = document.querySelector('a.boxPlakaty'); if (a == null) return Promise.reject('Invalid page structure'); a.hostname = 'www.fdb.cz'; return globalXHR(a.href).then(function({document}) { let imgs = 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') return globalXHR(url).then(function(response) { function heightExtractor(n) { let 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'; let result = Array.from(response.document.querySelectorAll('div.main > div[style] > a > img.thumb')).map(function(img) { let 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)}`, ]; }); result.caption = Array.from(response.document.querySelectorAll('body > div.bdinfo > div.blue_bar:first-of-type')).map(function(div) { let caption = div.childNodes[0].textContent.trim(); if (div.childNodes.length > 1) caption += ' (' + div.childNodes[1].textContent.trim() + ')'; return caption; }); return result; }); else break; 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(({responseText}) => [ /\b(?:images)\[1\]='(\S+?)'/.test(responseText) && RegExp.$1, /\b(?:images)\[0\]='(\S+?)'/.test(responseText) && RegExp.$1, ].map(src => origin.concat(src))); }); }); else 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)); }); else break; } return globalXHR(url, { headers: { 'Referer': url.origin } }).then(function({document}) { if (url.pathname.startsWith('/album/') && document.querySelector('div#tabbed-content-group > div.content-listing > div.pad-content-listing') != null) return new Chevereto(url.hostname).galleryResolver(url); let elem = document.querySelector('head > meta[name="generator"][content]'); if (elem != null && elem.content.toLowerCase() == 'bandcamp') { elem = document.querySelector('div#tralbumArt > a.popupImage'); elem = elem != null ? elem.href : getFromMeta(document); return httpParser.test(elem) ? elem.replace(/_\d+(?=\.\w+$)/, '_0') : notFound; } return getFromMeta(document) || notFound; }); })); } // don't clash with Upload Assistant if (document.getElementById('upload-assistant') != null) return imageHostUploaderInit(null, null, null, imageUrlResolver); function writeInfo() { let input = document.querySelector('input[name="summary"]'); if (input != null && !input.disabled && !input.value) input.value = 'Image update/rehost'; } const safeRehostSingleImage = imageUrl => imageHosts.rehostImages([imageUrl]).then(singleImageGetter, function(reason) { if (['redacted.ch'].includes(document.location.hostname) && imageUrl.includes('.img2go.com/dl/')) return forcedRehost(imageUrl); return Promise.reject(reason); }); function setImage(url) { return verifyImageUrl(url).then(imageUrl => { this.value = imageUrl; //this.disabled = true; this.style.opacity = 0.75; writeInfo(); const size = getRemoteFileSize(imageUrl); imagePreview(imageUrl, size); return checkImageSize(imageUrl, this, size).then(imageUrl => { return safeRehostSingleImage(imageUrl).then(imageUrl => { if (imageUrl == null) throw 'invalid image'; this.value = imageUrl; }); }).catch(reason => { this.value = imageUrl; logFail(reason + ' (not rehosted)'); }).then(() => { this.style.opacity = null; this.disabled = false; return imageUrl; }); }); } function inputDataHandler(evt, data) { const input = evt.currentTarget; console.assert(input instanceof HTMLInputElement, 'input instanceof HTMLInputElement'); const rehoster = imageUrl => safeRehostSingleImage(imageUrl).then(function(imageUrl) { if (!httpParser.test(imageUrl)) { console.warn('rehostImages returns invalid image URL:', imageUrl); throw 'invalid image URL'; } input.value = imageUrl; writeInfo(); }); if (!data) return true; if (data.files.length > 0) { if (data.files[0].type && !data.files[0].type.startsWith('image/')) return true; input.disabled = true; if (input.hTimer) { clearTimeout(input.hTimer); delete input.hTimer; } input.style.color = 'white'; input.style.backgroundColor = 'darkred'; let progressBar = { }; function progressHandler(worker, param = null) { if (param && typeof param == 'object') { if (param.readyState > 1 || progressBar.current != undefined && worker !== progressBar.current || Date.now() < progressBar.lastUpdate + 100) return; let pct = Math.floor(Math.min(param.done * 100 / param.total, 100)); if (pct <= progressBar.lastPct) return; input.value = 'Uploading... [' + (progressBar.lastPct = pct) + '%]'; progressBar.lastUpdate = Date.now(); } else if (param == null) { progressBar = { current: worker }; input.value = 'Uploading...'; } } const file = data.files[0]; input.disabled = true; checkImageSize(file, input, progressHandler).catch(function(reason) { logFail('Downsizing of source image not possible (' + reason + '), uploading original size'); return file; }).then(function(result) { const uploader = file => imageHosts.uploadImages([file], progressHandler).then(singleImageGetter).then(function(imageUrl) { input.value = imageUrl; imagePreview(imageUrl, file.size); writeInfo(); }); if (httpParser.test(result)) return rehoster(result).catch(function(reason) { logFail('Downsizing of source image failed (' + reason + '), uploading original size'); return uploader(file); }); if (result instanceof File) return uploader(result); console.warn('invalid checkImageSize(...) result:', result); return Promise.reject('invalid checkImageSize(...) result'); }).then(function() { input.style.backgroundColor = '#008000'; input.hTimer = setTimeout(function() { input.style.backgroundColor = null; input.style.color = null; delete input.hTimer; }, 10000); }, function(reason) { imageClear(evt); input.style.backgroundColor = null; input.style.color = null; Promise.resolve(reason).then(msg => { alert(msg) }); }).then(() => { input.disabled = false }); return false; } else if (data.items.length > 0) { let urls = data.getData('text/uri-list'); if (urls) urls = urls.split(/\r?\n/); else { urls = data.getData('text/x-moz-url'); if (urls) urls = urls.split(/\r?\n/).filter((item, ndx) => ndx % 2 == 0); else if (urls = data.getData('text/plain')) urls = urls.split(/\r?\n/); } if (!Array.isArray(urls) || urls.length <= 0) return true; input.disabled = true; console.time('Image URL Rehoster'); imageUrlResolver(urls[0], { altKey: evt.altKey, ctrlKey: evt.ctrlKey != (input.name == 'image[]'), shiftKey: evt.shiftKey, }).then(verifyImageUrl).then(function(imageUrl) { input.disabled = false; input.style.opacity = 0.75; input.value = imageUrl; const size = getRemoteFileSize(imageUrl); imagePreview(imageUrl, size); checkImageSize(imageUrl, input, size).then(rehoster).catch(function(reason) { input.value = imageUrl; Promise.resolve(reason).then(msg => { alert(msg + ' (not rehosted)') }); }).then(() => { console.timeEnd('Image URL Rehoster') }); }).catch(reason => { Promise.resolve(reason).then(alert) }).then(function() { input.style.opacity = null; input.disabled = false; }); return false; } return true; } 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)); } let opti_PNG = GM_getValue('optipng', false); function rehoster(promises, resultsHandler, target = null) { if (!Array.isArray(promises)) throw 'invalid parameter'; console.time('Image URL Resolver'); return Promise.all(promises).then(function(resolved) { let 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 (function() { if (!opti_PNG || !(target instanceof HTMLElement)) return Promise.resolve(resolvedUrls); return Promise.all(resolvedUrls.map(resolvedUrl => optiPNG(resolvedUrl).catch(reason => resolvedUrl))); })().then(srcUrls => imageHosts.rehostImages(srcUrls, RHProgressBar.prototype.update.bind(progressBar)).catch(function(reason) { logFail(reason + ' (not rehosted)'); RHProgressBar.prototype.update.call(progressBar, -1, false); return verifyImageUrls(srcUrls); }).then(function(results) { resolved.forEach(function(elem, index) { if (!elem.caption) return; if (!Array.isArray(results.captions)) results.captions = [ ]; results.captions.push(elem.caption); }); resultsHandler(results, arrayGrouping(resolved).flatten()); }).catch(reason => { Promise.resolve(reason).then(msg => { alert(msg) }) })).then(function() { RHProgressBar.prototype.cleanUp.call(progressBar); if (target instanceof HTMLElement) target.disabled = false; console.timeEnd('Image URL Resolver'); }); }); } function textAreaDropHandler(evt) { if (!evt.dataTransfer || evt.shiftKey) return true; const textArea = evt.currentTarget; console.assert(textArea instanceof HTMLTextAreaElement, 'textArea instanceof HTMLTextAreaElement'); 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; textArea.disabled = true; if (!['notwhat.cd'].some(hostname => document.domain == hostname)) var progressBar = new ULProgressBar(textArea, 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)) .catch(reason => { Promise.resolve(reason).then(msg => { alert(msg) }) }) .then(function() { ULProgressBar.prototype.cleanUp.call(progressBar); textArea.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(url => imageUrlResolver(url, { ctrlKey: !evt.ctrlKey })), resultsHandler, textArea).catch(function(reason) { if (evt.ctrlKey) textArea.value = textArea.value.slice(0, evt.rangeOffset) + content.join('\n') + textArea.value.slice(evt.rangeOffset); else { if (textArea.value.length > 0) textArea.value += '\n\n'; textArea.value += content.join('\n'); } }); evt.stopPropagation(); return false; } return true; function resultsHandler(results, groups = undefined) { if (results.length <= 0) return; if (evt.altKey && !textArea.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 = `
`; document.body.append(modal); let form = document.getElementById('ihh-template-selector'), btnInsert = form.querySelector('input#btn-insert'), btnCancel = form.querySelector('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) { let radio = document.createElement('input'); radio.type = 'radio'; radio.name = 'template'; radio.value = item[1]; radio.style = 'margin: 5px 15px 5px 0px; cursor: pointer;'; let 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); let br = document.createElement('br'); form.insertBefore(br, btnInsert); }); if (!results.some(result => typeof result == 'object' && httpParser.test(result.original) && httpParser.test(result.thumb))) disableItem(2, 7); if (!results.some(result => typeof result == 'object' && httpParser.test(result.original) && httpParser.test(result.share))) disableItem(3, 8); if (results.length % 2 != 0) disableItem(4, 5); form.onclick = evt => { evt.stopPropagation() }; btnInsert.onclick = function(evt) { let template = document.querySelector('form#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 }); function disableItem(...n) { n.forEach(function(n) { let 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 (textArea.noBBCode) template = 0; if (typeof template != 'number' || isNaN(template)) return; let code = '', nl = [6, 7, 8].includes(template) ? '