RandomClown / Anime Lighter

// ==UserScript==
// @name         Anime Lighter
// @namespace    horc.net
// @version      4.0.35
// @description  Filter for awesome anime trackers
// @author       RandomClown @ HoRC
// @license      MIT License
// @homepage     http://horc.bitbucket.org
// @icon         https://static.horc.net/app/anime-lighter/img/Logos/Anime%20Lighter.png?app=anime-lighter
// @grant        none
// @require      https://ajax.googleapis.com/ajax/libs/jquery/3.0.0/jquery.min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/react/15.1.0/react.min.js
// @require      https://cdnjs.cloudflare.com/ajax/libs/react/15.1.0/react-dom.min.js
// @require      http://static.horc.net/app/anime-lighter/js/common.js
// @match        http://horriblesubs.info
// @match        http://www.nyaa.se/*
// @match        https://openuserjs.org/scripts/RandomClown/Anime_Lighter
// ==/UserScript==
//
//
//     Source code available: https://bitbucket.org/horc/anime-lighter/
//
//

//////////////////////////////////////////////////
////////          Const & Styles          ////////

const HAL_URL_MAIN = 'https://openuserjs.org/scripts/RandomClown/Anime_Lighter';
const HAL_URL_GIT = 'https://bitbucket.org/horc/anime-lighter/overview';

const HAL_SAD_FACE = ['=A=', 'óuò', 'T~T', '(゚Д゚;)'];

const STYLESHEET = `
	.horc-lighter .horc-button:not([disabled]):hover {
		box-shadow: 0 5px 11px 0 rgba(0, 0, 0, 0.18), 0 4px 15px 0 rgba(0, 0, 0, 0.15);
	}

	.horc-lighter .horc-button:not([disabled]):hover .horc-button-color {
		opacity: 0.75;
	}

	.material-icons {
		font-family: 'Material Icons';
		font-weight: normal;
		font-style: normal;
		font-size: 24px; /* Preferred icon size */
		display: inline-block;
		line-height: 1;
		text-transform: none;
		letter-spacing: normal;
		word-wrap: normal;
		white-space: nowrap;
		direction: ltr;
		/* Support for all WebKit browsers. */
		-webkit-font-smoothing: antialiased;
		/* Support for Safari and Chrome. */
		text-rendering: optimizeLegibility;
		/* Support for Firefox. */
		-moz-osx-font-smoothing: grayscale;
		/* Support for IE. */
		font-feature-settings: 'liga';
	}

	/******************************************************/
	/*      Materialize - Progress Bar  */
	/*      http://materializecss.com  */

	.materialize-progress {
		position: 'relative',
		height: '4px',
		display: 'block',
		width: '100%',
		borderRadius: '2px',
		margin: '0.5rem 0 1rem 0',
		overflow: 'hidden',
	}

		.materialize-progress .materialize-indeterminate {
		}

			.materialize-progress .materialize-indeterminate:before {
				content: '''',
				position: 'absolute',
				backgroundColor: 'inherit',
				top: '0',
				left: '0',
				bottom: '0',
				will-change: 'left, right',
				Webkit-animation: 'indeterminate 2.1s cubic-bezier(0.65, 0.815, 0.735, 0.395) infinite',
				animation: 'indeterminate 2.1s cubic-bezier(0.65, 0.815, 0.735, 0.395) infinite',
			}

			.materialize-progress .materialize-indeterminate:after {
				content: '''',
				position: 'absolute',
				backgroundColor: 'inherit',
				top: '0',
				left: '0',
				bottom: '0',
				will-change: 'left, right',
				Webkit-animation: 'indeterminate-short 2.1s cubic-bezier(0.165, 0.84, 0.44, 1) infinite',
				animation: 'indeterminate-short 2.1s cubic-bezier(0.165, 0.84, 0.44, 1) infinite',
				Webkit-animation-delay: '1.15s',
				animation-delay: '1.15s',
			}

	@Webkit-keyframes indeterminate {
		0% {
			left: '-35%',
			right: '100%',
		}

		60% {
			left: '100%',
			right: '-90%',
		}

		100% {
			left: '100%',
			right: '-90%',
		}
	}

	@keyframes indeterminate {
		0% {
			left: '-35%',
			right: '100%',
		}

		60% {
			left: '100%',
			right: '-90%',
		}

		100% {
			left: '100%',
			right: '-90%',
		}
	}
`;

const STYLE = {
	/////////////////////////
	////////    UI stuff

	CHIP: {
		fontSize: '0.9rem',
		display: 'inline-block',
		margin: '0.2rem',
		padding: '0.5rem 1rem',
		borderRadius: '2rem',
		backgroundColor: '#bdbdbd',
	},

	BUTTON: {
		display: 'inline-table',
		height: '2.5rem',
		margin: '0.25rem',
		borderRadius: '0.75rem',
		overflow: 'hidden',
		transition: 'box-shadow 0.5s ease, transform 0.3s ease',
		boxShadow: '0 2px 5px 0 rgba(0, 0, 0, 0.16), 0 2px 10px 0 rgba(0, 0, 0, 0.12)',
		backgroundColor: '#ffffff',
		cursor: 'pointer',
		WebkitUserSelect: 'none',
		MozUserSelect: 'none',
		userSelect: 'none',
	},
	BUTTON__COLOR: {
		display: 'flex',
		alignItems: 'center',
		//height: 'calc(2.5rem - 2 * 0.5rem)',
		//padding: '0.5rem 1rem',
		//borderRadius: '0.75rem',
		height: '2.5rem',
		padding: 0,
		borderRadius: '2rem',
		transition: 'opacity 0.5s ease',
		opacity: 1,
	},
	BUTTON__INNER: {
		display: 'flex',
		alignItems: 'center',
		//height: 'calc(2.5rem - 2 * 0.5rem)',
		//padding: '0.5rem 1rem',
		//borderRadius: '0.75rem',
		height: '2.5rem',
		padding: 0,
		borderRadius: '2rem',
		transition: 'opacity 0.5s ease',
		opacity: 1,
	},

	CIRCLE: {
		width: '2.5rem',
		height: '2.5rem',
		borderRadius: '2rem',
	},

	ACTION_ICON: {
		borderRadius: '0.25rem',
		backgroundColor: 'rgba(0,0,0, 0.1)',
		cursor: 'pointer',
	},
	ACTION_ICON_NOTFIRST: {
		borderRadius: '0.25rem',
		backgroundColor: 'rgba(0,0,0, 0.1)',
		marginLeft: '0.25rem',
		cursor: 'pointer',
	},



	/////////////////////////
	////////    HAL stuff

	ANIME_LIGHTER: {
		fontFamily: 'Noto Sans',
		fontSize: '1rem',
		color: '#000000',
		backgroundColor: '#FFF8E1',
		margin: 0,
		padding: '0.5rem',
		borderRadius: '0.25rem',
	},


	PANEL_FILTER: {
		backgroundColor: '#FFF8E1',
		borderRadius: '0.25rem',
		marginTop: '0.5rem',
	},

	PANEL_FILTER__NAME: {
		fontSize: '1.15rem',
		letterSpacing: '-0.1rem',
		marginLeft: '0.25rem',
	},

	PANEL_FILTER__ITEM_NAME: {
		fontSize: '0.95rem',
		letterSpacing: '-0.05rem',
	},


	CONTEXT_MENU: {
		position: 'absolute',
		padding: '0.25rem 0.5rem',
		width: '24rem',
		boxShadow: '0 5px 11px 0 rgba(0, 0, 0, 0.18), 0 4px 15px 0 rgba(0, 0, 0, 0.15)',
		zIndex: '100',
	},

	CONTEXT_MENU__ENTRY: {
		transition: 'backgroundColor 0.3s ease',
	},


	HB_ANIME: {
		display: 'flex',
		flexFlow: 'column nowrap',
		margin: '0.5rem 0',
		backgroundColor: '#ffffff',
		borderRadius: '0.25rem',
		boxShadow: '0 5px 11px 0 rgba(0, 0, 0, 0.18), 0 4px 15px 0 rgba(0, 0, 0, 0.15)',
		backgroundSize: 'cover',
	},

	HB_ANIME__LIGHTEN_BACKGROUND: {
		padding: '1rem',
		backgroundColor: 'rgba(255,255,255, 0.75)',
	},

	HB_ANIME__CONTENT: {
		fontSize: '0.9em',
		padding: '0.2rem', /* !!! */
	},

	HB_ANIME__TITLE: {
		fontSize: '1.1em',
		fontWeight: 'bold',
		paddingBottom: '0.2rem',
	},

	HB_ANIME__ALT_TITLE: {
		color: '#606060',
		padding: '0.2rem',
		paddingLeft: '1rem',
	},

	HB_ANIME__META_BULLET: {
		padding: '0 0.5rem',
	},
};

////////          Const & Styles          ////////
//////////////////////////////////////////////////
//////////////////////////////////////////////////
////////              Loader              ////////

///	Dependencies:
/// <reference path="https://ajax.googleapis.com/ajax/libs/jquery/3.0.0/jquery.min.js" />

if (location.host !== 'openuserjs.org') {
	//	Anime Lighter App

	$('<style>').html(STYLESHEET).appendTo(document.head);

	Promise.all([
		loadstyle('https://fonts.googleapis.com/icon?family=Material+Icons'),
		loadstyle('https://fonts.googleapis.com/css?family=Noto+Sans'),
	]);
} else {
	//	Anime Lighter on OpenUserJS

	function new_panel() {
		return $('<div class="panel panel-default"><div class="panel-heading"><div class="panel-title"></div></div><div class="panel-body"></div></div>').insertBefore($('.panel:contains("Rating")'));
	}

	//	Like me!
	$(document).ready(() => {
		const $panel = new_panel();
		const $title = $panel.find('.panel-title');
		const $content = $panel.find('.panel-body');

		$title.html('<img style="height: 1.1em;" src="' + HAL_IMG_LOGO_ALC + '" /> Enjoy Anime Lighter?');

		$content.append('<p>Share your enjoyment.</p>');

		(() => { // Google Community
			$content.append('<div style="margin: 0.75rem 0;"><a href="https://plus.google.com/communities/104828447132071515050"><img style="height: 24px;" src="' + HAL_IMG_LOGO_GOOGLE + '" /></a> <a href="https://plus.google.com/communities/104828447132071515050">Google+ Community</a></div>');
		})();

		(() => { // Google
			$content.append('<script src="https://apis.google.com/js/platform.js" async defer>{parsetags: \'explicit\'}</script>');
			$content.append('<div style="margin: 0.5rem 0;" class="g-plusone" data-annotation="inline" data-href="https://openuserjs.org/scripts/RandomClown/Anime_Lighter"></div>');
		})();

		(() => { // Facebook
			(function (d, s, id) {
				var js, fjs = d.getElementsByTagName(s)[0];
				if (d.getElementById(id)) return;
				js = d.createElement(s); js.id = id;
				js.src = "//connect.facebook.net/en_US/sdk.js#xfbml=1&version=v2.7";
				fjs.parentNode.insertBefore(js, fjs);
			}(document, 'script', 'facebook-jssdk'));
			$content.append('<div style="margin: 0.5rem 0;" id="fb-root"></div>');
			$content.append('<div class="fb-like" data-href="https://openuserjs.org/scripts/RandomClown/Anime_Lighter" data-layout="standard" data-action="like" data-size="small" data-show-faces="true" data-share="false"></div>');
		})();
	});

	/*
	//	Disqus
	(() => {
		const $panel = new_panel();
		const $title = $panel.find('.panel-title');
		const $content = $panel.find('.panel-body');

		$title.html('<img style="height: 1.1em;" src="" /> Disqus');

		$content.append('<div id="disqus" style="background-color: #0d47a1;"><div id="disqus_thread" class="content"></div></div>');
		$content.append(`<script>var disqus_identifier = 'horc';var disqus_url = 'http://horc.net';var disqus_shortname = 'horc';(function () { var dsq = document.createElement('script'); dsq.type = 'text/javascript'; dsq.async = true; dsq.src = '//' + disqus_shortname + '.disqus.com/embed.js'; (document.getElementsByTagName('head')[0] || document.getElementsByTagName('body')[0]).appendChild(dsq); })();</script>`);
	})();
	*/
}

function loadscript(url) {
	return new Promise(function (resolve, reject) {
		var scripts = document.getElementsByTagName('script');
		for (var i = 0; i < scripts.length; ++i) {
			if (scripts[i].src === url) {
				resolve();
				return;
			}
		}

		var e = document.createElement('script');
		e.src = url;
		e.onload = function () {
			resolve();
		}
		e.onerror = function () {
			e.remove();
			reject('Failed to load: ' + url);
		}
		document.head.appendChild(e);
	});
}
function loadstyle(url) {
	return new Promise(function (resolve, reject) {
		var links = document.getElementsByTagName('link');
		for (var i = 0; i < links.length; ++i) {
			if (links[i].href === url) {
				resolve();
				return;
			}
		}


		var e = document.createElement('link');
		e.rel = 'stylesheet';
		e.type = 'text/css';
		e.href = url;
		e.onload = function () {
			resolve();
		}
		e.onerror = function () {
			e.remove();
			reject('Failed to load: ' + url);
		}
		document.head.appendChild(e);
	});
}

////////              Loader              ////////
//////////////////////////////////////////////////
//////////////////////////////////////////////////
////////               Host               ////////

///	Dependencies:
/// <reference path="https://ajax.googleapis.com/ajax/libs/jquery/3.0.0/jquery.min.js" />

switch (window.location.host) {
	case 'horriblesubs.info':
		document.HAL = function () {
			this.name = 'HorribleSubs';


			//		[currently unimplemented]
			//	Code to insert HAL mounting positions
			this.createPositions = () => { //	Create possible UI positions
				const $viewport = $('<div id="hal-viewport"></div>');
				this.viewport = $viewport[0];

				$('<div class="hal-spot">').insertBefore($('h2:contains("Releases")'));
				$('<div class="hal-spot">').insertAfter($('.episodecontainer'));
				$('<div class="hal-spot">').prependTo($('#sidebar'));
				$('<div class="hal-spot hal-default">').insertAfter($('#text-16'));
				$('<div class="hal-spot">').insertAfter($('#text-8'));
				$('<div class="hal-spot">').appendTo($('#sidebar'));

				$('.hal-default').append($viewport);
			};


			//	Magic code that needs to be run on this host
			this.run = () => {
				//	Find every way to update content

				var refresh = default_dynamic_content_refresh;
				$(document).ready(refresh);


				var refresh_enter = (e) => { if (e.which == 13) refresh(); };
				var refresh_click = (e) => { refresh(); };
				$('.searchbar').on('keyup', refresh_enter);
				$('.refreshbutton').on('keyup', refresh_enter);
				$('.refreshbutton').on('mouseup', refresh_click);
				$('.morebox').on('keyup', '.morebutton, .searchmorebutton', refresh_enter);
				$('.morebox').on('mouseup', '.morebutton, .searchmorebutton', refresh_click);
			};


			//	Stringified version of my personal styles
			this.default_filters = JSON.stringify({
				"b806197c42c3c2ba": {
					"name": "Watch", "enabled": true, "show": true, "order": "1", "style": ".episodecontainer .release-info.hal-$(filter-id) {\n\tfont-weight: bold;\n}"
				},
				"f60287e70cd7b38a": {
					"name": "Meh", "enabled": true, "show": false, "order": "2", "style": ".episodecontainer .release-info.hal-$(filter-id) {\n\topacity: 0.5;   /* makes it transparent */\n}"
				},
				"e7abf1bf3e855f5a": {
					"name": "Drop", "enabled": true, "show": false, "order": "3", "style": ".episodecontainer .release-info.hal-$(filter-id) {\n\tdisplay: none;   /* hides the entry */\n}"
				}
			});


			//	Format of CSS for this host
			this.default_styles = '.episodecontainer .release-info.hal-$(filter-id) {\n$(body)\n}';


			//	CSS selectors for the important bits
			const S = this.selectors = {
				list: '.episodecontainer',
				item: '.release-info',
				_title: '.rls-label',
				_number: '.rls-label',
			};


			//	Javascript to operate the selectors above
			const F = this.filters = {
				//	Listing container
				getList: () => {
					return $(S.list);
				},

				//	Episode item [in container]
				getItem: ($list) => {
					return $list.find(S.item);
				},

				//	Episode title [in item]
				getTitle: ($item) => {
					const contents = $item.find(S._title).contents();
					if (1 < contents.length) {
						//	Episode has a HS link
						//		<td class="rls-label">
						//			(08/21)
						//			<a>title</a>
						//			- 07
						//		</td>
						return contents[1].innerHTML.trim();
					} else {
						//	Episode does not a HS link
						//		<td class="rls-label">
						//			(08/21) title - 07
						//		</td>
						return contents.text().replace(/^\([^\(]*\) (.*?) -[^-]*$/, '$1').trim();
					}
				},

				//	Episode number [in item]
				getNumber: ($item) => {
					const contents = $item.find(S._title).contents();
					if (1 < contents.length) {
						//	Episode has a HS link
						//		<td class="rls-label">
						//			(08/21)
						//			<a>title</a>
						//			- 07
						//		</td>
						return contents[1].innerHTML.trim();
					} else {
						//	Episode does not a HS link
						//		<td class="rls-label">
						//			(08/21) title - 07
						//		</td>
						return contents.text().replace(/.*?([^-]*)$/, '$1').trim();
					}
					return $item.find(S._number).contents()[2].wholeText.replace(/-/, '').trim();
				},

				//	Unprocessed text for the bookmark
				getRaw: ($item) => {
					return $item.find(S._title).text();
				},
			};


			//	All episodes & their element
			this.parse_episodes = default_parse_episodes;


			this.insert_bookmark = function ($item) {
				if ($item.find('.hal-bookmark').length) return;


				const $bookmark = $('<i class="hal-bookmark material-icons">bookmark</i>')
					.css('width', '24px')
					.css('margin-right', '-24px')
					.css('vertical-align', 'middle')
					.css('pointer-events', 'none');


				//	HorribleSubs represents an item with a table.
				//	Find the last <td> & insert the bookmark
				$item.find('td:last').append($bookmark);
			};
		};
		break;
	case 'www.nyaa.se':
		'.tlist'
		'.tlistrow'
		break;
}

//	Generic function to get all episodes & their DOM element in 
function default_parse_episodes() {
	const F = document.HAL.filters;

	const results = [];
	const $list = F.getList();
	const $items = F.getItem($list);

	$items.map((i, item) => {
		const $item = $(item).addClass('hal-item');
		const title = F.getTitle($item);
		const number = F.getNumber($item);
		const raw = F.getRaw($item);

		results.push({
			title: title,
			clean: cleanTitle(title),
			number: number,
			element: item,
			raw: raw,
		});
	});

	return results;
}
function default_dynamic_content_refresh() {
	function hasContentChanged() {
		return content !== content_since_update || element !== element_since_update;
	}
	function shouldHALUpdate() {
		return hal_update_timeout <= hal_update_timer;
	}
	function hasNetworkFinished() {
		return network_timeout <= network_timer;
	}


	const F = document.HAL.filters;
	const check_interval = 50;      // ms
	const hal_update_timeout = 200; // ms - until updating react
	const network_timeout = 8000;   // ms - network timeout
	var hal_update_timer = 0;       // ms - max time until refresh
	var network_timer = 0;          // ms - max time until stop
	var content = '';               // content - text data to compare
	var element = '';               // element - HTML element to compare (since content is not enough)
	var content_since_update = '';  // content - since last HAL update
	var element_since_update = '';  // element - since last HAL update


	console.log('Refreshing HAL');


	clearInterval(document.HAL.interval_refresh);
	document.HAL.interval_refresh = setInterval(() => {
		const list = F.getList();
		content = list.text();
		element = F.getItem(list).filter(':first')[0];


		if (hasContentChanged(content)) {
			if (shouldHALUpdate(content)) {
				$(document).trigger('hal-refresh');

				console.log(' *');

				content_since_update = content;
				element_since_update = element;

				hal_update_timer = 0;
			}

			network_timer = 0;
		}

		if (hasNetworkFinished()) {
			clearInterval(document.HAL.interval_refresh);
			document.HAL.interval_refresh = null;
		}


		//	Increment timers
		hal_update_timer += check_interval;
		network_timer += check_interval;
	}, check_interval);
}

////////               Host               ////////
//////////////////////////////////////////////////
//////////////////////////////////////////////////
////////        HoRC Anime Lighter        ////////

/// <reference path="https://ajax.googleapis.com/ajax/libs/jquery/3.0.0/jquery.min.js" />
/// <reference path="https://cdnjs.cloudflare.com/ajax/libs/react/15.1.0/react.js" />
/// <reference path="https://cdnjs.cloudflare.com/ajax/libs/react/15.1.0/react-dom.js" />
/// <reference path="js/setup.js" />

/*

	To do:

		fix load bug. HAL loads faster than the page so episodes are not highlighted

		episode item needs to be interactable. right click to open, similar to filters
		needs: [move] [remove]

		inline all styles possible into react

		should all set an expire date (new requests after the date)
		& auto cleanup (if near 10mb):
			- storage__anime         (5d)
			- storage__search_slug   (7d)
			- storage__search_title  (7d)

*/

//    Wait for jQuery
'use strict';

var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };

var HAL_loading = setInterval(function () {
	if (typeof $ === 'undefined' || !$(document).jquery) return;
	clearInterval(HAL_loading);

	//	Load site module
	if (document.HAL === undefined) throw 'Failed to load any HAL module';
	document.HAL = new document.HAL();

	document.HAL.createPositions();
	document.HAL.run();

	console.log('	Host: ' + window.location.host);
	console.log('Loaded HAL module for: ' + document.HAL.name);

	//    Run main module
	setTimeout(function () {
		animelighter_render(); // render once to catch early mistakes
		function animelighter_render() {
			ReactDOM.render(React.createElement(AnimeLighter, null), document.HAL.viewport);
		}
		function animelighter_panic() {
			clearInterval(animelighter_interval);
			console.warn('HoRC Anime Lighter had too many fatal errors. Stopping.');
			animelighter_render(); // run once more to produce a stack trace
		}
		var animelighter_error_count = 0;
		var animelighter_interval = setInterval(function () {
			try {
				animelighter_render();

				//	Successful, so reduce error count
				if (0 < animelighter_error_count) {
					animelighter_error_count--;
				}
			} catch (err) {
				if (animelighter_error_count === 0) console.warn('HoRC Anime Lighter hit an error. Continuing for now.');
				console.warn(err);

				//	Failure, so increase error count
				animelighter_error_count += 1;

				//	If it's crashing every loop, then stop
				if (5 <= animelighter_error_count) {
					animelighter_panic();
				}
			}
		}, 42);
	}, 0);
}, 100);

/////////////////////////////////////////
////    JavaScript Storage Helpers

function storage_get_slug(title) {
	return sget(localStorage, 'hal-s-' + title, '');
}
function storage_set_slug(title, data) {
	return sset(localStorage, 'hal-s-' + title, data);
}
function storage_get_anime(slug) {
	return JSON.parse(sget(localStorage, 'hal-a-' + slug, '{}'));
}
function storage_set_anime(slug, data) {
	return JSON.parse(sset(localStorage, 'hal-a-' + slug, JSON.stringify(data)));
	return data;
}
function storage_get_filters() {
	return JSON.parse(sget(localStorage, 'hal-filters', document.HAL.default_filters));
}
function storage_set_filters(data) {
	return JSON.parse(sset(localStorage, 'hal-filters', JSON.stringify(data)));
	return data;
}
function storage_get_settings() {
	return JSON.parse(sget(localStorage, 'hal-settings', '{}'));
}
function storage_set_settings(data) {
	return JSON.parse(sset(localStorage, 'hal-settings', JSON.stringify(data)));
	return data;
}
function storage_get_bookmark() {
	return sget(localStorage, 'hal-bookmark-title', '');
}
function storage_set_bookmark(data) {
	return sset(localStorage, 'hal-bookmark-title', data);
	return data;
}
function storage_get_search_slug(slug) {
	return JSON.parse(sget(localStorage, 'hal-search-s-' + slug, '[]'));
}
function storage_set_search_slug(slug, data) {
	return JSON.parse(sset(localStorage, 'hal-search-s-' + slug, JSON.stringify(data)));
	return data;
}
function storage_get_search_title(title) {
	return JSON.parse(sget(sessionStorage, 'hal-search-t-' + title, '[]'));
}
function storage_set_search_title(title, data) {
	return JSON.parse(sset(sessionStorage, 'hal-search-t-' + title, JSON.stringify(data)));
	return data;
}
function parse_missing_image(url) {
	if (-1 < url.search('/missing.png')) return;
	if (-1 < url.search('/missing-anime-cover')) return;
	return url;
}

/////////////////////////////////////////
////    Anime Lighter

var AnimeLighter = React.createClass({
	displayName: 'AnimeLighter',

	getInitialState: function getInitialState() {
		var state = {
			filters: storage_get_filters(),
			host_episodes: [],

			settings: storage_get_settings(),

			bookmark_episode: '',

			context_menu_open: false,
			context_menu_x: 0,
			context_menu_y: 0,
			context_selected_host_episode: ''
		};

		if (typeof state.settings.bookmark_enabled) state.settings.bookmark_enabled = true;

		return state;
	},

	componentDidMount: function componentDidMount() {
		var _this = this;

		//	Add external refresh listener
		$(document).on('hal-refresh', this.on_refresh_episodes);
		this.on_refresh_episodes();

		//	Temporary save/load
		document.HAL['import'] = function () {
			$('body').css('transition', 'background-color 0.3s ease');
			$('body').css('background-color', '#ffffff');

			var $import = $('<textarea>').appendTo($('body').empty()).css('width', '40rem').css('height', '40rem').css('font-family', 'Consolas, monospace').attr('placeholder', 'Paste the Anime Lighter data to import').on('keyup', function () {
				try {
					var data = JSON.parse($import.val());

					if (Object['typeof'](data) !== 'object') {
						$('body').css('background-color', '#ffe082');
						return;
					}
					if (!Object.keys(data).length) {
						$('body').css('background-color', '#ffe082');
						return;
					}

					//	Delete current settnigs
					for (var key in localStorage) {
						if (key.substr(0, 4) !== 'hal-') continue;
						localStorage.removeItem(key);
					}

					//	Load new settings
					for (var key in data) {
						var value = data[key];
						localStorage.setItem(key, value);
					}

					$('body').css('background-color', '#a5d6a7');
				} catch (e) {
					$('body').css('background-color', '#ef9a9a');
				}
			});
		};
		document.HAL['export'] = function () {
			var data = {};

			for (var key in localStorage) {
				var value = localStorage[key];

				if (key.substr(0, 4) !== 'hal-') continue;
				if (key.substr(0, 10) === 'hal-search') continue;

				data[key] = value;
			}

			$('<pre>').html(JSON.stringify(data, null, '\t')).appendTo($('body').empty());
		};

		//	Bookmark stuff
		//		delegate from document:
		//	bookmark dragging
		//	hal-item drop zone & triggers set_bookmark here

		//	Context menu stuff
		(function () {
			var S = document.HAL.selectors;
			var F = document.HAL.filters;

			//	Add context menu listener
			$(document).on('contextmenu', S.list + ' ' + S.item, function (e) {
				if (e.ctrlKey) return;

				e.preventDefault();

				//	From right click, find item's root
				var $item = $(e.target);
				if (!$item.is(S.item)) $item = $item.closest(S.item);
				if (!$item.is(S.item)) throw '$(document).on(contextmenu): bad selector for S.item';

				//	Find in array
				var host_episode = {};
				for (var i in _this.state.host_episodes) {
					host_episode = _this.state.host_episodes[i];
					if (host_episode.element === $item[0]) {
						_this.setState({
							context_selected_host_episode: host_episode
						});
						break;
					}
				}

				//	Move the menu
				var x = e.clientX;
				var y = $item.position().top + 16;
				if (x < 8) x = 8;

				//const slug = storage_get_slug(host_episode.clean);
				//const anime = slug ? storage_get_anime(slug) : {};
				_this.setState({
					context_menu_open: true,
					context_menu_x: x,
					context_menu_y: y
				});
			});

			//	Block menu events on the menu
			$(document).on('contextmenu click mousedown touchstart', '.horc-context-menu', function (e) {
				e.stopPropagation();
			});

			//	Close event : Click out
			$(document).on('click mousedown touchstart', function (e) {
				//	if a horc-button, dont close
				if ($(e.target).closest('.horc-button').length) {
					e.stopPropagation();
				} else {
					_this.setState({ context_menu_open: false });
				}
			});

			//	Close event : Escape key
			$(document).on('keydown', function (e) {
				if (CODE_KEY[e.which] === 'esc') {
					_this.setState({ context_menu_open: false });
				}
			});
		})();
	},
	componentWillUnmount: function componentWillUnmount() {
		//	Delete event listeners
		$(document).off('hal-refresh');

		$(document).off('contextmenu');
		$(document).off('click mousedown touchstart');
	},

	on_refresh_episodes: function on_refresh_episodes() {
		var _this2 = this;

		var host_episodes = document.HAL.parse_episodes();
		var bookmark_episode = storage_get_bookmark();

		//	Mapping on the host_episode object
		host_episodes.map(function (host_episode, i) {
			//	Cache slug & anime object if known
			host_episode.slug = storage_get_slug(host_episode.clean);
			host_episode.anime = storage_get_anime(host_episode.slug);

			//	Check if bookmarked
			host_episode.bookmark = host_episode.raw === bookmark_episode;

			//	Update highlighting
			_this2.on_episode_update_highlight(host_episode);
		});

		this.setState({
			host_episodes: host_episodes,
			bookmark_episode: bookmark_episode
		});
	},

	on_episode_update_highlight: function on_episode_update_highlight(host_episode) {
		var element = host_episode.element;
		var $item = $(element);
		var slug = host_episode.slug;
		var filter = host_episode.anime.filter;

		//	Apply bookmark
		if (host_episode.bookmark) {
			document.HAL.insert_bookmark($item);
		} else {
			$item.find('.hal-bookmark').remove();
		}

		//	Apply filter style

		if (!slug || !filter) {
			var _attr = element.getAttribute('data-hal-filter');
			if (_attr) {
				element.setAttribute('data-hal-filter', '');
				$(element).removeClass('hal-' + _attr);
			}
			return;
		}

		var attr = element.getAttribute('data-hal-filter');
		if (attr !== filter) {
			element.setAttribute('data-hal-filter', filter);
			$(element).removeClass('hal-' + attr);
			$(element).addClass('hal-' + filter);
		}
	},

	on_set_bookmark: function on_set_bookmark(host_episode, state) {
		if (typeof state !== 'boolean') state = true;

		if (state) {
			storage_set_bookmark(host_episode.raw);
		} else {
			storage_set_bookmark('');
		}

		this.on_refresh_episodes();
	},

	on_set_filters: function on_set_filters(data) {
		storage_set_filters(data);
		this.setState({
			filters: data
		});
	},

	on_set_slug: function on_set_slug(host_episode, slug) {
		host_episode.slug = slug;
		storage_set_slug(host_episode.clean, slug);

		this.on_episode_update_highlight(host_episode);

		this.setState({
			context_selected_slug: host_episode.slug
		});

		this.on_refresh_episodes();
	},
	on_set_anime: function on_set_anime(host_episode, anime) {
		host_episode.anime = anime;
		storage_set_anime(host_episode.slug, anime);

		this.on_episode_update_highlight(host_episode);

		this.setState({
			context_selected_anime: host_episode.anime
		});

		this.on_refresh_episodes();
	},

	render: function render() {
		return React.createElement(
			'div',
			{ className: 'horc-lighter',
				style: STYLE.ANIME_LIGHTER },
			React.createElement(
				'div',
				null,
				React.createElement(TitleBar, null),
				React.createElement(Filters, { filters: this.state.filters,
					host_episodes: this.state.host_episodes,
					on_set_filters: this.on_set_filters })
			),
			this.state.context_menu_open ? React.createElement(ContextMenu, { menu_x: this.state.context_menu_x,
				menu_y: this.state.context_menu_y,
				host_episode: this.state.context_selected_host_episode,
				filters: this.state.filters,

				bookmark_episode: this.state.bookmark_episode,
				on_set_bookmark: this.on_set_bookmark,

				on_set_slug: this.on_set_slug,
				on_set_anime: this.on_set_anime,
				on_set_filters: this.on_set_filters,
				on_episode_update_highlight: this.on_episode_update_highlight }) : null
		);
	}
});

var TitleBar = React.createClass({
	displayName: 'TitleBar',

	open_settings: function open_settings(e) {
		e.preventDefault();

		this.props.on_open_settings();
	},

	render: function render() {
		return React.createElement(
			'div',
			{ className: 'horc-titlebar' },
			React.createElement(
				'div',
				{ style: {
						display: 'flex',
						flexFlow: 'row nowrap',
						alignItems: 'center'
					} },
				React.createElement(
					CircleButton,
					{ color: '#607D8B', href: HAL_URL_MAIN },
					React.createElement('img', { className: 'horc-logo', src: HAL_IMG_LOGO_ANIMELIGHTER })
				),
				React.createElement(
					'div',
					{ style: {
							flexGrow: '1',
							fontSize: '1.75rem',
							letterSpacing: '-0.1rem',
							paddingLeft: '0.25rem'
						} },
					'Lighter'
				)
			)
		);
	}
});

var Filters = React.createClass({
	displayName: 'Filters',

	getInitialState: function getInitialState() {
		return {
			settings_open: false
		};
	},

	on_set_filter: function on_set_filter(filter_id, new_settings) {
		var new_filters = _extends({}, this.props.filters);

		if (new_settings) {
			new_filters[filter_id] = new_settings;
		} else {
			delete new_filters[filter_id];
		}

		this.props.on_set_filters(new_filters);
	},

	open_settings: function open_settings() {
		this.setState({
			settings_open: !this.state.settings_open
		});
	},

	render: function render() {
		var _this3 = this;

		return React.createElement(
			'div',
			{ ref: 'filters',
				className: 'horc-filters',
				style: {
					display: 'flex',
					flexFlow: 'column'
				} },
			React.createElement(
				'div',
				{ style: {
						marginTop: '-24px',
						display: 'flex'
					} },
				React.createElement('div', { style: { flexGrow: 100 } }),
				React.createElement(
					'i',
					{ className: 'material-icons',
						style: STYLE.ACTION_ICON,
						onClick: this.open_settings },
					this.state.settings_open ? 'done' : 'settings'
				)
			),
			Object.keys(this.props.filters).map(function (id, i) {
				var filter = _this3.props.filters[id];
				return React.createElement(Filter, { key: i,
					id: id,
					filter: filter,
					host_episodes: _this3.props.host_episodes,
					settings_open: _this3.state.settings_open,
					on_set_filter: _this3.on_set_filter });
			})
		);
	}
});

var Filter = React.createClass({
	displayName: 'Filter',

	getInitialState: function getInitialState() {
		return {
			list_open: false, // list open state
			name_edit_box_open: false,
			style_edit_box_open: false
		};
	},

	componentWillMount: function componentWillMount() {
		this.temp_name = this.props.filter.name;
	},
	componentDidUpdate: function componentDidUpdate(prevProps) {
		if (this.props.filter.name !== prevProps.filter.name) {
			this.temp_slug = this.props.slug;
		}
	},

	on_enabled: function on_enabled(e) {
		e.preventDefault();

		this.props.on_set_filter(this.props.id, _extends({}, this.props.filter, {
			enabled: !this.props.filter.enabled
		}));
	},
	on_show: function on_show(e) {
		e.preventDefault();

		this.props.on_set_filter(this.props.id, _extends({}, this.props.filter, {
			show: !this.props.filter.show
		}));
	},
	on_edit: function on_edit(e) {
		e.preventDefault();

		var new_name = ReactDOM.findDOMNode(this.refs.name).value;
		var new_order = ReactDOM.findDOMNode(this.refs.order).value;
		var new_style = ReactDOM.findDOMNode(this.refs.style).value;

		this.props.on_set_filter(this.props.id, _extends({}, this.props.filter, {
			name: new_name,
			order: new_order,
			style: new_style
		}));

		this.temp_name = new_name;
	},
	on_delete: function on_delete(e) {
		e.preventDefault();

		if (confirm('Delete "' + this.props.filter.name + '"?')) {
			this.props.on_set_filter(this.props.id, null);
		}
	},

	on_open_name_edit_box: function on_open_name_edit_box(e) {
		e.preventDefault();

		this.setState({
			name_edit_box_open: !this.state.name_edit_box_open
		});
	},

	on_open_style_edit_box: function on_open_style_edit_box(e) {
		e.preventDefault();

		this.setState({
			style_edit_box_open: !this.state.style_edit_box_open
		});
	},

	on_style_box_save: function on_style_box_save(e) {
		if (e.ctrlKey && CODE_KEY[e.which] === 's') {
			e.preventDefault();

			var new_style = ReactDOM.findDOMNode(this.refs.style).value;

			this.props.on_set_filter(this.props.id, _extends({}, this.props.filter, {
				style: new_style
			}));
		}
	},

	render: function render() {
		var _this4 = this;

		var component_episodes = null;
		var component_filters = null;

		if (this.props.settings_open && this.state.name_edit_box_open) {
			component_filters = React.createElement(
				'form',
				{ key: 'form', onSubmit: this.on_edit },
				React.createElement(
					'div',
					null,
					React.createElement('input', { ref: 'name',
						type: 'text',
						style: {
							minWidth: '8rem',
							maxWidth: '16rem',
							width: '70%',
							backgroundColor: this.temp_name === this.props.filter.name ? null : '#fff9c4'
						},
						placeholder: 'New name',
						defaultValue: this.props.filter.name,
						onChange: function (e) {
							return _this4.temp_name = e.target.value;
						},
						required: true }),
					React.createElement('input', { ref: 'order',
						type: 'text',
						style: {
							width: '2.1rem'
						},
						placeholder: 'Order',
						defaultValue: this.props.filter.order,
						pattern: '\\d{1,3}' })
				),
				React.createElement(
					'div',
					null,
					React.createElement('textarea', { ref: 'style',
						style: {
							fontSize: '0.6rem',
							width: '100%',
							marginTop: '0.25rem',
							marginBottom: '0.25rem'
						},
						placeholder: 'Styles to apply',
						defaultValue: this.props.filter.style,
						onKeyDown: this.on_style_box_save })
				),
				React.createElement('input', { type: 'submit', style: { display: 'none' } })
			);
		}

		if (this.props.filter.show) {
			var seen_episodes = {};
			component_episodes = this.props.host_episodes.map(function (host_episode, i) {
				var slug = host_episode.slug;
				var filter = host_episode.anime.filter;

				//	Nothing set, so it's in no list
				if (!slug || !filter) return;

				//	Skip if seen this slug before (removes duplicates within the same list)
				if (seen_episodes[slug]) return;
				//	Mark it as seen
				seen_episodes[slug] = true;

				//	Check if in this filter
				if (filter !== _this4.props.id) return;

				return React.createElement(FilteredItem, { key: i,
					host_episode: host_episode });
			});
		}

		return React.createElement(
			'div',
			{
				className: 'horc-filter',
				style: _extends({}, STYLE.PANEL_FILTER, {
					order: this.props.filter.order
				}) },
			React.createElement(
				'div',
				{ className: 'horc-small-actions',
					style: {
						display: 'flex',
						flexFlow: 'row wrap',
						alignItems: 'center'
					} },
				React.createElement(
					'i',
					{ className: 'material-icons',
						style: STYLE.ACTION_ICON_NOTFIRST,
						onClick: this.on_show },
					this.props.filter.show ? 'expand_more' : 'chevron_right'
				),
				React.createElement(
					'div',
					{ className: 'horc-name',
						style: STYLE.PANEL_FILTER__NAME },
					this.props.filter.name
				),
				React.createElement('div', { style: { flexGrow: 100 } }),
				!this.props.settings_open ? React.createElement(
					'i',
					{ className: 'material-icons',
						style: STYLE.ACTION_ICON_NOTFIRST,
						onClick: this.on_enabled },
					this.props.filter.enabled ? 'label' : 'label_outline'
				) : null,
				this.props.settings_open ? React.createElement(
					'i',
					{ className: 'material-icons',
						style: STYLE.ACTION_ICON_NOTFIRST,
						onClick: this.on_open_name_edit_box },
					'mode_edit'
				) : null,
				this.props.settings_open ? React.createElement(
					'i',
					{ className: 'material-icons',
						style: STYLE.ACTION_ICON_NOTFIRST,
						onClick: this.on_delete },
					'delete'
				) : null
			),
			component_filters,
			React.createElement(
				'div',
				{ style: {
						display: 'flex',
						flexFlow: 'column nowrap'
					} },
				component_episodes
			),
			this.props.filter.enabled ? React.createElement(
				'style',
				{ 'data-hal-filter': this.props.filter.name },
				this.props.filter.style.replace(/\$\(filter-id\)/g, this.props.id)
			) : null
		);
	}
});

var FilteredItem = React.createClass({
	displayName: 'FilteredItem',

	render: function render() {
		return React.createElement(
			'div',
			{ style: STYLE.PANEL_FILTER__ITEM_NAME },
			this.props.host_episode.title
		);
	}
});

/////////////////////////////////////////
////    Context Menu

var ContextMenu = React.createClass({
	displayName: 'ContextMenu',

	on_set_slug: function on_set_slug(slug) {
		this.props.on_set_slug(this.props.host_episode, slug);
	},

	on_set_bookmark: function on_set_bookmark(e) {
		e.preventDefault();

		var bookmarked = this.props.host_episode.raw === this.props.bookmark_episode;

		this.props.on_set_bookmark(this.props.host_episode, !bookmarked);
	},

	on_new_filter: function on_new_filter(filter_name) {
		//	Check if exists
		for (var fid in this.props.filters) {
			if (this.props.filters[fid].name === filter_name) {
				return;
			}
		}

		//	Generate a new ID
		var id;
		for (id = Number.randHex(); (id in this.props.filters); id = Number.randHex()); // generate an ID

		var new_filter = {};
		new_filter[id] = {
			'name': filter_name,
			'enabled': true,
			'show': true,
			'order': '',
			'style': document.HAL.default_styles.replace('$(body)', '	font-weight: bold;')
		};

		var new_filters = Object.assign({}, this.props.filters, new_filter);
		this.props.on_set_filters(new_filters);
	},
	on_choose_filter: function on_choose_filter(filter_id) {
		var anime = _extends({}, this.props.host_episode.anime);
		if (anime['filter'] === filter_id) {
			delete anime['filter'];
		} else {
			anime['filter'] = filter_id;
		}
		this.props.on_set_anime(this.props.host_episode, anime);
	},

	render: function render() {
		var component_filters = [];
		var component_content = null;

		var bookmarked = this.props.host_episode.raw === this.props.bookmark_episode;

		//	Assemble filter components
		if (this.props.host_episode.slug) {
			for (var id in this.props.filters) {
				var filter = this.props.filters[id];

				component_filters.push(React.createElement(ContextMenuFilter, { key: id,
					id: id,
					host_episode: this.props.host_episode,
					filter: filter,
					on_choose_filter: this.on_choose_filter,
					on_episode_update_highlight: this.props.on_episode_update_highlight }));
			}
			if (!component_filters.length) {
				component_filters.push(React.createElement(
					'div',
					{ key: 'empty',
						style: STYLE.CONTEXT_MENU__ENTRY },
					React.createElement(
						'i',
						null,
						'No filters defined'
					)
				));
			}
		}

		//	Assemble content without slug
		if (this.props.host_episode.slug) {
			component_content = React.createElement(
				'div',
				{ style: {
						display: 'flex',
						flexFlow: 'column'
					} },
				React.createElement(
					'div',
					{ style: {
							display: 'flex',
							flexFlow: 'column'
						} },
					component_filters
				),
				React.createElement(ContextMenuFilter, _extends({}, this.props, {
					style: STYLE.CONTEXT_MENU__ENTRY,
					host_episode: this.props.host_episode,
					on_new_filter: this.on_new_filter,
					on_episode_update_highlight: this.props.on_episode_update_highlight })),
				React.createElement(HummingbirdChange, { host_episode: this.props.host_episode,
					on_set_slug: this.on_set_slug }),
				React.createElement(SearchHummingbird, { by: 'slug',
					host_episode: this.props.host_episode,
					on_set_slug: this.on_set_slug })
			);
		} else {
			component_content = React.createElement(
				'div',
				{ style: {
						display: 'flex',
						flexFlow: 'column'
					} },
				React.createElement(
					'div',
					{ style: {
							fontSize: '1.25rem',
							padding: '1rem',
							textAlign: 'center'
						} },
					'Which one is it?'
				),
				React.createElement(SearchHummingbird, { by: 'title',
					host_episode: this.props.host_episode,
					on_set_slug: this.on_set_slug })
			);
		}

		return React.createElement(
			'div',
			{ ref: 'menu',
				className: 'horc-context-menu',
				style: _extends({}, STYLE.CONTEXT_MENU, {
					top: this.props.menu_y,
					left: this.props.menu_x,
					backgroundColor: this.props.host_episode.slug ? '#ffffff' : '#e8eaf6'
				}) },
			React.createElement(
				'div',
				{ style: { display: 'flex', alignItems: 'center' } },
				React.createElement(
					'a',
					{ href: '#bookmark',
						onClick: this.on_set_bookmark },
					React.createElement(
						'i',
						{ className: 'material-icons' },
						bookmarked ? 'bookmark' : 'bookmark_border'
					)
				),
				React.createElement(
					'div',
					{ style: { flexGrow: 100, textAlign: 'center', fontWeight: 'bold' } },
					this.props.host_episode.title
				)
			),
			component_content
		);
	}
});

var ContextMenuFilter = React.createClass({
	displayName: 'ContextMenuFilter',

	on_new_filter: function on_new_filter(e) {
		e.preventDefault();

		var $name = ReactDOM.findDOMNode(this.refs.name);

		var value = $name.value.trim().replace(/\s+/g, ' ');
		$name.value = '';

		this.props.on_new_filter(value);
	},
	on_choose_filter: function on_choose_filter(e) {
		this.props.on_choose_filter(this.props.id);
		this.props.on_episode_update_highlight(this.props.host_episode);
	},

	render: function render() {
		if (this.props.filter) {
			var in_filter = false;
			if (this.props.host_episode.anime['filter'] && this.props.host_episode.anime['filter'] === this.props.id) in_filter = true;

			if (in_filter) {
				return React.createElement(
					'div',
					{ style: _extends({}, STYLE.CONTEXT_MENU__ENTRY, {
							order: this.props.filter.order,
							display: 'block',
							alignItems: 'center',
							padding: '0.25rem 1rem',
							borderBottom: '1px solid #cccccc',
							cursor: 'pointer'
						}),
						onClick: this.on_choose_filter },
					React.createElement(
						'a',
						{ style: {
								color: '#e64a19'
							},
							href: '#remove',
							onClick: function (e) {
								return e.preventDefault();
							} },
						React.createElement(
							'b',
							null,
							this.props.filter.name
						)
					)
				);
			} else {
				return React.createElement(
					'div',
					{ style: _extends({}, STYLE.CONTEXT_MENU__ENTRY, {
							order: this.props.filter.order,
							display: 'block',
							alignItems: 'center',
							padding: '0.25rem 1rem',
							borderBottom: '1px solid #cccccc',
							cursor: 'pointer'
						}),
						onClick: this.on_choose_filter },
					React.createElement(
						'a',
						{ style: {
								color: '#1976d2'
							},
							href: 'move',
							onClick: function (e) {
								return e.preventDefault();
							} },
						this.props.filter.name
					)
				);
			}
		} else {
			return React.createElement(
				'form',
				{ onSubmit: this.on_new_filter,
					style: _extends({}, STYLE.CONTEXT_MENU__ENTRY, {
						display: 'block',
						alignItems: 'center',
						padding: '0.25rem 1rem',
						borderBottom: '1px solid #cccccc'
					}) },
				React.createElement('input', { ref: 'name',
					type: 'text',
					placeholder: 'New personal filter',
					pattern: '\\s*[^\\s]+.*',
					required: true,
					style: { width: '100%' } })
			);
		}
	}
});

var HummingbirdChange = React.createClass({
	displayName: 'HummingbirdChange',

	componentWillMount: function componentWillMount() {
		this.temp_slug = this.props.host_episode.slug;
	},

	on_change_slug: function on_change_slug(e) {
		e.preventDefault();

		var slug = ReactDOM.findDOMNode(this.refs.slug).value;

		if (slug === this.props.host_episode.slug) return;

		this.temp_slug = slug;

		this.props.on_set_slug(slug);
	},

	render: function render() {
		var _this5 = this;

		var change_slug = 'hal-change-' + this.props.host_episode.slug;

		return React.createElement(
			'form',
			{ onSubmit: this.on_change_slug,
				style: {
					display: 'flex',
					flexFlow: 'row nowrap',
					alignItems: 'center',
					marginTop: '0.25rem'
				} },
			React.createElement(
				'div',
				{ style: { marginRight: '0.25rem' } },
				'Slug:'
			),
			' ',
			React.createElement('input', { ref: 'slug',
				type: 'text',
				placeholder: 'Clear the slug?',
				pattern: '[\\-\\w]*',
				defaultValue: this.props.host_episode.slug,
				style: {
					marginLeft: '0.5rem',
					width: '16rem',
					backgroundColor: this.temp_slug === this.props.host_episode.slug ? null : '#fff9c4'
				},
				onChange: function (e) {
					return _this5.temp_slug = e.target.value;
				},
				list: change_slug }),
			make_component_slug_links(this.props.host_episode.title),
			React.createElement(
				'datalist',
				{ ref: 'list', id: change_slug },
				React.createElement('option', { value: this.props.host_episode.slug })
			)
		);
	}
});

/////////////////////////////////////////
////    Bookmark

var Bookmark = React.createClass({
	displayName: 'Bookmark',

	render: function render() {
		return React.createElement(
			'div',
			{ className: 'horc-bookmark' },
			React.createElement('div', { className: 'horc-content' })
		);
	}
});

/////////////////////////////////////////
////    Hummingbird

//		Props:
//	title        |  The title to search by
//	slug         |  OR the slug to retrieve by
//	on_set_slug  |  To set the Hummingbird id
//	
//		States:
//	hummingbird_search_results  |  Results to display
var SearchHummingbird = React.createClass({
	displayName: 'SearchHummingbird',

	getInitialState: function getInitialState() {
		return {
			status: XHR_LOADING,
			hummingbird_search_results: [] };
	},

	// list of results returned from search
	componentDidMount: function componentDidMount() {
		this.mounted = true;
		this.on_hummingbird_search();

		this.last_slug = this.props.host_episode.slug;
		this.last_title = this.props.host_episode.title;
	},
	componentWillUnmount: function componentWillUnmount() {
		this.mounted = false;
	},
	componentDidUpdate: function componentDidUpdate(prevProps, prevState) {
		if (this.last_slug !== prevProps.host_episode.slug) {
			this.last_slug = prevProps.host_episode.slug;
			this.on_hummingbird_search();
		} else if (this.last_title !== prevProps.host_episode.title) {
			this.last_title = prevProps.host_episode.title;
			this.on_hummingbird_search();
		}
	},

	on_manual_set_slug: function on_manual_set_slug(e) {
		e.preventDefault();

		var slug = ReactDOM.findDOMNode(this.refs.slug).value;

		if (!slug.length) return;

		this.props.on_set_slug(slug);
	},

	on_hummingbird_search: function on_hummingbird_search() {
		if (this.props.by === 'slug') {
			this.on_hummingbird_search_slug();
		} else if (this.props.by === 'title') {
			this.on_hummingbird_search_title();
		}
	},
	on_hummingbird_search_slug: function on_hummingbird_search_slug() {
		var _this6 = this;

		if (!this.props.host_episode.slug) return;

		var slug = this.props.host_episode.slug;
		this.face = HAL_SAD_FACE[HAL_SAD_FACE.length * Math.random() | 0];

		var results = storage_get_search_slug(slug);
		if (results.length) {
			this.setState({
				status: XHR_OK, // its cached, so done.
				hummingbird_search_results: results
			});
		} else {
			this.setState({
				status: XHR_LOADING,
				hummingbird_search_results: []
			});

			xhr({
				url: 'https://hummingbird.horc.net/hummingbird.js',
				query: {
					by: 'slug',
					slug: slug.encode()
				}
			})['catch'](function (e) {
				if (_this6.mounted) {
					console.error(e);
					_this6.setState({
						status: e.status
					});
				}
			}).then(xhr.parse).then(function (res) {
				storage_set_search_slug(slug, res);

				// mounted & active, so update state
				if (_this6.mounted && slug === _this6.props.host_episode.slug) {
					_this6.setState({
						status: XHR_OK,
						hummingbird_search_results: res
					});
				}
			});
		}
	},
	on_hummingbird_search_title: function on_hummingbird_search_title() {
		var _this7 = this;

		if (!this.props.host_episode.title) return;

		var title = this.props.host_episode.title;
		var clean = this.props.host_episode.clean;

		this.face = HAL_SAD_FACE[HAL_SAD_FACE.length * Math.random() | 0];

		var results = storage_get_search_title(clean);
		if (results.length) {
			this.setState({
				status: XHR_OK,
				hummingbird_search_results: results
			});
		} else {
			this.setState({
				status: XHR_LOADING,
				hummingbird_search_results: []
			});

			xhr({
				url: 'https://hummingbird.horc.net/hummingbird.js',
				query: {
					by: 'title',
					title: title.encode()
				}
			})['catch'](function (e) {
				if (_this7.mounted) {
					console.error(e);
					_this7.setState({
						status: e.status
					});
				}
			}).then(xhr.parse).then(function (res) {
				storage_set_search_title(clean, res);

				// mounted & active, so update state
				if (_this7.mounted && title === _this7.props.host_episode.title) {
					_this7.setState({
						status: XHR_OK,
						hummingbird_search_results: res
					});
				}
			});
		}
	},

	render: function render() {
		var _this8 = this;

		var res = this.state.hummingbird_search_results;

		var search_style = {
			display: 'flex',
			flexFlow: 'column nowrap'
		};

		var component_manual_slug = null;
		if (this.props.by === 'title') {
			component_manual_slug = React.createElement(
				'form',
				{ onSubmit: this.on_manual_set_slug,
					style: {
						display: 'flex',
						flexFlow: 'row nowrap',
						alignItems: 'center'
					} },
				React.createElement(
					'div',
					null,
					'Slug:'
				),
				React.createElement('input', { ref: 'slug',
					type: 'text',
					placeholder: 'Manual input',
					pattern: '[\\-\\w]*',
					style: { marginLeft: '0.5rem', width: '16rem' } }),
				make_component_slug_links(this.props.host_episode.title)
			);
		}

		switch (this.state.status) {
			case XHR_LOADING:
				return React.createElement(
					'div',
					{ className: 'horc-search',
						style: search_style },
					component_manual_slug,
					React.createElement(
						'div',
						{ style: { paddingTop: '1rem' } },
						React.createElement(Preloader, { color: 'blue' })
					)
				);
			case XHR_OK:
				if (res.length) {
					return React.createElement(
						'div',
						{ className: 'horc-search',
							style: search_style },
						component_manual_slug,
						res.map(function (anime, i) {
							return React.createElement(HummingbirdItem, { key: i,
								anime: anime,
								search: _this8.props.host_episode.slug ? false : true,
								on_set_slug: _this8.props.on_set_slug });
						})
					);
				} else {
					return React.createElement(
						'div',
						{ className: 'horc-search',
							style: search_style },
						component_manual_slug,
						React.createElement(
							'div',
							{ style: { padding: '0.5rem' } },
							React.createElement(
								'b',
								null,
								'Couldn\'t find an anime by the slug entered. Things to try:'
							),
							React.createElement(
								'div',
								{ style: { padding: '0.5rem 1rem', lineHeight: '1.5rem' } },
								React.createElement(
									'div',
									null,
									'Enter the slug if you know it'
								),
								React.createElement(
									'div',
									null,
									'Clear the slug if you want to search again'
								),
								React.createElement(
									'div',
									{ style: {
											display: 'flex',
											flexFlow: 'row nowrap',
											alignItems: 'center'
										} },
									React.createElement(
										'a',
										{ href: 'https://hummingbird.me/search?scope=anime&query=' + encodeURIComponent(this.props.title),
											target: '_blank', style: { color: '#fd7532' } },
										'Find the slug through Hummingbird'
									),
									React.createElement(
										'i',
										{ className: 'material-icons', style: { fontSize: '18px' } },
										'exit_to_app'
									)
								),
								React.createElement(
									'div',
									{ style: {
											display: 'flex',
											flexFlow: 'row nowrap',
											alignItems: 'center'
										} },
									React.createElement(
										'a',
										{ href: 'https://www.google.com/#q=anime+' + encodeURIComponent(this.props.title),
											target: '_blank', style: { color: '#4caf50' } },
										'Google the alternate title'
									),
									React.createElement(
										'i',
										{ className: 'material-icons', style: { fontSize: '18px' } },
										'exit_to_app'
									)
								)
							)
						)
					);
				}
			case XHR_NETWORK_ERROR:
				return React.createElement(
					'div',
					{ className: 'horc-search' },
					component_manual_slug,
					React.createElement(
						'div',
						{ style: { paddingTop: '1rem' } },
						'The internet gave up',
						React.createElement('br', null),
						'Try again in a bit?'
					)
				);
			default:
				return React.createElement(
					'div',
					{ className: 'horc-search',
						style: search_style },
					component_manual_slug,
					React.createElement(
						'div',
						{ style: { paddingTop: '1rem' } },
						'Can\'t perform a search right now',
						React.createElement('br', null),
						'horc.net is down',
						React.createElement('br', null),
						this.face
					)
				);
		}
	}
});

//		Props:
//	anime        |  Parsed JSON response from Hummingbird API
//	on_set_slug  |  To set the Hummingbird id
var HummingbirdItem = React.createClass({
	displayName: 'HummingbirdItem',

	on_set_slug: function on_set_slug(e) {
		e.preventDefault();

		this.props.on_set_slug(this.props.anime.slug);
	},

	render: function render() {
		var anime = this.props.anime;

		var hummingbird_url = 'https://hummingbird.me/anime/' + this.props.anime.slug;

		//	Set cover image
		var bg_style = {};
		var cover = parse_missing_image(anime.cover_image) || null;
		var poster = parse_missing_image(anime.poster_image) || null;
		var bg_img = cover || poster || null;
		if (bg_img) bg_style.backgroundImage = 'url(' + bg_img + ')';

		//	Generate genres
		if (!this.genres_component) {
			this.genres_component = [];
			var container = this.props.anime.genres || [];
			for (var i = 0; i < container.length; i++) {
				var item = container[i];
				this.genres_component.push(React.createElement(
					'div',
					{ key: 'genre-' + i,
						className: 'horc-chip',
						style: STYLE.CHIP },
					item
				));
			}
		}

		//	Calculate meta airing date
		if (!this.airing) {
			var now = new Date();
			var air_s = new Date(anime.started_airing_date);
			var air_e = new Date(anime.finished_airing_date);
			if (anime.finished_airing_date === null) {
				this.airing = 'Airing from ';
				this.airing += anime.started_airing_date;
			} else {
				this.airing = (now < air_s ? 'Will air' : now < air_e ? 'Airing' : 'Aired') + ' from ';
				this.airing += anime.started_airing_date;
				this.airing += ' to ' + anime.finished_airing_date;
			}
		}

		if (!this.synopsis) {
			this.synopsis = this.props.anime.synopsis || '';
			if (200 < this.synopsis.length) this.synopsis = this.synopsis.substr(0, 200) + '...';
		}

		return React.createElement(
			'div',
			{ className: 'horc-anime',
				style: _extends({}, STYLE.HB_ANIME, bg_style) },
			React.createElement(
				'div',
				{ style: STYLE.HB_ANIME__LIGHTEN_BACKGROUND },
				React.createElement(
					'div',
					{ className: 'horc-anime-title',
						style: STYLE.HB_ANIME__TITLE },
					anime.titles.canonical
				),
				React.createElement(
					'div',
					{ style: STYLE.HB_ANIME__CONTENT },
					React.createElement(
						'div',
						{ className: 'horc-anime-alt-title',
							style: STYLE.HB_ANIME__ALT_TITLE },
						anime.titles.english
					),
					React.createElement(
						'div',
						{ className: 'horc-anime-alt-title',
							style: STYLE.HB_ANIME__ALT_TITLE },
						anime.titles.japanese
					),
					React.createElement(
						'div',
						{ className: 'horc-anime-genres' },
						this.genres_component
					),
					React.createElement(
						'div',
						{ className: 'horc-anime-meta' },
						React.createElement(
							'span',
							{ className: 'horc-anime-meta-info' },
							this.props.anime.show_type || '?'
						),
						React.createElement(
							'span',
							{ style: STYLE.HB_ANIME__META_BULLET },
							'•'
						),
						React.createElement(
							'span',
							{ className: 'horc-anime-meta-info' },
							this.props.anime.age_rating || '?'
						),
						React.createElement(
							'span',
							{ style: STYLE.HB_ANIME__META_BULLET },
							'•'
						),
						React.createElement(
							'span',
							{ className: 'horc-anime-meta-info' },
							this.props.anime.episode_count || '?',
							' episodes'
						),
						React.createElement(
							'span',
							{ style: STYLE.HB_ANIME__META_BULLET },
							'•'
						),
						React.createElement(
							'span',
							{ className: 'horc-anime-meta-info' },
							this.props.anime.episode_length,
							' min'
						),
						React.createElement(
							'span',
							{ style: STYLE.HB_ANIME__META_BULLET },
							'•'
						),
						React.createElement(
							'span',
							{ className: 'horc-anime-meta-info' },
							this.airing
						)
					),
					React.createElement(
						'div',
						{ className: 'horc-anime-synopsis' },
						this.synopsis
					),
					React.createElement(
						'div',
						{ className: 'horc-actions',
							style: {
								display: 'flex',
								flexFlow: 'row nowrap'
							} },
						this.props.search ? React.createElement(
							'button',
							{ style: {
									height: '2rem',
									marginTop: '1rem'
								},
								onClick: this.on_set_slug },
							'This One'
						)
						/*	Button class broke. Fix it?
      <Button color="#673AB7"
      		onClick={this.on_set_slug}>
      	<span style={{color: '#ffffff'}}>This One</span>
      </Button>
      */
						: undefined,
						React.createElement('div', { style: { flexGrow: 100 } }),
						React.createElement(
							CircleButton,
							{ color: '#fd7532',
								href: hummingbird_url,
								target: '_blank' },
							React.createElement('img', { className: 'horc-logo', src: HAL_IMG_LOGO_HUMMINGBIRD })
						)
					)
				)
			)
		);
	}
});

/////////////////////////////////////////
////    UI Stuff

//	BROKEN
var Button = React.createClass({
	displayName: 'Button',

	render: function render() {
		var mergeClass = reactAddClass("horc-button", this.props.className);
		var inner = React.createElement(
			'div',
			{ style: _extends({}, STYLE.BUTTON__COLOR, {
					backgroundColor: this.props.color
				}) },
			React.createElement(
				'div',
				{ style: {} },
				this.props.name,
				this.props.children
			)
		);

		return React.createElement(
			'a',
			{ className: mergeClass,
				style: _extends({}, STYLE.BUTTON),
				href: this.props.href,
				onClick: this.props.onClick,
				title: this.props.name,
				target: this.props.target },
			inner
		);
	}
});

var CircleButton = React.createClass({
	displayName: 'CircleButton',

	render: function render() {
		var mergeClass = reactAddClass("horc-button", this.props.className);
		var inner = React.createElement(
			'div',
			{ style: _extends({}, STYLE.BUTTON__COLOR, {
					backgroundColor: this.props.color
				}) },
			React.createElement(
				'div',
				{ style: {} },
				React.createElement(
					'div',
					{ style: { padding: this.props.padding } },
					this.props.name,
					this.props.children
				)
			)
		);

		return React.createElement(
			'a',
			{ className: mergeClass,
				style: _extends({}, STYLE.BUTTON, STYLE.CIRCLE),
				href: this.props.href,
				onClick: this.props.onClick,
				title: this.props.name,
				target: this.props.target },
			inner
		);
	}
});

var Preloader = React.createClass({
	displayName: 'Preloader',

	componentWillMount: function componentWillMount() {
		this.colors = {
			'red': { fg: '#f44336', bg: '#ef9a9a' },
			'pink': { fg: '#e91e63', bg: '#f48fb1' },
			'purple': { fg: '#9c27b0', bg: '#ce93d8' },
			'deep-purple': { fg: '#673ab7', bg: '#b39ddb' },
			'indigo': { fg: '#3f51b5', bg: '#9fa8da' },
			'blue': { fg: '#2196f3', bg: '#90caf9' },
			'light-blue': { fg: '#03a9f4', bg: '#81d4fa' },
			'cyan': { fg: '#00bcd4', bg: '#80deea' },
			'teal': { fg: '#009688', bg: '#80cbc4' },
			'green': { fg: '#4caf50', bg: '#a5d6a7' },
			'light-green': { fg: '#8bc34a', bg: '#c5e1a5' },
			'lime': { fg: '#cddc39', bg: '#e6ee9c' },
			'yellow': { fg: '#ffeb3b', bg: '#fff59d' },
			'amber': { fg: '#ffc107', bg: '#ffe082' },
			'orange': { fg: '#ff9800', bg: '#ffcc80' },
			'deep-orange': { fg: '#ff5722', bg: '#ffab91' },
			'brown': { fg: '#795548', bg: '#bcaaa4' },
			'grey': { fg: '#9e9e9e', bg: '#eeeeee' },
			'blue-grey': { fg: '#607d8b', bg: '#b0bec5' }
		};
	},

	render: function render() {
		var color = this.props.color || 'blue';
		var co = this.colors[color] || this.colors['blue'];
		var fg = co.fg;
		var bg = co.bg;
		return React.createElement(
			'div',
			{ className: 'materialize-progress', style: { backgroundColor: bg } },
			React.createElement('div', { className: 'materialize-indeterminate', style: { backgroundColor: fg } })
		);
	}
});

/////////////////////////////////////////
////	React Component helpers

function make_component_slug_links(title) {
	return [React.createElement(
		'a',
		{ key: 'h',
			href: 'https://hummingbird.me/search?scope=anime&query=' + encodeURIComponent(title || ''),
			target: '_blank',
			style: { color: '#fd7532', marginLeft: '0.5rem' },
			title: 'Hummingbird' },
		'H'
	), React.createElement(
		'a',
		{ key: 'g',
			href: 'https://www.google.com/#q=anime+' + encodeURIComponent(title || ''),
			target: '_blank',
			style: { color: '#4caf50', marginLeft: '0.5rem' },
			title: 'Google' },
		'G'
	), React.createElement(
		'i',
		{ className: 'material-icons',
			style: {
				fontSize: '18px',
				marginLeft: '0.5rem'
			} },
		'exit_to_app'
	)];
}

/////////////////////////////////////////
////    React Other Helpers

function reactAddClass(old_classes, new_classes) {
	if (new_classes) {
		if (old_classes) old_classes += ' ';
		old_classes += new_classes;
	}
	return old_classes;
}

////////        HoRC Anime Lighter        ////////
//////////////////////////////////////////////////
//////////////////////////////////////////////////
////////         Embedded Images          ////////

const HAL_IMG_LOGO_ANIMELIGHTER = '';
const HAL_IMG_LOGO_HUMMINGBIRD = '';
const HAL_IMG_LOGO_GOOGLE = '';
const HAL_IMG_LOGO_ALC = '';

////////         Embedded Images          ////////
//////////////////////////////////////////////////