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