Raw Source
jesus2099 / JASRAC. work importer/editor into MusicBrainz + MB-JASRAC-音楽の森 links + MB back search links

// ==UserScript==
// @name         JASRAC. work importer/editor into MusicBrainz + MB-JASRAC-音楽の森 links + MB back search links
// @version      2024.3.14
// @description  One click imports JASRAC works into MusicBrainz (name, iswc, type, credits, edit note, sort name, search hint) and マス歌詞®(mass-lyrics) and wikipedia links. It will do the same magic in work editor. Work links to both JASRAC and 音楽権利情報検索ナビ (ex-音楽の森 aka MINC and Music Forest) and back to MB
// @namespace    https://github.com/jesus2099/konami-command
// @supportURL   https://github.com/jesus2099/konami-command/labels/jasrac-mb-minc_WORK-IMPORT-CROSS-LINKING
// @downloadURL  https://github.com/jesus2099/konami-command/raw/master/jasrac-mb-minc_WORK-IMPORT-CROSS-LINKING.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        2011-01-14; https://web.archive.org/web/20131103163356/userscripts.org/scripts/show/94676 / https://web.archive.org/web/20141011084012/userscripts-mirror.org/scripts/show/94676
// @icon         
// @require      https://github.com/jesus2099/konami-command/raw/84cae2328f2dca0bc750e0961d669005771f16b8/lib/CONTROL-POMME.js?version=2024.3.14
// @require      https://github.com/jesus2099/konami-command/raw/de88f870c0e6c633e02f32695e32c4f50329fc3e/lib/SUPER.js?version=2022.3.24.224
// @grant        none
// @match        *://*.musicbrainz.org/work/*
// @include      /https?:\/\/www\.minc\.or\.jp\/((music|product)\/list|saku\/detail)/
// @match        *://www2.jasrac.or.jp/eJwid/main?trxID=*WORKS_CD=*
// @exclude      *.org/work/*/*edits*
// @run-at       document-end
// ==/UserScript==
"use strict";
let userjs = {
	id: "jesus2099userjs94676",
	name: GM_info.script.name.substr(4).replace(/\s/g, "\u00a0"),
	version: GM_info.script.version
};
/*
	https://github.com/jesus2099/konami-command/issues/14
	POST work credits NG https://chatlogs.musicbrainz.org/musicbrainz/2015/2015-04/2015-04-09.html#T16-34-38-396540
	GET JASRAC ID (work attributes) NG http://tickets.musicbrainz.org/browse/MBS-8341
*/
var MBS7313 = "This script has been partially fixed now but is back to VERY EXPERIMENTAL status!\n(ノ ゜Д゜)ノ 彡┻━┻ Work credits are back on import (not on edit yet). Aliases are still disabled (maybe forever?).";
var DEBUG = localStorage.getItem("jesus2099debug");
var RE_GUID = "[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}";
var reISWC = "T- ?\\d{3}\\.\\d{3}.\\d{3}-\\d";
var reCode = "\\d[A-Z\\d]\\d-\\d{4}-\\d";
var reAnnotCode = "(?:jasrac|作品コード)\\W+(" + reCode + ")";
var MBS = self.location.protocol + "//" + self.location.host;
var pagecat = self.location.href.match(new RegExp("(jasrac(?=\\.or\\.jp)|minc(?=\\.or\\.jp)|work(/" + RE_GUID + "/edit$|/create)|work)"));
var oldTitle = document.title;
var xhrForm = {}, xhrWork = {}, h1, iname;
var MBlookups = [];
var joblist = [];
var xhrJobs = {
	"workinfo-get": {
		info: "PLEASE WAIT (get current work info)",
		async: true,
		method: "get",
		init: function() {
			xhrJobs["workinfo-get"].url = MBS + "/ws/2/work/" + xhrWork.mbid + "?fmt=json&inc=annotation+aliases";
		},
		onload: function() {
			if (this.status > 199 && this.status < 400 && this.responseText.match(new RegExp(xhrWork.mbid))) {
				var res = JSON.parse(this.responseText);
				xhrWork.error = res.error;
				if (!xhrWork.error) {
					xhrWork.aliases = res.aliases;
					xhrWork.annotation = res.annotation ? res.annotation : "";
					var code = xhrWork.annotation.match(new RegExp(reAnnotCode, "i"));
					if (code) {
						insertBefore(createTag("div", {a: {class: "row"}}, [createTag("label", {}, "JASRAC作品コード:"), createTag("b", {s: {backgroundColor: background}}, createA(code[1], workLookupURL("jasrac", "code", code[1]), "JASRAC work code from annotation", "_blank"))]), /* xhrForm.name */ iname.parentNode);
					}
					aliasTable();
				}
				document.title = oldTitle;
			}
		},
	},
	"work-create/edit": {
		method: "post",
		async: true,
		init: function() {
			xhrJobs["work-create/edit"].info = (self.location.pathname.match(/^\/work\/create/) ? "create" : "edit") + " work";
			xhrJobs["work-create/edit"].url = xhrForm.form.getAttribute("action");
			var inps = xhrForm.form.querySelectorAll("form > div > fieldset:not(." + userjs.id + ") input[name]:not([name='']):not([type='button']), form > div > fieldset:not(." + userjs.id + ") textarea[name], form > div > fieldset:not(." + userjs.id + ") select[name]");
			xhrJobs["work-create/edit"].params = "";
			for (let inp = 0; inp < inps.length; inp++) {
				xhrJobs["work-create/edit"].params += (inp > 0 ? "&" : "") + inps[inp].getAttribute("name") + "=" + encodeURIComponent(inps[inp].value);
			}
		},
		onload: function() {
			var mbid = this.responseText.match(new RegExp("<h1>(?:<span class=\"mp\">)?<a[^>]+(" + RE_GUID + ")\""));
			xhrWork.id = this.responseText.match(new RegExp("<a[^>]+/work/merge_queue\\?add-to-merge=([0-9]+)\""));
			if (!xhrWork.edit && mbid) {
				xhrWork.mbid = mbid[1];
				if (self.location.pathname.match(/^\/work\/create/) && h1) {
					h1.appendChild(document.createTextNode(" "));
					h1.appendChild(createA(iname.value, "/work/" + xhrWork.mbid, null, "_blank"));
				}
			}
			if (xhrWork.mbid && xhrWork.id && xhrWork.mbid == mbid[1]) {
				xhrMachine();
			} else {
				alert("Work create (or edit) error #" + this.status + ".\n\n" + this.responseText);
				h1.style.setProperty("background-color", "pink");
				joblist = [];
				disable(xhrForm.form, false);
			}
		},
	},
	"batch-relationship-create": {
		method: "post",
		async: false,
		url: "/relationship-editor",
		init: function() {
			var inps = xhrForm.form.querySelectorAll("div#" + userjs.id + "wcs input.name.lookup-performed, form > div > fieldset:not(." + userjs.id + ") textarea.edit-note");
			xhrJobs["batch-relationship-create"].params = "rel-editor.as_auto_editor=" + (isAutoEdit() ? "1" : "0");
			var reli = 0;
			for (let inp = 0; inp < inps.length; inp++) {
				var input = inps[inp];
				if (input.classList.contains("lookup-performed")) {
					var row = getParent(input, "div", "row");
					var rel = "&rel-editor.rels." + (reli++) + ".";
					xhrJobs["batch-relationship-create"].params +=
						rel + "action=add" +
						+ rel + "link_type=" + row.querySelector("input[name='ar.link_type_id']").value
						+ rel + "period.ended=0"
						+ rel + "entity.0.gid=" + encodeURIComponent(row.querySelector("input[class='gid']").value)
						+ rel + "entity.0.type=" + input.parentNode.className.match(/artist|label/)
						+ rel + "entity.1.gid=" + xhrWork.mbid
						+ rel + "entity.1.type=work";
					if (row.querySelector("input[name='ar.attrs.additional']")) {
						xhrJobs["batch-relationship-create"].params += rel + "attrs.additional=1";
					}
					if (row.querySelector("input[name='ar.attrs.translated']")) { // TODO: obsolete, cf. da6c5d8a-ce13-474d-9375-61feb29039a5
						xhrJobs["batch-relationship-create"].params += rel + "attrs.translated=1";
					}
				} else if (input.tagName == "TEXTAREA") {
					xhrJobs["batch-relationship-create"].params += "&rel-editor.edit_note=" + encodeURIComponent(input.value.replace(/('''PERFORMERS'''|'''MAIN TITLE''')[^※]+※/, "''{…} cf. MB add work edit for full JASRAC extract''\n\n※").replace(/\u00a0(.+)/g, "") + "\n" + MBlinks());
				}
			}
		},
	},
	"annotation-get": {
		async: true,
		info: "get current work annotation",
		method: "get",
		init: function() {
			xhrJobs["annotation-get"].url = MBS + "/ws/2/work/" + xhrWork.mbid + "?inc=annotation";
		},
		onload: function() {
			if (this.status > 199 && this.status < 400 && this.responseText.match(new RegExp("<work.+id=\"" + xhrWork.mbid + "\">"))) {
				var oldannot = this.responseText.match(/<annotation><text>([^<]+)<\/text><\/annotation>/);
				if (!oldannot || oldannot && !oldannot[1].match(new RegExp(xhrWork.code))) {
					if (oldannot) {
						xhrWork.annotation = oldannot[1];
					}
				}
			}
			xhrMachine();
		},
	},
	"annotation-add": {
		async: false,
		info: "JASRAC work code annotation",
		method: "post",
		init: function() {
			xhrJobs["annotation-add"].url = MBS + "/work/" + xhrWork.mbid + "/edit_annotation";
			var curl = workLookupURL("jasrac", "code", xhrWork.code);
			xhrJobs["annotation-add"].params = "edit-annotation.text=" + encodeURIComponent("JASRAC: '''" + xhrWork.code + "''' ([http://tickets.musicbrainz.org/browse/MBS-7359|MBS-7359])" + (xhrWork.annotation ? "\n" + xhrWork.annotation : "")) + "&edit-annotation.changelog=" + encodeURIComponent("JASRAC: " + xhrWork.code + " (MBS-7359)") + "&edit-annotation.edit_note=" + encodeURIComponent("JASRAC: '''" + xhrWork.code + "''' (" + curl + ") ← requires JASRACへの直リンク ('''jasrac_DIRECT-LINK''')\nStill needed for JASRAC auto‐linking (until http://tickets.musicbrainz.org/browse/MBS-7359).\n\n" + MBlinks());
		},
	},
	"alias-add": {
		async: false,
		method: "post",
		init: function() {
			xhrJobs["alias-add"].url = MBS + "/work/" + xhrWork.mbid + "/add-alias";
			xhrJobs["alias-add"].params = "edit-alias.as_auto_editor=" + (isAutoEdit() ? "1" : "0");
			var newAlias = xhrWork.newAliases.shift();
			if (newAlias) {
				xhrJobs["alias-add"].info = newAlias["edit-alias.name"] + " alias";
				if (newAlias["edit-alias.sort_name"].length > 0 && newAlias["edit-alias.sort_name"] != newAlias["edit-alias.name"]) {
					xhrJobs["alias-add"].info += "(" + newAlias["edit-alias.sort_name"] + ")";
				}
				for (let p in newAlias) if (Object.prototype.hasOwnProperty.call(newAlias, p)) {
					xhrJobs["alias-add"].params += "&" + p + "=" + encodeURIComponent(newAlias[p]);
				}
				xhrJobs["alias-add"].params += "&edit-alias.period.ended=&edit-alias.edit_note=" + encodeURIComponent(teditnote.value.replace(/'''CREDITS'''[^※] + '''MAIN TITLE'''/, "'''MAIN TITLE'''") + "\n" + MBlinks());
			}
		},
	},
};
if (pagecat && !document.title.match(/slow down!/i)) {
	pagecat = pagecat[1].replace(new RegExp(RE_GUID + "/"), "");
	var background = "#FF6"; // favourite game
	var cOK = "#CFC"; // bad for ozone layer
	var cWARN = "gold";
	var cERR = "pink";
	var hasLyrics = "詞";
	var hasCredits = "作曲";
	var vocal = "vocal";
	var instrumental = "instrumental";
	if (DEBUG) console.log(userjs.id + " pagecat : " + pagecat);
	let sakuhinCode, sakuhin;
	switch (pagecat) {
		case "jasrac":
			waitForElement("#management-tabs", function() { // quick and dirty patch
				var workName;
				var iswc;
				var summary = "";
				var createWork = "https://musicbrainz.org/work/create?edit-work.name=";
				var isVocal = false;
				var tables = document.getElementsByTagName("table");
				if (tables) {
					var work = tables[1];
					if (true || work) { // quick and dirty patch
						workName = document.querySelector(".baseinfo--name").textContent.trim();
						sakuhinCode = document.querySelector(".baseinfo--code strong");
						sakuhinCode = {node: sakuhinCode, value: sakuhinCode.textContent};
						document.title = workName + " " + sakuhinCode.value + " " + document.title;
						createWork += encodeURIComponent(fullwidthToHalfwidth(workName)).replace(/%20/g, "+");
						summary += workName + " (work code '''" + sakuhinCode.value + "'''/" + sakuhinCode.value.replace(/-/g, "");
						iswc = document.querySelector(".baseinfo--iswc strong");
						if (iswc) {
							iswc = {node: iswc, value: iswc.textContent};
							summary += " — ISWC '''" + iswc.value + "'''/" + iswc.value.replace(/[-.]/g, "");
						}
						summary += ")\n";
						/* var srccred = tables[3];
						if (srccred) {
							var tmpcred = "";
							var credtr = srccred.getElementsByTagName("tr");
							if (credtr) {
								for (let icred = 2; icred < credtr.length; icred++) {
									var credtd = credtr[icred].getElementsByTagName("td");
									if (credtd) {
										var credit = {role: credtd[2].textContent.trim(), who: credtd[1].textContent.trim()};
										credit["trust"] = credtd[4].textContent.trim();
										if (credit.trust != "") {
											credit.trust = "\u00a0(信託状況:" + credit.trust; // + (!credit.trust.match(/全信託/) ? "sic" : "")
										}
										credit["manager"] = credtd[5].textContent.trim();
										if (credit.manager != "") {
											credit.trust += (credit.trust!="" ? "/" : "(") + credit.manager;
										}
										if (credit.trust != "") {
											credit.trust += ")";
										}
										if (!isVocal && credit.role.indexOf(hasLyrics) > -1) {
											isVocal = true;
										}
										credit["line"] = credit.role + ":" + credit.who + credit.trust + "\n";
										if (credit.who != "UNKNOWN PUBLISHER" && tmpcred.indexOf(credit.line) < 0) {
											tmpcred += credit.line;
										}
									}
								}
							}
							if (tmpcred != "") {
								summary += "\n'''CREDITS'''\n" + tmpcred;
							}
						}
						var perf = tables[6];
						if (perf) {
							var perfs = perf.getElementsByTagName("tr");
							var max = Math.min(perfs.length, 13);
							var tmpperf = "\n'''PERFORMERS'''" + (perfs.length > max ? " (" + (perfs.length - 3) + ")" : "") + "\n";
							var isperf = true;
							for (let iperf = 3; iperf < max; iperf++) {
								var artist = perfs[iperf].getElementsByTagName("td")[1].textContent.trim();
								if (artist == "") {
									isperf = false;
									break;
								}
								tmpperf += (iperf > 3 ? "\n" : "") + fullwidthToHalfwidth(artist);
							}
							if (perfs.length > max) tmpperf += "\n…";
							if (isperf) {
								summary += tmpperf + "\n";
							}
						}
						var alias = tables[5];
						if (alias) {
							var transtypes = ["'''genuine'''", "''yomikata''", "latin"];
							var aliases = alias.getElementsByTagName("tr");
							var tmptran = "\n'''MAIN TITLE'''";
							var hastran = false;
							var tmpali = "\n'''ALIASES/SUBTITLES/SEARCHES/TRANSLATIONS'''";
							var hasali = false;
							for (let iali = 2; iali < aliases.length; iali++) {
								var type = aliases[iali].getElementsByTagName("td")[0].textContent.match(/(正題|タイトルの続き)/);
								if (type) {
									type = type[1];
								} else {
									type = "";
								}
								var alis = aliases[iali].getElementsByTagName("div");
								for (let itran = 0; itran < 3; itran++) {
									var ali = alis[itran].textContent.trim();
									if(type == "正題") {
										if (!hastran) {
											hastran = true;
										}
										tmptran += "\n" + ali + " (" + transtypes[itran] + ")";
									} else {
										if (type == "タイトルの続き") {
											if (!ali.match(/^[--]$/)) {
												var prevalias = tmpali.substring(tmpali.lastIndexOf("\n")).split("◇");
												prevalias[itran] += ali;
												tmpali = tmpali.substring(0, tmpali.lastIndexOf("\n")) + prevalias.join("◇");
											}
										} else {
											if (!hasali) {
												hasali = true;
											}
											tmpali += (itran>0 ? "◇" : "\n") + ali;
										}
									}
								}
							}
							if (hastran) {
								summary += tmptran + "\n";
							}
							if (hasali) {
								summary += tmpali + "\n";
							}
							summary += "\n";
						} */
				/* -- vv ------ copiable full summary ------ vv -- */
						var tr = document.createElement("tr");
						var td = document.createElement("td");
						td.setAttribute("colspan", "3");
						td.style.setProperty("text-align", "center");
						td.style.setProperty("background-image", "/eJwid/img/kokoronelogo_A-2out.jpg");
						summary += "※ '''JASRAC work importer''' (" + userjs.version + ")\n" + workLookupURL("jasrac", "code", sakuhinCode.value) + " ← requires '''JASRAC direct links enabler'''\n" + workLookupURL("minc", "code", sakuhinCode.value) + " ← mirror, requires account\n";
						td.appendChild(document.createTextNode("click to select → "));
						var ta = createTag("textarea", {a: {name: "tsummary"}}, summary);
						ta.setAttribute("id", ta.getAttribute("name"));
						ta.style.setProperty("width", "40%");
						ta.setAttribute("rows", "1");
						ta.style.setProperty("color", "black");
						ta.style.setProperty("background", background);
						ta.addEventListener("focus", function(event) {
							this.setAttribute("rows", "20");
							this.select();
						}, false);
						ta.addEventListener("mousemove", function(event) {
							this.blur();
							this.focus();
						}, false);
						ta.addEventListener("mouseout", function(event) {
							this.setAttribute("rows", "1");
						}, false);
						td.appendChild(ta);
						td.appendChild(document.createTextNode(" ← CONTROL key + mouse over to expand + select"));
						tr.appendChild(td);
						work.appendChild(tr);
				/* -- vv ------ Add to MB ------ vv -- */
						var form = createTag("form", {a: {action: createWork.split("?")[0], method: "post", "accept-charset": "utf-8", title: "PLEASE REVIEW before final submission!"}, s: {display: "inline", background: background}});
						form.appendChild(createTag("input", {a: {type: "hidden", name: "edit-work.attributes.0.type_id", value: "3"}}));
						form.appendChild(createTag("input", {a: {type: "hidden", name: "edit-work.attributes.0.value", value: sakuhinCode.value}}));
						createWork += "&edit-work.attributes.0.type_id=3&edit-work.attributes.0.value=" + sakuhinCode.value;
						if (iswc) {
							form.appendChild(createTag("input", {a: {type: "hidden", name: "edit-work.iswcs.0", value: iswc.value}}));
							createWork += "&edit-work.iswcs.0=" + iswc.value;
						}
						if (isVocal) {
							form.appendChild(createTag("input", {a: {type: "hidden", name: "edit-work.type_id", value: "17"}}));
							createWork += "&edit-work.type_id=17";
						} else {
							form.appendChild(createTag("input", {a: {type: "hidden", name: "edit-work.language_id", value: "486"}}));
							createWork += "&edit-work.language_id=486";
						}
						/* https://musicbrainz.org/relationships */
						createWork += getWorkCredits({
							/* artist-work */
							"作詞": { nomatch: /^権利者 /, type: "3e48faba-ec01-47fd-8e89-30e81161661c" },
							"訳詞": { nomatch: /^権利者 /, type: "da6c5d8a-ce13-474d-9375-61feb29039a5" },
							"補詞": { nomatch: /^権利者 /, type: "3e48faba-ec01-47fd-8e89-30e81161661c", additional: "1" },
							"作曲": { nomatch: /^権利者 /, type: "d59d99ea-23d4-4a80-b066-edca32ee158f" },
							"作曲作詞": { type: "a255bca1-b157-4518-9108-7b147dc3fc68" },
							"不明": { type: "a255bca1-b157-4518-9108-7b147dc3fc68" }
						}, summary, createWork);
						createWork += getWorkCredits({
							/* label-work */
							"作詞": { match: /^権利者 (.+)$/, type: "05ee6f18-4517-342d-afdf-5897f64276e3" },
							"訳詞": { match: /^権利者 (.+)$/, type: "05ee6f18-4517-342d-afdf-5897f64276e3" },
							"補詞": { match: /^権利者 (.+)$/, type: "05ee6f18-4517-342d-afdf-5897f64276e3" },
							"作曲": { match: /^権利者 ㈱?(.+)$/, type: "05ee6f18-4517-342d-afdf-5897f64276e3" },
							"出版者": { type: "05ee6f18-4517-342d-afdf-5897f64276e3" },
							"サブ出版": { type: "05ee6f18-4517-342d-afdf-5897f64276e3" }
						}, summary, createWork);
						form.appendChild(createTag("input", {a: {type: "hidden", name: "edit-work.edit_note", value: summary}}));
						createWork += "&edit-work.edit_note=" + encodeURIComponent(summary).replace(/%20/g, "+");
							/* form.appendChild(createTag("a", {a: {title: MBS7313 + "\nImport this work in MusicBrainz (name, iswc, type, edit note)"}, s: {background: background, cursor: "pointer", textDecoration: "underline", color: "blue"}, e: {click: function(event) {
							this.parentNode.setAttribute("target", event.shiftKey || event.ctrlKey ? "_blank" : "_self");
							this.parentNode.submit();
							return stop(event);
						}}}, "Add to MB")) */
						form.appendChild(createTag("a", {a: {href: createWork, title: MBS7313 + "\nImport this work in MusicBrainz (name, iswc, type, edit note)"}, s: {background: background, cursor: "pointer", textDecoration: "blink line-through underline", color: "blue"}}, "Add to MusicBrainz"));
						var sakuhin = document.querySelector(".baseinfo--name").firstChild; // quick and dirty patch
						sakuhin.parentNode.appendChild(document.createTextNode(" ("));
						sakuhin.parentNode.appendChild(form);
						sakuhin.parentNode.appendChild(document.createTextNode(")"));
				/* -- vv ------ sakuhin links ------ vv -- */
						addAfter(document.createElement("sup"), sakuhin).appendChild(createA("MB", workLookupURL("mb", "name", workName), "Search this work name in MusicBrainz"));
						addAfter(document.createElement("sup"), sakuhin).appendChild(mincSearch("title", workName, "MF")).style.setProperty("text-decoration", "underline");
						addAfter(document.createTextNode(" "), sakuhin);
				/* -- vv ------ sakuhin code links ------ vv -- */
						sakuhinCode.node = replaceChildren(createA(sakuhinCode.value, workLookupURL("minc", "code", sakuhinCode.value), "This work in MINC"), sakuhinCode.node);
				/* -- vv ------ iswc links ------ vv -- */
						if (iswc) {
							replaceElement(createA(iswc.value, workLookupURL("mb", "iswc", iswc.value), "Search this ISWC in MusicBrainz"), iswc.node);
						}
				/* -- vv ------ Select music release rights, like CD, etc. ------ vv -- */
						// try 複製 then any then fallback on 複製 even if red, and some works like 000-0314-0 even have none
						(
							document.querySelector("div.management a[href^='#tab-00-']:not(.red)")
							?? document.querySelector("div.management a[href^='#tab-99-']:not(.red)")
							?? document.querySelector("div.management a[href^='#tab-00-']")
						)?.click();
				/* -- vv ------ Open performers section ------ vv -- */
						document.querySelector("section[data-role='artist'] a.btn-acd.close")?.click();
					}
				}
			}); // quick and dirty patch
			break;
		case "minc":
			setInterval(function() {
				var workIDCells = document.querySelectorAll(
					// product list page
					"tbody td[data-th='JASRAC作品コード']:not(." + userjs.id + "),"
					// saku detail page
					+ "div#summary-area > div.management > h5 > a.internal_link[href='#jasrac'] + p > span + span:not(." + userjs.id + "),"
					+ "div#jasrac-area > table > tbody > tr:nth-of-type(2) > th + td:not(." + userjs.id + ")"
				);
				for (var w = 0; w < workIDCells.length; w++) {
					workIDCells[w].classList.add(userjs.id);
					var sakuhinCode = workIDCells[w].firstChild.textContent.trim();
					replaceChildren(createA(sakuhinCode, workLookupURL("jasrac", "code", sakuhinCode), "Go to this work in JASRAC"), workIDCells[w]);
				}
			}, 1234);
			var sakuhinmei = document.querySelector("span[id$='_lbl_jas_nt_sakuhinnm']");
			sakuhinCode = document.querySelector("span[id$='_jas_sakucd']");
			var iswccode = document.querySelector("span[id$='_lbl_jas_nt_iswc']");
			if (sakuhinmei && sakuhinCode && iswccode) {
				sakuhinmei = getParent(sakuhinmei, "td");
				var sakuhinmei_v = sakuhinmei.textContent;
				sakuhinCode = getParent(sakuhinCode, "td");
				var sakuhinCode_v = sakuhinCode.textContent.match(new RegExp(reCode)) + "";
				iswccode = getParent(iswccode, "td");
				var iswccode_v = iswccode.textContent.match(new RegExp(reISWC));
				var jwForm = jasracSearch("title", sakuhinmei_v);
				var jwSubmit = jwForm.querySelector("a[target='_self']");
				jwSubmit.replaceChild(document.createTextNode("JW"), jwSubmit.firstChild);
				jwSubmit.setAttribute("title", "Search this work name in JASRAC");
				jwSubmit.style.setProperty("text-decoration", "underline");
				insertBefore(createTag("sup", {s: {float: "right"}}), sakuhinmei.firstChild).appendChild(jwForm);
				insertBefore(createTag("sup", {s: {float: "right"}}), sakuhinmei.firstChild).appendChild(createA("MB", workLookupURL("mb", "name", sakuhinmei_v), "Search this work name in MusicBrainz"));
				insertBefore(createTag("sup", {s: {float: "right"}}), sakuhinCode.firstChild).appendChild(createA("JW", workLookupURL("jasrac", "code", sakuhinCode_v), "Go to this work in JASRAC"));
				insertBefore(createTag("sup", {s: {float: "right"}}), sakuhinCode.firstChild).appendChild(createA("MB", workLookupURL("mb", "code", sakuhinCode_v), "Search this work code in MusicBrainz work annotation (#281 / SEARCH-434)"));
				if (iswccode_v) {
					iswccode_v += "";
					insertBefore(createTag("sup", {s: {float: "right"}}), iswccode.firstChild).appendChild(createA("JW", workLookupURL("jasrac", "iswc", iswccode_v), "Search this ISWC in JASRAC"));
					insertBefore(createTag("sup", {s: {float: "right"}}), iswccode.firstChild).appendChild(createA("MB", workLookupURL("mb", "iswc", iswccode_v.replace(/T- /, "T-")), "Search this ISWC in MusicBrainz"));
				}
			} else {
				var sakuhinCodeT = document.querySelectorAll("a[href^='SakCdInfo.aspx?SAKUHINCD='], a[href^='SakCDInfo.aspx?SAKUHINCD='], div#ctl00_ctl00_phMain_phDBMain_PanelDetail table.tbl > tbody > tr > td:nth-child(8)");
				for (let st = 0; st < sakuhinCodeT.length; st++) {
					sakuhinCode = "";
					let prec = sakuhinCodeT[st];
					if (sakuhinCodeT[st].tagName == "A") {
						sakuhinCode += sakuhinCodeT[st].getAttribute("href").match(/.{8}$/);
					} else {
						sakuhinCode += sakuhinCodeT[st].textContent.trim().replace(/-/, "");
						prec = prec.lastChild;
					}
					addAfter(createA("MB", workLookupURL("mb", "code", sakuhinCode), "Search this work code in MusicBrainz work annotation (#281 / SEARCH-434)"), prec);
					addAfter(createA("JW", workLookupURL("jasrac", "code", sakuhinCode), "Go to this work in JASRAC"), prec);
				}
			}
			break;
		case "work":
			if (getExtLinks()) {
				/* -- vv ------ JASRAC + ongakunomori sakuhin code link ------ vv -- */
				// var codes = new RegExp(reAnnotCode, "ig")
				var attributes = document.querySelector("h2.work-attributes + script[type='application/json']");
				if (attributes && attributes.textContent) {
					attributes = JSON.parse(attributes.textContent).attributes;
					setInterval(function() {
						// React hydrate in work attributes requires such daemon
						// retrieve collapsed JASRAC codes (when there are too many work attributes)
						var attributeList = document.querySelector("div#sidebar dl.work-attributes");
						var codes = [];
						for (var a = 0; a < attributes.length; a++) {
							if (attributes[a].typeID == 3 && !document.querySelector("div#sidebar dl.work-attributes > dd.work-attribute-jasrac-id[ref='" + attributes[a].value + "']")) {
								// {"id": 494106, "typeName": "JASRAC ID", "value_id": null, "value": "052-1235-9", "typeID": 3}
								if (codes.indexOf(attributes[a].value.replace(/-/g, "")) > -1) {
									if (confirm("Duplicate JASRAC work code " + attributes[a].value + " detected.\nDo you want to edit?")) {
										self.location.href = self.location.pathname + "/edit";
									}
								}
								codes.push(attributes[a].value.replace(/-/g, ""));
								// check if this is not already displayed
								var checkVisibleCodes = document.querySelectorAll("div#sidebar dl.work-attributes > dd.work-attribute-jasrac-id:not([ref='" + attributes[a].value + "'])");
								var codeIsVisible = false;
								for (var c = 0; c < checkVisibleCodes.length; c++) {
									if (checkVisibleCodes[c].textContent.trim() == attributes[a].value) {
										codeIsVisible = true;
										break;
									}
								}
								if (!codeIsVisible) {
									// if this JASRAC ID is not visible yet
									attributeList.insertBefore(createTag("dd", {a: {class: "work-attribute work-attribute-jasrac-id", ref: attributes[a].value}}, attributes[a].value), attributeList.firstChild);
									attributeList.insertBefore(createTag("dt", {}, attributes[a].typeName + ":"), attributeList.firstChild);
								}
							}
						}
						var jasracIDs = document.querySelectorAll("div#sidebar dl.work-attributes > dd.work-attribute-jasrac-id:not(." + userjs.id + ")");
						for (let i = 0; i < jasracIDs.length; i++) {
							var ddcode = jasracIDs[i].textContent.trim();
							replaceElement(createTag("a", {a: {href: workLookupURL("jasrac", "code", ddcode)}, s: {background: background}}, ddcode), jasracIDs[i].firstChild);
							jasracIDs[i].classList.add(userjs.id);
						}
					}, 1000);
					for (var a = 0; a < attributes.length; a++) {
						switch (attributes[a].typeID) {
							case 3:
								getExtLinks().appendChild(createTag("li", {a: {class: userjs.id + "jasrac no-favicon", ref: attributes[a].value}}, createTag("a", {a: {href: workLookupURL("jasrac", "code", attributes[a].value)}, s: {background: background}}, "JASRAC — " + attributes[a].value)));
								getExtLinks().appendChild(createTag("li", {a: {class: userjs.id + "minc no-favicon"}, ref: attributes[a].value}, createTag("a", {a: {href: workLookupURL("minc", "code", attributes[a].value)}, s: {background: background}}, "MINC — " + attributes[a].value)));
								break;
							case 33:
								getExtLinks().appendChild(createTag("li", {a: {class: userjs.id + "minc no-favicon"}, ref: attributes[a].value}, createTag("a", {a: {href: workLookupURL("minc", "NexTone", attributes[a].value)}, s: {background: background}}, "MINC — " + attributes[a].value)));
								break;
						}
					}
				}
				/* -- vv ------ JASRAC ISWC link ------ vv -- */
				var iswcs = document.querySelectorAll("div#sidebar > dl.properties > dd.iswc > a[href*='/iswc/'] code");
				for (let iswc = 0; iswc < iswcs.length; iswc++) {
					var iswct = iswcs[iswc].textContent;
					getExtLinks().appendChild(createTag("li", {a: {class: userjs.id + "jasrac no-favicon"}}, createA("JASRAC — " + iswct, workLookupURL("jasrac", "iswc", iswct.replace(/T-/, "T-+")), "Search this ISWC in JASRAC")));
				}
				/* -- vv ------ JASRAC name search link ------ vv -- */
				var title = document.querySelector("h1 a").textContent.trim();
				getExtLinks().appendChild(createTag("li", {a: {class: userjs.id + "jasrac no-favicon"}}, jasracSearch("title", title)));
				getExtLinks().appendChild(createTag("li", {a: {class: userjs.id + "minc no-favicon"}}, mincSearch("title", title)));
			}
			break;
		case "work/edit":
			xhrWork.edit = true;
			xhrWork.mbid = self.location.pathname.match(new RegExp(RE_GUID));
			xhrMachine(xhrJobs["workinfo-get"]);
			// fall through
		case "work/create":
			h1 = document.querySelector("h1");
			iname = document.getElementById("id-edit-work.name");
			xhrForm.form = getParent(iname, "form");
			insertBefore(createTag("p", {s: {color: "purple", border: "1px dashed #fcc", backgroundColor: "#ffc"}}, [createTag("h3", {}, userjs.name), MBS7313, " ☞ ", createA("Read more…", "https://github.com/jesus2099/konami-command/issues/14", null, "_blank")]), xhrForm.form);
				/* xhrForm.form.addEventListener("submit", function(event) {
				var inputs = xhrForm.form.querySelectorAll(xhrForm.originalInputs.css);
				var changed = !(xhrWork.edit) || (xhrForm.originalInputs.inputs.length != inputs.length);
				for (let i = 0; !changed && i < xhrForm.originalInputs.inputs.length; i++) {
					if (xhrForm.originalInputs.inputs[i].value != xhrForm.originalInputs.values[i]) {
						changed = true;
					}
				}
				if (changed) {
					joblist.push("work-create/edit");
				}
				var rels = this.querySelectorAll("div#" + userjs.id + "wcs input.name.lookup-performed").length;
				if (rels > 0) {
					xhrJobs["batch-relationship-create"].info = rels + " relationship" + (rels > 1 ? "s" : "");
					joblist.push("batch-relationship-create");
				}
				if (xhrWork.code && xhrWork.edit && !xhrWork.jasracidmatch) {//je sais pas si je comprendrai toujours ça la prochaine fois
					if (xhrWork.edit && typeof xhrWork.annotation == "undefined") {
						joblist.push("annotation-get");
					}
					if (!xhrWork.edit || typeof xhrWork.annotation == "string" && !xhrWork.annotation.match(new RegExp(xhrWork.code))) {
						joblist.push("annotation-add");
					}
				}
				var aliases = xhrForm.form.querySelectorAll("table#" + userjs.id + "alta > tfoot > tr > td > input." + userjs.id + "addit[type='checkbox']");
				xhrWork.newAliases = [];
				for (let a = 0; a < aliases.length; a++) {
					if (aliases[a].checked) {
						var tr = getParent(aliases[a], "tr");
						var n = tr.querySelector("input[name='edit-alias.name']");
						var sn = tr.querySelector("input[name='edit-alias.sort_name']");
						var t = tr.querySelector("*[name='edit-alias.type_id']");
						var p = tr.querySelector("input[name='edit-alias.primary_for_locale']");
						var l = tr.querySelector("input[name='edit-alias.locale']");
						if (
							n && typeof n.value == "string" && n.value.length > 0
							&& sn && typeof sn.value == "string"
							&& t && typeof t.value == "string"
							&& p && typeof p.checked == "boolean"
							&& l && typeof l.value == "string"
						) {
							xhrWork.newAliases.push({
								"edit-alias.name": n.value,
								"edit-alias.sort_name": sn.value.length>0 ? sn.value : n.value,
								"edit-alias.type_id": t.value,
								"edit-alias.primary_for_locale": p.checked ? "1" : "0",
								"edit-alias.locale": l.value,
							});
							joblist.push("alias-add");
						}
					}
				}
				if (joblist.length > 1 || joblist.length == 1 && joblist[0] != "work-create/edit") {
					stop(event);
					scrollTo(0, 0);
					disable(this, true);
					xhrMachine();
				}
			}, false); */
			xhrForm.submit = xhrForm.form.querySelector("div.row button.submit.positive[type='submit']");
			xhrForm.originalInputs = {inputs: [], values: [], css: "form > div > fieldset:not(." + userjs.id + ") input:not([type='button']), form > div > fieldset:not(." + userjs.id + ") select"};
			xhrForm.originalInputs.inputs = xhrForm.form.querySelectorAll(xhrForm.originalInputs.css);
			for (let i = 0; i < xhrForm.originalInputs.inputs.length; i++) {
				xhrForm.originalInputs.values.push(xhrForm.originalInputs.inputs[i].value);
			}
			var icomment = document.getElementById("id-edit-work.comment");
			var stypeid = document.getElementById("id-edit-work.type_id");
// TODO: restore language button feature (search slangid) https://github.com/jesus2099/konami-command/issues/321
			var slangid = document.querySelector("select[name='edit-work.languages.0']");
			var teditnote = document.getElementById("id-edit-work.edit_note");
			if (document.referrer.match(/jasrac\.or\.jp/) && (sakuhin = teditnote.value.match(new RegExp("^(.+) \\(work code '''(" + reCode + ")'''"))) !== null) {
				xhrWork.code = sakuhin[2];
				// workSortName(teditnote.value);
				// workCredits(teditnote.value);
			}
			iname.addEventListener("focus", function(event){ this.style.setProperty("background", ""); }, false);
			icomment.addEventListener("focus", function(event){ this.style.setProperty("background", ""); }, false);
			stypeid.addEventListener("focus", function(event){ this.style.setProperty("background", ""); }, false);
			stypeid.style.setProperty("width", "260px");
			var buttons = stypeid.parentNode.appendChild(createTag("span", {a: {class: "buttons"}}, [createTypeButton(vocal), createTypeButton(instrumental)]));
			stypeid.style.setProperty("width", (parseInt(self.getComputedStyle(stypeid).getPropertyValue("width"), 10) - parseInt(self.getComputedStyle(buttons).getPropertyValue("width"), 10)) + "px");
			// buttons = slangid.parentNode.appendChild(createTag("span", {a: {class: "buttons"}}, [createLangButton("日", 198), createLangButton("EN", 120)]));
			// slangid.style.setProperty("width", (parseInt(self.getComputedStyle(slangid).getPropertyValue("width"), 10) - parseInt(self.getComputedStyle(buttons).getPropertyValue("width"), 10)) + "px");
			teditnote.parentNode.appendChild(document.createElement("br"));
			var tjasrac = document.querySelector("div.workheader p.subheader") || document.querySelector("h1");
			tjasrac = tjasrac.appendChild(createTag("textarea", {a: {placeholder: "Paste JASRAC summary here"}}));
			addAfter(document.createTextNode(" ← Hover and paste JASRAC summary here for some magic :)"), tjasrac);
			tjasrac.setAttribute("tabindex", "-1");
			tjasrac.setAttribute("rows", "1");
			tjasrac.style.setProperty("overflow", "hidden");
			tjasrac.addEventListener("mouseover", function(event) {
				if (document.activeElement != this) {
					xhrForm.lastinput = document.activeElement;
				}
				this.blur();
				this.focus();
			}, false);
			tjasrac.addEventListener("focus", function(event) {
				this.setAttribute("rows", "4");
				this.select();
			}, false);
			tjasrac.addEventListener("mouseout", function(event) {
				this.setAttribute("rows", "1");
				if (xhrForm.lastinput && xhrForm.lastinput.focus) {
					xhrForm.lastinput.focus();
				}
			}, false);
			tjasrac.addEventListener("keyup", function(event) {
				sakuhinCode = this.value.match(new RegExp("work code '''(" + reCode + ")'''"));
				if (sakuhinCode && this.value.indexOf(hasCredits) != -1) {
					sakuhinCode = sakuhinCode[1];
					xhrWork.code = sakuhinCode;
					var sakuhin = this.value.match(/^(.+) \(work code/)[1];
					if (sakuhin == iname.value) {
						iname.style.setProperty("background", cOK);
					} else if (iname.value.match(new RegExp(sakuhin, "i"))) {
						iname.style.setProperty("background", cWARN);
					} else {
						iname.style.setProperty("background", cERR);
					}
					var workattrCSS = "form > div > fieldset > table#work-attributes";
					var workattrTypeCSS = "select";
					var workattrValueCSS = "input[type='text'][name$='.value']";
					var workattr = xhrForm.form.querySelectorAll(workattrCSS + " > tbody > tr");
					for (let wa = 0; wa < workattr.length - 1; wa++) {
						var type = workattr[wa].querySelector(workattrTypeCSS);
						var value = workattr[wa].querySelector(workattrValueCSS);
						if (type && value && type.value == "3" && value.value == xhrWork.code) {
							xhrWork.jasracidmatch = value;
							break;
						}
					}
					if (xhrWork.jasracidmatch) {
						xhrWork.jasracidmatch.style.setProperty("background", cOK);
					} else if (
						workattr.length > 1
						&& workattr[workattr.length - 2].querySelector(workattrValueCSS)
						&& workattr[workattr.length - 2].querySelector(workattrValueCSS).value == ""
					) {
						workattr[workattr.length - 2].querySelector(workattrTypeCSS).value = "3";
						workattr[workattr.length - 2].querySelector(workattrValueCSS).value = xhrWork.code;
						workattr[workattr.length - 2].querySelector(workattrValueCSS).style.setProperty("background", cWARN);
					} else {
						xhrForm.form.querySelector(workattrCSS + " button.add-item").click();
						var jasracidnew = xhrForm.form.querySelectorAll(workattrCSS + " > tbody > tr")[workattr.length - 1];
						jasracidnew.querySelector(workattrTypeCSS).value = "3";
						jasracidnew.querySelector(workattrValueCSS).style.setProperty("background", cWARN);
						jasracidnew.querySelector(workattrValueCSS).value = xhrWork.code;
					}
					var iswc = this.value.match(/— ISWC '''(T-[0-9]{3}\.[0-9]{3}.[0-9]{3}-[0-9])'''/); // reISWC when MBS-4727 fixed
					var insel = "form.edit-work div.form-row-text-list div.text-list-row input.value[name^='edit-work.iswcs.']";
					var iiswcs = document.querySelectorAll(insel);
					if (iiswcs.length > 0 && iswc) { // MBS-4727
						iswc = iswc[1];
						var iiswcm;
						for (let im = 0; im < iiswcs.length; im++) {
							if (iiswcs[im].value == iswc) {
								iiswcm = iiswcs[im];
								break;
							}
						}
						if (iiswcm) {
							iiswcm.style.setProperty("background", cOK);
						} else if (iiswcs[iiswcs.length - 1].value == "") {
							iiswcs[iiswcs.length - 1].value = iswc;
							iiswcs[iiswcs.length - 1].style.setProperty("background", cWARN);
						} else {
							document.querySelector("form.edit-work div.form-row-text-list div.form-row-add button.add-item").click();
							var iswcn = document.querySelectorAll(insel)[iiswcs.length];
							iswcn.style.setProperty("background", cWARN);
							iswcn.value = iswc;
						}
					}
					setType(this.value.indexOf(hasLyrics) != -1 ? vocal : instrumental);
					// workSortName(this.value);
					// workCredits(this.value);
					teditnote.value = this.value;
					this.style.setProperty("background", cOK);
					this.value = xhrWork.code + " " + sakuhin;
				} else {
					icomment.style.setProperty("background", "");
					stypeid.style.setProperty("background", "");
					this.style.setProperty("background", cERR);
				}
				scrollTo(0, 0);
			}, false);
			break;
		default:
			return false;
	}
}
function MBlinks() {
	return (xhrWork.id && self.location.href.match(/^\/work\/create/) ? "MB add work edit: " + MBS + "/search/edits?combinator=and&conditions.0.field=work&conditions.0.operator=%3D&conditions.0.name=" + (iname.value ? encodeURIComponent(iname.value) : "TA+GUEULE") + "&conditions.0.args.0=" + xhrWork.id[1] + "&conditions.1.field=type&conditions.1.operator=%3D&conditions.1.args=41\n" : "") + "MB work edit history: " + MBS + "/work/" + xhrWork.mbid + "/edits";
}
function workSortName(txt) {
	var sortname = txt.match(/(.+) \(''yomikata''\)/);
	var searchint = txt.match(/(.+) \(latin\)/);
	var alrows = [];
	var aldone = [];
	if (sortname) {
		alrows.push({name: iname.value, "sort-name": sortname[1], type: "1", locale: "ja", primary: "1"});
		aldone.push(iname.value);
		aldone.push(sortname[1]);
	}
	if (searchint) {
		searchint = swapTHE(searchint[1], true);
		alrows.push({name: searchint, "sort-name": searchint, type: "2"});
		aldone.push(searchint);
	}
	var aliases = txt.substring(txt.indexOf("ALIASES/SUBTITLES/SEARCHES/TRANSLATIONS")).split("\n");
	for (let al = 1; al < aliases.length && aliases[al].match(/^.*◇.*◇.*$/); al++) {
		var alias = fixSTR(aliases[al]).split("◇");
		if (alias[0].length > 0 && !alias[1].match(/^[--]?$/)) {
			alrows.push({name: alias[0], "sort-name": alias[1], type: "1", locale: "ja"});
			if (alias[2].length > 0) {
				let swapped = swapTHE(alias[2], true);
				alrows.push({name: swapped, "sort-name": swapped, type: "2"});
				aldone.push(swapped);
			}
		} else {
			for (let a = 0; a < 3; a++) {
				let swapped = swapTHE(alias[a], true);
				if (!alias[a].match(/^[--]?$/) && aldone.indexOf(swapped) == -1) {
					alrows.push({name: swapped, "sort-name": swapTHE(alias[a], false)});
					aldone.push(swapped);
				}
			}
		}
	}
	if (alrows.length > 0) {
		aliasTable(alrows, true);
	}
}
function workCredits(txt) {
	var wcs = document.getElementById(userjs.id + "wcs");
	if (!wcs) {
		wcs = addAfter(createTag("fieldset", {a: {class: userjs.id}}, [createTag("legend", {}, "Relationships"), createTag("p", {}, [createTag("b", {}, "JASRAC sometimes has wrong credits"), " so, please double-check with your booklet then only lookup for the correct relationship(s) you want to create.", document.createElement("br"), "If you change your mind and want to un-lookup one of them, just select and clear the text field.", document.createElement("br"), "Only green fields will queue relationship edits."]), createTag("div", {a: {id: userjs.id + "wcs"}})]), xhrForm.form.querySelector("form > div > fieldset")).querySelector("div#" + userjs.id + "wcs");
	}
	removeChildren(wcs);
	workCredit("artist", {
		"作詞": {nomatch: /^権利者 /, english: "lyrics", "ar.link_type_id": "165"},
		"訳詞": {nomatch: /^権利者 /, english: "translate lyrics", "ar.link_type_id": "165", "ar.attrs.translated": "1"}, // TODO: obsolete, cf. da6c5d8a-ce13-474d-9375-61feb29039a5
		"補詞": {nomatch: /^権利者 /, english: "additional lyrics", "ar.link_type_id": "165", "ar.attrs.additional": "1"},
		"作曲": {nomatch: /^権利者 /, english: "compose", "ar.link_type_id": "168"},
		"作曲作詞": [{english: "generic write", "ar.link_type_id": "167"}, {english: "compose", "ar.link_type_id": "168"}, {english: "lyrics", "ar.link_type_id": "165"}],
		"不明": [{english: "generic write", "ar.link_type_id": "167"}, {english: "compose", "ar.link_type_id": "168"}, {english: "lyrics", "ar.link_type_id": "165"}]
	}, txt, wcs);
	workCredit("label", {
		"作詞": {match: /^権利者 (.+)$/, english: "publish", "ar.link_type_id": "208"},
		"訳詞": {match: /^権利者 (.+)$/, english: "publish", "ar.link_type_id": "208"},
		"補詞": {match: /^権利者 (.+)$/, english: "publish", "ar.link_type_id": "208"},
		"作曲": {match: /^権利者 ㈱?(.+)$/, english: "publish", "ar.link_type_id": "208"},
		"出版者": {english: "publish", "ar.link_type_id": "208"},
		"サブ出版": {english: "sub-publish", "ar.link_type_id": "208"}
	}, txt, wcs);
}
function workCredit(enttype, credtypes, source, pTarget) {
	var i = pTarget.querySelectorAll("a[title^='reset'][ref]");
	i = i.length > 0 ? parseInt(i[i.length - 1].getAttribute("ref"), 10) + 1 : 0;
	for (let credtype in credtypes) if (Object.prototype.hasOwnProperty.call(credtypes, credtype)) {
		var ctype = credtypes[credtype].english ? [credtypes[credtype]] : credtypes[credtype];
		var cont = pTarget;
		for (let c = 0; c < ctype.length; c++) {
			var credit, credits = new RegExp("^" + credtype + ":([^\u00a0]+)(?:\u00a0(.+))?$", "igm");
			while ((credit = credits.exec(source)) !== null) {
				if (cont.tagName != "FIELDSET" && ctype.length > 1) {
					cont = cont.appendChild(createTag("fieldset", {a: {class: userjs.id}}, [createTag("legend", {}, "choose either")]));
				}
				credit = credit[1].trim();
				if (credtypes[credtype].nomatch && credit.match(credtypes[credtype].nomatch)) {
					continue;
				}
				if (credtypes[credtype].match) {
					if ((credit = credit.match(credtypes[credtype].match)) !== null) {
						credit = credit[1];
					} else {
						continue;
					}
				}
				var ilookupid = userjs.id + "ilookup" + i;
				var target = cont.appendChild(createTag("div", {a: {class: "row", title: credit}}));
				target.appendChild(createTag("label", {a: {for: ilookupid}}, ctype[c].english + ":"));
				target.appendChild(createTag("a", {a: {title: "reset\n" + credit, ref: i}, s: {cursor: "pointer"}, e: {click: function(event) {
					MBlookups[this.getAttribute("ref")].clear(true);
				}}}, "× "));
				var jQac = jQuery(target.appendChild(createTag("span", {a: {class: enttype + " autocomplete" + (ctype.length == 1 || c > 0 ? "" : " " + userjs.id + "manu"), ref: enttype}}, [
					createTag("img", {a: {src: "/static/images/icons/search.png", class: "search", alt: "search"}}),
					createTag("input", {a: {type: "text", class: "name", id: ilookupid, ref: credit}, s: {width: "150px"}}),
					createTag("input", {a: {type: "hidden", class: "id"}}),
					createTag("input", {a: {type: "hidden", class: "gid"}}),
				])));
				MBlookups.push(MB.Control.EntityAutocomplete({inputs: jQac}));
				MBlookups[MBlookups.length - 1].indexedSearch = false;
				target.querySelector("input.name").value = credit;
				jQac.bind("lookup-performed", function(event) {
					var remember = xhrForm.form.querySelector("fieldset.editnote label input.jesus2099remember[type='checkbox']");
					if (remember && remember.checked) {
						remember.click();
					}
					var name = this.querySelector("input.name");
					var gidv = this.querySelector("input.gid").value;
					var href = "/" + this.getAttribute("ref") + "/" + gidv;
					var rla = this.nextSibling;
					if (rla && rla.tagName == "A") {
						rla.setAttribute("href", href);
						replaceElement(document.createTextNode(name.value), rla.firstChild);
					} else {
						addAfter(createA(name.value, href, null, "_blank"), this).style.setProperty("white-space", "nowrap");
					}
				});
				if (ctype.length == 1 || c > 0) {
					jQac.bind("lookup-performed", function(event) {
						var name = this.querySelector("input.name");
						var gidv = this.querySelector("input.gid").value;
						var others = document.querySelectorAll("div#" + userjs.id + "wcs span.autocomplete");
						for (let ot = 0; ot < others.length; ot++) {
							if (
								!others[ot].classList.contains(userjs.id + "manu")
								&& others[ot].querySelector("input.name").getAttribute("ref") == name.getAttribute("ref")
								&& !others[ot].querySelector("input.name").classList.contains("lookup-performed")
							) {
								MBlookups[ot].options.select(document.createEvent("HTMLEvents"), {item: {name: name.value, id: this.querySelector("input.id").value, gid: gidv}});
							}
						}
					});
				}
				jQac.bind("cleared", function(event) {
					var acname = this.querySelector("input.name");
					acname.value = acname.getAttribute("ref");
					if (this.nextSibling.tagName == "A") {
						removeElement(this.nextSibling);
					}
				});
				for (let wap in ctype[c]) if (wap != "english" && Object.prototype.hasOwnProperty.call(ctype[c], wap)) {
					target.appendChild(createTag("input", {a: {type: "hidden", name: wap, value: ctype[c][wap]}}));
				}
				target.appendChild(document.createTextNode(" " + credtype));
				target.appendChild(document.createElement("br"));
				i++;
			}
		}
	}
}
function getWorkCredits(credtypes, source, url) {
	var newParams = "", r = url.lastIndexOf("&rels.");
	if (r > -1) {
		r = url.substr(r).match(/&rels\.(\d+)\./);
	} else {
		r = 0;
	}
	if (r) {
		r = parseInt(r[1], 10) + 1;
	}
	for (let credtype in credtypes) if (Object.prototype.hasOwnProperty.call(credtypes, credtype)) {
		var ctype = credtypes[credtype].type ? [credtypes[credtype]] : credtypes[credtype];
		for (let c = 0; c < ctype.length; c++) {
			var credit, credits = new RegExp("^" + credtype + ":([^\u00a0]+)(?:\u00a0(.+))?$", "igm");
			while ((credit = credits.exec(source)) !== null) {
				credit = credit[1].trim();
				if (credtypes[credtype].nomatch && credit.match(credtypes[credtype].nomatch)) {
					continue;
				}
				if (credtypes[credtype].match) {
					if ((credit = credit.match(credtypes[credtype].match)) !== null) {
						credit = credit[1];
					} else {
						continue;
					}
				}
				newParams += "&rels." + r + ".target=" + encodeURIComponent(fullwidthToHalfwidth(credit)) + "&rels." + r + ".direction=backward";
				for (let wap in ctype[c]) if (!wap.match(/match$/) && Object.prototype.hasOwnProperty.call(ctype[c], wap)) {
					newParams += "&rels." + r + "." + wap + "=" + ctype[c][wap];
				}
				r++;
			}
		}
	}
	return newParams;
}
function disable(w, dis) {
	var inputs = w.querySelectorAll("input, select, textarea, button");
	var len = inputs.length;
	if ((len) && len > 0) {
		for (let i = 0; i < len; i++) {
			if (dis) {
				inputs[i].setAttribute("disabled", "disabled");
			} else {
				inputs[i].removeAttribute("disabled");
			}
		}
		return true;
	} else {
		return false;
	}
}
function getExtLinks() {
	let el = document.querySelector("div#sidebar > ul.external_links");
	let sb, lu, lis;
	if (el) {
		lis = el.getElementsByTagName("li");
		if (lis.length == 1 && lis[0].textContent.match(/has no url relationships/i)) {
			removeElement(lis[0]);
		}
	} else if (!el && (sb = document.querySelector("div#sidebar"))) {
		lu = sb.querySelector("div#sidebar > p.lastupdate");
		if (lu) {
			lu = lu.previousSibling;
		} else {
			lu = sb.lastChild;
		}
		el = addAfter(createTag("ul", {a: {class: "external_links"}}), lu);
		addAfter(createTag("h2", {}, "External links"), lu);
	}
	return el;
}
function jasracSearch(type, query) {
	var formJASRAC = createTag("form", {a: {action: "https://www2.jasrac.or.jp/eJwid/main?trxID=A00401-3#result", method: "post", "accept-charset": "Shift_JIS"}, s: {display: "inline", background: background}});
	formJASRAC.appendChild(createTag("input", {a: {type: "hidden", name: "IN_DEFAULT_WORKS_KOUHO_MAX", value: "100"}}));
	formJASRAC.appendChild(createTag("input", {a: {type: "hidden", name: "IN_DEFAULT_WORKS_KOUHO_SEQ", value: "1"}}));
	switch (type) {
		case "title":
			if (!maybeJapanese(query)) {
				query = removeLeadingArticle(query).toLowerCase();
			}
			query = removeAccents(query);
			formJASRAC.appendChild(createTag("input", {a: {type: "hidden", name: "IN_WORKS_TITLE_NAME1", value: query}}));
			formJASRAC.appendChild(createTag("input", {a: {type: "hidden", name: "IN_WORKS_TITLE_CONDITION", value: "1"}}));
			break;
	}
	formJASRAC.appendChild(createTag("input", {a: {type: "hidden", name: "IN_DEFAULT_SEARCH_WORKS_NAIGAI", value: "0"}}));
	formJASRAC.appendChild(createTag("input", {a: {type: "hidden", name: "RESULT_CURRENT_PAGE", value: "1"}}));
	formJASRAC.appendChild(createCoolSubmit("JASRAC — " + query)).setAttribute("title", formJASRAC.getAttribute("action") + "…");
	return formJASRAC;
}
// form charset trick to allow query from MS932 JASRAC site (and any other sites) to utf-8 MINC site
function mincSearch(type, query, label) {
	var formMINC = createTag("form", {a: {action: "https://www.minc.or.jp/music/list/", method: "get", "accept-charset": "utf-8"}, s: {display: "inline", background: background}});
	switch (type) {
		case "title":
			if (!maybeJapanese(query)) {
				query = removeLeadingArticle(query);
			}
			query = halfwidthToFullwidth(query.toUpperCase());
			formMINC.appendChild(createTag("input", {a: {type: "hidden", name: "tr", value: query}}));
			formMINC.appendChild(createTag("input", {a: {type: "hidden", name: "type", value: "search-form-title"}}));
			formMINC.appendChild(createTag("input", {a: {type: "hidden", name: "match", value: "1"}}));
			break;
	}
	formMINC.appendChild(createCoolSubmit(label ? label : "MINC — " + query)).setAttribute("title", formMINC.getAttribute("action") + "…");
	return formMINC;
}
function removeLeadingArticle(title) {
	return title.replace(/^((?:LA|LE|LES|UN|UNE|DES|A|AN|THE) |L\W)/i, "");
}
function removeAccents(title) {
	return title.replace(/[đ]/g, "d").replace(/[ñ]/g, "n").replace(/[áàảãạăắằẳẵặâấầẩẫậ]/g, "a").replace(/[ïîíìỉĩị]/g, "i").replace(/[úùủũụưứừửữựû]/g, "u").replace(/[éèẻẽẹêếềểễệ]/g, "e").replace(/[óòỏõọơớờởỡợôốồổỗộ]/g, "o").replace(/[ýỳỷỹỵ]/g, "y");
}
function swapTHE(n, swap) {
	return fullwidthToHalfwidth(n.replace(/^(.+)\s{2}(A|THE|UN|UNE|L)$/i, swap ? "$2 $1" : "$1, $2")).replace(/\w\(/g, " (").replace(/\)\w/g, ") ").trim();
}
function toCamelCase(s) {
	return s.toLowerCase().replace(/\b(.)/g, function($1) { return $1.toUpperCase(); });
}
function setType(type) {
	switch (type.toLowerCase()) {
		case vocal.toLowerCase():
			if (stypeid.value == 17) {
				stypeid.style.setProperty("background", cOK);
			} else {
				stypeid.style.setProperty("background", cWARN);
				stypeid.value = 17;
			}
			if (slangid.selectedIndex !== 0 && slangid.value != 486) {
				slangid.style.setProperty("background", cOK);
			} else {
				slangid.style.setProperty("background", cWARN);
				slangid.value = 198;
			}
			break;
		case instrumental.toLowerCase():
			if (stypeid.selectedIndex === 0) {
				stypeid.style.setProperty("background", cOK);
			} else {
				stypeid.style.setProperty("background", cWARN);
				stypeid.selectedIndex = 0;
			}
			if (slangid.value == 486) {
				slangid.style.setProperty("background", cOK);
			} else {
				slangid.style.setProperty("background", cWARN);
				slangid.value = 486;
			}
			break;
	}
	stypeid.focus();
}
function workLookupURL(db, type, q) {
	switch (db) {
		case "mb":
			switch (type) {
				case "name": return "https://musicbrainz.org/search?type=work&limit=100&query=" + encodeURIComponent(q);
				case "code": return "https://musicbrainz.org/search?type=annotation&limit=100&method=advanced&query=type%3Awork+AND+text%3A" + q;
				case "iswc": return "https://musicbrainz.org/iswc/" + q;
			}
			break;
		case "jasrac":
			switch (type) {
				case "name": return "http://www2.jasrac.or.jp/eJwid/main?trxID=A00401-3&IN_DEFAULT_WORKS_KOUHO_MAX=100&IN_DEFAULT_WORKS_KOUHO_SEQ=1&IN_WORKS_TITLE_NAME1=" + q + "&IN_DEFAULT_SEARCH_WORKS_NAIGAI=0&RESULT_CURRENT_PAGE=1";
				case "code": return "http://www2.jasrac.or.jp/eJwid/main?trxID=F20101&WORKS_CD=" + q.replace(/-/g, "") + "&subSessionID=001&subSession=start";
				case "iswc": return "http://www2.jasrac.or.jp/eJwid/main?trxID=A00401-3&IN_DEFAULT_WORKS_KOUHO_MAX=100&IN_DEFAULT_WORKS_KOUHO_SEQ=1&IN_ISWC=" + q.replace(/ /, "+") + "&IN_DEFAULT_SEARCH_WORKS_NAIGAI=0&RESULT_CURRENT_PAGE=1";
			}
			break;
		case "minc":
			switch (type) {
				case "code": return "https://www.minc.or.jp/music/list/?tr=%q%&type=search-form-sakuhin&match=1".replace(/%q%/, q);
				case "NexTone": return "https://www.minc.or.jp/music/list/?tr=%q%&type=search-form-sakuhin&match=2".replace(/%q%/, q);
			}
			break;
	}
}
function aliasTable(add, clear) {
	var alta = document.getElementById(userjs.id + "alta");
	if (!alta) {
		alta = addAfter(createTag("fieldset", {a: {class: userjs.id}}, [createTag("legend", {}, "Aliases"), createTag("p", {}, "You can add some aliases including (but not limited to) the default work sort-name and a latin search-hint. The JASRAC readings are almost always already cool. Sometimes you may fix few ツ→ッ but it’s rare. You can visually check the existing aliases."), createTag("table", {a: {id: userjs.id + "alta"}}, [
			createTag("thead", {}, createTag("tr", {}, [createTag("th", {}, "Name"), createTag("th", {}, "Sort name"), createTag("th", {}, "Type"), createTag("th", {}, "Locale"), createTag("th", {}, "Add?")])),
			document.createElement("tfoot"), document.createElement("tbody")
		])]), xhrForm.form.querySelector("form > div > fieldset")).querySelector("table#" + userjs.id + "alta");
	}
	var aliases = add ? add : xhrWork.aliases;
	var cont = alta.getElementsByTagName(add ? "tfoot" : "tbody")[0];
	if (clear || !add) {
		var headr = createTag("tr", {}, createTag("th", {a: {colspan: 5}, s: {borderTop: "1px solid black", color: "#666"}}));
		if (clear) {
			removeChildren(cont);
			var th = cont.appendChild(headr).firstChild;
			th.appendChild(document.createTextNode("Alias suggestions ("));
			th.appendChild(createTag("a", {s: {cursor: "pointer"}, e: {click: function(event) {
				aliasTable([{name: ""}]);
			}}}, "add another alias"));
			th.appendChild(document.createTextNode(")"));
		} else if (!add) {
			cont.appendChild(headr).firstChild.appendChild(createA((aliases.length > 0 ? aliases.length : "no") + " existing aliase" + (aliases.length != 1 ? "s" : ""), "/work/" + xhrWork.mbid + "/aliases", "Work aliases page", "_blank"));
		}
	}
	for (let a = 0; a < aliases.length; a++) {
		var tr = cont.appendChild(document.createElement("tr"));
		if (!add && a % 2 == 0) {
			tr.style.setProperty("background-color", "#f2f2f2");
		}
		var td = tr.appendChild(createTag("td", {}, add ? createTag("input", {a: {name: "edit-alias.name", value: aliases[a].name, title: aliases[a].name}, s: {width: "10em"}}) : aliases[a].name));
		if (add || aliases[a].name != aliases[a]["sort-name"]) {
			tr.appendChild(createTag("td", {}, add ? createTag("input", {a: {name: "edit-alias.sort_name", value: aliases[a]["sort-name"] ? aliases[a]["sort-name"] : ""}, s: {width: "10em"}}) : aliases[a]["sort-name"]));
		} else {
			td.setAttribute("colspan", "2");
		}
		var se = tr.appendChild(createTag("td", {}, add ? createTag("select", {a: {name: "edit-alias.type_id"}, s: {width: "8em"}, e: {change: function(event) {
			var inps = this.parentNode.parentNode.querySelectorAll("input:not(." + userjs.id + "addit)");
			for (let i = 0; i < inps.length; i++) {
				if (i < 2) {
					inps[i].value = (this.value == "1" && inps[2].value == "en" ? toCamelCase(inps[i].value) : inps[i].value.toUpperCase());
				}
				if (i > 0) {
					inps[i].disabled = this.value == "2" ? true : false;
				}
			}
		}}}, [createTag("option", {}, " "), createTag("option", {a: {value: "1"}}, "work name"), createTag("option", {a: {value: "2"}}, "search hint")]) : (aliases[a].type ? aliases[a].type.toLowerCase() : ""))).querySelector("select");
		td = tr.appendChild(createTag("td", {}, add ? createTag("input", {a: {name: "edit-alias.locale", value: (aliases[a].locale ? aliases[a].locale : "")}, s: {width: "3em"}, e: {keyup: function(event) {
			var lang = this.value.match(/^$|^en$/);
			if (lang && event.key != "Tab") {
				var titles = this.parentNode.parentNode.querySelectorAll("input[name^='edit-alias.'][name$='name']");
				for (let t = 0; t < titles.length; t++) {
					titles[t].value = lang == "en" ? toCamelCase(titles[t].value) : titles[t].value.toUpperCase();
				}
			}
		}}}) : (aliases[a].locale ? aliases[a].locale : "")));
		if (add) {
			var cb = td.appendChild(createTag("input", {a: {name: "edit-alias.primary_for_locale", value: "1", type: "checkbox", title: "primary"}}));
			if (aliases[a].primary) {
				cb.checked = true; // for display
				cb.setAttribute("checked", "checked"); // for reset
			}
		} else if (aliases[a].primary) {
			replaceElement(document.createTextNode("primary " + td.textContent), td.firstChild);
			td.setAttribute("title", "primary");
		}
		tr.appendChild(createTag("td", {}, add ? createTag("input", {a: {type: "checkbox", title: "add this work alias?", class: userjs.id + "addit"}}) : ""));
		if (se) { // se is the work type <select>
			if (aliases[a].type) {
				se.value = aliases[a].type;
				sendEvent(se, "change");
				se.querySelector("option[value='" + aliases[a].type + "']").setAttribute("selected", "selected"); // for reset
			}
			se.addEventListener("change", function(event) {
				var addit = this.parentNode.parentNode.querySelector("input." + userjs.id + "addit");
				if (!addit.checked) {
					addit.click();
				}
				if (this.value == "1") {
					this.parentNode.nextSibling.firstChild.focus();
				}
			}, false);
			xhrForm.form.addEventListener("reset", function(event) {
				var ops = document.querySelectorAll("#" + userjs.id + "alta select option[selected]");
				for (let o = 0; o < ops.length; o++) {
					ops[o].parentNode.value = ops[o].value;
					sendEvent(ops[o].parentNode, "change");
				}
			}, false);
			if (!xhrWork.edit && a < 2 && !(aliases[a].type == "2" && aliases[a].name == iname.value)) {
				getParent(se, "tr").querySelector("input." + userjs.id + "addit[type='checkbox']").click();
			}
			var wname = tr.querySelector("input[name='edit-alias.name']");
			var wsname = tr.querySelector("input[name='edit-alias.sort_name']");
			var wnameaddit = tr.querySelector("input." + userjs.id + "addit[type='checkbox']");
			if (clear && a == 0 && aliases[a].type == "1") {
				if (wname) {
					iname.addEventListener("keyup", function(event) {
						var wname = document.querySelector("table#" + userjs.id + "alta > tfoot > tr > td > input[name='edit-alias.name']");
						var wnameaddit = document.querySelector("table#" + userjs.id + "alta > tfoot > tr > td > input." + userjs.id + "addit[type='checkbox']");
						if (wname) {
							if (this.value != wname.value && wnameaddit && wnameaddit.checked) {
								wnameaddit.click();
							}
							wname.value = this.value;
						}
					}, false);
					wname.setAttribute("title", "work name is always used here");
					wname.setAttribute("readonly", "readonly");
				}
				addAfter(document.createTextNode(se.options[se.selectedIndex].textContent), se);
				replaceElement(createTag("input", {a: {type: "hidden", name: se.getAttribute("name"), value: se.value}}), se);
			} else if (!clear && add) {
				wname.focus();
				wnameaddit.click();
			}
			if (wname && wsname) {
				wname.setAttribute("ref", wname.value);
				wname.addEventListener("focus", function(event) {
					this.setAttribute("ref", this.value);
				}, false);
				wname.addEventListener("keyup", function(event) {
					var wsname = this.parentNode.nextSibling.firstChild;
					if (wsname.value == this.getAttribute("ref")) {
						wsname.value = this.value;
					}
					this.setAttribute("ref", this.value);
				}, false);
				if (wsname.value.length > 0) {
					wsname.setAttribute("title", wsname.value + (wsname.value == wname.value ? "\n(=name)" : ""));
				}
			}
			if (aliases.length == 1 && aliases[a].name == "" && !wnameaddit.checked) {
				// OMG I’M LOST
				wnameaddit.click();
			}
		}
	}
	return alta;
}
function createA(text, link, title, tgt) {
	var a = document.createElement("a");
	a.style.setProperty("background", background);
	a.style.setProperty("padding", "0 4px");
	a.setAttribute("target", tgt || "_self");
	if (typeof link == "string") {
		a.setAttribute("href", link);
	} else {
		a.style.setProperty("cursor", "pointer");
		a.addEventListener("click", link, false);
	}
	if (title) {
		a.setAttribute("title", title);
	}
	a.appendChild(document.createTextNode(text));
	return a;
}
function createCoolSubmit(txt) {
	var a = createA(txt, function(event) {
		if (event.button == 0) {
			if (CONTROL_POMME.new_tab_mod_keys(event)) {
				this.parentNode.setAttribute("target", "_blank");
			} else if (CONTROL_POMME.new_bg_tab_mod_keys(event)) {
				this.parentNode.setAttribute("target", weirdobg());
			} else {
				this.parentNode.setAttribute("target", "_self");
			}
			this.parentNode.submit();
		}
	});
	a.addEventListener("mousedown", function (event) {
		event.preventDefault();
		if (event.button == 1) {
			this.parentNode.setAttribute("target", weirdobg());
			this.parentNode.submit();
		}
	}, false);
	return a;
}
function weirdobg() {
	var weirdo = userjs.id + (new Date().getTime());
	try {
		self.open("", weirdo).blur();
	} catch (error) {}
	self.focus();
	return weirdo;
}
function createTypeButton(type) {
	return createTag("input", {a: {type: "button", value: type.charAt(0).toUpperCase(), title: type, tabindex: "-1", class: "styled-button"}, s: {width: "10px", padding: "1px 8px"}, e: {click: function(event) {
		setType(this.getAttribute("title"));
	}}});
}
function createLangButton(txt, val) {
	return createTag("input", {a: {type: "button", value: txt, title: val, tabindex: "-1", class: "styled-button"}, s: {width: "10px", padding: "1px 8px", float: "none"}, e: {click: function(event) {
		var slang;
		var title = this.getAttribute("title");
		if (title && title.length > 0 && (slang = getSibling(this.parentNode, "select", null, true))) {
			slang.value = this.getAttribute("title");
			slang.focus();
		}
	}}});
}
function isAutoEdit() {
	var ae = xhrForm.form.querySelector("div.auto-editor > input[type='checkbox'][name='edit-work.as_auto_editor']");
	return (ae && ae.checked);
}
function insertBefore(newSibling, element) {
	if (newSibling && element && element.parentNode) {
		return element.parentNode.insertBefore(newSibling, element);
	} else {
		return null;
	}
}
function fixSTR(s) {
	return s.replace(/(^|◇)[**]/g, "$1").replace(/~/g, "〜");
}
function fullwidthToHalfwidth(s) {
	return s.replace(/[!-}]/g, function(a) {
		return String.fromCharCode(a.charCodeAt(0) - 65248);
	}).replace(/\u3000/g, "\u0020").replace(/~/g, "〜");
}
function halfwidthToFullwidth(s) {
	return s.replace(/[!-}]/g, function(a) {
		return String.fromCharCode(a.charCodeAt(0) + 65248);
	}).replace(/\u0020/g, "\u3000");
}
function maybeJapanese(text) {
	return fullwidthToHalfwidth(text).match(/[ぁ-\uFFFF]/);
}
function xhrMachine(_job) {
	var job;
	if (_job) {
		job = _job;
	} else {
		if (joblist.length > 0) {
			job = xhrJobs[joblist.shift()];
		} else if (xhrWork.mbid) {
			self.location.href = "/work/" + xhrWork.mbid;
			return;
		} else {
			alert("MAXI ERROR NO MBID (OMG BBQ WTF?)\nMaybe reload everything and try again…");
			disable(xhrForm.form, false);
			return;
		}
	}
	var xhr = new XMLHttpRequest();
	var async = (typeof job.async == "boolean" ? job.async : true);
	if (job.onreadystatechange) {
		xhr.onreadystatechange = job.onreadystatechange;
	}
	if (job.onload) {
		xhr.onload = job.onload;
	} else if (!async) {
		xhr.onload = function(event) {
			xhrMachine();
		};
		async = true;
	}
	if (job.onerror) {
		xhr.onerror = job.onerror;
	}
	if (job.init) {
		job.init(xhr);
	}
	if (!job.ninja) {
		document.title = "⌛ " + (job.info ? job.info : "loading");
	}
	if (h1) {
		if (h1.textContent.trim().match(/add work/i)) {
			removeChildren(h1);
		}
		h1.appendChild(createTag("div", {}, " " + document.title.replace(/ /g, "\u00a0")));
	}
	xhr.open(job.method ? job.method : "get", job.url, async);
	if (job.params) {
		xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
	}
	if (DEBUG) console.log(job.info + "\n" + job.url + "\n" + (job.params ? job.params.replace(/&/g, "\n") : ""));
	xhr.send(job.params);
	if (!async) {
		xhrMachine();
	}
}
function removeElement(element) {
	return element.parentNode.removeChild(element);
}
function replaceElement(newElement, oldElement) {
	return oldElement.parentNode.replaceChild(newElement, oldElement);
}