// ==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;
}
}