Airdates.tv enhancer

Control how many past weeks you want to see and many other enhancements

目前為 2017-12-11 提交的版本,檢視 最新版本

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

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

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name        Airdates.tv enhancer
// @namespace   V@no
// @author      V@no
// @description Control how many past weeks you want to see and many other enhancements
// @include     http://www.airdates.tv/*
// @include     https://www.airdates.tv/*
// @include     https://disqus.com/embed/comments/*
// @icon        
// @license     MIT
// @version     1.18.1
// @run-at      document-start
// @grant       none
// ==/UserScript==

//tab = 2 spaces


//to add custom links, uncomment this block of code:

//begin custom links
/*
let loo = 1000;
(function loop()
{
	if ((typeof(engines) == "undefined" || !engines.length) && loo--)
		return setTimeout(loop, 0);

//	Available tags in the url (must be in capital letters):
//	MONKEY = full name of the episode
//	MONKEY_N = full name without episode number (S01E32)
//	MONKEY_ID = show ID on this website
//	MONKEY_ARCHIVELINK = url to the archive listing on this website
//	{WIKI_TITLE} = wikipedia path for the show
	window.engines = window.engines.concat([
		{
			host: "example.com", //domain name, used as ID
			href: "http://example.com/path?monkey=MONKEY&monkey_n=MONKEY_N&monkey_id=MONKEY_ID&monkey_archivelink=MONKEY_ARCHIVELINK&wiki_title={WIKI_TITLE}", //url to the website
			name: "test" //name will be displayed in the list
		},
		{
			host: "blah.com", //domain name, used as ID
			href: "http://example.com/another/path?monkey=MONKEY&monkey_n=MONKEY_N&monkey_id=MONKEY_ID&monkey_archivelink=MONKEY_ARCHIVELINK&wiki_title={WIKI_TITLE}", //url to the website
			name: "second example" //name will be displayed in the list
		},
	]);
})()
*/

//end custom links


let log = console.log;

function ls(id, data)
{
	let r;
	if (typeof(data) == "undefined")
	{
		r = localStorage.getItem(id);
		try
		{
			r = JSON.parse(r);
		}catch(e){}
	}
	else
	{
		try
		{
			r = localStorage.setItem(id, JSON.stringify(data));
		}
		catch(e){}
	}
	return r;
}
function cs(id, data)
{
	let r;
	if (typeof(data) == "undefined")
	{
		r = readCookie(id);
		try
		{
			r = JSON.parse(r);
		}catch(e){}
	}
	else
	{
		try
		{
			r = createCookie(id, JSON.stringify(data));
		}
		catch(e){}
	}
	return r;
}
cs.list = ["cm","sh","sn","s","sr","n","w","wa","middleClick"];

let func = function(event)
{
	let _today,
			_hidden = ls("hidden") || [],
			showHidden = cs("sh") ? true : false,
			_assignColor = assignColor;
/*
work around for expand/collapse same series in one day
*/
window.assignColor = function assignColor ( seriesId, color, permanent )
{
	let r = _assignColor(seriesId, color, permanent),
			css = $("#css_"+seriesId),
			html = css.html();

	if (html)
		css.html(html.replace(/(\s+)(\.activeOnly)([^\{]+)\{/, "$1$2$3:not(.multi),body:not(.collapseMulti) $2$3,$2 .day.expand $3,$2 .day.opened $3{"));

	return r;
};
/*
fixing browser history inflating after each page refresh and prev/next history jump don't work
*/
var prevPath;
function loadArchiveFromPathname(originalPath,highlightSelector)
{
	path = originalPath||document.location.pathname;
	var match = path.match(/^\/archive\/([0-9]+)-([0-9]+)$/);
	var ym = match && match.length == 3? (~~match[1]*12+~~match[2]-1) : ymToday;
//start fix - browser history inflating after each page refresh
	if (originalPath && prevPath != originalPath)
	{
		prevPath = originalPath;
		history.pushState({originalPath:originalPath,highlightSelector:highlightSelector}, "", originalPath);
	}

//end fix
	$("#archive-prev").attr("href", "/archive/" + ym2str(ym-1));
	$("#archive-next").attr("href", "/archive/" + ym2str(ym+1));
	$("body").toggleClass("archive", !!match&&match.length==3);
	$("#leaveArchive").text(ym>ymToday?"Leave future":"Leave archive");

	var whenDone = function(){
		// set title
		$("#archive-current").text((!!match&&match.length==3)?ym2str(ym):"Go waste your time");
		// highlight today with a happy yellow background
		$( "div.day[data-date='" + today.getFullYear() + pad2( today.getMonth() + 1 ) + pad2( today.getDate() ) + "']" ).addClass( "today" ).attr( "id", "today" );
		markSearchResults();

		var $hl = $(highlightSelector||".does-not-exist-and-never-will");
		if($hl.get().length>0){
			// blink
			for( var i = 0; i < 8; i++ ) $hl.animate( {"opacity":i%2}, 1 ).delay( 100 );

			// scroll to
			var val = function(){return $hl.offset().top-Math.max(document.documentElement.clientHeight, window.innerHeight || 0)/2;};
			$('html, body').animate({
						scrollTop: val()
				},{duration:500, step:function(top,info){
					info.end = val();
				}});
		}
//injecting userscript function execution
		showPast(function()
		{
//adding watched checkboxes
			$("div.day > div.entry").each(watched.attach);
//collapse multiple entries of the same series in one day
			$("div.day").each(collapseMulti);
		});
		showHideLoad();
	};

	if(ymCurrent != ym)
	{
		$(".days").load("/_archive/" + ~~(ym/12) + "-" + ~~((ym%12)+1), function()
		{
			ymCurrent = ym;
			whenDone();
		});
	}
	else
	{
		whenDone();
	}
}

//collapse multiple entries of the same series in one day
let collapseMulti = function collapseMulti(i, day)
{
	if (day.list)
	{
		collapseMulti.setTitle(day.list, collapseMulti.enabled && !$(day).hasClass("opened") ? "_titleCollapsed" : "_titleOrig");
		return;
	}
	day.list = {};
	let list = {};
	day.addEventListener("mouseenter", function(e)
	{
		collapseMulti.mouseOver(e, day);
	}, true);
	day.addEventListener("mouseleave", function(e)
	{
		collapseMulti.mouseOut(e, day);
	}, true);

	$(day).find("div.entry").each(function(i, entry)
	{
		let id = entry.getAttribute("data-series-id");
		entry._title = $(entry).find(".title");
		entry._title._titleOrig = entry._title.text();
		if (list[id])
			$(entry).toggleClass("multi", true);
		else
			list[id] = [];

		list[id].push(entry);
	});
	for(let i in list)
	{
		if (list[i].length < 2)
			continue;

		list[i][0]._title._titleCollapsed = list[i][0]._title.text() + "-" + list[i][list[i].length - 1]._title.text().replace(/.* ([^ ]+)$/, "$1");
		day.list[i] = list[i][0]._title;
		if (collapseMulti.enabled)
			$(day.list[i]).html(day.list[i]._titleCollapsed);
	}
};

collapseMulti.prev = null;

collapseMulti.setTitle = function(list, title)
{
	for (let i in list)
		$(list[i]).html(list[i][title]);
};

collapseMulti.mouseOver = function(e, day)
{
	if (!collapseMulti.enabled)
		return;

	if (collapseMulti.prev)
	{
		if (collapseMulti.prev == day)
		{
//log("over"+$(day).attr("data-date"))
//			clearTimeout(collapseMulti.timer);
			return;
		}
		else
		{
//			$(collapseMulti.prev).toggleClass("expand", false);
			if (!$(collapseMulti.prev).hasClass("opened"))
				collapseMulti.setTitle(collapseMulti.prev.list, "_titleCollapsed");
		}
	}
	collapseMulti.setTitle(day.list, "_titleOrig");
	$(day).toggleClass("expand", true);
	collapseMulti.prev = day;
};

collapseMulti.mouseOut = function(e, day, id)
{
	if (!collapseMulti.enabled || e.target != day)
		return;

//	clearTimeout(collapseMulti.timer);
	collapseMulti.timer = setTimeout(function()
	{
//log("out"+$(day).attr("data-date"));
		$(day).toggleClass("expand", false);
		if (!$(day).hasClass("opened"))
			collapseMulti.setTitle(day.list, "_titleCollapsed");

		collapseMulti.prev = null;
	}, 300);
};

collapseMulti.onOff = function(e, id, checked)
{
	collapseMulti.enabled = checked;
	$("div.day").each(collapseMulti);
};

//fixing prev/next history jump does nothing
$(window).on("popstate", function(e)
{
	let state = e.originalEvent.state;
	prevPath = (state ? state.originalPath : undefined) || e.target.location.pathname;

	loadArchiveFromPathname(prevPath, state ? state.highlightSelector : undefined);
});
/*
end fixing browser history inflating after each page refresh and prev/next history jump don't work
*/

window.loadArchiveFromPathname = loadArchiveFromPathname;
function showPast(callback)
{

// hide all past days
	let	d = new Date();

	_today = $("div.day[data-date='" + d.getFullYear() + pad2((new Date()).getMonth() + 1) + pad2((new Date()).getDate()) + "']");

	if (!_today || $(".archive").length)
		return;

	let	firstDay = $(".days").children().first(),
			firstDate = firstDay.attr("data-date"),
			div = [document.createElement("div")],
			y = firstDate.slice(0, 4),
			m = firstDate.slice(4, 6);

	if (parseInt(firstDate.slice(6, 8)) < 15)
		div.push(document.createElement("div"));

	let c = div.length;
	for (let n = 0; n < div.length; n++)
	{
		let d = new Date(y, m - n - 1);
		$(div[n]).load("/_archive/" + d.getFullYear() + "-" + pad2(d.getMonth() + 1), pastLoadDone);
	}
	function pastLoadDone()
	{
		if (--c > 0)
			return;

		let found = false,
				func = function (i, o)
				{
					if ($(this).attr("data-date") == firstDate)
						found = true;

					if (found)
						$(this).remove();
				};
		for (let n = 0; n < div.length; n++)
		{
			found = false;
		//remove any duplicate dates that already exist;
			$(div[n]).children().each(func);
			$(div[n]).children().prependTo($("div.days"));
		}
		pastLoaded();
		if (callback)
			callback();
	}
}//showPast()

function pastLoaded()
{
	let	hasLoaded = $("#pastWeeks").length,
			stop = false;

	$("div.days").children().each(function(i, o)
	{
		if ($(this).is(_today))
		{
			$(this).attr("today", "true");
			stop = true;
		}
		if (stop)
		{
			$(this).removeClass("past");
		}
		else
		{
			$(this).addClass("past");
		}
	});

	let week = ["mon", "tue", "wed", "thu", "fri", "sat", "sun"],
			match = $(_today).find(".date").text().toLowerCase().match(/([^,]{3})/),
			daysNum = match ? week.indexOf(match[1]) : -1;

	let prev = _today.prev();
	//there is no identication of which week a day belong to, so we must make sure that previous days of current week don't count as previous week.
	if (daysNum > 0)
	{
		for(let i = 0; i < daysNum  && prev; i++)
		{
			prev.removeClass("past");
			prev = prev.prev();
		}
	}
	//remove more then 4 weeks worth of days.
	for(let i = 0; i < 27 && prev; i++)
	{
		prev = prev.prev();
	}
	if (prev)
	{
		let found = false,
				func = function(i, o)
				{
					if (found)
						return;

					if ($(this).is(prev))
						found = true;
					else
						$(this).remove();
				};
		$("div.days").children().each(func);
	}
	let	daysPast = $('div.past'),
			weeks = parseInt(readCookieRaw("w")),
			weeksMax = Math.round((daysPast.length) / 7);//how many past weeks do we have available?

	if (isNaN(weeks))
		weeks = readCookieRaw("p") == "1" ? weeksMax : 0;

	if (weeks > weeksMax)
		weeks = weeksMax;

	//main function that shows/hides past days
	function showWeeks()
	{

		//get week numbers from dropdown
		let weeks = parseInt($("#pastWeeks").val());

		//just some sanity check
		if (weeks * 7 > daysPast.length)
			weeks = weeksMax;

		if (weeks < 1)
			weeks = 0;

		for(let i = weeksMax; i > 0; i--)
			$( "div.calendar" ).toggleClass("showPast" + i, (weeks >= i));

		createCookie("w", weeks);
	}
	//add new class pastNN to each past day, where NN is a week number.
	let func = function(i)
	{
		$(this).addClass("past" +  (Math.ceil((daysPast.length - i) / 7 % (weeksMax + 1))));
	};
	daysPast.each(func);

	if (!hasLoaded)
	{
		//create dropdown menu with number of available past weeks
		let dropdown = document.createElement("select");
		dropdown.id = "pastWeeks";

		for(let i = 0; i <= weeksMax; i++)
		{
			let option = document.createElement("option");
			option.value = i;
			option.text = i;
			option.selected = (weeks == i);
			dropdown.appendChild(option);
		}

		//add event listener to dropdown
		$(dropdown).change(showWeeks);

		let span = document.createElement("span");
		span.id = "past-weeks";
		span.appendChild(document.createTextNode("Show past weeks:"));
		span.appendChild(dropdown);
		//insert our dropdown and new text into document
		$(".calendar").prepend(span);

		//hide "Show/Hide past month" links
		$("#past-showing").toggle(true);
		//togglePast();//make sure we disable "Show past month"
		$("#past-showing").toggle(false);
		$("#past-hidden").toggle(false);
	}
	showWeeks();
}//pastLoaded()

var prevOpened = null,
		prevParentOpened = null,
		prevParentOpenTimer = null;
//adding attribute "opened" to the entry allows us show/hide things from CSS based on entry state
$("div.days").on("click", "div.entry div.title", function(e)
{
	let $entry = $( this ).parent(),
			parent = $entry.parent();

	if (prevOpened)
	{
		let po = prevOpened,
				ppo = prevParentOpened;

		prevParentOpenTimer = setTimeout(function()
		{
			collapseMulti.setTitle(ppo[0].list, collapseMulti.enabled && !ppo.hasClass("expand")? "_titleCollapsed" : "_titleOrig");

			po.parent().toggleClass("opened", false);
		}, 400);

		prevOpened.attr("opened", "");
		setTimeout(function()
		{
			po.removeAttr("opened");
		}, 300);
		prevOpened = null;
	}


	if ($entry.attr("opened") === undefined)
	{
		$entry.attr("opened", "");
		parent.toggleClass("opened", true);
		collapseMulti.setTitle(parent[0].list, "_titleOrig");
//idealy this should've been done via "on complete" function submitted for slideUp/slideDown
		if (prevParentOpened && parent[0] == prevParentOpened[0])
			clearTimeout(prevParentOpenTimer);

		setTimeout(function()
		{
			$entry.attr("opened", "1");
		}, 300);

		prevParentOpened = parent;
		prevOpened = $entry;
	}
});

//create stylesheet. A little trick to have multi-line text in javascript
let	style = document.createElement("style"),
		css = function(){/*
div.showPast1 div.past1,
div.showPast2 div.past2,
div.showPast3 div.past3,
div.showPast4 div.past4,
div.showPast5 div.past5,
div.showPast6 div.past6,
div.showPast7 div.past7,
div.showPast8 div.past8,
div.showPast9 div.past9
{
	display: block;
}

/*
div.calendar.searching > #past-weeks
{
	text-decoration: line-through;
}
*//*
#past-weeks
{
	margin-left: 0.2%;
}

#pastWeeks
{
	margin-left: 0.5em;
}

.archive #past-weeks
{
	text-decoration: line-through;
}


/* today column border *//*
div.today
{
	border-color: #FFC800;
}


/* higlighted title under cursor *//*
div.entry > input[type="checkbox"]:hover + div.title,
div.title:hover
{
	background-color: #ffffd5;
	-webkit-transition: background 0s;
	color: black;
}

div[white] div.entry:hover div.title,
div[white] div.title:hover
{
	color: black;
}


/* highlight opened entry *//*
.entry[opened]
{
	border: 1px solid black;
}

.entry[opened][white]
{
	border-color: grey;
}

div[opened] + div
{
	border-top: 0;
}

.engines input[type="checkbox"]
{
	width: unset !important;
	vertical-align: top;
}
.engines img
{
	vertical-align: text-bottom !important;
}

div.entry,
.date
{
	cursor: pointer;
}

.filter
{
	margin-left: 1em;
}

.past,
.showhide0,
span[checked] > .checkoff,
span:not([checked]) > .checkon,
/*separate showing searies and season premiere*//*
div.calendar.showNew div.entry:not(.series-premiere):not(.searchResult),
div.calendar.showReturn div.entry:not(.season-premiere):not(.searchResult)
{
	display: none;
}


/*separate showing searies and season premiere*//*
#nu-showing,
#nu-hidden
{
	display: none !important;
}

div.calendar.showNew div.entry.series-premiere:not(.searchResult),
div.calendar.showReturn div.entry.season-premiere:not(.searchResult)
{
	display: block;
}


/*Search field overlaps text below in Firefox*//*
#searchFieldContainer
{
	background-color: unset;
}


/*Watched*//*
body:not(.enableWatched) div.entry > input[type="checkbox"]
{
	display: none;
}

body.enableWatched div.entry[watched] > div.title
{
	text-decoration: line-through;
}

div.entry > input[type="checkbox"]
{
	vertical-align: bottom;
	width: unset !important;
	float: left;
}


/*Past related*//*
.past
{
	opacity: 0.5;
}

body.collapseMulti div.day:not(.expand):not(.opened) div.entry.multi
{
	display: none !important;
}
*/};

style.innerHTML = css.toString().slice(14,-3).split("*//*").join("*/");
$("head").append(style);


//fix incorrect initial color in colorpicker
//fix clicking outside of colorpicker saves selected color instead of discarding
//fix close colorpicker by pressing escape button

var editingSeriesId = -1;
var clone = $("#colorPickerHolder").clone();
$("#colorPickerHolder").remove();
var picker = clone.colorPicker(
{
	animationSpeed: 0,
	opacity: false,
	buildCallback: function($elm)
	{
		let cp = this;
		$elm
			.append('<div class="cp-disp"><input type="button" value="save"> <input type="button" value="cancel"></div>')
			.on( "click", "input", function()
			{
				if (!this.value)
					return;

				if (this.value == 'save')
				{
					assignColor( editingSeriesId, "#" + cp.color.colors.HEX, true );
					editingSeriesId = -1;
				}
				cp.toggle();
			});
		$("body").on("keydown", function(e)
		{
//ESC(27) = cancel
			if (e.which == 27 && $elm.is(":visible"))
				cp.toggle();
		});
	},
	cssAddon: ".cp-disp{ padding-bottom: 2px; clear:both; }",
	renderCallback: function($elm, toggled) {
		var colors = this.color.colors;
		// on show
		if( toggled === true ){
		}
		// on change
		else if( toggled === undefined ){
//preview new color
			assignColor( editingSeriesId, "#" + colors.HEX, false );
		}
		// on hide
		else if( toggled === false ){
//restore entry color
			if( editingSeriesId > 0 )
				loadColor( editingSeriesId );

			editingSeriesId = -1;
			picker.detach().appendTo("body");
		}

	}
}).attr("id", "colorPickerHolderNew"); //replace ID so it won't initialize in main.js

$("body").off("click", ".picker");

// now hook up the picker to the picker icons
$("body").on("click", ".picker", function(e){
	if( !$(e.target).hasClass("picker") ) return;

	editingSeriesId = $(this).parents("[data-series-id]").data("series-id");
//set initial color in colorpicker based on current entry or default to white
	picker.val(coalesce(DB.savedColors[editingSeriesId], "#ffffff"));
	picker.detach().appendTo(this).click();
});

let engines = window.engines;
engines.add = function(id)
{
	for(let i = 0; i < _engines.length; i++)
	{
		if (_engines[i].name == id || _engines[i].host == id)
			return;
	}
	for(let i = 0; i < engines.length; i++)
	{
		if (engines[i].name == id || engines[i].host == id)
		{
			_enginesList.push(engines[i].name);
			_engines.push(engines[i]);
			cs("middleClick", _enginesList);
			return;
		}
	}
};

engines.remove = function(id)
{
	for(let i = 0; i < _engines.length; i++)
	{
		if (_engines[i].name == id || _engines[i].host == id)
		{
			let n = _enginesList.indexOf(_engines[i].name);
			if (n != -1)
				_enginesList.splice(n, 1);

			cs("middleClick", _enginesList);
			_engines.splice(i, 1);
			return;
		}
	}
};

engines.check = function(id)
{
	for(let i = 0; i < _engines.length; i++)
	{
		if (_engines[i].name == id || _engines[i].host == id)
			return i;
	}
	return -1;
};

//middle click on day's title opens selected engines for user's shows
let _engines = [],
		_enginesList = cs("middleClick") || [];

for(let i = 0; i < engines.length; i++)
{
	let n = _enginesList.indexOf(engines[i].name);
	if (n != -1)
		_engines.push(engines[i]);

	if (engines[i].name == "Piratebay")
	{
		//sort by date instead of seeds
		engines[i].href = engines[i].href.replace(/\.se\//, ".org").replace(/\/0\/7\/0$/, "/0/3/0");
	}
}

function middleClick(e, search)
{
	if (e.button != 1)
		return;

	e.stopPropagation();
	e.preventDefault();
	let parent,
			entry = true;
	if (search)
		parent = $(search);
	else if ($(this).is("div.title"))
		parent = $(this).parent();
	else
	{
		parent = $(this).parent().find("div.entry");
		entry = false;
	}
	let func = function(e)
	{
		let el = $(this),
				MONKEY_ID = encodeURIComponent( el.data("series-id") );
		if (!entry && !DB.getColor(MONKEY_ID))
			return;

		let title = el.children("div.title").text().replace("?", "").replace(/[0-9]+-[0-9]+-[0-9]+/, ""),
//				MONKEY = encodeURIComponent( title ),
				MONKEY = encodeURIComponent( title.replace(/ E[0-9]+/g, "") ),
				MONKEY_N = encodeURIComponent( title.replace( /S[0-9]+E[0-9]+$/g, '' ) ),
				WIKI_TITLE = encodeURIComponent( el.data("series-source") );

		$.each( _engines, function( i, engine )
		{
			let href = engine.href
				.replace( "MONKEY_ID", MONKEY_ID )
				.replace( "MONKEY_N", MONKEY_N )
				.replace( "MONKEY", MONKEY )
				.replace( "{WIKI_TITLE}", WIKI_TITLE );

			href = fixLink(href, engine.name);
			window.open(href, (MONKEY + engine.name).replace(/ /g, "").replace(/%.{2}/g, ""));
		});
	};
	parent.each(func);
}
let markSearchResults = window.markSearchResults;
window.markSearchResults = function()
{
	let entries = $("#searchResults").find("div.entry");
	entries.each(watched.attach);
	$("div.day").each(collapseMulti);

	entries.on("mousedown", function(e)
	{
		middleClick(e, this);
	});
	return markSearchResults();
};
$("div.date,div.title").on("mousedown", middleClick);

//sanitizing engine links
$("body").on("click", "div.entry div.title", function(e)
{
	let obj = this;
	if (obj.inited)
		return;

	obj.inited = true;
	setTimeout(function()
	{
		let showHideObj = document.createElement("a"),
				id = $(obj.parentNode).attr("data-series-id"),
				show = document.createElement("span"),
				hide = document.createElement("span");

		show.innerHTML = "Show";
		show.className = "showhide0";
		hide.innerHTML = "Hide";
		hide.className = "showhide1";
		showHideObj.appendChild(show);
		showHideObj.appendChild(hide);
		showHideObj.appendChild(document.createTextNode(" this show"));

		showHideObj.className = "showhide";
		showHideObj.href = "#";
		showHideObj.addEventListener("click", function(e)
		{
			e.stopPropagation();
			e.preventDefault();
			showHide(parseInt(obj.parentNode.getAttribute("data-series-id")), 2);
		}, false);
		$(obj).parent().find(".engines").append(showHideObj).children().each(function(i, o)
		{
			if (o.tagName != "A")
				return;

			let checkbox = document.createElement("input"),
					img = o.previousSibling;

			if (img.tagName != "IMG")
				img = o;

			let engine = img.src ? img.src.match(/\?domain=(.*)/)[1] : null;
			checkbox.engine = engine;
			checkbox.type = "checkbox";
			checkbox.title = "Open with middle click on title";
			checkbox.checked = engines.check(engine) != -1;
			o.parentNode.insertBefore(checkbox, img);
			if (engine == "airdates.tv" || o.className == "showhide")
			{
				checkbox.style.visibility = "hidden";
				return;
			}
			o.href = fixLink(o.href, o.text);

			$(checkbox).on("click", "", function(e)
			{
				if (checkbox.checked)
					engines.add(checkbox.engine);
				else
					engines.remove(checkbox.engine);
			});

		});
	});
});

function fixLink(link, engine)
{
	link = link.replace("%3F", "").replace(/[0-9]+-[0-9]+-[0-9]+(%20)?/, "").replace(/( |%20)E[0-9]+/, "");
	if (engine == "Piratebay")
		link = link.replace(/['"`!]/g, '');
	return link;
}

function command(id, val)
{
	if (!command.list[id])
		return false;

	try
	{
		command.list[id].func({preventDefault:function(){},stopPropagation:function(){}}, val);
	}
	catch(e)
	{
		return false;
	}
	return true;
}
command.list = {};
command.add = function(id, objId, func)
{
	command.list[id] = {
		id: id,
		objId: objId,
		func: func
	};
};
function createCheckbox(id, label, cookie, callback)
{
	let span = document.createElement("span"),
			checkon = document.createElement("span"),
			checkoff = document.createElement("span"),
			a = document.createElement("a"),
			check = cs(cookie) ? true : false;
	a.className = "filter " + id;
	checkon.className = "checkon nu";
	checkoff.className = "checkoff nu";
	checkon.innerHTML = "☑";
	checkoff.innerHTML = "☐";
	a.href = "#";
	span.appendChild(checkon);
	span.appendChild(checkoff);
	span.appendChild(document.createTextNode(label));
	a.appendChild(span);
	$(a).insertBefore("#nu-showing");
	let func = function(e, val)
	{
		e.preventDefault();
		e.stopPropagation();
		let check = span.hasAttribute("checked");
		if (val !== undefined)
			check = !val;

		if (check)
			span.removeAttribute("checked");
		else
			span.setAttribute("checked", "checked");

		$(".calendar").toggleClass(id, !check);
		$("body").toggleClass(id, !check);
		cs(cookie, check ? 0 : 1);

		return (typeof(callback) == "function") ? callback(e, id, !check) : e;
	};
	command.add(cookie, id, func);
	a.addEventListener("click", func, false);
	if (check)
		span.setAttribute("checked", "checked");

	$(".calendar").toggleClass(id, check);
	$("body").toggleClass(id, check);

	return a;
}

function showHideLoad()
{

	if (showHideLoad.inited)
		return;

	if (cs("sh") === null)
	{
		cs("sh", cs("showhidden") ? 1 : 0);
		eraseCookie("showhidden");
	}

	$(".days").before(createCheckbox("showHidden", "Show hidden", "sh"));
	$(".days").before(createCheckbox("collapseMulti", "Collapse multiple", "cm", collapseMulti.onOff));
	$(".days").before(createCheckbox("enableWatched", "Enable watched", "wa"));
	collapseMulti.enabled = cs("cm");
	for(let i = 0; i < _hidden.length; i++)
	{
		showHide(_hidden[i], 1);
	}
	showHideLoad.inited = true;


	let clearColors = $(".clearColors"),
			clearHidden = clearColors.clone(false),
			clearWatched = clearColors.clone(false);

	clearHidden.removeClass();
	clearHidden.addClass(".clearHidden");
	clearHidden.html("Clear hidden");
	clearHidden.insertAfter(clearColors);
	clearWatched.removeClass();
	clearWatched.addClass(".clearWatched");
	clearWatched.html("Clear watched");
	clearWatched.insertAfter(clearHidden);
	clearColors.after(" | ");
	clearHidden.after(" | ");
	clearHidden.on( "click", function()
	{
		let array = [];
		for(let i = 0; i < _hidden.length; i++)
			array.push(_hidden[i]);

		for(let i = 0; i < array.length; i++)
			showHide(array[i], 0);

		alert( "Done!" );
		return false;
	});
	clearWatched.on( "click", function()
	{
		watched._list = {};
		watched.save();
		$("div.entry").each(function(i, entry)
		{
			if (entry._input)
				watched.update(entry, false);
		});

		alert( "Done!" );
		return false;
	});

	$("body").off("click", ".importColors");
	$("body").off("click", ".exportColors");
	$("body").on("click", ".importColors", function()
	{
		let str = prompt( "Please enter the crazy text!" ),
				reload = false;
		if( str )
		{
			let hidden = 0,
					colors = 0,
					watchedNum = 0;
			$.each( str.split( ";" ), function( i, e )
			{
				let color = e.split("=").concat([true] );
				if (color.length == 3)
				{
					assignColor.apply( null, color );
					colors++;
				}
				else
				{
					let json = null;
					try
					{
						json = JSON.parse(color[0]);
					}
					catch(e){}
					if (json)
					{
						if ("hidden" in json)
						{
							hidden = json.hidden.length;
							for(let i = 1; i < json.hidden.length; i++)
								showHide(parseInt(json.hidden[i]), 1);
						}
						if ("watched" in json)
						{
							watchedNum = 0;
							for(let id in json.watched)
							{
								for(let i = 0; i < json.watched[id].length; i++)
								{
									let ep = json.watched[id][i];
									watched.add(id, ep);
									watchedNum++;
									$('div.entry[data-series-id="' + id + '"]').each(function(n, entry)
									{
										if (watched.title(entry) == ep)
											watched.update(entry, true);
									});
								}
							}
						}
						if ("settings" in json)
						{
							for(let i in json.settings)
							{
								if (cs.list.indexOf(i) == -1)
									continue;

								if (!command(i, json.settings[i]))
								{
									cs(i, json.settings[i]);
									reload = true;
								}
							}
						}
					}
					else
					{
						color = color[0].split(",");
						hidden = color.length - 1;
						for(let i = 1; i < color.length; i++)
							showHide(parseInt(color[i]), 1);
					}
				}
			});
			let txt = "Imported " + colors + " colors" + ((hidden || watchedNum) ? " and marked " : "");
			if (hidden)
				txt += hidden + " as hidden";
			if (watchedNum)
				txt += (hidden ? ", " : " ") + watchedNum + " as watched";

			alert(txt);
			if (reload)
				window.location.reload();
		}
		return false;
	});
	$("body").on( "click", ".exportColors", function()
	{
		let str = $.map(DB.savedColors,function(e,i){return i + "=" + e;}).join(";");
		str = str.replace(/;?[0-9]+=#FFFFFF/i, "");
		let settings = {};
		for(let i = 0; i < cs.list.length; i++)
		{
			let v = cs(cs.list[i]);
			if (v !== null)
			{
				settings[cs.list[i]] = v;
			}
		}
		let obj = {};
		for(let i in _hidden)
		{
			obj.hidden = _hidden;
			break;
		}
		for(let i in watched._list)
		{
			obj.watched = watched._list;
			break;
		}
		for(let i in settings)
		{
			obj.settings = settings;
			break;
		}
		for(let i in obj)
		{
			str += ";" + JSON.stringify(obj);
			break;
		}


		if (str)
				prompt( "This is the crazy text. \nYou can save it in a normal textfile and/or import it to another computer/browser.", str );
		else
				alert("Nothing to export");
		return false;
	});


} //showHideLoad()
//fix when removing color it still saves it in database as #FFFFFF
window.DB._setColor = window.DB.setColor;
window.DB.setColor = function setColor(id, c)
{
	if (c == "#FFFFFF")
		c = "";

	this._setColor(id, c);
};

function showHide(id, t)
{
	let hidden = _hidden.indexOf(id);
	if (typeof(t) == "undefined")
		return hidden != -1;

	if (t == 2)
		t = hidden == -1 ? 1 : 0;

	switch(t)
	{
		case 0:
			_hidden.splice(hidden, 1);
			hidden = false;
			break;
		case 1:
			if (hidden == -1)
				_hidden.push(id);

			hidden = true;
			break;
	}
	let css = $("#css" + id);
	if (hidden)
	{
 		if (!css.length)
 		{
			let style = "<style id='css" + id + "'>"+
									"div.calendar:not(.showHidden).activeOnly .entry[data-series-id='" + id + "'][opened]:not(.multi){ display: block !important; }"+
									"div.calendar:not(.showHidden) .entry[data-series-id='" + id + "']:not([opened]):not(.searchResult){ display: none !important; }"+
									".entry[data-series-id='" + id + "'] .title{ font-style: italic; }"+
									".entry[data-series-id='" + id + "']{ opacity: 0.3;}"+
									".entry[data-series-id='" + id + "'] .showhide0{ display: inline; }"+
									".entry[data-series-id='" + id + "'] .showhide1{ display: none; }"+
									"</style>";
			$(style).appendTo("body");
		}
	}
	else
		css.remove();

	ls("hidden", _hidden);
	return hidden;
}

//splitting New/Returning into two separate filters
if (cs("sn") === null)
{
	let n = cs("n") ? 1 : 0;
	cs("sn", n);
	cs("sr", n);
}
//hiding old combined checkboxes
$( "#nu-showing" ).toggle(false);
$( "#nu-hidden" ).toggle(false);
//add new separate checkboxes
createCheckbox("showNew", "New shows", "sn");
createCheckbox("showReturn", "Returning shows", "sr");

//fix auto clear search field on click
setTimeout(function()
{
	$(document).off("click", ".onX, .x").on('touchstart click', '.onX', function( ev )
	{
		ev.preventDefault();
		$(this).removeClass('x onX').val('').change();
	});
//fix no X button after page refresh and browser auto fill search bar with previous text
	$('input.clearable').trigger("input");
});

//fix paste via right click
$("#searchBecauseNoOneChecks").on( "input", function(e)
{
	$(this).trigger("change");
});

//watched checkbox
function watched(entry)
{
	if (entry._input.checked)
		watched.add(entry.getAttribute("data-series-id"), watched.title(entry));
	else
		watched.remove(entry.getAttribute("data-series-id"), watched.title(entry));

	watched.update(entry, entry._input.checked);
}
watched._list = ls("watched") || {};
watched._saving = false;
watched.add = function(id, episode)
{
	if (!watched._list[id])
		watched._list[id] = [];

	if (watched._list[id].indexOf(episode) == -1)
		watched._list[id].push(episode);

	watched.save();
};

watched.remove = function(id, episode)
{
	if (!watched._list[id])
		return;

	let n = watched._list[id].indexOf(episode);
	if (n == -1)
		return;

	watched._list[id].splice(n, 1);
	if (!watched._list[id].length)
		delete watched._list[id];

	watched.save();
};

watched.save = function(f)
{
	if (watched._saving && !f)
		return;

	clearTimeout(watched._saving);
	watched._saving = setTimeout(function()
	{
		watched._saving = false;
		ls("watched", watched._list);
	}, 500);
};

watched.has = function(entry)
{
	let id = entry.getAttribute("data-series-id");
	return watched._list[id] && watched._list[id].indexOf(watched.title(entry)) != -1;
};

watched.title = function(entry)
{
	let txt = $(entry).find("div.title").text();
	return txt.substring(txt.lastIndexOf(" ") + 1).replace(/\s+$/g, "");
};

watched.update = function(entry, enable)
{
	if (enable)
	{
		entry.title = "Watched";
		entry.setAttribute("watched", "");
	}
	else
	{
		entry.title = "Not watched";
		entry.removeAttribute("watched");
	}
	entry._input.checked = enable;

};

watched.attach = function(i,entry)
{
	if (entry._input)
		return;

	let input = document.createElement("input");
	input.type = "checkbox";
	input.checked = watched.has(entry);
	entry._input = input;
	watched.update(entry, input.checked);

	input.addEventListener("change", function(e)
	{
		let title = watched.title(entry);
		$('div.entry[data-series-id="' + entry.getAttribute("data-series-id") + '"]').each(function(i, entry)
		{
			if (watched.title(entry) == title)
				watched.update(entry, input.checked);
		});
		watched(entry);
	}, false);
	entry.insertBefore(input, entry.firstChild);
};


if (!event)
{
//injecting userscript function execution
	showPast(function()
	{
//adding watched checkboxes
		$("div.day > div.entry").each(watched.attach);
//collapse multiple entries of the same series in one day
		$("div.day").each(collapseMulti);
	});
	showHideLoad();
}

};//func()

if (window.top !== window.self)
{
	func = function(){};

//disqus troll filter
	if (window.location.href.indexOf("disqus.com") != -1)
	{
		let trollHide = false;
		let trollList = ls("trolls"),
				trollTimer,
				comments = {};

		if (!trollList || typeof(trollList) != "object")
			trollList = ["Tubasing"];

		let isTroll = function (name)
		{
			return trollList.indexOf(name);
		};

		let trollSave = function ()
		{
			clearTimeout(trollTimer);
			trollTimer = setTimeout(function()
			{
				ls("trolls", trollList);
			}, 1000);
		};

		let trollAdd = function (name)
		{
			if (isTroll(name) == -1)
				trollList.push(name);

			trollSave();
		};

		let trollRemove = function (name)
		{
			let n = isTroll(name);
			if (n != -1)
				trollList.splice(n, 1);

			trollSave();
		};

		let isMatch = function (needle, array)
		{
			let r = -1;
			for(let i = 0; i < array.length; i++)
			{
				r = needle.indexOf(array[i]);
				if (r != -1)
					break;
			}
			return r;
		};
		let findParent = function (obj, c)
		{
			if (typeof(c) != "object")
				c = [c];

			if (obj && obj.className && isMatch(obj.className, c) != -1)
				return obj;

			if (obj)
				return findParent(obj.parentNode, c);

			return null;
		};
		let censor = function (a, b, c, d, e)
		{
			let p = "blah",
					r = "",
					caps = true,
					t,
					i,
					isStr = isNaN(a.replace(/['`].*/, ''));

			if ((a.length < 2 && isStr) || (a.length < 7 && a.match(/['`]/) && isStr) || a.match(/^\$?[0-9]+k?/i))
				return a;

			for(i = 0; i < a.length; i++)
				if (a[i] == a[i].toLowerCase())
				{
					caps = false;
					break;
				}

			for(i = 0; i < p.length; i++)
			{
				t = p[i];

				if (a.length < i + 1)
					continue;

				if (caps || (a.length > i && a[i] != a[i].toLowerCase()))
					t = t.toUpperCase();

				r += t;
			}
			return r;
		};

		let toggleTroll = function (comment, f)
		{
			//f = 0: orig
			//f = 1: troll
			//f = 2: toggle
			f = typeof(f) == "undefined" ? 2 : f;
			let parent = comment._parent,
					t = true;
			if (f == 1 || (f == 2 && parent.getAttribute("troll") != "true"))
			{
				if (trollHide)
					comment.innerHTML = '<img src="' + trollSrc + '" class="trollComment">';
				else
					censorText(comment);
			}
			else
			{
				censorText(comment, true);
				t = false;
			}
			parent.setAttribute("troll", t);
			return t;
		};

		let censorText = function (obj, orig)
		{
			let c = obj.childNodes;
			for(let i = 0; i < c.length; i++)
			{
				if (c[i].nodeName == "#text")
				{
					if (!("nodeValueOrig" in c[i]))
						c[i].nodeValueOrig = c[i].nodeValue;

					let t;
					if (orig)
						t = c[i].nodeValueOrig;
					else
					{
						if (!("nodeValueTroll" in c[i]))
							c[i].nodeValueTroll = c[i].nodeValue.replace(/(\w+(['`]\w+)?)/g, censor);

						t = c[i].nodeValueTroll;
					}

					c[i].nodeValue = t;
				}
				else if (c[i].childNodes.length && c[i].className != "see-more")
					censorText(c[i], orig);
			}
			return obj.innerHTML;
		};

		window.addEventListener("load", function(e)
		{
			let timer = setInterval (function()
			{
				let posts = document.getElementById("post-list");
				if (!posts)
					return;

				clearInterval(timer);
				timer = setInterval (function()
				{
					let names = posts.getElementsByClassName("author");
					if (!names.length)
						return;
					clearInterval(timer);

					let style = document.createElement("style");
					style.innerHTML = function(){/*
.trollComment
{
	cursor: pointer;
	opacity: 0.4;
	transform: scaleX(-1)
}
.post-byline:not([troll="true"]) img.troll
{
	opacity: 0.1;
	transform: scaleX(-1);
}
img.troll
{
	background-image: url("");
	width: 16px;
	height: 16px;
	cursor: pointer;
	margin-left: 0.3em;
	vertical-align: top;
	display: inline-block;
}
.trollMenuImg
{
	vertical-align: top;
}
.trollBox
{
	height: auto;
	line-height: 1.3em;
}
*/
					}.toString().slice(14,-3).split("*//*").join("*/");
					document.getElementsByTagName("head")[0].appendChild(style);
					for(let i = 0; i < names.length; i++)
					{

						let body = findParent(names[i], "post-content");
						if (!body)
							continue;

						let parent = findParent(names[i], "post-byline"),
								img = document.createElement("img"),
								name = names[i].innerText,
								troll = (isTroll(name) != -1),
								post = body.getElementsByClassName("post-message")[0];

						post._parent = parent;

						img.className = "troll";
						parent.insertBefore(img, parent.firstChild.nextSibling);

						parent.setAttribute("troll", troll);

						names[i].innerHTML = names[i].innerHTML.replace("Tubasing", "Tubashit");

						if (typeof(comments[name]) == "undefined")
							comments[name] = [];

						comments[name].push({
							parent: parent,
							post: post
						});

						img.addEventListener("click", function(e)
						{
							let troll = toggleTroll(post);
							if (troll)
								trollAdd(name);
							else
								trollRemove(name);

							for(let i = 0; i < comments[name].length; i++)
							{
								if (post === comments[name][i].post)
									continue;

								toggleTroll(comments[name][i].post);
							}
						}, false);



						let timer,
								prev,
								eventHandler = function(e)
								{
									if (parent.getAttribute("troll") != "true")
										return;

									let type = e.type == "mouseenter";

									if (prev == e.type)
										return;

									prev = e.type;
									clearTimeout(timer);
									timer = setTimeout(function()
									{
										censorText(post, type);
									}, type ? 100 : 1000);
								};
						body.addEventListener("mouseenter", eventHandler, false);
						body.addEventListener("mouseleave", eventHandler, false);

						if (!troll)
							continue;

						censorText(post);
					}
				}, 100);
			}, 100);
		}, false);
	}
}

if (document.readyState != "loading")
	func();
else
	document.addEventListener("DOMContentLoaded", func ,true);