[GMT] Flexible Search Links

Appends versatile search links bar to linkbar

// ==UserScript==
// @name         [GMT] Flexible Search Links
// @namespace    https://greasyfork.org/users/321857-anakunda
// @version      1.62.0
// @description  Appends versatile search links bar to linkbar
// @author       Anakunda
// @copyright    © 2025 Anakunda (https://greasyfork.org/users/321857)
// @license      GPL-3.0-or-later
// @match        https://*/torrents.php?id=*
// @match        https://*/artist.php?id=*
// @match        https://*/requests.php?action=view&id=*
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_deleteValue
// @grant        GM_listValues
// @grant        GM_registerMenuCommand
// @require      https://openuserjs.org/src/libs/Anakunda/libCtxtMenu.min.js
// ==/UserScript==

'use strict';

if (document.querySelector('div.sidebar > div.box_artists') == null) return; // not a music category
const header = document.querySelector('div#content div.header');
if (header == null) throw 'Unexpected page structure';
for (let fn of ['GM_getValue', 'GM_setValue'/*, 'GM_listValues'*/])
    if (!(fn in window)) throw 'GM extensions not available';

if (typeof GM_deleteValue == 'function') {
    var menu = document.createElement('menu');
    menu.type = 'context';
    menu.id = 'context-1d19ca90-5242-418a-b6d3-d9a9fd5e5cfc';
    menu.innerHTML = '<menuitem label="Remove this link" icon="" /><menuitem label="-" />';
    menu.deleter = function(searchLinks, branch) {
        if (typeof searchLinks != 'object') throw 'Invalid argument (searchLinks)';
        if (!branch) throw 'Invalid argument (branch)';
        if (!(this.invoker instanceof HTMLAnchorElement)) throw 'Invoker not set';
        if (!(this.invoker.textContent in searchLinks)) {
            console.debug('searchLinks:', Object.keys(searchLinks), this.invoker.textContent);
            throw '"' + this.invoker.textContent + '" not a key of searchLinks';
        }
        if (!confirm('Are you sure to remove ' + this.invoker.textContent + ' from search bar?')) return false;
        delete searchLinks[this.invoker.textContent];
        if (this.invoker.nextSibling != null && this.invoker.nextSibling.nodeType == 3) this.invoker.nextSibling.remove();
            else if (this.invoker.previousSibling.textContent == divisor) this.invoker.previousSibling.remove();
        this.invoker.remove();
        GM_setValue(branch, searchLinks);
        alert('Site was removed. To restore it, use reset command from script\'s submenu');
    }.bind(menu);
    menu.callerSetter = function(evt) { this.invoker = evt.currentTarget }.bind(menu);
    document.body.append(menu);
}

const searchBox = document.createElement('div'), divisor = ' | ';
searchBox.className = 'searchbox';
searchBox.style = 'text-align: center; padding-bottom: 5px; margin-top: -1px;';
const fileName = document.location.pathname.replace(/^.*\//, '').toLowerCase();
let searchLinks = GM_getValue(fileName);

String.prototype.toASCII = function() { return this.normalize('NFKD').replace(/[\x00-\x1F\u0300-\u036F]/gu, '') };

switch (fileName) {
    case 'torrents.php':
    case 'requests.php': {
        const defaultSearchLinks = fileName == 'torrents.php' ? {
            'Orpheus': 'https://orpheus.network/torrents.php?action=advanced&artistname=${real_artists}&groupname=${album}',
            'RuTracker': 'https://rutracker.org/forum/tracker.php?nm=${real_artists_quoted}+${album_quoted}',
            'Google': 'https://www.google.com/search?q=${artists_quoted}+${album_quoted}+${year}',
            'Google (Images)': 'https://www.google.com/search?q=${artists_quoted}+${album_quoted}+${year}&tbm=isch',
            'Wikipedia': 'https://www.wikipedia.org/w/index.php?search=${artists}+${album}',
            'Discogs': 'https://www.discogs.com/search/?title=${album}&artist=${real_artists}&type=all&layout=med',
            'Discogs (title/year)': 'https://www.discogs.com/search/?title=${album}&year=${year}&type=all&layout=med',
			'MusicBrainz': 'https://musicbrainz.org/search?query=artistname:${artist_quoted} AND release:${album_quoted}&type=release_group&method=advanced',
            'AllMusic': 'https://www.allmusic.com/search/all/${real_artists_quoted}%20${album_quoted}',
            'Rate Your Music': 'https://rateyourmusic.com/search?searchterm=${real_artists_quoted}+${album_quoted}&searchtype=l',
            'Album of the Year': 'https://www.albumoftheyear.org/search/?q=${real_artists}+${album}',
			'Apple Music': 'https://music.apple.com/search?term=${artists}+${album}',
            'Deezer': 'https://www.deezer.com/search/${artists_quoted}%20${album_quoted}/album',
            'Spotify': 'https://open.spotify.com/search/${artists_quoted}%20${album_quoted}',
            'Tidal': 'https://listen.tidal.com/search/albums?q=${real_artists_quoted}+${album_quoted}',
            'Qobuz': 'https://www.qobuz.com/search?q=${real_artist}+${album}&i=boutique',
            'HighResAudio': 'https://www.highresaudio.com/en/search/?artist=${real_artist_quoted}&album=${album_quoted}&sort=-releaseDate',
            'Bandcamp': 'https://bandcamp.com/search?q=${real_artist_quoted}+${album_quoted}&item_type=a',
            'Mora': 'https://mora.jp/search/top?keyWord=${real_artists_quoted}+${album_quoted}',
            'e-onkyo': 'https://www.e-onkyo.com/search/search.aspx?q=${real_artists_quoted}+${album_quoted}',
            '7digital': 'https://uk.7digital.com/search?q=${real_artists_quoted}+${album_quoted}',
            'Boomkat': 'https://boomkat.com/products?q[keywords]=${album_quoted}',
            'Bleep': 'https://bleep.com/search/query?q=${album}',
            'SoundCloud': 'https://soundcloud.com/search/albums?q=${real_artists_quoted}+${album_quoted}',
            'Amazon Music': 'https://music.amazon.com/search/${real_artists_quoted}%20${album_quoted}',
            'YouTube Music': 'https://music.youtube.com/search?q=${real_artists_quoted}%20${album_quoted}',
            'Presto Jazz': 'https://www.prestomusic.com/jazz/search?search_query=${real_artists_quoted}%20${album_quoted}',
            'Presto Classical': 'https://www.prestomusic.com/classical/search?search_query=${real_artists_quoted}%20${album_quoted}',
            'ProStudioMasters': 'https://www.prostudiomasters.com/search?cs=1&q=${real_artists_quoted}+${album_quoted}',
            'Acoustic Sounds': 'https://store.acousticsounds.com/index.cfm?get=results&Artist=${real_artists}&Album=${album}',
            'Beatport': 'https://www.beatport.com/search/releases?q=${real_artists_quoted}+${album_quoted}',
            'Beatsource': 'https://www.beatsource.com/search?q=${real_artists_quoted}+${album_quoted}',
            'Juno Download': 'https://www.junodownload.com/search/?solrorder=relevancy&q%5Ball%5D%5B%5D=${real_artists}%20${album}',
            'Traxsource': 'https://www.traxsource.com/search/titles?term=${real_artists_quoted}+${album_quoted}',
            'Last.fm': 'https://www.last.fm/search?q=${real_artists_quoted}+${album_quoted}',
            'OTOTOY': 'https://ototoy.jp/find/?q=${album_quoted}',
            'Recochoku (レコチョク)': 'https://recochoku.jp/search/all?q=${real_artist}+${album}',
            'NetEase': 'https://music.163.com/#/search/m/?s=${real_artists_quoted}%20${album_quoted}&type=10',
            'QQ音乐': 'https://y.qq.com/portal/search.html#t=album&w=${real_artists_quoted}%20${album_quoted}',
        } : {
            'Orpheus': 'https://orpheus.network/torrents.php?action=advanced&artistname=${real_artists}&groupname=${album}',
            'RuTracker': 'https://rutracker.org/forum/tracker.php?nm=${real_artists_quoted}+${album_quoted}',
            'Google': 'https://www.google.com/search?q=${artists_quoted}+${album_quoted}+${year}',
            'Google (Images)': 'https://www.google.com/search?q=${artists_quoted}+${album_quoted}+${year}&tbm=isch',
            'Wikipedia': 'https://www.wikipedia.org/w/index.php?search=${artists}+${album}',
            'Discogs': 'https://www.discogs.com/search/?title=${album}&artist=${real_artists}&type=all&layout=med',
            'Discogs (title/year)': 'https://www.discogs.com/search/?title=${album}&year=${year}&type=all&layout=med',
            'Discogs (label/cat№)': 'https://www.discogs.com/search/?label=${label}&catno=${cat_no}&type=all&layout=med',
			'MusicBrainz': 'https://musicbrainz.org/search?query=artistname:${artist_quoted} AND release:${album_quoted}&type=release_group&method=advanced',
            'AllMusic': 'https://www.allmusic.com/search/all/${real_artists_quoted}%20${album_quoted}',
            'Rate Your Music': 'https://rateyourmusic.com/search?searchterm=${real_artists_quoted}+${album_quoted}&searchtype=l',
            'Album of the Year': 'https://www.albumoftheyear.org/search/?q=${real_artists_quoted}+${album_quoted}',
			'Apple Music': 'https://music.apple.com/search?term=${artists}+${album}',
            'Deezer': 'https://www.deezer.com/search/${artists_quoted}%20${album_quoted}/album',
            'Spotify': 'https://open.spotify.com/search/${artists_quoted}%20${album_quoted}',
            'Tidal': 'https://listen.tidal.com/search/albums?q=${real_artists_quoted}+${album_quoted}',
            'Qobuz': 'https://www.qobuz.com/search?q=${real_artist}+${album}&i=boutique',
            'HighResAudio': 'https://www.highresaudio.com/en/search/?artist=${real_artist_quoted}&album=${album_quoted}&sort=-releaseDate',
            'Bandcamp': 'https://bandcamp.com/search?q=${real_artist_quoted}+${album_quoted}&item_type=a',
            'Mora': 'https://mora.jp/search/top?keyWord=${real_artists_quoted}+${album_quoted}',
            'e-onkyo': 'https://www.e-onkyo.com/search/search.aspx?q=${real_artists_quoted}+${album_quoted}',
            '7digital': 'https://uk.7digital.com/search?q=${real_artists_quoted}+${album_quoted}',
            'Boomkat': 'https://boomkat.com/products?q[keywords]=${album_quoted}',
            'Bleep': 'https://bleep.com/search/query?q=${album}',
            'SoundCloud': 'https://soundcloud.com/search/albums?q=${real_artists_quoted}+${album_quoted}',
            'Amazon Music': 'https://music.amazon.com/search/${real_artists_quoted}%20${album_quoted}',
            'YouTube Music': 'https://music.youtube.com/search?q=${real_artists_quoted}%20${album_quoted}',
            'Presto Jazz': 'https://www.prestomusic.com/jazz/search?search_query=${real_artists_quoted}%20${album_quoted}',
            'Presto Classical': 'https://www.prestomusic.com/classical/search?search_query=${real_artists_quoted}%20${album_quoted}',
            'ProStudioMasters': 'https://www.prostudiomasters.com/search?cs=1&q=${real_artists_quoted}+${album_quoted}',
            'Acoustic Sounds': 'https://store.acousticsounds.com/index.cfm?get=results&Artist=${real_artists}&Album=${album}',
            'Beatport': 'https://www.beatport.com/search/releases?q=${real_artists_quoted}+${album_quoted}',
            'Beatsource': 'https://www.beatsource.com/search?q=${real_artists_quoted}+${album_quoted}',
            'Juno Download': 'https://www.junodownload.com/search/?solrorder=relevancy&q%5Ball%5D%5B%5D=${real_artists}%20${album}',
            'Traxsource': 'https://www.traxsource.com/search/titles?term=${real_artists_quoted}+${album_quoted}',
            'Last.fm': 'https://www.last.fm/search?q=${real_artists_quoted}+${album_quoted}',
            'OTOTOY': 'https://ototoy.jp/find/?q=${album_quoted}',
            'Recochoku (レコチョク)': 'https://recochoku.jp/search/all?q=${real_artist}+${album}',
            'NetEase': 'https://music.163.com/#/search/m/?s=${real_artists_quoted}%20${album_quoted}&type=10',
            'QQ音乐': 'https://y.qq.com/portal/search.html#t=album&w=${real_artists_quoted}%20${album_quoted}',
        };
        if (typeof searchLinks != 'object') GM_setValue(fileName, searchLinks = defaultSearchLinks);
        //console.debug('searchLinks:', searchLinks);
        if (typeof GM_registerMenuCommand == 'function' && typeof GM_deleteValue == 'function')
            GM_registerMenuCommand('Reset links to default', function() {
                if (!confirm('Are you sure to discard current configuration?')) return;
                GM_setValue(fileName, searchLinks = defaultSearchLinks);
                if (header.querySelector('div.searchbox') == null) header.append(searchBox);
                searchBox.build();
            }, 'R');
        if (Object.keys(searchLinks) <= 0) return;
        menu.onclick = evt => menu.deleter(searchLinks, fileName);
        let full_title = header.querySelector('h2 > span:last-of-type');
        if (full_title != null) {full_title = full_title.textContent.trim()}else{
            const opsTitle = header.querySelector('h2 >a:last-of-type');
            if (opsTitle) {full_title = opsTitle.textContent.trim()} else throw 'Unexpected page structure';}
        const title = full_title.replace(/\s+(?:EP|E\.\s?P\.|\(EP\)|\(E\.\s?P\.\)|-\s*EP|-\s*E\.\s?P\.|\(Live\)|- Live)$/, '');
        let albumArtist = header.querySelector('div.header > h2 > a:first-of-type');
        if (albumArtist != null) albumArtist = albumArtist.textContent.trim();
        let releaseType, label, cat_no;
        switch (fileName) {
            case 'torrents.php':
                releaseType = header.querySelector('div.header > h2');
                if (releaseType != null) releaseType = /\[([^\[\]]*)\]$/.exec(releaseType.textContent.trim());
                releaseType = releaseType != null && releaseType[1] || undefined;
                if (/^\d{4}/.test(releaseType)) releaseType = releaseType.slice(5);
                label = cat_no = '';
                break;
            case 'requests.php': {
                function getValue(label) {
                    if (label) for (let tr of document.body.querySelectorAll('div.main_column > table > tbody > tr'))
                        if ([0, 1].every(ndx => tr.children[ndx] != null)
                                && tr.children[0].textContent.trim().toLowerCase() == label.toLowerCase())
                            return tr.children[1].textContent.trim();
                    return '';
                }
                releaseType = getValue('Release type');
                label = getValue('Record label');
                cat_no = getValue('Catalogue number');
                break;
            }
        }
        const isComp = releaseType == 'Compilation', VA = 'Various Artists';
        let year = header.querySelector('h2');
        if (year != null) year = /\[(\d{4})\]/.exec(year.lastChild.textContent);
        year = year != null ? parseInt(year[1]) : '????';
		console.assert(year >= 1900 && year < 1e4, 'year >= 1900 && year < 1e4');
        const mainArtists = Array.from(document.querySelectorAll((fileName == 'torrents.php' ?
			'ul#artist_list > li.artist_main' : 'ul > li.artists_main') + ' > a[href]')).map(a => a.textContent.trim());
		const metaNames = {
			artist: isComp ? VA : mainArtists.length > 0 ? mainArtists[0] : '',
			real_artist: isComp ? '' : mainArtists.length > 0 ? mainArtists[0] : '',
			artists: isComp ? VA : mainArtists.slice(0, 3).join(' '),
			real_artists: isComp ? '' : mainArtists.slice(0, 3).join(' '),
			all_artists: isComp ? VA : mainArtists.join(' '),
			album_artist: isComp ? VA : albumArtist ? albumArtist : '',
			real_album_artist: isComp ? '' : albumArtist ? albumArtist : '',
			album: title,
			release_type: releaseType ? releaseType : '',
		};
		for (let quoted of [false, true]) for (let asc of [false, true])
			for (let raw of [false, true]) for (let key in metaNames) {
				let value = metaNames[key];
				if (asc) { value = value.toASCII(); key += '_asc'; }
				if (quoted) { value = '"' + value + '"'; key += '_quoted'; }
				if (raw) key = 'raw_' + key; else value = encodeURIComponent(value);
				//console.debug('Metaname:', key, 'Value:', value);
				window.eval(`${key} = unescape('${escape(value)}')`);
			}
        (searchBox.build = function() {
            this.textContent = 'Lookup on: ';
            for (let key in searchLinks) {
                if (this.lastChild.nodeName == 'A') this.append(divisor);
                let a = document.createElement('A');
                a.textContent = key;
                try { a.href = eval('`' + searchLinks[key] + '`') } catch(e) {
                    console.error('Invalid URL format for', key, searchLinks[key], e);
                    continue;
                }
                a.target = '_blank';
                if (typeof GM_deleteValue == 'function' && menu instanceof HTMLElement) {
                    a.setAttribute('contextmenu', menu.id);
                    a.oncontextmenu = menu.callerSetter;
                }
				a.style = 'white-space: nowrap;';
                this.append(a);
            }
        }.bind(searchBox))();
        header.append(searchBox);
        break;
    }
    case 'artist.php': {
        const defaultSearchLinks = {
            'Orpheus': 'https://orpheus.network/artist.php?artistname=${artist}',
            'RuTracker': 'https://rutracker.org/forum/tracker.php?nm=${artist_quoted}',
            'Google': 'https://www.google.com/search?q=${artist_quoted}',
            'Google (Images)': 'https://www.google.com/search?q=${artist_quoted}&tbm=isch',
            'Wikipedia': 'https://www.wikipedia.org/w/index.php?search=${artist_quoted}',
            'Discogs': 'https://www.discogs.com/search/?q=${artist}&type=artist&layout=med',
            'MusicBrainz': 'https://musicbrainz.org/search?query=${artist_quoted}&type=artist',
            'AllMusic': 'https://www.allmusic.com/search/artists/${artist}',
            'Rate Your Music': 'https://rateyourmusic.com/search?searchterm=${artist_quoted}&searchtype=a',
            'Album of the Year': 'https://www.albumoftheyear.org/search/?q=${artist}',
            'Artist Info': 'https://music.metason.net/artistinfo?name=${artist}',
            'Apple Music': 'https://music.apple.com/search?term=${artist_quoted}',
            'Deezer': 'https://www.deezer.com/search/${artist}/artist',
            'Spotify': 'https://open.spotify.com/search/${artist_quoted}',
            'Tidal': 'https://listen.tidal.com/search/artists?q=${artist_quoted}',
            'Qobuz': 'https://www.qobuz.com/search?q=${artist}&i=boutique',
            'HighResAudio': 'https://www.highresaudio.com/en/search/?artist=${artist_quoted}',
            'Bandcamp': 'https://bandcamp.com/search?q=${artist_quoted}&item_type=b',
            'Mora': 'https://mora.jp/search/top?keyWord=${artist_quoted}',
            'e-onkyo': 'https://www.e-onkyo.com/search/search.aspx?q=${artist_quoted}',
            '7digital': 'https://uk.7digital.com/search?q=${artist_quoted}',
            'Boomkat': 'https://boomkat.com/products?q[keywords]=${artist_quoted}',
            'Bleep': 'https://bleep.com/search/query?q=${artist}',
            'SoundCloud': 'https://soundcloud.com/search/people?q=${artist_quoted}',
            'Amazon Music': 'https://music.amazon.com/search/${artist_quoted}',
            'YouTube Music': 'https://music.youtube.com/search?q=${artist_quoted}',
            'Presto Jazz': 'https://www.prestomusic.com/jazz/search?search_query=${artist_quoted}',
            'Presto Classical': 'https://www.prestomusic.com/classical/search?search_query=${artist_quoted}',
            'ProStudioMasters': 'https://www.prostudiomasters.com/search?cs=1&q=${artist_quoted}',
            'Acoustic Sounds': 'https://store.acousticsounds.com/index.cfm?get=results&Artist=${artist}',
            'Beatport': 'https://www.beatport.com/search/artists?q=${artist_quoted}',
            'Beatsource': 'https://www.beatsource.com/search?q=${artist_quoted}',
            'Juno Download': 'https://www.junodownload.com/search/?solrorder=relevancy&q%5Ball%5D%5B%5D=${artist}',
            'Traxsource': 'https://www.traxsource.com/search/artists?term=${artist_quoted}',
            'Last.fm': 'https://www.last.fm/search/artists?q=${artist_quoted}',
            'OTOTOY': 'https://ototoy.jp/find/?q=${artist_quoted}',
            'Recochoku (レコチョク)': 'https://recochoku.jp/search/artist?q=${artist}',
            'NetEase': 'https://music.163.com/#/search/m/?s=${artist_quoted}&type=100',
            'QQ音乐': 'https://y.qq.com/portal/search.html#t=artist&w=${artist_quoted}',
        };
        if (typeof searchLinks != 'object') GM_setValue(fileName, searchLinks = defaultSearchLinks);
        //console.debug('searchLinks:', searchLinks);
        if (typeof GM_registerMenuCommand == 'function' && typeof GM_deleteValue == 'function')
            GM_registerMenuCommand('Reset links to default', function() {
                if (!confirm('Are you sure to discard current configuration?')) return;
                GM_setValue(fileName, searchLinks = defaultSearchLinks);
                if (header.querySelector('div.searchbox') == null) header.append(searchBox);
                searchBox.build();
            }, 'R');
        if (Object.keys(searchLinks) <= 0) return;
        menu.onclick = evt => menu.deleter(searchLinks, fileName);
        let h2 = header.querySelector('h2');
        if (h2 == null) throw 'Unexpected page structure';
        const artist = encodeURIComponent(h2.textContent.trim()),
                    artist_asc = encodeURIComponent(h2.textContent.trim().toASCII()),
                    artist_quoted = encodeURIComponent('"' + h2.textContent.trim() + '"'),
                    artist_asc_quoted = encodeURIComponent('"' + h2.textContent.trim().toASCII() + '"');
        searchBox.style.marginBottom = '1em';
        (searchBox.build = function() {
            this.textContent = 'Lookup on: ';
            for (let key in searchLinks) {
                if (this.lastChild.nodeName == 'A') this.append(divisor);
                let a = document.createElement('A');
                a.textContent = key;
                try { a.href = eval('`' + searchLinks[key] + '`') } catch(e) {
                    console.error('Invalid URL format for', key, searchLinks[key], e);
                    continue;
                }
                a.target = '_blank';
                if (typeof GM_deleteValue == 'function' && menu) {
                    a.setAttribute('contextmenu', menu.id);
                    a.oncontextmenu = menu.callerSetter;
                }
				a.style = 'white-space: nowrap;';
                this.append(a);
            }
        }.bind(searchBox))();
        header.append(searchBox);
        break;
    }
}