Tobias.Kelmandia / Tsumino Enhanced

"use strict";

// ==UserScript==
// @name				Tsumino Enhanced
// @namespace			http://codingtoby.com
// @version				2.0.3.11
// @description			Adds a collection of customizable tweaks, enhancements, and new features to Tsumino.com.
// @author				Toby
// @include				/((http)(s)?(\:\/\/)(www\.)?(tsumino\.com)(\/)?([\s\S]*))/
// @exclude				/((http)(s)?(\:\/\/)(www\.)?(tsumino\.com)(\/)?(Read\/AuthProcess)/
// @exclude				http://www.tsumino.com/bhome/*
// @require				https://ajax.googleapis.com/ajax/libs/jquery/2.1.4/jquery.min.js
// @require				http://js.codingtoby.com/semantic.min.js
// @require				https://cdnjs.cloudflare.com/ajax/libs/bean/1.0.15/bean.min.js
// @require				https://cdnjs.cloudflare.com/ajax/libs/velocity/1.2.3/velocity.min.js
// @grant				GM_setValue
// @grant				GM_getValue
// @grant				GM_deleteValue
// @grant				unsafeWindow
// @grant				GM_openInTab
// @grant				GM_xmlhttpRequest
// @run-at				document-start
// ==/UserScript==


/*************************************************************************************
 * Open source libraries.
 *************************************************************************************/

/*************************************************************************************
 * This one adds arrayBuffer support to jQuery's ajax method.
 * -------------------------
 * jquery.binarytransport.js
 * @description. jQuery ajax transport for making binary data type requests.
 * @version 1.0
 * @author Henry Algus <henryalgus@gmail.com>
 *************************************************************************************/

// use this transport for "binary" data type
$.ajaxTransport( "+binary", function (options, originalOptions, jqXHR)
{
	if ( window.FormData &&
		 (options.dataType && options.dataType == "binary"
		  || options.data && (window.ArrayBuffer && options.data instanceof ArrayBuffer
							  || window.Blob && options.data instanceof Blob)) )
	{
		return {
			send  : function (headers, callback)
			{
				var xhr                                             = new XMLHttpRequest,
					url                                             = options.url,
					type                                            = options.type,
					async                                           = options.async || true,
					dataType = options.responseType || "blob", data = options.data || null,
					username                                        = options.username || null,
					password                                        = options.password || null;
				xhr.addEventListener( "load", function ()
				{
					var data                 = {};
					data[ options.dataType ] = xhr.response;
					callback( xhr.status, xhr.statusText, data, xhr.getAllResponseHeaders() )
				} );
				xhr.open( type, url, async, username, password );
				for (var i in headers)
				{
					xhr.setRequestHeader( i, headers[ i ] );
				}
				xhr.responseType = dataType;
				xhr.send( data )
			},
			abort : function () {jqXHR.abort()}
		}
	}
} );


/*************************************************************************************
 * Tsumino Enhanced
 *************************************************************************************/

// Establish Tsumino Enhanced
(function (w, $)
{
	// Main object - Metadata
	var TE                = {
		name           : GM_info[ "script" ][ "name" ],
		version        : GM_info[ "script" ][ "version" ],
		status         : {},
		updateLocation : "https://openuserjs.org/scripts/Tobias.Kelmandia/Tsumino_Enhanced",
		installLocation: "https://openuserjs.org/install/Tobias.Kelmandia/Tsumino_Enhanced.user.js"
	};
	TE.status.pagesLoaded = {};

	// Tsumino Enhanced Configuration
	TE.config = {
		debug        : true,
		verboseDebug : true,
		pfRange      : 5,
		preload      : true
	};

	// User's current location.
	TE.myLocation = w.location.href;


	/*************************************************************************************
	 * User Configuration
	 *************************************************************************************/
	TE.User = {};
	(function ()
	{
		if ( GM_getValue( "TE_settings" ) )
		{
			var TE_settings = GM_getValue( "TE_settings" );
			TE.User         = JSON.parse( TE_settings );
		}
	})();

	/*************************************************************************************
	 * Detect which features the user's browser has.
	 *************************************************************************************/
	TE.ft = {};
	TE.ft.logGroups = typeof w.console.group === 'function';


	/*************************************************************************************
	 * Tsumino Site Configuration.
	 *************************************************************************************/

		// Define prefixes for all major site pages.
	TE.site = {
		account    : {
			prefix : "/Account/Home",
			regex  : "(\/Account\/Home*)"
		},
		auth       : {
			prefix : "/Read/Auth/",
			regex  : "(\/Read\/Auth\/[\\s\\S]*)"
		},
		baseURL    : {root : TE.myLocation.split( ".com" )[ 0 ] + ".com"},
		book       : {
			prefix : "/Book/Info/",
			regex  : "(\/Book\/Info\/[\\s\\S]*)"
		},
		browse     : {
			prefix : "/Browse/Index/",
			regex  : "(\/Browse\/[\\s\\S]*)||(\/*)"
		},
		error      : {
			prefix : "/Error/Index/",
			regex  : "(\/Error\/Index\/[\\s\\S]*)"
		},
		image      : {
			prefix : "/Image/Object/?data=",
			regex  : "(\/Image\/Object\/\\?data=[\\s\\S]*)"
		},
		login      : {
			prefix : "/Account/Login",
			regex  : "(\/Account\/Login[\\s\\S]*)"
		},
		manageTags : {
			prefix : "/Account/ManageTags",
			regex  : "(\/Account\/ManageTags[\\s\\S]*)"
		},
		reader     : {
			prefix : "/Read/View/",
			regex  : "(\/Read\/View\/[\\s\\S]*)"
		},
		search     : {
			prefix : "/Search",
			regex  : "(\/Search[\\s\\S]*)"
		},
		forum      : {
			prefix : "/Forum",
			regex  : "(\/Forum[\\s\\S]*)"
		}
	};

	var temp              = TE.site.baseURL.root;
	TE.site.baseURL.regex = temp.replace( /([.*+?\^=!:${}()\|\[\]\/\\])/g, "\\$1" );

	// Location Checking object.
	TE.on = {};

	// Create full URLs and do location checking.
	for (var key in TE.site)
	{
		if ( TE.site.hasOwnProperty( key ) )
		{
			var obj = TE.site[ key ];
			if ( obj.hasOwnProperty( "prefix" ) )
			{
				// Create Full URLs.
				obj[ "url" ] = TE.site.baseURL.root + obj[ "prefix" ];

				// Perform location checking.
				if ( obj[ "prefix" ] )
				{
					TE.on[ key ] = !!RegExp( "^(" + TE.site.baseURL.regex + ")" + obj[ "regex" ] + "$" ).test( TE.myLocation );
				}
			}
		}
	}

	TE.on[ "tsumino" ] = !!RegExp( "^(" + TE.site.baseURL.regex + ")([\\s\\S]*)$" ).test( TE.myLocation );

	// Prepare prefetch.
	TE.status.prefetch = {};

	TE.status.numRandStrings = 0;

	/*************************************************************************************
	 * Utility Functions
	 *************************************************************************************/
	TE.fn = TE.prototype = {
		// Logging to console with Timestamps.
		randomString : function()
		{
			TE.status.numRandStrings++;
			var text = TE.status.numRandStrings + "_";
			var possible = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-";

			var baseline = 9;
			var ceiling = (Math.floor(Math.random() * 10));
			var numChars = baseline + ceiling;

			for( var i=0; i < numChars; i++ )
			{
				text += possible.charAt(Math.floor(Math.random() * possible.length));
			}
			return text;
		},
		log             : function ()
		{
			if ( (arguments.length > 0) && (TE.config.debug) )
			{
				var date  = new Date(), hr, min, sec, mil, timeStamp;
				hr        = date.getHours();
				min       = date.getMinutes();
				sec       = date.getSeconds();
				mil       = date.getMilliseconds();
				timeStamp = hr + ":" + min + ":" + sec + ":" + mil;
				if ( TE.ft.logGroups )
				{
					if ( arguments[ 0 ] == "gname" )
					{
						console.group( arguments[ 1 ] + " - [" + timeStamp + "]" );
						for (var i = 2 ; i < arguments.length ; ++i)
						{
							console.log( arguments[ i ] );
						}
					}
					else
					{
						console.group( timeStamp );
						for (var i = 0 ; i < arguments.length ; ++i)
						{
							console.log( arguments[ i ] );
						}
					}
					console.groupEnd();
				}
				else
				{
					if ( arguments[ 0 ] == "gname" )
					{
						console.log( "----- " + arguments[ 1 ] + " -----" );
						for (var i = 0 ; i < arguments.length ; ++i)
						{
							console.log( arguments[ i ] );
						}
					}
					else
					{
						console.log( "[" + timeStamp + "]" );
						for (var i = 0 ; i < arguments.length ; ++i)
						{
							console.log( arguments[ i ] );
						}
					}
					console.log();
				}
			}
		},
		vbLog           : function ()
		{
			if ( TE.config.verboseDebug )
			{
				if ( arguments.length > 0 )
				{
					this.log.apply( this.log, arguments );
				}
			}
		},
		errorMsg        : function (code, situation, error)
		{
			this.log( "gname", TE.name, "An error was detected while:", situation, "Error Code: " + code, error );
		},
		replaceAll      : function (str, find, replace)
		{
			// Escape regex.
			find = find.replace( /([.*+?^=!:${}()|\[\]\/\\])/g, "\\$1" );
			return str.replace( new RegExp( find, 'g' ), replace );
		},
		load            : function (pageNumber, imageUrl)
		{
			var dfd = jQuery.Deferred(), authUrl = TE.site.baseURL.root + TE.site.auth.prefix + TE.book.id + "/" + pageNumber;

			if ( (TE.status.pagesLoaded[ pageNumber ] != "working") && (TE.status.pagesLoaded[ pageNumber ] != "done") )
			{
				TE.status.pagesLoaded[ pageNumber ] = "working";
				$( "#te_readerMessageDisplay" ).append( `
				<div id="te_loading_message" class="ui segment"><br />
				<div class="ui active dimmer"><div class="ui text loader">Loading...</div></div><br /></div>
				` );

				this.vbLog( "gname", "TE.load", "Loading Image: " + pageNumber + "...", TE.site.baseURL.root + imageUrl );
				var downloadStart = new Date();
				// Make an ajax request expecting a binary (arraybuffer) datatype.
				var loadImage = $.ajax( {
					method           : "GET",
					url              : imageUrl,
					dataType         : "binary",
					processData      : false,
					responseType     : 'arraybuffer',
					success          : $.proxy( function (data, textStatus, request)
					{
						// Put the response headers into an array.
						var rh = loadImage.getAllResponseHeaders(), rha = rh.split( "\r\n" );

						// Create a proper object from the response header array.
						var responseHeader = {};
						for (var i = 0 ; i < rha.length ; i++)
						{
							var thisRH = rha[ i ];
							thisRH     = thisRH.split( ": " );
							if ( thisRH[ 0 ] != "" )
							{
								responseHeader[ thisRH[ 0 ] ] = thisRH[ 1 ];
							}
						}

						// Local logging to examine response headers.
						TE.vbLog("gname","TE.fn.load","Response Headers",responseHeader);

						// Content-Type is undefined if Tsumino requires us to solve a captcha.
						if ( typeof responseHeader[ "Content-Type" ] === "undefined" )
						{
							// Redirect to the auth page.
							w.location.href = authUrl;
						}
						else
						{
							var downloadComplete = new Date();
							TE.vbLog( "gname", "TE.load", "Content Type: " + responseHeader[ "Content-Type" ] );

							// If we're dealing with a JPEG image.
							// (Why is it 'images/jpeg' instead of 'image/jpeg'? Typo by Tsumino devs?)
							if ( responseHeader[ "Content-Type" ] == "images/jpeg" )
							{
								TE.vbLog( "gname", "TE.load", "Image data loaded.", "Running conversions..." );
								var startTime = new Date();

								// Use Uint8Array to view the arrayBuffer response data.
								var typedArray = new Uint8Array( data );

								// Determine number of bytes for the assembly loop.
								var numBytes     = typedArray.length;
								var binaryString = "";

								// Convert it into a useable binary string.
								for (i = 0 ; i < numBytes ; i++)
								{
									binaryString = binaryString + String.fromCharCode( typedArray[ i ] );
								}

								// And finally encode the binary string as base64.
								var encodedBS = btoa( binaryString );

								var endTime     = new Date();
								var dlTime      = downloadComplete - downloadStart;
								var runTime     = endTime - startTime;
								var dlTimeDISP  = dlTime + "ms";
								var runTimeDISP = runTime + "ms";
								if ( dlTime >= 1000 )
								{ dlTimeDISP = (dlTime / 1000) + "s"; }
								if ( runTime >= 1000 )
								{ runTimeDISP = (runTime / 1000) + "s"; }
								TE.vbLog( "gname", "TE.load", "Conversions completed.",
									"Image downloaded in: " + dlTimeDISP + ".", "Total time spent on conversion: " + runTimeDISP + "." );

								// Take the base64 string and prepend it so it can be used as a dataURI.
								var dataURI = "data:image/jpeg;base64," + encodedBS;

								// Add a hidden image to the page so the dataURI can be harvested from its source later.
								$( "body" ).append( "<img id='te_loadImage_" + pageNumber + "' src='" + dataURI + "' style='display:none;'>" );

								// And we're done.
								this.vbLog( "gname", "TE.load", "Image " + pageNumber + " loaded." );
								$( "#te_loading_message" ).remove();
								TE.status.pagesLoaded[ pageNumber ] = "done";
								dfd.resolve();
							}
						}
					}, this ), error : $.proxy( function (request, status, error)
					{
						this.log( "gname", "TE.load", "Error retrieving image.", request, status, error );
					}, this )
				} );
			}
			else if ( TE.status.pagesLoaded[ pageNumber ] == "working" )
			{
				var checkAgain = function ()
				{
					setTimeout( function ()
					{
						if ( TE.status.pagesLoaded[ pageNumber ] == "done" )
						{
							dfd.resolve();
						}
						else
						{
							checkAgain();
						}
					}, 500 );
				};

				checkAgain();
			}
			else if ( TE.status.pagesLoaded[ pageNumber ] == "done" )
			{
				dfd.resolve();
			}
			return dfd.promise();
		},
		prefetch        : {
			init : function (pageNumber)
			{
				var dfd = jQuery.Deferred();
				TE.vbLog( "gname", "Prefetch Init", "Initializing..." );
				var pfRange = TE.config.pfRange, pfStart = (pageNumber - pfRange), pfEnd = (pageNumber + pfRange);
				if ( pfStart < 1 )
				{
					pfStart = 1;
				}
				if ( pfEnd > TE.book.totalPages )
				{
					pfEnd = TE.book.totalPages;
				}
				var thisRange = pfEnd - pfStart;
				if ( thisRange == 1 )
				{
					thisRange++;
				}

				var timestamp                   = new Date().getTime();
				TE.status.prefetch[ timestamp ] = 0;

				TE.vbLog( "gname", "Prefetch Init", "Base: " + pageNumber, "Start: " + pfStart, "End: " + pfEnd );
				for (var i = pfStart ; i <= pfEnd ; i++)
				{
					if ( TE.status.prefetch[ TE.book.id ][ i ] == "" )
					{
						$.when( this.get( i ) ).then( function ()
						{
							TE.status.prefetch[ timestamp ]++;
							if ( TE.status.prefetch[ timestamp ] == thisRange )
							{
								dfd.resolve();
							}
						} );
					}
				}
				return dfd.promise();
			},
			get  : function (pageNumber)
			{
				TE.status.prefetch[ TE.book.id ][ pageNumber ] = "working";
				var dfd                                        = jQuery.Deferred();
				var url                                        = TE.site.baseURL.root + "/Image/Image/" + TE.book.id + "/" + pageNumber;
				TE.status.prefetch[ TE.book.id ][ pageNumber ] = url;
				if ( TE.config.preload )
				{
					$.when( TE.fn.load( pageNumber, url ) ).then( function ()
					{
						setTimeout(function()
						{
							dfd.resolve();
						},500);
					} );
					//TE.fn.load(pageNumber, pfImgSrc);
				}
				else
				{
					dfd.resolve();
				}

				//TE.vbLog( "gname", "Prefetch", url );

				/*
				 $.get( url ).done( function (data)
				 {
				 data                                           = TE.fn.scrubAjaxData( data );
				 var pfImg                                      = $( data ).find( "img.reader-img" );
				 var pfImgSrc                                   = $( pfImg ).attr( "data-content" );
				 TE.status.prefetch[ TE.book.id ][ pageNumber ] = pfImgSrc;
				 TE.vbLog( "gname", "Prefetch", "Prefetched image src for page " + pageNumber, TE.site.baseURL.root + pfImgSrc );
				 if ( TE.config.preload )
				 {
				 $.when( TE.fn.load( pageNumber, pfImgSrc ) ).then( function ()
				 {
				 dfd.resolve();
				 } );
				 //TE.fn.load(pageNumber, pfImgSrc);
				 }
				 else
				 {
				 dfd.resolve();
				 }
				 } );
				 */
				return dfd.promise();
			}
		},
		camelize        : function (str)
		{
			return str.replace( /(?:^\w|[A-Z]|\b\w|\s+)/g, function (match, index)
			{
				if ( +match === 0 )
				{
					return "";
				}
				return index == 0 ? match.toLowerCase() : match.toUpperCase();
			} );
		},
		updateSettings  : function ()
		{
			GM_setValue( "TE_settings", JSON.stringify( TE.User ) );
		},
		scrubAjaxData   : function (data)
		{
			data = TE.fn.replaceAll( data, "src=", "data-content=" );
			data = TE.fn.replaceAll( data, "script", "div" );
			data = TE.fn.replaceAll( data, "type=", "data-notype=" );
			data = TE.fn.replaceAll( data, "link", "div" );
			return data;
		},
		checkForUpdates : function ()
		{
			var dfd = jQuery.Deferred();

			if ( typeof TE.User.tsuminoEnhanced === "undefined" )
			{
				TE.User.tsuminoEnhanced                 = {};
				TE.User.tsuminoEnhanced.lastUpdateCheck = parseInt( new Date().getTime() );
				TE.User.tsuminoEnhanced.upToDate        = true;
				this.updateSettings();
			}

			if ( TE.User.tsuminoEnhanced.latestVersion != TE.version )
			{
				TE.User.tsuminoEnhanced.upToDate = false;
				this.updateSettings();
			}
			else
			{
				TE.User.tsuminoEnhanced.upToDate = true;
				this.updateSettings();
			}

			var now       = parseInt( new Date().getTime() );
			var oneMinute = 60000;
			var oneHour   = oneMinute * 60;
			var oneDay    = oneHour * 24;


			if ( now >= (parseInt( TE.User.tsuminoEnhanced.lastUpdateCheck ) + oneMinute) )
			{
				GM_xmlhttpRequest({
					method: "GET",
					url: TE.updateLocation,
					onload: $.proxy(function(response) {
						var scrubbed = this.scrubAjaxData(response.responseText);
						TE.log( "gname", TE.name, "Checking for updates..." );
						TE.User.tsuminoEnhanced.lastUpdateCheck = parseInt( new Date().getTime() );
						var latestVersion                       = $(scrubbed).find( "code" )[ 0 ];
						latestVersion                           = $( latestVersion ).text();
						TE.User.tsuminoEnhanced.latestVersion   = latestVersion;
						if ( TE.User.tsuminoEnhanced.latestVersion != TE.version )
						{
							TE.log( "gname", TE.name, "An update is available!" );
							TE.User.tsuminoEnhanced.upToDate = false;
							this.updateSettings();
							dfd.resolve();
						}
						else
						{
							TE.log( "gname", TE.name, TE.name + " is up to date!" );
							TE.User.tsuminoEnhanced.upToDate = true;
							this.updateSettings();
							dfd.resolve();
						}
					},this)
				});
			}
			else
			{
				dfd.resolve();
			}

			return dfd.promise();
		}
	};

	// Alias specific commonly used utility functions to the main namespace.
	TE.log            = TE.fn.log;
	TE.vbLog          = TE.fn.vbLog;
	TE.errorMsg       = TE.fn.errorMsg;
	TE.replaceAll     = TE.fn.replaceAll;
	TE.prefetch       = TE.fn.prefetch;
	TE.load           = TE.fn.load;
	TE.updateSettings = TE.fn.updateSettings;
	TE.randomString   = TE.fn.randomString;


	TE.config.internalIDs = {};
	TE.config.internalIDs.teConfigLink = TE.randomString();
	TE.status.idPrefix = TE.randomString() + "_";

	/*************************************************************************************
	 * Tsumino Enhanced User Interface settings.
	 * Stylesheets, data URI, etc.
	 *************************************************************************************/
		// User Interface object.
	TE.ui = {};

	/* Tsumino Enhanced CSS.
	 ** Minified so it's easier to include in the script.
	 **
	 ** Beautify here:	http://www.cleancss.com/css-beautify/
	 ** Minify here:		http://cssminifier.com/
	 */
	TE.ui.css        = {};
	TE.ui.css.master = `
	.te_mainColor { color:#22a7f0; }
	.te_en_incompatible{background-color:rgba(255,0,0,.1);border:2px solid rgba(255,0,0,0);border-radius:5px;padding:.5em}
	@font-face { font-family: "Icons";
	src: url("https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.1.8/themes/default/assets/fonts/icons.eot");
	src: url("https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.1.8/themes/default/assets/fonts/icons.eot?#iefix")
	format("embedded-opentype"),
	url("https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.1.8/themes/default/assets/fonts/icons.woff2") format("woff2"),
	url("https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.1.8/themes/default/assets/fonts/icons.woff") format("woff"),
	url("https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.1.8/themes/default/assets/fonts/icons.ttf") format("truetype"),
	url("https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.1.8/themes/default/assets/fonts/icons.svg#icons") format("svg");
	font-style: normal; font-weight: normal; font-variant: normal; text-decoration: inherit; text-transform: none; }


	#te_readerCurrentImage { margin: 0 auto; cursor: pointer; }
	`;


	// Browsing Tweaks CSS.
	TE.ui.css.browsingTweaks = {};

	// Browsing tweaks - Master CSS.
	TE.ui.css.browsingTweaks.master = `
	.te_browsetweak_infobutton,.te_browsetweak_readbutton {
		position:absolute;border:3px solid #fff;bottom:10px;
		padding:10px;margin-left:5%;margin-right:5%;
		font-size:17px;color:#fff;width:42.5%;display:inline-block;
		text-decoration:none;box-sizing:border-box }
	.te_browsetweak_readbutton{right:0;box-sizing:border-box}
	.te_browsetweak_infobutton:focus,.te_browsetweak_readbutton:focus{text-decoration:none}
	.te_browsetweak_infobutton:hover,.te_browsetweak_infobutton:visited,
	.te_browsetweak_readbutton:hover,.te_browsetweak_readbutton:visited
		{background-color:#22a7f0;color:#fff;text-decoration:none}
	.te_browsetweak_infobutton{left:0}
		`;

	// More Books CSS.
	TE.ui.css.browsingTweaks.moreBooks =
		`
		@media(min-width:768px) { .overlay-title { font-size:.8em; } .col-sm-4 { width: 25% } }
		@media(min-width:992px) { .col-md-3 { width: 20% } }
		`;

	// Record Keeper CSS.
	TE.ui.css.recordKeeper = `
		.te_recordKeeper_finished:hover { border: 3px solid rgba(0,125,0,.8) !important; }
		.te_recordKeeper_started:hover { border: 3px solid rgba(190,190,90,.8) !important; }
	`;

	// Tsumino Enhanced Favicon Data URI
	TE.ui.favicon
		= ``;

	TE.ui.mainColor = "#22a7f0";

	/*************************************************************************************
	 * Classes
	 *************************************************************************************/


	/*************************************************************************************
	 * Class: Enhancement
	 *************************************************************************************/
	TE.Enhancement = {
		/*************************************************************************************
		 * Class:     Enhancement
		 * Subclass: main
		 *
		 * Master Enhancement class.
		 *
		 * name:          Name of the Enhancement. String only.
		 * description:     Description of the Enhancement.
		 *                    Can be a string or boolean false.
		 *                    False will indicate no description.
		 * options:          Should be an object collection of option subclasses, or boolean false.
		 *                    False will indicate no options.
		 *                    If no options are provided:
		 *                         - The Enhancement will not appear on the configuration page.
		 *                         - The Enhancement will be activated automatically.
		 *                    If options is not false, one option must use the "enable" key.
		 * section:          The section of the config page the Enhancement will appear in.
		 * incompatible: Array of Enhancements that this one is incompatible with.
		 * fn:               Object containing all actual Enhancement functionality.
		 *                    "init" key should be used for activation.
		 *                    "upgradeHandling" key should be used for upgrade handling.
		 *
		 * TE.Enhancement.main(name,displayName,description,options,section,incompatible,fn)
		 *************************************************************************************/
		main   : function (name, description, options, section, incompatible, fn)
		{
			try
			{
				if ( typeof name !== "string" )
				{
					throw new Error( "Enhancement name must be defined as a string." );
				}
				if ( (typeof description !== "string") && (description != false) )
				{
					throw new Error( "Enhancement description must be defined as a string." );
				}
				if ( (typeof options !== "object") && (options != false) )
				{
					throw new Error( "Enhancement options must be defined as an object." );
				}
				if ( (options != false) || (section != false) )
				{
					if ( typeof section !== "string" )
					{
						throw new Error( "Enhancement section must be defined as a string." );
					}
				}
				if ( (typeof incompatible !== "object") && (incompatible != false) )
				{
					throw new Error( "Enhancement incompatibilities must be defined as 'false', or an array." );
				}

				if ( typeof fn !== "object" )
				{
					throw new Error( "Enhancement functionality must be defined as an object." );
				}

				this.name         = name;
				this.shortName    = TE.fn.camelize( name );
				this.description  = description;
				this.options      = options;
				this.section      = section;
				this.incompatible = incompatible;
				this.fn           = fn;
			}
			catch (error)
			{
				TE.errorMsg( "CD01", "Creating an Enhancement class object.", error );
			}
		}, /*************************************************************************************
		 * Class:     Enhancment
		 * Subclass: option
		 *************************************************************************************/
		option : {
			/*************************************************************************************
			 * Master Class:     Enhancment
			 * Parent Class:     option
			 * Subclass:          main
			 *
			 * Primary Option class.
			 *
			 * type:               The type of the Option.
			 *                    Must be a string of one of the following:
			 *                    "enable" - Switch that enables the Enhancement.
			 *                         Options of this type require no further parameters.
			 *                    "toggle" - Renders a checkbox.
			 *                    "radio" - Radio buttons. Requires arguments.
			 *                    "dropdown" - Dropdown menu. Requires arguments.
			 * name:          Name of the option.
			 * description:     Description of the Option.
			 *                    Can be a string or boolean false.
			 *                    False will indicate no description.
			 * defaultValue: The default value of the option.
			 * params:     Must be an object of the "params" subclass, or boolean false.
			 *                    False indicates no params.
			 *
			 * TE.Enhancement.option.main(type,name,description,defaultValue,arguments)
			 *************************************************************************************/
			main   : function (type, name, description, defaultValue, params)
			{
				try
				{
					if ( typeof type !== "string" )
					{
						throw new Error( "Option type must be defined as a string." );
					}
					if ( type != "enable" )
					{
						if ( typeof name !== "string" )
						{
							throw new Error( "Option name must be defined as a string." );
						}
						if ( typeof description === "undefined" )
						{
							description = false;
						}

						if ( type == "toggle" )
						{
						}
						if ( (type == "dropdown") && (typeof v !== "object") )
						{
							throw new Error( "You must define params for option type dropdown." );
						}
						if ( (type == "radio") && (typeof params !== "object") )
						{
							if ( typeof params === "undefined" )
							{
								throw new Error( "You must define params for option type radio." );
							}
						}
						if ( type == "dropdown" )
						{
							if ( typeof params === "undefined" )
							{
								throw new Error( "Option params must be defined with type dropdown." );
							}
						}

						this.type         = type;
						this.name         = name;
						this.shortName    = TE.fn.camelize( name );
						this.description  = description;
						this.defaultValue = defaultValue;
						this.params       = params;
					}
					else
					{
						this.type         = type;
						this.name         = "enable";
						this.shortName    = "enable";
						this.description  = false;
						this.defaultValue = false;
						this.params       = false;
					}
				}
				catch (error)
				{
					TE.errorMsg( "CD02", "Creating an Enhancement.option.main class object.", error );
				}
			}, /*************************************************************************************
			 * Master Class:     Enhancment
			 * Parent Class:     option
			 * Subclass:          params
			 *
			 * params for radio and dropdown type options.
			 *
			 * type:               The type of the param.
			 *                    Must be a string of one of the following:
			 *                         "range" - For generating a
			 *************************************************************************************/
			params : function (type, options)
			{
				try
				{
					if ( typeof type !== "string" )
					{
						throw new Error( "Type must be defined as a string." );
					}
					if ( type == "range" )
					{
						if ( typeof options !== "object" )
						{
							throw new Error( "options must be defined as an object." );
						}
						else
						{
							if ( typeof options.rangeStart !== "number" )
							{
								throw new Error( "Range start must be defined as a number." );
							}
							if ( typeof options.rangeEnd !== "number" )
							{
								throw new Error( "Range end must be defined as a number." );
							}
						}
					}
					else
					{
						throw new Error( "Range is the only acceptable type right now." );
					}

					this.type   = type;
					this.params = params;
				}
				catch (error)
				{
					TE.log( "gname", "ERROR", "Error defining params class:", error );
					TE.errorMsg( "CD03", "Creating an Enhancement.option.params class object.", error );
				}
			}
		}
	};


	//Fix the navbar.
	TE.fixNavbar = function ()
	{
		$( "nav .tsumino-nav-items li" ).click( function ()
		{
			var thisLink = $( this ).find( "a:first" )[ 0 ];

			if ( $( thisLink ).text() == "Browse " )
			{
				$( "#te_navMenu" ).toggleClass( "tsumino-nav-visible" );
			}
			else if ( $( thisLink ).prop( "id" ) == TE.config.internalIDs.teConfigLink )
			{
				$( "#te_config_modal" )
					.modal( "show" )
					.modal( "refresh" );
			}
			else
			{
				window.location.href = $( thisLink ).prop( "href" );
			}
		} );
	};

	/*************************************************************************************
	 * Enhance Page - Core Functionality
	 *  + Creates IDs for important elements.
	 *  + Gathers data for storage in the TE object.
	 *************************************************************************************/
	TE.enhancePage = function ()
	{
		var dfd = jQuery.Deferred();
		TE.vbLog( "gname", "TE.enhancePage", "Started working..." );
		$( document ).ready( function ()
		{
			/*************************************************************************************
			 * All pages.
			 *************************************************************************************/

			if ( !TE.on.forum )
			{
				// Prepare config modal
				$( "body" ).append( `<div id="te_config_modal" class="ui fullscreen basic modal"></div>` );
				TE.settings.render();

				// The navigation bar at the top.
				$( "nav" ).attr( "id", "te_siteNavbar" );

				// Replace favicon.
				$( "link[rel='icon']" ).attr( "href", TE.ui.favicon );


				// Add Tsumino Enhanced config link to navbar.
				var navbar = $( ".tsumino-nav-left" )[ 0 ];
				$( navbar ).attr( "id", "te_navbarMain" );

				// Add ID to browse button link and swap href from # to javascript:;.
				var browseButton = $( "#te_navbarMain" ).find( "a[href=#]" )[ 0 ];
				$( browseButton ).prop( "id", "te_navBrowse" );
				$( "#te_navBrowse" ).prop( "href", "javascript:;" );

				// Add ID to nav menu.
				var navMenu = $( "#te_navBrowse" ).siblings()[ 0 ];
				$( navMenu ).prop( "id", "te_navMenu" );
				TE.fixNavbar();

				if ( !TE.User.tsuminoEnhanced.upToDate )
				{
					$( w ).load( function ()
					{
						$( "#"+TE.config.internalIDs.teConfigLink ).append( "&nbsp;&nbsp;<i class='ui red icon upload'></i>" );
						$(".search-wrapper input").css("margin-left","8em");
						$( "#"+TE.config.internalIDs.teConfigLink ).parent().popup( {
							title : 'An update is available!'
						} );
						$( "#"+TE.config.internalIDs.teConfigLink ).parent().css( "background-color", "#333333" );
						setTimeout( function ()
						{
							$( "#"+TE.config.internalIDs.teConfigLink ).parent()
																	   .velocity( {backgroundColor : "#2D5467"} )
																	   .velocity( {backgroundColor : "#333333"} )
																	   .velocity( {backgroundColor : "#2D5467"} )
																	   .velocity( {backgroundColor : "#333333"} );
						}, 1 );
					} );
				}

				// ID the primary content area.
				var pageContent = $( "div.container-fluid" )[ 0 ];
				$( pageContent ).attr( "id", "te_pageContent" );

				var footer = $( "div.nav-footer" )[ 0 ];
				$( footer ).attr( "id", "te_page_footer" );


				var teNLC_className = TE.fn.randomString();
				var teNLC_height = $("nav.tsumino-nav").height();
				var teNLC_left = $("#te_navbarMain").offset().left;
				teNLC_left += $("#te_navbarMain").width();
				var teNLC_top = 0;

				TE.ui.css.master = TE.ui.css.master + `
					.a`+teNLC_className+`
					{
						float: left;
						position: absolute;
						top:`+teNLC_top+`px;
						left:`+teNLC_left+`px;
						height: `+teNLC_height+`px;
						z-index:999;
						border-left: 1px solid #282828;
					}
					.a`+TE.config.internalIDs.teConfigLink+`
					{
						margin:10px 5px;
					}
				`;

				$( "body" ).append( `
				<div class="a`+teNLC_className+`">
					<div id="`+TE.config.internalIDs.teConfigLink+`" class="a`+TE.config.internalIDs.teConfigLink+`">
						<a href='javascript:;' style='color:` + TE.ui.mainColor + ` !important;'>ENHANCED</a>
					</div>
				</div>
					` );
				$( "#"+TE.config.internalIDs.teConfigLink ).click( function ()
				{
					$( "#te_config_modal" ).modal( "show" );
					$( "#te_config_modal" ).modal( "refresh" );
				} );

				$(".search-wrapper input").css("margin-left","6.5em");


				$( "head" )
				// Include Semantic CSS
					.prepend( "<link rel='stylesheet' href='http://js.codingtoby.com/semantic.min.css' />" )
					// Apply Tsumino Enhanced CSS.
					.append( "<style>" + TE.ui.css.master + "</style>" );
			}

			/*************************************************************************************
			 * Book & Reader
			 *************************************************************************************/
			if ( TE.on.reader || TE.on.book )
			{
				// Create the book object.
				TE.book = {};

				// Reader only.
				if ( TE.on.reader && false )
				{
					// Create IDs.
					$( ".reader-page" ).attr( "id", "te_readerPageMain" );
					var imageBlock = $( "#te_readerPageMain" ).children()[ 0 ];
					$( imageBlock ).attr( "id", "te_imageBlock" );
					$( ".reader-btn" ).attr( "id", "te_readerButtonContainer" );
					//$( "img.reader-img" ).attr( "id", "te_readerCurrentImage" );

					var readInfo = TE.myLocation;
					readInfo     = readInfo.replace( TE.site.reader.url, "" );
					readInfo     = readInfo.split( "/" );

					// Book ID.
					TE.book.id = parseInt( readInfo[ 0 ] );
					TE.log(readInfo[ 1 ]);

					// Book title.
					var bookTitle = $( "title" ).text();
					TE.book.title = bookTitle.replace( "Tsumino - Free Premium Doujinshi and Hentai Manga: Read ", "" );

					// Pagination setup.
					var pagination = $( "#te_readerButtonContainer" ).find( "h1" )[ 0 ];
					$( pagination ).attr( "id", "te_readerPagination" );
					var pagesInfo = $( "#te_readerPagination" ).text();
					pagesInfo     = pagesInfo.split( " Page " )[ 1 ];
					pagesInfo     = pagesInfo.split( " of " );

					// Current Page.
					if(readInfo.length > 1)
					{
						TE.book.currentPage    = parseInt( readInfo[ 1 ] );
					}
					else
					{
						TE.book.currentPage    = 1;
					}
					TE.book.currentPageURL = TE.site.reader.prefix + TE.book.id + "/" + TE.book.currentPage;

					$( "title" ).text( "Tsumino - " + TE.book.title + " - Page " + TE.book.currentPage );

					// Origin page.
					TE.book.originPage = TE.book.currentPage;

					// Total Pages.
					TE.book.totalPages = parseInt( pagesInfo[ 1 ] );

					// Next page.
					TE.book.nextPage = TE.book.currentPage + 1;
					if ( TE.book.nextPage > TE.book.totalPages )
					{
						TE.book.nextPage = false;
					}
					else
					{
						TE.book.nextPageURL = TE.site.reader.prefix + TE.book.id + "/" + TE.book.nextPage;
					}

					// Previous page.
					TE.book.prevPage = TE.book.currentPage - 1;
					if ( TE.book.prevPage == 0 )
					{
						TE.book.prevPage = false;
					}
					else
					{
						TE.book.prevPageURL = TE.site.reader.prefix + TE.book.id + "/" + TE.book.prevPage;
					}

					// Rewrite pagination section.
					$( "#te_readerPagination" ).before( `<div id="te_readerMessageDisplay" style="margin-bottom: 25px;"></div>` );
					$( "#te_readerPagination" ).html( "Page <span id='te_currentPage'></span> of <span id='te_totalPages'></span>" );
					$( "#te_currentPage" ).html( TE.book.currentPage );
					$( "#te_totalPages" ).html( TE.book.totalPages );

					// Rename Return button to 'book info' and give it an ID.
					var bookInfoButton = $( "a[href*='" + TE.site.book.prefix + "']:contains('RETURN')" );
					$( bookInfoButton ).attr( "id", "te_bookInfoButton" );
					$( "#te_bookInfoButton" )
						.attr("href",TE.site.book.prefix + TE.book.id)
						.text( "BOOK INFO" );

					// Add a return button that takes you to the index.
					$( "#te_bookInfoButton" ).after( `
						<a class='book-read-button button-stack' id='te_returnToIndexButton'><i class='fa fa-home'></i> BACK TO INDEX</a>
						` );
					var returnToIndexLink = sessionStorage.getItem( "te_returnLink" );
					if ( typeof returnToIndexLink === "object" )
					{
						$( "#te_returnToIndexButton" ).attr( "href", TE.site.baseURL.root );
					}
					else
					{
						$( "#te_returnToIndexButton" ).attr( "href", returnToIndexLink );
					}


					// Enhance Previous Page button.
					if ( $( "a:contains(' PREV')" ).length )
					{
						$( "a:contains(' PREV')" ).attr( "id", "te_prevButton" );
					}
					else
					{
						$( "#jump-page" ).before( `
							<a id="te_prevButton" class="book-read-button button-stack" style="margin-right: 10px;"><i class="fa fa-arrow-left"></i> PREV</a>
							` );
						$( "#te_prevButton" ).css( "display", "none" );
					}
					$( "#te_prevButton" ).attr( "href", TE.book.prevPageURL );
					if ( TE.book.currentPage > 1 )
					{
						$( "#te_prevButton" ).css( "display", "inline" );
					}
					else
					{
						$( "#te_prevButton" ).css( "display", "none" );
					}

					// Enhance Next Page Button
					if ( $( "a:contains('NEXT ')" ).length )
					{
						$( "a:contains('NEXT ')" ).attr( "id", "te_nextButton" );
					}
					else
					{
						$( "#jump-page" ).after( `
							<a id="te_nextButton" class="book-read-button button-stack">NEXT <i class="fa fa-arrow-right"></i></a>
							` );
						$( "#te_nextButton" ).css( "display", "none" );
					}
					$( "#te_nextButton" ).attr( "href", TE.book.nextPageURL );
					if ( TE.book.currentPage < TE.book.totalPages )
					{
						$( "#te_nextButton" ).css( "display", "inline" );
					}
					else
					{
						$( "#te_nextButton" ).css( "display", "none" );
					}


					$("#image-container")
						.after(`
						<div id="te_imageContainer">
							<img class="img-responsive" src="/Image/Image/`+TE.book.id+`/`+TE.book.currentPage+`" id="te_readerCurrentImage">
						</div>
						`)
						.remove();



					// Enhance Image link.
					/*
					 var imageLink = $( "#te_readerCurrentImage" ).parent();
					 $( imageLink ).attr( "id", "te_imageLink" );
					 $( "#te_imageLink" ).attr( "href", TE.book.nextPageURL );
					 */

					TE.vbLog( "gname", "TE.book", TE.book );
				}
				else if ( TE.on.book )
				{
					var readInfo = TE.myLocation;
					readInfo     = readInfo.replace( TE.site.book.url, "" );
					readInfo     = readInfo.split( "/" );

					// Book ID.
					TE.book.id = parseInt( readInfo[ 0 ] );

					// Fix tag display bug causing unwanted line breaks.
					$( ".book-tag:contains(' ')" ).css( "white-space", "nowrap" );
					$( ".book-tag:contains('-')" ).css( "white-space", "nowrap" );

					// Read Online button.
					$( "a.book-read-button:contains(' READ ONLINE')" ).attr( "id", "te_readOnlineButton" );

					var indexButton = $( "a:contains(' BACK TO INDEX')" );
					$( indexButton ).attr( "id", "te_backToIndexButton" );
					var returnToIndexLink = sessionStorage.getItem( "te_returnLink" );
					if ( typeof returnToIndexLink !== "object" )
					{
						$( "#te_backToIndexButton" ).attr( "href", returnToIndexLink );
					}
				}
			}
			if ( TE.on.browse )
			{
				sessionStorage.setItem( "te_returnLink", TE.myLocation );

				var browsePage         = $( "div.browse-page" );
				var ctProper           = $( "div.row.push-in" );
				var bookshelfContainer = $( ctProper ).children()[ 0 ];
				$( bookshelfContainer ).attr( "id", "te_bookshelfContainer" );
				var sidebarContainer = $( ctProper ).children()[ 1 ];
				$( sidebarContainer ).attr( "id", "te_sidebarContainer" );

				var bookshelf = $( browsePage ).find( "div.row.row-no-padding" );
				$( bookshelf ).attr( "id", "te_bookshelf" );
				$( bookshelf ).children().each( function ()
				{
					var thisLinkUrl = $( this ).find( "a.overlay-button" ).attr( "href" );
					var temp        = thisLinkUrl.replace( TE.site.book.prefix, "" );
					temp            = temp.split( "/" );
					var thisBookID  = temp[ 0 ];
					$( this ).attr( "id", "te_book_" + thisBookID + "_masterContainer" );

					$( this ).find( "div.book-grid-item" ).attr( "id", "te_book_" + thisBookID + "_container" );

					var thisOverlay = $( "#te_book_" + thisBookID + "_container" ).find( "div.overlay" );
					$( thisOverlay ).attr( "id", "te_book_" + thisBookID + "_overlay" );

					var thisData = $( "#te_book_" + thisBookID + "_overlay" ).find( "div.overlay-data" );
					$( thisData ).attr( "id", "te_book_" + thisBookID + "_data" );

					var thisPages = $( "#te_book_" + thisBookID + "_data" ).find( "div.overlay-sub" );
					$( thisPages ).attr( "id", "te_book_" + thisBookID + "_pagesContainer" );

					var bottomTitle = $( this ).find( "a.title" );
					$( bottomTitle ).attr( "id", "te_book_" + thisBookID + "_bottomTitle" );
					$( "#te_book_" + thisBookID + "_bottomTitle" ).attr( "href", "javascript:;" );
				} );
				//TE.vbLog( "gname", "TE.enhancePage", bookshelf );
			}
			TE.vbLog( "gname", "TE.enhancePage", "Finished working." );
			dfd.resolve();
		} );

		return dfd.promise();
	};


	/*************************************************************************************
	 * Enhancement Code
	 * Create all Enhancements within anonymous functions.
	 *************************************************************************************/
		// Object for storing all Enhancements
	TE.Enhancements = {};

	/*******************************************************
	 * Hidden Enhancements
	 * Important functionality.
	 * Customization either not yet written or not required.
	 *******************************************************/


	/*******************************************************
	 * General Enhancements
	 *******************************************************/
	(function ()
	{
		/*******************************************************
		 * Unstickied Header - General Enhancement
		 *******************************************************/
		var name, shortName, description, options, section, incompatible, main;
		name         = "Unstickied Header";
		shortName    = TE.fn.camelize( name );
		description  = "The Tsumino navigation bar will no longer follow you as you scroll down.";
		options      = [];
		section      = "General";
		incompatible = false;
		main         = {
			init : function ()
			{
				this.run();
			},
			run  : function ()
			{
				$( "#te_siteNavbar" ).css( "position", "absolute" );
			}
		};
		//TE.Enhancement.option.main(type,name,description,defaultValue,arguments)
		var opt1 = {
			type         : "enable",
			name         : false,
			description  : false,
			defaultValue : false,
			arguments    : false
		};
		options.push( new TE.Enhancement.option.main( opt1.type, opt1.name, opt1.description, opt1.defaultValue, opt1.arguments ) );
		TE.Enhancements[ shortName ] = new TE.Enhancement.main( name, description, options, section, incompatible, main );
	})();


	(function ()
	{
		/*******************************************************
		 * Record Keeper - General Enhancement
		 *******************************************************/
		var name, shortName, description, options, section, incompatible, main;
		name         = "Record Keeper";
		shortName    = TE.fn.camelize( name );
		description  = `
		The aptly named Record Keeper Enhancement keeps a record of what Doujin you've read.<br />
		This record includes Doujin IDs, the last page you read, and whether or not you finished reading a Doujin.<br />
		While browsing, unread Doujin will retain the normal blue border.<br />
		Doujin you have started but haven't finished will have a yellow border.<br />
		Doujin you've finished reading will have a green border.<br />
		Additionally, the information overlay will now contain this information.<br />
		There will also be a 'Continue Reading' button on the book info page if you have previously read a Doujin past the first page.
		`;
		options      = [];
		section      = "General";
		incompatible = false;
		main         = {};


		main = {
			init : function ()
			{
				if ( typeof TE.User.recordKeeper.data !== "object" )
				{
					TE.User.recordKeeper.data = {};
					TE.updateSettings();
				}
				if ( TE.on.browse )
				{
					$.when( TE.status.enhancePage ).done( function ()
					{
						$( "style" ).append( TE.ui.css.recordKeeper );
						TE.log( "gname", name, "Initializining..." );
						for (var key in TE.User.recordKeeper.data)
						{
							if ( TE.User.recordKeeper.data.hasOwnProperty( key ) )
							{
								var obj = TE.User.recordKeeper.data[ key ];
								if ( obj[ "finished" ] )
								{
									$( "#te_book_" + key + "_pagesContainer" ).append( "<br />Finished!" );
									$( "#te_book_" + key + "_container" ).addClass( "te_recordKeeper_finished" );
									$( "#te_book_" + key + "_bottomTitle" ).css( "border-top", "3px solid rgba(0,125,0,.8)" );
								}
								if ( (obj[ "lastSeen" ] > 1) && (!obj[ "finished" ]) )
								{
									$( "#te_book_" + key + "_pagesContainer" ).text( "Read " + obj[ "lastSeen" ] + " / " + obj[ "totalPages" ]
																					 + " pages." );
									$( "#te_book_" + key + "_container" ).addClass( "te_recordKeeper_started" );
									$( "#te_book_" + key + "_bottomTitle" ).css( "border-top", "3px solid rgba(190,190,90,.8)" );
								}
							}
						}
					} );
				}
				if ( TE.on.book )
				{
					$.when( TE.status.enhancePage ).done( $.proxy( function ()
					{
						TE.log( "gname", name, "Initializining..." );
						if ( typeof TE.User.recordKeeper.data[ TE.book.id ] === "object" )
						{
							if ( TE.User.recordKeeper.data[ TE.book.id ][ "lastSeen" ] > 1 )
							{
								//TE.User.recordKeeper.data[ TE.book.id ][ "lastSeen" ];
								var oldButton       = $( "#te_readOnlineButton" ).html();
								var starOver        = oldButton.replace( " READ ONLINE", " START OVER" );
								var continueReading = oldButton.replace( " READ ONLINE", " CONTINUE READING" );
								$( "#te_readOnlineButton" ).html( starOver );
								var resumeUrl = TE.site.reader.url + TE.book.id + "/" + TE.User.recordKeeper.data[ TE.book.id ][ "lastSeen" ];
								$( "#te_readOnlineButton" ).before( `
									<a id='te_resumeButton' class='book-read-button button-stack' href='` + resumeUrl + `'></a>
								` );
								$( "#te_resumeButton" ).html( continueReading );
							}
						}
					}, this ) );
				}
				if ( TE.on.reader && false )
				{
					$.when( TE.status.enhancePage ).done( $.proxy( function ()
					{
						TE.log( "gname", name, "Initializining..." );
						if ( typeof TE.User[ shortName ].data !== "object" )
						{
							TE.User[ shortName ].data = {};
						}
						if ( typeof TE.User[ shortName ].data[ TE.book.id ] !== "object" )
						{
							TE.User[ shortName ].data[ TE.book.id ] = {
								totalPages : TE.book.totalPages, lastSeen : TE.book.currentPage, finished : false
							};
						}
						this.update();
					}, this ) );
				}
			},

			update : function ()
			{
				TE.User[ shortName ].data[ TE.book.id ].lastSeen = TE.book.currentPage;
				if ( !TE.User[ shortName ].data[ TE.book.id ][ "finished" ] )
				{
					if ( TE.book.totalPages == TE.book.currentPage )
					{
						TE.User[ shortName ].data[ TE.book.id ][ "finished" ] = true;
					}
				}
				TE.updateSettings();
			}
		};

		//TE.Enhancement.option.main(type,name,description,defaultValue,arguments)
		var opt1 = {
			type         : "enable",
			name         : false,
			description  : false,
			defaultValue : false,
			arguments    : false
		};
		var opt2 = {
			type         : "toggle",
			name         : "Show Messages",
			description  : "Displays loading messages while preparing images for display.",
			defaultValue : true,
			arguments    : false
		};
		options.push( new TE.Enhancement.option.main( opt1.type, opt1.name, opt1.description, opt1.defaultValue, opt1.arguments ) );
		//options.push(new TE.Enhancement.option.main(opt2.type,opt2.name,opt2.description,opt2.defaultValue,opt2.arguments));

		TE.Enhancements[ shortName ] = new TE.Enhancement.main( name, description, options, section, incompatible, main );
	})();


	(function ()
	{
		/*******************************************************
		 * Browse Tags Lite - Browsing Enhancement
		 *******************************************************/
		var name, shortName, description, options, section, incompatible, main;
		name         = "Browse Tags Lite";
		shortName    = TE.fn.camelize( name );
		description  = `Adds a new link to Tsumino's browse menu. "ALL TAGS (LITE)"<br />
						Clicking it launches the Browse Tags Lite modal window.<br />
						Tags are sorted alphabetically and categorized by the first letter.<br />
						Click a letter at the top to view tags that start with that letter.<br />
						The tag list is downloaded automatically the first time you open BTL.`;
		options      = [];
		section      = "Browsing";
		incompatible = false;
		main         = {
			init             : function ()
			{
				var firstLoad = true;
				$.when( TE.status.enhancePage ).then( $.proxy( function ()
				{
					$( "ul#te_navMenu" ).append( `<li id="te_browseTagsLite_menuLink"><a>All Tags (Lite)</a></li>` );

					var alphabet = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
					alphabet     = alphabet.split( "" );

					$( "body" ).append( `
					<div id="te_browseTagsLite_modal" class="ui fullscreen long basic modal">
						<button id="te_btl_closeModal" class="ui icon button right floated"><i class="remove icon"></i></button>
						<div class="header"><h1><span class="te_mainColor">Browse Tags Lite</span></h1></div>
						<div id="te_browseTagsLite_modalBody" class="content" style="font-size: 1.4em;">
							<div id="te_btl_mainDisplay">

							</div>
						</div>
					</div>
					` );

					$( "#te_browseTagsLite_modalBody" ).prepend( `<div id="te_browseTagsLite_tabs" class="ui top attached inverted large tabular menu"></div>` );


					/*
					 $( "#te_browseTagsLite_tabs" ).prepend( `<a class="item te_btl_tabButton" data-tab="te_btl_tab_all">*</a>` );
					 $( "#te_btl_mainDisplay" ).append( `
					 <div id="te_btl_tab_all" class="ui bottom attached inverted tab segment" data-tab="te_btl_tab_all">
					 <h1>ALL</h1>
					 Tags go here
					 </div>
					 ` );
					 */


					// Populate button list
					for (var i = 0 ; i < alphabet.length ; i++)
					{
						$( "#te_browseTagsLite_tabs" ).append(
							`<a id="te_btl_btn_` + alphabet[ i ] + `" class="item te_btl_tabButton" data-tab="te_btl_tab_` + alphabet[ i ] + `">`
							+ alphabet[ i ] + `</a>` );
						$( "#te_btl_mainDisplay" ).append( `
							<div id="te_btl_tab_` + alphabet[ i ] + `" class="ui bottom attached inverted tab segment" data-tab="te_btl_tab_`
														   + alphabet[ i ] + `">
							</div>
							` );
					}
					$( "#te_browseTagsLite_tabs" ).append( `
						<a id="te_btl_btn_reload" class="icon item"><i class="icon refresh" style="color:` + TE.ui.mainColor + `"></i></a>
						` );

					// Check to see if the user has a copy of the tag list.
					if ( TE.User.tsuminoEnhanced.tagList )
					{
						this.renderTagList();
						$( "#te_btl_btn_B" ).addClass( "active" );
						$( "#te_btl_tab_B" ).addClass( "active" );
					}

					$( "#te_btl_btn_reload" ).click( $.proxy( function ()
					{
						$( "#te_btl_btn_reload" ).addClass( "disabled" );
						$( "#te_browseTagsLite_modal" ).append( `
							<div id="te_btl_downloadingTagListLoader">
								<div class="ui active dimmer">
									<div class="ui text loader">Downloading Tag List...</div>
								</div>
							</div>
						` );
						$.when( this.updateTagList() ).then( $.proxy( function ()
						{
							$( "#te_btl_downloadingTagListLoader" ).remove();
							$( "#te_btl_btn_reload" ).removeClass( "disabled" );
							this.renderTagList();
							$( "#te_btl_btn_A" ).addClass( "active" );
							$( "#te_btl_tab_A" ).addClass( "active" );
						}, this ) );
					}, this ) );


					// Initialize tabs.
					$( ".tabular.menu .item" ).tab();

					// Initialize modal.
					$( "#te_browseTagsLite_modal" ).modal(
						{
							onVisible      : function ()
							{
								if ( !TE.User.tsuminoEnhanced.tagList )
								{
									$( "#te_btl_btn_reload" ).click();
								}

								if ( firstLoad )
								{
									$( "#te_browseTagsLite_modal" ).modal( "refresh" );
									$( "#te_btl_btn_A" ).click();
									firstLoad = false;
								}
							},
							observeChanges : true
						} );

					$( "#te_browseTagsLite_menuLink" ).click( function ()
					{
						$( "#te_browseTagsLite_modal" ).modal( "show" );
					} );

					$( ".te_btl_tabButton" ).click( function ()
					{
						$( "#te_browseTagsLite_modal" ).modal( "refresh" );
					} );

					// Close button
					$( "#te_btl_closeModal" ).click( function ()
					{
						$( "#te_browseTagsLite_modal" ).modal( "hide" );
					} );


					$( "#te_btl_btn_reload" ).popup( {
						title : 'Reload the Tag List from the Server'
					} );


				}, this ) );
			}, updateTagList : function ()
			{
				var dfd                     = jQuery.Deferred();
				TE.status.tagListDownloaded = false;
				TE.status.tagListPage       = 1;
				TE.status.tagList           = [];

				bean.on( TE.status.tagList, "completed", function ()
				{
					TE.log( "gname", "Browse Tags Lite", "Tag list downloaded." );
					TE.status.tagListDownloaded     = true;
					TE.User.tsuminoEnhanced.tagList = TE.status.tagList;
					TE.updateSettings();
					dfd.resolve();
				} );

				this.loadTagPages( TE.status.tagListPage );

				TE.log( "gname", "Browse Tags Lite", "Retrieving tags..." );
				return dfd.promise();
			},
			loadTagPages     : function (pageNum)
			{
				if ( !TE.status.tagListDownloaded )
				{
					$.ajax( {
						method  : "GET",
						url     : TE.site.baseURL.root + TE.site.browseTags.prefix,
						data    : {infpage : pageNum},
						success : $.proxy( function (data)
						{
							if ( data.trim() != "" )
							{
								data = TE.fn.scrubAjaxData( data );
								$( data ).find( "a" ).each( function ()
								{
									TE.status.tagList.push( $( this ).text() );
								} );
								TE.status.tagListPage++;
								this.loadTagPages( TE.status.tagListPage );
							}
							else
							{
								bean.fire( TE.status.tagList, "completed" );
							}
						}, this ),
						error   : function () {}
					} );
				}
			},
			renderTagList    : function ()
			{
				var numTags         = 0;
				var tagCounter      = [];
				var thisTag         = "";
				var urlFormattedTag = "";
				var thisTagUrl      = "";
				var idFormattedTag  = "";
				var thisTagFL       = "";
				var alphabet        = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
				alphabet            = alphabet.split( "" );

				for (var i = 0 ; i < alphabet.length ; i++)
				{
					tagCounter[ alphabet[ i ] ] = 0;
					$( "#te_btl_tab_" + alphabet[ i ] ).html( `
						<h1>` + alphabet[ i ] + `</h1>
						<hr />` );

					$( "#te_btl_tab_" + alphabet[ i ] ).removeClass( "active" );
					$( "#te_btl_btn_" + alphabet[ i ] ).removeClass( "active" );
				}

				for (var i = 0 ; i < TE.User.tsuminoEnhanced.tagList.length ; i++)
				{
					numTags++;
					thisTag = TE.User.tsuminoEnhanced.tagList[ i ];
					thisTag = thisTag.trim();
					thisTag = thisTag.replace( /\w\S*/g, function (txt)
					{
						return txt.charAt( 0 ).toUpperCase() + txt.substr( 1 ).toLowerCase();
					} );

					urlFormattedTag = TE.fn.replaceAll( thisTag, " ", "+" );
					thisTagUrl      = TE.site.baseURL.root + "/Browse/Tag?name=" + urlFormattedTag;
					idFormattedTag  = TE.fn.replaceAll( thisTag, " ", "_" );
					thisTagFL       = thisTag.charAt( 0 );
					tagCounter[ thisTagFL ]++;
					$( "#te_btl_tab_" + thisTagFL ).append( `
						<a class="book-tag" id="te_btl_tagLink_"` + idFormattedTag + `" href="` + thisTagUrl + `">` + thisTag + `</a>
						<br />
						` );
				}

				// Remove links from empty sections and add tag counters to populated sections.
				for (i = 0 ; i < alphabet.length ; i++)
				{
					if ( tagCounter[ alphabet[ i ] ] == 0 )
					{
						$( "#te_btl_tab_" + alphabet[ i ] ).append( "<br />No tags that start with '" + alphabet[ i ] + "'.<br />" );
					}
				}
			}

		};

		//TE.Enhancement.option.main(type,name,description,defaultValue,arguments)
		var opt1 = {
			type         : "enable",
			name         : false,
			description  : false,
			defaultValue : false,
			arguments    : false
		};
		options.push( new TE.Enhancement.option.main( opt1.type, opt1.name, opt1.description, opt1.defaultValue, opt1.arguments ) );
		TE.Enhancements[ shortName ] = new TE.Enhancement.main( name, description, options, section, incompatible, main );
	})();

	(function ()
	{
		/*******************************************************
		 * Browsing Tweaks - Browsing Enhancement
		 *******************************************************/
		var name, shortName, description, options, section, incompatible, main;
		name         = 'Browsing Tweaks';
		shortName    = TE.fn.camelize( name );
		description  = "A collection of customizations to browsing.";
		options      = [];
		section      = "Browsing";
		incompatible = false;
		main         = {};

		main = {
			init : function ()
			{
				if ( TE.on.browse )
				{
					$.when( TE.status.enhancePage ).done( function ()
					{
						if ( typeof TE.User[ shortName ] !== "undefined" )
						{
							if ( TE.User[ shortName ].removeSidebar )
							{
								$( "#te_sidebarContainer" ).remove();
								$( "#te_bookshelfContainer" ).css( "width", "100%" );
							}

							if ( TE.User[ shortName ].moreBooks )
							{
								$( "style" ).append( "@media(min-width:768px) { .overlay-title { font-size:.8em; } .col-sm-4 { width: 25% } }" );
								$( "style" ).append( "@media(min-width:992px) { .col-md-3 { width: 20% } }" );
							}
							if ( TE.User[ shortName ].skipInfo )
							{
								// Apply new CSS.
								$( "style" ).append( TE.ui.css.browsingTweaks.master );
								$( "div.overlay" ).each( function ()
								{
									// Get Book ID
									var bookID = $( this ).attr( "id" );
									bookID     = bookID.replace( "te_book_", "" );
									bookID     = bookID.replace( "_overlay", "" );
									bookID     = parseInt( bookID );

									// Replace old class on view button.
									var viewInfoButton = $( this ).find( "a.overlay-button" );
									var linkURL = TE.site.reader.prefix + bookID + "/1";
									$( viewInfoButton ).after(`<a href="`+linkURL+`" class="te_browsetweak_readbutton">READ</a>`);
									var readButton = $( this ).next();
									$( viewInfoButton ).text( "INFO" );
									$( viewInfoButton ).removeClass( "overlay-button" );
									$( viewInfoButton ).addClass( "te_browsetweak_infobutton" );
									// Add new read button.


									if ( TE.User.recordKeeper )
									{
										if ( TE.User.recordKeeper.data[ bookID ] )
										{
											linkURL = TE.site.reader.prefix + bookID + "/" + TE.User.recordKeeper.data[ bookID ][ 'lastSeen' ];
											$( readButton ).attr( "href", linkURL );
										}
									}
								} );
							}
							if ( TE.User[ shortName ].fitTitles )
							{
								$( "div.overlay" ).each( function ()
								{
									var thisTitle   = $( this ).find( ".overlay-title" ).text();
									var titleLength = thisTitle.split( "" );
									titleLength     = titleLength.length;
									if ( (titleLength > 40) && (titleLength < 50) )
									{
										$( this ).find( ".overlay-title" ).css( "font-size", ".75em" );
									}
									else if ( (titleLength > 50) )
									{
										$( this ).find( ".overlay-title" ).css( "font-size", ".65em" );
									}
								} );
							}
							if ( TE.User[ shortName ].thumbnailLinks )
							{
								$( "div.overlay" ).each( function ()
								{
									$( this ).on( "mousedown", $.proxy( function (e)
									{
										// Only fire if the div itself is clicked, ignoring children.
										if ( e.target == this )
										{
											// Left Mouse
											if ( (e.which == 1) )
											{
												TE.vbLog( "Left mouse button clicked." );
												if ( TE.User[ shortName ].skipInfo )
												{
													var thisLink = $( this ).find( "a.te_browsetweak_readbutton" ).attr( "href" );
												}
												else
												{
													var thisLink = $( this ).find( "a.overlay-button" ).attr( "href" );
												}
												thisLink = TE.site.baseURL.root + thisLink;
												if ( e.ctrlKey )
												{
													w.open( thisLink );
												}
												else
												{
													w.location.href = thisLink;
												}

											}

											// Middle Mouse
											if ( (e.which == 2) )
											{
												TE.vbLog( "Middle mouse button clicked." );
												if ( TE.User[ shortName ].skipInfo )
												{
													var thisLink = $( this ).find( "a.te_browsetweak_readbutton" ).attr( "href" );
												}
												else
												{
													var thisLink = $( this ).find( "a.overlay-button" ).attr( "href" );
												}
												thisLink = TE.site.baseURL.root + thisLink;
												w.open( thisLink );
											}
										}
										e.preventDefault();
									}, this ) );
								} );
							}
						}
					} );
				}
			}
		};

		//TE.Enhancement.option.main(type,name,description,defaultValue,arguments)
		var opt1 = {
			type         : "toggle",
			name         : "Remove Sidebar",
			description  : "Removes the &quot;random picks&quot; sidebar.",
			defaultValue : false,
			arguments    : false
		};
		var opt2 = {
			type         : "toggle",
			name         : "More Books",
			description  : "Displays one extra book per row.",
			defaultValue : false,
			arguments    : false
		};
		var opt3 = {
			type         : "toggle",
			name         : "Thumbnail Links",
			description  : "Clicking anywhere on the thumbnail image will load the Doujin.",
			defaultValue : false,
			arguments    : false
		};
		var opt4 = {
			type         : "toggle",
			name         : "Skip Info",
			description  : "Skips the book info page and takes you directly to the reader.",
			defaultValue : false,
			arguments    : false
		};
		var opt5 = {
			type         : "toggle",
			name         : "Fit Titles",
			description  : "Makes sure doujin titles fit appropriately.",
			defaultValue : false,
			arguments    : false
		};

		options.push( new TE.Enhancement.option.main( opt1.type, opt1.name, opt1.description, opt1.defaultValue, opt1.arguments ) );
		options.push( new TE.Enhancement.option.main( opt2.type, opt2.name, opt2.description, opt2.defaultValue, opt2.arguments ) );
		options.push( new TE.Enhancement.option.main( opt3.type, opt3.name, opt3.description, opt3.defaultValue, opt3.arguments ) );
		options.push( new TE.Enhancement.option.main( opt4.type, opt4.name, opt4.description, opt4.defaultValue, opt4.arguments ) );
		options.push( new TE.Enhancement.option.main( opt5.type, opt5.name, opt5.description, opt5.defaultValue, opt5.arguments ) );
		TE.Enhancements[ shortName ] = new TE.Enhancement.main( name, description, options, section, incompatible, main );
	})();


	/*******************************************************
	 * Reader Enhancements
	 *******************************************************/
	(function ()
	{
		/*******************************************************
		 * Automatic Repositioning - Reader Enhancement
		 *******************************************************/
		var name, shortName, description, options, section, incompatible, main;
		name         = "Automatic Repositioning";
		shortName    = TE.fn.camelize( name );
		description  = "Automatically scrolls you to the top of the image.";
		options      = [];
		section      = "Reader";
		incompatible = false;
		main         = {
			init   : function ()
			{
				if ( TE.on.reader )
				{
					$.when( TE.status.enhancePage ).done( $.proxy( function ()
					{
						TE.Enhancements.unstickiedHeader.fn.run();
						this.run();
					}, this ) );
				}
			}, run : function ()
			{
				var imgPos = $( "#te_imageBlock" ).offset().top;
				$( "html, body" ).animate( {scrollTop : imgPos}, 300 );
			}
		};

		var opt1 = {
			type         : "enable",
			name         : false,
			description  : false,
			defaultValue : false,
			arguments    : false
		};
		options.push( new TE.Enhancement.option.main( opt1.type, opt1.name, opt1.description, opt1.defaultValue, opt1.arguments ) );
		TE.Enhancements[ shortName ] = new TE.Enhancement.main( name, description, options, section, incompatible, main );
	})();


	(function ()
	{
		/*******************************************************
		 * Seamless Viewing - Reader Enhancement
		 *******************************************************/
		var name, shortName, description, options, section, incompatible, main;
		name         = "Seamless Viewing";
		shortName    = TE.fn.camelize( name );
		description  = `Negates the need to load the entire Tsumino webpage again every time you flip through a Doujin.<br />
			This means faster load times, and not losing sight of the previous page until the instant the new page is loaded.<br />
			Seamless Viewing leaves the previous image in place until the new one is ready.<br />
			Once the new image is ready, you are automatically scrolled up to the top of the image.`;
		options      = [];
		section      = "Reader";
		incompatible = [ "Infinity Scrolling" ];
		main         = {};

		main = {
			replaceKeybinds : function ()
			{
				// Disable default Tsumino Reader Keybinds.
				unsafeWindow.$( document ).off( "keydown" );

				// Use Classic Seamless Viewing keybinds instead.
				$( document ).keydown( $.proxy( function (e)
				{
					var bk  = $.proxy( function ()
					{
						this.changePage( TE.book.prevPage );
					}, this );
					var fwd = $.proxy( function ()
					{
						this.changePage( TE.book.nextPage );
					}, this );
					if ( (!e.ctrlKey) && (!e.altKey) )
					{
						switch (e.which)
						{
							case 87: // w
								w.scrollBy( 0, -100 );
								break;
							case 83: // s
								w.scrollBy( 0, 100 );
								break;
							case 8: //back
							case 37: //left
							case 65: //a
								bk();
								break;
							case 32: //space
							case 13: //enter
							case 39: //right
							case 68: //d
								fwd();
								break;
							default:
								return;
						}
						e.preventDefault();
					}
				}, this ) );
			}, changePage   : function (pageNumber)
			{
				var dfd = jQuery.Deferred();

				function changePageCommon(pageNumber)
				{
					pageNumber = parseInt( pageNumber );
					// Update page and location variables.
					TE.book.currentPage = pageNumber;
					TE.book.prevPage    = pageNumber - 1;
					TE.book.nextPage    = pageNumber + 1;
					if ( TE.book.nextPage > TE.book.totalPages )
					{
						TE.book.nextPage = false;
					}
					if ( TE.book.prevPage <= 0 )
					{
						TE.book.prevPage = false;
					}
					TE.book.currentPageURL = TE.site.reader.prefix + TE.book.id + "/" + TE.book.currentPage;

					// Get the dataURI from the source of loader's hidden image.
					var newImageSrc = $( "#te_loadImage_" + pageNumber ).attr( "src" );
					//var newImageSrc = TE.site.image.prefix + TE.book.id + "/" + TE.book.currentPage;

					// Remove the loader's hidden image.
					$( "#te_readerCurrentImage" ).attr( "src", newImageSrc );

					// Reposition.
					TE.Enhancements.automaticRepositioning.fn.run();

					// If Record Keeper is Enabled.
					if ( TE.User.recordKeeper.enable )
					{
						TE.Enhancements.recordKeeper.fn.update();
					}

					// If Page Jumper is Enabled.
					if ( TE.User.pageJumper.enable )
					{
						$( "#te_pageJumper" ).val( pageNumber );
						$( "#te_pageJumper" ).dropdown( "set selected", pageNumber );
					}

					// Prefetch new pages.
					// TE.fn.prefetch.init( TE.book.currentPage );

					// Update title.
					$( "title" ).text( "Tsumino - " + TE.book.title + " - Page " + TE.book.currentPage );

					// Update links.
					this.updateLinks();

					// Update history and window location.
					if ( (!history.state) || (history.state && history.state.pageNumber != TE.book.currentPage) )
					{
						w.history.pushState( {pageNumber : TE.book.currentPage}, $( "title" ).text(), TE.book.currentPageURL );
					}
					TE.log( "gname", name, "Image " + pageNumber + " has been placed in the reader." );
					unsafeWindow.ga('send', 'pageview', TE.site.reader.prefix + TE.book.id + '/' + TE.book.currentPage);
					dfd.resolve();
				}

				var cpc = changePageCommon.bind( this );
				// Make sure the page is in range first.
				if ( (pageNumber <= TE.book.totalPages) && (pageNumber > 0) )
				{
					if ( TE.status.pagesLoaded[ pageNumber ] == "done" )
					{
						cpc( pageNumber );
					}
					else
					{
						if ( (TE.status.prefetch[ TE.book.id ][ pageNumber ] != "") && (TE.status.prefetch[ TE.book.id ][ pageNumber ] != "working") )
						{
							//TE.status.load = TE.load( pageNumber, TE.status.prefetch[ TE.book.id ][ pageNumber ] );
							// Once the requested page is loaded, continue.
							/*
							$.when( TE.status.load ).then( $.proxy( function ()
							{
								if ( TE.status.pagesLoaded[ pageNumber ] == "done" )
								{
									TE.log( "CPC going" );
									cpc( pageNumber );
								}
								else
								{
									TE.log( "CPC ERROR" );
								}
							}, this ) );
							*/
							$( "body" ).append( `
								<img id="te_loadImage_"` + pageNumber + `" src="/Image/Image/"` + TE.book.id + `/` + pageNumber + `" style="display:none;">
							` );
							$("#te_loadImage_" + pageNumber).load(function ()
							{
								cpc ( pageNumber );
							});
							if ($("#te_loadImage_" + pageNumber).complete)
							{
								$(this).load();
							}
						}
						else
						{
							TE.log( "gname", name, "Prefetch is still initializing..." )
						}
					}
				}
				// If the user requested a page that was less than 1 or greater than the total number of pages, stop.
				else
				{
					if ( pageNumber == false )
					{
						w.location.href = TE.site.book.url + TE.book.id;
					}
					TE.log( "gname", "Seamless Viewing", "Image " + pageNumber + " is out of range and will not be loaded." );
					dfd.resolve();
				}
				return dfd.promise();
			}, updateLinks  : function ()
			{
				TE.vbLog( "gname", name, "Updating links... " );
				// Remove old click binds from links.
				$( "#te_prevButton" ).off( "click" );
				$( "#te_nextButton" ).off( "click" );
				$( "#te_readerCurrentImage" ).off( "click" );

				// Establish updated click binds.
				if ( TE.book.currentPage <= TE.book.totalPages )
				{
					$( "#te_nextButton" ).css( "display", "inline" );
					$( "#te_nextButton" ).click( $.proxy( function ()
					{
						this.changePage( TE.book.nextPage );
					}, this ) );
					$( "#te_readerCurrentImage" ).click( $.proxy( function ()
					{
						this.changePage( TE.book.nextPage );
					}, this ) );
				}
				if ( TE.book.currentPage == TE.book.totalPages )
				{
					$( "#te_nextButton" ).css( "display", "none" );
					$( "#te_readerCurrentImage" ).click( $.proxy( function ()
					{
						this.changePage( TE.book.nextPage );
					}, this ) );
				}

				if ( TE.book.currentPage > 1 )
				{
					$( "#te_prevButton" ).css( "display", "inline" );
					$( "#te_prevButton" ).click( $.proxy( function ()
					{
						this.changePage( TE.book.prevPage );
					}, this ) );

				}
				else
				{
					$( "#te_prevButton" ).css( "display", "none" );
				}

				$( "#te_currentPage" ).html( "<a href='" + TE.book.currentPageURL + "'>" + TE.book.currentPage + "</a>" );
			}, init         : function ()
			{
				if ( TE.on.reader )
				{
					$.when( TE.status.enhancePage ).done( $.proxy( function ()
					{
						w.history.replaceState( {pageNumber : TE.book.currentPage}, $( "title" ).text(), TE.book.currentPageURL );

						// Allow history navigation to work with Seamless Viewing.
						$( w ).on( "popstate", $.proxy( function ()
						{
							if ( history.state )
							{
								this.changePage( history.state.pageNumber );
							}
						}, this ) );

						TE.log( "gname", name, "Initializining..." );

						// Replace default Tsumino reader keybinds with Enhanced Seamless Viewing keybinds.
						this.replaceKeybinds();

						// Automatic Repositioning.
						TE.Enhancements.automaticRepositioning.fn.run();

						// Unstickied Header.
						TE.Enhancements.unstickiedHeader.fn.run();

						// Remove default Tsumino doujin navigation links.
						$( "#te_prevButton" ).attr( "href", "javascript:;" );
						$( "#te_nextButton" ).attr( "href", "javascript:;" );
						$( "#te_imageLink" ).attr( "href", "javascript:;" );

						// Prepare Prefetch
						TE.status.prefetch[ TE.book.id ] = {};
						for (var i = 1 ; i <= TE.book.totalPages ; i++)
						{
							TE.status.prefetch[ TE.book.id ][ i ] = "";
						}
						TE.fn.prefetch.init( TE.book.currentPage );

						// Update doujin navigation links.
						this.updateLinks();
						$( "body" ).append( "<img id='te_loadImage_" + TE.book.currentPage + "' style='display:none;'>" );

						// "Cache" the first image that loads for later.
						var originImage    = new Image();
						originImage.onload = function ()
						{
							var canvas    = document.createElement( "canvas" );
							canvas.width  = this.naturalWidth;
							canvas.height = this.naturalHeight;
							canvas.getContext( "2d" ).drawImage( this, 0, 0 );
							var newSrc = canvas.toDataURL( "image/jpeg" );
							$( "#te_loadImage_" + TE.book.currentPage ).attr( "src", newSrc );
						};
						originImage.src    = $( "#te_readerCurrentImage" ).attr( "src" );

					}, this ) );
				}
				else if ( TE.on.auth )
				{
					// Reserved
				}
			}
		};

		//TE.Enhancement.option.main(type,name,description,defaultValue,arguments)
		var opt1 = {
			type         : "enable",
			name         : false,
			description  : false,
			defaultValue : false,
			arguments    : false
		};
		var opt2 = {
			type         : "toggle",
			name         : "Show Messages",
			description  : "Displays loading messages while preparing images for display.",
			defaultValue : true,
			arguments    : false
		};

		options.push( new TE.Enhancement.option.main( opt1.type, opt1.name, opt1.description, opt1.defaultValue, opt1.arguments ) );
		//options.push(new TE.Enhancement.option.main(opt2.type,opt2.name,opt2.description,opt2.defaultValue,opt2.arguments));

		TE.Enhancements[ shortName ] = new TE.Enhancement.main( name, description, options, section, incompatible, main );
	})();


	(function ()
	{
		/*******************************************************
		 * Page Jumper - Reader Enhancement
		 *******************************************************/
		var name, shortName, description, options, section, incompatible, main;
		name         = "Page Jumper";
		shortName    = TE.fn.camelize( name );
		description  = "Adds a dropdown box to the Reader that lets you skip directly to a page.";
		options      = [];
		section      = "Reader";
		incompatible = false;
		main         = {};

		main = {
			init : function ()
			{
				if ( TE.on.reader )
				{
					$.when( TE.status.enhancePage ).done( $.proxy( function ()
					{
						$( "#te_readerPagination" ).after( `
							<h1 style='display:inline;'>Jump to page: </h1>
							<select class='ui compact search dropdown' id='te_pageJumper'></select><br />
							` );
						for (var i = 1 ; i <= TE.book.totalPages ; i++)
						{
							$( "#te_pageJumper" ).append( "<option value='" + i + "'>" + i + "</option>" );
						}
						$( "#te_pageJumper" ).val( TE.book.currentPage );

						$( "#te_pageJumper" ).change( $.proxy( function ()
						{
							// Seamless Viewing Compatibility
							if ( TE.User.seamlessViewing.enable )
							{
								var pageNumber = parseInt( $( "#te_pageJumper" ).val() );

								$.when( TE.fn.prefetch.init( pageNumber ) ).then( function ()
								{
									TE.Enhancements.seamlessViewing.fn.changePage( pageNumber );
								} );

							}
							// Vanilla Tsumino
							else
							{
								w.location.href = TE.site.reader.url + TE.book.id + "/" + pageNumber;
							}
						}, this ) );
						$( "#te_pageJumper" ).dropdown();
					}, this ) );
				}
			}
		};

		//TE.Enhancement.option.main(type,name,description,defaultValue,arguments)
		var opt1 = {
			type         : "enable",
			name         : false,
			description  : false,
			defaultValue : false,
			arguments    : false
		};
		options.push( new TE.Enhancement.option.main( opt1.type, opt1.name, opt1.description, opt1.defaultValue, opt1.arguments ) );

		TE.Enhancements[ shortName ] = new TE.Enhancement.main( name, description, options, section, incompatible, main );
	})();


	(function ()
	{
		/*******************************************************
		 * Infinity Scrolling - Reader Enhancement
		 *******************************************************/
		var name                  = "Infinity Scrolling",
			shortName             = TE.fn.camelize( name ),
			description           = "Scroll down to load images.",
			options = [], section = "Reader",
			incompatible          = [ "Seamless Viewing" ],
			main                  = {};

		main = {
			init : function ()
			{
				if ( TE.on.reader )
				{
					$.when( TE.status.enhancePage ).done( function ()
					{

					} );
				}
			}

		};

		//TE.Enhancement.option.main(type,name,description,defaultValue,arguments)
		var opt1 = {
			type         : "enable",
			name         : false,
			description  : false,
			defaultValue : false,
			arguments    : false
		};
		options.push( new TE.Enhancement.option.main( opt1.type, opt1.name, opt1.description, opt1.defaultValue, opt1.arguments ) );

		//TE.Enhancements[shortName] = new TE.Enhancement.main(name,description,options,section,incompatible,main);
	})();

	(function ()
	{
		/*******************************************************
		 * Automatic Repositioning - Reader Enhancement
		 *******************************************************/
		var name, shortName, description, options, section, incompatible, main;
		name         = "Automatic Update Installation";
		shortName    = TE.fn.camelize( name );
		description  = `If your version of Tsumino Enhanced is out of date, it will automatically attempt to update.<br />
						You must still accept your browser's prompt to install the update.`;
		options      = [];
		section      = "TsuminoEnhanced";
		incompatible = false;
		main         = {
			init   : function ()
			{
				// No INIT.
			}
		};

		var opt1 = {
			type         : "enable",
			name         : false,
			description  : false,
			defaultValue : false,
			arguments    : false
		};
		options.push( new TE.Enhancement.option.main( opt1.type, opt1.name, opt1.description, opt1.defaultValue, opt1.arguments ) );
		TE.Enhancements[ shortName ] = new TE.Enhancement.main( name, description, options, section, incompatible, main );
	})();



	/*************************************************************************************
	 * Tsumino Enhanced Settings Page
	 *************************************************************************************/
	TE.settings = {
		render    : function ()
		{
			if ( !TE.User.readNews )
			{
				TE.User.readNews = {};
			}

			$( "#te_config_modal" ).html( "<div id='te_settings' style='font-size: 1.4em;'></div>" );
			$( "#te_config_modal" ).prepend( `
				<div class="header"><h1><span class="te_mainColor">Tsumino Enhanced</span>
				<span id="te_version" class="small">` + TE.version + `</span></h1></div>
				` );
			if ( !TE.User.tsuminoEnhanced.upToDate )
			{
				var installLocation = TE.updateLocation;
				installLocation     = installLocation.replace( "/scripts/", "/install/" );
				installLocation     = installLocation + ".user.js";
				//$("#te_version").append("&nbsp;&nbsp;-&nbsp;&nbsp;<a style='color:#ff0000; text-decoration:none;' href='" + installLocation + "'>Update!</a>");
				$( "#te_version" ).after( `&nbsp;&nbsp;
					<div id="te_updateButton" class="ui big labeled button" tabindex="0">
						<div class="ui red button"><i class="upload icon"></i>Update!</div><a class="ui basic red left pointing label">`
										  + TE.User.tsuminoEnhanced.latestVersion + `</a></div>` );
				//$("#te_config_modal").append("<a id='te_secretUpdateLink' style='display:none;' href='" + installLocation + "'>Update!</a>")
				$( "#te_updateButton" ).click( function ()
				{
					$( "body" ).append( `
					<div id="te_refresh_modal" class="ui basic modal">
					<div class="header" style="font-size: 3em;"><i class="upload icon"></i> Updating Tsumino Enhanced...</div>
					<div class="image content"><div class="image"><i class="refresh icon"></i></div>
					<div class="description" style="font-size: 1.5em;">
						<p>You will be prompted to install the update in just a moment.</p>
						<p>In order for the update to take effect, you must refresh this page after it has finished installing.</p>
						<p>Do you wish to refresh the page now?</p>
						</div></div>
					<div class="actions">
						<button id="te_refreshPageButton" class="massive fluid gray ui button disabled">Refresh</button>
					</div></span>` );
					$( "#te_config_modal" ).modal( "hide" );
					$( "#te_refresh_modal" ).modal();
					$( "#te_refresh_modal" ).modal( "show" );
					$( "#te_refresh_modal" ).modal( "refresh" );
					setTimeout( function ()
					{
						w.location.href = installLocation;
						$("#te_refreshPageButton").removeClass("disabled");
						$("#te_refreshPageButton").removeClass("gray");
						$("#te_refreshPageButton").addClass("green");
					}, 5000 );
					$( "#te_refreshPageButton" ).click( function ()
					{
						if(!$("#te_refreshPageButton").hasClass("disabled"))
						{
							w.location.reload();
						}
					} );
				} );

				$( "#te_updateButton" ).popup( {
					title : 'Click here to begin the update process.'
				} );
			}

			// Settings page navigation structure.
			//$("#te_settings").prepend("<div id='te_tabContainer' class='te_configTab'><nav><ul><li id='tab_generalEnhancements'><a href='javascript:;'>General</a></li><li id='tab_browsingEnhancements'><a href='javascript:;'>Browsing</a></li><li id='tab_readerEnhancements'><a href='javascript:;'>Reader</a></li><li id='tab_teNews'><a href='javascript:;'>TE News</a></li><li id='tab_searchEnhancements'><a href='javascript:;'>Search</a></li></ul></nav></div>");

			$( "#te_settings" ).prepend( `<div id="te_settings_tabs" class="ui top attached inverted large tabular menu"></div>` );
			$( "#te_settings_tabs" ).append( `<a class="item active" data-tab="generalEnhancements">General Enhancements</a>` );
			$( "#te_settings_tabs" ).append( `<a class="item" data-tab="browsingEnhancements">Browsing Enhancements</a>` );
			$( "#te_settings_tabs" ).append( `<a class="item" data-tab="readerEnhancements">Reader Enhancements</a>` );
			$( "#te_settings_tabs" ).append( `<a class="item" data-tab="TE_options">TE Configuration</a>` );
			$( "#te_settings_tabs" ).append( `<a class="item" data-tab="teAbout">About</a>` );

			$( "#te_settings" ).append( `
				<div id="te_settings_tab_generalEnhancements" class="ui bottom attached inverted tab segment active" data-tab="generalEnhancements"></div>
			` );
			$( "#te_settings" ).append( `
				<div id="te_settings_tab_browsingEnhancements" class="ui bottom attached inverted tab segment" data-tab="browsingEnhancements"></div>
			` );
			$( "#te_settings" ).append( `
				<div id="te_settings_tab_readerEnhancements" class="ui bottom attached inverted tab segment" data-tab="readerEnhancements"></div>
			` );
			$( "#te_settings" ).append( `
				<div id="te_settings_tab_TE_options" class="ui bottom attached inverted tab segment" data-tab="TE_options"></div>
			` );
			$( "#te_settings" ).append( `
				<div id="te_settings_tab_teAbout" class="ui bottom attached tab segment inverted" data-tab="teAbout"></div>
			` );

			var tobyAvatar = "";
			var tobyCard   = `

<div class="ui centered grid">
	<div class="seven wide column">
		<div class="ui fluid blue card">
			<div class="content">
				<h1 class="header">Tsumino Enhanced</h1>
				<div class="description">
					<table class="ui single line definition blue left aligned table">
						<tbody>
						<tr>
							<td class="left aligned">Current Version</td>
							<td>` + TE.version + `</td>
						</tr>
						<tr>
							<td class="left aligned">Latest Version</td>
							<td>` + TE.User.tsuminoEnhanced.latestVersion + `</td>
						</tr>
						<tr>
							<td>Changelog</td>
							<td><button id="te_tobyLinks_teChangelog" data-content="Check out the list of changes."
							class="ui large circular inverted blue icon button teTooltip"><i class="external icon"></i></button></td>
						</tr>
						<tr>
							<td>Requests &amp; Feedback</td>
							<td><button id="te_tobyLinks_teRaF" data-content="Influence TE's development."
							class="ui large circular inverted blue icon button teTooltip"><i class="external icon"></i></button></td>
						</tr>
						<tr class="top aligned">
							<td class="left aligned collapsing">Library Dependencies</td>
							<td>
								<ul class="ui list">
									<li class="teTooltip" data-content="General purpose library.">jQuery 2.1.14</li>
									<li class="teTooltip" data-content="Gives TE a nicer UI.">SemanticUI 2.1.8</li>
									<li class="teTooltip" data-content="Event management library.">Bean 1.0.15</li>
									<li class="teTooltip" data-content="A more versatile animation library.">Velocity 1.2.3</li>
								</ul>
							</td>
						</tr>
						<tr class="top aligned">
							<td class="left aligned">@grant Permissions</td>
							<td>
								<ul class="ui list">
									<li class="teTooltip" data-content="Used for saving your settings.">GM_setValue</li>
									<li class="teTooltip" data-content="Used for reading your settings.">GM_getValue</li>
									<li class="teTooltip" data-content="Used for cleaning up depreciated values.">GM_deleteValue</li>
									<li class="teTooltip" data-content="Used for opening links in new tabs.">GM_openInTab</li>
									<li class="teTooltip" data-content="Used for overriding Tsumino hotkeys.">unsafeWindow</li>
								</ul>
							</td>
						</tr>
						</tbody>
					</table>
				</div>
			</div>
		</div>
	</div>
	<div class="five wide column">
		<div class="ui purple card" style="width:280px;">
			<div class="image">
				<img src="` + tobyAvatar + `" />
			</div>
			<div class="content">
				<div class="header">Toby</div>
				<div class="meta">
					<a>Web Developer</a>
				</div>
				<div class="description">
					<p>I'm the guy that made this.</p>
				</div>
			</div>
			<div class="extra content">
				<button id="te_tobylinks_home" data-content="Visit my project blog!" class="ui large circular inverted violet icon button teTooltip">
					<i class="large home icon"></i>
				</button>
				<button id="te_tobylinks_github" data-content="Take a look at my GitHub profile!" class="ui large circular black icon button teTooltip">
					<i class="large github alternate icon"></i>
				</button>
				<button id="te_tobylinks_openuserjs" data-content="Check out my other userscripts!" class="ui large circular grey icon button teTooltip">
					<i class="large code icon"></i>
				</button>
				<button id="te_tobylinks_skype" data-content="Chat with me on Skype!" class="ui large circular inverted blue icon button teTooltip">
					<i class="large skype icon"></i>
				</button>
				<button id="te_tobylinks_tsuminopm" data-content="Send me a PM on the Tsumino Forums!" class="ui large circular inverted pink icon button teTooltip">
					<i class="large mail outline icon"></i>
				</button>
			</div>
		</div>
	</div>
</div>
			`;

			$( "#te_settings_tab_teAbout" ).append( tobyCard );

			$( "#te_tobyLinks_teChangelog" ).click( function ()
			{
				GM_openInTab( "http://codingtoby.com/category/userscripts/tsumino-enhanced/te-updates/" );
			} );
			$( "#te_tobyLinks_teRaF" ).click( function ()
			{
				GM_openInTab( "http://codingtoby.com/userscripts/tsumino-enhanced/requests-and-feedback/#respond" );
			} );
			$( "#te_tobylinks_home" ).click( function ()
			{
				GM_openInTab( "http://codingtoby.com/" );
			} );
			$( "#te_tobylinks_github" ).click( function ()
			{
				GM_openInTab( "https://github.com/tobiaskelmandia" );
			} );
			$( "#te_tobylinks_openuserjs" ).click( function ()
			{
				GM_openInTab( "https://openuserjs.org/users/Tobias.Kelmandia/scripts" );
			} );
			$( "#te_tobylinks_skype" ).click( function ()
			{
				w.location.href = "skype:tobias_kelmandia?chat";
			} );
			$( "#te_tobylinks_tsuminopm" ).click( function ()
			{
				GM_openInTab( "http://www.tsumino.com/Forum/ucp.php?i=pm&mode=compose&u=191" );
			} );

			// Initialize Tooltips
			$( ".teTooltip" ).popup();

			// Populate Sections.
			for (var key in TE.Enhancements)
			{
				if ( TE.Enhancements.hasOwnProperty( key ) )
				{
					var obj = TE.Enhancements[ key ];

					if ( obj[ "section" ] != false )
					{
						// Determine which section to append to.
						var sectionID = "";
						if ( obj[ "section" ] == "General" )
						{
							sectionID = "#te_settings_tab_generalEnhancements";
						}
						else if ( obj[ "section" ] == "Browsing" )
						{
							sectionID = "#te_settings_tab_browsingEnhancements";
						}
						else if ( obj[ "section" ] == "Reader" )
						{
							sectionID = "#te_settings_tab_readerEnhancements";
						}
						else if ( obj[ "section" ] == "TsuminoEnhanced" )
						{
							sectionID = "#te_settings_tab_TE_options";
						}

						// Append the Enhancement's options group to the section.
						$( sectionID ).append( "<div id='" + obj[ "shortName" ] + "_group' class='te_optionGroup'></div>" );

						// Add the description.
						if ( obj[ "description" ] != false )
						{
							$( "#" + obj[ "shortName" ] + "_group" ).append( "<div class='te_optionDescription'>" + obj[ "description" ] + "</div>" );
						}

						// Add the primary options area.
						$( "#" + obj[ "shortName" ] + "_group" ).append( "<div id='te_options_" + obj[ "shortName" ] + "'></div>" );

						var noEnable = true;
						if ( obj[ "options" ] != false )
						{
							// Display all options.
							for (var oKey in obj[ "options" ])
							{
								if ( obj[ "options" ].hasOwnProperty( oKey ) )
								{
									var option = obj[ "options" ][ oKey ];
									// Write the Enable option.
									if ( option[ "type" ] == "enable" )
									{
										$( "#" + obj[ "shortName" ] + "_group" ).prepend( `
											<div class="ui middle aligned toggle checkbox">
												<input id="tes_` + obj[ "shortName" ] + `_enable" name="tes_` + obj[ "shortName" ] + `_enable"  type="checkbox">
												<label for="tes_` + obj[ "shortName" ] + `_enable">
													<a class="ui huge blue label">` + obj[ "name" ] + `</a>
												</label>
											 </div>` );


										$( "#enhancement_header_" + obj[ "shortName" ] ).click( {obj : obj}, function (event)
										{
											TE.log( $( "#tes_" + event.data.obj[ "shortName" ] + "_enable" ) );
											if ( $( "#tes_" + event.data.obj[ "shortName" ] + "_enable" ).prop( "checked" ) == true )
											{
												$( "#tes_" + event.data.obj[ "shortName" ] + "_enable" ).prop( "checked", false );
											}
											else
											{
												$( "#tes_" + event.data.obj[ "shortName" ] + "_enable" ).prop( "checked", true );
											}
										} );
										noEnable = false;
									}
									// Write Toggle Options.
									if ( option[ "type" ] == "toggle" )
									{
										$( "#te_options_" + obj[ "shortName" ] ).append( `<br />
											<div class="ui middle aligned toggle checkbox">
												<input id="tes_` + obj[ "shortName" ] + `_` + option[ "shortName" ] + `"
												name="tes_` + obj[ "shortName" ] + `_` + option[ "shortName" ] + `" type="checkbox" class="te_subOption" />
												<label for="tes_` + obj[ "shortName" ] + `_` + option[ "shortName" ] + `">
													<a class="ui huge blue label">` + option[ "name" ] + `</a>
												</label>
											</div>` );

										// Write the option's description.
										if ( option[ "description" ] != false )
										{
											$( "#" + obj[ "shortName" ] + "_optionContainer_" + option[ 'shortName' ] ).append( "<br />"
																																+ option[ "description" ] );
										}
									}
								}
							}
						}

						// If there was no "enable" option found:
						if ( noEnable )
						{
							// Display the title without a switch.
							$( "#" + obj[ "shortName" ] + "_group" ).prepend( "<h2 class='te_enhancementName'>" + obj[ "name" ] + "</h2>" );

							// Apply user settings.
							if ( typeof TE.User[ obj[ "shortName" ] ] !== "undefined" )
							{
								for (var oKey in obj[ "options" ])
								{
									var option = obj[ "options" ][ oKey ];
									if ( option[ "type" ] == "toggle" )
									{
										$( "#tes_" + obj[ "shortName" ] + "_" + option[ 'shortName' ] )
											.prop( "checked", TE.User[ obj[ "shortName" ] ][ option[ 'shortName' ] ] );
									}
								}
							}
						}
						else
						{
							// Apply user settings.
							if ( typeof TE.User[ obj[ "shortName" ] ] !== "undefined" )
							{
								for (var oKey in obj[ "options" ])
								{
									var option = obj[ "options" ][ oKey ];
									if ( TE.User[ obj[ "shortName" ] ][ "enable" ] == true )
									{
										if ( option[ "type" ] == "enable" )
										{
											$( "#tes_" + obj[ "shortName" ] + "_" + option[ 'shortName' ] ).prop( "checked", true );
										}
										else
										{
											if ( option[ "type" ] == "toggle" )
											{
												$( "#tes_" + obj[ "shortName" ] + "_" + option[ 'shortName' ] )
													.prop( "checked", TE.User[ obj[ "shortName" ] ][ option[ 'shortName' ] ] );
											}
										}
									}
									else
									{
										if ( option[ "type" ] != "enable" )
										{
											$( "#tes_" + obj[ "shortName" ] + "_" + option[ 'shortName' ] ).prop( "checked", false );
											$( "#tes_" + obj[ "shortName" ] + "_" + option[ 'shortName' ] ).prop( "disabled", "disabled" );
										}
									}
								}
							}
							else
							{
								for (var oKey in obj[ "options" ])
								{
									if ( obj[ "options" ].hasOwnProperty( oKey ) )
									{
										var option = obj[ "options" ][ oKey ];
										if ( option[ "type" ] != "enable" )
										{
											if ( !$( "#tes_" + obj[ "shortName" ] + "_" + option[ 'shortName' ] ).prop( "disabled" ) )
											{
												$( "#tes_" + obj[ "shortName" ] + "_" + option[ 'shortName' ] ).prop( "disabled", "disabled" );
											}
										}
									}
								}
							}

							// Apply default values to options when enabling an Enhancement.
							$( "#tes_" + obj[ "shortName" ] + "_enable" ).change( {obj : obj}, function (event)
							{
								if ( $( "#tes_" + event.data.obj[ "shortName" ] + "_enable" ).prop( "checked" ) == true )
								{
									for (var oKey in event.data.obj[ "options" ])
									{
										if ( event.data.obj[ "options" ].hasOwnProperty( oKey ) )
										{
											var option = event.data.obj[ "options" ][ oKey ];
											if ( option[ "type" ] == "toggle" )
											{
												$( "#tes_" + event.data.obj[ "shortName" ] + "_" + option[ 'shortName' ] ).removeProp( "disabled" );
												$( "#tes_" + event.data.obj[ "shortName" ] + "_"
												   + option[ 'shortName' ] ).prop( "checked", option[ 'defaultValue' ] );
											}
										}
									}
								}
								// Disable Enhancement options unless the enhancement is enabled.
								else
								{
									for (var oKey in event.data.obj[ "options" ])
									{
										if ( event.data.obj[ "options" ].hasOwnProperty( oKey ) )
										{
											var option = event.data.obj[ "options" ][ oKey ];
											if ( option[ "type" ] != "enable" )
											{
												if ( option[ "type" ] == "toggle" )
												{
													$( "#tes_" + event.data.obj[ "shortName" ] + "_"
													   + option[ 'shortName' ] ).prop( "checked", false );
												}
												$( "#tes_" + event.data.obj[ "shortName" ] + "_"
												   + option[ 'shortName' ] ).prop( "disabled", "disabled" );
											}
										}
									}
								}
							} );
						}
						// List any incompatibilities.
						if ( obj[ "incompatible" ] != false )
						{
							$( "#" + obj[ "shortName" ] + "_group" ).append( "<br /><div class='te_en_incompatible' id='" + obj[ "shortName" ]
																			 + "_incompatible'>This Enhancement is incompatible with: </div>" );
							var punct = "";
							for (var i = 0 ; i < obj[ "incompatible" ].length ; i++)
							{
								$( "#" + obj[ "shortName" ] + "_incompatible" ).append( "<span class='te_enhancementColor'>"
																						+ obj[ "incompatible" ][ i ] + "</span>" );
								if ( i + 1 == obj[ "incompatible" ].length )
								{
									punct = ".";
								}
								else
								{
									punct = ", ";
								}
								$( "#" + obj[ "shortName" ] + "_incompatible" ).append( punct );
								$( "#" + obj[ "shortName" ] + "_group" ).append( "<br /><br />" );
							}
						}
						else
						{
							$( "#" + obj[ "shortName" ] + "_group" ).append( "<br /><br />" );
						}
					}
				}
			}


			//$("#te_settingsBody").append("<div id='searchEnhancements' class='te_options'></div>");
			//$("#te_settings").append("<div id='forumEnhancements' class='te_options'></div>");

			// Create Buttons
			$( "#te_settings" ).append( "<br /><br /><div id='te_buttonContainer'></div><br /><br />" );
			$( "#te_buttonContainer" ).append( "<a id='te_saveAndCloseButton' class='book-read-button'>Save &amp; Reload</a>&nbsp;&nbsp;" );
			$( "#te_saveAndCloseButton" ).click( $.proxy( function ()
			{
				this.save();
				location.reload();
			}, this ) );
			$( "#te_buttonContainer" ).append( "<a id='te_applySettingsButton' class='book-read-button'>Apply</a>&nbsp;&nbsp;" );
			$( "#te_applySettingsButton" ).click( $.proxy( function ()
			{
				this.save();
			}, this ) );
			$( "#te_buttonContainer" ).append( "<a id='te_cancelSettingsButton' class='book-read-button'>Cancel</a>" );
			$( "#te_cancelSettingsButton" ).click( $.proxy( function ()
			{
				this.remove();
			}, this ) );


			$( ".menu .item" ).tab();
			$( "#te_config_modal" ).modal(
				{
					onVisible      : function ()
					{
						$( "#te_config_modal" ).modal( "refresh" );
					},
					observeChanges : true
				} );

			$( ".menu .item" ).click( function ()
			{
				$( "#te_config_modal" ).modal( "refresh" );
			} );
		}, remove : function ()
		{
			$( "#te_config_modal" ).modal( "hide" );
		}, save   : function ()
		{
			// Find all enhancement groups within the main settings area.
			var enhancementGroups = $( "#te_settings" ).find( "div[id*='_group']" ), enhancementSettings = {};

			// Loop through the groups to get all the individual settings.
			for (var i = 0 ; i < enhancementGroups.length ; i++)
			{
				var thisEnhancement                = enhancementGroups[ i ];
				var thisEnhName                    = $( thisEnhancement ).attr( "id" );
				thisEnhName                        = thisEnhName.replace( "_group", "" );
				var thisEnhSettings                = $( thisEnhancement ).find( "*[id*='tes_" + thisEnhName + "_']" );
				enhancementSettings[ thisEnhName ] = {};
				for (var es = 0 ; es < thisEnhSettings.length ; es++)
				{
					var thisEnSetting = $( thisEnhSettings )[ es ], thisEnSettingName = $( thisEnSetting ).attr( "id" );
					thisEnSettingName = thisEnSettingName.replace( "tes_" + thisEnhName + "_", "" );
					if ( $( thisEnSetting ).prop( "tagName" ) == "INPUT" )
					{
						if ( $( thisEnSetting ).prop( "type" ) == "checkbox" )
						{
							enhancementSettings[ thisEnhName ][ thisEnSettingName ] = $( thisEnSetting ).prop( "checked" );
						}
					}
				}
			}
			$.extend( true, TE.User, enhancementSettings );
			TE.updateSettings();
		}
	};

	/*************************************************************************************
	 * Tsumino Enhanced Initialization Code
	 *************************************************************************************/
		// Initialization.
	TE.init = function ()
	{
		TE.User.seamlessViewing.enable = false;
		TE.User.automaticRepositioning.enable = false;
		TE.User.pageJumper.enable = false;
		TE.User.unstickiedHeader.enable = false;

		$.when(TE.fn.checkForUpdates()).then(function()
		{
			if(!TE.User.tsuminoEnhanced.upToDate)
			{
				if(TE.User.automaticUpdateInstallation)
				{
					if(TE.User.automaticUpdateInstallation.enable)
					{
						GM_openInTab(TE.installLocation);
					}
				}
			}
		});

		// Output initializating messages to the console.
		var debugState = "Disabled";
		if ( TE.config.debug )
		{
			if ( TE.config.verboseDebug )
			{
				debugState = "Verbose";
			}
			else
			{
				debugState = "Standard";
			}
		}

		// Check which Enhancements the user has enabled.
		var enabledEnhancements = [], eeLongNames = "", autoOn = [];
		for (var key in TE.User)
		{
			if ( TE.User.hasOwnProperty( key ) )
			{
				var obj = TE.User[ key ];
				for (var prop in obj)
				{
					if ( obj.hasOwnProperty( prop ) )
					{
						if ( (prop == "enable") && (obj[ prop ] == true) )
						{
							// Add enabled Enhancements to the appropriate array.
							if ( typeof TE.Enhancements[ key ] !== "undefined" )
							{
								enabledEnhancements.push( key );
								eeLongNames = eeLongNames + "[X] " + TE.Enhancements[ key ].name + "\r\n";
							}
						}
						else if ( (prop == "enable") && (obj[ prop ] == false) )
						{
							eeLongNames = eeLongNames + "[ ] " + TE.Enhancements[ key ].name + "\r\n";
						}
					}
				}
			}
		}
		// Check for automatic enhancements.
		for (var key in TE.Enhancements)
		{
			if ( TE.Enhancements.hasOwnProperty( key ) )
			{
				var obj = TE.Enhancements[ key ];
				autoOn.push( obj[ 'shortName' ] );
				for (var prop in obj)
				{
					if ( obj.hasOwnProperty( prop ) )
					{
						if ( prop == "options" )
						{
							for (var optNum in obj[ 'options' ])
							{
								for (var opt in obj[ 'options' ][ optNum ])
								{
									if ( (opt == "type") && (obj[ 'options' ][ optNum ][ opt ] == "enable") )
									{
										var thisIndex = autoOn.indexOf( obj[ 'shortName' ] );
										if ( thisIndex > -1 )
										{
											autoOn.splice( thisIndex, 1 );
										}
									}
								}
							}
						}
					}
				}
			}
		}
		// Enable automatic enhancements.
		for (i = 0 ; i < autoOn.length ; i++)
		{
			enabledEnhancements.push( autoOn[ i ] );
			eeLongNames = eeLongNames + "[X] " + TE.Enhancements[ autoOn[ i ] ].name + "\r\n";
		}



		// Output initialization messages.
		TE.log( "gname", TE.name,
			"Version:	" + TE.version,
			"Latest:		" + TE.User.tsuminoEnhanced.latestVersion,
			"Up2Date:	" + TE.User.tsuminoEnhanced.upToDate,
			"Debugging:	" + debugState,
			"Enhancements:", eeLongNames );
		TE.vbLog( "gname", TE.name, "Current Settings:", TE.User );
		//TE.vbLog( "gname", "TE.site", TE.site );
		TE.vbLog( "gname", "TE.on", TE.on );
		TE.vbLog( "gname", "TE.Enhancements", TE.Enhancements );

		// Set up TE.status.enhancePage for Enhancements that require it to run.
		TE.status.enhancePage = TE.enhancePage();

		// Initialize all enabled Enhancements.
		for (var i = 0 ; i < enabledEnhancements.length ; i++)
		{
			if ( typeof TE.Enhancements[ enabledEnhancements[ i ] ] !== "undefined" )
			{
				TE.Enhancements[ enabledEnhancements[ i ] ].fn.init();
			}
		}
	};

	var tempMyLocation = TE.myLocation;
	// Initialize Tsumino Enhanced.
	if ( !TE.on.forum )
	{
		TE.init();
	}
	else if((TE.on.forum) && ( (tempMyLocation.indexOf("#p1472") != -1) || (tempMyLocation.indexOf("p=1472") != -1) ) )
	{
		var messageID = TE.randomString();
		$(document).ready(function()
		{
			$("#p1472").after(`
			<div class="post" id="`+messageID+`">
				<div class="inner">
					<div class="postbody">
						<span style="font-size: 2em">A message from <span style="color:`+TE.ui.mainColor+`">Tsumino Enhanced</span></span>
						<hr />
						<div class="content">
							If you're here, that means that Tsumino has updated their detection script again.<br />
							The good news is that you don't appear to be banned yet.<br />
							If you wish to continue using Tsumino Enhanced, make sure you're using the latest version.<br />
							You can still manually update over on
							<a href="https://openuserjs.org/scripts/Tobias.Kelmandia/Tsumino_Enhanced">OpenUserJS</a>
							by clicking the install button on that page.<br />
							If you're using the latest version and still get taken here, you can disable TE until the new
							detection method has been defeated.<br />
							You can find the latest news for Tsumino Enhanced over on the
							<a href="http://codingtoby.com/" target="_blank">development blog</a>.<br />
							If you're interested, you can also read
							<a href="http://codingtoby.com/userscripts/tsumino-enhanced/regarding-tsuminos-stance-on-userscripts/"
								target="_blank">my response to Tweety's post</a>.<br />
						</div>
					</div>
				</div>
			</div>
		`);
		});
		$(w).load(function()
		{
			var scrollTo = $( "#" + messageID ).offset().top;
			scrollTo -= ($(".nav-tabs").height() + 10);
			$( "html, body" ).animate( {scrollTop : scrollTo}, 300 );
		});
	}

})( typeof window === "undefined" ? this : window, this.jQuery );