artificeren / Fix BanniNation

// ==UserScript==
// @name		Fix BanniNation
// @description	fixes up various parts of the bn ui
// @version		25
// @namespace	http://www.bannination.com/fixbn
// @include		http://www.bannination.com/*
// @include		http://bannination.com/*
// @grant		GM_getValue
// @grant		GM_setValue
// @grant		GM_log
// @grant		GM_addStyle
// @grant		GM_getResourceText

// @require		https://ajax.aspnetcdn.com/ajax/jquery/jquery-1.11.0.min.js
// @require		https://ajax.aspnetcdn.com/ajax/jquery.migrate/jquery-migrate-1.2.1.min.js
// @require		https://ajax.aspnetcdn.com/ajax/jquery.ui/1.10.3/jquery-ui.min.js
// @require		https://raw.github.com/markitup/1.x/master/markitup/jquery.markitup.js
// @require		https://raw.github.com/sizzlemctwizzle/GM_config/master/gm_config.js
// @require		https://raw.github.com/medialize/URI.js/gh-pages/src/URI.min.js
// @require		https://raw.github.com/accursoft/caret/master/jquery.caret.js
// @require		https://raw.github.com/dimsemenov/Magnific-Popup/master/dist/jquery.magnific-popup.js
// @require		https://raw.github.com/needim/noty/master/js/noty/packaged/jquery.noty.packaged.min.js
// @require		https://raw.github.com/bgrins/spectrum/master/spectrum.js
// @require		https://raw.github.com/ksylvest/jquery-age/master/javascripts/jquery.age.js
// @require		https://raw.github.com/artificeren/jqSmartTag/master/site/script/jquery.smartTag.js
// @require		https://raw.github.com/silvestreh/onScreen/master/jquery.onscreen.js

// @resource	juipepper	https://ajax.aspnetcdn.com/ajax/jquery.ui/1.8.10/themes/pepper-grinder/jquery-ui.css
// @resource	magnificcss https://raw.github.com/dimsemenov/Magnific-Popup/master/dist/magnific-popup.css
// @resource	spectrumcss	https://raw.github.com/bgrins/spectrum/master/spectrum.css
// ==/UserScript==
/* jshint  browser: true*/
/* global $, jQuery, URI, GM_config, GM_addStyle, GM_getResourceText, GM_configStruct, GM_getValue, GM_setValue, noty */

// main code
var __fixbn = null;
try {
	(function ($) {
		"use strict";

		var FixbN;
		var $bind = function (fn, me) { return function () { return fn.apply(me, arguments); }; };
		var $root = null;

		FixbN = (function () {
			// ctor
			function FixbN() {
				this.fix = $bind(this.fix, this);
				this.fixAllPages = $bind(this.fixAllPages, this);
				this.fixHeadlinesPages = $bind(this.fixHeadlinesPages, this);
				this.fixCommentsPage = $bind(this.fixCommentsPage, this);
				this.fixQueuePage = $bind(this.fixQueuePage, this);
				this.fixTaggers = $bind(this.fixTaggers, this);
				this.createDateUrl = $bind(this.createDateUrl, this);

				var mainConfigFrame = $("<div style='display:none;' />")[0];
				document.body.appendChild(mainConfigFrame);
				GM_config.init({
					'id': 'FixbNconfig',
					'title': 'Fix bN Global Settings',
					'fields': {
						'switchColumns': {
							'label': 'Switch Headline and Comment Columns',
							'type': 'checkbox',
							'default': true
						},
						 'hideModThreads': {
							'label': 'Hide all Moderated Threads',
							'type': 'checkbox',
							'default': false
						},
						'fixedHeader': {
							'label': "Keep Header at Top of Window",
							'type': 'checkbox',
							'default': true
						},
						'scrollArrow': {
							'label': 'Show a Floating Arrow for Scrolling',
							'type': 'checkbox',
							'default': true
						},
						'stickyTagger': {
							'label': "Show a Floating Tagger in Comment Threads",
							'type': 'checkbox',
							'default': true
						},
						'tagNotyDuration': {
							'label': 'Tag Notification Duration',
							'type': 'int',
							'min': 3,
							'max': 30,
							'default': 6
						},
						'repliesOveride': {
							'label': "Override the masterbn replies notification",
							'type': 'checkbox',
							'default': true
						},
						'ignoreReplies': {
							'label': "Ignoring a User Also Ignores All Replies To That User",
							'type': 'checkbox',
							'default': true
						},
						'showRepliesToMe': {
							'label': "Show a List of Comments That Are Replies To Me",
							'type': 'checkbox',
							'default': true
						},
						'nabbitVisibility': {
							'label': "Enable Nabbit User Images (Can Be Overridden Per-User)",
							'type': 'checkbox',
							'default': true
						},
						'highScoreThreshold': {
							'label': "Mark Posts With Scores Over:",
							'type': 'int',
							'default': '4'
						},
						'autoLinkComments': {
							'label': "Auto-linkify pasted URLs",
							'type': 'checkbox',
							'default': false
						}
					},
					'frame': mainConfigFrame
				});
				GM_config.onSave = function () { if (GM_config.isOpen) { GM_config.close(); window.location.reload(); } };
				GM_config.onOpen = function (doc, win, frame) {
					var notyDuration = $(frame).find("input[id$='tagNotyDuration']");
					notyDuration.css("width", "3em");

					$(frame).find("input[id$='highScoreThreshold']").css("width", "3em");

					var wrapper = $(frame).find("div#FixbNconfig_wrapper");
					var extras = $("<div style='float:right; text-align:right; font-weight: bold; margin-top:40px; margin-right:10px;'></div>");
					wrapper.prepend(extras);

					
					$("<p><a style='font-size:14px; color:black;' href='http://www.bannination.com/comments/5182401/'>Bannination.com Feedback Thread</a></p>").appendTo(extras);
					$("<p><a style='font-size:14px; color:black;' href='https://github.com/artificeren/FixbN/issues?state=open'>Fix bN GitHub Project Feedback</a></p>").appendTo(extras);

					var bitCoinDonate = $('<div style="font-size:14px;margin:0 auto;width:300px" class="blockchain-btn" data-address="1Fe2d6giUvTMJniufMyivybg6v6HQGpeXG" data-shared="false"> <div class="blockchain stage-begin"><p style="text-decoration:underline;cursor:pointer;"> Donate Bitcoin to Fix bN </p></div> <div class="blockchain stage-loading" style="text-align:center"> <img src=""/> </div> <div class="blockchain stage-ready"> <p>Bitcoin Address: <span style="font-size:12px;">[[address]]</span></p> <p class="qr-code"></p> </div> <div class="blockchain stage-paid"> <p>Donation of <b>[[value]] BTC</b> Received. Thank You.</p> </div> <div class="blockchain stage-error"> <font color="red">[[error]]</font> </div> </div>');
					bitCoinDonate.appendTo(extras).bitcoin();

					$(frame).find("#FixbNconfig_saveBtn").text(" OK ");
					$(frame).find("#FixbNconfig_closeBtn").text("Cancel");
				};
				
			}

			FixbN.prototype.fix = function () {

				this.$root = $("html, body");

				try {
					var firstMenuItem = $("div#menu ul:first li:first");
					$("<li><span title='Fix bN Version {0}' style='text-decoration:underline;cursor:pointer;'>fix bN</span></li>".fex(GM_info.script.version))
						.insertAfter(firstMenuItem)
						.click(function () { GM_config.open(); })
						.attr("title", "Settings for the Fix bN addon");
					firstMenuItem.after("\r\n\r\n");

				} catch (ex) {
					console.error("FixbN failed adding options menu item", ex);
				}

				try {
					this.fixAllPages();
				} catch (ex) {
					console.error("FixbN failed the allPages fixup", ex);
				}

				try {
					this.fixHeadlinesPages();
				} catch (ex) {
					console.error("FixbN failed the Headlines page fixup", ex);
				}

				try {
					this.fixCommentsPage();
				} catch (ex) {
					console.error("FixbN failed the Comments page fixup", ex);
				}

				try {
					this.fixQueuePage();
				} catch (ex) {
					console.error("FixbN failed the Queue page fixup", ex);
				}

				try {
					this.fixTaggers();
				} catch (ex) {
					console.error("FixbN failed the Tagging fixup", ex);
				}

			};

			FixbN.prototype.fixAllPages = function () {
				// there was a bug where jsessionid caused problems, not sure if still needed
				if ($("#banner a").first().attr("href").indexOf("jsessionid") > 0) {
					document.location.reload();
				}

				// optionally keep the header fixed to the top of the window
				if (GM_config.get("fixedHeader")) {
					var header = $("div#header");
					header.addClass("fixedHeader");
					$("div#main").addClass("fixedHeader");
					var $style = $("<style type='text/css'>").appendTo('head'); 

					var bindTopMargin = function() {
						$("body").css("margin-top", $("div#header").height());
						var $style = $("<style type='text/css'>").appendTo('head');
						var adjustment = $("div#header").height() + 10;
						$style.html("div#main.fixedHeader a:target { padding-top: " + adjustment + "px; margin-top: -" + adjustment + "px; }");
					};
					$(window).resize(bindTopMargin);
					bindTopMargin();
					window.setTimeout(bindTopMargin, 200);

				} else {
					$("<style type='text/css'> div#main a:target { padding-top: 50px; margin-top: -50px; } </style>").appendTo('head');
				}

				// scroll To Bottom/Top buttons
				if (GM_config.get("scrollArrow")) {
					var scrollToBottom = $("<div class='fbnScrollToBottom' style='width:64px; height:64px;'><img title='Scroll To Bottom' alt='Scroll To Bottom' src='' /></div>");
					scrollToBottom
						.prependTo($("div#main"))
						.click((function () { this.$root.animate({ scrollTop: $(document).height() - $(window).height() }); }).bind(this));

					var scrollToTop = $("<div class='fbnScrollToTop' style='display: none; width:64px; height:64px;'><img title='Scroll To Top' alt='Scroll To Top' src='' /></div>");
					scrollToTop
						.prependTo($("div#main"))
						.click((function () { this.$root.animate({ scrollTop: 0 }); }).bind(this));

					$("div#footer").onScreen({
						doIn: function () {
							$("div.fbnScrollToBottom").hide();
							$("div.fbnScrollToTop").show();
						},
						doOut: function () {
							$("div.fbnScrollToTop").hide();
							$("div.fbnScrollToBottom").show();
						}
					});
				}

				// split the menu
				var leftMenu = $("<ul class='leftMenu'></ul>");
				leftMenu
				.append($("div#menu a[href$='/queue']").closest("li"))
				.append("<li><a href='{0}'>today</a></li>".fex(this.createDateUrl(0)))
				.append("<li><a href='{0}'>yesterday</a></li>".fex(this.createDateUrl(-1)))
				.append("<li><a href='{0}' title='Damn You, Dictionary! This Is A Great Word!'>ereyesterday</a>".fex(this.createDateUrl(-2)))
				.append($("div#menu a[href^='http://wiki']").closest("li"))
				.append("<li><a href='/comments/1000' class='fbnBgLink " + $("div#welcome a[href='/comments/1000']").attr("class") + "'>beer garden</a></li>")
				.prependTo("div#menu");

				if ((new Date()).getHours() < 8 && $("div#menu a:contains('dangerous mode')").length > 0) {
					leftMenu.append("<li class='nsfw'><a href='/comments/901' class='nsfw'>strip club</a></li>");
				}

				// consolidate the profile links
				try {
					var profilePopup = $("<div class='profilePopup' style='min-width:200px;margin:0px;padding:10px;color:white;position:absolute;display:none;right:0px;background:#373737;z-index:1001;' />");
					if (GM_config.get("fixedHeader")) {
						profilePopup.css("position", "fixed");
					}
					profilePopup.appendTo("body");
					var selfItem = $("div#menu li:contains('logged in as')");
					if (selfItem.length !== 0) {
						var selfLink = selfItem.find("a");
						selfLink.remove();
						selfItem.empty().append(selfLink).append(" ");

						var originalLogoutLink = selfLink.closest("li").next("li").find("a");
						originalLogoutLink.closest("li").hide();
						var list = selfLink.closest("ul");

						originalLogoutLink.appendTo(profilePopup);
						var profileLink = selfLink.clone();
						profileLink.text("profile");
						profilePopup.prepend(profileLink);

						profileLink.css("float", "right").addClass("topLink");
						originalLogoutLink.css("float", "left").addClass("topLink");

						var recentCommentWrapper = $("<div class='recentCommentWrapper' style='clear:both;margin-top:5px;' />");
						recentCommentWrapper.appendTo(profilePopup);
						var loadComments = function () {
							var userPageUrl = profileLink.attr("href");
							$.ajax({
								url: userPageUrl,
								dataType: "text",
								success: function (data, textStatus, jqXHR) {
									var comments = data.substring(data.indexOf("<h2>Recent Comments</h2>"));
									comments = comments.substring(0, comments.indexOf("<h2>Account Settings</h2>") - 1);
									recentCommentWrapper.css("overflow-y", "scroll").css("height", $(window).height() - 200);
									recentCommentWrapper.html(comments);
									profilePopup.css("width", $("div#menu").width() / 1.5).position({
										my: "right top",
										at: "right bottom",
										of: $("div#menu"),
										collision: "none"
									});
									recentCommentWrapper.find("h2").css({ "cursor": "pointer" }).click(loadComments);
								},
								error: function () {
									window.location.href = userPageUrl;
								}
							});

						};
						selfLink.click(function (event) {
							event.preventDefault();
							try {
								profilePopup.toggle();
								if (profilePopup.is(":visible")) {
									profilePopup.position({
										my: "right top",
										at: "right bottom",
										of: $("div#menu"),
										collision: "fit"
									});
									loadComments();
								}
							} catch (ex) {
								console.error(ex);
							}
						});

					}
				} catch (ex) {
					console.error("FixbN Failed consolidating profile links", ex);
				}

				// fix the masterbn replies box
				if (GM_config.get("repliesOveride")) {
					$("div#replies").hide().data("replies", $("div#replies").html());
					var replyNoty = null;
					window.setInterval(function () {
						var mbnReplies = null;
						try {
							var replies = $("div#replies");
							if (replies.html() === replies.data("replies")) {
								return;
							}
							replies.data("replies", replies.html());
							mbnReplies = [];
							var mbnRepliesConfigsPromises = [];
							replies.contents().filter(function () {
								return this.nodeType === 3; //Node.TEXT_NODE
							}).each(function () {
								try {
									var words = this;
									var rplyUsername = words.nodeValue.trim();

									if (rplyUsername.indexOf("has replied to your comment in thread") > 0) {
										rplyUsername = rplyUsername.substring(0, rplyUsername.indexOf("has replied to your comment in thread")).trim();

										var link = $(words.nextSibling);

										mbnReplies.push(
											{
												username: rplyUsername,
												url: link.attr("href"),
												threadTitle: link.text()
											});
										mbnRepliesConfigsPromises.push(__userConfig.getConfigPromise(rplyUsername, null));
									}
								} catch (ex) {
									console.error("FixbN Failed creating masterbn reply info", ex);
								}

							});

							$.when.apply(this, mbnRepliesConfigsPromises).then(function () {
								try {
									var mbnConfigs = {};
									for (var x = 0; x < arguments.length; x++) {
										var replierConfig = arguments[x];
										mbnConfigs[replierConfig.bnUsername] = replierConfig;
										replierConfig.onSave.callbacks.add(function () {
											$("div#replies").data("replies", "force refresh");
										});
									}

									var mbnRepliesHtml = [];
									for (var i = 0; i < mbnReplies.length; i++) {
										var replyInfo = mbnReplies[i];
										if (mbnConfigs[replyInfo.username].get("visibility") !== "Ignore") {
											mbnRepliesHtml.push("<span class='notyReply'><a href='{url}' title='{threadTitle}'>{username}</a></span>".fex(replyInfo));
										}
									}

									if (mbnRepliesHtml.length > 0) {
										if (replyNoty === null) {
											replyNoty = new noty({
												layout: 'topCenter',
												text: mbnRepliesHtml.join(" <span class='notyReplyDivider'> | </span> "),
												type: "information",
												timeout: false,
												force: true,
												callback: {
													onClose: function (theNoty) {
														replyNoty = null;
														$.ajax("/post?action=clear_notifcations", { dataType: "text", global: false });
													}
												}
											});
										} else {
											replyNoty.text = mbnRepliesHtml.join(" <span class='notyReplyDivider'> | </span> ");
										}
									}

								} catch (ex) {
									console.error("FixbN Failed creating noty for masterbn replies", ex);
								}

							});

						} catch (ex) {
							console.error("FixbN Failed masterbn reply override", ex);
						}

					}, 500);
				}
				
			};

			FixbN.prototype.fixHeadlinesPages = function () {
				if (!bnurl.isHeadlinesPage()) {
					return;
				}

				// fuck the welcome
				$("div#welcome").css("display", "none");
			   
				//hide moderated threads
				if (GM_config.get("hideModThreads")) {
						$("div.moder").css("display", "none");
				}
			
				// fix bydate stories
				if (bnurl.isHeadlinesByDate()) {
					var stories = $("table#stories tr");
					if (stories.length) {
						// the bydate table has an incorrect number of columns in the header
						stories[0].removeChild(stories[0].cells[1]);

						// move headline after the discuss link
						if (GM_config.get("switchColumns")) {
							stories.each(function () {
								$(this).append(this.cells[0]);
							});
						}
					}

					// stories should show nsfw tags
					$("table#stories tr.nsfw a.storylink").css({
						"padding-right": "85px",
						"background": "url('http://www.bannination.com/img/nsfw.jpg') no-repeat scroll right center transparent"
					});

				}

				// fix by-score headlines
				if (bnurl.isHeadlinesByScore()) { 

					// fix up the tag cloud
					$("div#tag_cloud").css("width", "90%");

					
					if (GM_config.get("fixedHeader")) {
						$("div#tag_cloud span").click((function (evt) {
							var c = "tag_" + $(evt.target).text().replace(/[^a-z]/g, "");
							window.setTimeout(function () {
								try { 
									this.$root.animate({
										scrollTop: $("." + c + ":first").offset().top - $("div#header").height() - 10
									}, 50);
								} catch (ex) {
									console.error("Fix bN Failed correcting tag scroll", ex);
								}
							}.bind(this), 600);
						}).bind(this));
					}
					

					// move the welcome title to the top of the tag cloud
					var bnTitleTagline = $("div#welcome p").first().text();
					$("div#tag_cloud center").first().text(bnTitleTagline);
					if ($("div#tag_cloud center").last().text() === "[tag cloud hidden]") {
						$("div#tag_cloud").html("<center>" + bnTitleTagline + "</center>");
					}

					// queue link can get extra info from the main page welcome
					var queueEntries = $("div#welcome a[href='/queue']").text().replace(/[^\d\.]/g, '');
					if (queueEntries && queueEntries !== "") {
						$("div#menu a[href$='/queue']").append(" (" + queueEntries + ")");
					}

					// move headline after the discuss link
					if (GM_config.get("switchColumns")) {
						$("div.storytable_row").each(function () {
							var storyRow = $(this);
							storyRow.append(storyRow.children("div.headline_cell"));
						});
					}

				}
			};

			FixbN.prototype.fixCommentsPage = function () {
				if (!bnurl.isCommentsPage()) {
					return;
				}

				var threadId = $("h1").first().attr("id").substring(1);
				if (!threadId) {
					return;
				}

				// add calendar to beer garden
				var beerGardenHeader = $("h1#s1000");
				if (beerGardenHeader.length > 0) {
					var bgDatePicker = $("<input type='hidden' id='bgDatePicker' />");
					bgDatePicker.prependTo("h1#s1000").datepicker({
						showOn: "button",
						buttonImage: "https://cdn1.iconfinder.com/data/icons/Pretty_office_icon_part_2/16/event.png",
						buttonImageOnly: true,
						changeMonth: true,
						changeYear: true,
						showAnim: 'fold',
						onSelect: function (dateText, cal) {
							var selectedDate = new Date(dateText);
							window.location.href = "/date/{year}/{month}/{day}/1000".fex({ year: selectedDate.getFullYear(), month: selectedDate.getMonth() + 1, day: selectedDate.getDate() });
						}
					});
				}

				// store the userid and username and score in each header data
				$("div.ch").each(function () {
					var header = $(this);
					header.data("uname", header.find("span.ui").text());
					var uidSpan = header.find("span.uid");
					if (uidSpan.length === 0) {
						header.data("uid", 0);
					} else {
						header.data("uid", uidSpan.text());
					}
					var score = header.find("span.score span").text();
					header.data("score", score);
				});

				// clear out tag instructions
				var tagCloud = $("td.current_tags");
				if (tagCloud.text().indexOf("add relevant short tags using the box to the left") > 0) {
					tagCloud.empty();
				}
				if (tagCloud.text().indexOf("[tag cloud hidden]") > 0) {
					$("div#tag_story").css("display", "none");
				}

				// more clickable refresh link
				$("a.commentslink")
					.last()
					.clone()
					.empty()
					.removeClass("commentslink")
					.addClass("commentsLinkButton")
					.append("<img title='check for more comments' src='' />")
					.prependTo("div#comment_form form");

				// score/sunlight
				$.fn.sunlight.defaults.handler = function (sunlight) {
					this.attr("title", sunlight.blame).text("score (" + sunlight.cool + "|" + sunlight.uncool + ")");
					this.closest("div.ch").data("score", parseInt(sunlight.cool) - parseInt(sunlight.uncool));
					this.closest("div.ch").find("span.uid").userDecoration("update");
				};
				$.fn.sunlight.defaults.threadId = threadId;

				// auto-load my own comment sunlight
				$("div.uc span.score").css("cursor", "pointer").sunlight();
				// click to load/refresh all comment sunlight
				$("div.ch span.score").css("cursor", "pointer").sunlight({ event: 'click' });

				// nabbit and pretty dates
				$.fn.nabbit.defaults.userId = function () { return this.closest("div.ch").data("uid"); };
				$.fn.nabbit.defaults.username = function () { return this.closest("div.ch").data("uname"); };

				$("div.ch span.uid").nabbit({
					imgSize: 'small',
					imgClass: 'nabbitSmall',
					imgTitle: $.fn.nabbit.defaults.userId
				});

				// pretty dates
				var timeSpans = $("div.ch span.time");
				timeSpans.each(function () {
					var bigSpot = $(this);

					var postTime = bigSpot.text();
					var cleanTime = (postTime || "").replace(/-/g, "/").replace(/\.0/g, " PST"); // HACK!
					bigSpot.attr("datetime", cleanTime);
					bigSpot.attr("title", postTime);
					bigSpot.age({ suffixes: { past: "ago", future: "from now" } });

				});

				// nabbit big images
				timeSpans.nabbit({ 
					imgSize: 'large',
					imgClass: 'nabbitBig',
					imgTitle: function () { return this.text(); },
					loaded: function (img) {
						this.next("span.time").remove();
						this.after($("<span class='time'></span>").append(img));
					}
				});

				// anon
				$("div.ch.u0").find("span.ui").html(function (index, text) {
					return "~ " + text.substring(30);
				}).after("<span class='uid'><img title='settings' src='https://cdn1.iconfinder.com/data/icons/hamburg/32/settings.png' style='height:1em; width:1em;' /></span>");

				// add out-of-page quotes to the style
				/* jshint -W064 */
				GM_addStyle("div.cb a[href^='/comments/{0}'] { color:black; }".fex(threadId));
				/* jshint +W064 */

				// add user decoration to userid span
				$("div.ch span.uid").userDecoration({
					'threadId': threadId,
					onImageReplaced: function () {
						$(this).magnificPopup({ type: 'image', verticalFit: true, closeOnContentClick: true, showCloseBtn: false });
					}
				});

				// html comment editor
				$('textarea').markItUp({
					/* jshint ignore:start */
					resizeHandle: false,
					markupSet: [
						{ name: 'Bold', key: 'B', openWith: '<b>', closeWith: '</b>', className: 'miuBold' },
						{ name: 'Italic', key: 'I', openWith: '<i>', closeWith: '</i>', className: 'miuItalic' },
						{ name: 'Stroke through', key: 'S', openWith: '<strike>', closeWith: '</strike>', className: 'miuStrike' },
						{
							name: 'Colors', className: 'palette', dropMenu: [
								{ name: 'Yellow', openWith: '<font color="#FCE94F">', closeWith: '</font>', className: "col1-1" },
								{ name: 'Yellow', openWith: '<font color="#EDD400">', closeWith: '</font>', className: "col1-2" },
								{ name: 'Yellow', openWith: '<font color="#C4A000">', closeWith: '</font>', className: "col1-3" },

								{ name: 'Orange', openWith: '<font color="#FCAF3E">', closeWith: '</font>', className: "col2-1" },
								{ name: 'Orange', openWith: '<font color="#F57900">', closeWith: '</font>', className: "col2-2" },
								{ name: 'Orange', openWith: '<font color="#CE5C00">', closeWith: '</font>', className: "col2-3" },

								{ name: 'Brown', openWith: '<font color="#E9B96E">', closeWith: '</font>', className: "col3-1" },
								{ name: 'Brown', openWith: '<font color="#C17D11">', closeWith: '</font>', className: "col3-2" },
								{ name: 'Brown', openWith: '<font color="#8F5902">', closeWith: '</font>', className: "col3-3" },

								{ name: 'Green', openWith: '<font color="#8AE234">', closeWith: '</font>', className: "col4-1" },
								{ name: 'Green', openWith: '<font color="#73D216">', closeWith: '</font>', className: "col4-2" },
								{ name: 'Green', openWith: '<font color="#4E9A06">', closeWith: '</font>', className: "col4-3" },

								{ name: 'Blue', openWith: '<font color="#729FCF">', closeWith: '</font>', className: "col5-1" },
								{ name: 'Blue', openWith: '<font color="#3465A4">', closeWith: '</font>', className: "col5-2" },
								{ name: 'Blue', openWith: '<font color="#204A87">', closeWith: '</font>', className: "col5-3" },

								{ name: 'Purple', openWith: '<font color="#AD7FA8">', closeWith: '</font>', className: "col6-1" },
								{ name: 'Purple', openWith: '<font color="#75507B">', closeWith: '</font>', className: "col6-2" },
								{ name: 'Purple', openWith: '<font color="#5C3566">', closeWith: '</font>', className: "col6-3" },

								{ name: 'Red', openWith: '<font color="#EF2929">', closeWith: '</font>', className: "col7-1" },
								{ name: 'Red', openWith: '<font color="#CC0000">', closeWith: '</font>', className: "col7-2" },
								{ name: 'Red', openWith: '<font color="#A40000">', closeWith: '</font>', className: "col7-3" },

								{ name: 'Gray', openWith: '<font color="#FFFFFF">', closeWith: '</font>', className: "col8-1" },
								{ name: 'Gray', openWith: '<font color="#D3D7CF">', closeWith: '</font>', className: "col8-2" },
								{ name: 'Gray', openWith: '<font color="#BABDB">', closeWith: '</font>6', className: "col8-3" },

								{ name: 'Gray', openWith: '<font color="#888A85">', closeWith: '</font>', className: "col9-1" },
								{ name: 'Gray', openWith: '<font color="#555753">', closeWith: '</font>', className: "col9-2" },
								{ name: 'Gray', openWith: '<font color="#000000">', closeWith: '</font>', className: "col9-3" }
							]
						},
						{ name: 'Pre', className: 'miuPre', openWith: '<pre>', closeWith: '</pre>' },
						{ separator: '---------------' },
						{ name: 'Ul', openWith: '<ul>\n', closeWith: '</ul>\n', className: 'miuUList' },
						{ name: 'Ol', openWith: '<ol>\n', closeWith: '</ol>\n', className: 'miuOList' },
						{ name: 'Li', openWith: '<li>', closeWith: '</li>', className: 'miuListitem' },
						{ separator: '---------------' },
						{ name: 'Picture', key: 'P', replaceWith: '<img src="[![Source:!:http://]!]" />', className: 'miuImage' },
						{ name: 'Link', key: 'L', openWith: '<a href="[![Link:!:http://]!]"(!( title="[![Title]!]")!)>', closeWith: '</a>', placeHolder: 'Your text to link...', className: 'miuLink' },
						{ separator: '---------------' },
						{ name: 'Clean', replaceWith: function (markitup) { return markitup.selection.replace(/<(.*?)>/g, ""); }, className: 'miuClean' },
						{ name: 'Symbols', className: 'miuSymbols', dropMenu: [
								{ name: 'Tags', openWith: '&lt;', closeWith: '&gt;' },
								{ name: '&', openWith: '&amp;' }
							]
						}
					]
					/* jshint ignore:end */
				});

				// intercept comment pastes
				$('textarea').each(function(){
					$(this).data("oldVal", $(this).val()).data("editing", false);
				}).bind('input propertychange', function (evt) {
					if (!GM_config.get("autoLinkComments")) {
						return;
					}
					try {
						var commentBox = $(this);

						var oldVal = commentBox.data("oldVal");
						var newVal = commentBox.val();
						commentBox.data("oldVal", newVal);

						if (commentBox.data("editing")) {
							console.log("leaving: editing");
							return;
						}

						if (newVal.length > oldVal.length) {
							commentBox.data("editing", true);
							try {
								var change = diff(oldVal, newVal);
								if (oldVal.indexOf(change) > 0 || change.length <= 1) {
									return;
								}
								var url = new URI(change);
								if (url.is("url") && url.protocol().indexOf("http") === 0) {
									var tag = "";

									switch (url.suffix()) {
										case "gif":
										case "jpg":
										case "jpeg":
										case "png":
											tag = "<img src='{0}' />".fex(change);
											break;
										default:
											tag = "<a href='{0}'>{0}</a>".fex(change);
											break;
									}

									var parts = newVal.split(change);
									var result = parts[0] + tag + parts[1];
									commentBox.val(result);
								}
							} catch (ex) {
								console.error(ex);
							} finally {
								commentBox.data("editing", false);
							}
						}
					} catch (ex) {
						console.log("FixbN Failed intercepting comment changes", ex);
					}
				});

				// any selective reply child quotes have font tags around them, let's style them
				/* jshint -W014 */
				$("div.cb font").filter(function () {
					var quoteChildren = $(this).contents();
					return (
						(quoteChildren.length === 3 && quoteChildren[0].tagName === "A" && quoteChildren[2].tagName === "I")
						||
						(quoteChildren.length === 2 && quoteChildren[0].tagName === "A" && quoteChildren[1].tagName === "I")
					);
					
				}).addClass("commentquote");
				/* jshint +W014 */

				// any selective reply images have been converted to links, let's style them
				$("div.cb a").filter(function () {
					var text = $(this).text();
					return text === "[quoted image removed]" || text === "image removed";
				}).magnificPopup({ type: 'image', verticalFit: true, closeOnContentClick: true, showCloseBtn: false });

				// comment warning
				var commentForm = $("div#comment_form form");
				var warning = $("div#comment_warning");
				warning.css('display', 'none');
				if (warning.text().length > 0) {
					if (warning.text().indexOf("Not Safe") > 0) {
						commentForm.append("<h2 style='display:inline;color:red;'>Not Safe For Work Allowed</h2>");
					} else {
						commentForm.append("<h2 style='display:inline;color:green;'>Keep Thread Safe For Work</h2>");
					}
								var modindex = warning.text().indexOf("thread."); //Add moderated thread text back in *wushupork 05/03/2014
					if (modindex > 0) {
						var modtext = warning.text().substr((modindex+7));
						commentForm.append("<br /><br />This is a moderated thread. "+modtext); 
					}
					commentForm.css('width:90%;');
				}

				// comment preview
				$("div#comment_form form :submit[value='preview']").click(function (event) {
					event.preventDefault();

					if ($("textarea#comment").val().trim() === "") {
						return;
					}

					var values = $("div#comment_form form").serializeArray();
					values.push({ name: 'button', value: 'preview' });

					$.ajax({
						url: $("div#comment_form form").attr("action"),
						type: "POST",
						dataType: "text",
						data: values,
						success: function (data) {
							var commentHtml = data.substring(data.indexOf('<div class="cb" id="c0">'));
							commentHtml = commentHtml.substring(0, commentHtml.indexOf('<div id="comment_form">'));
							var previewComment = $("<div class='ch uc'><span class='ui'>Comment Preview</span></div>" + $.trim(commentHtml));

							previewComment.find("font").filter(function () {
								var quoteChildren = $(this).contents();
								/* jshint -W014 */
								return (
									(quoteChildren.length === 3 && quoteChildren[0].tagName === "A" && quoteChildren[2].tagName === "I")
									||
									(quoteChildren.length === 2 && quoteChildren[0].tagName === "A" && quoteChildren[1].tagName === "I")
									);
								/* jshint +W014 */
							}).addClass("commentquote");

							$.magnificPopup.open({
								showCloseBtn: false,
								items: {
									src: previewComment,
									type: 'inline'
								}
							});
						}
					});

				});

				// "replies to me" that I can see
				if (GM_config.get("showRepliesToMe")) {
					var myUserName = $("div#comment_form form input#username").val();
					if (myUserName) {
						var repliesToMeList = $("<ul></ul>");
						var repliesToMe = $("<div class='fbnRepliesToMe'><span class='label'>Replies To Me</span><span class='replyToMeListHolder' /></div>");
						repliesToMe.find("span.replyToMeListHolder").append(repliesToMeList);
						var visibleCommentLinks = $("div.cb:visible:not(.fbnIgnored) a[href^='#'], div.cb:visible:not(.fbnIgnored) a[href^='/comments']");
						visibleCommentLinks.filter(function () {
							return $(this).text() === myUserName;
						}).each(function () {
							var replyToMeHeader = $(this).closest("div.cb").prev("div.ch");
							if (replyToMeHeader.data("uname") !== myUserName) {
								repliesToMeList.append("<li><a href='#{0}'>{1}</a></li>".fex(replyToMeHeader.attr("id").substring(1), replyToMeHeader.data("uname")));
							}
						});
						if (repliesToMeList.find("li").length > 0) {
							repliesToMe.insertAfter($("div#main h1").first());
							repliesToMeList.parent().hide();
							repliesToMe.find("span.label").click(function () {
								repliesToMeList.parent().toggle(250);
							});
						}
					}
				}

				// add comment body smarttags/menu
				var contextMenu = $("<div id='selectionMenu' style='display:none;position:absolute;'><div id='selectionMenuQuote'>Quote this</div><hr /><div id='selectionMenuTag'>Tag this</div></div>");
				$("body").append(contextMenu);
				contextMenu.click(function (event) {
					var menu = $(this);
					try {
						var item = $(event.target);
						var text = menu.data("selectionText");
						var commentInfo = __fixbn.findCommentInfo(menu.closest("div.cb").find("div.reply a"));
						var cleanHtml = __fixbn.cleanSelection(text, commentInfo);

						switch (event.target.id) {
							case "selectionMenuQuote":
								var commentBox = $("#comment");
								if (commentBox.length) {
									$("html, body").animate({ scrollTop: $(document).height() - $(window).height() });
									var originalComment = commentBox.val();
									if (originalComment !== "") { originalComment = originalComment + "\n"; }
									try {
										commentBox.data("editing", true);
										commentBox.val(originalComment + cleanHtml + "\n\n").focus().scrollTop(commentBox[0].scrollHeight).caret(-1);
										commentBox.data("oldVal", commentBox.val());
									} catch (ex) {
										console.error(ex);
									} finally {
										commentBox.data("editing", false);
										console.log("set editing to " + commentBox.data("editing"));
									}
								} else {
									alert("You must log in to comment");
								}
								break;
							case "selectionMenuTag":
								if (GM_config.get("stickyTagger")) {
									var stickyTaggerInput = $("div.stickyTagger input");
									stickyTaggerInput.show(250);
									stickyTaggerInput.val(text);
								} else {
									alert("Enable the Floating Tagger in options to use selection tagging.");
								}
								break;
						}
					} catch (ex) {
						console.error(ex);
					}
					menu.fadeOut();
				});
				$("div.cb").smartTag({
					tagWidth: "32px",
					tagHeight: "32px",
					onClick: function (event) {
						try {
							var tag = $(this);
							var container = $(this).closest("div.cb");
							var menu = $("#selectionMenu");
							menu.data("selectionText", event.data);
							container.append(menu);
							menu.css({ top: parseInt(tag.css("top")) + parseInt(tag.css("height")), left: tag.css("left") }).fadeIn();
						} catch (ex) {
							console.error(ex);
						}
					},
					//onShowing		
					onHiding: function () {
						var menu = $("#selectionMenu");
						menu.fadeOut();
						$("body").append(menu);
					}
				});

				// make sure the standard reply scrolls the comment textarea
				if ($("#comment").length) {

					$("div.reply a").click((function () {
						//window.setTimeout(function () {
						try {
							$("html, body").animate({ scrollTop: $(document).height() - $(window).height() });
							$("#comment").focus().scrollTop($("#comment")[0].scrollHeight).caret(-1);
						} catch (ex) {
							console.error("FixbN failed scroll comment box", ex);
						}
						//}, 500);
					}));
				} else {
					$("div.reply").hide();
				}
			};

			FixbN.prototype.fixQueuePage = function () {
				if (!bnurl.isQueuePage()) {
					return;
				}

				$("input.submitform[name='headline']").attr("maxlength", "254");
			};

			FixbN.prototype.fixTaggers = function () {
				var notification = null;

				$.fn.tagn.defaults.beforeSubmit = function (tagSet) {
					var tagInput = $(this);
					tagInput.val("");
				};
				$.fn.tagn.defaults.taggingComplete = function (tagSet) {
					try {
						var accepted = 0;
						var result = $("<ul></ul>");
						for (var i = 0; i < tagSet.tags.length; i++) {
							var tag = tagSet.tags[i];
							var status = tagSet.tags[i].status;
							switch (status) {
								case "accepted":
									accepted = Math.max(accepted, parseInt(tag.message));
									result.append("<li>{0} accepted</li>".fex(tag.value));
									break;
								case "matched":
									result.append("<li>{0} matched with {1}</li>".fex(tag.value, tag.message));
									break;
								default:
									result.append("<li>{0} rejected: {1}</li>".fex(tag.value, tag.message));
									break;
							}
						}
						result.prepend("<li>{0} total tags accepted for this thread".fex(accepted.toString()));

						new noty({ 'timeout': GM_config.get("tagNotyDuration") * 1000, 'type': 'information', 'killer': true, 'text': result.html() });
					} catch (ex) {
						console.error("Fix bN Failed to notify of tagging status", ex);
					}
				};

				// heh, i couldn't disable the other submit handlers, so I just hide and copy that fucker
				if (bnurl.isCommentsPage()) {
					var storyId = $("form input[name='storyid']").val();

					var oldForm = $("form input[id^='tag']").closest("form");
					if (oldForm.length > 0) {
						var newForm = oldForm.clone(false);
						newForm.find("label[id^='tagsaccepted']").attr("id", "tagnResult").addClass("tagnResult");
						newForm.addClass("tagnForm").insertAfter(oldForm);
						oldForm.css("display", "none");
						newForm.find("input[id^='tag']").tagn({ "storyId": storyId });

						if (GM_config.get("stickyTagger")) {
							var stickyTagger = $("<div class='stickyTagger'><form><label for='stickyTaggerInput'>Tag</label><input id='stickyTaggerInput' /></form></div>");
							$("div#main").append(stickyTagger);
							stickyTagger.find("label").click(function () {
								var tagger = $("#" + $(this).attr("for"));
								tagger.toggle(250);
							});
							stickyTagger.find("input").tagn({ "storyId": storyId });
						}
					}
				}

				if (bnurl.isQueuePage()) {
					$(".storyentry form input[id^='tag']").each(function () {
						var queueStoryId = this.id.substring(3);
						var that = $(this);
						var oldQueueForm = that.closest("form");
						var newQueueForm = oldQueueForm.clone(false).addClass("tagnForm").insertAfter(oldQueueForm);
						newQueueForm.find("label[id^='tagsaccepted']").addClass("tagnResult");
						newQueueForm.find("input[id^='tag']").tagn({ "storyId": queueStoryId });
						oldQueueForm.css("display", "none");
					});
				}
			};

			FixbN.prototype.createDateUrl = function (relativeDays) {
				var linkDate = new Date();
				linkDate.setDate(linkDate.getDate() + relativeDays);
				return "/date/" + linkDate.getFullYear() + "/" + (linkDate.getMonth() + 1) + "/" + linkDate.getDate();
			};

			FixbN.prototype.findCommentInfo = function (replyLink) {
				return eval($(replyLink).attr("href").replace("javascript:reply", "(function(c,u){return { 'commentId': c, 'username': u };})") + ";"); // jshint ignore:line
			};

			FixbN.prototype.cleanSelection = function (selectionHtml, commentInfo) {
				var quote = $("<quote />").append(selectionHtml);
				quote.find("img").replaceWith(function () {
					return "<a href='" + $(this).attr("src") + "'>[quoted image removed]</a>";
				});
				quote.find("br").remove();

				var quotesToWrap = [];
				var quoteContents = quote.contents();
				/* jshint -W014 */
				quoteContents.each(function (index) {
					if (this.nodeType !== 3
						&& this.tagName === "A"
						&& quoteContents.length > index + 2
						&& quoteContents[index + 1].nodeType === 3
						&& quoteContents[index + 2].tagName === "I"
						) {

						quotesToWrap.push(index);
					}
				});
				/* jshint +W014 */
				var i;
				for (i = 0; i < quotesToWrap.length; i++) {
					var indexOfA = quotesToWrap[i];
					$(quoteContents[indexOfA]).add(quoteContents[indexOfA + 1]).add(quoteContents[indexOfA + 2]).wrapAll("<font class='commentquote' ></font>");
				}

				var result = quote.html();
				result = result.replace("</i></font>\n\n", "</i></font>\n").replace(/^[\n\r\t ]+/, "").replace(/[\n\r\t ]+$/, "");
				// wrap in a quote
				result = '<quote user="' + commentInfo.username + '" cid="' + commentInfo.commentId + '">' + result + "</quote>";

				return result;
			};

			return FixbN;

		})();

		__fixbn = new FixbN();
	})(jQuery);
} catch (ex) {
	console.error("FixbN Failed declaring __fixbn", ex);
}

// user config class
var __userConfig = null;
try {
	__userConfig = (function ($) {
		"use strict";

		var _configPromiseCache = {};
		var _idStore = null;

		function _getConfigPromise(username, userid) {
			username = _cleanUsername(username);

			var result = _configPromiseCache[username];
			if (!result) {
				var dfd = $.Deferred();

				if (typeof userid !== "undefined" && userid !== null && !isNaN(userid)) {
					_saveUserId(username, userid);
					dfd.resolve(_createConfig(username, userid));
				} else {
					var userIdPromise = _getUserIdPromise(username);
					userIdPromise.always(function (result, evt) {

						if (typeof result !== "undefined" && result !== null && typeof result.userId !== "undefined") {
							dfd.resolve(_createConfig(username, result.userId));
						} else {

							// get userid from ajax
							console.log("FixbN Looking up userId on bannination.com/users/{0}".fex(username));
							var userPageUrl = "http://www.bannination.com/users/{0}".fex(username);
							$.ajax({
								url: userPageUrl,
								dataType: "text",
								success: function (data, textStatus, jqXHR) {
									// <h1>The user bleh was not found</h1>
									// <h1> Profile for artificeren (757)</h1>
									if (data.indexOf("Profile for") > 0) {
										//lolregex
										var match = data.match("(?:<h1> Profile for [^\(]* )(.*)(?:</h1>)"); // jshint ignore:line
										var result = match[1].replace(/\(/g, '').replace(/\)/g, '');
										_saveUserId(username, result);
										dfd.resolve(_createConfig(username, result));
									}
									else {
										console.error("FixbN Error retrieving userid for user {0}".fex(username));
										dfd.reject(null);
									}
								},
								error: function () {
									console.error("FixbN Error retrieving userid for user {0}".fex(username));
									dfd.reject(null);
								}
							});
						}
					});

				}

				_configPromiseCache[username] = dfd.promise();
			}
			return _configPromiseCache[username];
		}

		function _cleanUsername(username) {
			return (username.indexOf("~") !== 0 && username.indexOf("someone who may or may not be") !== 0) ? username : "~Anonymous";
		}

		function _createConfig(username, userId) {
			var configId = 'FixbNUser' + userId;
			var configKey = username;

			var userConfigFrame = $("<div style='display:none;' />")[0];
			document.body.appendChild(userConfigFrame);
			var config = new GM_configStruct();
			config.init({
				'id': configId,
				'title': 'Fix bN Settings for User: {0}'.fex(configKey),
				'fields':
				{
					'visibility':
					{
						'label': 'User Comment Visibility',
						'type': 'radio',
						'options': ['Normal', 'Ignore'],
						'default': 'Normal'
					},
					'ignoreReplies':
					{
						'label': 'User Being Quoted Visibility',
						'type': 'radio',
						'options': ['Shown', 'Default', 'Hidden'],
						'default': 'Default'
					},
					'blockImages':
					{
						'label': 'User Comment Image Visibility',
						'type': 'radio',
						'options': ['Normal', 'Removed'],
						'default': 'Normal'
					},
					'nabbitVisibility':
					{
						'label': 'User Nabbit Image Visibility',
						'type': 'radio',
						'options': ['Shown', 'Default', 'Hidden'],
						'default': 'Default'
					},
					'overrideIgnoreReplies':
					{
						'label': "Show This User's Replies to Ignored Users",
						'type': 'checkbox',
						'default': false
					},
					'headColor':
					{
						'label': 'Comment Header Font Color',
						'type': 'text',
						'size': 25,
						'default': ''
					},
					'headBackColor':
					{
						'label': 'Comment Header Background Color',
						'type': 'text',
						'size': 25,
						'default': ''
					}
				},
				'frame': userConfigFrame
			});
			var openCallbacks = $.Callbacks();
			openCallbacks.fire.callbacks = openCallbacks;
			config.onOpen = openCallbacks.fire;

			var saveCallbacks = $.Callbacks();
			saveCallbacks.fire.callbacks = saveCallbacks;
			config.onSave = saveCallbacks.fire;

			var closeCallbacks = $.Callbacks();
			closeCallbacks.fire.callbacks = closeCallbacks;
			config.onClose = closeCallbacks.fire;

			var resetCallbacks = $.Callbacks();
			resetCallbacks.fire.callbacks = resetCallbacks;
			config.onReset = resetCallbacks.fire;

			config.onSave.callbacks.add((function () {
				if (this.isOpen) {
					this.close();
				}
			}).bind(config));

			config.onOpen.callbacks.add(function (doc, win, frame) {
				$(frame).find("input[id$='_field_headColor'], input[id$='_field_headBackColor']").spectrum({
					clickoutFiresChange: true,
					preferredFormat: "hex6",
					showInput: true,
					showButtons: true,
					allowEmpty: true,
					showPalette: true,
					showSelectionPalette: true,
					palette: [
						"#EEEEEE",
						"#807373",
						"#955050",
						"#373737"
					],
					localStorageKey: "spectrum.fixbn.bannination"
				});
				$(frame).find(".saveclose_buttons[id$='_saveBtn']").text(" OK ");
				$(frame).find(".saveclose_buttons[id$='_closeBtn']").text("Cancel");
			});

			config.bnUsername = username;
			return config;
		}

		function _getUserIdPromise(username) {
			return __fixbndb.idStore().get(username);
		}

		function _saveUserId(username, userId) {
			try {
				$.when(__fixbndb.idStore().get(username)).then(
					function (item, evt) {
						if ( typeof item !== "undefined" && item !== null ) {
							if (item.userId !== userId) {
								__fixbndb.idStore().put({ "username": username, "userId": userId }, username);
							
							}
						} else {
							__fixbndb.idStore().add({ "username": username, "userId": userId }, username);
							
						}
					},
					function (error, evt) {
						__fixbndb.idStore().add({ "username": username, "userId": userId }, username);
					}
				);
				
			} catch (ex) {
				console.error("FixbN Failed saving userId to local storage", ex);
			}

		}

		return {
			getConfigPromise: _getConfigPromise,
			getCleanUsername: _cleanUsername
		};

	})(jQuery);
} catch (ex) {
	console.error("FixbN Failed declaring __userConfig", ex);
}

// local database
var __fixbndb = null;
try {
	__fixbndb = (function ($) {

		function _idStore() {
			return $.indexedDB("FixbN").objectStore("Users", {
				"keypath": "username",
				"autoincrement": false
			});
		}

		return {
			idStore: _idStore
		};

	})(jQuery);
} catch (ex) {
	console.error("FixbN Failed declaring __fixbndb", ex);
}

// bannination.com url tests
var bnurl = (function () {

	var url = new URI();

	function _isHeadlinesPage() {
		result = (url.path() === "/" || url.path() === "/pages/main/index.vm" || (url.segment().length > 0 && url.segment(0).toLowerCase() === "date" && url.segment().length < 5));
		return result;
	}

	function _isHeadlinesByScore() {
		result = _isHeadlinesPage && ( url.path() === "/" || url.path() === "/pages/main/index.vm" );
		return result;
	}

	function _isHeadlinesByDate() {
		result = _isHeadlinesPage() && url.segment().length > 0 && url.segment(0).toLowerCase() === "date";
		return result;
	}

	function _isCommentsPage() {
		result = url.segment().length > 0 && (url.segment(0).toLowerCase() === "comments" || (url.segment(0).toLowerCase() === "date" && url.segment().length >= 5));
		return result;
	}

	function _isCommentsPageTagable() {
		result = _isCommentsPage() && (url.segment(url.segment.length - 1) !== "1000");
		return result;
	}

	function _isQueuePage() {
		result = url.segment.length > 0 && url.segment(0).toLowerCase() === "queue";
		return result;
	}

	return {
		isHeadlinesPage: _isHeadlinesPage,
		isHeadlinesByScore: _isHeadlinesByScore,
		isHeadlinesByDate: _isHeadlinesByDate,
		isCommentsPage: _isCommentsPage,
		isCommentsPageTagable: _isCommentsPageTagable,
		isQueuePage: _isQueuePage
	};

})();

// nabbit jquery plugin
(function ($) {
	"use strict";

	var Nabbit;
	var $bind = function (fn, me) { return function () { return fn.apply(me, arguments); }; };

	Nabbit = (function () {

		// ctor
		function Nabbit($el, settings) {
			if (settings === null) {
				settings = {};
			}

			this.uid = "";
			this.uname = "";
			this.config = null;

			this.attach = $bind(this.attach, this);
			this.update = $bind(this.update, this);

			this.$el = $el;
			this.settings = $.extend({}, $.fn.nabbit.defaults, settings);

			if (this.settings.userId) {
				if (typeof (this.settings.userId) === "function") {
					this.uid = $bind(this.settings.userId, this.$el)();
				} else {
					this.uid = this.settings.userId.toString();
				}
			} else {
				this.uid = this.$el.text();
			}
			if (this.settings.username) {
				if (typeof (this.settings.username) === "function") {
					this.uname = $bind(this.settings.username, this.$el)();
				} else {
					this.uname = this.settings.username.toString();
				}
			}

			$.when(__userConfig.getConfigPromise(this.uname, this.uid)).then(this.attach);
		}

		Nabbit.prototype.attach = function (userConfig) {

			this.config = userConfig;
			this.config.onSave.callbacks.add(this.update);

			this.update();
		};

		Nabbit.prototype.update = function () {
			var img = null;
			var imgsrc = "";
			var title = "";
			if (this.settings.imgTitle) {
				if ($.type(this.settings.imgTitle) === "function") {
					title = $bind(this.settings.imgTitle, this.$el)();
				} else if (this.settings.imgTitle !== null) {
					title = this.settings.imgTitle.toString();
				}
			}

			var imgHtml = "<img alt='{title}' title='{title}' {size} />";
			if (title !== "") {
				imgHtml = imgHtml.fex({ 'title': title });
			} else {
				imgHtml = imgHtml.fex({ 'title': "" });
			}

			switch (this.settings.imgSize) {
				case "small":
					img = $(imgHtml.fex({ 'size': "width='58' height='18'" }));
					imgsrc = "http://webmonkees.com/naBBits/" + this.uid + ".gif";
					break;
				case "large":
					img = $(imgHtml.fex({ 'size': "width='180' height='18' align='top' border='0'" }));
					imgsrc = "http://webmonkees.com/naBBits/m" + this.uid + ".png";
					break;
				default:
			}

			var globalNabbitVisibility = GM_config.get("nabbitVisibility");
			var userNabbitVisibility = this.config.get("nabbitVisibility");
			
			var visClass = "";
			switch (userNabbitVisibility) {
				case "Shown":
					visClass = "nabbitForceOn";
					break;
				case "Hidden":
					visClass = "nabbitOff";
					break;
				case "Default":
					visClass = globalNabbitVisibility ? "nabbitOn" : "nabbitOff";
					break;
			}
			img.addClass(visClass);

			if ($.type(this.settings.imgClass) === "string" && this.settings.imgClass !== "") {
				img.addClass(this.settings.imgClass);
			}

			img.load($bind(function (evt) {
				if ($.type(this.settings.loaded) === 'function') {
					$bind(this.settings.loaded, this.$el)(evt.target);
				} else {
					this.$el.empty().append(evt.target);
				}
			}, this)).attr("src", imgsrc);

		};

		return Nabbit;

	})();

	$.fn.extend({
		nabbit: function (options) {
			if (options === null) {
				options = {};
			}
			return this.each(function () {
				return new Nabbit($(this), options);
			});
		}
	});

	// settings default
	$.fn.nabbit.defaults = {
		userId: null, // string or function that returns the nabbit userId
		username: null, // string or function that returns the username
		imgTitle: null, // string of function that returns the title for the created img
		loaded: null, // function that handles the img loaded event, recieves the loaded image
		imgClass: '', // string which is added to the img css class
		imgSize: 'small' // choices: small, large
	};

})(jQuery);

// bN Sunlight jquery plugin
(function ($) {
	"use strict";

	var Sunlight;
	var $bind = function (fn, me) { return function () { return fn.apply(me, arguments); }; };

	Sunlight = (function () {

		// ctor
		function Sunlight($el, settings) {
			if (settings === null) {
				settings = {};
			}

			this.attach = $bind(this.attach, this);
			this.load = $bind(this.load, this);

			this.$el = $el;
			this.settings = $.extend({}, $.fn.sunlight.defaults, settings);

			this.attach();
		}

		Sunlight.prototype.attach = function () {
			if (this.settings && this.settings.event && this.settings.event !== "") {
				this.$el.on(this.settings.event, this.load);
			} else {
				this.load();
			}
		};

		Sunlight.prototype.load = function () {
			var commentId = this.$el.closest("div.ch").attr("id").substring(1);

			//http://www.bannination.com/comments/5181856/sunlight/5087531
			var sunlightUrl = "http://www.bannination.com/comments/" + this.settings.threadId + "/sunlight/" + commentId;

			this.$el.data("originalcursor", this.$el.css("cursor"));
			this.$el.css("cursor", "progress");
			$.ajax({
				url: sunlightUrl,
				dataType: "text",
				success: $bind(function (data) {

					var sunlight = {
						cool: 0,
						uncool: 0,
						blame: ""
					};

					var cools = data.match(new RegExp(" Cool from (.*)\n", "g"));
					var uncools = data.match(new RegExp(" Uncool from (.*)\n", "g"));

					var i;
					if (cools && cools.length) {
						sunlight.cool = cools.length;
						for (i = 0; i < cools.length; i++) {
							sunlight.blame += "* " + cools[i];
						}
					}
					if (uncools && uncools.length) {
						sunlight.uncool = uncools.length;
						for (i = 0; i < uncools.length; i++) {
							sunlight.blame += "* " + uncools[i];
						}
					}

					if (sunlight.blame.length === 0) { sunlight.blame = "No Votes"; }

					if (this.settings && this.settings.handler) {
						($bind(this.settings.handler, this.$el))(sunlight);
					}
				}, this),
				complete: $bind(function () {
					this.$el.css("cursor", this.$el.data("originalcursor"));
				}, this)
			});
		};

		return Sunlight;

	})();

	$.fn.extend({
		sunlight: function (options) {
			if (options === null) {
				options = {};
			}
			return this.each(function () {
				return new Sunlight($(this), options);
			});
		}
	});

	// settings default
	$.fn.sunlight.defaults = {
		threadId: '0',
		handler: null,
		event: ''
	};

})(jQuery);

// bN User Decoration jquery plugin
try {
	(function ($) {
		"use strict";

		var UserDecoration;
		var $bind = function (fn, me) { return function () { return fn.apply(me, arguments); }; };

		UserDecoration = (function () {
			// ctor
			function UserDecoration($el, settings) {
				if (settings === null) {
					settings = {};
				}

				this.userConfig = null;
				this.childConfigs = {};
				this.userId = "0";
				this.userName = "";

				this.attach = $bind(this.attach, this);
				this.showUI = $bind(this.showUI, this);
				this.updateAll = $bind(this.updateAll, this);
				this.update = $bind(this.update, this);

				this.onImageReplaced = $.Callbacks();

				this.$el = $el;
				this.$el.data("userDecoration", this);
				this.settings = $.extend({}, $.fn.userDecoration.defaults, settings);

				if (typeof this.settings.onImageReplaced === "function") {
					this.onImageReplaced.add(this.settings.onImageReplaced);
				}

				this.attach();
			}

			// attach to element
			UserDecoration.prototype.attach = function () {
				var header = this.$el.closest("div.ch");
				this.userId = header.data("uid");
				this.userName = header.data("uname");
				var body = $("div.cb.u" + this.userId + "[id$='" + header.attr("id").substring(1) + "']");
			
				var peep = $("<span class='peep' >ಠ_ಠ</span>");
				peep.click( $bind(function () {
					this.$el.closest("div.ch").removeClass("fbnIgnored");
					var body = $("div.cb.u" + this.userId + "[id$='" + header.attr("id").substring(1) + "']");
					body.slideDown('fast').removeClass("fbnIgnored");
				}, this));
				header.prepend(peep);

				var me = this;

				// Get the configs for every quoted user and create a promise array from them
				var quotedConfigPromises = [];
				var threadId = $("form input[name='storyid']").val();
				var quoteLinks = body.find("a[href^='#'], a[href^='/comments/{0}']".fex(threadId));
				quoteLinks.each(function () {
					var link = $(this);
					var quotedUsername = link.text().replace("someone who may or may not be", "~");
					var quotePromise = __userConfig.getConfigPromise(quotedUsername);
					quotePromise.done(function (quotedConfig) {
						me.childConfigs[quotedConfig.bnUsername] = quotedConfig;
						quotedConfigPromises.push(quotePromise);
					});
				});

				// when I have the main config and then all child configs, 
				// attach to all of their save events for updating, and then update to current settings
				$.when(__userConfig.getConfigPromise(me.userName, me.userId))
					.then(function (config) {
						$.when.apply(this, quotedConfigPromises).then(function () {
							me.userConfig = config;
							me.userConfig.onSave.callbacks.add(me.update);
							$.each(me.childConfigs, function (index, value) {
								value.onSave.callbacks.add(me.update);
							});
							me.$el.css("cursor", "pointer").click(me.showUI);
							me.update();
						});
					});

			};

			UserDecoration.prototype.update = function () {
				if (this.userConfig === null) {
					return;
				}

				var header = this.$el.closest("div.ch.u" + this.userId);
				var body = $("div.cb.u" + this.userId + "[id$='" + header.attr("id").substring(1) + "']");
				var me = this;

				// score hilighting
				var score = header.data("score");
				if (score && score > parseInt(GM_config.get("highScoreThreshold"))) {
					header.addClass("highScore");
				} else {
					header.removeClass("highScore");
				}

				// image blocking
				var blockImages = this.userConfig.get("blockImages");
				switch (blockImages) {
					case "Normal":
						body.find("a.fbnBlockedImage").replaceWith(function () {
							var blockedImage = $(this);
							var unblockedImage = $("<img src='" + blockedImage.attr("href") + "' />");
							var height = blockedImage.data("height");
							var width = blockedImage.data("width");
							if (height) {
								unblockedImage.attr("height", height).css("height", height);
							}
							if (width) {
								unblockedImage.attr("width", width).css("width", width);
							}
							return unblockedImage;
						});
						break;
					case "Removed":
						body.find("img").replaceWith(function () {
							var unBlockedImage = $(this);
							var blockedImageLink = $("<a class='fbnBlockedImage' href='" + unBlockedImage.attr("src") + "'>[image blocked]</a>");
							blockedImageLink.data("width", unBlockedImage.attr("width"));
							blockedImageLink.data("height", unBlockedImage.attr("height"));
							me.onImageReplaced.fireWith(blockedImageLink, blockedImageLink);
							return blockedImageLink;
						});
						break;
				}

				// header colors
				header
					.css("background-color", this.userConfig.get("headBackColor"))
					.css("color", this.userConfig.get("headColor"))
					.find("a").css("color", this.userConfig.get("headColor"))
				;

				// quote style and ignored reply visibility
				var visibility = this.userConfig.get("visibility");

				var threadId = $("form input[name='storyid']").val();
				var quoteLinks = body.find("a[href^='#'], a[href^='/comments/{0}']".fex(threadId));
				quoteLinks.each(function () {
					try {
						var link = $(this);
						link.text(link.text().replace("someone who may or may not be", "~"));
						var quotedUsername = link.text();

						var quotedConfig = me.childConfigs[ __userConfig.getCleanUsername( quotedUsername )];
						if (typeof quotedConfig === "undefined") {
							console.error("FixbN Failed acquiring config for quoted user {0} on a comment by {1}".fex(quotedUsername, me.userName));
							return;
						}

						// styling quotes
						link.css({ "background-color": quotedConfig.get("headBackColor"), "color": quotedConfig.get("headColor") });

						// ignoring replies as needed
						try {

							var replyVisibile = true;

							// first only bother checking if the quoted user is on ignore at all
							var quoteIgnore = (quotedConfig.get("visibility") === "Ignore");
							if (quoteIgnore) {
								

								// since they are on Ignore, then determine if they have a reply override on them
								var gIgnore = GM_config.get("ignoreReplies");
								var uIgnore = quotedConfig.get("ignoreReplies");

								switch (uIgnore) {
									case "Shown":
										quoteIgnore = false;
										break;
									case "Hidden":
										quoteIgnore = true;
										break;
									default:
										quoteIgnore = gIgnore;
										break;
								}

								if (quoteIgnore) {
									//ok, this person is set to be reply-ignored, but prefered user settings can override that

									if (!(me.userConfig.get("overrideIgnoreReplies"))) {

										// BAM! Ignored Reply!
										replyVisibile = false;

									}

								}

							}
							visibility = ( visibility === "Normal" ) && replyVisibile ? "Normal" : "Ignore";

						} catch (ex) {
							console.error("FixbN Failed determining quoted user's visibility", ex);
						}

					} catch (ex) {
						console.error(ex);
					}
				});

				switch (visibility) {
					case 'Normal':
						body.filter(".fbnIgnored").slideDown('fast').removeClass("fbnIgnored");
						header.removeClass("fbnIgnored");
						break;
					case 'Ignore':
						body.filter(":visible").addClass("fbnIgnored").slideUp('fast');
						header.addClass("fbnIgnored");
						break;
				}

			};

			UserDecoration.prototype.showUI = function (evt) {
				if (evt) {
					evt.preventDefault();
				}

				this.userConfig.open();
			};

			return UserDecoration;
		})();

		$.fn.extend({
			userDecoration: function (options) {
				if (typeof options === "string") {
					switch (options) {
						case "update":
							try {
								$(this).data("userDecoration").update();
							} catch (ex) {
								console.error(ex);
								console.log(this);
								console.error($(this).data("userDecoration"));
							}
							break;
					}
					return $(this);
				} else {
					if (options === null) {
						options = {};
					}
					return this.each(function () {
						return new UserDecoration($(this), options);
					});
				}
			}
		});

		// settings default
		$.fn.userDecoration.defaults = {
			threadId: 0,
			onImageReplaced: function () { }
		};

	})(jQuery);
} catch (ex) {
	console.error("FixbN Failed declaring UserDecoration", ex);
}

// bN taggination replacement
(function ($) {
	"use strict";

	var Tagn;
	var $bind = function (fn, me) { return function () { return fn.apply(me, arguments); }; };

	var TAG_ACCEPTED = "1";
	var INVALID_CHARACTER_IN_TAG = "2";
	var TAG_TOO_LONG = "3";
	var TOO_MANY_REPEATED_WORDS = "4";
	var TAG_MATCH = "5";

	Tagn = (function () {
		
		/*
		 * @constructor
		 */
		function Tagn($el, settings) {
			if (settings === null) {
				settings = {};
			}

			this.attach = $bind(this.attach, this);
			this.formSubmit = $bind(this.formSubmit, this);
			this.submitTags = $bind(this.submitTags, this);
			this.tagSubmitSuccess = $bind(this.tagSubmitSuccess, this);
			this.tagSubmitError = $bind(this.tagSubmitError, this);

			this.$el = $el;
			this.settings = $.extend({}, $.fn.tagn.defaults, settings);

			this.tagSet = {};

			this.attach();
		}

		Tagn.prototype.attach = function () {
			var theForm = this.$el.closest("form");
			theForm.get(0).submit = null;
			theForm
				.unbind("submit")
				.off("submit")
				.on("submit", this.formSubmit);
		};

		Tagn.prototype.formSubmit = function (evt) {
			evt.preventDefault();

			this.createTagSet(this.$el.val());

			if (typeof this.settings.beforeSubmit === "function") {
				this.settings.beforeSubmit.bind(this.$el)(this.tagSet);
			}

			this.submitTags().done(function () {
				this.settings.taggingComplete.bind(this.$el)(this.tagSet);
			}.bind(this));

			return false;
		};

		Tagn.prototype.submitTags = function() {
			var promises = [];

			try {
				var url = "";
				if (bnurl.isQueuePage()) {
					url = this.settings.queueTagUrl;
				} else {
					url = this.settings.articleTagUrl;
				}

				url = url.fex({ "storyId": this.settings.storyId });
				var me = this;

				$.each(this.tagSet.tags, function (index, tag) {
					var def = new $.Deferred();

					if (tag.status === "valid") {

						// call ajax
						$.ajax({
							url: url.fex({ "tag": tag.value }),
							type: "GET",
							dataType: "text",
							context: me,
							beforeSend: function (jqXHR, settings) {
								jqXHR.stamp = me.tagSet.stamp;
								jqXHR.tag = tag;
							},
							success: me.tagSubmitSuccess,
							error: me.tagSubmitError,
							complete: function (jqXHR, textStatus) {
								def.resolve();
							}
						});
					} else {
						$bind(me.settings.tagUpdated, me.$el)(me.tagSet);
						def.resolve();
					}

					promises.push(def);
				});
			} catch (ex) {
				console.error("Fix bN Failed sending tags", ex);
			}

			return $.when.apply(undefined, promises).promise();
		};

		Tagn.prototype.createTagSet = function (text) {
			// Split the tags separated by commas into an array.
			var initialTags = text.split(",");
			var cleaningRegex = /[^A-Za-z0-9$\.&'\s\-_]/g;

			this.tagSet = {
				"stamp": $.now(),
				"tags": [],
			};
			
			// Clean the tags and only attempt to submit valid ones.
			for (var i = 0; i < initialTags.length; ++i) {
				var currentTag = initialTags[i].trim().replace(cleaningRegex, "").trim();
				var tag = {
					'value': currentTag,
					'status': "",
					'message': ""
				};
				if (2 <= currentTag.length && currentTag.length <= 30) {
					tag.status = "valid";
				} else {
					tag.status = "invalid";
					tag.message = (currentTag.length > 5) ? "Tag too long" : "Tag too short";
				}

				this.tagSet.tags.push(tag);
			}
		};

		Tagn.prototype.tagSubmitSuccess = function (data, textStatus, jqXHR) {
			if (jqXHR.stamp !== this.tagSet.stamp) {
				return;
			}

			var tag = jqXHR.tag;
			var responseCode = data.substring(0, 1);
			switch (responseCode) {
				case TAG_ACCEPTED:
					tag.status = "accepted";
					tag.message = data.substring(2, 99).trim();
					break;
				case TAG_MATCH:
					var user = data.substring(2, 99).trim();
					tag.status = "matched";
					tag.message = (user.length === 0) ? "the auto-tagger" : user;
					break;
				case INVALID_CHARACTER_IN_TAG:
					tag.status = "rejected";
					tag.message = "Invalid character in tag";
					break;
				case TAG_TOO_LONG:
					tag.status = "rejected";
					tag.message = "Tag too long";
					break;
				case TOO_MANY_REPEATED_WORDS:
					tag.status = "rejected";
					tag.message = "Too many repeated words";
					break;
				default:
					if (data.indexOf("You must be logged in to do that") > 0) {
						tag.status = "rejected";
						tag.message = "You must be logged in to do that";
					} else {
						tag.status = "unknown";
						tag.message = "unknown response from server";
					}
					break;
			}

			$bind(this.settings.tagUpdated, this.$el)(this.tagSet);
		};

		Tagn.prototype.tagSubmitError = function (jqXHR, textStatus, errorThrown) {
			if (jqXHR.stamp !== this.tagSet.stamp) {
				return;
			}

			var tag = jqXHR.tag;
			tag.status = String(textStatus);
			tag.message = String(errorThrown);

			$bind(this.settings.tagUpdated, this.$el)(this.tagSet);
		};

		return Tagn;

	})();

	$.fn.extend({
		tagn: function (options) {
			if (options === null) {
				options = {};
			}
			return this.each(function () {
				return new Tagn($(this), options);
			});
		}
	});

	$.fn.tagn.defaults = {
		storyId: "",
		articleTagUrl: "/post?action=addtag&storyid={storyId}&tag={tag}",
		queueTagUrl: "/post?action=addqueuetag&channel_id=1&story_queue_id={storyId}&tag={tag}",

		beforeSubmit: function () { },
		tagUpdated: function () { },
		taggingComplete: function () { }
	};

})(jQuery);

// bitcoin donate jquery plugin
try {
	(function ($) {
		"use strict";

		var $bind = function (fn, me) { return function () { return fn.apply(me, arguments); }; };
		var root = "https://blockchain.info/";

		var Bitcoin = (function () {

			// @constructor
			function Bitcoin($el, settings) {
				if (settings === null) {
					settings = {};
				}

				this.attach = $bind(this.attach, this);

				this.$el = $el;
				this.settings = $.extend({}, $.fn.tagn.defaults, settings);

				this.attach();
			}

			Bitcoin.prototype.attach = function () {
				var button = this.$el;

				button.find(".blockchain").hide();
				button.find('.stage-begin').trigger('show').show();

				button.click(function () {
					var receivers_address = $(this).data('address');
					var shared = $(this).data('shared');
					var test = $(this).data('test');

					if (!shared) { shared = false; }

					var callback_url = $(this).data('callback');
					if (!callback_url) { callback_url = ''; }

					button.find('.blockchain').hide();
					button.find('.stage-loading').trigger('show').show();

					$.ajax({
						type: "GET",
						dataType: 'json',
						url: root + 'api/receive',
						data: { method: 'create', address: encodeURIComponent(receivers_address), shared: shared, callback: callback_url },
						success: function (response) {
							button.find('.qr-code').empty();
							button.find('.blockchain').hide();

							if (!response || !response.input_address) {
								button.find('.stage-error').trigger('show').show().html(button.find('.stage-error').html().replace('[[error]]', 'Unknown Error'));
								return;
							}

							function checkBalance() {
								$.ajax({
									type: "GET",
									url: root + 'q/getreceivedbyaddress/' + response.input_address,
									data: { format: 'plain' },
									success: function (response) {
										if (!response) { return; }

										var value = parseInt(response);

										if (value > 0 || test) {
											button.find('.blockchain').hide();
											button.find('.stage-paid').trigger('show').show().html(button.find('.stage-paid').html().replace('[[value]]', value / 100000000));
										} else {
											setTimeout(checkBalance, 5000);
										}
									}
								});
							}

							try {
								var ws = new WebSocket('ws://ws.blockchain.info/inv');
								if (!ws) { return; }

								ws.onmessage = function (msg) {
									try {
										var obj = $.parseJSON(msg.data);
										var result = 0;

										if (obj.op == 'utx') {
											var tx = obj.x;
																				
											for (var i = 0; i < tx.out.length; i++) {
												var output = tx.out[i];

												if (output.addr == response.input_address) {
													result += parseInt(output.value);
												}
											}
										}

										button.find('.blockchain').hide();
										button.find('.stage-paid').trigger('show').show().html(button.find('.stage-paid').html().replace('[[value]]', result / 100000000));

										ws.close();
									} catch (ex) {
										console.error(ex);
										console.error(ex.data);
									}
								};

								ws.onopen = function () {
									ws.send('{"op":"addr_sub", "addr":"' + response.input_address + '"}');
								};
							} catch (ex) {
								console.error(ex);
							}

							button.find('.stage-ready').trigger('show').show().html(button.find('.stage-ready').html().replace('[[address]]', response.input_address));
							button.find('.qr-code').html('<img style="margin:5px" src="' + root + 'qr?data=' + response.input_address + '&size=125">');
							button.unbind();

							///Check for incoming payment
							setTimeout(checkBalance, 5000);
						},
						error: function (ex) {
							button.find('.blockchain').hide();
							button.find('.stage-error').show().trigger('show').html(button.find('.stage-error').html().replace('[[error]]', ex.responseText));
						}
					});

				});
			};

			return Bitcoin;
		})();

		$.fn.extend({
			bitcoin: function (options) {
				if (options === null) {
					options = {};
				}
				return this.each(function () {
					return new Bitcoin($(this), options);
				});
			}
		});

	})(jQuery);
} catch (ex) {
	console.error("FixbN Failed declaring bitcoin", ex);
}

/*
string format function
attached to the string prototype
*/
if ("function" !== typeof "".fex) { // add fex() if one does not exist already
	String.prototype.fex = (function () { // closure to store regex
		"use strict";

		var rxp = /\{\{|\}\}|\{(\w+)\}/g;

		/*
		* String formatting function. Input string can use either indexed or named placeholders {0} or {foo}, use {{ }} to escape placeholders. Parameters can either be a series of strings, an array, or simple object with name/stringvalue pairs
		* @example
		* "foo {0} {1}".fex('bar', 'zap'); // returns "foo bar zap"
		* "foo {0} {1}".fex(['bar', 'zap']); // returns "foo bar zap"
		* "foo {second} {third}".fex({'second':'bar', 'third':'zap'}); // returns "foo bar zap"
		* @param {(...string|string[]|object} replacements - the replacement value(s) to be inserted into the string
		* @this String
		* @returns {string} The formatted string
		*/
		return function () { // the fex function
			var args = arguments;
			if (!args || args.length === 0) {
				return this.toString();
			}
			var col = {};
			if (args.length === 1) {
				var arg = args[0];
				if (arg === null) {
					arg = "null";
				}
				switch ((typeof arg)) {
					case "object":
						col = arg;

						break;
					case "string":
						col[0] = arg;

						break;
					default:
						return this.toString();
				}
			}
			if (args.length > 1) {
				col = Array.prototype.slice.call(args);
			}

			return this.replace(rxp, function (match, property) {
				if (property === null || typeof (property) === "undefined") {
					return match;
				}
				if (match === "{{") { return "{"; }
				if (match === "}}") { return "}"; }
				return (typeof col[property] !== "undefined") ? col[property] : match;
			});
		};
	}());
}
 
/*
 * string trim function
 * attached to the string prototype to support chaining
 */
if ("function" !== typeof "".trim) {
	String.prototype.trim = function () {
		return $.trim(this);
	};
}

// simple diff that assumes one addition to "n"
function diff(o, n) {
	var i = 0;
	for (i = 0; i < o.length; i++) {
		if (o[i] !== n[i]) {
			break;
		}
	}
	o = o.substring(i);
	n = n.substring(i);

	var l = 0;
	for (i = 0; i < n.length; i++) {
		if (n.substring(i) === o) {
			break;
		}
	}

	return n.substring(0, i + 1);
}

/* jshint ignore:start */
try {
	(function ($, undefined) {
		'use strict';
		var indexedDB = window.indexedDB || window.mozIndexedDB || window.webkitIndexedDB || window.msIndexedDB;
		var IDBKeyRange = window.IDBKeyRange || window.webkitIDBKeyRange;
		var IDBCursor = window.IDBCursor || window.webkitIDBCursor || {};
		if (typeof IDBCursor.PREV === "undefined") {
			IDBCursor.PREV = "prev";
		}
		if (typeof IDBCursor.NEXT === "undefined") {
			IDBCursor.NEXT = "next";
		}

		/**
		 * Best to use the constant IDBTransaction since older version support numeric types while the latest spec
		 * supports strings
		 */
		var IDBTransaction = window.IDBTransaction || window.webkitIDBTransaction;

		function getDefaultTransaction(mode) {
			var result = null;
			switch (mode) {
				case 0:
				case 1:
				case "readwrite":
				case "readonly":
					result = mode;
					break;
				default:
					result = IDBTransaction.READ_WRITE || "readwrite";
			}
			return result;
		}

		$.extend({
			/**
			 * The IndexedDB object used to open databases
			 * @param {Object} dbName - name of the database
			 * @param {Object} config - version, onupgradeneeded, onversionchange, schema
			 */
			"indexedDB": function (dbName, config) {
				if (config) {
					// Parse the config argument
					if (typeof config === "number") config = {
						"version": config
					};

					var version = config.version;
					if (config.schema && !version) {
						var max = -1;
						for (var key in config.schema) {
							max = max > key ? max : key;
						}
						version = config.version || max;
					}
				}


				var wrap = {
					"request": function (req, args) {
						return $.Deferred(function (dfd) {
							try {
								var idbRequest = typeof req === "function" ? req(args) : req;
								idbRequest.onsuccess = function (e) {

									dfd.resolveWith(idbRequest, [idbRequest.result, e]);
								};
								idbRequest.onerror = function (e) {

									dfd.rejectWith(idbRequest, [idbRequest.error, e]);
								};
								if (typeof idbRequest.onblocked !== "undefined" && idbRequest.onblocked === null) {
									idbRequest.onblocked = function (e) {

										var res;
										try {
											res = idbRequest.result;
										} catch (e) {
											res = null; // Required for Older Chrome versions, accessing result causes error 
										}
										dfd.notifyWith(idbRequest, [res, e]);
									};
								}
								if (typeof idbRequest.onupgradeneeded !== "undefined" && idbRequest.onupgradeneeded === null) {
									idbRequest.onupgradeneeded = function (e) {

										dfd.notifyWith(idbRequest, [idbRequest.result, e]);
									};
								}
							} catch (e) {
								e.name = "exception";
								dfd.rejectWith(idbRequest, ["exception", e]);
							}
						});
					},
					// Wraps the IDBTransaction to return promises, and other dependent methods
					"transaction": function (idbTransaction) {
						return {
							"objectStore": function (storeName) {
								try {
									return wrap.objectStore(idbTransaction.objectStore(storeName));
								} catch (e) {
									idbTransaction.readyState !== idbTransaction.DONE && idbTransaction.abort();
									return wrap.objectStore(null);
								}
							},
							"createObjectStore": function (storeName, storeParams) {
								try {
									return wrap.objectStore(idbTransaction.db.createObjectStore(storeName, storeParams));
								} catch (e) {
									idbTransaction.readyState !== idbTransaction.DONE && idbTransaction.abort();
								}
							},
							"deleteObjectStore": function (storeName) {
								try {
									idbTransaction.db.deleteObjectStore(storeName);
								} catch (e) {
									idbTransaction.readyState !== idbTransaction.DONE && idbTransaction.abort();
								}
							},
							"abort": function () {
								idbTransaction.abort();
							}
						};
					},
					"objectStore": function (idbObjectStore) {
						var result = {};
						// Define CRUD operations
						var crudOps = ["add", "put", "get", "delete", "clear", "count"];
						for (var i = 0; i < crudOps.length; i++) {
							result[crudOps[i]] = (function (op) {
								return function () {
									return wrap.request(function (args) {
										return idbObjectStore[op].apply(idbObjectStore, args);
									}, arguments);
								};
							})(crudOps[i]);
						}

						result.each = function (callback, range, direction) {
							return wrap.cursor(function () {
								if (direction) {
									return idbObjectStore.openCursor(wrap.range(range), direction);
								} else {
									return idbObjectStore.openCursor(wrap.range(range));
								}
							}, callback);
						};

						result.index = function (name) {
							return wrap.index(function () {
								return idbObjectStore.index(name);
							});
						};

						result.createIndex = function (prop, options, indexName) {
							if (arguments.length === 2 && typeof options === "string") {
								indexName = arguments[1];
								options = null;
							}
							if (!indexName) {
								indexName = prop;
							}
							return wrap.index(function () {
								return idbObjectStore.createIndex(indexName, prop, options);
							});
						};

						result.deleteIndex = function (indexName) {
							return idbObjectStore.deleteIndex(indexName);
						};

						return result;
					},

					"range": function (r) {
						if ($.isArray(r)) {
							if (r.length === 1) {
								return IDBKeyRange.only(r[0]);
							} else {
								return IDBKeyRange.bound(r[0], r[1], (typeof r[2] === 'undefined') ? false : r[2], (typeof r[3] === 'undefined') ? false : r[3]);
							}
						} else if (typeof r === "undefined") {
							return null;
						} else {
							return r;
						}
					},

					"cursor": function (idbCursor, callback) {
						return $.Deferred(function (dfd) {
							try {

								var cursorReq = typeof idbCursor === "function" ? idbCursor() : idbCursor;
								cursorReq.onsuccess = function (e) {

									if (!cursorReq.result) {
										dfd.resolveWith(cursorReq, [null, e]);
										return;
									}
									var elem = {
										// Delete, update do not move 
										"delete": function () {
											return wrap.request(function () {
												return cursorReq.result["delete"]();
											});
										},
										"update": function (data) {
											return wrap.request(function () {
												return cursorReq.result["update"](data);
											});
										},
										"next": function (key) {
											this.data = key;
										},
										"key": cursorReq.result.key,
										"value": cursorReq.result.value
									};

									dfd.notifyWith(cursorReq, [elem, e]);
									var result = callback.apply(cursorReq, [elem]);

									try {
										if (result === false) {
											dfd.resolveWith(cursorReq, [null, e]);
										} else if (typeof result === "number") {
											cursorReq.result["advance"].apply(cursorReq.result, [result]);
										} else {
											if (elem.data) cursorReq.result["continue"].apply(cursorReq.result, [elem.data]);
											else cursorReq.result["continue"]();
										}
									} catch (e) {

										dfd.rejectWith(cursorReq, [cursorReq.result, e]);
									}
								};
								cursorReq.onerror = function (e) {

									dfd.rejectWith(cursorReq, [cursorReq.result, e]);
								};
							} catch (e) {

								e.type = "exception";
								dfd.rejectWith(cursorReq, [null, e]);
							}
						});
					},

					"index": function (index) {
						try {
							var idbIndex = (typeof index === "function" ? index() : index);
						} catch (e) {
							idbIndex = null;
						}

						return {
							"each": function (callback, range, direction) {
								return wrap.cursor(function () {
									if (direction) {
										return idbIndex.openCursor(wrap.range(range), direction);
									} else {
										return idbIndex.openCursor(wrap.range(range));
									}

								}, callback);
							},
							"eachKey": function (callback, range, direction) {
								return wrap.cursor(function () {
									if (direction) {
										return idbIndex.openKeyCursor(wrap.range(range), direction);
									} else {
										return idbIndex.openKeyCursor(wrap.range(range));
									}
								}, callback);
							},
							"get": function (key) {
								if (typeof idbIndex.get === "function") {
									return wrap.request(idbIndex.get(key));
								} else {
									return idbIndex.openCursor(wrap.range(key));
								}
							},
							"count": function () {
								if (typeof idbIndex.count === "function") {
									return wrap.request(idbIndex.count());
								} else {
									throw "Count not implemented for cursors";
								}
							},
							"getKey": function (key) {
								if (typeof idbIndex.getKey === "function") {
									return wrap.request(idbIndex.getKey(key));
								} else {
									return idbIndex.openKeyCursor(wrap.range(key));
								}
							}
						};
					}
				};


				// Start with opening the database
				var dbPromise = wrap.request(function () {

					return version ? indexedDB.open(dbName, parseInt(version)) : indexedDB.open(dbName);
				});
				dbPromise.then(function (db, e) {

					db.onversionchange = function () {
						// Try to automatically close the database if there is a version change request
						if (!(config && config.onversionchange && config.onversionchange() !== false)) {
							db.close();
						}
					};
				}, function (error, e) {

					// Nothing much to do if an error occurs
				}, function (db, e) {
					if (e && e.type === "upgradeneeded") {
						if (config && config.schema) {
							// Assuming that version is always an integer 

							for (var i = e.oldVersion + 1; i <= e.newVersion; i++) {
								typeof config.schema[i] === "function" && config.schema[i].call(this, wrap.transaction(this.transaction));
							}
						}
						if (config && typeof config.upgrade === "function") {
							config.upgrade.call(this, wrap.transaction(this.transaction));
						}
					}
				});

				return $.extend(dbPromise, {
					"cmp": function (key1, key2) {
						return indexedDB.cmp(key1, key2);
					},
					"deleteDatabase": function () {
						// Kinda looks ugly coz DB is opened before it needs to be deleted. 
						// Blame it on the API 
						return $.Deferred(function (dfd) {
							dbPromise.then(function (db, e) {
								db.close();
								wrap.request(function () {
									return indexedDB.deleteDatabase(dbName);
								}).then(function (result, e) {
									dfd.resolveWith(this, [result, e]);
								}, function (error, e) {
									dfd.rejectWith(this, [error, e]);
								}, function (db, e) {
									dfd.notifyWith(this, [db, e]);
								});
							}, function (error, e) {
								dfd.rejectWith(this, [error, e]);
							}, function (db, e) {
								dfd.notifyWith(this, [db, e]);
							});
						});
					},
					"transaction": function (storeNames, mode) {
						!$.isArray(storeNames) && (storeNames = [storeNames]);
						mode = getDefaultTransaction(mode);
						return $.Deferred(function (dfd) {
							dbPromise.then(function (db, e) {
								var idbTransaction;
								try {

									idbTransaction = db.transaction(storeNames, mode);

									idbTransaction.onabort = idbTransaction.onerror = function (e) {
										dfd.rejectWith(idbTransaction, [e]);
									};
									idbTransaction.oncomplete = function (e) {
										dfd.resolveWith(idbTransaction, [e]);
									};
								} catch (e) {

									e.type = "exception";
									dfd.rejectWith(this, [e]);
									return;
								}
								try {
									dfd.notifyWith(idbTransaction, [wrap.transaction(idbTransaction)]);
								} catch (e) {
									e.type = "exception";
									dfd.rejectWith(this, [e]);
								}
							}, function (err, e) {
								dfd.rejectWith(this, [e, err]);
							}, function (res, e) {

								//dfd.notifyWith(this, ["", e]);
							});

						});
					},
					"objectStore": function (storeName, mode) {
						var me = this,
							result = {};

						function op(callback) {
							return $.Deferred(function (dfd) {
								function onTransactionProgress(trans, callback) {
									try {

										callback(trans.objectStore(storeName)).then(function (result, e) {
											dfd.resolveWith(this, [result, e]);
										}, function (err, e) {
											dfd.rejectWith(this, [err, e]);
										});
									} catch (e) {

										e.name = "exception";
										dfd.rejectWith(trans, [e, e]);
									}
								}
								me.transaction(storeName, getDefaultTransaction(mode)).then(function () {

									// Nothing to do when transaction is complete
								}, function (err, e) {
									// If transaction fails, CrudOp fails
									if (err.code === err.NOT_FOUND_ERR && (mode === true || typeof mode === "object")) {

										var db = this.result;
										db.close();
										dbPromise = wrap.request(function () {

											return indexedDB.open(dbName, (parseInt(db.version, 10) || 1) + 1);
										});
										dbPromise.then(function (db, e) {

											db.onversionchange = function () {
												// Try to automatically close the database if there is a version change request
												if (!(config && config.onversionchange && config.onversionchange() !== false)) {
													db.close();
												}
											};
											me.transaction(storeName, getDefaultTransaction(mode)).then(function () {

												// Nothing much to do
											}, function (err, e) {
												dfd.rejectWith(this, [err, e]);
											}, function (trans, e) {

												onTransactionProgress(trans, callback);
											});
										}, function (err, e) {
											dfd.rejectWith(this, [err, e]);
										}, function (db, e) {
											if (e.type === "upgradeneeded") {
												try {

													db.createObjectStore(storeName, mode === true ? {
														"autoIncrement": true
													} : mode);

												} catch (ex) {

													dfd.rejectWith(this, [ex, e]);
												}
											}
										});
									} else {
										dfd.rejectWith(this, [err, e]);
									}
								}, function (trans) {

									onTransactionProgress(trans, callback);
								});
							});
						}

						function crudOp(opName, args) {
							return op(function (wrappedObjectStore) {
								return wrappedObjectStore[opName].apply(wrappedObjectStore, args);
							});
						}

						function indexOp(opName, indexName, args) {
							return op(function (wrappedObjectStore) {
								var index = wrappedObjectStore.index(indexName);
								return index[opName].apply(index[opName], args);
							});
						}

						var crud = ["add", "delete", "get", "put", "clear", "count", "each"];
						for (var i = 0; i < crud.length; i++) {
							result[crud[i]] = (function (op) {
								return function () {
									return crudOp(op, arguments);
								};
							})(crud[i]);
						}

						result.index = function (indexName) {
							return {
								"each": function (callback, range, direction) {
									return indexOp("each", indexName, [callback, range, direction]);
								},
								"eachKey": function (callback, range, direction) {
									return indexOp("eachKey", indexName, [callback, range, direction]);
								},
								"get": function (key) {
									return indexOp("get", indexName, [key]);
								},
								"count": function () {
									return indexOp("count", indexName, []);
								},
								"getKey": function (key) {
									return indexOp("getKey", indexName, [key]);
								}
							};
						};

						return result;
					}
				});
			}
		});

		$.indexedDB.IDBCursor = IDBCursor;
		$.indexedDB.IDBTransaction = IDBTransaction;
		$.idb = $.indexedDB;
	})(jQuery);
} catch (ex) {
	console.error("FixbN Failed Declaring indexedDB", ex);
}
/* jshint ignore:end */

// The kickoff function
$(document).ready(function () {
	"use strict";

	// general style changes
	/* jshint -W064 */
	GM_addStyle((function () {
		return [
			"div#selectionMenu { box-shadow:0px 0px 5px black; background:#eeeeee; border:4px solid black; border-radius:5px; padding:5px; padding-left:8px; }",
			"div#selectionMenu div { cursor:pointer; }",
			"div#selectionMenu  hr { margin:3px; padding:0px; }",
			"div#menu { vertical-align:middle; }",
			"div#menu ul { font-size: 0.8em; white-space: nowrap; clear:both; padding:0px; margin-bottom:5px; }",
			"div#menu ul li img.fbnIcon { display:inline;float:none;margin:0px;padding:0px; height:20px;}",
			"div#menu ul.leftMenu { float:left; padding:0px; }",
			"div#menu ul.leftMenu li { margin:0px; padding:0px 0.5em; border:none; border-right:1px solid white; }",
			"div#menu ul.leftMenu li:last-child { border:none; }",
			"div.profilePopup a { color: white; }",
			"div.profilePopup a.topLink { padding-bottom:5px; }",
			"div.recentCommentWrapper { font-size:12px; padding-top:5px; }",
			"div.recentCommentWrapper a:visited { color:#aaa; }",
			"div.fbnRepliesToMe { float:left; background:#807373; border: 1px solid black; margin-left:10px; margin-right:10px; margin-bottom:10px; font-size:0.75em; }",
			"div.fbnRepliesToMe span { color:white; cursor:pointer; padding-left:0.5em; padding-right:0.5em; }",
			"div.fbnRepliesToMe ul { display: inline; }",
			"div.fbnRepliesToMe ul li { display: inline-block; white-space:nowrap; border-left: 1px solid white; padding-left: 0.5em; padding-right:0.5em; margin:0em; }",
			"div.fbnRepliesToMe ul a { color:white; }",
			"div.fbnRepliesToMe ul a:visited { color:#bbb; }",
			".notyReply a { color: white; }",
			"div.fbnScrollToBottom { margin:0px; padding:0px; border:none; position:fixed; top:-5px; left:auto; right:-3px; z-index:1001; }",
			"div.fbnScrollToBottom img { margin:0px; padding:0px; border:none; cursor:pointer; }",
			"div.fbnScrollToTop { margin:0px; padding:0px; border:none; position:fixed; top:-5px; left:auto; right:-3px; z-index:1001; }",
			"div.fbnScrollToTop img { margin:0px; padding:0px; border:none; cursor:pointer; }",
			"div.mfp-content div.ch { margin-left: 5em; margin-right:5em; border:10px solid white; border-bottom-width:0px; }",
			"div.mfp-content div.cb { margin-left: 5em; margin-right:5em; border:10px solid white; border-top-width:0px;}",
			"div.stickyTagger {margin:0.25em; padding:0em; background:#807373; position: fixed; left: auto; right: 0; bottom: 0; text-align: right; border:1px solid black; box-shadow: 0px 0px 5px black;}",
			"div.stickyTagger label { color:white; cursor:pointer; padding:2em 1em 2em 1em; font-size:1.5em; }",
			"div.stickyTagger input { margin-right:1em; display:none; font-size:1.25em; margin-bottom:3px; width: 30em; }",
			"div#header.fixedHeader { position:fixed; top:0px; box-shadow: 0px 0px 10px black; z-index:1000; }",
			"div.form { width:98%; min-width:511px; }",
			"div#tag_cloud span { cursor:pointer; }",
			"div#footer { color: #CCCCCC; font-size: 0.5em; }",
			"div#footer a { font-size: 1.5em; }",
			"a.commentsLinkButton { float:right; margin-top:-15px; margin-right:-10px; }",
			"div.ch.fbnIgnored img { opacity: 0; }",
			"div.ch.highScore { padding-left: 25px; background-image: url(); background-repeat: no-repeat; background-attachment: scroll; background-position:left center;  }",
			"div.ch img.nabbitOff { opacity: 0; }",
			"div.ch img.nabbitForceOn { opacity: 1; }",
			"div.ch span.peep { display: none; float: left; }",
			"div.ch.fbnIgnored span.peep { display: inline; float: left; cursor:pointer; font-weight:bold; padding: 0px 0.5em 0px 0.5em; margin: 0.25em 0px 0.5em 0px; }",
			"div.cb ul, div.cb pre { padding:0px; margin:0px; } div.cb li { padding:0px; margin:0px 0px 0px 1em; }",
			"div.cb .commentquote { display: block; padding:0px 0px 0px 0.5em; margin:1em 0px 0px 0.5em; border-left: 2px solid #807373; font-size: 0.75em; }",
			"div.cb a[href^='#'] { color:black; }",
			"div#menu a.a1{padding-right:25px;background:url(../img/bncombined.png) no-repeat top;background-position:right -360px;}",
			"div#menu a.a1{padding-right:25px;background:url(../img/bncombined.png) no-repeat top;background-position:right -360px;}",
			"div#menu a.a2{padding-right:25px;background:url(../img/bncombined.png) no-repeat top;background-position:right -432px;}",
			"div#menu a.a3{padding-right:25px;background:url(../img/bncombined.png) no-repeat center;background-position:right -504px;}",
			"div#menu a.a4{padding-right:25px;background:url(../img/bncombined.png) no-repeat center;background-position:right -576px;}",
			"div#menu a.a5{padding-right:25px;background:url(../img/bncombined.png) no-repeat center;background-position:right -648px;}",
			"div#menu a.a1{padding-right:25px;background:url(../img/bncombined.png) no-repeat top;background-position:right -360px;}"
		].join('\n') + '\n';
	})()); 

	// gm_config styling
	GM_addStyle((function () {
		return [
			"div[id$='_wrapper'] * { font-family: calibri,arial,tahoma,myriad pro,sans-serif; }",
			"div[id$='_wrapper'] { background: #FFF; padding: 0px; height: 99%; background-repeat: no-repeat; background-position: top right; background-size: contain; background-image:url(); }",

			"div[id$='_wrapper'] .block { display: block; }",
			"div[id$='_wrapper'] .center { text-align: center; }",
			"div[id$='_wrapper'] .indent40 { margin-left: 40%; }",

			"div[id$='_wrapper'] .config_header { font-size: 2em; margin: -5px -5px 0px -5px; background-color: #807373; color:white; white-space: nowrap;}",
			"div[id$='_wrapper'] .config_desc, div[id$='_wrapper'] .section_desc, div[id$='_wrapper'] .reset { font-size: 0.75em; }",

			"div[id$='_wrapper'] .config_var { margin: 0.5em; padding: 0.5em; }",
			"div[id$='_wrapper'] .config_var input[type='checkbox'] { float:left; position:relative; top:2px; margin-right:5px; }",
			"div[id$='_wrapper'] input[type='radio'] { margin-left: 0.5em; margin-top: 0.5em; }",
			"div[id$='_wrapper'] .field_label { font-size: 1em; font-weight: bold; margin-right: 1em; }",
			"div[id$='_wrapper'] .radio_label { font-size: 1em; margin-right: 1em; }",

			"div[id$='_wrapper'] .section_header_holder { margin-top: 1em; }",
			"div[id$='_wrapper'] .section_header { background: #414141; border: 1px solid #000; color: #FFF; font-size: 13pt; margin: 0; }",
			"div[id$='_wrapper'] .section_desc { background: #EFEFEF; border: 1px solid #CCC; color: #575757; font-size: 9pt; margin: 0 0 6px; }",

			"div[id$='_wrapper'] div[id$='_buttons_holder'] { position:absolute; bottom:0px; right:0px; left:0px; color: #000; text-align: right; padding-right:0.5em; padding-bottom:0.5em; }",
			"div[id$='_wrapper'] div.saveclose_buttons { margin:0em 2em 2em 0em; }",
			"div[id$='_wrapper'] button[id$='_saveBtn'] { font-weight:bold; }",
			"div[id$='_wrapper'] button[id$='_closeBtn'] { font-weight:bold; }"
		].join('\n') + '\n';
	})()); 
	/* jshint +W064 */

	// add markItUp! style information (including base64 images).
	// converted by Stephen Cronin
	/* jshint ignore:start */
	GM_addStyle((function () {
		return "\
.markItUp .miuBold   a { background-image:url(); } \
.markItUp .miuItalic a { background-image:url(); } \
.markItUp .miuItalic a { background-image:url(); } \
.markItUp .miuStrike a { background-image:url(); } \
.markItUp .miuPara a { background-image:url(); } \
.markItUp .miuQuote a { background-image:url(); } \
.markItUp .miuUList a { background-image:url(); } \
.markItUp .miuOList a { background-image:url(); } \
.markItUp .miuListitem a { background-image:url(); } \
.markItUp .miuImage a { background-image:url(); } \
.markItUp .miuLink a { background-image:url(); } \
.markItUp .miuPre a { background-image:url(); } \
.markItUp .miuClean a { background-image:url(); } \
.markItUp .miuSymbols a { background-image:url(); } \
.markItUp * { margin:0px; padding:0px; outline:none; } \
.markItUp a:link, .markItUp a:visited { color:#000; text-decoration:none; } \
.markItUp  { width:100% !important; margin:5px 0 5px 0; } \
.markItUpContainer  { border: 2px solid #807373; padding:5px 10px 0px 0px; font:1em Calibri, Verdana, Arial, Helvetica, sans-serif; color:white; } \
.markItUpEditor { font:1em Calibri, Verdana, Arial, Helvetica, sans-serif; background-color: #eeeeee; padding:5px !important; width:100% !important; border:none; margin:0px; height:320px; clear:both; display:block; line-height:18px; overflow:auto; } \
.markItUpPreviewFrame	{ overflow:auto; background-color:#FFFFFF; border:1px solid #3C769D; width:99.9%; height:300px; margin:5px 0; } \
.markItUpFooter { width:100%; cursor:n-resize; } \
.markItUpResizeHandle { overflow:hidden; width:22px; height:5px; margin-left:auto; margin-right:auto; cursor:n-resize; background-image:url(); } \
.markItUpHeader ul { padding-left:3px; } \
.markItUpHeader ul li	{ list-style:none; float:left; position:relative; } \
.markItUpHeader ul li ul{ display:none; } \
.markItUpHeader ul li:hover > ul{ display:block; } \
.markItUpHeader ul .markItUpDropMenu { margin-right:5px; background:transparent url() no-repeat 115% 50%; } \
.markItUpHeader ul .markItUpDropMenu li { margin-right:0px; } \
.markItUpHeader ul .markItUpSeparator { margin:0 10px; width:1px; height:16px; overflow:hidden; background-color:#CCC; } \
.markItUpHeader ul ul .markItUpSeparator { width:auto; height:1px; margin:0px; } \
.markItUpHeader ul ul { display:none; position:absolute; top:18px; left:0px; background:#F5F5F5; border:1px solid #3C769D; height:inherit; } \
.markItUpHeader ul ul li { float:none; border-bottom:1px solid #3C769D; } \
.markItUpHeader ul ul .markItUpDropMenu { background:#F5F5F5 url() no-repeat 100% 50%; } \
.markItUpHeader ul ul ul { position:absolute; top:-1px; left:150px; } \
.markItUpHeader ul ul ul li { float:none; } \
.markItUpHeader ul a { display:block; width:16px; height:16px; text-indent:-10000px; background-repeat:no-repeat; padding:3px; margin:0px; } \
.markItUpHeader ul ul a { display:block; padding-left:0px; text-indent:0; width:120px; padding:5px 5px 5px 25px; background-position:2px 50%; } \
.markItUpHeader ul ul a:hover  { color:#FFF; background-color:#3C769D; } \
.markItUp .palette a { background-image:url(); } \
.markItUp .palette ul { width:81px; padding:1px; } \
.markItUp .palette li { border:1px solid white; width:25px;	height:25px; overflow:hidden; padding:0px; margin:0px; float:left; } \
.markItUp .palette ul a { width:25px;	height:25px; } \
.markItUp .palette ul a:hover { background-color:transparent; } \
.markItUp .palette .col1-1 a { background:#FCE94F; } \
.markItUp .palette .col1-2 a { background:#EDD400; } \
.markItUp .palette .col1-3 a { background:#C4A000; } \
.markItUp .palette .col2-1 a { background:#FCAF3E; } \
.markItUp .palette .col2-2 a { background:#F57900; } \
.markItUp .palette .col2-3 a { background:#CE5C00; } \
.markItUp .palette .col3-1 a { background:#E9B96E; } \
.markItUp .palette .col3-2 a { background:#C17D11; } \
.markItUp .palette .col3-3 a { background:#8F5902; } \
.markItUp .palette .col4-1 a { background:#8AE234; } \
.markItUp .palette .col4-2 a { background:#73D216; } \
.markItUp .palette .col4-3 a { background:#4E9A06; } \
.markItUp .palette .col5-1 a { background:#729FCF; } \
.markItUp .palette .col5-2 a { background:#3465A4; } \
.markItUp .palette .col5-3 a { background:#204A87; } \
.markItUp .palette .col6-1 a { background:#AD7FA8; } \
.markItUp .palette .col6-2 a { background:#75507B; } \
.markItUp .palette .col6-3 a { background:#5C3566; } \
.markItUp .palette .col7-1 a { background:#EF2929; } \
.markItUp .palette .col7-2 a { background:#CC0000; } \
.markItUp .palette .col7-3 a { background:#A40000; } \
.markItUp .palette .col8-1 a { background:#FFFFFF; } \
.markItUp .palette .col8-2 a { background:#D3D7CF; } \
.markItUp .palette .col8-3 a { background:#BABDB6; } \
.markItUp .palette .col9-1 a { background:#888A85; } \
.markItUp .palette .col9-2 a { background:#555753; } \
.markItUp .palette .col9-3 a { background:#000000; } \
";
	})());
	/* jshint ignore:end */

	/* jshint -W064 */
	try {
		GM_addStyle(GM_getResourceText('magnificcss')); 
		GM_addStyle(GM_getResourceText('juipepper'));
		GM_addStyle(GM_getResourceText('spectrumcss'));
	} catch (ex) {
		console.error("Fix bN Failed applying css resources", ex);
	}
	/* jshint +W064 */

	__fixbn.fix();

});
console.info("Fix bN v" + GM_info.script.version);