DeviantTidy

Performs a variety of functions on DeviantArt pages to improve its look and usability. For full details, see http://www.deviantart.com/deviation/45622809/

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name         DeviantTidy
// @namespace    devianttidy
// @description  Performs a variety of functions on DeviantArt pages to improve its look and usability. For full details, see http://www.deviantart.com/deviation/45622809/
// @version      4.7.9
// @icon         
// @require      https://cdnjs.cloudflare.com/ajax/libs/jquery/2.2.0/jquery.min.js
// @match        *://*.deviantart.com/*
// @run-at       document-start
// @grant        GM_addStyle
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_registerMenuCommand
// @grant        GM_xmlhttpRequest
// @grant        unsafeWindow
// ==/UserScript==


(function () {
	"use strict";

	// Create a DOM element (tag, [properties,] children)
	var $E = function() {
		if (arguments.length === 0) {return;}

		function applyObj(to, obj) {
			for (var prop in obj) {
				if (obj.hasOwnProperty(prop)) {
					if (typeof obj[prop] === 'object') {
						applyObj(to[prop], obj[prop]);
					}
					else {
						to[prop] = obj[prop];
					}
				}
			}
		}

		var elm = document.createElement(arguments[0]);

		[arguments[1], arguments[2]].forEach(function(arg, idx, ary) {
			if (typeof arg === 'object') {
				if (arg instanceof Array) {
					arg.forEach(function(append, idx, ary) {
						elm.appendChild((typeof append === 'string') ? document.createTextNode(append) : append);
					});
				}
				else {
					for (var prop in arg) {
						if (arg.hasOwnProperty(prop)) {
							if (prop === 'events') {
								var events = arg[prop];
								for (var evt in events) {
									if (events.hasOwnProperty(evt)) {
										elm.addEventListener(evt.replace(/^on/, ''), events[evt], false);
									}
								}
							}
							else {
								if (typeof arg[prop] === 'object') {
									applyObj(elm[prop], arg[prop]);
								}
								else {
									elm[prop] = arg[prop];
								}
							}
						}
					}
				}
			}
		});
		return elm;
	};


	// Determines whether we're within a dynamically-created deviation page
	var inDynamicPage = function() {return $('.minibrowse-container').size() > 0;};


	// Gets the logged-in user name, or '' if not logged in
	var getUsername = function() {var d = unsafeWindow.deviantART.deviant; return d && d.loggedIn ? d.username : '';};


	// The DeviantTidy-specific modal interface
	var devianttidydialog = {
		node: null,
		body: null,
		timer: null,

		open: function(header, content, autoClose) {
			// Reset timer and set a new one if requested.
			clearTimeout(this.timer);
			if (autoClose) {
				this.timer = window.setTimeout(this.close.bind(this), autoClose);
			}

			// If dialog is open, close it and start a new one
			this.close();

			if (typeof header !== 'string') {return;}
			if (typeof content === 'string') {
				content = [$E('div', {className: 'ppp c'}, [content])];
			}

			this.body = $E('div', {className: 'ppp dialog-body'}, content);
			this.node = this.createPopup(header);
			devianttidy.body.appendChild(this.node);
			this.resizePopup();
			window.addEventListener('resize', this.resizePopup.bind(this));
			$('#devianttidy-dialog-close').focus();
		},

		createPopup: function(header) {
			return $E('div', {className:'devianttidy-dialog', style:{display:'none'}}, [
				$E('div', [
					$E('div', {className:'gr-box gr-genericbox'}, [
						$E('i', {className:'gr1'}, [$E('i')]),
						$E('i', {className:'gr2'}, [$E('i')]),
						$E('i', {className:'gr3'}, [$E('i')]),
						$E('div', {className:'gr-top'}, [
							$E('i', {className:'tri'}),
							$E('div', {className:'gr'}, [
								$E('h2', [
									$E('a', {href:devianttidy.homepage, title:"DeviantTidy Homepage"}, [
										$E('img', {className:'dialog-icon', src:devianttidyicons.dt})
									]),
									header
								]),
								$E('a', {className: 'dialog-close', id: 'devianttidy-dialog-close', href: '#', events: {click: this.close.bind(this)}}, [
									$E('img', {src: devianttidyicons.close})
								])
							])
						]),
						$E('div', {className:'gr-body'}, [this.body]),
						$E('i', {className:'gr3 gb'}),
						$E('i', {className:'gr2 gb'}),
						$E('i', {className:'gr1 gb gb1'})
					])
				])
			]);
		},

		resizePopup: function() {
			var maxPanelHeight = 900;
			var minWindowHeight = 250;

			// Set maximum body height given window height
			var ih = window.innerHeight;
			var h = ih && ih > minWindowHeight ? (ih < maxPanelHeight ? ih : maxPanelHeight) : minWindowHeight;
			this.body.style.maxHeight = (h * 0.9 - 60) + 'px';

			// Set vertical alignment, given popup height
			var gr = this.node.childNodes[0];
			gr.style.marginTop = (gr.offsetHeight ? -gr.offsetHeight / 2 : -minWindowHeight) + 'px';
		},

		close: function() {
			if (this.node) {
				var oldNode = this.node;
				this.node = null;
				$(oldNode).remove();
			}
		}
	};


	// Embedded image data for interface.
	var devianttidyicons = {
		dt: '',
		close: '',
		down: '',
		up: ''
	};

	// Quickly generate buttons using DA's button theme
	var makeButton = function(label, primary, clickEvent) {
		return $E('button', {className: primary ? 'smbutton smbutton-green' : 'smbutton', events: {click: clickEvent}}, [
			$E('span', [label])
		]);
	};

	// Compiled, minified CSS from devianttidy.less
	var devianttidycss = "body.dt-limit-width{margin-left:auto!important;margin-right:auto!important}body.dt-limit-width.dt-limit-width.l1{max-width:1200px!important}body.dt-limit-width.dt-limit-width.l2{max-width:1400px!important}body.dt-limit-width.dt-limit-width.l3{max-width:1600px!important}#artist-comments hr,#output hr:not([class]),.devianttidy-dialog hr,.previewcontainer hr,.thought .body hr{display:block!important;border:1px solid transparent!important;border-top-color:#9DB1B0!important;border-bottom-color:#E9EFE8!important}.mc-ctrl,.mcb-note-box,.mcbox>.ch-ctrl,div.mcbox-inner-preview{border-radius:0!important}#output a.a,#output div.alink a{text-decoration:none!important}#output a.a:hover,#output div.alink a:hover{text-decoration:underline!important}#output a.a:visited{opacity:.7}body>div.drag-and-collect{display:none!important}.smbutton-blue:focus{outline:#000 dotted 1px!important}#overhead-collect.dt-top-nav-fixed{position:fixed!important;z-index:151!important}:not(.oh-eax)>#oh-mainmenu #more7-main.dt-hide-nav-labels>a:hover{min-width:8em!important}:not(.oh-eax)>#oh-mainmenu #more7-main.dt-hide-nav-labels>a:not(:hover){font-size:0!important;padding-right:0!important}:not(.oh-eax)>#oh-mainmenu #more7-main.dt-hide-nav-labels>a:not(:hover)>sup{display:none!important}body.dt-hide-core-ad #oh-menu-upgrade{display:none!important}.friendmachine .controls{padding-left:5px!important}.friendmachine .friendmachine>.readout>dl>dd.f{line-height:18px!important}.friendmachine .friendmachine>.readout>dl{margin-bottom:8px!important}textarea{font-size:12px!important;font-family:verdana,sans-serif}select{border:1px solid #ccc}.bubbleview>div.policy-page,.text.text-ii,p.critique-recommendation{width:auto!important;max-width:100%!important}table.zebra,table.zebra tr,table.zebra tr>*{border-collapse:collapse!important}#deviantlist td{padding:0 3px!important}#deviantlist tbody tr:hover td,#deviantlist tr.even:hover td{background:#DEE8E5}#tblGroups td.c input,#tblGroups+form td.c input{width:80px!important}body .cc-avatar{margin-top:1px!important}.ccomment{margin-bottom:8px!important}.cc-signature{float:none!important;font-size:90%;overflow-y:auto!important;max-height:15em!important;padding-bottom:1px!important}body.dt-scroll-comments .ctext .text-ii{overflow-y:auto!important;padding-bottom:1px}body.dt-scroll-comments.s1 .ctext .text-ii{max-height:17em!important}body.dt-scroll-comments.s2 .ctext .text-ii{max-height:34em!important}.dt-floating-comment{border:1px solid rgba(255,255,255,.5);background:rgba(211,223,209,.8);z-index:101!important;display:block!important;position:fixed!important;bottom:0!important;left:0;right:0;padding:15px 20px 0!important}.dev-view-about{z-index:101!important}div.talk-post div.pager-holder,div.talk-post div.pager2,div.talk-post textarea{height:150px}.talk-tower div.nest{padding-left:12px!important;margin-bottom:8px!important;border-left:solid 1px transparent!important}.talk-tower div.nest:hover{border-color:#a6b2a6!important}#deviant ul.list[style^=border-top]{border:none!important;margin-top:0!important;padding-top:0!important}body.dt-hide-group-box #gruze-main #gmi-GroupMemberZone{display:none!important}#any-joinrequest-module>.gr-configform{padding-left:0!important}.submit_to_groups .second_option>textarea{margin-left:0!important}#gmi-GMRoleEditor #gmi-BPPDropDown>div[style]{padding-left:30px!important}.frame-button.submit,body.dt-hide-morelikethis .mlt-link{display:none!important}body.dt-collapse-sidebar #deviant td.gruze-sidebar:not(:hover){width:15px!important}body.dt-collapse-sidebar #deviant td.gruze-sidebar:not(:hover)>*{display:none!important}body.dt-collapse-sidebar #browse2:not(.shopModuleBrowse) #browse-sidebar:not(:hover){max-width:15px!important}body.dt-collapse-sidebar #browse2:not(.shopModuleBrowse) #browse-sidebar:not(:hover)>*{display:none!important}body.dt-collapse-sidebar td+.gruze-sidebar:not(:hover){width:15px!important}body.dt-collapse-sidebar td+.gruze-sidebar:not(:hover)>*{display:none!important}body.editmode #modalspace>.modal{width:700px!important;margin-left:-350px!important}body.editmode #modalspace>.modal #dnd_deck_container,body.editmode #modalspace>.modal #dnd_deck_picker,body.editmode #modalspace>.modal>form{width:auto!important}body.editmode #modalspace>.modal input[type=text]{width:100%!important}body.editmode #modalspace>.modal textarea.css,body.editmode #modalspace>.modal textarea.text{height:250px!important}span.shadow>a.lit,span.shadow>span.blogthumb>div{font-size:86%!important;font-family:arial,sans-serif!important}span.shadow>a.lit>q>strong{display:none!important}.gr-shoutbox div.pp>dl.shouts dd,.gr-shoutbox div.pp>dl.shouts dt{padding-left:20px!important}.gr-shoutbox div.h.p{width:auto!important;padding-right:85px!important;position:relative!important}.gr-shoutbox div.h.p dt{display:none!important}.gr-shoutbox dl.shouts .timestamp{font-size:80%!important;opacity:.7!important}.gr-shoutbox dl.shouts input[type=text]{width:100%!important}.gr-shoutbox dl.shouts input[type=submit]{position:absolute!important;right:6px!important;width:55px!important;top:5px!important}.dd-heading{margin:0!important}#deviation_critiques div.critique,div.critique_feedback{width:auto!important;margin-right:135px!important}body.dt-hide-share-buttons #gmi-ResourceViewShare{display:none!important}.ile_edit_in_muroimport{margin-top:0!important}.ile_edit_in_muroimport>span.button-title,body.dt-hide-sidebar-thumbs .deviation-mlt-preview .stream,body.dt-hide-sidebar-thumbs h3>span.tiny-avatar{display:none!important}body.dt-hide-sidebar-thumbs h3.group_featured_title,body.dt-hide-sidebar-thumbs h3.more-from-da-title{background:0 0!important;padding-left:0!important}.dev-meta-producttabs>#printtabscontainer:not(:hover) #print-button{border-radius:5px!important;border-bottom:solid 1px #9ead98}.dev-meta-producttabs>#printtabscontainer:not(:hover)>#buy-tabs{display:none!important}.print-submit-help-bubble{display:none!important}#pointsdownload_widget:not(:hover)>.pdw_details{display:none!important}#pointsdownload_widget:not(:hover) #pdw_button_download{border-radius:5px!important}div.group_featured_list{position:relative!important;padding:32px 0 0!important;min-height:35px;max-height:285px;overflow-y:auto;overflow-x:hidden}.not-in-group{text-align:center;margin-top:1em}.submit_to_groups_button{display:inline!important}.submit_to_groups_link{margin:0!important}#groups_links{position:absolute!important;top:0!important;padding:0!important}#all_groups{float:right!important;padding:4px 0 0 1em!important}.dev-metainfo-copy-control{clear:both;margin-top:0!important}.dev-metainfo-copy-control br{display:none!important}.dev-metainfo-copy-control strong{display:inline-block;min-width:96px}.dev-view-about-content{display:block!important;opacity:1!important}body.dt-hide-forum-icons #thread #reply form table table,body.dt-hide-forum-icons #thread .forum img:not([src*='/lock.']):not([src*='/sticky.']),body.dt-hide-forum-icons .mcbox-preview-forum .mcb-icon>img{display:none}#thread .forum br{display:none}#thread .forum .d-started-by a[title],#thread .forum span[title]{margin-left:10px;opacity:.5}#thread .forum .d-latest-reply,#thread .forum .d-started-by{white-space:nowrap}#thread .forum tr.thread td{padding-top:3px!important;padding-bottom:3px!important}#help-container .mglist,div[style*='rgb(222, 233, 229)']{background:0 0!important;padding:0 0 8px!important}.mglist li{padding-bottom:0!important}#messages h2.mczone-title{margin-right:0!important}#messages .mczone{border-bottom:none!important}#messages .messages-menu div.header img{display:none!important}#messages .messages-folder-zone a.maybedrop{background-position:0 -450px!important}#messages .no-folder-notice{font-size:90%!important}#messages .mczone-empty,#messages .talkmessage a.h+img,#messages .talkmessage div.h+a.h{display:none!important}.talkmessage table,.talkmessage td{width:100%!important}.talkmessage>table>*>*>td:first-child{width:0!important}div.message-simulator{padding:0!important}div.mcbox-inner-full-comment div.mcb-whoicon{top:0!important}div.mcbox-sel>div>span.mcx{top:4px!important;right:4px!important}div.mcbox-sel>div>span.mcdx{top:4px!important;right:22px!important}div.mcbox-sel-thumb>div>span.mcdx,div.mcbox-sel-thumb>div>span.mcx{margin-right:-2px!important}.talkmessage .mcb-body{width:auto!important}.talkmessage-taller{min-height:102px!important}body.dt-scroll-comments .talkmessage .mcb-body{overflow-y:auto!important}body.dt-scroll-comments.s1 .talkmessage .mcb-body{max-height:10em!important}body.dt-scroll-comments.s2 .talkmessage .mcb-body{max-height:20em!important}.mcbox-leech{margin-top:-4px!important;margin-left:0!important;border-left:none!important}.mcbox-leech.mcbox-sel{margin-left:-1px!important}.mcbox-full .mcbox-inner{margin-bottom:5px!important}.talk-post .inputs{padding:4px 0!important}.mcbox-inner-full-stack .talkmessage-comment.al{width:90%!important;min-height:0!important;padding:4px 6px!important}.mcbox-inner-full-stack a.ts-lnk{color:inherit!important}.popup2-mcbox-comment{width:500px!important;height:auto!important;min-height:150px;max-height:270px}#messages .mcb-tab{margin-top:25px!important}#deviantART-v7 #messages .mcb-tab>a{padding:0!important}#messages .mcb-tab>a>.tabtext{border-radius:0!important}#notes .left-column{width:40%!important}#notes .right-column{width:59%!important}#notes li{padding-top:3px!important;padding-bottom:3px!important}body.dt-scroll-comments #notes:not(.note-modal) .previewcontainer{height:auto!important}body.dt-scroll-comments.s1 #notes:not(.note-modal) .previewcontainer{max-height:20em!important}body.dt-scroll-comments.s2 #notes:not(.note-modal) .previewcontainer{max-height:40em!important}#solid-gone .altview+.sleekadbubble,#solid-gone .sleekadbubble+.altview,#solid-gone>img[src*=fella],#solid-gone>img[src*=fella]+div{display:none}#solid-gone div.altview{margin-left:auto!important;margin-right:auto!important}#solid-gone input[style='width: 120px']{width:180px!important}#solid-gone #forgot-container{margin-left:0;width:auto}.devianttidy-dialog{display:block!important;position:fixed!important;top:0;left:0;bottom:0;right:0;background:rgba(0,0,0,.5);z-index:200!important;padding:2em}.devianttidy-dialog>div{position:absolute;left:50%;top:50%;margin-left:-30em!important;width:60em}.devianttidy-dialog a{color:#3B5A4A!important}.devianttidy-dialog .dialog-icon{padding-right:.3em}.devianttidy-dialog .dialog-close{position:absolute;top:2px;right:6px;cursor:pointer;padding:4px}.devianttidy-dialog .dialog-close>img{margin:0}.devianttidy-dialog .dialog-body{margin:8px!important;overflow-y:auto}.devianttidy-dialog .dialog-category{margin-top:.5em;font-weight:700}.devianttidy-dialog .dialog-control{position:relative;margin-left:26px}.devianttidy-dialog .dialog-control input{position:absolute;left:-18px;margin-top:0}.devianttidy-dialog .dialog-control select{position:absolute;right:0;margin-top:-4px;border:1px solid #ccc}.devianttidy-dialog .hint{font-size:90%;color:#676}.devianttidy-dialog .dialog-buttons{margin-top:1em}.devianttidy-dialog .dialog-buttons button{margin:0 .3em}";

	// Bundle up these utility methods into a single container object that can be passed to other scripts
	var devianttidyutils = {
		$: $,
		$E: $E,
		inDynamicPage: inDynamicPage,
		getUsername: getUsername,
		devianttidyicons: devianttidyicons,
		devianttidydialog: devianttidydialog
	};

	// The DeviantTidy application
	var devianttidy = {
		version: '4.7.9',
		debug: false,
		homepage: 'https://www.deviantart.com/deviation/45622809/',
		body: null,
		
		changelist: [
			"Updated: DeviantTidy will look for updates using HTTPS because DA now officially supports it.",
			"Fixed: Options link moved from the page footer to the user menu on the top navigation.",
			"Fixed: Modal dialogs will resize if the window is resized after they're opened."
		],

		log: function(message, alertme) {
			if (!this.debug) {return;}
			console.log(message);
			if (alertme) {
				alert("DeviantTidy Debug Message:\n\n" + message);
			}
		},

		preload: function() {
			GM_addStyle(devianttidycss);
		},

		start: function() {
			if (document.readyState !== "interactive") {
				return;
			}

			// Add the CSS again as a workaround for Firefox 55.
			// Internal changes may have broken GM_addStyle during preload.
			GM_addStyle(devianttidycss);
			
			this.body = $('body')[0];

			if (!Function.prototype.bind) {
				alert("DeviantTidy requires an up-to-date browser in order to function."); return;
			}
			if (unsafeWindow.devianttidy) {
				this.log("Another instance of DeviantTidy has already loaded!"); return;
			}

			// Allow this application and its utilities to be accessed by other scripts through unsafeWindow
			unsafeWindow.devianttidy = this;
			unsafeWindow.devianttidyutils = devianttidyutils;

			// Fresh update?
			if (GM_getValue('version') !== this.version) {
				GM_setValue('version', this.version);

				devianttidydialog.open('DeviantTidy ' + this.version + ' Installed', [
					$E('div', {className: 'pp'}, [
						"You can view all available options by clicking 'DeviantTidy Options' under the user menu on the header navigation. ",
						$E('a', {href: "#", className: 'a', events: {click: devianttidy.preferences}}, ["Configure DeviantTidy right now"]),
						"."
					]),
					$E('hr'),
					$E('div', {className: 'pp'}, this.changelist.map(function(c){return $E('div', [c]);})),
					$E('hr'),
					$E('div', {className: 'pp'}, ["You will not see this message again.	 Close this panel to continue browsing."]),
					$E('div', {className: 'pp c'}, [makeButton("Cheers!", true, function() {devianttidydialog.close();})])
				]);
			}

			// Run through all options
			this.dispatch();

			// Silently look for updates every 2 days - alert user if new version is available
			if (location.href.indexOf('http://my.deviantart') === 0 || location.href.indexOf('http://www.deviantart') === 0) {
				var now = new Date();
				var last = GM_getValue('last_updated', 0);
				if (!last || Date.parse(last).valueOf() < now - 48 * 3600 * 1000) {
					this.update(true);
				}
			}

			// Add Greasemonkey menu item
			GM_registerMenuCommand("DeviantTidy Options", this.preferences);

			// Add the Options link to the header nav
			$('#oh-menu-deviant li.oh-menu-list-item .i8').closest('li').after(
				$E('li', {className: 'oh-menu-list-item'}, [
					$E('a', {id: 'devianttidy-options-link', className: 'mi iconset-messages', href: '#', events: {click: this.preferences}}, [
						$E('i', {className: 'i8'}),
						'DeviantTidy Options'
					])
				])
			);
		},

		preferences: function() {
			var controls = [];

			// Generate options controls
			for (var o in devianttidy.options) {
				// Options without descriptions are hidden functions, but their preferences can be set manually
				var option = devianttidy.options[o];

				if (option.category) {
					controls.push($E('div', {className: 'p dialog-category'}, [option.category]));
				}

				if (option.description) {
					var control;
					var control_id = 'devianttidy-control-' + o;
					var description = [option.description, $E('b', [(option.custom ? ' (add-on)' : '')])];

					if (option.choices) {
						// If a list of choices is provided, the options form a drop-down list
						var selections = [];

						for (var c in option.choices) {
							selections.push($E('option', {value: c}, [option.choices[c]]));
						}

						selections[option.pref !== undefined ? option.pref : option.initial].selected = true;

						control = [
							$E('select', {id: control_id, name: o, events: {change: function() {devianttidy.options[this.name].pref = this.value;}}}, selections),
							$E('label', {htmlFor: control_id}, description)
						];
					}
					else {
						// Otherwise, use a checkbox
						control = [
							$E('input', {type: 'checkbox', id: control_id, name: o, checked: option.pref, events: {change: function() {devianttidy.options[this.name].pref = this.checked ? 1 : 0;}}}),
							$E('label', {htmlFor: control_id}, description)
						];
					}

					if (option.hint) {
						control.push($E('div', {className: 'hint'}, [option.hint]));
					}

					controls.push($E('div', {className: 'p dialog-control'}, control));
				}
			}

			devianttidydialog.open("DeviantTidy Options", [
				$E('div', {className: 'p r'}, [
					$E('a', {href: devianttidy.homepage}, ["DeviantTidy"]),
					" version " + devianttidy.version + ". ",
					$E('a', {href: "#", events: {click: function() {devianttidy.update();}}}, ["Check for updates"])
				]),
				$E('div', {className: 'p'}, controls),
				$E('div', {className: 'dialog-buttons c'}, [
					makeButton("Save & Reload", true, function() {devianttidy.save(); devianttidy.reload();}),
					makeButton("Reset", false, function() {devianttidy.reset();}),
					makeButton("Cancel", false, function() {devianttidydialog.close();})
				])
			]);

			return false;
		},

		load: function() {
			for (var o in this.options) {
				this.options[o].pref = parseInt(GM_getValue("options." + o, this.options[o].initial));
			}
		},

		save: function() {
			for (var o in this.options) {
				GM_setValue("options." + o, this.options[o].pref);
			}
		},

		reset: function() {
			if (confirm("This will reset all DeviantTidy options to their default values, and then reload the page.")) {
				for (var o in this.options) {
					this.options[o].pref = this.options[o].initial;
				}
				this.save();
				this.reload();
			}
		},

		dispatch: function() {
			this.load();

			var dispatch_start = new Date();
			var dispatch_log = [];
			var dispatch_count = 0;
			var dispatch_fails = 0;

			for (var o in this.options) {
				var option = this.options[o];

				// A lazy option should only run when the pref is not 0.
				// If dispatcher is run again, don't repeat functions that were already run.
				if ((option.lazy && option.pref === 0) || option.dispatched) {
					continue;
				}

				try {
					var dispatch_time = new Date();
					var dispatch_result = option.method(option.pref);
					dispatch_log.push("	 + " + o + ": " + dispatch_result + " (" + (new Date() - dispatch_time) + "ms)");
				}
				catch (e) {
					dispatch_fails++;
					dispatch_log.push("	 ! " + o + ": FAILED (" + e.message + " - line " + e.lineNumber + ")");
				}

				option.dispatched = true;
				dispatch_count++;
			}

			if(this.debug) {
				var elapsed = new Date() - dispatch_start;
				this.log("Dispatched " + dispatch_count + " function(s) in " + elapsed + "ms.\n" + dispatch_log.join("\n"));

				if (dispatch_fails > 0) {
					this.log(dispatch_fails + " dispatch method(s) failed. Check the Error Console for details.", true);
				}
			}
		},

		reload: function() {
			devianttidydialog.open('Reloading...', "Reloading page.	 Please wait...");
			location.reload();
		},

		update: function(quiet) {
			this.log("Looking for updates...");
			GM_setValue('last_updated', new Date().toString());

			if (!quiet) {
				devianttidydialog.open('Looking for Updates...', "Checking for a new version of DeviantTidy. Please wait...");
			}

			var update_error = function(jqXHR, message) {
				if (!quiet) {
					devianttidydialog.open('Error', [
						$E('div', {className: 'ppp c'}, [
							"Unable to get the version information from ",
							$E('a', {href: devianttidy.homepage}, [devianttidy.homepage]),
							".	Please try visiting the page yourself to check for updates."
						]),
						$E('div', {className: 'ppp c'}, [
							(typeof message === "string") ? message : "No error details available."
						])
					]);
				}
			};

			var update_success = function(html) {
				var version_text = html.match(/<b><\/b><b><i>version ([\d.]+)<\/i><\/b><b><\/b>/i);
				if (!version_text) {
					update_error(null, "Couldn't find version number string on the page.");
					return;
				}

				var message;
				var version_number = version_text[1];

				if (version_number === devianttidy.version) {
					devianttidy.log("No newer version available.");
					if (quiet) {return;}
					message = ["Your version of DeviantTidy is up to date!"];
				}
				else {
					message = [$E('b', ["DeviantTidy " + version_number + " is available. "])];
					message.push($E('a', {href: devianttidy.homepage}, ["Go to the DeviantTidy homepage"]));
					message.push(" to update your style and script.");
				}
				devianttidydialog.open('Update Status', [$E('div', {className: 'ppp c'}, message)]);
			};

			GM_xmlhttpRequest({
				method: "GET",
				url: devianttidy.homepage,
				onload: function(response) {
					if (response.status === 200) {update_success(response.responseText);}
					else {update_error(null, "Response code: " + response.status);}
				},
				onerror: update_error
			});
		},

		extend: function(object) {
			// Use this function to add your own options to DeviantTidy.
			// Review the example add-on in the project source for documentation.
			if (typeof object.name !== 'string' || typeof object.method !== 'function' || typeof object.description !== 'string') {
				this.log("Attempt to extend with a malformed add-on");
				this.log(object);
				alert("DeviantTidy doesn't like the structure of the option you tried to add.\n" +
					"Please check that you set the required parameters and that their types are correct.");
				return;
			}
			else if (this.options[object.name]) {
				this.log("Attempt to extend resulted in a name-clash.");
				alert("The DeviantTidy option '" + object.name + "' already exists.	 You cannot extend it.");
				return;
			}

			if (typeof object.initial === 'undefined') {
				object.initial = 1;
			}
			object.custom = true;
			this.options[object.name] = object;
			this.log("Extended with custom function '" + object.name + "'");

			// We must delegate this to a timeout because executing it under unsafeWindow will
			// result in access violations when calling GM functions
			window.setTimeout(function() {devianttidy.dispatch();}, 1);
		},

		options: {
			'hide_forum_icons': {
				category: "Hidden Elements",
				description: "Hide forum thread icons",
				initial: 1,
				lazy: true,
				method: function(pref) {
					$(devianttidy.body).addClass("dt-hide-forum-icons");
					return true;
				}
			},
			'hide_nav_labels': {
				description: "Hide text labels on the sticky navigation bar until I hover over them",
				initial: 0,
				lazy: true,
				method: function(pref) {
					$('#more7-main').addClass("dt-hide-nav-labels");
					return true;
				}
			},
			'no_group_box': {
				description: "Hide the blue 'Contribute' box at the top of all group pages",
				initial: 0,
				lazy: true,
				method: function(pref) {
					$(devianttidy.body).addClass('dt-hide-group-box');
					return true;
				}
			},
			'no_share_buttons': {
				description: "Hide the social sharing buttons on deviation pages",
				initial: 0,
				lazy: true,
				method: function(pref) {
					$(devianttidy.body).addClass('dt-hide-share-buttons');
					return true;
				}
			},
			'hide_morelikethis': {
				description: "Hide the 'More like this' links on gallery thumbnails",
				initial: 0,
				lazy: true,
				method: function(pref) {
					$(devianttidy.body).addClass('dt-hide-morelikethis');
					return true;
				}
			},
			'hide_core_ad': {
				description: "Hide the 'Upgrade to CORE' ad in the header",
				initial: 0,
				lazy: true,
				method: function(pref) {
					$(devianttidy.body).addClass('dt-hide-core-ad');
					return true;
				}
			},
			'hide_sidebar_thumbs': {
				description: "Hide 'More from...' thumbnails in the deviation page sidebar",
				initial: 1,
				lazy: true,
				method: function(pref) {
					$(devianttidy.body).addClass('dt-hide-sidebar-thumbs');
					return true;
				}
			},
			'collapse_sidebar': {
				category: "UI Tweaks",
				description: "Collapse the folders/categories sidebar on galleries and browse pages",
				hint: "The sidebar will be invisible until you mouse-over the left edge of the page.",
				initial: 0,
				lazy: true,
				method: function(pref) {
					$(devianttidy.body).addClass('dt-collapse-sidebar');
					return true;
				}
			},
			'top_nav_fixed': {
				description: "Fix the navigation bar to always be visible at the top of the screen",
				initial: 0,
				lazy: true,
				method: function(pref) {
					$('#overhead-collect').addClass('dt-top-nav-fixed');
					return true;
				}
			},
			'scroll_comments': {
				description: "Add scrollbars to long comments and notes",
				initial: 2,
				lazy: true,
				choices: ["Disabled", "Small (approx 15 lines)", "Large (approx 30 lines)"],
				method: function(pref) {
					$(devianttidy.body).addClass("dt-scroll-comments s" + pref);
					return true;
				}
			},
			'limit_width': {
				description: "Limit the maximum width of pages on wide screens",
				initial: 0,
				lazy: true,
				choices: ["No limit", "1200 pixels", "1400 pixels", "1600 pixels"],
				method: function(pref) {
					$(devianttidy.body).addClass("dt-limit-width l" + pref);
					return true;
				}
			},
			'short_titles': {
				category: "Page Titles",
				description: "Shorten page titles by removing DeviantArt prefixes and suffixes",
				hint: "For instance, a window or tab named 'Spyed on DeviantArt' will be shortened to 'Spyed'.",
				initial: 1,
				lazy: true,
				method: function(pref) {
					document.title = document.title.replace(/^(DeviantArt): where ART meets application!$|^DeviantArt: | on DeviantArt$| DeviantArt( \w+)$| - DeviantArt$/i, '$1$2');
					return true;
				}
			},
			'message_center_title': {
				description: "Show message count in the Notification Center page title",
				initial: 1,
				lazy: true,
				method: function(pref) {
					if (window.location.href.indexOf('//www.deviantart.com/notifications/') < 0) {return -1;}
					var msgs = 0;
					var m;
					$("#overhead .oh-keep > a > span, #oh-menu-split > a > span").each(function() {
						m = parseInt($(this).text().replace(',', ''), 10);
						if (!isNaN(m)) {msgs += m;}
					});
					if (msgs) {
						document.title = msgs + " Notification" + (msgs === 1 ? "" : "s");
					}
					return msgs;
				}
			},
			'note_title': {
				description: "Show the subject of the selected note in the page title",
				initial: 1,
				lazy: true,
				method: function(pref) {
					if (window.location.href.indexOf('//www.deviantart.com/notifications/notes/') < 0) {return -1;}

					var mutationHandler = function () {
						var subjectElement = notesBlock.find('.mcb-title');
						if (subjectElement.size() === 1) {
							document.title = subjectElement.text() + " - Notes";
						}
					};

					var notesBlock = $('.notes-right');
					new MutationObserver (mutationHandler).observe(notesBlock[0], { childList: true, subtree: true });
					mutationHandler();
					return true;
				}
			},
			'strip_outgoing_links': {
				category: "Extra Features",
				description: "Strip DeviantArt redirects from external links",
				initial: 1,
				choices: ["No", "Yes", "Prompt"],
				lazy: true,
				method: function(pref) {
					var prefix = /https?:\/\/www.deviantart.com\/users\/outgoing\?(.+)/i;

					// Check every link that gets focus and apply rules based on its href.
					$(devianttidy.body).on('focus', 'a.external', function(evt) {
						var link = $(this);
						var href = link.attr('href');
						if (href) {
							var matches = prefix.exec(href);
							if (matches.length == 2) {
								link.attr('href', matches[1]);
								link.addClass('dt-external-link');
							}
						}
					});

					// If prompt mode is on, add a click listener to external links.
					if (pref === 2) {
						$(devianttidy.body).on('click', 'a.dt-external-link', function(evt) {
							var href = $(this).attr('href');
							var msg = "This external link redirects to:\n\n" + href + "\n\n" + "Press OK to follow.";
							return confirm(msg);
						});
					}

					return true;
				}
			},
			'redirect_on_login': {
				description: "Redirect to a specific page after logging in",
				initial: 0,
				lazy: true,
				choices: ["Disabled", "Notification Center", "Channels", "My Profile Page"],
				method: function(pref) {
					var username = getUsername();
					var username_old = GM_getValue('username', username);
					GM_setValue('username', username);

					if (!username || username_old === username) {return -2;}

					var redirects = {
						1: {url: 'http://www.deviantart.com/notifications/', name: 'Notification Center'},
						2: {url: 'http://www.deviantart.com/channels/', name: 'Channels'},
						3: {url: 'http://' + username.toLowerCase() + '.deviantart.com/', name: 'your profile page'}
					};

					var url = redirects[pref].url;
					if (location.href.indexOf(url) === 0) {return -1;}

					devianttidydialog.open("Logged In", "Going to " + redirects[pref].name + "...");
					location.href = url;
					return true;
				}
			},
			'disable_dnd': {
				description: "Disable drag-and-drop thumbnail collecting",
				hint: "Note that while drag-and-drop is disabled, you will be unable to perform actions like selecting and moving items in your notifications or gallery.",
				initial: 0,
				lazy: true,
				method: function(pref) {
					if (unsafeWindow.DDD) {return delete unsafeWindow.DDD;}
					return false;
				}
			},
			'floating_comment_key': {
				description: "Shortcut key for the Floating Comment feature (ALT + SHIFT + ...)",
				initial: 2,
				choices: "ABCDEFGHIJKLMNOPQRSTUVWXYZ".split(''),
				lazy: false,
				method: function(pref) {
					var comment_toggle = function() {
						try {
							var form = $('form[id="cooler-comment-submit"]:last')[0];
							var textarea = $(form).find('.writer,textarea')[0];

							if (!form || !textarea) {
								devianttidydialog.open("Unavailable", [$E('p', {className: 'p c'}, ["No comment box found."])], 1500);
								return false;
							}

							devianttidydialog.close();
							if (form.className.indexOf('dt-floating-comment') >= 0) {
								form.className = form.className.substr(0, form.className.length - 20);
								textarea.blur();
							}
							else {
								form.className += ' dt-floating-comment';
								textarea.focus();
								textarea.click();
							}

							return false;
						}
						catch (e) {
							devianttidy.log("Error getting floating comment box: " + e.message, true);
						}
					};

					// Make an invisible link with access key C to listen for this keystroke.
					devianttidy.body.appendChild($E('a', {
						id: 'dt-floating-comment-link',
						href: '#',
						style: {display: 'none'},
						accessKey: "ABCDEFGHIJKLMNOPQRSTUVWXYZ".split('')[pref],
						events: {click: comment_toggle}
					}));

					return 1;
				}
			},
			'keyboard_browsing': {
				description: "Use left/right arrow keys to browse galleries and notifications",
				initial: 1,
				lazy: true,
				method: function(pref) {
					window.addEventListener('keyup', function(e) {
						// Respond only to keystrokes without modifiers.
						if (e.ctrlKey || e.shiftKey || e.altKey) {return;}

						// Determines whether we're within a dynamically-created deviation page.
						// If viewing a deviation, don't override deviation key listeners.
						if (inDynamicPage()) {return;}

						var evt = e || window.event;
						var target = evt.target;

						while (target.nodeType === 3 && target.parentNode !== null) {
							target = target.parentNode;
						}

						var node = target.nodeName;
						if (node === 'TEXTAREA' || node === 'SELECT' || target.hasAttribute('contenteditable')) {
							return;
						}
						else if (node === 'INPUT') {
							// On browse pages, the search box is always focused on load.
							// Continue anyway if the search box is empty, or if its value is
							// exactly equal to the current search criteria.
							if (target.name !== 'q') {return;}
							var urldecode = function(str) {
								return decodeURIComponent((str+'').replace(/\+/g, '%20'));
							};
							if (target.value && !urldecode(location.href).match('q=' + target.value + "($|&)")) {return;}
						}

						var find;

						switch (evt.keyCode) {
							case 37: find = ".shadow a.l:eq(0), .pagination li.prev a"; break;
							case 39: find = ".shadow a.r:eq(0), .pagination li.next a, .pagination .load_more"; break;
							default: return;
						}

						var link = $(find);
						if (!link.length) {
							devianttidy.log("No link found ", true);
							return;
						}

						link[0].click();
					}, true);

					return true;
				}
			}
		}
	};

	devianttidy.preload();
	document.onreadystatechange = devianttidy.start.bind(devianttidy);
}());