// ==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/ // @downloadURL https://update.greasyfork.cloud/scripts/1027/WhatCD%20%3A%20Album%20Art%20Display.user.js // @updateURL https://update.greasyfork.cloud/scripts/1027/WhatCD%20%3A%20Album%20Art%20Display.meta.js // ==/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 = '
Cover Art
'; 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: 'Album Art Display

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