// ==UserScript== // // @name Discogs Scout // @version 3.0 // @namespace https://github.com/Purfview/Discogs-Scout // @description Auto search for music on torrent, local drive, ddl, streaming, predb, and other sites. Adds links to Discogs pages from various sites. // @icon  // @license MIT // // @homepage https://github.com/Purfview/Discogs-Scout // @supportURL https://github.com/Purfview/Discogs-Scout/issues // // @compatible firefox // @compatible opera // @compatible chrome // @compatible safari (it doesn't support the sites with logins) // @compatible edge // // @require https://cdn.jsdelivr.net/gh/sizzlemctwizzle/GM_config@43fd0fe4de1166f343883511e53546e87840aeaf/gm_config.js // @require https://code.jquery.com/jquery-3.5.1.min.js // @require https://greasemonkey.github.io/gm4-polyfill/gm4-polyfill.js // // @include https://www.discogs.com/artist/* // @include https://www.discogs.com/master/* // @include https://www.discogs.com/release/* // @include https://www.discogs.com/label/* // @include https://www.discogs.com/user/*/collection* // @include https://www.discogs.com/wantlist* // @include https://www.discogs.com/mywantlist* // @include https://www.discogs.com/lists/* // // @include https://www.discogs.com/*/artist/* // @include https://www.discogs.com/*/master/* // @include https://www.discogs.com/*/release/* // @include https://www.discogs.com/*/label/* // @include https://www.discogs.com/*/user/*/collection* // @include https://www.discogs.com/*/wantlist* // @include https://www.discogs.com/*/mywantlist* // @include https://www.discogs.com/*/lists/* // // @connect * // @grant GM_getValue // @grant GM_setValue // @grant GM_addStyle // @grant GM_openInTab // @grant GM_xmlhttpRequest // @grant GM_registerMenuCommand // @grant GM.getValue // @grant GM.setValue // @grant GM.openInTab // @grant GM.xmlHttpRequest // @grant GM.registerMenuCommand // @grant GM.notification // // @run-at document-start // @noframes // // @downloadURL none // ==/UserScript== /* //============================================================================== // Version History: //============================================================================== 3.0 - Fixed: Release and Artist pages. 2.7.1 - Removed: PREovh. 2.7 - Adjusted "large covers" notification. 2.6 - New feature: Support for other sites's languages. Fixed: Bug with 'replaceSpecials', it was deleting 'f'. "E.P" removal. Added: NZBKing, Alterportal. Updated: Tidal. 2.5 - Fixed issues with a page reload/refresh. 2.4.1 - Added: HQMusic, BitSearch. 2.4 - Fixed: Inconsistency with mPOST formattings ["+" replacement with space for 2 & 3 formattings]. New feature: Added duplicate keys support for mPOST 1 formatting. Added: PunkTorrents, KrayTracker. 2.3 - Case insensitive "EP" & "E.P." removal. 2.2 - New feature: Option to remove "EP" & "E.P." from the end of release titles. [enabled by default] 2.1.1 - Added: Bemaniso, AnimeBytes. 2.1 - Fixed: Own wantlist page wasn't working. 2.0 - New feature: Support for the new version of the master pages (beta). New feature: Support for the wantlist/collection/label/list pages. New feature: Searching for Various/No Artist ect. albums should work now. Fixed: Old version of the master pages stopped working. Fixed: mPOST didn't work for the icon sites. Fixed: POST form submit didn't work on the release pages (jQuery issue, replaced with pure JS). Fixed: Searchable sites with mPOST didn't work if auto-search was disabled. Added: Voidtools-App, TPB-Proxy, RMT, Metal-Tracker, YesAsia, PandaCD, RlsBB, Sharing-DB, Mp3db, BoxAlbums, CorruptNet-Pre, CorruptNet-Trace, LeakedAlbums, Lossless-music, RockBox, TakeMetal, Core Radio, GetMetal Club, Gabber.od.ua, 1Gabba, FLAC Attack, zHouse, 1Techno, 1Trance, The Last Disaster, eBay, Spotify, Spotify-App, JunoDownload, Apple Music, Bleep, Qobuz, Deezer, Tidal. 1.7 - Fixed: Possible bug with sites (if it's mPOST and there are special chars in the name). Removed: PREcBurns, TPB-Proxy. 1.6 - New feature: Support for new Master release pages. 1.5 - Added: KG-Release. Removed: preFYP. Fix to keep the bars order consistent. Remove text in brackets from %release% names. New attribute 'replaceSpecials' to remove non-latin and special characters. 1.4.1 - Tweak: JPop. 1.4 - Added: LiB, LiB-Req, JPop, JPop-Req. The settings tweak: If site has 'goToUrl' attribute then show its hostname instead of 'searchUrl'. New feature: 'Pre databases' section in the settings. Added preDBs: PREcBurns, PreDB.de, PreDB.me, PreDB.org, PreDB.pw, preFYP, PREovh, srrDB, xREL. 1.3 - Fixed: "+" in band/release. Fixed: "&" in band/release. 1.2 - New setting to put all links into a single bar. 1.1 - Added: Bunch of sites. Blue color for requests. Tweaked default rate limit. 1.0 - First usable version. New feature: Support for Release/Master pages. New feature: Three bars for sites: 1st for no-auto-search sites, 2nd for searchable download sites 3rd for searchable other sites/tools. New feature: Auto open the settings if no sites selected. Added: First bunch of sites. Updated: OPS (advanced search is broken, switched to basic). 0.3 - Fixed OPS. Check if there are releases on artist page. 0.2.1 - Added the update links and OPS. 0.1 - Initial alpha test release. //============================================================================== // A list of all the sites. //============================================================================== -= Each site is a dictionary with the following attributes: =- # 'name': The site name, full or abbreviated, must be unique. # 'icon' (optional): Icon for the site. If not defined then script looks at homesite/favicon.ico. Can be URL or Base64 string (www.base64-image.de). # 'searchUrl': The URL to perform the search against, see below for how to tailor the string to a site. # 'matchRegex': The string which appears if the searchUrl *doesn't* return a result. # 'bar': 1,2 or 3 integer. Places site at 1st, 2nd or 3rd bar element. # 'positiveMatch' (optional): Changes the test to return true if the searchUrl *does* return a result that matches matchRegex. # 'SpaceEncode' (optional): Changes the character used to encode spaces in band/release names. Default is '+'. # 'replaceSpecials' (optional): Remove non latin and special characters in band/release names. # 'goToUrl' (optional): Most of the time the same URLs that are used for checking are the ones that are used to actually get to the movie, but this allows overriding that. # 'loggedOutRegex' (optional): If any text on the page matches this regex, the site is treated as being logged out, rather than mising the movie. This option is not effected by positiveMatch. # 'rateLimit' (optional): Connection rate limit in milliseconds. Default is 500. On the Artist/List pages if rateLimit<=1000 then it will be increased by a factor of 4. # 'mPOST': HTTP request by POST method. For the sites that doesn't support GET. Right mouse click won't submit such request. Atm 'goToUrl' not supported with it. Examples (3 types of formating): 1) 'cat1=4&cat2=6&filter=%tt%' // (supports duplicate keys) 2) '{"cat1":4,"cat2":6,"filter":"all=%band%+%release%&sort=date"}' 3) '{key:"cat",value:"4"},{key:"cat",value:"6"},{key:"filter",value:"%band%+%release%"}' // (supports duplicate keys) Note: only these special chars are allowed in a site name if mPOST: .- (). # 'ignore404' (optional): Ignores all 4** HTTP errors. # 'ignoreEmpty' (optional): Use it if an empty response means that no results found, otherwise by default it means 'logged_out'. -= Search URL parameters: =- # %band%: Band's name. # %release%: Album/EP/Single name. */ var icon_sites = [ { 'name': 'AllMusic', 'icon': '', 'searchUrl': 'https://www.allmusic.com/search/albums/"%release%"', 'bar': 1}, { 'name': 'Amazon', 'icon': '', 'searchUrl': 'https://www.amazon.com/s?k=%band%+%release%&i=music-intl-ship&ref=nb_sb_noss', 'bar': 1}, { 'name': 'Apple Music', 'icon': '', 'searchUrl': 'https://music.apple.com/search?term=%band%+%release%', 'bar': 1}, { 'name': 'Bandcamp', 'icon': '', 'searchUrl': 'https://bandcamp.com/search?q=%band%+%release%', 'bar': 1}, { 'name': 'Beatport', 'icon': '', 'searchUrl': 'https://www.beatport.com/search?q=%band%+%release%', 'bar': 1}, { 'name': 'Bleep', 'icon': '', 'searchUrl': 'https://bleep.com/search/query?q=%release%', 'bar': 1}, { 'name': 'CorruptNet-Trace', 'icon': '', 'searchUrl': 'https://trace.corrupt-net.org/search.php?search=%band%+%release%+type:FLAC|MP3', 'replaceSpecials': true, 'bar': 1}, { 'name': 'Decks', 'icon': '', 'searchUrl': 'https://www.decks.de/decks/workfloor/search_db.php?such=%band%+%release%&wosuch=all&wassuch=atl&where=', 'bar': 1}, { 'name': 'Deejay', 'icon': '', 'searchUrl': 'https://www.deejay.de/%band%+%release%', 'bar': 1}, { 'name': 'Deezer', 'icon': '', 'searchUrl': 'https://www.deezer.com/search/%band%+%release%/album', 'bar': 1}, { 'name': 'eBay', 'icon': '', 'searchUrl': 'https://www.ebay.com/sch/11233/i.html?_nkw=%band%+%release%&_sop=15', 'bar': 1}, { 'name': 'Genius', 'icon': '', 'searchUrl': 'https://genius.com/search?q=%band%+%release%', 'bar': 1}, { 'name': 'Google', 'icon': '', 'searchUrl': 'https://www.google.com/search?q="%band%"+"%release%"', 'bar': 1}, { 'name': 'Juno', 'icon': '', 'searchUrl': 'https://www.juno.co.uk/search/?q[all][]=%band%+%release%&hide_forthcoming=0', 'bar': 1}, { 'name': 'JunoDownload', 'icon': '', 'searchUrl': 'https://www.junodownload.com/search/?q[all][]=%band%+%release%', 'bar': 1}, { 'name': 'Last.fm', 'icon': '', 'searchUrl': 'https://www.last.fm/search?q=%band%+%release%', 'bar': 1}, { 'name': 'MusicBrainz', 'icon': '', 'searchUrl': 'https://musicbrainz.org/search?query="%release%"+AND+artist:"%band%"&type=release_group&limit=25&method=advanced', 'bar': 1}, { 'name': 'Qobuz', 'icon': '', 'searchUrl': 'https://www.qobuz.com/search?q=%band%+%release%', 'bar': 1}, { 'name': 'RYM', 'icon': '', 'searchUrl': 'https://rateyourmusic.com/search?searchterm=%band%+%release%', 'bar': 1}, { 'name': 'SoundCloud', 'icon': '', 'searchUrl': 'https://soundcloud.com/search/albums?q=%band%+%release%', 'bar': 1}, { 'name': 'Spotify', 'icon': '', 'searchUrl': 'https://open.spotify.com/search/%band%+%release%/albums', 'spaceEncode': ' ', 'bar': 1}, { 'name': 'Spotify-App', 'icon': '', 'searchUrl': 'spotify:search:artist:%band% album:%release%', 'spaceEncode': ' ', 'bar': 1}, { 'name': 'Tidal', 'icon': '', 'searchUrl': 'https://listen.tidal.com/search?q=%band%+%release%', 'bar': 1}, { 'name': 'TPB-Proxy', 'icon': '', 'searchUrl': 'https://tpb.one/search.php?q=%band%+%release%&cat=101,104', 'bar': 1}, { 'name': 'Wikipedia', 'icon': '', 'searchUrl': 'https://en.wikipedia.org/w/index.php?search=%band%+%release%&go=Go', 'bar': 1}, { 'name': 'YouTube', 'icon': '', 'searchUrl': 'https://www.youtube.com/results?search_query=%band%+%release%', 'bar': 1} ]; var public_sites = [ { 'name': '1Gabba', 'icon': '', 'searchUrl': 'https://1gabba.pw/frontpage?title=%band%+-+%release%', 'loggedOutRegex': /Cloudflare|Ray ID/, 'matchRegex': /class="node/, 'positiveMatch': true, 'bar': 2}, { 'name': '1Techno', 'icon': '', 'searchUrl': 'https://1techno.org/?t=%band%+-+%release%', 'loggedOutRegex': /Cloudflare|Ray ID/, 'matchRegex': /find any results/, 'bar': 2}, { 'name': '1Trance', 'icon': '', 'searchUrl': 'https://1trance.org/?t=%band%+-+%release%', 'loggedOutRegex': /Cloudflare|Ray ID/, 'matchRegex': /find any results/, 'bar': 2}, { 'name': 'Alterportal', 'icon': '', 'searchUrl': 'https://alterportal.net/index.php?do=search&subaction=search&titleonly=3&story=%band%+%release%', 'loggedOutRegex': /Cloudflare|Ray ID/, 'matchRegex': /никаких результатов/, 'bar': 2}, { 'name': 'BitSearch', 'icon': '', 'searchUrl': 'https://bitsearch.to/search?q=%band%+%release%&category=7&subcat=', 'loggedOutRegex': /Cloudflare|Ray ID/, 'matchRegex': /Found 0Total hits : 0Not FoundRemember meRemember meYes<|Nichts gefunden!|Aucune requête trouvée!|Nada Encontrado!|Нет результатов!|何も見つかりませんでした/, 'bar': 2}, { 'name': 'KG-Release', 'icon': '', 'searchUrl': 'https://karagarga.in/browse.php?search="%release%"&search_type=title&cat=2', 'loggedOutRegex': /Cloudflare|Ray ID|Not logged in!/, 'matchRegex': /No torrents found/, 'bar': 2}, { 'name': 'KrayTracker', 'icon': '', 'searchUrl': 'https://kraytracker.com/torrents.php?searchstr=%band%+%release%', 'loggedOutRegex': /Cloudflare|Ray ID|>Remember meRemember meKeep me logged in.Keep me logged in.Remember meRemember meRemember meRemember me0 results0 results<\/tbody>/, 'replaceSpecials': true, 'bar': 3}, { 'name': 'srrDB', 'icon': '', 'searchUrl': 'https://api.srrdb.com/v1/search/%band%/%release%', 'goToUrl': 'https://www.srrdb.com/browse/%band%/%release%', 'matchRegex': /resultsCount":"0/, 'spaceEncode': '/', 'replaceSpecials': true, 'bar': 3}, { 'name': 'xREL', 'icon': '', 'searchUrl': 'https://www.xrel.to/search.html?xrel_search_query=%band%+%release%&lang=en_US', 'matchRegex': /not return any results/, 'bar': 3} ]; var sites = public_sites.concat(private_sites, other_sites, pre_databases); //============================================================================== // Replace Search URL parameters //============================================================================== // URLs for tests: // https://www.discogs.com/artist/2274430-Tuning-Circuits // https://www.discogs.com/master/2385727-Twice-Formula-Of-Love-OT3 // https://www.discogs.com/master/1257041-For-King-Country-Crave // https://www.discogs.com/master/1787931-Baauer-Planets-Mad async function replaceSearchUrlParams(site, band, release, mPOSTsearch) { var search_url = ('mPOST' in site && !mPOSTsearch) ? site['mPOST'] : site['searchUrl']; var space_replace = ('spaceEncode' in site) ? site['spaceEncode'] : '+'; let band_str = band.replace(/\(\d+\)/g, '') // delete brackets with numbers inside .replace(/\*/g, '') // delete * .replace(/^Unknown Artist$/, '') // delete string if Various and ect.: .replace(/^No Artist$/, '') .replace(/^Anonymous$/, '') .replace(/^Various$/, '').trim(); let release_str = release.replace(/\(.+\)/g, '') // delete brackets with anything inside .replace(/-|–/g, ' ').trim(); // replace - – with space release_str = GM_config.get('remove_ep') ? release_str.replace(/\sEP+$|\sE\.P\.+$|\sE\.P/i, '') : release_str // remove " EP"," E.P." & " E.P" from the end if (site['replaceSpecials'] === true) { // Replace non latin | Special chars remove. Not included "`", "-", ".", "_". band_str = band_str.trim().replace(/[\u0250-\ue007]/g, ' ').replace(/\'/g, '').replace(/\¬|\!|\"|\£|\$|\%|\^|\&|\*|\(|\)|\+|\=|\||\\|\[|\]|\;|\#|\,|\?|\/|\{|\}|\:|\@|\~|\<|\>/g, ' '); release_str = release_str.trim().replace(/[\u0250-\ue007]/g, ' ').replace(/\'/g, '').replace(/\¬|\!|\"|\£|\$|\%|\^|\&|\*|\(|\)|\+|\=|\||\\|\[|\]|\;|\#|\,|\?|\/|\{|\}|\:|\@|\~|\<|\>/g, ' '); } // encode illegal chars band_str = band_str.replace(/\+/g, '%2B').replace(/&/g, '%26').replace(/#/g, '%23').replace(/=/g, '%3D').replace(/\s+/g, space_replace).trim(); release_str = release_str.replace(/\+/g, '%2B').replace(/&/g, '%26').replace(/#/g, '%23').replace(/=/g, '%3D').replace(/\s+/g, space_replace).trim(); var s = search_url.replace(/%band%/g, band_str) .replace(/%release%/g, release_str); return s; } //============================================================================== // Get site's icon //============================================================================== function getFavicon(site, hide_on_err) { var favicon; if (typeof(hide_on_err) === 'undefined') { hide_on_err = false; } else if (hide_on_err === false) { return; } if ('icon' in site) { favicon = site['icon']; } else { var url = new URL(site['searchUrl']); favicon = url.origin + '/favicon.ico'; } const size = GM_config.get('mod_icons_size'); const border = parseInt(GM_config.get('iconsborder_size')) *2; var iconsize = (site['bar'] == 1) ? size : GM_config.get('auto_search') ? size - border : size; var title = site['name']; var img = $('').attr({'style': '-moz-opacity: 0.4; border: 0', 'width': iconsize, 'height': iconsize, 'src': favicon, 'title': title, 'alt': site['name']}); if (hide_on_err) { img.attr('onerror', "this.style.display='none';"); } return img; } //============================================================================== // Create elements and add search links //============================================================================== function addLink(elem, site_name, target, site, state, scout_tick, post_data) { // State should always be one of the values defined in valid_states. if ($.inArray(state, valid_states) < 0) { console.log("Unknown state: " + state); } var link = $('').attr('href', target).attr('target', '_blank').attr('rel', 'noreferrer'); // Link and add Form element for POST method. if ('mPOST' in site) { var form_name = site['name'] + '-form-' + scout_tick; form_name = form_name.replace(/\s|\.|\(|\)/g, '-'); var placebo_url = new URL(target).origin; link = $('').attr('href', placebo_url).attr('onclick', "document.getElementById('"+form_name+"').submit(); return false;").attr('target', '_blank').attr('rel', 'noreferrer'); //var data = (post_data.match('{')) ? post_data.replace(/\+/g, ' ') : '{"' + post_data.replace(/&/g, '","').replace(/=/g, '":"').replace(/\+/g, ' ') + '"}'; var data = (post_data.match('{')) ? post_data.replace(/\+/g, ' ') : '{key:"' + post_data.replace(/&/g, '"},{key:"').replace(/=/g, '",value:"').replace(/\+/g, ' ') + '"}'; var addform = $('
'); addform.attr('id', form_name); addform.attr('action', target); addform.attr('method', 'post'); addform.attr('style', 'display: none;'); addform.attr('target', '_blank'); addform.attr('rel', 'noreferrer'); if (data.match('},{')) { const dataArray = (new Function("return [" +data+ "];")()); dataArray.forEach(function (item, index) { let addinput = $(""); addinput.attr('type', 'text'); addinput.attr('name', item.key); addinput.attr('value', item.value); addform.append(addinput); $('body').append(addform); }); } else { data = JSON.parse(data); for (const name in data) { let addinput = $(""); addinput.attr('type', 'text'); addinput.attr('name', name); addinput.attr('value', data[name]); addform.append(addinput); $('body').append(addform); } } } // Icon appearance. let icon; const border_width = GM_config.get('iconsborder_size'); if (GM_config.get('auto_search') && site['bar'] != 1) { icon = getFavicon(site); icon.css({'border-width': border_width, 'border-style': 'solid', 'border-radius': '2px', 'margin': '1px 2px 2px'}); if (state == 'error' || state == 'logged_out') { (GM_config.get('highlight_sites').split(',').includes(site['name'])) ? icon.css('border-color', 'rgb(255,0,0)') : icon.css('border-color', 'rgb(180,0,0)'); } else if (state == 'missing') { (GM_config.get('highlight_sites').split(',').includes(site['name'])) ? icon.css('border-color', 'rgb(255,255,0)') : icon.css('border-color', 'rgb(230,200,100)'); } else if (state == 'found') { (GM_config.get('highlight_sites').split(',').includes(site['name'])) ? icon.css('border-color', 'rgb(0,220,0)') : icon.css('border-color', 'rgb(0,130,0)'); if ((site['name']).match('-Req')) icon.css('border-color', 'rgb(50,50,200)'); } link.append(icon); } else { icon = getFavicon(site); icon.css({'border-width': '0px', 'border-style': 'solid', 'border-radius': '2px', 'margin': '1px 2px 2px'}); (GM_config.get('highlight_sites').split(',').includes(site['name'])) ? icon.css('border-color', 'rgb(0,220,0)') : icon.css('border-color', 'rgb(0,130,0)'); if ((site['name']).match('-Req')) icon.css('border-color', 'rgb(50,50,200)'); link.append(icon); } // Create elements on Release/Master pages. if (onReleasePage) { const background = GM_config.get('greybackground_view') ? 'rgb(51, 51, 51)' : ''; if ($('.result_box_main').length == 0) { $(elem).after($('
').addClass('result_box_main')); $('.result_box_main').css({'background-color': background, 'padding': '4px 4px 0px 4px'}); $('.result_box_main').append($('
').addClass('result_bar_1st')); $.each(valid_states, function(i, name) { $('.result_bar_1st').append(""+''); }); $('.result_box_main').append($('
').addClass('result_bar_2nd')); $.each(valid_states, function(i, name) { $('.result_bar_2nd').append(""+''); }); $('.result_box_main').append($('
').addClass('result_bar_3rd')); $.each(valid_states, function(i, name) { $('.result_bar_3rd').append(""+''); }); } // Add links to elements on Release/Master pages. if (site['bar'] == 1 || GM_config.get('all_in_one_bar')) { $('#discogscout1_' + state).append(link); } else if (site['bar'] == 2) { $('#discogscout2_' + state).append(link); } else if (site['bar'] == 3) { $('#discogscout3_' + state).append(link); } } else { // Create elements on Artist/Label/Collection/Wantlist/MyWantlist/List pages. const background = GM_config.get('greybackground_view') ? 'rgb(51, 51, 51)' : ''; if ($('.result_box_main' + scout_tick).length == 0) { $(elem).after($('').append($('',{'colspan':'11'}).addClass('result_box_main' + scout_tick))); $('.result_box_main' + scout_tick).css({'background-color': background, 'padding': '0px 4px'}); $('.result_box_main' + scout_tick).append($('
').addClass('result_bar_1st' + scout_tick)); $.each(valid_states, function(i, name) { $('.result_bar_1st' + scout_tick).append(""+''); }); $('.result_box_main' + scout_tick).append($('
').addClass('result_bar_2nd' + scout_tick)); $.each(valid_states, function(i, name) { $('.result_bar_2nd' + scout_tick).append(""+''); }); $('.result_box_main' + scout_tick).append($('
').addClass('result_bar_3rd' + scout_tick)); $.each(valid_states, function(i, name) { $('.result_bar_3rd' + scout_tick).append(""+''); }); } // Add links to elements on Artist/Label/Collection/Wantlist/MyWantlist/List pages. if (site['bar'] == 1 || GM_config.get('all_in_one_bar')) { $('#discogscout1_' + state + scout_tick).append(link); } else if (site['bar'] == 2) { $('#discogscout2_' + state + scout_tick).append(link); } else if (site['bar'] == 3) { $('#discogscout3_' + state + scout_tick).append(link); } } } //============================================================================== // Determine whether a site should be displayed //============================================================================== async function maybeAddLink(elem, site_name, search_url, site, scout_tick, band, release) { // Connection rate limiter per domain. var set_rate = ('rateLimit' in site) ? site['rateLimit'] : 500; var rate = (set_rate > 1000) ? set_rate : set_rate * 4; var domain = search_url.split('/')[2]; var now = (new Date())*1; var lastLoaded = window.localStorage[domain+'_lastLoaded']; if (!lastLoaded) { lastLoaded = now - 50000; } else { lastLoaded = parseInt(lastLoaded); } if (now - lastLoaded < rate) { window.setTimeout(maybeAddLink.bind(undefined, elem, site['name'], search_url, site, scout_tick, band, release), rate); return; } else { window.localStorage[domain+'_lastLoaded'] = (new Date())*1; } var success_match = ('positiveMatch' in site) ? site['positiveMatch'] : false; var target = search_url; if ('goToUrl' in site) { target = await replaceSearchUrlParams({'searchUrl': site['goToUrl'], 'spaceEncode': ('spaceEncode' in site) ? site['spaceEncode'] : '+', 'replaceSpecials': ('replaceSpecials' in site) ? site['replaceSpecials'] : false}, band, release); } // Check for results with POST method. if ('mPOST' in site) { const post_data = await replaceSearchUrlParams(site, band, release); GM.xmlHttpRequest({ method: 'POST', timeout: parseInt(GM_config.get('timeout_ms')), url: search_url, data: post_data, headers: { 'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8' }, onload: function(response) { if (GM_config.get('debug_sites')) { const name = site['name']; console.log(name + " POST Response Status: " + response.status + "\n "); console.log(name + " POST Response Headers: " + response.responseHeaders + "\n "); console.log(name + " POST Response: " + response.responseText + "\n "); } if (response.responseHeaders.indexOf('efresh: 0; url') > -1 || response.status > 499 || (response.status > 399 && !site.ignore404) || (response.responseText == "" && !site.ignoreEmpty)) { addLink(elem, site_name, target, site, 'logged_out', scout_tick, post_data); } else if (site['positiveMatch'] && site['loggedOutRegex'] && String(response.responseText).match(site['loggedOutRegex'])) { addLink(elem, site_name, target, site, 'logged_out', scout_tick, post_data); } else if (String(response.responseText).match(site['matchRegex']) ? !(success_match) : success_match) { if (!GM_config.get('hide_missing')) { addLink(elem, site_name, target, site, 'missing', scout_tick, post_data); } } else if (site['loggedOutRegex'] && String(response.responseText).match(site['loggedOutRegex'])) { addLink(elem, site_name, target, site, 'logged_out', scout_tick, post_data); } else { addLink(elem, site_name, target, site, 'found', scout_tick, post_data); } }, onerror: function() { addLink(elem, site_name, target, site, 'error', scout_tick, post_data); console.log("Discogs Scout (POST-Request Error. Site): " +site_name); }, onabort: function() { addLink(elem, site_name, target, site, 'error', scout_tick, post_data); console.log("Discogs Scout (POST-Request aborted. Site): " +site_name); }, ontimeout: function() { addLink(elem, site_name, target, site, 'error', scout_tick, post_data); console.log("Discogs Scout (POST-Request timed out. Site): " +site_name); } }); return; } // Request header tweaks let reqHeader = {}; // Check for results with GET method. GM.xmlHttpRequest({ method: 'GET', headers: reqHeader, timeout: parseInt(GM_config.get('timeout_ms')), url: search_url, onload: function(response) { if (GM_config.get('debug_sites')) { const name = site['name']; console.log(name + " GET Response Status: " + response.status + "\n "); console.log(name + " GET Response Headers: " + response.responseHeaders + "\n "); console.log(name + " GET Response: " + response.responseText + "\n "); } if (response.responseHeaders.indexOf('efresh: 0; url') > -1 || response.status > 499 || (response.status > 399 && !site.ignore404) || (response.responseText == "" && !site.ignoreEmpty)) { addLink(elem, site_name, target, site, 'logged_out', scout_tick); } else if (site['positiveMatch'] && site['loggedOutRegex'] && String(response.responseText).match(site['loggedOutRegex'])) { addLink(elem, site_name, target, site, 'logged_out', scout_tick); } else if (String(response.responseText).match(site['matchRegex']) ? !(success_match) : success_match) { if (!GM_config.get('hide_missing')) { addLink(elem, site_name, target, site, 'missing', scout_tick); } } else if (site['loggedOutRegex'] && String(response.responseText).match(site['loggedOutRegex'])) { addLink(elem, site_name, target, site, 'logged_out', scout_tick); } else { addLink(elem, site_name, target, site, 'found', scout_tick); } }, onerror: function() { addLink(elem, site_name, target, site, 'error', scout_tick); console.log("Discogs Scout (GET-Request Error. Site): " +site_name); }, onabort: function() { addLink(elem, site_name, target, site, 'error', scout_tick); console.log("Discogs Scout (GET-Request aborted. Site): " +site_name); }, ontimeout: function() { addLink(elem, site_name, target, site, 'error', scout_tick); console.log("Discogs Scout (GET-Request timed out. Site): " +site_name); } }); } //============================================================================== // Perform code for sites //============================================================================== function perform(elem, band, release, scout_tick) { let site_shown = false; $.each(icon_sites, async function(index, site) { if (site['show']) { site_shown = true; const searchUrl = await replaceSearchUrlParams(site, band, release, true); const post_data = await replaceSearchUrlParams(site, band, release, false); // run on non-post sites too to keep order of icons addLink(elem, site['name'], searchUrl, site, 'found', scout_tick, post_data); } }); $.each(sites, async function(index, site) { if (site['show']) { site_shown = true; var searchUrl = await replaceSearchUrlParams(site, band, release, true); if ('goToUrl' in site && GM_config.get('auto_search')) { maybeAddLink(elem, site['name'], searchUrl, site, scout_tick, band, release); } if ('goToUrl' in site && !GM_config.get('auto_search')) { searchUrl = await replaceSearchUrlParams({'searchUrl': site['goToUrl'], 'spaceEncode': ('spaceEncode' in site) ? site['spaceEncode'] : '+', 'replaceSpecials': ('replaceSpecials' in site) ? site['replaceSpecials'] : false}, band, release); addLink(elem, site['name'], searchUrl, site, 'found', scout_tick); } if (!('goToUrl' in site) && GM_config.get('auto_search')) { maybeAddLink(elem, site['name'], searchUrl, site, scout_tick, band, release); } if (!('goToUrl' in site) && !GM_config.get('auto_search')){ const post_data = await replaceSearchUrlParams(site, band, release, false); // run on non-post sites too to keep order of icons addLink(elem, site['name'], searchUrl, site, 'found', scout_tick, post_data); } } }); // Open settings if no sites selected: if (!site_shown) { GM_config.open(); } } //============================================================================== // Artist Page code //============================================================================== async function performArtist() { const band = $('meta[property="og\:title"]').attr('content').replace(/\(\d+\)/, '').trim(); // Wait for dynamic content to be loaded: await sleep(1000); if($('.textWithCoversRow_3IhZ3').length == 0) { await sleep(1000); if($('.textWithCoversRow_3IhZ3').length == 0) { await sleep(1000); } } if($('.textWithCoversRow_3IhZ3').length !== 0) { $('.textWithCoversRow_3IhZ3').each(function() { const elem = $(this); const release = $(this).find('.title_oY1q1>.link_1ctor').text(); let scout_tick = window.localStorage['_discogscout_tick']; if (!scout_tick) { scout_tick = 1; window.localStorage['_discogscout_tick'] = scout_tick; } perform(elem, band, release, scout_tick); scout_tick = parseInt(scout_tick) + 1; window.localStorage['_discogscout_tick'] = scout_tick; }); startObserver2(); } else { GM.notification("Artist page code error or No releases!", "Discogs Scout"); console.log("Discogs Scout: Artist page code error or No releases!"); } } function startObserver2() { console.log('Discogs Scout: Starting Observer2.'); if ($('.releasesPath_2MuQ7').length) { const obscfg = {childList: true}; const obs = new MutationObserver(start_performArtist); obs.observe($('.releasesPath_2MuQ7')[0], obscfg); } } function start_performArtist(mutation, observer) { console.log('Discogs Scout: Observer2 is triggered.'); observer.disconnect(); performArtist(); } //============================================================================== // Release Page code //============================================================================== async function performRelease() { const elem = $('[class^=body]'); // This won't work properly if " - " is in band's name: // const title = $('meta[property="og\:title"]').attr('content').trim(); // const band = title.replace(/ - .+/, '').trim(); // const release = title.replace(/.+? - /, '').trim(); let band = ""; let release = ""; if (Boolean(location.href.match('/master/'))) { if ($('[id*=profile_title]').length > 0) { // the old version of the master page band = $('[id*=profile_title]').find('a').text().trim(); release = $('[id*=profile_title]').children().last().text().trim(); } else if ($('#master_schema').length > 0) { // the new version of the master page (beta) band = $('[class^=body]').find('h1>span>a.link_15cpV:first').text().trim(); release = JSON.parse(document.getElementById('master_schema').textContent)['@graph'][0]['name']; } } else if (Boolean(location.href.match('/release/'))) { if ($('#release_schema').length > 0) { band = $('[class^=body]').find('h1>span>a.link_15cpV:first').text().trim(); release = JSON.parse(document.getElementById('release_schema').textContent)['name']; } } if (band == "" || release == "") { GM.notification("Release code error!", "Discogs Scout"); console.log("Discogs Scout: Release code error!"); return; } perform(elem, band, release); } //============================================================================== // Label/Collection/Wantlist/List Page code //============================================================================== function performList() { if($('.shortcut_navigable').length !== 0) { $('.shortcut_navigable').each(function() { const elem = $(this); let band = ""; let release = ""; if (onLabelPage) { // https://www.discogs.com/label/34268-Roadrunner-Records band = $(this).find('.artist>a').text().trim(); release = $(this).find('.title>a' ).text().trim(); } else if (onWantlistPage || onCollectionPage) { // https://www.discogs.com/wantlist?user=GPX // https://www.discogs.com/user/GPX/collection band = $(this).find('.release_title>a:eq(0)').text().trim(); if ($(this).find('.release_title>a[href*="/release/"]').length > 0) { release = $(this).find('.release_title>a[href*="/release/"]:eq(0)').text().trim(); } else if ($(this).find('.release_title>a[href*="/master/"]').length > 0) { release = $(this).find('.release_title>a[href*="/master/"]:eq(0)').text().trim(); } } else if (onMyCollectionPage) { // own collection page band = $(this).find('.collection-card-title>a:eq(0)').text().trim(); if ($(this).find('.collection-card-title>a[href*="/release/"]').length > 0) { release = $(this).find('.collection-card-title>a[href*="/release/"]:eq(0)').text().trim(); } else if ($(this).find('.collection-card-title>a[href*="/master/"]').length > 0) { release = $(this).find('.collection-card-title>a[href*="/master/"]:eq(0)').text().trim(); } } else if (onMyWantlistPage) { // own wantlist page // https://www.discogs.com/mywantlist band = $(this).find('.release_title>a:eq(0)').text().trim(); if ($(this).find('.release_title_link>a[href*="/release/"]').length > 0) { release = $(this).find('.release_title_link>a[href*="/release/"]:eq(0)').text().trim(); } else if ($(this).find('.release_title_link>a[href*="/master/"]').length > 0) { release = $(this).find('.release_title_link>a[href*="/master/"]:eq(0)').text().trim(); } } else if (onListPage) { // This won't work properly if " - " is in band's name or there are more than one artist in title: const title = $(this).find('.listitem_title>a:eq(0)').text().trim(); band = title.replace(/ - .+/, '').trim(); release = title.replace(/.+? - /, '').trim(); } if (band == "" || release == "") { GM.notification("List page code error 1!", "Discogs Scout"); console.log("Discogs Scout: List page code error 1!"); Discogs_Scout__Generate_Not_Defined_Error_To_Stop_The_Script(); } let scout_tick = window.localStorage['_discogscout_tick']; if (!scout_tick) { scout_tick = 1; window.localStorage['_discogscout_tick'] = scout_tick; } perform(elem, band, release, scout_tick); scout_tick = parseInt(scout_tick) + 1; window.localStorage['_discogscout_tick'] = scout_tick; }); } else { GM.notification("List page code error 2!", "Discogs Scout"); console.log("Discogs Scout: List page code error 2!"); return; } } function sleep(ms) { return new Promise(resolve => setTimeout(resolve, ms)); } //============================================================================// //================================ MAIN ====================================// //============================================================================// //============================================================================== // Polyfill for GM3 notifications //============================================================================== if (typeof GM.notification === "undefined") { this.GM_notification = function(options) { const opts = {}; if (typeof options === "string") { opts.text = options; opts.title = arguments[1]; opts.image = arguments[2]; opts.onclick = arguments[3]; } else { Object.keys(options).forEach(function(key) { opts[key] = options[key]; }); } checkPermission(); function checkPermission() { if (Notification.permission === "granted") { fireNotice(opts); } else if (Notification.permission === "denied") { alert("User has denied notifications for this page/site!"); // eslint-disable-next-line no-useless-return return; } else { Notification.requestPermission(function(permission) { console.log("New permission: ", permission); checkPermission(); }); } } function fireNotice(ntcOptions) { if (ntcOptions.text && !ntcOptions.body) { ntcOptions.body = ntcOptions.text; } var ntfctn = new Notification(ntcOptions.title, ntcOptions); if (ntcOptions.onclick) { ntfctn.onclick = ntcOptions.onclick; } if (ntcOptions.timeout) { setTimeout(function() { ntfctn.close(); }, ntcOptions.timeout); } } }; GM.notification = GM_notification; } //============================================================================== // Settings Menu (GM_config) //============================================================================== // To have consistent spacing in different browsers. var set_cfg_iconsize_spacing = "   "; var timeout_ms_spacing = " "; if (navigator.userAgent.toLowerCase().indexOf('firefox') > -1) { set_cfg_iconsize_spacing = "  "; timeout_ms_spacing = ""; } var config_fields = { 'aftertitle': { 'section': ' ', 'label': '  ', 'type': 'hidden' }, 'mod_icons_size': { 'label': 'Size of the icons (pixels):    ', 'type': 'text', 'default': '32' }, 'iconsborder_size': { 'label': 'Size of the icons border:     ', 'type': 'select', 'options': ['2px', '3px', '4px', '5px', '6px'], 'default': '3px' }, 'cfg_iconsize': { 'label': 'Size of the settings icons:' + set_cfg_iconsize_spacing, 'type': 'text', 'default': '22' }, 'timeout_ms': { 'label': 'Timeout requests after:      ' + timeout_ms_spacing, 'type': 'select', 'options': ['10000 ms', '20000 ms', '30000 ms', '45000 ms', '60000 ms'], 'default': '30000 ms' }, 'debug_sites': { 'type': 'checkbox', 'label': 'Debug (the searchable sites)?', 'default': false }, 'auto_search': { 'type': 'checkbox', 'label': 'Auto-search sites for results?', 'default': true }, 'hide_missing': { 'type': 'checkbox', 'label': "Hide link if search didn't found results?", 'default': false }, 'greybackground_view': { 'type': 'checkbox', 'label': 'Enable grey background for the links?', 'default': true }, 'all_in_one_bar': { 'type': 'checkbox', 'label': 'Put all links into one bar?', 'default': false }, 'run_artistpages': { 'type': 'checkbox', 'label': 'Enable the script on Artist/Label/Collection/Wantlist/List pages?', 'default': false }, 'remove_ep': { 'type': 'checkbox', 'label': 'Remove "EP", "E.P." & "E.P" from the end of release titles?', 'default': true }, 'highlight_sites': { 'label': 'Highlight sites:      ', 'type': 'text', 'default': 'RED,OPS' } }; //============================================================================== // Add sites to Settings (GM_config) //============================================================================== $.each(icon_sites, function(index, site) { config_fields['show_' + site['name']] = { 'section': (index == 0) ? ['Icon sites (no search):'] : '', 'type': 'checkbox', 'label': ' ' + site['name'] }; }); $.each(public_sites, function(index, site) { config_fields['show_' + site['name']] = { 'section': (index == 0) ? ['Public download sites:'] : '', 'type': 'checkbox', 'label': ' ' + site['name'] }; }); $.each(private_sites, function(index, site) { config_fields['show_' + site['name']] = { 'section': (index == 0) ? ['Private download sites:'] : '', 'type': 'checkbox', 'label': ' ' + site['name'] }; }); $.each(other_sites, function(index, site) { config_fields['show_' + site['name']] = { 'section': (index == 0) ? ['Other sites/tools:'] : '', 'type': 'checkbox', 'label': ' ' + site['name'] }; }); $.each(pre_databases, function(index, site) { config_fields['show_' + site['name']] = { 'section': (index == 0) ? ['Pre databases:'] : '', 'type': 'checkbox', 'label': ' ' + site['name'] }; }); //============================================================================== // Initialize and register GM_config //============================================================================== GM_config.init({ 'id': 'discogs_scout', 'title': 'Discogs Scout Settings', 'fields': config_fields, 'css': `#discogs_scout_section_header_1, #discogs_scout_section_header_2, #discogs_scout_section_header_3, \ #discogs_scout_section_header_4, #discogs_scout_section_header_5 { \ background: #00ab00 !important; \ color: black !important; \ font-weight: bold !important; \ border: 0px !important; \ padding-left: 0px !important; \ text-align: middle !important;}\ .field_label { \ display: flex !important; \ align-items: center !important; \ font-weight: normal !important;}\ .config_var { \ margin-top: 2px !important; \ margin-bottom: 2px !important; \ display: flex !important; \ align-items: center !important;}\ #discogs_scout_aftertitle_var { \ margin-top: 0px !important; \ margin-bottom: 0px !important;}\ input { \ margin-top: 0px !important; \ margin-bottom: 0px !important;}\ .grey_link { \ margin-left: 4px !important;}\ #discogs_scout_section_header_0 { \ font-weight: bold !important; \ border: 0px !important; \ margin-top: 0px !important; \ background: #bfbfbf !important;}\ #discogs_scout_header { \ background: black !important; \ color: white !important;}\ #discogs_scout_section_0 { \ margin-top: 0px !important;}`, 'events': { 'open': function() { // Iframe position. this.frame.style.top = '50px'; this.frame.style.left = 'auto'; this.frame.style.right = '150px'; this.frame.style.height = '90%'; this.frame.style.width = '450px'; $('#discogs_scout').contents().find('input#discogs_scout_field_mod_icons_size').attr('size', '1'); $('#discogs_scout').contents().find('input#discogs_scout_field_cfg_iconsize').attr('size', '1'); const modVersion = 'Discogs Scout v' + GM.info.script.version; const modUrl = 'https://greasyfork.org/en/scripts/439452-discogs-scout'; $('#discogs_scout').contents().find('#discogs_scout_section_header_0').append($(''+modVersion+'')); $('#discogs_scout').contents().find('#discogs_scout_section_header_0').find('a').css({ 'text-decoration': 'none', 'color': '#cb0000' }); $('#discogs_scout').contents().find('#discogs_scout_section_1').find('.field_label').each(function(index, label) { var url = (icon_sites[index].goToUrl) ? new URL(icon_sites[index].goToUrl) : new URL(icon_sites[index].searchUrl); $(label).append(' ' + '' + (/www./.test(url.hostname) ? url.hostname.match(/www.(.*)/)[1] : url.hostname) + ''); $(label).prepend(getFavicon(icon_sites[index], true)); }); $('#discogs_scout').contents().find('#discogs_scout_section_2').find('.field_label').each(function(index, label) { var url = (public_sites[index].goToUrl) ? new URL(public_sites[index].goToUrl) : new URL(public_sites[index].searchUrl); $(label).append(' ' + '' + (/www./.test(url.hostname) ? url.hostname.match(/www.(.*)/)[1] : url.hostname) + ''); $(label).prepend(getFavicon(public_sites[index], true)); }); $('#discogs_scout').contents().find('#discogs_scout_section_3').find('.field_label').each(function(index, label) { var url = (private_sites[index].goToUrl) ? new URL(private_sites[index].goToUrl) : new URL(private_sites[index].searchUrl); $(label).append(' ' + '' + (/www./.test(url.hostname) ? url.hostname.match(/www.(.*)/)[1] : url.hostname) + ''); $(label).prepend(getFavicon(private_sites[index], true)); }); $('#discogs_scout').contents().find('#discogs_scout_section_4').find('.field_label').each(function(index, label) { var url = (other_sites[index].goToUrl) ? new URL(other_sites[index].goToUrl) : new URL(other_sites[index].searchUrl); $(label).append(' ' + '' + (/www./.test(url.hostname) ? url.hostname.match(/www.(.*)/)[1] : url.hostname) + ''); $(label).prepend(getFavicon(other_sites[index], true)); }); $('#discogs_scout').contents().find('#discogs_scout_section_5').find('.field_label').each(function(index, label) { var url = (pre_databases[index].goToUrl) ? new URL(pre_databases[index].goToUrl) : new URL(pre_databases[index].searchUrl); $(label).append(' ' + '' + (/www./.test(url.hostname) ? url.hostname.match(/www.(.*)/)[1] : url.hostname) + ''); $(label).prepend(getFavicon(pre_databases[index], true)); }); $('#discogs_scout').contents().find("img").css({"margin-right": "4px", "width": GM_config.get('cfg_iconsize'), "height": GM_config.get('cfg_iconsize')}); }, 'close': function() { window.location.reload(true); } } }); GM.registerMenuCommand('Discogs Scout Settings', function() {GM_config.open();}); //============================================================================== // Fetch per-site values from GM_config //============================================================================== $.each(icon_sites, function(index, site) { site['show'] = GM_config.get('show_' + site['name']); }); $.each(sites, function(index, site) { site['show'] = GM_config.get('show_' + site['name']); }); //============================================================================== // Global variables //============================================================================== // For internal use (order matters). const valid_states = [ 'found', 'missing', 'logged_out', 'error' ]; var onArtistPage = false; var onReleasePage = false; var onLabelPage = false; var onMyCollectionPage = false; var onCollectionPage = false; var onMyWantlistPage = false; var onWantlistPage = false; var onListPage = false; if (Boolean(location.href.match('/artist/'))) { // artist page? onArtistPage = true; if (Boolean(location.href.match('type='))) { if (Boolean(location.href.match('subtype=Videos'))) { onArtistPage = false; } else if (!Boolean(location.href.match('type=Releases'))) { onArtistPage = false; } } } else if (Boolean(location.href.match('/release/')) || Boolean(location.href.match('/master/'))) { // release page? onReleasePage = true; } else if (Boolean(location.href.match('/label/'))) { // label page? onLabelPage = true; } else if (Boolean(location.href.match('/collection')) && Boolean(location.href.match('/user/'))) { // collection page? onCollectionPage = true; } else if (Boolean(location.href.match('/wantlist'))) { // wantlist page? onWantlistPage = true; } else if (Boolean(location.href.match('/mywantlist'))) { // own wantlist page? onMyWantlistPage = true; } else if (Boolean(location.href.match('/lists/')) && !Boolean(location.href.match('/byuser/'))) { // list page? onListPage = true; } //============================================================================== // Stuff for /release/ pages (to start after reflow) //============================================================================== function startObserver() { console.log('Discogs Scout (Initialization): Starting Observer.'); if ($('[class^=body]').length) { addDummyElem(); const obscfg = {childList: true}; const obs = new MutationObserver(checkDummyElem); obs.observe($('[class^=body]')[0], obscfg); } else { GM.notification("Element not found! Please report it.", "Discogs Scout"); console.log("Discogs Scout (Start Error): Element not found! Please report it."); return; } } function addDummyElem() { const temp = $('').attr('id','temp_scout').css({'display':'none'}); $('[class^=body]').append(temp); setTimeout(function(){ temp.remove(); }, 2000); } function checkDummyElem(mutation, observer) { if (!$('#temp_scout').length) { observer.disconnect(); startDiscogsScout(); } } //============================================================================== // Start: Add links to sites //============================================================================== function startDiscogsScout() { // We don't want to run on these urls if (Boolean(location.href.match('/image')) || Boolean(location.href.match('/history'))) { console.log("Discogs Scout: Not starting. [Report it if you think that it should start here!]"); return; } else if (Boolean(location.href.match('layout=big')) || $('.cards_layout_large').length > 0) { if (!GM_config.get('run_artistpages') && !onReleasePage) { console.log("Discogs Scout: Not starting. [Report it if you think that it should start here!]"); return; } else { GM.notification("Large covers layout is not supported!", "Discogs Scout"); console.log("Discogs Scout: Not starting. [Report it if you think that it should start here!]"); return; } } // Check if we are on the own collection page if (onCollectionPage && $('.collection-row').length > 0) { onMyCollectionPage = true; onCollectionPage = false; } if (onReleasePage) { console.log("Discogs Scout: Starting a release page."); performRelease(); } else if (GM_config.get('run_artistpages')) { if (onArtistPage) { console.log("Discogs Scout: Starting an artist page."); performArtist(); } else if (onLabelPage) { console.log("Discogs Scout: Starting a label page."); performList(); } else if (onMyCollectionPage) { console.log("Discogs Scout: Starting a mycollection page."); performList(); } else if (onCollectionPage) { console.log("Discogs Scout: Starting a collection page."); performList(); } else if (onWantlistPage) { console.log("Discogs Scout: Starting a wantlist page."); performList(); } else if (onMyWantlistPage) { console.log("Discogs Scout: Starting a mywantlist page."); performList(); } else if (onListPage) { console.log("Discogs Scout: Starting a list page."); performList(); } else { console.log("Discogs Scout: Not starting. [Report it if you think that it should start here!]"); } } else { console.log("Discogs Scout: Not starting."); } } if (document.readyState !== 'loading') { console.log('Discogs Scout (Initialization): Document is already ready.'); if (onReleasePage) { startObserver(); } else { startDiscogsScout(); } } else { console.log('Discogs Scout (Initialization): Waiting for "DOMContentLoaded" event.'); if (onReleasePage) { document.addEventListener('DOMContentLoaded', startObserver); } else { document.addEventListener('DOMContentLoaded', startDiscogsScout); } }