NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name mb. INLINE STUFF // @version 2024.7.31 // @description musicbrainz.org: Release page: Inline recording names, comments, ISRC and AcoustID. Direct CAA add link if none. Highlight duplicates in releases and edits. Recording page: millisecond display, spot track length and title variations. // @namespace https://github.com/jesus2099/konami-command // @supportURL https://github.com/jesus2099/konami-command/labels/mb_INLINE-STUFF // @downloadURL https://github.com/jesus2099/konami-command/raw/master/mb_INLINE-STUFF.user.js // @author jesus2099 // @licence CC-BY-NC-SA-4.0; https://creativecommons.org/licenses/by-nc-sa/4.0/ // @licence GPL-3.0-or-later; http://www.gnu.org/licenses/gpl-3.0.txt // @since 2010-07-09; https://web.archive.org/web/20131103163358/userscripts.org/scripts/show/81127 / https://web.archive.org/web/20141011084022/userscripts-mirror.org/scripts/show/81127 / https://web.archive.org/web/20131104205636/userscripts.org/scripts/show/166877 / https://web.archive.org/web/20141011084010/userscripts-mirror.org/scripts/show/166877 // @icon data:image/gif;base64,R0lGODlhEAAQAKEDAP+/3/9/vwAAAP///yH/C05FVFNDQVBFMi4wAwEAAAAh/glqZXN1czIwOTkAIfkEAQACAwAsAAAAABAAEAAAAkCcL5nHlgFiWE3AiMFkNnvBed42CCJgmlsnplhyonIEZ8ElQY8U66X+oZF2ogkIYcFpKI6b4uls3pyKqfGJzRYAACH5BAEIAAMALAgABQAFAAMAAAIFhI8ioAUAIfkEAQgAAwAsCAAGAAUAAgAAAgSEDHgFADs= // @require https://github.com/jesus2099/konami-command/raw/ab3d205ab8a9897ac3ef23075fda26bed07ca342/lib/COOL-BUBBLES.js?version=2016.6.1.1310 // @require https://github.com/jesus2099/konami-command/raw/de88f870c0e6c633e02f32695e32c4f50329fc3e/lib/SUPER.js?version=2022.3.24.224 // @grant none // @include /^https?:\/\/(\w+\.)?musicbrainz\.org\/[^/]+\/[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}\/(open_)?edits/ // @include /^https?:\/\/(\w+\.)?musicbrainz\.org\/artist\/[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}\/recordings/ // @include /^https?:\/\/(\w+\.)?musicbrainz\.org\/edit\/\d+/ // @include /^https?:\/\/(\w+\.)?musicbrainz\.org\/edit\/subscribed/ // @include /^https?:\/\/(\w+\.)?musicbrainz\.org\/event\/[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}(?!\/(add-)?event-art)/ // @include /^https?:\/\/(\w+\.)?musicbrainz\.org\/isrc\/[a-z]{2}[a-z0-9]{3}[0-9]{2}[0-9]{5}/ // @include /^https?:\/\/(\w+\.)?musicbrainz\.org\/recording\/[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}(?!\/edit$)/ // @include /^https?:\/\/(\w+\.)?musicbrainz\.org\/release\/[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}([^/]|$|\/disc\/\d+)/ // @include /^https?:\/\/(\w+\.)?musicbrainz\.org\/search\/edits\?/ // @include /^https?:\/\/(\w+\.)?musicbrainz\.org\/user\/[^/]+\/edits/ // @run-at document-idle // ==/UserScript== "use strict"; /* - --- - --- - --- - START OF CONFIGURATION - --- - --- - --- - */ var dupeColour = "pink"; var infoColour = "lightcyan"; var contractFingerPrints = true; /* more compact AcoustIDs but brwoser can still inline search/find full AcoustID */ /* track/recording name diff: null for no marking and no inline recording disambiguation comment either, "" for no name change, %br% is return to new line feed ex.: "%track-name%*", "%track-name% (%recording-name%)" or even %recording-name% */ var markTrackRecNameDiff = "%track-name%%br%%recording-name%"; var recUseInRelationshipLink = "+relate"; /* null or delete for no such tool */ var recAddToMergeLink = "+merge"; /* null or delete for no such tool */ var formatISRC = true; // Turn USJT19900112 into coloured US-JT1-99-00112 (country-label-year-number) var trackLengthDiffInfo = 5000; // ms var trackLengthDiffWarn = 15000; // ms /* - --- - --- - --- - END OF CONFIGURATION - --- - --- - --- - */ var userjs = "jesus2099userjs81127"; var DEBUG = localStorage.getItem("jesus2099debug"); var MBS = location.protocol + "//" + location.host; var hasDupes = { ISRC: 0, AcoustID: 0 }; var shownisrcs = []; var shownacoustids = []; var shownworks = {count: 0}; var texts = { de: { add_cover_art: "Cover‐Art hinzufügen", add_event_art: "Event‐Art hinzufügen" }, en: { add_cover_art: "Add cover art", add_event_art: "Add event art" }, es: { add_cover_art: "Añadir imágenes", add_event_art: "Añadir imágenes" }, et: { add_cover_art: "Lisa kaanepiltid", add_event_art: "Lisa pildid" }, fr: { add_cover_art: "Ajouter des images", add_event_art: "Ajouter des images" }, it: { add_cover_art: "Aggiungi copertine", add_event_art: "Aggiungi immagini dell’evento" }, ja: { add_cover_art: "カバーアートを追加", add_event_art: "イベントアートを追加" }, nl: { add_cover_art: "Afbeeldingen toevoegen", add_event_art: "Afbeeldingen toevoegen" } }; var lang = document.querySelector("html[lang]"); lang = lang && lang.getAttribute("lang") || "en"; if (lang) { if (texts[lang]) { texts = texts[lang]; } else { texts = texts["en"]; } } debug(texts); var isrcURL = "/isrc/%s"; var acoustidURL = "//acoustid.org/track/%s"; var releasewsURL = "/ws/2/release/%s/?inc=recordings+isrcs"; /* http://wiki.musicbrainz.org/XMLWebService#release_resources */ var str_GUID = "[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}"; var re_GUID = new RegExp(str_GUID, "i"); var AcoustIDlinkingURL = "//acoustid.org/edit/toggle-track-mbid?track_gid=%acoustid&mbid=%mbid&state=%state"; var css_recording = "td:not(.pos):not(.video) > a[href^='/recording/'], td:not(.pos):not(.video) > :not(div):not(.ars) a[href^='/recording/']"; var css_work = "td:not(.pos):not(.video) div.ars > dl.ars > dd > a[href^='/work/'], td:not(.pos):not(.video) div.ars > dl.ars > dd > span.mp > a[href^='/work/']"; var tracksHtml = null; var page_type; if (location.pathname.match(/\/show\/edit\/|\/mod\/search\/|\/edit|\/edits|\/open_edits/i)) { page_type = "edits"; } else { var entity_type = location.pathname.match(/^\/(event|isrc|recording|release)/i); if (entity_type) { page_type = entity_type[1]; } else if (location.pathname.match(/\/recordings/i)) { page_type = "recordings"; } } debug("Page type: " + page_type); const pageMbid = location.pathname.match(re_GUID); var paginatedRelease = document.querySelector("div#content table.tbl.medium > tbody[style*='display'][style*='none'], div#content table.tbl.medium > tbody a.load-tracks"); var css = document.createElement("style"); css.setAttribute("type", "text/css"); document.head.appendChild(css); css = css.sheet; css.insertRule("div#page table.add-isrcs tbody a[href^='/isrc/'], div#page table.merge-recordings tbody a[href^='/isrc/'], div#page table.remove-isrc tbody a[href^='/isrc/'], div#page table." + userjs + "-has-isrcs tbody a[href^='/isrc/'] { white-space: nowrap !important; }", 0); css.insertRule("div#content." + userjs + "hide-recdis table.tbl.medium span." + userjs + "recdis.comment { display: none; }", 0); if (page_type) { switch (page_type) { case "release": // React hydrate clumsy workaround setTimeout(function() { // CAA tab / Add link addIMGTabLink(); // Tracklist stuff css.insertRule("a[" + userjs + "recname] { text-shadow: 1px 2px 2px #999; color: maroon }", 0); if (contractFingerPrints) { css.insertRule("div.ars[class^='ars AcoustID'] code { display: inline-block; overflow-x: hidden; vertical-align: bottom; width: 6ch}", 0); } if (pageMbid && (tracksHtml = document.querySelectorAll("div#content table.tbl.medium > tbody > tr[id]:not(.subh)")).length > 0) { if (recUseInRelationshipLink || recAddToMergeLink) { for (var ith = 0; ith < tracksHtml.length; ith++) { var toolzone = tracksHtml[ith].querySelector("td.treleases"); if (toolzone) { toolzone = toolzone.appendChild(document.createElement("div")); toolzone.className = userjs + "toolzone"; toolzone.style.setProperty("display", "none"); var rec = tracksHtml[ith].querySelector(css_recording); if (recUseInRelationshipLink && rec) { toolzone.appendChild(createA(recUseInRelationshipLink, rec.getAttribute("href") + "/relate", "Use this recording in a relationship…")); } var rat = tracksHtml[ith].querySelector("span.star-rating a.set-rating"); if (recAddToMergeLink && rat) { if (recUseInRelationshipLink) { toolzone.appendChild(document.createElement("br")); } toolzone.appendChild(createA(recAddToMergeLink, "/recording/merge_queue?add-to-merge=" + rat.getAttribute("href").match(/id=([0-9]+)/)[1], "Merge this recording…")); } toolzone = toolzone.parentNode.appendChild(document.createElement("div")); toolzone.className = userjs + "editbutt"; toolzone.style.setProperty("display", "none"); toolzone.appendChild(createA("Edit", rec.getAttribute("href") + "/edit", "Edit this recording")); toolzone = toolzone.parentNode.appendChild(document.createElement("div")); toolzone.className = userjs + "openEdits"; toolzone.style.setProperty("display", "none"); toolzone.appendChild(createA("Open", rec.getAttribute("href") + "/open_edits", "Recording open edits")); toolzone.appendChild(document.createTextNode(" ")); toolzone.appendChild(createA("edits", rec.getAttribute("href") + "/edits", "Recording edit history")); } var works = tracksHtml[ith].querySelectorAll(css_work); if (works) { for (var w = 0; w < works.length; w++) { var workid = works[w].getAttribute("href").match(new RegExp("/work/(" + str_GUID + ")$")); if (workid) { if (!shownworks[workid[1]]) { shownworks[workid[1]] = 0; shownworks.count++; } shownworks[workid[1]]++; } } } } } idCount("Track", tracksHtml.length); if (shownworks.count > 0) { idCount("Work", shownworks.count); } let xhr = new XMLHttpRequest(); xhr.onreadystatechange = isrcFish; coolBubble.info("Loading “" + document.querySelector("h1").textContent + "” shadow release…"); xhr.open("GET", MBS + releasewsURL.replace(/%s/, pageMbid), true); xhr.overrideMimeType("text/xml"); xhr.send(null); } }, 1000); break; case "event": // React hydrate clumsy workaround setTimeout(function() { // EAA tab / Add link addIMGTabLink(); }, 1000); break; case "isrc": if (formatISRC) { let ISRCcode = document.querySelector("h1 code"); ISRCcode.replaceChild(coolifyISRC(ISRCcode.textContent), ISRCcode.firstChild); } break; case "recordings": document.querySelector("div#content table.tbl").classList.add(userjs + "-has-isrcs"); // for later duplicate spot // fall through case "edits": var iedits = document.querySelectorAll("div#page table.add-isrcs, div#page table.merge-recordings, div#page table.remove-isrc, div#page table." + userjs + "-has-isrcs"); for (var ied = 0; ied < iedits.length; ied++) { shownisrcs = []; var as = iedits[ied].getElementsByTagName("a"); for (var ia = 0; ia < as.length; ia++) { var href = as[ia].getAttribute("href").match(/isrc[=/]([^?]+)$/); if (href) { as[ia].replaceChild(coolifyISRC(as[ia].textContent), as[ia].firstChild); if (shownisrcs[href[1]]) { hasDupes.ISRC++; shownisrcs[href[1]].style.setProperty("background-color", dupeColour); as[ia].style.setProperty("background-color", dupeColour); } else { shownisrcs[href[1]] = as[ia]; } } } } break; case "recording": // format sidebar ISRCs if (formatISRC) { var sideBarISRCs = document.querySelectorAll("div#sidebar dd.isrc a[href^='/isrc']"); for (var i = 0; i < sideBarISRCs.length; i++) { sideBarISRCs[i].replaceChild(coolifyISRC(sideBarISRCs[i].textContent), sideBarISRCs[i].firstChild); } } // from mb_INLINE-TRACK-ARTIST var tracks = document.querySelectorAll("div#content table.tbl > tbody > tr"); if (pageMbid && tracks.length > 0) { /* locate length, track title and release artist columns */ var lengthColumnIndex = getParent(tracks[0], "table").querySelector("thead > tr > th.treleases").cellIndex + 1; var trackTitleColumnIndex = lengthColumnIndex - 1; if (trackTitleColumnIndex && lengthColumnIndex) { let xhr = new XMLHttpRequest(); xhr.addEventListener("load", function(event) { var wsRecording = this.responseXML; if ( this.status == 200 && (wsRecording = wsRecording.documentElement) ) { var wsRecordingLength = wsRecording.querySelector("recording > length"); wsRecordingLength = wsRecordingLength ? parseInt(wsRecordingLength.textContent, 10) : 0; var trackLengthCell = document.querySelector("div#sidebar dl.properties dd.length"); if (trackLengthCell) { trackLengthCell.replaceChild(document.createTextNode(time(wsRecordingLength)), trackLengthCell.firstChild); } var wsTracks = wsRecording.querySelectorAll("recording[id='" + pageMbid + "'] > release-list > release > medium-list > medium > track-list > track"); for (var wst = 0; wst < wsTracks.length; wst++) { var wsRelease = getParent(wsTracks[wst], "release"); var wsReleaseMBID; var wsPosition = wsTracks[wst].parentNode.parentNode.querySelector("position"); var wsTrackPosition = wsTracks[wst].parentNode.getAttribute("offset"); if ( wsRelease && (wsReleaseMBID = wsRelease.getAttribute("id")) && wsPosition && (wsPosition = wsPosition.textContent) && wsTrackPosition && (wsTrackPosition = parseInt(wsTrackPosition, 10) + 1) ) { for (var t = 0; t < tracks.length; t++) { if (tracks[t].querySelector("a[href*='/release/']") && tracks[t].querySelector("a[href*='/release/']").getAttribute("href").indexOf(wsReleaseMBID) > 0 && tracks[t].querySelector("td:first-of-type").textContent.trim() == wsPosition + "." + wsTrackPosition) { /* display recording/track title discrepency */ var trackTitleCell = tracks[t].querySelector("td:nth-child(" + trackTitleColumnIndex + ")"); if (trackTitleCell) { var trackTitle = document.querySelector("h1 a"); var wsTrackTitle = wsTracks[wst].querySelector("title"); if (trackTitle && wsTrackTitle && trackTitle.textContent != wsTrackTitle.textContent) { trackTitleCell.firstChild.setAttribute("title", "≠ " + trackTitle.textContent); trackTitleCell.firstChild.classList.add("name-variation"); } } /* display recording/track length discrepency */ trackLengthCell = tracks[t].querySelector("td:nth-child(" + lengthColumnIndex + ")"); if (trackLengthCell) { var wsTrackLength = wsTracks[wst].querySelector("length"); if (wsTrackLength && (wsTrackLength = parseInt(wsTrackLength.textContent, 10))) { let trackLengthSpan = createTag("span", {}, document.createTextNode(time(wsTrackLength))); trackLengthCell.replaceChild(trackLengthSpan, trackLengthCell.firstChild); if (wsTrackLength <= wsRecordingLength - trackLengthDiffInfo || wsTrackLength >= wsRecordingLength + trackLengthDiffInfo) { trackLengthSpan.classList.add("name-variation"); trackLengthSpan.setAttribute("title", "more than " + time(trackLengthDiffInfo) + " away from " + time(wsRecordingLength)); if (wsTrackLength <= wsRecordingLength - trackLengthDiffWarn || wsTrackLength >= wsRecordingLength + trackLengthDiffWarn) { trackLengthSpan.classList.add("warn-lengths"); trackLengthSpan.setAttribute("title", "more than " + time(trackLengthDiffWarn) + " away from " + time(wsRecordingLength)); } } } } break; } } } } } else { coolBubble.error("Error " + this.status + (this.statusText ? " “" + this.statusText + "”" : "") + " while fetching inline track stuff."); } }); xhr.addEventListener("error", function(event) { coolBubble.error("Error " + this.status + (this.statusText ? " “" + this.statusText + "”" : "") + " while fetching inline track stuff."); }); coolBubble.info("Loading “" + document.querySelector("h1").textContent + "” shadow recording…"); xhr.open("get", MBS + "/ws/2/recording?query=rid:" + pageMbid, true); xhr.overrideMimeType("text/xml"); xhr.send(null); } } } } function createA(text, link, title, target) { var a = document.createElement("a"); if (typeof text == "string") { a.appendChild(document.createTextNode(text)); } else { a.appendChild(text); } if (link) { a.setAttribute("href", link); } else { a.style.setProperty("cursor", "pointer"); } if (title) { a.setAttribute("title", title); } if (target) { a.setAttribute("target", target); } return a; } function isrcFish() { if (this.readyState == 4 && this.status == 200 && tracksHtml) { if (document.body.classList.contains("MMR2099userjs120382")) { // linked to mb_MASS-MERGE-RECORDINGS coolBubble.warn("INLINE STUFF abandoned in favour of MASS MERGE session."); } else { var res = this.responseXML; var isrcNet = {}; var recnameNet = {}; var acoustidNet = []; var tracks = res.evaluate("//mb:recording", res, nsr, XPathResult.ANY_TYPE, null); var track; for (var pos = 1; (track = tracks.iterateNext()) !== null; pos++) { var trackMBID = track.getAttribute("id"); if (acoustidNet.indexOf(trackMBID) < 0) { acoustidNet.push(trackMBID); } isrcNet[trackMBID] = []; recnameNet[trackMBID] = {}; var isrcs = res.evaluate(".//mb:isrc", track, nsr, XPathResult.ANY_TYPE, null); var isrc; while ((isrc = isrcs.iterateNext()) !== null) { isrcNet[trackMBID].push(isrc.getAttribute("id")); } var recnames = res.evaluate(".//mb:title/text()", track, nsr, XPathResult.ANY_TYPE, null); var recname; while ((recname = recnames.iterateNext()) !== null) { recnameNet[trackMBID].name = recname.textContent; } var recdisambigs = res.evaluate(".//mb:disambiguation/text()", track, nsr, XPathResult.ANY_TYPE, null); var recdisambig; while ((recdisambig = recdisambigs.iterateNext()) !== null) { recnameNet[trackMBID].comment = recdisambig.textContent; } } acoustidFishBatch(acoustidNet); for (var i = 0; i < tracksHtml.length; i++) { var aRec = tracksHtml[i].querySelector(css_recording); if (aRec) { var mbid = aRec.getAttribute("href").match(re_GUID); if (mbid) { mbid = mbid[0]; } var trackTitleCell = tracksHtml[i].querySelector("td:not(.pos):not(.video)"); if (isrcNet[mbid].length > 0) { insertBeforeARS(trackTitleCell, createStuffFragment("ISRC", isrcNet[mbid], shownisrcs, isrcURL, null, mbid)); } var sameCompleteName = aRec.textContent == recnameNet[mbid].name + " (" + recnameNet[mbid].comment + ")"; if (aRec.textContent != recnameNet[mbid].name && !sameCompleteName) { aRec.setAttribute("title", "track name: " + aRec.textContent + "\n≠rec. name: " + recnameNet[mbid].name); if (markTrackRecNameDiff) { if (typeof markTrackRecNameDiff == "string") { var trackNameFragment = document.createDocumentFragment(); var trackNameRows = markTrackRecNameDiff.replace(/%track-name%/ig, aRec.textContent).replace(/%recording-name%/ig, recnameNet[mbid].name).split("%br%"); for (var row = 0; row < trackNameRows.length; row++) { if (row > 0) { trackNameFragment.appendChild(document.createElement("br")); } trackNameFragment.appendChild(document.createTextNode(trackNameRows[row])); } aRec.setAttribute(userjs + "recname", aRec.textContent); // linked in mb_MASS-MERGE-RECORDINGS, mb_PLAIN-TEXT-TRACKLIST, mb_COLLECTION-HIGHLIGHTER aRec.replaceChild(trackNameFragment, aRec.firstChild); } } } if (markTrackRecNameDiff && recnameNet[mbid].comment && !sameCompleteName) { // .userjsrecdis linked in mb_MASS-MERGE-RECORDINGS, mb_PLAIN-TEXT-TRACKLIST addAfter(createTag("fragment", {}, [ " ", createTag("span", {a: {class: userjs + "recdis comment"}}, "(" + recnameNet[mbid].comment + ")") ]), aRec); } } } shownisrcs = count(shownisrcs); idCount("ISRC", shownisrcs); } } else if (this.readyState == 4 && this.status > 200) { coolBubble.error("Error " + this.status + (this.statusText ? " “" + this.statusText + "”" : "") + " while fetching MB inline stuff."); } } function createStuffFragment(stufftype, stuffs, shownstuffs, url, trackid, recid) { var td = document.createElement("dd"); for (var i = 0; i < stuffs.length; i++) { var adisabled = (stufftype == "AcoustID" && stuffs[i][1]); var stuff = (stufftype == "AcoustID" ? stuffs[i][0] : stuffs[i]); if (i > 0) { td.appendChild(document.createTextNode(", ")); } var a = document.createElement("a"); a.style.setProperty("white-space", "nowrap"); a.setAttribute("href", url.replace(/%s/, stuff)); var code = document.createElement("code"); if (stufftype == "ISRC") { code = formatISRC ? coolifyISRC(stuff) : document.createTextNode(stuff); } else { // AcoustID code.appendChild(document.createTextNode(stuff)); } a.appendChild(code); if (adisabled) { code.style.setProperty("text-decoration", "line-through"); code.style.setProperty("opacity", ".2"); a.addEventListener("mouseover", function(event) { this.firstChild.style.removeProperty("text-decoration"); this.firstChild.style.removeProperty("opacity"); }, false); a.addEventListener("mouseout", function(event) { this.firstChild.style.setProperty("text-decoration", "line-through"); this.firstChild.style.setProperty("opacity", ".2"); }, false); } td.appendChild(a); if (stufftype == "ISRC" || !adisabled) { if (shownstuffs[stuff]) { var bgColour = dupeColour; if (recid && shownstuffs[stuff]["recid"] == recid) { bgColour = infoColour; } else { hasDupes[stufftype]++; } if (shownstuffs[stuff]["a"].style.getPropertyValue("background-color") != dupeColour) { shownstuffs[stuff]["a"].style.setProperty("background-color", bgColour); } a.style.setProperty("background-color", bgColour); } else { shownstuffs[stuff] = { "a": a, "trackid": trackid, "recid": recid }; } } if (stufftype == "AcoustID") { a.parentNode.appendChild(togAID(recid, stuff, adisabled)); } } var tr = document.createElement("dl"); tr.className = "ars"; tr.appendChild(document.createElement("dt").appendChild(document.createTextNode(stufftype + (stuffs.length > 1 ? "s" : "") + ":")).parentNode); tr.appendChild(td); var table = document.createElement("div"); table.className = "ars " + stufftype + "81127"; table.style.setProperty("display", localStorage.getItem("hide" + stufftype + "81127") == "1" ? "none" : "block"); table.appendChild(tr); return table; } function togAID(rec, aid, dis) { var acoustracktoggle = createA(dis ? "+" : "×", AcoustIDlinkingURL.replace(/%acoustid/, aid).replace(/%mbid/, rec).replace(/%state/, dis ? "0" : "1"), dis ? "Link AcoustID to this recording" : "Unlink AcoustID from this recording", "_blank"); acoustracktoggle.style.setProperty("color", dis ? "green" : "red"); return acoustracktoggle; } function insertBeforeARS(par, chi) { var ars = par.getElementsByClassName("ars"); if (ars.length > 0) { return par.insertBefore(chi, ars[0]); } else { return par.appendChild(chi); } } function nsr(prefix) { switch (prefix) { case "mb": return "http://musicbrainz.org/ns/mmd-2.0#"; default: return null; } } var bonusclicks = []; function idCount(type, count) { var idCountZone = document.querySelector("div#sidebar div#" + userjs + "idcountzone"); if (!idCountZone) { idCountZone = document.querySelector("div#sidebar > dl.properties").appendChild(document.createElement("div")); idCountZone.setAttribute("id", userjs + "idcountzone"); idCountZone.style.setProperty("background", "#fef"); idCountZone.style.setProperty("border", "1px dashed silver"); idCountZone.setAttribute("title", GM_info.script.name + " version " + GM_info.script.version); idCountZone.appendChild(createTag("dd", {}, createTag("label", {}, [ createTag("input", {a: {type: "checkbox", ref: "recdis"}, e: {click: function(event) { var content = document.querySelector("div#content"); if (content) { localStorage.setItem(userjs + "hide-recdis", event.target.checked ? "0" : "1"); if (event.target.checked) { content.classList.remove(userjs + "hide-recdis"); } else { content.classList.add(userjs + "hide-recdis"); } } }}}), " Show recording comments" ]))); if (localStorage.getItem(userjs + "hide-recdis") !== "1") { idCountZone.querySelector("input[ref='recdis']").click(); } var showOE = idCountZone.appendChild(document.createElement("dd")).appendChild(document.createElement("label")).appendChild(document.createElement("input")); showOE.setAttribute("type", "checkbox"); showOE.parentNode.appendChild(document.createTextNode(" Show recording open edits")); showOE.addEventListener("click", function(event) { var openEditLinks = document.querySelectorAll("div." + userjs + "openEdits"); for (var oe = 0; oe < openEditLinks.length; oe++) { openEditLinks[oe].style.setProperty("display", this.checked ? "block" : "none"); } }); var showTZ = idCountZone.appendChild(document.createElement("dd")).appendChild(document.createElement("label")).appendChild(document.createElement("input")); showTZ.setAttribute("type", "checkbox"); showTZ.parentNode.appendChild(document.createTextNode(" Show relate/merge tools")); showTZ.addEventListener("click", function(event) { var tzs = document.querySelectorAll("div." + userjs + "toolzone"); for (var tz = 0; tz < tzs.length; tz++) { tzs[tz].style.setProperty("display", this.checked ? "block" : "none"); } }, false); var showEB = idCountZone.appendChild(document.createElement("dd")).appendChild(document.createElement("label")).appendChild(document.createElement("input")); showEB.setAttribute("type", "checkbox"); showEB.parentNode.appendChild(document.createTextNode(" Show Edit rec. buttons")); showEB.addEventListener("click", function(event) { var ebs = document.querySelectorAll("div." + userjs + "editbutt"); for (var eb = 0; eb < ebs.length; eb++) { ebs[eb].style.setProperty("display", this.checked ? "block" : "none"); } }, false); } if (count != 0 && (!paginatedRelease || !type.match(/track|work/i))) { var errorMsg = { "-20": "acoustid.org unreachable", "-21": "Strange result from acoustid.org" }; var cooldt = idCountZone.appendChild(document.createElement("dt")).appendChild(document.createTextNode(type + (count > 1 ? "s" : "") + ":")).parentNode; var cooldd = idCountZone.appendChild(document.createElement("dd")).appendChild(document.createTextNode(count < 0 ? errorMsg[count] + " (or\u00a0something\u00a0like\u00a0that)" : count)).parentNode; if (count < 0) { cooldt.setAttribute("title", "Error " + count); cooldt.style.setProperty("background-color", dupeColour); } else if (count >= 0 && type != "Track" && type != "Work") { if (hasDupes[type] > 0) { cooldt.setAttribute("title", "There " + (hasDupes[type] == 1 ? "is 1" : "are " + hasDupes[type]) + " duplicate" + (hasDupes[type] == 1 ? "" : "s")); cooldt.style.setProperty("background-color", dupeColour); } cooldd.appendChild(document.createTextNode(" (")); var typetoggle = cooldd.appendChild(createA(localStorage.getItem("hide" + type + "81127") == "1" ? "show" : "hide", null, "shift+click to hide/show all")); typetoggle.style.setProperty("cursor", "pointer"); typetoggle.setAttribute("id", "tog81127" + type); typetoggle.addEventListener("click", function(event) { var type = this.getAttribute("id").match(/^tog81127([a-z]+)$/i)[1]; var show = (this.textContent == "show"); localStorage.setItem("hide" + type + "81127", show ? "0" : "1"); var togstuffs = document.getElementsByClassName(type + 81127); for (var itog = 0; itog < togstuffs.length; itog++) { togstuffs[itog].style.setProperty("display", show ? "block" : "none"); } this.replaceChild(document.createTextNode(show ? "hide" : "show"), this.firstChild); if (event.shiftKey && bonusclicks.length == 0) { let showHideARSButt = document.querySelector("a." + (show ? "show" : "hide") + "-credits"); if (showHideARSButt) { showHideARSButt[0].click(); } else { var ars = document.querySelectorAll("div.ars"); for (var iar = 0; iar < ars.length; iar++) { ars[iar].style.setProperty("display", show ? "block" : "none"); } } let showToolZones = document.querySelectorAll("div#sidebar div#" + userjs + "idcountzone input[type='checkbox']"); if (showToolZones) { for (var stz = 0; stz < showToolZones.length; stz++) { if (showToolZones[stz].checked != show) showToolZones[stz].click(); } } var otogs = ["AcoustID", "ISRC"]; for (var iotog = 0; iotog < otogs.length; iotog++) { var togid = "tog81127" + otogs[iotog]; var toglnk = document.getElementById(togid); if (this.id != togid && toglnk && toglnk.textContent == (show ? "show" : "hide")) { bonusclicks.push(toglnk); } } } if (bonusclicks.length > 0) { bonusclicks.pop().click(); } }, false); // onclick cooldd.appendChild(document.createTextNode(")")); } } } function coolifyISRC(isrc) { function truc(txt, i) { var ivt = document.createElement("span"); ivt.appendChild(document.createTextNode(txt)); if (i) { ivt.style.setProperty("color", "red"); ivt.style.setProperty("margin", "0 .1em"); ivt.style.setProperty("text-shadow", "1px 2px 2px yellow"); } return ivt; } if (isrc.match(/[a-z]{2}-?[a-z0-9]{3}-?[0-9]{2}-?[0-9]{5}/i)) { var coolISRC = document.createElement("code"); coolISRC.appendChild(truc(isrc.substr(0, 2), true)); coolISRC.appendChild(truc("-" + isrc.substr(2, 3) + "-", false)); coolISRC.appendChild(truc(isrc.substr(5, 2), true)); coolISRC.appendChild(truc("-" + isrc.substr(7, 5), false)); return coolISRC; } else { return document.createTextNode(isrc); } } // https://web.archive.org/web/20130130164824/tiffanybbrown.com/presentations/2011/xhr2/ function acoustidFishBatch(recids) { if (recids.length > 0) { let xhr = new XMLHttpRequest(); xhr.onload = function(event) { var res = this.responseXML; var wsstatus, mbids; if ( this.status == 200 && (res = res.documentElement) && (wsstatus = res.querySelector("response > status")) && wsstatus.textContent == "ok" && (mbids = res.querySelectorAll("response > mbids > mbid > mbid:not(:empty)")) ) { var acoustids = {}; for (var m = 0; m < mbids.length; m++) { var mbid = mbids[m].textContent; if (mbid && mbid.match(re_GUID)) { if (!acoustids[mbid]) { acoustids[mbid] = []; } var trackids = mbids[m].parentNode.querySelectorAll("track > id:not(:empty)"); for (var ti = 0; ti < trackids.length; ti++) { var trackid = trackids[ti].firstChild.textContent; if (trackid && trackid.match(re_GUID)) { var disabled = trackids[ti].parentNode.querySelector("disabled"); disabled = disabled && disabled.textContent.match(/true/i); var duo = [trackid, disabled]; if (disabled) { acoustids[mbid].push(duo); } else { acoustids[mbid].unshift(duo); } } } } } for (var th = 0; th < tracksHtml.length; th++) { var recmbid, trackTitleCell; if ( (recmbid = tracksHtml[th].querySelector(css_recording)) && (recmbid = recmbid.getAttribute("href").match(new RegExp("/(" + str_GUID + ")$"))) && (recmbid = recmbid[1]) && acoustids[recmbid].length > 0 && (trackTitleCell = tracksHtml[th].querySelector("td:not(.pos):not(.video)")) ) { insertBeforeARS(trackTitleCell, createStuffFragment("AcoustID", acoustids[recmbid], shownacoustids, acoustidURL, null, recmbid)); } } } else { shownacoustids = -21; coolBubble.error("Error parsing AcoustIDs."); } shownacoustids = count(shownacoustids); idCount("AcoustID", shownacoustids); }; xhr.onerror = function(event) { idCount("AcoustID", -20); coolBubble.error("Error " + this.status + (this.statusText ? " “" + this.statusText + "”" : "") + " while fetching AcoustIDs."); }; coolBubble.info("Loading AcoustIDs…"); xhr.open("post", "//api.acoustid.org/v2/track/list_by_mbid", true); var params = "client=A6AsOfBc&format=xml&batch=1&disabled=1"; for (var m = 0; m < recids.length; m++) { params += "&mbid=" + recids[m]; } xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded"); xhr.overrideMimeType("text/xml"); xhr.send(params); } } function addIMGTabLink() { var IMGtab = document.querySelector("div.tabs > ul.tabs > li > a[href$='-art']"); if (IMGtab && IMGtab.textContent.match(/\(0\)$/)) { IMGtab.setAttribute("href", IMGtab.getAttribute("href").replace(/(cover|event)-art/, "add-$1-art")); IMGtab.style.setProperty("background-color", "#FF6"); IMGtab.replaceChild(document.createTextNode(texts["add_" + (page_type == "release" ? "cover" : "event") + "_art"]), IMGtab.firstChild); } } function count(hash) { var count = 0; for (var key in hash) if (Object.prototype.hasOwnProperty.call(hash, key)) { count += 1; } return count; } function time(_ms) { var ms = typeof _ms == "string" ? parseInt(_ms, 10) : _ms; if (ms > 0) { var d = new Date(ms); return (d.getUTCHours() > 0 ? d.getUTCHours() + ":" + (d.getUTCMinutes() / 100).toFixed(2).slice(2) : d.getUTCMinutes()) + ":" + (d.getUTCSeconds() / 100).toFixed(2).slice(2) + (d.getUTCMilliseconds() > 0 ? "." + (d.getUTCMilliseconds() / 1000).toFixed(3).slice(2) : ""); } return "?:??"; } function debug(text) { if (DEBUG) { console.debug(text); } }