Reddit Enhancer

Extends Reddit and adds some useful Functions. (Requires frisch's UserScript Extender)

// ==UserScript==
// @name        Reddit Enhancer
// @namespace   http://null.frisch-live.de/
// @version     1.85
// @description Extends Reddit and adds some useful Functions. (Requires frisch's UserScript Extender)
// @author      frisch
// @include     http://*.reddit.com/*
// @include     https://*.reddit.com/*
// @exclude     https://*.reddit.com/message/*
// @exclude     http://*.reddit.com/message/*
// @exclude     https://*.reddit.com/search?*
// @grant       GM_getValue
// @grant       GM_setValue
// @grant       GM_openInTab
// ==/UserScript==
console.log("Initializing Reddit Enhancer...");

var settings = {};

var enumKeys = {
	'NONE': -1,
	'BACKSPACE': 8,
	'TAB': 9,
	'ENTER': 13,
	'SHIFT': 16,
	'CTRL': 17,
	'ALT': 18,
	'PAUSE/BREAK': 19,
	'CAPS LOCK': 20,
	'ESCAPE': 27,
	'SPACE': 32,
	'PAGE UP': 33,
	'PAGE DOWN': 34,
	'END': 35,
	'HOME': 36,
	'LEFT ARROW': 37,
	'UP ARROW': 38,
	'RIGHT ARROW': 39,
	'DOWN ARROW': 40,
	'INSERT': 45,
	'DELETE': 46,
	'0': 48,
	'1': 49,
	'2': 50,
	'3': 51,
	'4': 52,
	'5': 53,
	'6': 54,
	'7': 55,
	'8': 56,
	'9': 57,
	'A': 65,
	'B': 66,
	'C': 67,
	'D': 68,
	'E': 69,
	'F': 70,
	'G': 71,
	'H': 72,
	'I': 73,
	'J': 74,
	'K': 75,
	'L': 76,
	'M': 77,
	'N': 78,
	'O': 79,
	'P': 80,
	'Q': 81,
	'R': 82,
	'S': 83,
	'T': 84,
	'U': 85,
	'V': 86,
	'W': 87,
	'X': 88,
	'Y': 89,
	'Z': 90,
	'LEFT WINDOW KEY': 91,
	'RIGHT WINDOW KEY': 92,
	'SELECT KEY': 93,
	'NUMPAD 0': 96,
	'NUMPAD 1': 97,
	'NUMPAD 2': 98,
	'NUMPAD 3': 99,
	'NUMPAD 4': 100,
	'NUMPAD 5': 101,
	'NUMPAD 6': 102,
	'NUMPAD 7': 103,
	'NUMPAD 8': 104,
	'NUMPAD 9': 105,
	'MULTIPLY': 106,
	'ADD': 107,
	'SUBTRACT': 109,
	'DECIMAL POINT': 110,
	'DIVIDE': 111,
	'F1': 112,
	'F2': 113,
	'F3': 114,
	'F4': 115,
	'F5': 116,
	'F6': 117,
	'F7': 118,
	'F8': 119,
	'F9': 120,
	'F10': 121,
	'F11': 122,
	'F12': 123,
	'NUM LOCK': 144,
	'SCROLL LOCK': 145,
	'SEMI-COLON': 186,
	'EQUAL SIGN': 187,
	'COMMA': 188,
	'DASH': 189,
	'PERIOD': 190,
	'FORWARD SLASH': 191,
	'GRAVE ACCENT': 192,
	'OPEN BRACKET': 219,
	'BACK SLASH': 220,
	'CLOSE BRAKET': 221,
	'SINGLE QUOTE': 222,
};
enumKeys.hasValue = function(val){
	for(var k in this)
		if(this[k] === val) {
			return true;
		}

	return false;
};

// User Script Enhancer Properties and Functions
var jq = document.fExt.jq;
var fExtCreateStyle = document.fExt.createStyle;
var fExtAddSub = document.fExt.ctxMenu.addCtxSub;
var fExtMessage = document.fExt.message;
var fExtAddCtxItem = document.fExt.ctxMenu.addCtxItem;
var fExtAddCtxSeparator = document.fExt.ctxMenu.addSeparator;
var fExtClipboard = document.fExt.clipboard;
var fExtGetSelection = document.fExt.getSelection;
var fExtPopup = document.fExt.popup;
var fExtSetLoading = document.fExt.setLoading;

var typeCheck = Object.prototype.toString;
var numberOfSitesLoaded = 1;
var isHiding = false;

var ctxItemUsrHdr, ctxItemUsrHdrPg, ctxItemUsrUnhdr, ctxItemSbHdr, ctxItemSbUnhdr, ctxItemKeywHdr, ctxItemKeywUnhdr, ctxUndo, ctxRedo, ctxItemCpyHtml, ctxItemSett;
var upVoteLinks, downVoteLinks, rSubRedditLinks, rAuthorLinks, postList;

var loc = location.href;
var isSinglePost = location.href.indexOf("/comments/") > 0 || location.href.indexOf("/submit$") > 0;
var isUserPage = location.href.indexOf("/user/") > 0 || location.href.indexOf("/u/") > 0;
var isUserSubmittedPage = isUserPage && location.href.indexOf("/submitted") > 0;
var isMyMulti = location.href.indexOf("/me/m/") > 0;
var isFriendsPage = location.href.indexOf("/r/friends") > 0;
var isSubmitPage = location.href.indexOf("/submit?") > 0;
var isMySavedPage = location.href.indexOf(settings.accountName + "/saved") > 0;
var hiderElements;
var headerHeight = jq("#header").outerHeight(true);
var hideLinks = [];
var userHideLinks = [];
var subHideLinks = [];
var duplicateLinks = [];
var keywordsHideLinks = [];
var rTitleLinks;
var rEnhSub = fExtAddSub("Reddit Enhancer", undefined);
var pstItems = [];
pstItems.onlyVisible = function(){
	return this.filter(function(item){
		return jq("#" + item).is(":visible");
	});
};
pstItems.Next = function(steps){
	return GetFromArray(pstItems.onlyVisible(), pstItems.selected, steps ? steps : 1);
};
pstItems.Previous = function(steps){
	return GetFromArray(pstItems.onlyVisible(), pstItems.selected, steps ? steps * -1 : -1);
};
pstItems.First = function(){
	var visiblePosts = pstItems.onlyVisible();
	return visiblePosts.length > 0 ? visiblePosts[0] : undefined;
};
pstItems.Last = function(){
	var visiblePosts = pstItems.onlyVisible();
	return visiblePosts.length > 0 ? visiblePosts[visiblePosts.length - 1] : undefined;
};

function GetFromArray(array, item, steps) {
	if(array.length === 0)
		return undefined;
	else if(item && jq.inArray(item, array) < 0)
		return undefined;

	var ind = 0;
	if(item)
		ind = array.indexOf(item);

	if(ind < 0)
		ind = 0;

	var stepIndex = (steps < 0 ? steps * -1 : steps);
	var modifier = (steps < 0 ? -1 : 1);

	ind += steps;
	while(ind >= array.length)
		ind -= array.length;

	if(ind < 0)
		ind = array.length - 1;

	return array[ind] === item ? undefined : array[ind];
}

var reloadCancelled = true;
var resourcesLinks = [];
var subColors = {};
var settingTranslation = [
	{ key: "accountName", translation: "Your Profilename", hint: "Required to determine some pages (e.g. your saved pages)" },

	{ key: "markNSFW", translation: "Mark NSFW posts", hint: "Highlights NSFW posts" },
	{ key: "replaceTopNew", translation: "Replace Top with New", hint: "Instead of landing on the Top-Page of a reddit, you'll be redirected to the New-Page instead" },
	{ key: "sortByVotes", translation: "Sort by Votes", hint: "Sorts the posts by votes" },
	{ key: "autoExpandSelectedPost", translation: "Autoexpand Posts", hint: "Causes selected Posts to automatically collapse/expand" },
	{ key: "removeDuplicates", translation: "Remove Duplicates", hint: "Removes duplicate posts by checking the link/source of a post" },
	{ key: "maximumNumberOfSitesToLoad", translation: "Autoload # of Pages", hint: "Let's you load more than one page at once. Currently buggy and will be worked on." },

	{ key: "animationItemTimeout", translation: "Animation Speed", hint: "" },
	{ key: "hideItemTimeout", translation: "Posthiding Timeout", hint: "" },
	{ key: "imagePreloadingTimeout", translation: "Preloading timeout", hint: "" },

	{ key: "hideUsers", translation: "Hide Users", hint: "If true, users in the 'Users to hide'-list will be hidden" },
	{ key: "hideUsersNSFWonly", translation: "Hide Users NSFW posts only", hint: "Only hide NSFW posts by the users" },
	{ key: "usersToHide", translation: "Users to hide", hint: "" },

	{ key: "hideSubs", translation: "Hide Subs", hint: "If true, subsreddits in the 'Subs to hide'-list will be hidden" },
	{ key: "subsToHide", translation: "Subs to hide", hint: "" },

	{ key: "hideKeywords", translation: "Hide Keywords", hint: "If true, posts containing any keyword fromt he 'Keywords to hide'-list will be hidden" },
	{ key: "keywordsToHide", translation: "Keywords to hide", hint: "" },

	{ key: "panelView", translation: "Panelview", hint: "Not yet implemented" },

	{ key: "shortcuts", translation: "Keyboard Shortcuts", hint: "" },
	{ key: "voteUp", translation: "Upvote selected Post", hint: "" },
	{ key: "previousPost", translation: "Select previous Post", hint: "" },
	{ key: "voteDown", translation: "Downvote selected Post", hint: "" },
	{ key: "toggleSelected", translation: "Expand/Collapse selected Post", hint: "" },
	{ key: "hideSelected", translation: "Hide selected Post", hint: "" },
	{ key: "nextPost", translation: "Select next Post", hint: "" },
	{ key: "hideAll", translation: "Hide all Posts", hint: "Also reloads the page after hiding all posts (can be cancelled)" },

	{ key: "undoLast", translation: "Undo last action", hint: "Activates the latest element in the undo list" },
	{ key: "redoLast", translation: "Redo last action", hint: "Activates the latest element in the redo list" },

	{ key: "previousImage", translation: "Display previous Image", hint: "Used for RES-Albums" },
	{ key: "nextImage", translation: "Display next Image", hint: "Used for RES-Albums" },

	{ key: "zoomIn", translation: "Zoom in", hint: "Used for Images" },
	{ key: "zoomOut", translation: "Zoom out", hint: "Used for Images" },
	{ key: "rotateLeft", translation: "Rotate left", hint: "Used for Images" },
	{ key: "rotateRight", translation: "Rotate right", hint: "Used for Images" }
];

// Images
var imgBooks = "";
var imgDuplicate = "";
var imgDelCat = "";
var imgAbort = "";
var imgDelUser = "";
var imgTree = "";

// Variables - End

// Initialization - Start
GMLoad();

// Initialization - Styles - Start
jq("body").css("margin-bottom", "0");
// Initialization - Styles - End

// Initialization - Controls - Start
if (isSinglePost) {
	jq("div.media-preview").attr("style","");
	jq("#upvoter,#downvoter,#expander,#hideAll,#hideSelected,#toggleUserHidden,#toggleSubsHidden,#toggleKeywordsHidden,#toggleDuplicateHidden").addClass("disabled");
}

// Initialization - Controls - Settings Dialog
var settingsDialog = '<div style="display:none" id="rEnhSettings"><div class="tableContainer"><table cellspacing="0" cellpadding="0" border="0" width="700px"><colgroup><col width="200px" /><col width="auto" /></colgroup>';
settingsDialog += '<thead><tr><th class="label">Setting</th><th class="setting">Value</th></tr></thead>';

function BuildSettings(settingsObject) {
	for (var key in settingsObject) {
		var varType = typeof settingsObject[key];

		if (varType === "object") {
			if (typeCheck.call(settingsObject[key]) === '[object Array]')
				AddSettingToDialog(key, "list");
			else {
				settingsDialog += '<tr><td class="label" colspan="2"><b>' + key + '</b></td></tr>';

				if (key === 'shortcuts') {
					for (var shortcut in settingsObject[key]) {
						AddSettingToDialog(shortcut, 'key');
					}
				} else BuildSettings(settingsObject[key]);
			}
		} else AddSettingToDialog(key, varType);
	}
}

function getTranslationObject(key) {
	for (var i in settingTranslation) {
		if (settingTranslation[i].key === key) {
			return settingTranslation[i];
		}
	}

	return { key: key, translation: key, hint: "" };
}

function AddSettingToDialog(key, varType) {
	var transObject = getTranslationObject(key);
	settingsDialog += '<tr title="' + transObject.hint + '">';
	if (varType === "list") {
		settingsDialog += '<td class="label" colspan="2">' + transObject.translation + '</td>';
		settingsDialog += '</tr>';
		settingsDialog += '<tr title="' + transObject.hint + '">';
		settingsDialog += '<td class="setting" colspan="2"><textarea name="' + key + '" class="value" cols="80" rows="7" title="Separate entries by using comma"></textarea></td>';
	} else {
		settingsDialog += '<td class="label">' + transObject.translation + '</td>';
		if (varType === "key") {
			settingsDialog += '<td><select class="value" name="' + key + '">';
			for (var knownKey in enumKeys) {
				settingsDialog += '<option value="' + enumKeys[knownKey] + '">' + knownKey + '</option>';
			}
			settingsDialog += '</select><input type="Button" class="DetectShortcut" value="Assign" /></td>';
		} else {
			var inputType;
			switch (varType) {
				case "boolean":
					inputType = "checkbox";
					break;
				case "number":
					inputType = "number";
					break;
				case "string":
					inputType = "text";
					break;
				default:
					console.log("Cannot add setting " + key + " to settings dialog because unkown type " + varType);
					return;
			}

			settingsDialog += '<td class="setting">' + '<input type="' + inputType + '" class="value" name="' + key + '"></td>';
		}
	}
	settingsDialog += '</tr>';
}

jq(document).on("click", "input.DetectShortcut", function(e) {
	fExtMessage("Press the key to assign or press ESC to unassign.");
	jq(document).bind("keydown", { sender: this }, AssignShortcut);
});

function AssignShortcut(e){
	var sender = e.data.sender;
	var jqSender = jq(sender);

	e.preventDefault();

	var kc = e.which;

	if(kc != enumKeys.ESCAPE) {
		if(!enumKeys.hasValue(kc)){
			fExtMessage("Invalid Key! Choose a different key or press ESC to unassign.");
			return false;
		}
		else
			jqSender.parent().find("select").val(kc);
	}
	else
		jqSender.parent().find("select").val(enumKeys.NONE);


	fExtMessage(undefined);

	jq(document).unbind("keydown", AssignShortcut);
	return false;
}

BuildSettings(settings);

settingsDialog += '</table></div>';
settingsDialog += '<div class="settingsButtons"><a href="#" id="renhSettingsClose">Close</a><a href="#" id="renhSettingsSave">Save</a></div>';
settingsDialog += '</div>';
jq("body").append(settingsDialog);
// Initialization - Controls - End

// Moving the navigation buttons
var jqNavButtons = jq("div.nav-buttons");
if (jqNavButtons) {
	if (jqNavButtons.length === 1 && settings.maximumNumberOfSitesToLoad > 1) {
		GrabContent(jq("body").html());
	} else MainInitialization();
} else MainInitialization();

function MainInitialization() {
	InitializeElements();
	InitializeStyles();
	InitializeEventHandlers();


	var fnHandleShortcut;
	if (!isSinglePost) {
		fnHandleShortcut = handleShortcutForOverview;
		postList = jq("#siteTable").find("div.thing");

		postList.each(function(){
			jq(this).find("div.midcol").find("div.unvoted").each(function(index, item) {
				var jqItem = jq(item);
				var unvotedVal = jqItem.text();
				var multiplier = unvotedVal.indexOf("k") >= 0 ? 1000 : 1;
				var pts = parseFloat(unvotedVal.replace("k",""));
				if(isNaN(pts))
					pts = 0;
				else
					pts = pts * multiplier;
				if (pts > 100) jqItem.parent().css("background-color", "#99FF99");
				else if (pts > 0) jqItem.parent().css("background-color", "#CCFFCC");
				else if (pts < -10) jqItem.parent().css("background-color", "#FF9999");
				else if (pts < 0) jqItem.parent().css("background-color", "#FFCCCC");
				jqItem.parents("div.thing:first").data("VoteValue",pts);
			});
		});

		if (!isUserPage) {
			if(settings.sortByVotes){
				// Marks Rating of Posts
				postList.sort(function(a, b) {
					var scoreA = parseInt(jq(a).data("VoteValue"));
					var scoreB = parseInt(jq(b).data("VoteValue"));

					if (scoreA > scoreB)
						return -1;
					else if (scoreA < scoreB)
						return 1;
					else
						return 0;
				});

				postList.detach();
				postList.appendTo(jq("#siteTable"));
			}
		}

		for(var i = 0; i < postList.length; i++){
			var post = jq(postList[i]);
			if(!post.is(":visible"))
				continue;

			var hideBtn = post.find("form.hide-button");
			var eventData = { thingID: post.attr("id") };
			hideBtn.bind("click", eventData, handleHideClick);
		}
	}
	else {
		jq("div.thing:first").find("img, video").each(function() {
			jq(this).attr("style","width: auto !important");
		});
		jq("div.thing:first").find("iframe").each(function() {
			jq(this).attr("style","width: 100% !important");
		});
		var pstID = jq("div.thing:first").attr("id");

		fnHandleShortcut = handleShortcutForSinglePage;
		postList = jq(".sitetable").find("div.thing").not("#" + pstID);

		postList.each(function(index, item) {
			pstItems.push(item.getAttribute("id"));
		});
	}

	jq(document).bind("keydown", fnHandleShortcut);

	jq(document).on("focusin", "input, textarea, select", function(e) {
		jq(document).unbind("keydown", fnHandleShortcut);
	});
	jq(document).on("focusout", "input, textarea, select", function(e) {
		jq(document).bind("keydown", fnHandleShortcut);
	});

	// Removal of sponsor divs
	jq("[class^='sponsor'").remove();

	upVoteLinks = jq(".up:not(.archived)");
	downVoteLinks = jq(".down:not(.archived)");
	rSubRedditLinks = jq("a.subreddit");
	rAuthorLinks = jq("a.author");

	rAuthorLinks.each(function(index, item) {
		var href = item.getAttribute("href");
		if (href.indexOf("/submitted") === -1)
			item.setAttribute("href", href + "/submitted/");
	});

	// Holy crap that new outbound url remapping is annoying af... here let's fix this shit
	jq("a[data-href-url]").each(function() {
		var jqThis = jq(this);
		var currentDataHref = this.getAttribute("data-href-url");
		this.setAttribute("data-outbound-url", currentDataHref);
		this.setAttribute("href", currentDataHref);
	});

	if (settings.replaceTopNew) {
		jq("a.subreddit").each(function(index, item) {
			var hrefLink = item.getAttribute("href");
			if (!hrefLink.match(".*/$")) {
				hrefLink = hrefLink + "/";
			}
			hrefLink = hrefLink + "new";
			item.setAttribute("href", hrefLink);
		});
	}

	jq("span.nsfw-stamp").each(function(index, item) {
		var jqItem = jq(item).parent("li");
		var parent = jqItem.parents("ul.flat-list.buttons");
		if (settings.markNSFW) {
			jqItem.parents("div.entry:first").addClass("NsfwItem");
		}
		jqItem.detach().appendTo(parent);
		jqItem.parents("div.thing:first").addClass("NsfwPost");
	});

	jq("span.spoiler-stamp").each(function(index, item) {
		var jqItem = jq(item).parent("li");
		var parent = jqItem.parents("ul.flat-list.buttons");
		jqItem.detach().appendTo(parent);
		jqItem.parents("div.thing:first").addClass("SpoilerPost");
	});

	// remove Gild and Share button
	jq("li.give-gold-button, li.share").remove();
	jq("li a:contains('promoted')").parent().remove();
	jq("li a:contains('gilded')").parent().remove();
	jq("li a:contains('rising')").parent().remove();
	jq("li a:contains('controversial')").parent().remove();

	// highlight hide/unhide
	var hideSubsFromLocation = false;
	var hideUsersFromLocation = false;

	if (!isMyMulti && !isMySavedPage && !isSinglePost && !isFriendsPage) {
		hideSubsFromLocation = true;
		hideUsersFromLocation = true;
	}

	InitializeContextMenu();

	if (!isSinglePost) {
		jq("a.comments.empty").each(function(index, item) {
			Mark(jq(item), "uncommented");
		});

		jq("span.domain").each(function(index, span) {
			var item = jq(span).find("a");
			var jqItem = jq(item);
			var parentItem = jqItem.parents("div.entry");
			var txt = jqItem.text();

			if (txt.indexOf("youtu") >= 0)
				Mark(parentItem, "video");
			else if (txt.indexOf("imgur") >= 0 || txt.indexOf("gfycat") >= 0)
				Mark(parentItem, "image");
		});

		postList.find("a.title").each(function() {
			var text = this.text;
			if (text.length > 180) {
				var spacerInd = text.substring(0, 175).lastIndexOf(" ");
				this.setAttribute("title", "..." + text.substring(spacerInd));
				this.text = text.substring(0, spacerInd) + "... (more)";
			}
		});

		postList.each(function(index, item) {
			pstItems.push(item.getAttribute("id"));
		});

		for(var i = 0; i < pstItems.length; i++){
			var jqThing = jq("#" + pstItems[i]);
			var tagline = jqThing.find("p.tagline");
			tagline.detach();
			var subReddit = tagline.find("a.subreddit");
			subReddit.detach();
			var first = jqThing.find("div.entry p:first");
			var jqDiv = jq("<span class='thingUserSubInfo'></span>");
			jqDiv.insertBefore(first);
			var flatlistButtons = jqThing.find("ul.flat-list.buttons");
			flatlistButtons.detach();

			tagline.appendTo(jqDiv);
			subReddit.appendTo(jqDiv);
			flatlistButtons.appendTo(jqDiv);


			if (settings.removeDuplicates) {
				var resource = jqThing.find("a.thumbnail").get(0);
				if(resource) {
					if (resourcesLinks.indexOf(resource.href) >= 0) {
						if (!jqThing.hasClass("duplicateHidden"))
							jqThing.addClass("duplicateHidden");

						AddToHideLinks(jqThing);
						duplicateLinks.push(jqThing);
					} else
						resourcesLinks.push(resource.href);
				}
			}
		}

		jq("div.thing.promotedlink").each(function() {
			AddToHideLinks(jq(this));
		});

		if (!isFriendsPage && !isUserSubmittedPage) {
			rAuthorLinks.each(function(index, item) {
				var jqItem = jq(item);
				var userName = jqItem.text().replace("/r/", "").replace("r/","");
				var userIndex = jq.inArray(userName, settings.usersToHide);
				var itemThing = jqItem.parents("div.thing:first");
				if (userIndex >= 0) {
					var hideThisPost = settings.hideUsers &&
						!isUserPage &&
						(settings.hideUsersNSFWonly && jqItem.parents("div.entry").hasClass("NsfwItem") || !settings.hideUsersNSFWonly);

					if (hideThisPost)
						AddToHideLinks(itemThing);

					userHideLinks.push(itemThing);
					itemThing.addClass("userHidden");
				}
			});
		}

		var hideThisPost = settings.hideKeywords && !isUserPage;
		for (var iKW = 0; iKW < settings.keywordsToHide.length; iKW++) {
			var kwItem = settings.keywordsToHide[iKW];
			if (kwItem) {
				jq("a.title:contains(" + kwItem + ")").each(function(jndex, jtem) {
					var thing = jq(jtem).parents("div.thing:first");

					if (hideThisPost)
						AddToHideLinks(thing);

					keywordsHideLinks.push(thing);
					if (!thing.hasClass("keywordsHidden"))
						thing.addClass("keywordsHidden");
				});
			}
		}

		if (loc.indexOf("/r/all") >= 0 || loc.indexOf("/r/popular") >= 0) {
			rSubRedditLinks.each(function(index, item) {
				var subName = item.text.replace("/r/", "").replace("r/","");
				var subIndex = jq.inArray(subName, settings.subsToHide);
				if (subIndex >= 0) {
					var itemThing = jq(item).parents("div.thing:first");
					if (hideSubsFromLocation)
						AddToHideLinks(itemThing);
					subHideLinks.push(itemThing);
					itemThing.addClass("subsHidden");
				}
			});
		}

		jq("#toggleSubsHidden").find("span").text(subHideLinks.length);
		jq("#toggleUserHidden").find("span").text(userHideLinks.length);
		jq("#toggleKeywordsHidden").find("span").text(keywordsHideLinks.length);
		jq("#toggleDuplicateHidden").find("span").text(duplicateLinks.length);

		if(hideLinks.length > 0)
			HideAndAction(hideLinks, 0, null, "enhance");
		else
			InitializeHider();

		var subs = jq("a.subreddit");
		if (subs.length > 0) {
			subs.each(function(index, item) {
				var jqItem = jq(item);
				var subText = jqItem.text().replace("/r/", "").replace("r/","");
				var subColor = subColors[subText];

				if (!subColor) {
					subColor = getRandomColor();
					subColors[subText] = subColor;
				}

				//jqItem.detach().prependTo(eleParent);
				jqItem.text(subText);

				jqItem.parents("div.thing:first").css("background-color", subColor);
			});
		} else {
			jq("span.domain").each(function(index, span) {
				var item = jq(span).children("a");

				if(item){
					var subText = item.text().replace("/r/", "").replace("r/","");
					var subColor = subColors[subText];

					if (!subColor) {
						subColor = getRandomColor();
						subColors[subText] = subColor;
					}

					item.parents("div.thing:first").css("background-color", subColor);
				}
			});
		}

		jq(".last-clicked").removeClass("last-clicked");
		jq(".promotedlink").remove();

		jq("span.author:contains('[deleted]')").parents("div.thing").each(function(){
			jq(this).find("a:contains('hide')").click();
		});

		if (userHideLinks.length === 0) {
			jq("#toggleUserHidden").attr("style", "background-color: #808080 !important; color: #CFCFCF; border: 1px solid black;");
		} else {
			jq("#toggleUserHidden").click(function(e) {
				ToggleVisibility(e, this, ".userHidden");
				return false;
			});
		}
		if (subHideLinks.length === 0) {
			jq("#toggleSubsHidden").attr("style", "background-color: #808080 !important; color: #CFCFCF; border: 1px solid black;");
		} else {
			jq("#toggleSubsHidden").click(function(e) {
				ToggleVisibility(e, this, ".subsHidden");
				return false;
			});
		}
		if (duplicateLinks.length === 0) {
			jq("#toggleDuplicateHidden").attr("style", "background-color: #808080 !important; color: #CFCFCF; border: 1px solid black;");
		} else {
			jq("#toggleDuplicateHidden").click(function(e) {
				ToggleVisibility(e, this, ".duplicateHidden");
				return false;
			});
		}
		if (keywordsHideLinks.length === 0) {
			jq("#toggleKeywordsHidden").attr("style", "background-color: #808080 !important; color: #CFCFCF; border: 1px solid black;");
		} else {
			jq("#toggleKeywordsHidden").click(function(e) {
				ToggleVisibility(e, this, ".keywordsHidden");
				return false;
			});
		}

		var navButton = jq("div.nav-buttons").find(".next-button").find("a:first");
		if (navButton.length === 1) {
			navButton.prepend().appendTo("#postNavigation");
			var navButtonPrev = jq("div.nav-buttons .prev-button a");
			if (navButtonPrev)
				navButtonPrev.prepend().appendTo("#postNavigation");
		}
		jq("div.nav-buttons").remove();
	}

	fExtMessage(undefined);
}
// Initialization - End

// Events - Start
function HideAllClick(e) {
	if(e)
		e.preventDefault();

	if(jq("#hideAll").text().indexOf("?") >= 0)
		return;

	isHiding = true;

	var siteTable = jq("#siteTable");
	siteTable.css("display", "block");
	postList.css("display", "block");
	siteTable.css("height", siteTable.outerHeight(true) + "px");

	siteTable.find("div.thing").each(function(index, thing) {
		var jqThing = jq(thing);
		var offs = jqThing.offset();
		var top = parseInt(offs.top);
		var width = jqThing.width();
		var height = jqThing.height();

		setTimeout(function() {
			jqThing.css({
				top: top,
				left: 0,
				width: width,
				height: height,
				position: "absolute"
			});
		}, settings.animationItemTimeout);
	});


	reloadCancelled = false;
	setTimeout(function() {
		if(pstItems.selected)
			pstItems.jqSelected.animate({ scrollTop: pstItems.jqSelected.offset().top }, 0);

		HideAndAction(hiderElements, 0, null, "reload");
	}, settings.hideItemTimeout);
	jq(this).hide();

	return false;
}

function HideSelectedClick(e) {
	if (pstItems.selected) {
		var selPst = pstItems.selected;
		jq("#nextPost").click();
		Hide(jq("#" + selPst), false, false);
	}
	else
		MakeActivePost(undefined, pstItems.Next());

	if(e)
		e.preventDefault();
	return false;
}

// Kontext-Menü
jq("#fExtContextMenu").on("fExtContextMenuOpening", function(event, actor) {
	var txt = actor.text();
	var jqThing = getThingForCtxActor(actor);
	var usrTxt = 'Select a User';
	var subTxt = 'Select a Subreddit';
	var keywTxt = 'Select word(s)';

	ctxItemUsrHdr.Toggle(false);
	ctxItemUsrHdrPg.Toggle(false);
	ctxItemUsrUnhdr.Toggle(false);

	ctxItemSbHdr.Toggle(false);
	ctxItemSbUnhdr.Toggle(false);
	if(jqThing.length === 1) {
		var usrTxtFnd = jqThing.find(".author:first").text();
		var subTxtFnd = jqThing.find(".subreddit:first").text().replace("/r/","");

		if(usrTxtFnd !== "") {
			usrTxt = usrTxtFnd;
			var usrIndex = jq.inArray(usrTxt, settings.usersToHide);
			ctxItemUsrHdr.Toggle(usrIndex < 0);
			ctxItemUsrHdrPg.Toggle(usrIndex < 0);
			ctxItemUsrUnhdr.Toggle(usrIndex >= 0);
		}
		if(subTxtFnd !== "") {
			subTxt = subTxtFnd;
			var subIndex = jq.inArray(subTxt, settings.subsToHide);

			ctxItemSbHdr.Toggle(subIndex < 0);
			ctxItemSbUnhdr.Toggle(subIndex >= 0);
		}
	}

	var selText = fExtGetSelection();
	var keywIndex = jq.inArray(selText, settings.keywordsToHide);
	bCanHide = selText !== "" && keywIndex < 0;
	bCanUnhide = selText !== "" && keywIndex >= 0;

	ctxItemKeywHdr.Toggle(bCanHide);
	ctxItemKeywUnhdr.Toggle(bCanUnhide);
	ctxItemKeywHdr.ItemText("Hide Keywords - '" + (selText || keywTxt) + "'");
	ctxItemKeywUnhdr.ItemText("Unhide Keywords - '" + (selText || keywTxt) + "'");

	ctxItemUsrHdr.ItemText("Hide User - " + usrTxt);
	ctxItemUsrHdrPg.ItemText("Hide User and open Page - " + usrTxt);
	ctxItemUsrUnhdr.ItemText("Unhide User - " + usrTxt);

	ctxItemSbHdr.ItemText("Hide Sub - " + subTxt);
	ctxItemSbUnhdr.ItemText("Unhide Sub - " + subTxt);

	ctxRedo.Toggle(ctxRedo.find("li.ctxItem").length > 0);
	ctxUndo.Toggle(ctxUndo.find("li.ctxItem").length > 0);
});

if (!isSinglePost && !isSubmitPage) {
	jq(document).on("click", "div.thing, div.thing *", function(e) {
		if(e.target.nodeName === "A" || e.target.nodeName === "LI")
			return true;

		var jqThing;
		if(e.target.nodeName === "DIV" && jq.inArray("thing", e.target.classList) >= 0)
			jqThing = jq(e.target);
		else if(jq(e.target).parents("div.thing:first").length === 1 && (e.target.nodeName === "SPAN" || (e.target.nodeName === "DIV" && jq.inArray("entry", e.target.classList) >= 0)))
			jqThing = jq(e.target).parents("div.thing:first");

		if (jqThing && jqThing.is(":visible") && jqThing.attr("id") != pstItems.selected) {
			return MakeActivePost(pstItems.selected, jqThing.attr("id"));
		}
		else return true;
	});
}

jq("#topPost").click(function() {
	MakeActivePost(pstItems.selected, pstItems.First());
	return false;
});

jq("#botPost").click(function() {
	MakeActivePost(pstItems.selected, pstItems.Last());
	return false;
});

jq("#prevPost").click(function() {
	MakeActivePost(pstItems.selected, pstItems.Previous());
	return false;
});

jq("#nextPost").click(function() {
	MakeActivePost(pstItems.selected, pstItems.Next());
	return false;
});

jq("#frPost").click(function() {
	var pst = pstItems.selected;
	var pstNew = pst;
	pstNew = pstItems.Previous(5);

	MakeActivePost(pst, pstNew);
	return false;
});

jq("#ffPost").click(function() {
	var pst = pstItems.selected;
	var pstNew = pst;

	MakeActivePost(pstItems.selected, pstItems.Next(5));
	return false;
});

// Events - Settings
function SetSettingsValues(settingsObject) {
	for (var key in settingsObject) {
		var varType = typeof settingsObject[key];

		SetSettingValue(key, varType, settingsObject[key]);
	}
}

function SetSettingValue(key, varType, value) {
	var settingsElement = jq("#rEnhSettings").find("[name='" + key + "']");

	switch (varType) {
		case "boolean":
			if (value)
				settingsElement.attr("checked", "true");
			else
				settingsElement.removeAttr("checked");
			break;
		case "number":
			settingsElement.val(value);
			break;
		case "string":
			settingsElement.val(value);
			break;
		case "object":
			if (typeCheck.call(value) === '[object Array]')
				settingsElement.val(value.toString());
			else
				SetSettingsValues(value);
			break;
		default:
			console.log("Cannot get setting " + key + " to settings dialog because unkown type " + varType);
			break;
	}
}

function ShowSettings() {
	SetSettingsValues(settings);

	jq("#rEnhSettings").show();
	return false;
}

function SaveSettingsDialog(e) {
	e.preventDefault();

	jq("#rEnhSettings").find("input, textarea, select").each(function() {
		var val;
		var key = this.name;
		var tag = this.tagName;

		switch (tag) {
			case "INPUT":
				var type = this.type;
				switch (type) {
					case "checkbox":
						val = Boolean(this.checked);
						break;
					case "number":
						val = parseFloat(this.value);
						break;
					case "text":
						val = this.value;
						break;
					case "button":
						break;
					default:
						console.log("unhandled input type " + type + " for key " + key);
						break;
				}
				break;
			case "TEXTAREA":
				val = this.value.split(",");
				break;
			case "SELECT":
				val = parseInt(this.value);
				break;
			default:
				break;
		}

		if (val !== undefined) {
			if (key in settings)
				settings[key] = val;
			else if (key in settings.shortcuts)
				settings.shortcuts[key] = val;
		}
	});
	GMSave();

	fExtPopup("Settings saved!");

	return false;
}
// Events - End

// Functions - Start
function handleShortcutForOverview(e) {
	switch (e.which) {
		case settings.shortcuts.voteUp:
			Vote(pstItems.selected, "up");
			break;
		case settings.shortcuts.previousPost:
			jq("#prevPost").click();
			break;
		case settings.shortcuts.voteDown:
			Vote(pstItems.selected, "down");
			break;
		case settings.shortcuts.nextImage:
			var imgActorNext = jq(".pstSelected").find("div.res-gallery-individual-controls").find("div.res-gallery-next");
			if(imgActorNext.length > 0)
				imgActorNext.get(0).click();
			break;
		case settings.shortcuts.toggleSelected:
			if (pstItems.selected)
				pstItems.jqSelected.find(".expando-button").get(0).click();
			break;
		case settings.shortcuts.previousImage:
			var imgActorPrev = jq(".pstSelected").find("div.res-gallery-individual-controls").find("div.res-gallery-previous");
			if(imgActorPrev.length > 0)
				imgActorPrev.get(0).click();
			break;
		case settings.shortcuts.hideSelected:
			HideSelectedClick();
			break;
		case settings.shortcuts.nextPost:
			jq("#nextPost").click();
			break;
		case settings.shortcuts.hideAll:
			if (reloadCancelled)
				HideAllClick();
			break;
		case settings.shortcuts.zoomIn:
			if (pstItems.selected)
				pstItems.jqSelected.find("img, video, iframe").each(function() { document.fExt.zoomIn(this, 20); });
			break;
		case settings.shortcuts.zoomOut:
			if (pstItems.selected)
				pstItems.jqSelected.find("img, video, iframe").each(function() { document.fExt.zoomOut(this, 20); });
			break;
		case settings.shortcuts.rotateLeft:
			if (pstItems.selected)
				pstItems.jqSelected.find("img, video, iframe").each(function() { document.fExt.rotate(this, -90); });
			break;
		case settings.shortcuts.rotateRight:
			if (pstItems.selected)
				pstItems.jqSelected.find("img, video, iframe").each(function() { document.fExt.rotate(this, 90); });
			break;
		case settings.shortcuts.undoLast:
			if(ctxUndo.Items.length > 0)
				ctxUndo.Items[ctxUndo.Items.length - 1].Trigger();
			break;
		case settings.shortcuts.redoLast:
			if(ctxRedo.Items.length > 0)
				ctxRedo.Items[ctxRedo.Items.length - 1].Trigger();
			break;
		default:
			return true;
	}
	return false;
}

var jqPost = jq("div.thing:first");
var jqPostID = jqPost.attr("id");
function handleShortcutForSinglePage(e) {
	switch (e.which) {
		case settings.shortcuts.voteUp:
			Vote(jqPostID, "up");
			break;
		case settings.shortcuts.previousPost:
			jq("#prevPost").click();
			break;
		case settings.shortcuts.voteDown:
			Vote(jqPostID, "down");
			break;
		case settings.shortcuts.nextImage:
			var imgActorNext = jqPost.find("div.res-gallery-individual-controls").find("div.res-gallery-next");
			if(imgActorNext.length > 0)
				imgActorNext.get(0).click();
			break;
		case settings.shortcuts.toggleSelected:
			pstItems.jqSelected.find("a.expand:first").click();
			break;
		case settings.shortcuts.previousImage:
			var imgActorPrev = jqPost.find("div.res-gallery-individual-controls").find("div.res-gallery-previous");
			if(imgActorPrev.length > 0)
				imgActorPrev.get(0).click();
			break;
		case settings.shortcuts.nextPost:
			jq("#nextPost").click();
			break;
		case settings.shortcuts.zoomIn:
			jqPost.find("img, video, iframe").each(function() { document.fExt.zoomIn(this, 20); });
			break;
		case settings.shortcuts.zoomOut:
			jqPost.find("img, video, iframe").each(function() { document.fExt.zoomOut(this, 20); });
			break;
		case settings.shortcuts.rotateLeft:
			jqPost.find("img, video, iframe").each(function() { document.fExt.rotate(this, -90); });
			break;
		case settings.shortcuts.rotateRight:
			jqPost.find("img, video, iframe").each(function() { document.fExt.rotate(this, 90); });
			break;
		default:
			return true;
	}
	return false;
}

function InitializeElements() {
	var rPostNavigation = jq('<div id="postNavigation"><a href="#" id="topPost">Top</a><a href="#" id="frPost">&lt;&lt;</a><a href="#" id="prevPost">&lt;</a><a href="#" id="nextPost">&gt;</a><a href="#" id="ffPost">&gt;&gt;</a><a href="#" id="botPost">Bottom</a></div>');
	rPostNavigation.appendTo("body");

	var rTabMenu = jq("ul.tabmenu");
	jq('<li class="spacerLi"></li>').appendTo(rTabMenu);

	jq('<li><span class="">All Posts:</span></li>').appendTo(rTabMenu);
	jq('<li><a href="#" class="choice tabButton" title="Updvotes all displayed posts" id="upvoter">Upvote</a></li>').appendTo(rTabMenu);
	jq('<li><a href="#" class="choice tabButton" title="Downvotes all displayed posts" id="downvoter">Downvote</a></li>').appendTo(rTabMenu);

	jq('<li><a href="#" class="choice tabButton collapsed" title="Expands all posts" id="expander">Expand</a></li>').appendTo(rTabMenu);

	jq('<li class="spacerLi"></li>').appendTo(rTabMenu);
	jq('<li><span class="">Hide:</span></li>').appendTo(rTabMenu);
	jq('<li><a href="#" class="choice tabButton" style="background-color: rgb(255, 138, 138) !important;" title="Hides all displayed posts which prevents them from showing up again" id="hideAll">All (<span>?</span>)</a></li>').appendTo(rTabMenu);
	jq('<li><a href="#" class="choice tabButton" style="display: none; background-color: rgb(255, 138, 138) !important;" title="Hides all displayed posts which prevents them from showing up again" id="hideSelected">Selected</a></li>').appendTo(rTabMenu);

	jq('<li class="spacerLi"></li>').appendTo(rTabMenu);
	jq('<li><span class="">Toggle hidden:</span></li>').appendTo(rTabMenu);
	jq('<li><a href="#" class="choice tabButton" style="background-color: rgb(175, 255, 171) !important;" title="Toggles the hidden User posts" id="toggleUserHidden">Users (<span>?</span>)</a></li>').appendTo(rTabMenu);
	jq('<li><a href="#" class="choice tabButton" style="background-color: rgb(175, 255, 171) !important;" title="Toggles the hidden Sub posts" id="toggleSubsHidden">Subs (<span>?</span>)</a></li>').appendTo(rTabMenu);
	jq('<li><a href="#" class="choice tabButton" style="background-color: rgb(175, 255, 171) !important;" title="Toggles the hidden Keyword posts" id="toggleKeywordsHidden">Keywords (<span>?</span>)</a></li>').appendTo(rTabMenu);
	jq('<li><a href="#" class="choice tabButton" style="background-color: rgb(175, 255, 171) !important;" title="Toggles the duplicate posts" id="toggleDuplicateHidden">Duplicates (<span>?</span>)</a></li>').appendTo(rTabMenu);

	jq('<li class="spacerLi"></li>').appendTo(rTabMenu);
	if (loc.indexOf("show=all") < 0) {
		var parmChar = "?";
		if(loc.indexOf("?") >= 0)
			parmChar = "&";
		jq('<li><a href="' + loc.replace('#', '') + parmChar + 'show=all" title="Show all posts of this sub, including previously hidden posts" class="choice tabButton" style="background-color: rgb(187, 187, 255) !important;" id="showAll">Show All</a></li>').appendTo(rTabMenu);
	}
}

function InitializeStyles() {
	fExtCreateStyle("#header { z-index: 107; }");
	fExtCreateStyle("#siteTable { padding: 0 64px; }");
	fExtCreateStyle("body.with-listing-chooser>.content, body.with-listing-chooser .footer-parent { margin-left: 0 !important; }");
	fExtCreateStyle("div.listing-chooser { left: 0 !important; }");
	fExtCreateStyle("div.side { right: 0; }");
	fExtCreateStyle("div.side, div.listing-chooser { z-index: 106 !important; position: fixed !important; height: 90%; top: " + headerHeight + "px !important; margin: 0px; border: 1px solid black; width: 5px !important; overflow-y: hidden !important; padding: 8px !important; }");
	fExtCreateStyle("div.side:hover { width: 500px !important; overflow-y: auto !important; max-height: 100vh; }");
	fExtCreateStyle("div.listing-chooser:hover { width: 500px !important; overflow-y: auto !important; max-height: 100vh; }");
	fExtCreateStyle("span.domain { font-weight: bold; }");

	fExtCreateStyle("div.content { padding-top: 80px  !important; }");
	fExtCreateStyle(".tabButton { color: #333333 !important; background-color: rgb(252, 255, 215) !important; border: 1px solid black; }");
	fExtCreateStyle(".spacerLi { width: 50px; }");
	fExtCreateStyle("#postNavigation { position: fixed !important; top: 27px; width: 500px; margin: 0 auto; display: block; z-index: 108; /*font-size: 12px; font-weight: bold; */ left: 0; right: 0; } ");
	fExtCreateStyle("#postNavigation a { background-color: #ffffff;  border: 1px solid black;  padding: 8px 14px;  margin: 0 1px; }");
	fExtCreateStyle("#postNavigation a:hover { background-color: #dfdfdf; }");
	fExtCreateStyle(".pstSelected, .pstSelected * { z-index: 105; }");

	fExtCreateStyle(".midcol { width: 32px !important; }");
	fExtCreateStyle(".RES-keyNav-activeElement, .RES-keyNav-activeElement .md-container { background-color: transparent !important; }");
	fExtCreateStyle("div.md { border: none !important; }");

	if (isSinglePost) {
		fExtCreateStyle(".livePreview { width: 1600px !important; }");
		fExtCreateStyle(".livePreview p, .livePreview ol { width: 1500px !important; }");
		fExtCreateStyle("div.usertext-edit { max-width: 50%; };");
		fExtCreateStyle("div.usertext-edit div.md { width: 100%; max-width: 100%; }");
		fExtCreateStyle("div.usertext-edit div.md textarea { width: 100%; }");
		fExtCreateStyle(".pstSelected p.tagline { background-color: #DDDDFF; }");
	} else {
		fExtCreateStyle(".md { max-width: none !important; }");
		fExtCreateStyle(".madeVisible *,div.expando * { z-index: 105; }");
		fExtCreateStyle("a.madeVisible img, a.madeVisible video { max-height: 100vh !important; max-width: 95vw !important;}");
		fExtCreateStyle("body { margin-bottom: 650px !important; }");
		fExtCreateStyle("img.RESImage { border-bottom: #000 solid 3px !important; padding: 0 !important; }");
		fExtCreateStyle("div.thing { min-height: 80px; padding: 5px 0; margin: 0 0 8px 0; border: 1px solid black; }");
		fExtCreateStyle("div.thing.pstSelected div.entry>div:not(.expando-button) { background-color: #fff !important; padding: 10px 20px; border: 1px solid black; }");
		fExtCreateStyle("div.thing div.entry img, div.thing div.entry video { display: none !important; }");
		fExtCreateStyle("div.thing.pstSelected div.entry img, div.thing.pstSelected div.entry video { display: block !important; }");
		if (settings.markNSFW) {
			fExtCreateStyle(".NsfwPost { background-color: rgb(255, 81, 81) !important; }");
			fExtCreateStyle(".NsfwItem .madeVisible { display: none; }");
			fExtCreateStyle(".pstSelected .NsfwItem .madeVisible { display: initial !important; }");
			fExtCreateStyle(".pstSelected .NsfwItem .madeVisible div { width: auto !important; }");
			fExtCreateStyle(".pstSelected .NsfwItem * { z-index: 104 !important; }");
		}
		fExtCreateStyle(".entry .buttons li { line-height: inherit !important; width: 100px; }");
		fExtCreateStyle(".thingUserSubInfo { padding: 4px 8px; height: 26px; display: block; }");
		fExtCreateStyle(".thingUserSubInfo>* { float: left; clear: none; border: 1px solid black; margin: 0 2px; }");
		fExtCreateStyle(".thingUserSubInfo * { font-size: 12px !important; }");
		fExtCreateStyle(".thingUserSubInfo .nsfw-stamp { display: inline !important; }");
		fExtCreateStyle(".thingUserSubInfo p.tagline { width: 450px; }");
		fExtCreateStyle(".thingUserSubInfo a.subreddit { width: 160px; }");
		fExtCreateStyle("ul.flat-list.buttons { float: left; clear: none; padding: 0; position: absolute; left: 870px; }");
		fExtCreateStyle("ul.flat-list.buttons>li { cursor: pointer; }");
		fExtCreateStyle("ul.flat-list.buttons>li, .thingUserSubInfo p.tagline, .thingUserSubInfo a.subreddit { padding: 6px !important; background-color: #eeeeee !important;  text-align: center; }");
		fExtCreateStyle("ul.flat-list.buttons>li:hover, .thingUserSubInfo p.tagline:hover, .thingUserSubInfo a.subreddit:hover { background-color: #cee3f8 !important; }");
		//fExtCreateStyle("a.subreddit { float: left; background-color: white !important; color: #888; font-weight: bold; padding: 0 4px !important; }");
		// fExtCreateStyle("p.tagline { float: left; position: relative; background-color: white; color: #888; font-weight: bold; padding: 0 1px; }");
		fExtCreateStyle("p.title { color: #000000 !important; float: left; }");
		fExtCreateStyle("p.title a { color: #000000 !important; }");
		fExtCreateStyle("div.top-matter { height: 48px !important; }");
		//fExtCreateStyle("ul.flat-list.buttons { padding-bottom: 40px; width: 800px; margin: 20px auto 0 auto; font-size: larger; text-align: center; left: 0; right: 0; clear: both; }");
		//fExtCreateStyle("ul.flat-list.buttons li { padding: 4px; background-color: #fff; margin: 2px; border: 1px solid #999; }");
		fExtCreateStyle(".usertext * { z-index: 1; }");
		fExtCreateStyle("div.thing div.entry div.expando:not(.expando-uninitialized) { display: block !important; }");
		fExtCreateStyle("div.thing div.entry div.expando span { display: none; }");
	}
	fExtCreateStyle(".toggleHidden, div.thing.hidden { background: repeating-linear-gradient(45deg, transparent, transparent 35px, rgba(255,255,255,.5) 35px, rgba(255,255,255,.5) 70px); }");

	if(settings.hideUsers)
		fExtCreateStyle(".userHidden .author { text-decoration: line-through; }");
	if(settings.hideSubs)
		fExtCreateStyle(".subsHidden .subreddit { text-decoration: line-through; }");

	fExtCreateStyle(".userHidden a.title, .subsHidden a.title, .duplicateHidden a.title { background-position: 0 !important; background-repeat: no-repeat !important; padding: 8px 8px 8px 26px; }");
	fExtCreateStyle(".userHidden a.title { background-image: url('" + imgDelUser + "') !important; }");
	fExtCreateStyle(".subsHidden a.title { background-image: url('" + imgDelCat + "') !important; }");
	fExtCreateStyle(".duplicateHidden a.title { background-image: url('" + imgDuplicate + "') !important; }");

	fExtCreateStyle("ul.tabmenu .disabled { background-color: #dfdfdf !important; }");

	fExtCreateStyle("#rEnhSettings { position: fixed; height: 750px; top: 50px; border: 1px solid black; width: 750px; padding: 12px; z-index: 1000; background-color: #FEFEFE; right: 0; left: 0; margin: 0 auto; }");
	fExtCreateStyle("#rEnhSettings table { }");
	fExtCreateStyle("#rEnhSettings table td.label, #rEnhSettings table td.value { }");
	fExtCreateStyle("#rEnhSettings table td.label { float: left; clear: right; font-size: larger; }");
	fExtCreateStyle("#rEnhSettings table td.value { float: right; clear: left; }");
	fExtCreateStyle("#rEnhSettings table thead { font-size: large; }");
	fExtCreateStyle("#rEnhSettings div.tableContainer { overflow-y: scroll; height: 700px; }");
	fExtCreateStyle("#rEnhSettings div.settingsButtons { height: 50px;  }");
	fExtCreateStyle("#rEnhSettings * { z-index: 1000; margin: 4px; }");
	fExtCreateStyle("#rEnhSettings td.label {  }");
	fExtCreateStyle("#rEnhSettings a { padding: 4px; border: 1px solid grey; float: right; }");
}

function InitializeHider() {
	if(!hiderElements) {
		hiderElements = [];
		for(var i = 0; i < postList.length; i++){
			var post = jq(postList[i]);
			if(!post.is(":visible"))
				continue;

			hiderElements.push(post);
		}

		if (hiderElements.length === 0)
			jq("#hideAll").attr("style", "background-color: #808080 !important; color: #CFCFCF; border: 1px solid black;");

		var len = jq("form.hide-button").length;
		if(len > 0)
			jq("#hideAll").text("All (" + len + ")");
		else
			jq("#hideAll").text("None");

		jq(".hide-button:contains(hidden)").click(function(e){
			e.preventDefault();

			jq(this).parents("div.thing:first").hide();

			return false;
		});
	}
}

function VoteAndNext(sender, voteLinks, ind) {
	var nextVoteItem = voteLinks[ind];
	jq(nextVoteItem).click();

	ind++;
	if (voteLinks.length > ind) {
		animateControl(sender);

		setTimeout(function() {
			VoteAndNext(sender, voteLinks, ind);
		}, settings.animationItemTimeout);
	} else {
		jq("#downvoter").text("Downvote");
		jq("#upvoter").text("Upvote");
	}
}

function ExpandAndNext(expandLinks, ind) {
	var jqExpander = jq("#expander");
	var expandItem = expandLinks[ind];
	jq(expandItem).click();

	if (expandLinks.length > ind) {
		animateControl(jqExpander);
		setTimeout(function() {
			ExpandAndNext(expandLinks, ind + 1);
		}, settings.animationItemTimeout);
	}
	else  {
		if(jqExpander.hasClass("collapsed")) {
			jqExpander.text("Collapse");
			jqExpander.removeClass("collapsed");
			jqExpander.addClass("expanded");
		}
		else {
			jqExpander.text("Expand");
			jqExpander.addClass("collapsed");
			jqExpander.removeClass("expanded");
		}
	}
}

function animateControl(sender) {
	var txt = jq(sender).text();
	if (txt.length >= 3)
		txt = ".";
	else
		txt = txt + ".";

	jq(sender).text(txt);
}

function HideAndAction(hideLinks, index, previousLink, action) {
	var jqLink = jq(previousLink);
	var item = hideLinks[index];
	if (jqLink.find("form.hide-button").length === 0) {
		if (hideLinks.length > index) {
			var perc = index * 100 / hideLinks.length;
			fExtSetLoading(perc);
			if (action !== "hideObject")
				fExtMessage(action + " (" + index + "/" + hideLinks.length + ") ...");

			Hide(item, action === "reload" || (action == "enhance" && jq(item).find("a.author.friend").length > 0 && duplicateLinks.indexOf(item) < 0));

			var actArgs = [hideLinks, index + 1, previousLink, action];
			ActionWhenElementIsInvisible(item, HideAndAction, actArgs);
		} else {
			isHiding = false;
			fExtSetLoading(100);
			fExtMessage(undefined);
			switch (action) {
				case "reload":
					ReloadLooper(5);
					break;
				case "enhance":
					if (pstItems.onlyVisible().length === 0)
						ReloadLooper(5);
					else
						InitializeHider();
					break;
				case "nextPost":
					if (pstItems.onlyVisible().length === 0)
						ReloadLooper(5);
					else
						jq("#nextPost").click();
					break;
				default:
					break;
			}
		}
	} else setTimeout(function() { HideAndAction(hideLinks, index, previousLink, action); }, settings.hideItemTimeout);
}

function ReloadLooper(loops) {
	if (reloadCancelled) {
		fExtMessage();
		return;
	} else {
		if (loops === 0) {
			fExtMessage("Reloading...");
			setTimeout(function() {
				location.reload();
			}, 1000);
		} else {
			var msg = "Reloading in " + loops + " seconds... (click here to cancel)";
			fExtMessage(msg);
			document.title = msg;
			setTimeout(function() {
				ReloadLooper(loops - 1);
			}, 1000);
		}
	}
}

function Hide(item, displayAfter, handled) {
	var jqHideButton = item.find("form.hide-button");

	if (!isHiding && pstItems.selected === item.attr("id")) {
		MakeActivePost(item.attr("id"), pstItems.Next());
	}

	if(!handled) {
		jqHideButton.unbind("click", handleHideClick);

		jqHideButton.find("a:first").click();

		var args = [item, displayAfter, true];
		AddUndo(Unhide, args, Hide, args, item.find("p.title").text());

		jqHideButton.bind("click", handleHideClick);
	}

	item.hide(settings.animationItemTimeout);

	if (displayAfter) {
		ActionWhenElementIsInvisible(item, item.show, undefined);
	}

	if(pstItems.onlyVisible().length === 0) {
		reloadCancelled = false;
		ReloadLooper(5);
	}

	return true;
}

function Unhide(item, displayAfter, handled) {
	jq(item).show();

	if(!handled){
		var args = [item, displayAfter, true];
		AddUndo(Hide, args, Unhide, args, item.find("p.title").text());
	}

	return true;
}

function Mark(item, itemType) {
	item.addClass("markedItem");
	item.addClass(itemType);
}

function MakeActivePost(oldId, newId) {
	if (oldId === undefined || newId === undefined || oldId !== newId) {
		if (newId) {
			if (newId !== pstItems.selected) {
				pstItems.selected = newId;
				pstItems.jqSelected = jq("#" + newId);
				if (!pstItems.jqSelected.hasClass("pstSelected"))
					pstItems.jqSelected.addClass("pstSelected");

				if (settings.autoExpandSelectedPost) {
					var jsExpander = pstItems.jqSelected.find(".expando-button.collapsed").get(0);
					if (jsExpander)
						jsExpander.click();
				}
			}
		}
		else {
			pstItems.selected = undefined;
			pstItems.jqSelected = undefined;
		}

		if (oldId) {
			var jqOldPost = jq("#" + oldId);
			jqOldPost.removeClass("pstSelected");

			if (settings.autoExpandSelectedPost) {
				var jsCollapser = jqOldPost.find(".expando-button.expanded").get(0);
				if (jsCollapser)
					jsCollapser.click();
			}
		}
	}

	if(pstItems.jqSelected)
		scrollTo(pstItems.jqSelected);

	return false;
}

// Image Preloading - Start
var preloadIndex = 0;
var preloadComplete = 0;
var preloadingImages = jq("a.title[href*='imgur'],a.title[href*='gfycat']").not(".thumbnail");
for(var i = 0; i < preloadingImages.length; i++){
	var img = preloadingImages.get(i);

	if (img.href.indexOf("https://") < 0) {
		img.href = img.href.replace("http://", "https://");
	}
}

function preloadSuccess(e){
	console.log("Preloading image " + preloadIndex + " successful!");
	preload();
}

function preloadError(e){
	console.log("Error preloading image " + preloadIndex);
	preload();
}

var preloadingImage = document.createElement('IMG');
preloadingImage.style.display = "block";
preloadingImage.style.position = "absolute";
preloadingImage.style.top = 0;
preloadingImage.style.left = 0;
preloadingImage.src= "about:blank";
preloadingImage.width=1;
preloadingImage.height=1;
preloadingImage.onerror = preloadError;
preloadingImage.onload = preloadSuccess;
document.body.appendChild(preloadingImage);

function preload() {
	if(preloadIndex >= preloadingImages.length) {
		preloadingImage.remove();
		return;
	}

	var href = preloadingImages.get(preloadIndex).href;
	preloadIndex++;

	console.log("Preloading Image " + preloadIndex + " of " + preloadingImages.length + "(" + href + ")...");
	preloadingImage.src = href;
}

setTimeout(function(){ preload(); }, 1000);
// Image preloading - End

function scrollTo(item) {
	var jqItem = jq(item);
	var offs = jqItem.offset();
	var topPos = offs.top - headerHeight;

	jq(document.body).scrollTop(topPos);
	//jq(document.body).animate({ scrollTop: topPos }, settings.animationItemTimeout);
}

function AddToHideLinks(elm) {
	if (!elm.hasClass("toggleHidden"))
		elm.addClass("toggleHidden");

	if (jq.inArray(elm, hideLinks) < 0)
		hideLinks.push(elm);
}

function Vote(id, direction) {
	if (id !== undefined) {
		var element;

		switch (direction) {
			case "up":
				element = jq("#" + id).find(".arrow.up, .arrow.upmod");
				break;
			case "down":
				element = jq("#" + id).find(".arrow.down, .arrow.downmod");
				break;
			default:
				break;
		}

		if (element.length > 0) {
			element.click();
		}
	}
}

function AddUndo(fnUndo, argsUndo, fnRedo, argsRedo, suffix) {
	if (ctxUndo.Items.length === 10) {
		var removeThis = ctxUndo.Items[0];
		ctxUndo.Items.splice(0, 1);
		removeThis.remove();
	}
	if(!suffix)
		suffix = "";
	else if(suffix.length > 25)
		suffix = suffix.substring(0, 25) + "...";

	var ctxItem = fExtAddCtxItem(suffix && suffix.length > 0 ? fnRedo.name + " - " + suffix : fnRedo.name, ctxUndo);
	ctxItem.ClickCloses = false;
	ctxItem.fnUndo = fnUndo;
	ctxItem.argsUndo = argsUndo;
	ctxItem.fnRedo = fnRedo;
	ctxItem.argsRedo = argsRedo;
	ctxItem.Action = function(event, sender, actor) {
		sender.fnUndo.apply(this, sender.argsUndo);
		AddRedo(sender.fnUndo, sender.argsUndo, sender.fnRedo, sender.argsRedo, suffix);
		sender.Remove();
	};
}

function AddRedo(fnUndo, argsUndo, fnRedo, argsRedo, suffix) {
	if (ctxRedo.Items.length === 10) {
		ctxRedo.Items[0].remove();
		ctxRedo.Items.splice(0, 1);
	}
	if(!suffix)
		suffix = "";
	else if(suffix.length > 25)
		suffix = suffix.substring(0, 25) + "...";

	var ctxItem = fExtAddCtxItem(suffix && suffix.length > 0 ? fnUndo.name + " - " + suffix : fnUndo.name, ctxRedo);
	ctxItem.ClickCloses = false;
	ctxItem.fnUndo = fnUndo;
	ctxItem.argsUndo = argsUndo;
	ctxItem.fnRedo = fnRedo;
	ctxItem.argsRedo = argsRedo;
	ctxItem.Action = function(event, sender, actor) {
		sender.fnRedo.apply(this, sender.argsRedo);
		AddUndo(sender.fnUndo, sender.argsUndo, sender.fnRedo, sender.argsRedo, suffix);
		sender.remove();
	};
}

function ToggleVisibility(event, sender, className) {
	event.preventDefault();
	if (jq(sender).data("state") === "shown") {
		jq(sender).data("state","hidden");
		jq(className).hide();
	} else {
		jq(sender).data("state","shown");
		jq(className).show();
	}
}

function HideObject(containingArray, arrayName, text, selector) {
	if (jq.inArray(text, containingArray) < 0) {
		fExtPopup("Adding '" + text + "' from '" + arrayName + "' to hide.");
		containingArray.push(text);
		GMSave();

		var links = [];
		var selectNext = false;

		jq(selector).parents("div.thing").each(function(index, item) {
			var jqItem = jq(item);
			selectNext = selectNext || item.id === pstItems.selected;

			if (jqItem.find("form.hide-button").length > 0)
				links.push(jqItem);
		});

		HideAndAction(links, 0, null, "hideObject");

		return true;
	}
	return false;
}

function UnhideObject(containingArray, arrayName, text, selector) {
	var arrayIndex = jq.inArray(text, containingArray);
	if (arrayIndex >= 0) {
		fExtPopup("Removing '" + text + "' from '" + arrayName + "'.");
		containingArray.splice(arrayIndex, 1);
		GMSave();
		return true;
	}
	return false;
}

function GrabContent(htmlContent) {
	fExtMessage("Loading additional Pages (" + numberOfSitesLoaded + " of max " + settings.maximumNumberOfSitesToLoad + ")");

	var navButtonNext = jq(htmlContent).find("div.nav-buttons").find(".next-button").find("a").get(0);
	if (numberOfSitesLoaded < settings.maximumNumberOfSitesToLoad && navButtonNext) {
		numberOfSitesLoaded += 1;
		var srcUrl = navButtonNext.href;
		//siteTable
		jq.ajax({
			url: srcUrl,
			method: "get",
			success: function(data) {
				jq(data).find("div#siteTable").find("div.thing").each(function() {
					jq(this).detach().appendTo("div#siteTable");
				});

				GrabContent(data);
			},
			error: function(data) {
				console.log(data);
			}
		});
	} else {
		if (navButtonNext)
			jq(navButtonNext).detach().appendTo("#postNavigation");

		fExtMessage(numberOfSitesLoaded + " Pages loaded, starting initialization...");

		MainInitialization();
	}
}

function InitializeContextMenu() {
	ctxItemUsrHdr = fExtAddCtxItem("Hide User", rEnhSub);
	ctxItemUsrHdr.Action = function(event, sender, actor) {
		var jqThing = getThingForCtxActor(actor);
		var user = jqThing.find(".author:first").text();
		var selector = "a.author:contains(" + user + ")";
		var args = [settings.usersToHide, "Users", user, selector];
		if (HideObject.apply(sender, args))
			AddUndo(UnhideObject, args, HideObject, args, user);
	};

	ctxItemUsrHdrPg = fExtAddCtxItem("Hide User and open Userpage", rEnhSub);
	ctxItemUsrHdrPg.Action = function(event, sender, actor) {
		ctxItemUsrHdr.Action(event, sender, actor);
		var jqThing = getThingForCtxActor(actor);
		var user = jqThing.find(".author:first").text();
		GM_openInTab("https://www.reddit.com/u/" + user + "/submitted/", true);
	};

	ctxItemUsrUnhdr = fExtAddCtxItem("Unhide User", rEnhSub);
	ctxItemUsrUnhdr.Action = function(event, sender, actor) {
		var jqThing = getThingForCtxActor(actor);
		var user = jqThing.find(".author:first").text();
		var selector = "a.author:contains(" + user + ")";
		var args = [settings.usersToHide, "Users", user, selector];
		if (UnhideObject.apply(sender, args)) {
			jq("a.author:contains(" + user + ")").parents("div.thing").each(function(index, item) {
				jq(item).show();
			});
			AddUndo(HideObject, args, UnhideObject, args, user);
		}
	};

	fExtAddCtxSeparator(rEnhSub);

	ctxItemSbHdr = fExtAddCtxItem("Hide Sub", rEnhSub);
	ctxItemSbHdr.Action = function(event, sender, actor) {
		var jqThing = getThingForCtxActor(actor);
		var sub = jqThing.find(".subreddit:first").text().replace("/r/", "").replace("r/","");
		var selector = "a.subreddit:contains(" + sub + ")";
		var args = [settings.subsToHide, "Subs", sub, selector];
		if (HideObject.apply(sender, args))
			AddUndo(UnhideObject, args, HideObject, args, sub);
	};

	ctxItemSbUnhdr = fExtAddCtxItem("Unhide Sub", rEnhSub);
	ctxItemSbUnhdr.Action = function(event, sender, actor) {
		var jqThing = getThingForCtxActor(actor);
		var sub = jqThing.find(".subreddit:first").text().replace("/r/", "").replace("r/","");
		var selector = "a.subreddit:contains(" + sub + ")";
		var args = [settings.subsToHide, "Subs", sub, selector];
		if (UnhideObject.apply(sender, args)) {
			jq(selector).parents("div.thing").each(function(index, item) {
				jq(item).show();
			});
			AddUndo(HideObject, args, UnhideObject, args, sub);
		}
	};

	fExtAddCtxSeparator(rEnhSub);

	ctxItemKeywHdr = fExtAddCtxItem("Hide Keywords", rEnhSub);
	ctxItemKeywHdr.Action = function(event, sender, actor) {
		var keyword = fExtGetSelection().trim();
		var selector = "a.title:contains(" + keyword + ")";
		var args = [settings.keywordsToHide, "Keywords", keyword, selector];
		if (HideObject.apply(sender, args))
			AddUndo(UnhideObject, args, HideObject, args, keyword);
	};

	ctxItemKeywUnhdr = fExtAddCtxItem("Unhide Keywords", rEnhSub);
	ctxItemKeywUnhdr.Action = function(event, sender, actor) {
		var keyword = fExtGetSelection();
		var selector = "a.title:contains(" + keyword + ")";
		var args = [settings.keywordsToHide, "Keywords", keyword, selector];
		if (UnhideObject.apply(sender, args)) {
			jq(selector).parents("div.thing").each(function(index, item) {
				jq(item).show();
			});
			AddUndo(HideObject, args, UnhideObject, args, keyword);
		}
	};

	fExtAddCtxSeparator(rEnhSub);

	// Undo/Redo
	ctxUndo = fExtAddSub("Undo", rEnhSub);
	ctxRedo = fExtAddSub("Redo", rEnhSub);

	fExtAddCtxSeparator(rEnhSub);

	ctxItemCpyHtml = fExtAddCtxItem("Copy HTML", rEnhSub);
	ctxItemCpyHtml.Action = function(event, sender, actor) {
		fExtClipboard("Copy", jq('div.content').html());
	};

	fExtAddCtxSeparator(rEnhSub);

	ctxItemSett = fExtAddCtxItem("REnhancer Settings", rEnhSub);
	ctxItemSett.Action = function() { ShowSettings(); };
}

function InitializeEventHandlers() {
	jq("#fExtMessage").click(function() { reloadCancelled = true; });
	jq("#hideAll").click(HideAllClick);
	jq("#hideSelected").click(HideSelectedClick);
	jq("#expander").click(function(e) {
		e.preventDefault();
		if(jq("#expander").hasClass("expanded"))
			ExpandAndNext(jq("div.expando-button.expanded:visible"), 0);
		else
			ExpandAndNext(jq("div.expando-button.collapsed:visible"), 0);
		return false;
	});
	jq("#renhSettingsSave").click(SaveSettingsDialog);
	jq("#upvoter").click(function(e) {
		e.preventDefault();
		VoteAndNext(jq(this), upVoteLinks, 0);
		return false;
	});
	jq("#downvoter").click(function(e) {
		e.preventDefault();
		VoteAndNext(jq(this), downVoteLinks, 0);
		return false;
	});
	jq("#renhSettingsClose").click(function(e) {
		jq("#rEnhSettings").hide();
		return false;
	});

	jq("ul.flat-list.buttons>li:not(.report-button)").click(function(e){
		if(e.target.tagName !== "A" && e.target.tagName !== "FORM") {
			var targetItem = jq(e.target).find("a").first().get(0);
			if(targetItem) targetItem.click(e);
			else return true;
		}
		else return true;
	});
}

function handleHideClick(e){
	var jqThing;
	if (e.data)
		jqThing = jq("#" + e.data.thingID);
	else
		jqThing = jq(this).parents("div.thing:first");

	var args = [jqThing, false, true];
	AddUndo(Unhide, args, Hide, args, jqThing.find("p.title").text());

	Hide(jqThing, undefined, true);
}

function ActionWhenElementIsInvisible(element, action, args, step) {
	if (element.hasClass("hidden") || element.find("a[text=hidden]").length === 1 || step === 3)
		action.apply(element, args);
	else {
		if(isNaN(step))
			step = 0;

		step++;

		setTimeout(function() {
			ActionWhenElementIsInvisible(element, action, args, step);
		}, settings.hideItemTimeout);
	}
}

// Functions - Load and Save
function GMLoadValue(valueName) {
	var ret = GM_getValue(valueName);
	if (ret !== undefined)
		return ret.toString().split(",");

	return [];
}

function GMSave() {
	GM_setValue("settings", JSON.stringify(settings));
}

function GMLoad() {
	settings = { // Defaults
		accountName: "enter your account name here, e.g. throwaway001",
		markNSFW: true,
		replaceTopNew: false,
		sortByVotes: false,
		autoExpandSelectedPost: true,
		removeDuplicates: true,
		maximumNumberOfSitesToLoad: 1,

		animationItemTimeout: 500,
		imagePreloadingTimeout: 250,
		hideItemTimeout: 750,

		hideUsers: true,
		hideUsersNSFWonly: true,
		usersToHide: undefined,

		hideSubs: true,
		subsToHide: undefined,

		hideKeywords: true,
		keywordsToHide: undefined,

		shortcuts: {
			nextPost: 98,
			previousPost: 104,
			toggleSelected: 101,
			hideSelected: 99,

			hideAll: 97,

			voteUp: 105,
			voteDown: 103,

			nextImage: 102,
			previousImage: 100,
			zoomIn: 107,
			zoomOut: 109,
			rotateLeft: 111,
			rotateRight: 106,

			undoLast: -1,
			redoLast: -1,
		},

		panelView: false,
	};

	var lSettings = GM_getValue("settings");
	if (lSettings !== undefined) {
		settingsFromLoad(settings, JSON.parse(lSettings.toString()));
	}

	if (!settings.usersToHide)
		settings.usersToHide = GMLoadValue("usersToHide");
	if (!settings.subsToHide)
		settings.subsToHide = GMLoadValue("subsToHide");
	if (!settings.keywordsToHide)
		settings.keywordsToHide = GMLoadValue("keywordsToHide");
}

function settingsFromLoad(setSettings, getSettings){
	for (var k in getSettings) {
		var varType = typeof getSettings[k];
		var varTypeB = typeof setSettings[k];
		switch (varType) {
			case "boolean":
				setSettings[k] = Boolean(getSettings[k]);
				break;
			case "number":
				setSettings[k] = parseInt(getSettings[k]);
				break;
			case "object":
				if(Array.isArray(getSettings[k]))
					setSettings[k] = getSettings[k];
				else
					settingsFromLoad(setSettings[k], getSettings[k]);
				break;
			case "string":
			default:
				setSettings[k] = getSettings[k];
				break;
		}
	}
}

// Functions - Misc
function getThingForCtxActor(actor){
	return jq(actor).hasClass("thing") ? jq(actor) : jq(actor).parents("div.thing:first");
}

function grabExtension(src) {
	var start, end;

	start = src.lastIndexOf(".") + 1;
	if (src.lastIndexOf("?") >= 0)
		end = src.lastIndexOf("?");
	else
		end = src.length;

	return src.substring(start, end);
}

function colorObject(r, g, b, a){
	var rv = {
		red: 0,
		blue: 0,
		green: 0,
		alpha: 100
	};

	if(r)
		rv.red = parseInt(r);
	if(g)
		rv.green = parseInt(g);
	if(b)
		rv.blue = parseInt(b);
	if(a)
		rv.alpha = parseInt(a);

	rv.getRgba = function(){
		return "rgba(" + this.red + "," + this.green + "," + this.blue + ", " + (this.alpha / 100) + ")";
	};
	rv.getHex = function(){
		var strR = this.red.toString(16);
		var strG = this.blue.toString(16);
		var strB = this.green.toString(16);
		if(strR.length === 1) strR = "0" + strR;
		if(strG.length === 1) strG = "0" + strG;
		if(strB.length === 1) strB = "0" + strB;

		return "#" + strR + strG + strB;
	};

	return rv;
}

function getRandomColor() {
	var color = '#';

	// Neu rgb-style
	var min = 155, max = 255;
	var cO = colorObject(Math.random() * (max - min) + min, Math.random() * (max - min) + min, Math.random() * (max - min) + min);

	if(cO.red > cO.green && cO.red > cO.blue)
		color = getRandomColor();
	else
		color = cO.getHex();

	return color;
}
// Functions - End