Reddit Enhancer

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name        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