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();
}());