Greasy Fork is available in English.

What.CD : Album Art Display

View album art.

このスクリプトの質問や評価の投稿はこちら通報はこちらへお寄せください。
// ==UserScript==
// @id             what-album-art-display
// @name           What.CD : Album Art Display
// @namespace      hateradio)))
// @author         hateradio
// @version        4.9.5
// @description    View album art.
// @homepage       https://userscripts.org/scripts/show/114153
// @icon           
// @icon64         
// @screenshot     
// @include        http*://*what.cd/artist.php?*
// @include        http*://*what.cd/user.php?*
// @include        http*://*what.cd/torrents.php*
// @include        http*://*what.cd/top10.php*
// @updated        2012-11-05
// @since          2011-09-27
// @grant          GM_xmlhttpRequest
// (c) 2011+, hateradio
// Icon from http://openiconlibrary.sourceforge.net/
// ==/UserScript==

// @match          *://*.what.cd/artist.php?*
// @match          *://*.what.cd/user.php?*
// @match          *://*.what.cd/torrents.php*
// @match          *://*.what.cd/top10.php*

(function () {
	'use strict';
	var greaseWindow, strg, update, art;

	// Obj+
	Object.LEN = function (a) { var i = 0, j; for (j in a) { if (a.hasOwnProperty(j)) { ++i; } } return i; };
	Object.SFT = function (a) { var i; for (i in a) { if (a.hasOwnProperty(i)) { delete a[i]; break; } } };

	greaseWindow = (function (a) {
		try {
			a = unsafeWindow === window ? a : unsafeWindow;
		} finally {
			return a || (function () {
				a = document.createElement('p');
				a.setAttribute('onclick', 'return window;');
				return a.onclick();
			}());
		}
	}());

	strg = {
		on: (function () { try { var s = window.localStorage; if (s.getItem && s.setItem && s.removeItem) { return true; } } catch (e) { return false; } }()),
		read: function (key) { return this.on ? JSON.parse(window.localStorage.getItem(key)) : false; },
		save: function (key, dat) { return this.on ? !window.localStorage.setItem(key, JSON.stringify(dat)) : false; },
		wipe: function (key) { return this.on ? !window.localStorage.removeItem(key) : false; },
		zero: function (o) { var k; for (k in o) { if (o.hasOwnProperty(k)) { return false; } } return true; }
	};

	update = {
		name: 'What.CD : Album Art Display',
		version: 4950,
		key: 'ujs_WhatAlbumArtDisplay',
		callback: 'wcdaadudpt',
		page: 'https://userscripts.org/scripts/show/114153',
		urij: 'https://dl.dropbox.com/u/14626536/userscripts/updt/wcdaad/wcdaadudpt.json',
		uric: 'https://dl.dropbox.com/u/14626536/userscripts/updt/wcdaad/wcdaadudpt.js', // Allow dropbox.com to run scripts.
		checkchrome : true,
		interval: 5,
		day: +new Date(),
		top: document.head || document.getElementsByTagName('head')[0],
		time: function () { return +new Date(this.day + (1000 * 60 * 60 * 24 * this.interval)); },
		css: function (t) {
			if (!this.style) { this.style = document.createElement('style'); this.style.type = 'text/css'; this.top.appendChild(this.style); }
			this.style.appendChild(document.createTextNode(t + '\n'));
		},
		js: function (t) {
			var j = document.createElement('script');
			j.type = 'text/javascript';
			j[(/^https?\:\/\//i.test(t) ? 'src' : 'textContent')] = t;
			this.top.appendChild(j);
			return j;
		},
		notification: function (j) {
			try {
				if (this.version < j.version) {
					localStorage.setItem(this.key, JSON.stringify({date: this.time(), version: j.version}));
					this.link();
				}
			} catch (e) {}
		},
		link: function () {
			var a = document.createElement('a');
			a.href = this.page;
			a.className = 'userscriptupdater';
			a.title = 'Update now.';
			a.textContent = 'An update for ' + this.name + ' is available.';
			document.body.appendChild(a);
		},
		check: function (opt) {
			if (this.gmu === true || !strg.on) { return; }
			var stored = strg.read(this.key);
			this.csstxt();
			if (opt || strg.zero(stored) || stored.date < this.day) {
				this.page = this.page || (stored && stored.page ? stored.page : false);
				strg.save(this.key, {date: this.time(), version: this.version, page: this.page});
				if (!opt && this.gmxhr()) {
					return GM_xmlhttpRequest({method: 'GET', url: update.urij, onload: function (r) { update.notification(JSON.parse(r.responseText)); }, onerror: function () { update.check(1); }});
				}
				greaseWindow[this.callback] = function (json) { update.notification(json); };
			} else if (this.version < stored.version) { this.link(); }
		},
		gmu: (function () { try { return GM_updatingEnabled; } catch (e) {} }()),
		gmxhr: function () { if (!(this.checkchrome === true && typeof (chrome) === 'object') && typeof (GM_xmlhttpRequest) === 'function') { return true; } },
		csstxt: function () {
			if (!this.pop) { this.pop = true; this.css('.userscriptupdater,.userscriptupdater:visited{-moz-box-shadow:0 0 6px #787878;-webkit-box-shadow:0 0 6px #787878;box-shadow:0 0 6px #787878;border:1px solid #777;-moz-border-radius:4px;border-radius:4px;cursor:pointer;color:#555;font-family:Arial, Verdana, sans-serif;font-size:11px;font-weight:700;text-align:justify;min-height:45px;position:fixed;z-index:999999;right:10px;top:10px;width:170px;background:#ebebeb url() no-repeat 13px 15px;padding:12px 20px 10px 65px}.userscriptupdater:hover,.userscriptupdater:visited:hover{color:#55698c!important;background-position:13px -85px;border-color:#8f8d96}'); }
		}
	};
	update.check();

	art = {
		jkey: greaseWindow.authkey,
		uri: document.location.href,
		div: document.getElementById('discog_table'),
		uli: document.createElement('ul'),
		cvr: document.createElement('div'),
		act: document.location.pathname.substring(1).replace('.php', ''),
		img: [], // detected ids
		mem: strg.read('AlbArtDispCache'),
		usr: strg.read('whatartdisplaysettings'),
		map: false, // setting to show collage
		showMap: false, // on only for artists pages
		ren: false,
		top: true,
		siz: 96, // img size
		max: 50,
		lag: 500,
		reg: /(?:\/torrents\.php\?id\=(\d+))/,
		exc: ['remix', 'remixed', 'mixtape', 'unknown', 'bootleg', 'interview'], // not for these
		box: {
			td: function (i, location) {
				return location.parentNode.insertBefore(art.elm('td', {className: i >= art.max ? 'gm_albumartdisplay' : 'gm_albumartdisplay gm_albumartdisplay_loading'}), location);
			},
			map: function () {
				art.cvr.id = 'coverhead';
				art.cvr.className = 'box';
				art.div.parentNode.insertBefore(art.cvr, art.div);
				art.cvr.innerHTML = '<div id="coverhead" class="head"><strong>Cover Art</strong></div>';
				art.uli.className = 'collage_images';
				art.cvr.appendChild(art.uli);
			},
			mapAdd: function (uri, id, text) {
				return art.elm('img', {className: '_albumartdisplay' + id, title: text, width: 117, src: uri},
					art.elm('a', {className: uri ? 'gm_albumartdisplay' : 'gm_albumartdisplay_loading', href: '/torrents.php?id=' + id, title: 'Loading . . .'},
						art.elm('li', false, art.uli)));
			},
			shiftCol: function (col) {
				// col.colSpan = col.colSpan ? col.colSpan + 1 : 1;
				col.colSpan += 1;
			}
		},
		mod: function (node, selectors) {
			// console.log('n: ' + node, 'm: ' + this.map, 's: ' + selectors);
			var A,
				C,
				a,
				b = document.querySelectorAll(selectors[0]),
				c = b.length,
				d = document.querySelectorAll(selectors[1]),
				e = d.length,
				tt,
				id,
				j = -1; //console.log(c, e);
			if (this.showMap && this.map) { this.box.map(); }
			while (e--) { this.box.shiftCol(d[e]); }
			while (++j < c) {
				A = b[j];
				C = this.box.td(j, node === 1 ? A.parentNode.parentNode : A.parentNode);
				id = A.href.match(this.reg)[1];
				tt = A.textContent;
				if (j < this.max) {
					a = this.mem[id] || '';
					if (a) {
						this.max++;
						C.className = 'gm_albumartdisplay';
					} else if (this.img[id]) {
						// Depending on the page, there might be several links/images
						// of the same group ID. Ignore duplicates and add one to the max.
						this.max++;
					} else {
						this.img.push(id);
						C.title = 'Loading . . .';
					}
					if (this.showMap && this.map) {
						this.box.mapAdd.call(this, a, id, tt);
					}
				}
				this.elm('img', {className: '_albumartdisplay' + id, title: tt, width: this.siz, height: this.siz, $onclick: 'lightbox.init(this,' + this.siz + ');', src: a, _display: a ? '' : 'none'}, C);
			}
			this.max--;
			this.run();
		},
		opt: function () {
			var A = document.getElementById('collagecovers') || document.getElementById('hidecollage'), B, C, D;
			if (A) {
				C = this.elm('tr', {innerHTML: '<td class="label"><strong>Album Art Display</strong></label></td><td><p><strong class="important_text" id="_albumartdisplaym"></strong></p><div id="_albumartdisplaysettings"></div></td>'});
				A.parentNode.parentNode.parentNode.insertBefore(C, A.parentNode.parentNode);
				D = document.getElementById('_albumartdisplaysettings');
				this.pm = document.getElementById('_albumartdisplaym');
				B = this.elm('input', {id: 'albumdisplaymap', type: 'checkbox', checked: this.map}, D);
				B.addEventListener('click', this.set, false);
				A = this.elm('label', {$for: 'albumdisplaymap', textContent: ' Show collage in artist pages. '}, D);
				B = this.elm('input', {id: 'albumdisplayren', type: 'checkbox', checked: this.ren}, D);
				B.addEventListener('click', this.set, false);
				A = this.elm('label', {$for: 'albumdisplayren', textContent: ' Show album art in torrent/notification pages. '}, D);
				B = this.elm('input', {id: 'albumdisplaytop', type: 'checkbox', checked: this.top}, D);
				B.addEventListener('click', this.set, false);
				A = this.elm('label', {$for: 'albumdisplaytop', textContent: ' Show album art on the Top10.'}, D);
			}
		},
		init: function () { //console.log('N'+Object.LEN(this.mem));
			update.css('a.gm_albumartdisplay_loading{display:block;height:117px}.gm_albumartdisplay_loading{background:transparent url(http://whatimg.com/i/97804798653144081223.gif) no-repeat center center;cursor:progress}.gm_albumartdisplay_loading img{opacity:0 !important}.gm_albumartdisplay img{cursor:pointer}td.gm_albumartdisplay{min-width:' + this.siz + 'px !important;height:' + this.siz + 'px;padding:0;margin:0}td.gm_albumartdisplay img{opacity:.9}td.gm_albumartdisplay img:hover{opacity:1}');
			this.mem = strg.zero(this.mem) ? {} : this.mem;
			if (!strg.zero(this.usr)) {
				this.map = this.usr.map;
				this.ren = this.usr.ren;
				this.top = this.usr.top; // console.log(this.map,this.ren);
			}
			this.page();
		},
		page: function () {
			var img = document.querySelector('.sidebar .box_albumart img[onclick]');
			if (/(?:\/user\.php)/.test(this.uri)) {
				this.opt();
			} else if (this.reg.test(this.uri)) { //console.log('album page');
				if (img) { // console.log(img.src);
					this.img = this.elm('img', {src: img.src, id: RegExp.lastParen});
					this.img.addEventListener('load', art.mix, false);
					this.img.addEventListener('error', art.mix, false);
				}
			} else if (/(?:\/artist\.php)/.test(this.uri)) {
				this.showMap = true;
				this.sel(0, 0, 1);
			} else if (this.ren && /(?:\/torrents\.php)/.test(this.uri)) {
				if (this.uri.indexOf('action=notify') !== -1) {
					this.sel(0, 2, 1);
				} else if (this.uri.indexOf('userid') !== -1) {
					this.sel(1, 4, 2);
				} else {
					this.sel(2, 1, 2);
				}
			} else if (this.top && this.act === 'top10') {
				if (this.uri.indexOf('type=users') === -1 && this.uri.indexOf('type=tags') === -1) {
					this.sel(0, 3, 1);
				}
			} // console.log('m: '+RegExp.lastMatch);
		},
		sel: function (a, b, c) { //console.log(a, b, c);
			var x = [];
			switch (a) {
			case 0:
				x[0] = '.torrent a[href^="torrents.php?id"], table:not(#torrents_' + this.exc.join('):not(#torrents_') + ') .group a[href^="torrents.php?id"], .group_torrent strong a[href^="torrents.php?id"]';
				break;
			case 1:
				x[0] = 'table td a[href^="torrents.php?id"]';
				break;
			case 2:
				x[0] = 'table .cats_col+td a[href^="torrents.php?id"]';
				break;
			default:
				return;
			}
			switch (b) {
			case 0:
				x[1] = ['table.torrent_table:not(#torrents_', this.exc.join('):not(#torrents_'), ') .group_torrent td:first-child, table:not(#torrents_', this.exc.join('):not(#torrents_'), ') td.small:nth-child(', c, ')'].join('');
				break;
			case 1:
				x[1] = '.small.cats_col, tr.group_torrent td[colspan]';
				break;
			case 2:
				x[1] = 'form[id^="notificationform"] .small.cats_col';
				break;
			case 3:
				x[1] = '#top10 .colhead td:nth-child(2), #top10 .colhead_dark td:nth-child(2), .group_torrent td[colspan]';
				break;
			case 4:
				x[1] = '.colhead td:nth-child(2), tr.group_torrent td[colspan]';
				break;
			default:
				return;
			}
			this.mod(c, x);
		},
		run: function () { //console.log('this.img: ' + this.img);
			setTimeout(art.exe, art.lag);
		},
		exe: function () {
			var id = art.img.shift();
			if (id) {
				art.xhr(id);
				art.run();
			}
		},
		xhr: function (id) {
			var req = new XMLHttpRequest();
			req.id = id;
			req.pic = document.getElementsByClassName('_albumartdisplay' + req.id);
			req.pln = req.pic.length; // console.log('p ln: '+req.id, req.pln);
			req.pix = function (e) { art.mix.call(req, e); };
			req.open('get', ['ajax.php?action=torrentgroupalbumart&id=', req.id, '&auth=', this.jkey].join(''), true);
			req.onload = this.dat;
			req.onerror = this.bad;
			req.send(null);
		},
		bad: function () { //console.log('not found!'+this.id);
			art.mux.call(this, 'error');
		},
		dat: function () {
			var j;
			try {
				j = JSON.parse(this.responseText); //console.log(j);
				this.pic[0].onerror = this.pix;
				this.pic[0].onload = this.pix;
				this.pic[0].src = j.response.wikiImage;
			} catch (e) {
				art.bad.call(this);
			}
		},
		mix: function (evt) { // evt - load or error - is attached to an image or xhr
			var a = evt.type, b = evt.target; // console.log('mix: ', this, a, b.src, this.id);

			switch (a) {
			case 'load':
				art.mem[this.id] = b.src;
				break;
			case 'error':
				delete art.mem[this.id];
				break;
			default:
				return;
			}

			if (this.responseText) {
				this.pic[0].onerror = null;
				if (a === 'error') {
					this.pic[0].src = 'static/common/noartwork/music.png';
					return;
				}
				this.pic[0].onload = null;
				art.mux.call(this, a);
			} else {
				this.removeEventListener('error', art.mix, false);
			}
			art.put();
		},
		mux: function (a) {
			var i = this.pln, p, q;
			while (i--) {
				p = this.pic[i];
				q = p.parentNode;
				q.className = 'gm_albumartdisplay';
				p.src = this.pic[0].src;
				p.alt = a === 'load' ? 'Loaded' : 'No artwork';
				p.removeAttribute('style');
				q.removeAttribute('title');
			}
		},
		put: function () {
			if (strg.on) {
				if (Object.LEN(this.mem) > 1500) { Object.SFT(this.mem); }
				strg.save('AlbArtDispCache', this.mem);
			}
		},
		set: function () {
			switch (this.id) {
			case 'albumdisplaymap':
				art.map = this.checked;
				break;
			case 'albumdisplayren':
				art.ren = this.checked;
				break;
			case 'albumdisplaytop':
				art.top = this.checked;
				break;
			default:
				return false;
			}
			art.pm.textContent = strg.save('whatartdisplaysettings', {map: art.map, ren: art.ren, top: art.top}) ? 'Saved.' : 'The setting could not be saved.'; // console.log('saved: '+strg.read('whatartdisplaysettings'));
			window.setTimeout(function () { art.pm.textContent = ''; }, 2500);
		},
		elm: function (t, o, e, p) {
			var a, b, c = document.createElement(t);
			if (typeof (o) === 'object') { for (a in o) { if (o.hasOwnProperty(a)) { b = a.charAt(0); switch (b) { case '_': c.style[a.substring(1)] = o[a]; break; case '$': c.setAttribute(a.substring(1), o[a]); break; default: c[a] = o[a]; break; } } } }
			if (e) { p ? c.appendChild(e) : e.appendChild(c); }
			return c;
		}
	};
	art.init();
}());