// ==UserScript== // // @name Discogs Scout // @version 1.0 // @namespace https://github.com/Purfview/Discogs-Scout // @description Auto search for music on torrent 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/* // // @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 ================================== 1.0 - First usable version. New feature: Support for Release/Master pages. New feature: Three bars for sites: 1st for non-searchable 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. ==============================================================================*/ 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': 'Google', 'icon': '', 'searchUrl': 'https://www.google.com/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': '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': 'RuT', 'icon': '', 'searchUrl': 'https://rutracker.org/forum/tracker.php?nm=%band%+%release%', 'loggedOutRegex': /Введите ваше имя/, 'matchRegex': 'Не найдено', 'bar': 2}, { 'name': 'RuT-Discography', 'icon': '', 'searchUrl': 'https://rutracker.org/forum/tracker.php?nm=%band%+Дискография', 'loggedOutRegex': /Введите ваше имя/, 'matchRegex': 'Не найдено', 'bar': 2} ]; var private_sites = [ { 'name': 'OPS', 'icon': '', 'searchUrl': 'https://orpheus.network/torrents.php?searchstr=%band%+%release%&filter_cat[1]=1', 'loggedOutRegex': /Cloudflare|Ray ID|>Remember meRemember me0 results').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; var placebo_url = new URL(target).origin; link = $('').attr('href', placebo_url).attr('onclick', "$('#" + form_name + "').submit(); return false;").attr('target', '_blank').attr('rel', 'noreferrer'); var data = (post_data.match('{')) ? post_data : '{"' + post_data.replace(/&/g, '","').replace(/=/g, '":"').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 (post_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)'); } 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)'); link.append(icon); } // Create elements on Artist pages. if (onArtistPage) { //const bar_height = (parseInt(GM_config.get('mod_icons_size')) +6) +"px"; const background = GM_config.get('greybackground_view') ? 'rgb(51, 51, 51)' : ''; if ($('.result_box_main' + scout_tick).length == 0) { $(elem).after($('').append($('',{'colspan':'9'}).addClass('result_box_main' + scout_tick))); $('.result_box_main' + scout_tick).css({'background-color': background, 'padding': '0px 4px'}); } if (site['bar'] == 1) { if ($('.result_bar_1st' + scout_tick).length == 0) { $('.result_box_main' + scout_tick).append($('
').addClass('result_bar_1st' + scout_tick)); //$('.result_bar_1st' + scout_tick).css({'height': bar_height}); $.each(valid_states, function(i, name) { $('.result_bar_1st' + scout_tick).append(""+''); }); } } if (site['bar'] == 2) { if ($('.result_bar_2nd' + scout_tick).length == 0) { $('.result_box_main' + scout_tick).append($('
').addClass('result_bar_2nd' + scout_tick)); //$('.result_bar_2nd' + scout_tick).css({'height': bar_height}); $.each(valid_states, function(i, name) { $('.result_bar_2nd' + scout_tick).append(""+''); }); } } if (site['bar'] == 3) { if ($('.result_bar_3rd' + scout_tick).length == 0) { $('.result_box_main' + scout_tick).append($('
').addClass('result_bar_3rd' + scout_tick)); //$('.result_bar_3rd' + scout_tick).css({'height': bar_height}); $.each(valid_states, function(i, name) { $('.result_bar_3rd' + scout_tick).append(""+''); }); } } // Add links to elements on Artist pages. if (site['bar'] == 1) { $('#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); } } // Create elements on Release/Master pages. if (onReleasePage) { //const bar_height = (parseInt(GM_config.get('mod_icons_size')) +6) +"px"; 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'}); } if (site['bar'] == 1) { if ($('.result_bar_1st').length == 0) { $('.result_box_main').append($('
').addClass('result_bar_1st')); //$('.result_bar_1st').css({'height': bar_height}); $.each(valid_states, function(i, name) { $('.result_bar_1st').append(""+''); }); } } if (site['bar'] == 2) { if ($('.result_bar_2nd').length == 0) { $('.result_box_main').append($('
').addClass('result_bar_2nd')); //$('.result_bar_2nd').css({'height': bar_height}); $.each(valid_states, function(i, name) { $('.result_bar_2nd').append(""+''); }); } } if (site['bar'] == 3) { if ($('.result_bar_3rd').length == 0) { $('.result_box_main').append($('
').addClass('result_bar_3rd')); //$('.result_bar_3rd').css({'height': bar_height}); $.each(valid_states, function(i, name) { $('.result_bar_3rd').append(""+''); }); } } // Add links to elements on Release/Master pages. if (site['bar'] == 1) { $('#discogscout1_' + state).append(link); } else if (site['bar'] == 2) { $('#discogscout2_' + state).append(link); } else if (site['bar'] == 3) { $('#discogscout3_' + state).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'] : 200; 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'] : '+'}, 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; var searchUrl = await replaceSearchUrlParams(site, band, release, true); addLink(elem, site['name'], searchUrl, site, 'found', scout_tick); } }); $.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'] : '+'}, 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')){ addLink(elem, site['name'], searchUrl, site, 'found', scout_tick); } } }); // Open settings if no sites selected: if (!site_shown) { GM_config.open(); } } //============================================================================== // Artist Page code //============================================================================== function performArtist() { // Check if artist has releases if (!Boolean($('.credit_type:contains("Releases")').text().match('Releases'))) { console.log("Discogs Scout: Artist page doesn't have releases! Quitting...") return; } const band = $('meta[property="og\:title"]').attr('content').replace(/\(\d+\)/, '').trim(); if($('.card').length !== 0) { $('.card').each(function() { const elem = $(this); const release = $(this).find('.title>a').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; }); } } //============================================================================== // 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(/ - .+/, '').replace(/\(\d+\)/, '').trim(); // const release = title.replace(/.+? - /, '').trim(); let band, release; if (Boolean(location.href.match('/master/'))) { band = $('#profile_title').find('a').text().replace(/\(\d+\)/, '').trim(); release = $('#profile_title').children().last().text().trim(); } else if (Boolean(location.href.match('/release/'))) { band = $('[class^=body]').find('h1>span>a.link_1ctor:first').text().replace(/\(\d+\)/, '').trim(); release = JSON.parse(document.getElementById('release_schema').textContent)['name']; } perform(elem, band, release); } function sleep(ms) { return new Promise(resolve => setTimeout(resolve, ms)); } //============================================================================// //================================ MAIN ====================================// //============================================================================// //============================================================================== // 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 }, 'run_artistpages': { 'type': 'checkbox', 'label': 'Enable the script on Artist pages?', 'default': false }, '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'] + (site['TV'] ? ' (TV)' : '') }; }); //============================================================================== // 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 { \ 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 = 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 = 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 = 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 = 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("img").css({"margin-right": "4px", "width": GM_config.get('cfg_iconsize'), "height": GM_config.get('cfg_iconsize')}); }, 'close': function() { location.reload(); } } }); 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' ]; // Are we on an artist page? var onArtistPage = Boolean(location.href.match('/artist/')); if (onArtistPage && Boolean(location.href.match('type='))) { if (Boolean(location.href.match('subtype=Videos'))) { onArtistPage = false; } else if (!Boolean(location.href.match('type=Releases'))) { onArtistPage = false; } } // Are we on a release page? var onReleasePage = false; if (!Boolean(location.href.match('/artist/'))) { if (Boolean(location.href.match('/release/')) || Boolean(location.href.match('/master/'))) { onReleasePage = true; } } //============================================================================== // Stuff for /release/ pages (to start after reflow) //============================================================================== function startObserver() { if ($('[class^=body]').length) { addDummyElem(); const obscfg = {childList: true}; const obs = new MutationObserver(checkDummyElem); obs.observe($('[class^=body]')[0], obscfg); } else { console.log("Discogs Scout (Start Error): Element not found! Please report it."); } } 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() { if (onArtistPage && GM_config.get('run_artistpages')) { console.log("Discogs Scout: Starting an artist page.") performArtist(); } else if (onReleasePage) { console.log("Discogs Scout: Starting a release page.") performRelease(); } } if (Boolean(location.href.match('/release/'))) { document.addEventListener('DOMContentLoaded', startObserver); } else { document.addEventListener('DOMContentLoaded', startDiscogsScout); }