mb. AUTO-FOCUS + KEYBOARD-SELECT

musicbrainz.org: MOUSE-LESS EDITING ! Cleverly focuses fields in various musicbrainz edit pages and allows keyboard selection of relationship types as well as some release editor keyboard navigation performance features

目前為 2016-06-16 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         mb. AUTO-FOCUS + KEYBOARD-SELECT
// @version      2016.6.15
// @changelog    https://github.com/jesus2099/konami-command/commits/master/mb_AUTO-FOCUS-KEYBOARD-SELECT.user.js
// @description  musicbrainz.org: MOUSE-LESS EDITING ! Cleverly focuses fields in various musicbrainz edit pages and allows keyboard selection of relationship types as well as some release editor keyboard navigation performance features
// @homepage     http://userscripts-mirror.org/scripts/show/135547
// @supportURL   https://github.com/jesus2099/konami-command/labels/mb_AUTO-FOCUS-KEYBOARD-SELECT
// @compatible   opera(12.18.1872)+violentmonkey  my setup
// @namespace    https://github.com/jesus2099/konami-command
// @author       PATATE12
// @licence      CC BY-NC-SA 3.0 (https://creativecommons.org/licenses/by-nc-sa/3.0/)
// @since        2012-06-08
// @icon         
// @grant        none
// @match        *://*.mbsandbox.org/*/*/add-alias
// @match        *://*.mbsandbox.org/*/*/alias/*/delete
// @match        *://*.mbsandbox.org/*/*/alias/*/edit
// @match        *://*.mbsandbox.org/*/*/edit_annotation*
// @match        *://*.mbsandbox.org/*/*/tags*
// @match        *://*.mbsandbox.org/artist/*/edit
// @match        *://*.mbsandbox.org/artist/create
// @match        *://*.mbsandbox.org/cdtoc/*/set-durations?tracklist=*
// @match        *://*.mbsandbox.org/cdtoc/attach*
// @match        *://*.mbsandbox.org/cdtoc/move*
// @match        *://*.mbsandbox.org/cdtoc/remove*
// @match        *://*.mbsandbox.org/edit/*/cancel*
// @match        *://*.mbsandbox.org/edit/relationship/create*
// @match        *://*.mbsandbox.org/edit/relationship/create_url*
// @match        *://*.mbsandbox.org/edit/relationship/delete*
// @match        *://*.mbsandbox.org/edit/relationship/edit*
// @match        *://*.mbsandbox.org/isrc/delete*
// @match        *://*.mbsandbox.org/label/*/edit
// @match        *://*.mbsandbox.org/label/create
// @match        *://*.mbsandbox.org/recording/*/add-isrc
// @match        *://*.mbsandbox.org/recording/*/edit
// @match        *://*.mbsandbox.org/recording/*/remove-puid*
// @match        *://*.mbsandbox.org/recording/create*
// @match        *://*.mbsandbox.org/release/*/add-cover-art
// @match        *://*.mbsandbox.org/release/*/edit
// @match        *://*.mbsandbox.org/release/*/edit-cover-art/*
// @match        *://*.mbsandbox.org/release/*/remove-cover-art/*
// @match        *://*.mbsandbox.org/release/*/reorder-cover-art
// @match        *://*.mbsandbox.org/release/add*
// @match        *://*.mbsandbox.org/release-group/*/edit
// @match        *://*.mbsandbox.org/release-group/create*
// @match        *://*.mbsandbox.org/url/*/edit
// @match        *://*.mbsandbox.org/work/*/add-iswc
// @match        *://*.mbsandbox.org/work/*/edit
// @match        *://*.mbsandbox.org/work/create*
// @match        *://*.musicbrainz.org/*/*/add-alias
// @match        *://*.musicbrainz.org/*/*/alias/*/delete
// @match        *://*.musicbrainz.org/*/*/alias/*/edit
// @match        *://*.musicbrainz.org/*/*/edit_annotation*
// @match        *://*.musicbrainz.org/*/*/tags*
// @match        *://*.musicbrainz.org/artist/*/edit
// @match        *://*.musicbrainz.org/artist/create
// @match        *://*.musicbrainz.org/cdtoc/*/set-durations?tracklist=*
// @match        *://*.musicbrainz.org/cdtoc/attach*
// @match        *://*.musicbrainz.org/cdtoc/move*
// @match        *://*.musicbrainz.org/cdtoc/remove*
// @match        *://*.musicbrainz.org/edit/*/cancel*
// @match        *://*.musicbrainz.org/edit/relationship/create*
// @match        *://*.musicbrainz.org/edit/relationship/create_url*
// @match        *://*.musicbrainz.org/edit/relationship/delete*
// @match        *://*.musicbrainz.org/edit/relationship/edit*
// @match        *://*.musicbrainz.org/isrc/delete*
// @match        *://*.musicbrainz.org/label/*/edit
// @match        *://*.musicbrainz.org/label/create
// @match        *://*.musicbrainz.org/recording/*/add-isrc
// @match        *://*.musicbrainz.org/recording/*/edit
// @match        *://*.musicbrainz.org/recording/*/remove-puid*
// @match        *://*.musicbrainz.org/recording/create*
// @match        *://*.musicbrainz.org/release/*/add-cover-art
// @match        *://*.musicbrainz.org/release/*/edit
// @match        *://*.musicbrainz.org/release/*/edit-cover-art/*
// @match        *://*.musicbrainz.org/release/*/remove-cover-art/*
// @match        *://*.musicbrainz.org/release/*/reorder-cover-art
// @match        *://*.musicbrainz.org/release/add*
// @match        *://*.musicbrainz.org/release-group/*/edit
// @match        *://*.musicbrainz.org/release-group/create*
// @match        *://*.musicbrainz.org/url/*/edit
// @match        *://*.musicbrainz.org/work/*/add-iswc
// @match        *://*.musicbrainz.org/work/*/edit
// @match        *://*.musicbrainz.org/work/create*
// @run-at       document-end
// ==/UserScript==
/* ---------- configuration below ---------- */
const autoFocus = true; /* focuses on most clever field http://tickets.musicbrainz.org/browse/MBS-2213 */
const selectText = false; /* selects the focused field’s text */
const keyboardSelect = true; /* allows relationship keyboard shortcuts */
const moreURLmatch = true; /* more URL patterns matching in add/edit links (blog, etc.) */
const experimentalTracklistEditorKeyboardNavUpDown = true; /* press UP↓/↑DOWN keys to navigate through track positions, names and lengths */
/* ---------- configuration above ---------- */
/*work in progress, don't refrain from requesting more pages and/or fields*/
function what() {
	var w;
	switch (self.location.pathname.replace(/[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}/, "*").replace(/[0-9]+/, "*")) {
		case "/artist/*/add-alias":
		case "/work/*/add-alias":
		case "/label/*/add-alias":
		case "/artist/*/alias/*/edit":
		case "/label/*/alias/*/edit":
		case "/work/*/alias/*/edit":
			w = document.querySelector("input[id='id-edit-alias.name']");
			break;
		case "/artist/create":
		case "/artist/*/edit":
			w = document.querySelector("input[id='id-edit-artist.name']");
			break;
		case "/artist/*/edit_annotation":
		case "/label/*/edit_annotation":
		case "/recording/*/edit_annotation":
		case "/release/*/edit_annotation":
		case "/release-group/*/edit_annotation":
		case "/work/*/edit_annotation":
			w = document.querySelector("textarea[id='id-edit-annotation.text']");
			break;
		case "/cdtoc/attach":
		case "/cdtoc/move":
			w = (
				document.querySelector("input[id='id-filter-artist.query']") ||
				document.querySelector("input[id='id-filter-release.query']")
			);
			break;
		case "/edit/relationship/create":
		case "/edit/relationship/create-recordings":
		case "/edit/relationship/edit":
			w = document.querySelector("select[id='id-ar.link_type_id']");
			if (keyboardSelect && w) {
				var kbdsel = {
					/*artist-artist*/
						103: "m", //member
						106: ".", //(unblocking member)
						108: "a", //performs as
					/*artist-work*/
						165: "l", //lyrics
						167: "w", //write
						168: "c", //compose
						170: ".", //(unblocking compose)
					/*artist-recording*/
						148: "i", //instrument
						149: "v", //vocal
						150: "op", //orchestra perf
						151: "c", //conducted
						153: "r", //remix
						156: "p", //performs
						297: "a", //arrange
						300: "o", //orchestrate
					/*artist-release*/
						19: "ill", //illust(/design)
						20: "ph", //photo
						24: "l", //liner notes
						26: "mix", //mix
						27: "d", //design
						28: "e", //engineer
						30: "pr", //produce
						36: "rec", //record
						40: "o", //orchestrate
						42: "mas", //master
						44: "ins", //instrument
						45: "op", //orchestra perf
						46: "c", //conducted
						47: "rem", //remix
						51: "pe", //performs
						60: "v", //vocal
						295: "a", //arrange
					/*recording-work*/
						244: "m", //medley
						278: "r", //recording of
				};
				for (var rtype in kbdsel) {
					if (kbdsel.hasOwnProperty(rtype) && (opt = w.querySelector("option[value='"+rtype+"']"))) {
						opt.replaceChild(document.createTextNode(kbdsel[rtype].toUpperCase()+"."+opt.textContent), opt.firstChild);
					}
				}
			}
			break;
		case "/edit/relationship/create_url":
			w = document.querySelector("input[id='id-ar.url']");
			if (moreURLmatch && (type = document.querySelector("select[id='id-ar.link_type_id']"))) {
				w.addEventListener("keyup", function(e) {
					var urlmatch = {
						199: /6109\.jp|ameblo|blog|cocolog|instagram\.com|jugem\.jp|plaza.rakuten\.co\.jp|tumblr\.com/i, //blog
						288: /sonymusic\.co\.jp|\/disco/i, //discography
					};
					for (var utype in urlmatch) {
						if (urlmatch.hasOwnProperty(utype) && (opt = type.querySelector("option[value='"+utype+"']")) && w.value.match(urlmatch[utype])) {
							type.value = utype;
							break;
						}
					}
				}, false);
			}
			break;
		case "/label/create":
		case "/label/*/edit":
			w = document.querySelector("input[id='id-edit-label.name']");
			break;
		case "/recording/create":
		case "/recording/*/edit":
			w = document.querySelector("input[id='id-edit-recording.name']");
			break;
		case "/recording/*/add-isrc":
			w = document.querySelector("input[id='id-add-isrc.isrc']");
			break;
		case "/release/add":
		case "/release/*/edit":
			if (experimentalTracklistEditorKeyboardNavUpDown && (atl = document.querySelector("div#tracklist"))) {
				atl.addEventListener("focus", function(e) {
					var tgt = e.target;
					if (tgt && tgt.classList.contains("track-length") && tgt.value.match(/^(NaN:aN|\?:\?\?)$/)) {
						tgt.value = "";
					}
				}, true);
				atl.addEventListener("keyup", function(e) {
					var tgt = e.target;
					var val;
					if (tgt && tgt.classList.contains("track-length") && (val = tgt.value.match(/(\d+)[^\d]+(\d+)/))) {
						tgt.value = val[1]+":"+val[2];
					}
				}, true);
				atl.addEventListener("keydown", function(e) {
					if (e.target.className.match(/pos|track-(name|length)/) && (e.keyCode == 38 || e.keyCode == 40)) {
						e.preventDefault();
						var its = atl.querySelectorAll("input."+e.target.className);
						var it;
						for (it=0; it<its.length; it++) {
							if (its[it] == e.target) break;
						}
						it += e.keyCode==38?-1:1;
						if (it >= 0 && it < its.length) {
							if (
								e.shiftKey && !e.target.className.match(/pos|track-length/) ||
								!e.shiftKey && e.target.className.match(/pos|track-length/) ||
								e.target.selectionStart == 0 && e.target.selectionEnd == e.target.value.length
							) {
								its[it].select();
							}
							else {
								if (e.target.selectionStart == e.target.selectionEnd && e.target.selectionStart == e.target.value.length) {
									its[it].selectionStart = its[it].value.length;
									its[it].selectionEnd = its[it].value.length;
								}
								else {
									its[it].selectionStart = e.target.selectionStart;
									its[it].selectionEnd = e.target.selectionEnd;
								}
							}
							its[it].focus();
						}
					}
				}, false);
			}
			w = (
				document.querySelector("input#id-name") || 
				document.querySelector("select[id$='.format_id']")
			);
			break;
		case "/release/*/add-cover-art":
			w = document.querySelector("input[id='id-add-cover-art.comment']");
			break;
		case "/release/*/edit-cover-art/*":
			w = document.querySelector("input[id='id-edit-cover-art.comment']");
			break;
		case "/artist/*/tags":
		case "/label/*/tags":
		case "/recording/*/tags":
		case "/release/*/tags":
		case "/release-group/*/tags":
		case "/work/*/tags":
			w = document.querySelector("textarea[id='id-tag.tags']");
			break;
		case "/release-group/create":
		case "/url/*/edit":
			w = document.querySelector("input[id='id-edit-url.url']");
			break;
		case "/release-group/*/edit":
			w = document.querySelector("input[id='id-edit-release-group.name']");
			break;
		case "/work/create":
		case "/work/*/edit":
			w = document.querySelector("input[id='id-edit-work.name']");
			break;
		case "/work/*/add-iswc":
			w = document.querySelector("input[id='id-add-iswc.iswc']");
			break;
	}
	return w?w:document.querySelector("textarea[id$='edit_note']");
}
if (w = what()) {
	if (autoFocus) {
		w.focus();
		if (w.getAttribute("type") == "text" && (wlen = w.value.length)) {
			if (selectText) { w.select(); }
			else { w.setSelectionRange(wlen, wlen); }
		}
		highlight(w);
	}
}
var hli;
var rgbi = 0;
function highlight(f) {
	hli = setInterval(function(){hl(f)}, 50);
}
function hl(f) {
	f.style.setProperty("background-color", "rgb(255,255,"+rgbi+")");
	rgbi += 50;
	if (rgbi >= 255) {
		clearInterval(hli);
		f.style.removeProperty("background-color");
	}
}