// ==UserScript== // @name            4chan imgur thumbnail // @version         1.0 // @namespace       anon.4chan.org // @description     Scans for 4chan posts that contain a specific link but no file and embeds an image as a thumbnail. Supports Imgur, 4chan, YouTube, Derpibooru and Vocaroo links as well as direct image links (jpg, png, gif). // @include         http*://boards.4chan.org/* // @grant           GM_xmlhttpRequest // @updateURL       https://bitbucket.org/murkeli/userscripts/raw/master/4chanimgur.user.js // @downloadURL     https://bitbucket.org/murkeli/userscripts/raw/master/4chanimgur.user.js // ==/UserScript==   (function() {       var namespace = '4chanimgur.';     var version = '1.0';     var home = 'https://userscripts.org/scripts/show/161995';     var d = document;       var $ = function(selector, context, forceArray) {         if ($.null(context)) {             context = d.body;         }         if ($.null(context.querySelectorAll)) return null;         var nodes = context.querySelectorAll(selector);         if (!forceArray) {             if (nodes.length == 1) {                 return nodes[0];             } else if (nodes.length == 0) {                 return null;             }         }         return nodes;     };       $.extend = function(object, properties) {         var key, val;         for (key in properties) {             val = properties[key];             object[key] = val;         }     };       $.extend($, {         null: function(value) {             return (value === undefined || value === null);         },         notnull: function(value) {             return !$.null(value);         },         json: function(url, data, callback) {             if (typeof GM_xmlhttpRequest != 'function') {                 return false;             }             GM_xmlhttpRequest({                 method: 'GET',                 url:    url,                 onload: function(req) {                     try {                         var info = JSON.parse(req.responseText);                         callback.call(self, data, info);                     } catch(ex) {                         console.log('JSON data from ' + url + ' could not be parsed: ' + ex + '\nHTTP: ' + req.status + '\nResponse: ' + req.responseText);                     }                 },                 onerror: function() {                     console.log('Request to ' + url + ' failed.');                 }             });         },         on: function(elem, events, handler) {             var arr = events.split(' ');             for (var i = 0, c = arr.length; i < c; i++) {                 elem.addEventListener(arr[i], handler, false);             }         },         off: function(elem, events, handler) {             var arr = events.split(' ');             for (var i = 0, c = arr.length; i < c; i++) {                 elem.removeEventListener(arr[i], handler, false);             }         },         sibling: function(n, elem) {             var r = [];             for ( ; n; n = n.nextSibling ) {                 if (n.nodeType === 1 && n !== elem) {                     r.push(n);                 }             }             return r;         },         siblings: function(elem) {             return this.sibling((elem.parentNode || {}).firstChild, elem);         },         each: function(list, callback) {             if ($.null(list)) return;             for (var i = 0, c = list.length; i < c; i++) {                 if (callback.call(list[i], list[i], i) === false) {                     break;                 }             }         },         attr: function(elem, attr, value) {             if (!elem.getAttribute) return '';             if (!value) {                 return elem.getAttribute(attr);             } else {                 return elem.setAttribute(attr, value);             }         },         data: function(elem, key, value) {             if (!elem.getAttribute) return '';             if ($.null(value)) {                 return elem.getAttribute('data-' + key);             } else {                 return elem.setAttribute('data-' + key, value);             }         },         strip: function(elem) {             if (elem.innerHTML) {                 return elem.innerHTML.replace(//g, '').replace(/&/g, '&').replace(/
/g, ' ');             }             return '';         },         nodes: function(nodes) {             var frag, node, _i, _len;             if (!(nodes instanceof Array)) {                 return nodes;             }             frag = d.createDocumentFragment();             for (_i = 0, _len = nodes.length; _i < _len; _i++) {                 node = nodes[_i];                 frag.appendChild(node);             }             return frag;         },         append: function(parent, elem) {             return parent.appendChild($.nodes(elem));         },         prepend: function(parent, children) {             return parent.insertBefore($.nodes(children), parent.firstChild);         },         after: function(root, elem) {             return root.parentNode.insertBefore($.nodes(elem), root.nextSibling);         },         before: function(root, elem) {             return root.parentNode.insertBefore($.nodes(elem), root);         },         remove: function(elem) {             return elem.parentNode.removeChild($.nodes(elem));         },         replace: function(root, elem) {             return root.parentNode.replaceChild($.nodes(elem), root);         },         create: function(tag, properties) {             var elem = d.createElement(tag);             if (properties) {                 $.extend(elem, properties);             }             return elem;         },         tn: function(text) {             return d.createTextNode(text);         },         id: function(id) {             return d.getElementById(id);         },         hasClass: function(elem, className) {             return (' ' + elem.className + ' ').replace(/[\n\t]/g, ' ').indexOf(className) > -1;         },         addClass: function(elem, className) {             return elem.classList.add(className);         },         rmClass: function(elem, className) {             return elem.classList.remove(className);         },         addStyle: function(css) {             var style = $.create('style', {                 type: 'text/css',                 textContent: css             });             var f = function() {                 var root;                 if (root = d.head || d.documentElement) {                     return $.append(root, style);                 } else {                     return setTimeout(f, 20);                 }             };             f();             return style;         },         inArray: function(arr, obj) {             return arr.indexOf(obj) > -1;         },     });         var Hover = {         limit: 20,         elapsed: function() {             var now = Date.now();             if (!Hover.lastTimestamp || now - Hover.lastTimestamp >= Hover.limit) {                 Hover.lastTimestamp = now;                 return true;             }             return false;         },         image: {             set: function(img) {                 var img = $('#imgur-thumbnail-hover');                 if ($.null(img)) {                     img = $.create('img', { id: 'imgur-thumbnail-hover' });                     $.append(d.body, img);                 }                 img.style.display = 'block';                 img.src = $.data(this, 'image-url');             },             update: function(e) {                 if (!Hover.elapsed()) return;                 var img = $('#imgur-thumbnail-hover');                 if ($.notnull(img)) {                     var style = img.style;                     var clientX, clientY, top, height;                     height = img.offsetHeight;                     clientX = e.clientX, clientY = e.clientY;                     _ref = d.documentElement, clientHeight = _ref.clientHeight, clientWidth = _ref.clientWidth;                     top = clientY - 120;                     style.top = clientHeight <= height || top <= 0 ? '0px' : top + height >= clientHeight ? clientHeight - height + 'px' : top + 'px';                     if (clientX <= clientWidth - 400) {                         style.left = clientX + 45 + 'px';                         style.right = null;                     } else {                         style.left = null;                         style.right = clientWidth - clientX + 45 + 'px';                     }                 }             },             hide: function() {                 var img = $('#imgur-thumbnail-hover');                 if ($.notnull(img)) {                     img.style.display = 'none';                     img.src = '';                 }             },         },         init: function(img) {             var processor = $.data(img, 'processor');             if (Config.get([processor, 'hoverExpand'], Options.processors[processor]['hoverExpand'][0]) === true) {                 $.on(img, 'mouseover', Hover.image.set);                 $.on(img, 'mousemove', Hover.image.update);                 $.on(img, 'mouseout', Hover.image.hide);             }         },         suspend: function(img) {             $.off(img, 'mouseover', Hover.image.set);             $.off(img, 'mousemove', Hover.image.update);             $.off(img, 'mouseout', Hover.image.hide);             Hover.image.hide();         },     };         var Resources = {         placeholderThumb: '',         loadingThumb: '',         errorThumb: '',         externalImage: '',     };             var Thumb = function(props) {         var self = this;                 $.extend(self, props);                         self.buildFile = function(fileId) {             var container = $.create('div', { id: fileId, className: 'file imgur-container' });             var div = $.create('div', { className: 'fileInfo' });             var span = $.create('span', { className: 'fileText', innerHTML: 'Link: ' });             var fia = $.create('a', { target: '_BLANK', href: self.link, innerHTML: self.name + ' ' });             var extImg = $.create('img', { src: Resources.externalImage });             var fa = $.create('a', { target: '_BLANK', href: self.link, className: 'fileThumb' });             var img = $.create('img', { className: 'imgur-image thumb' });             $.data(img, 'processor', self.processor);             $.data(img, 'thumb-url', self.thumbUrl);             $.data(img, 'image-url', self.imageUrl);             $.append(fia, extImg);             $.append(span, fia);             $.append(div, span);             $.append(container, div);             $.append(fa, img);             $.append(container, fa);             return container;         };                 self.initInlineExpand = function(img) {             if (Config.get([self.processor, 'inlineExpand'], Options.processors[self.processor]['inlineExpand'][0]) === false) return;             $.on(img, 'click', function(e) {                 e.preventDefault();                 if ($.hasClass(img, 'thumb')) {                     $.rmClass(img, 'thumb');                     img.src = self.imageUrl;                     Hover.suspend(img);                 } else {                     $.addClass(img, 'thumb');                     img.src = self.thumbUrl;                     Hover.init(img);                 }             });         };                 self.loadImage = function(img) {             img.src = Resources.loadingThumb;             var image = new Image();             image.onload = function() {                 img.src = self.thumbUrl;                 Hover.init(img);                 self.initInlineExpand(img);             };             image.onerror = function() {                 console.log("Could not load image: " + self.imageUrl);                 img.src = Resources.errorThumb;             };             image.src = self.thumbUrl;         };                 self.placehold = function(post, file) {             var fileId = 'f' + (/m(\d+)/.exec(post.id))[1];             var file = file || self.buildFile(fileId);             var img = $('img.imgur-image', file)             img.src = Resources.placeholderThumb;             var handler = function(e) {                 e.preventDefault();                 Thumb.preloaded.push(fileId);                 $.off(img, 'click', handler);                 self.loadImage(img);             };             $.on(img, 'click', handler);             $.before(post, file);         };                 self.plant = function(post, file) {             var fileId = 'f' + post.parentNode.id.substring(1); //this is the troublesome line; joot changed the structure of posts slightly; no more ids in the blockquotes!             if (Config.get([self.processor, 'preload'], Options.processors[self.processor]['preload'][0]) === false                 && !Thumb.isPreloaded(fileId)) {                     return self.placehold(post, file);             }             var file = file || self.buildFile(fileId);             var img = $('img.imgur-image', file)             $.before(post, file);             self.loadImage(img);         };     };         $.extend(Thumb, {         preloaded: [],         isPreloaded: function(id) {             return $.inArray(Thumb.preloaded, id);         },     });         var EmbedObject = function(props) {         var self = this;                 $.extend(self, props);                 self.buildObject = function(fileId) {             var container = $.create('div', { id: fileId, className: 'file imgur-container' });             var div = $.create('div', { className: 'fileInfo' });             var span = $.create('span', { className: 'fileText', innerHTML: 'Link: ' });             var fia = $.create('a', { target: '_BLANK', href: self.link, innerHTML: self.name + ' ' });             var extImg = $.create('img', { src: Resources.externalImage });             var obj = $.create('object', { className: 'imgur-embed', width: self.width, height: self.height });             for (var i = 0, c = self.params.length; i < c; i++) {                 var p = $.create('param', { name: self.params[i].name, value: self.params[i].value });                 $.append(obj, p);             }             var emb = $.create('embed', { src: self.embed, width: self.width, height: self.height });             $.append(obj, emb);             $.append(fia, extImg);             $.append(span, fia);             $.append(div, span);             $.append(container, div);             $.append(container, obj);             return container;         };                 self.plant = function(post, obj) {             var fileId = 'f' + (/m(\d+)/.exec(post.id))[1];             var obj = obj || self.buildObject(fileId);             $.before(post, obj);         };     };       var Main = {         processors: [],         getFile: function(post) {             var siblings = $.siblings(post);             var res;             $.each(siblings, function(sibling) {                 if ($.hasClass(sibling, 'file')) {                     res = sibling;                     return false;                 }             });             return res;         },         qualify: function(post, qualifier) {             return post.innerHTML.toLowerCase().indexOf(qualifier.toLowerCase()) >= 0;         },         sortProcessors: function() {             Main.processors.sort(function(a, b) {                 if (a.priority < b.priority)                     return -1;                 else if (a.priority > b.priority)                     return 1;                 return 0;             });         },         attach: function(processor) {             var exists = false;             $.each(Main.processors, function(proc) {                 if (proc.name == processor.name) {                     exists = true;                     return false;                 }             });             if (!exists) {                 //console.log("attaching " + processor.name);                 Main.processors.push(new processor);                 Main.sortProcessors();             }         },         detach: function(type) {             $.each(Main.processors, function(proc, index) {                 if (proc.name == type) {                     //console.log("detaching " + type);                     Main.processors.splice(index, 1);                     Main.sortProcessors();                     return false;                 }             });         },         process: function(target) {             if (Main.processors.length == 0) return;             var posts = $('blockquote', target, true);             $.each(posts, function(post) {                 var file = Main.getFile(post);                 if ($.null(file)) {                     $.each(Main.processors, function(processor) {                         if (Main.qualify(post, processor.qualifier)) {                             return !processor.process(post, target);                         }                     });                 } else if ($.hasClass(file, 'imgur-container')) {                     var img = $('img.imgur-image', file);                     if (!img) return;                     var thumb = new Thumb({                         processor: $.data(img, 'processor'),                         thumbUrl: $.data(img, 'thumb-url'),                         imageUrl: $.data(img, 'image-url'),                     });                     thumb.plant(post, file);                 }             });         },         init: function() {             Main.process(d.body);                         $.on(d, 'DOMNodeInserted', function(e) {                 Main.process(e.target);             });         },     };         var Processors = {         'Imgur': function() {             var self = this;                         self.priority = 1;             self.name = 'Imgur';             self.regex = /((?:i\.)?imgur.com\/)(\w{4,7})(\.?(?:jpg|png|gif))?/i;             self.qualifier = 'imgur.com/';             self.imageBase = 'http://i.imgur.com/';                         self.process = function(post) {                 self.regex.lastIndex = 0;                 var match = self.regex.exec($.strip(post));                 if (match && $.notnull(match[2])) {                     var base = match[1];                     var filename = match[2];                     var extension = '';                     if (match.length >= 3 && $.notnull(match[3])) {                         extension = match[3];                     }                     var name = base + filename + extension;                     var link = '//' + base + filename + extension;                     var imageUrl = self.imageBase + filename + (extension.length > 0 ? extension : '.jpg');                     var thumbUrl;                     var tinyThumb = Config.get([self.name, 'tinyThumb'], Options.processors[self.name]['tinyThumb'][0]);                     var autoGif = Config.get([self.name, 'autoGif'], Options.processors[self.name]['autoGif'][0]);                     if (!tinyThumb && autoGif && extension == '.gif') {                         thumbUrl = self.imageBase + filename + extension;                     } else {                         thumbUrl = self.imageBase + filename + (tinyThumb ? 's.jpg' : 'm.jpg');                     }                     var thumb = new Thumb({                         processor: self.name,                         name: name,                         link: link,                         imageUrl: imageUrl,                         thumbUrl: thumbUrl                     });                     thumb.plant(post);                     return true;                 }                 return false;             };         },         'Fourchan': function() {             var self = this;                         self.priority = 2;             self.name = 'Fourchan';             self.regex = /images\.4chan\.org\/(a|b|c|d|e|f|g|gif|h|hr|k|m|o|p|r|s|t|u|v|vg|w|wg|i|ic|r9k|cm|hm|y|3|adv|an|cgl|ck|co|diy|fa|fit|hc|int|jp|lit|mlp|mu|n|po|pol|sci|soc|sp|tg|toy|trv|tv|vp|wsg|x|rs|q|asp|gd|lgbt|out|vr)\/src\/(\d+?)\.(?:jpg|png|gif)/i;             self.qualifier = 'images.4chan.org';             self.url = 'http://thumbs.4chan.org/{board}/thumb/{stamp}s.jpg';                         self.process = function(post) {                 self.regex.lastIndex = 0;                 var match = self.regex.exec($.strip(post));                 if (match && $.notnull(match[1]) && $.notnull(match[2])) {                     var name = match[0];                     var board = match[1];                     var stamp = match[2];                     var link = '//' + name;                     var thumbUrl = self.url.replace('{board}', board).replace('{stamp}', stamp);                     var thumb = new Thumb({                         processor: self.name,                         name: name,                         link: link,                         imageUrl: link,                         thumbUrl: thumbUrl                     });                     thumb.plant(post);                     return true;                 }                 return false;             };         },         'YouTube': function() {             var self = this;                         self.priority = 3;             self.name = 'YouTube';             //self.regex = /(?:youtube(?:-nocookie)?\.com\/(?:[^\/]+\/.+\/|(?:v|e(?:mbed)?)\/|.*[?&]v=)|youtu\.be\/)([^"&?\/ ]{11})/i;             self.regex = /(?:[0-9A-Z-]+\.)?(?:youtu\.be\/|youtube(?:-nocookie)?\.com\S*[^\w\-\s])([\w\-]{11})(?=[^\w\-]|$)(?![?=&+%\w]*(?:['"][^<>]*>|<\/a>))[?=&#+%\w-]*/i;             self.qualifier = 'youtu';             self.url = '//i.ytimg.com/vi/{id}/0.jpg';                         self.process = function(post) {                 self.regex.lastIndex = 0;                 var match = self.regex.exec($.strip(post));                 if (match && $.notnull(match[1])) {                     var id = match[1];                     var name = match[0];                     var link = '//' + match[0];                     var imageUrl = self.url.replace('{id}', id);                     var thumb = new Thumb({                         processor: self.name,                         name: name,                         link: link,                         imageUrl: imageUrl,                         thumbUrl: imageUrl                     });                     thumb.plant(post);                     return true;                 }                 return false;             };         },         'Derpibooru': function() {             var self = this;                         self.priority = 5;             self.name = 'Derpibooru';             self.regex = /(derpiboo(?:\.ru|ru\.org)\/)(\d+)/i;             self.qualifier = 'derpiboo';             self.cache = {};                         self.processJSON = function(data, info) {                 if ($.notnull(info)) {                     self.cache[data.id] = info;                 } else {                     info = self.cache[data.id];                 }                 var name = data.base + info.id_number;                 var link = data.url;                 var thumb = new Thumb({                     processor: self.name,                     name: name,                     link: link,                     imageUrl: info.image,                     thumbUrl: info.image                 });                 thumb.plant(data.post);             };                         self.process = function(post) {                 self.regex.lastIndex = 0;                 var match = self.regex.exec($.strip(post));                 if (match && $.notnull(match[1]) && $.notnull(match[2])) {                     var data = {                         post: post,                         base: match[1],                         id: match[2],                     };                     data.url = '//' + data.base + data.id;                     if (self.cache[data.id]) {                         self.processJSON(data);                     } else {                         $.json(data.url + '.json', data, self.processJSON);                     }                     return true;                 }                 return false;             };         },         'Vocaroo': function() {             var self = this;                         self.priority = 10;             self.name = 'Vocaroo';             self.regex = /vocaroo\.com\/i\/([a-z0-9]+)\b/i;             self.qualifier = 'vocaroo.com';                         self.process = function(post, target) {                 self.regex.lastIndex = 0;                 var match = self.regex.exec($.strip(post));                 if (match && $.notnull(match[1])) {                     var id = match[1];                     var link = '//vocaroo.com/i/' + id;                     var autoplay = (Config.get([self.name, 'autoplay'], Options.processors[self.name]['autoplay'][0]) ? '1' : '0');                     if (target === d.body) autoplay = 0;                     var obj = new EmbedObject({                         processor: self.name,                         name: match[0],                         link: link,                         width: 148,                         height: 44,                         params: [                             { name: 'movie', value: 'http://vocaroo.com/player.swf?playMediaID=' + id + '&autoplay=' + autoplay },                             { name: 'wmode', value: 'transparent' },                         ],                         embed: 'http://vocaroo.com/player.swf?playMediaID=' + id + '&autoplay=' + autoplay,                     });                     obj.plant(post);                     return true;                 }                 return false;             };         },         'Generic': function() {             var self = this;                         self.priority = 10;             self.name = 'Generic';             //self.regex = /https?:\/\/[\-A-Z0-9+&@#\/%?=~_|!:,.;]*[\-A-Z0-9+&@#\/%=~_|]?(?:\.(?:jpe?g|png|gif))/i;             self.regex = /https?:\/\/([\-A-Z0-9.]+)\/[\-A-Z0-9+&@#\/%=~_|]+\.(?:jpe?g|png|gif)/i;             self.qualifier = 'http';                         self.isAllowedDomain = function(domain) {                 var disallowed = Config.get([self.name, 'disallowedDomains'], Options.processors[self.name]['disallowedDomains'][0]).split(',');                 for (var i = 0, c = disallowed.length; i < c; i++) {                     var dom = disallowed[i].trim();                     if (dom.length > 0 && domain.indexOf(dom) > -1) return false;                 }                 if (!Config.get([self.name, 'whitelistOnly'], Options.processors[self.name]['whitelistOnly'][0])) {                     return true;                 }                 var allowed = Config.get([self.name, 'allowedDomains'], Options.processors[self.name]['allowedDomains'][0]).split(',');                 for (var i = 0, c = allowed.length; i < c; i++) {                     var dom = allowed[i].trim();                     if (dom.length > 0 && domain.indexOf(dom) > -1) return true;                 }                 return false;             };                         self.process = function(post) {                 self.regex.lastIndex = 0;                 var match = self.regex.exec($.strip(post));                 if (match && $.notnull(match[0])) {                     var imageUrl = match[0];                     var domain = match[1];                     if (self.isAllowedDomain(domain)) {                         var thumb = new Thumb({                             processor: self.name,                             name: imageUrl,                             link: imageUrl,                             imageUrl: imageUrl,                             thumbUrl: imageUrl                         });                         thumb.plant(post);                         return true;                     }                 }                 return false;             };         },     };         var Options = {         processors: {             'Imgur': {                 enabled: [true, 'Enabled', 'Enable imgur.com link thumbnails'],                 preload: [true, 'Auto-Load', 'Load thumbnail automatically instead of waiting for user action'],                 inlineExpand: [true, 'Inline Expand', 'Click the thumbnail to switch to the full image'],                 hoverExpand: [true, 'Hover Expand', 'Hover the thumbnail to show the full image'],                 autoGif: [true, 'Auto-Gif', 'Use animated gifs instead of non-animated thumbnails'],                 tinyThumb: [false, 'Tiny Thumb', 'Use smallest available thumbnail (overrides auto-gif)'],             },             'Fourchan': {                 enabled: [true, 'Enabled', 'Enable 4chan link thumbnails'],                 preload: [true, 'Auto-Load', 'Load thumbnail automatically instead of waiting for user action'],                 inlineExpand: [true, 'Inline Expand', 'Click the thumbnail to switch to the full image'],                 hoverExpand: [true, 'Hover Expand', 'Hover the thumbnail to show the full image'],             },             'YouTube': {                 enabled: [true, 'Enabled', 'Enable YouTube link thumbnails'],                 preload: [true, 'Auto-Load', 'Load thumbnail automatically instead of waiting for user action'],                 inlineExpand: [false, 'Inline Expand', 'Click the thumbnail to switch to the full image'],                 hoverExpand: [true, 'Hover Expand', 'Hover the thumbnail to show the full image'],             },             'Derpibooru': {                 enabled: [true, 'Enabled', 'Enable derpibooru.org link thumbnails (probably doesn\'t work on Opera)'],                 preload: [false, 'Auto-Load', 'Load thumbnail automatically instead of waiting for user action'],                 inlineExpand: [true, 'Inline Expand', 'Click the thumbnail to switch to the full image'],                 hoverExpand: [true, 'Hover Expand', 'Hover the thumbnail to show the full image'],             },             'Vocaroo': {                 enabled: [true, 'Enabled', 'Enable vocaroo.com embedding'],                 autoplay: [false, 'Auto-Play', 'Automatically play new embeds (never on page load)'],             },             'Generic': {                 enabled: [true, 'Enabled', 'Enable other image link thumbnails'],                 whitelistOnly: [false, 'Allowed Only', 'Only enable for whitelisted domains (comma-separated, match-based)'],                 allowedDomains: ['tumblr.com,derpicdn.net', 'Allowed Domains'],                 disallowedDomains: ['', 'Disallowed Domains'],                 preload: [false, 'Auto-Load', 'Load thumbnail automatically instead of waiting for user action (not recommended)'],                 inlineExpand: [true, 'Inline Expand', 'Click the thumbnail to switch to the full image'],                 hoverExpand: [true, 'Hover Expand', 'Hover the thumbnail to show the full image'],             },         },     };         var Config = {         get: function(key, def) {             if (key instanceof Array) {                 key = key.join('.');             }             var val = JSON.parse(localStorage.getItem(namespace + key));             if ($.null(val)) return def;             return val;         },         set: function(key, value) {             return localStorage.setItem(namespace + key, JSON.stringify(value));         },         remove: function(key) {             return localStorage.removeItem(namespace + key);         },         checkbox: {             change: function() {                 Config.set(this.name, this.checked);                 var opt = this.name.split('.');                 if (opt[1] == 'enabled') {                     if (this.checked)                         Main.attach(Processors[opt[0]]);                     else                         Main.detach(opt[0]);                 }                 return this.checked;             },         },         textbox: {             change: function() {                 Config.set(this.name, this.value);                 return this.value;             },         },     };         var Menu = {         init: function() {             var ref = ['boardNavDesktop', 'boardNavDesktopFoot', 'shortcuts'];             for (var i = 0; i < ref.length; i++) {                 var elem = $.id(ref[i]);                 if ($.null(elem)) continue;                                 var a = $.create('span', {                     className: 'imgur-settings-link',                     innerHTML: '[Imgur Settings]',                 });                 $.on(a, 'click', Menu.dialog);                 $.append(elem, a);             }             if (!Config.get('firstrun')) {                 Config.set('firstrun', true);                 return Menu.dialog();             }         },         dialog: function() {             var menu = $.create('div', {                 id: 'imgur-menu',                 innerHTML: '\                    
\                        

4chan imgur thumbnail v' + version + '

\                    
\                    
\                    
\                        
\                    
\                 ',             });             var label, key, ref, opt, procs, content;             ref = Options.processors;             procs = $('.processors', menu);             content = $('.content', menu);             for (label in ref) {                 var ul = $.create('ul', { innerHTML: '' + label + '' });                 opt = ref[label];                 for (key in opt) {                     var arr = opt[key];                     var confKey = [label, key].join('.');                     var def = arr[0];                     var text = arr[1];                     switch (typeof def) {                         case 'boolean':                             var checked = Config.get(confKey, arr[0]) ? true : false;                             var description = arr[2];                             var li = $.create('li', {                                 innerHTML: ': ' + description + '',                             });                             $.on($('input', li), 'change', Config.checkbox.change);                             break;                         case 'string':                             var value = Config.get(confKey, arr[0]);                             var li = $.create('li', {                                 innerHTML: ': ',                             });                             $.on($('input', li), 'change', Config.textbox.change);                             break;                     }                     $.append(ul, li);                 }                 $.append(procs, ul);             }                         overlay = $.create('div', {                 id: 'imgur-overlay'             });             $.on(overlay, 'click', Menu.close);             $.on(menu, 'click', function(e) {                 return e.stopPropagation();             });             $.append(overlay, menu);             $.append(d.body, overlay);         },         close: function() {             $.remove(this);         },     };         $.addStyle('\         #imgur-overlay {\             position: fixed;\             top: 0;\             left: 0;\             width: 100%;\             height: 100%;\             text-align: center;\             background: rgba(0,0,0,.5);\             z-index: 1;\         }\         #imgur-overlay:after {\             content: "";\             display: inline-block;\             height: 100%;\             vertical-align: middle;\         }\         #imgur-menu {\             background: #D6DAF0;\             border: 1px solid rgba(0, 0, 0, 0.25);\             color:#000;\             display: inline-block;\             padding: 5px;\             position: relative;\             text-align: left;\             vertical-align: middle;\             min-width: 350px;\             max-width: 100%;\             max-height: 100%;\         }\         #imgur-menu .info {\             font-size: 10px;\         }\         #imgur-menu .warning {\             font-weight: bold;\             color: #f00;\         }\         #imgur-menu .bar {\             font-size: 16px;\             font-weight: bold;\             margin-left: 5px;\         }\         #imgur-menu .bar p {\             margin: 0 0 5px;\         }\         #imgur-menu .content {\             height: 500px;\             overflow: auto;\             padding: 0 10px 10px;\         }\         #imgur-menu ul {\             padding: 0;\             list-style: none;\         }\         #imgur-menu label {\             text-decoration: underline;\         }\         #imgur-menu input[type="text"] {\             width: 300px;\             margin-top: 3px;\         }\         .imgur-settings-link {\             float: right;\             margin-right: .25em;\         }\         .imgur-embed { float: left; margin: 3px 20px 5px; }\         .imgur-image {\             border: 1px solid #00a !important;\             padding: 2px;\         }\         .imgur-image.thumb {\             max-height: 125px !important;\             max-width: 125px !important;\         }\         #imgur-thumbnail-hover {\             position: fixed;\             max-height: 97%;\             max-width: 75%;\             padding-bottom: 18px;\         }\     ');         Menu.init();       for (label in Options.processors) {         var opt = Options.processors[label];         if (Config.get([label, 'enabled'], opt.enabled[0])) {             Main.attach(Processors[label]);         }     }         Main.init();     })();