Control how many past weeks you want to see and many other enhancements
当前为
// ==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.19.3
// @run-at document-start
// @grant none
// ==/UserScript==
//tab = 2 spaces
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){log(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;
}
let loo = 1000;
let cl = ls("customLinks") || {};
(function loop()
{
if ((typeof(engines) == "undefined" || !engines.length) && --loo)
return setTimeout(loop, 0);
if (!loo)
return;
// 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
let list = [
//begin custom links
/*
{
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
remove: false //set to true if you want clear this item from local storage;
},
{
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
remove: true //set to true if you want clear this item from local storage;
},
{
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
remove: false //set to true if you want clear this item from local storage;
},
*/
//end custom links
];
let listNew = [],
clNew = {},
remove = [];
for(let l = 0; l < list.length; l++)
{
if (cl[list[l].host] && !cl[list[l].host].remove)
cl[list[l].host] = list[l];
}
for(let l in cl)
{
if (!cl[l].remove)
{
delete cl[l].remove;
listNew.push(cl[l]);
clNew[l] = cl[l];
}
else
{
remove.push(l);
}
}
list = [];
for(let i = 0; i < listNew.length; i++)
{
if (remove.indexOf(listNew[i].host) == -1)
{
list.push(listNew[i]);
clNew[listNew[i].host] = listNew[i];
}
}
ls("customLinks", clNew);
window.engines = window.engines.concat(list);
})();
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");
$(list[i][0]).toggleClass("multif", true);
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;
}
body.collapseMulti div.day:not(.expand):not(.opened) div.entry.multif div.title:after
{
content: "";
background-color: black;
border: 1px solid white;
border-right-width: 0;
}
body.collapseMulti div.day:not(.expand):not(.opened) div.entry.multif div.title:after
{
position: absolute;
height: 100%;
right: -1px;
top: -1px;
width: 2px;
}
*/};//css
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]+/, ""),
let title = this._title && this._title._titleOrig ? this._title._titleOrig : el.children("div.title").text();
title = title.replace("?", "").replace(/[0-9]+-[0-9]+-[0-9]+/, "");
// MONKEY = encodeURIComponent( title ),
let 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);
}
$("div.date,div.title").on("mousedown", middleClick);
let _markSearchResults = markSearchResults;
window.markSearchResults = function markSearchResults()
{
if (showMyShows.box)
return
let entries = $("#searchResults").find("div.entry");
entries.each(watched.attach);
$("div.day").each(collapseMulti);
entries.on("mousedown", function(e)
{
middleClick(e, this);
});
return _markSearchResults();
};
//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(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 (o.className.indexOf("archive-link") != -1 && showMyShows.box)
{
$(o).toggleClass("archive-link", false);
o.textContent = "Show all episodes";
o.href = "javascript:search('info:" + id + "');";
}
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 = "";
if (c)
DB.infoAdd(id);
else
DB.infoRemove(id);
if (showMyShows.box)
showMyShows();
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");
});
//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._title && entry._title._titleOrig ? entry._title._titleOrig : $(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();
}
//list of user's shows
DB.infoNameClean = function(entry)
{
let title = entry._title && entry._title._titleOrig ? entry._title._titleOrig : $(entry).find("div.title").text();
return title.substring(0, title.lastIndexOf(" "));
};
DB.infoSave = function()
{
ls("info", DB.info);
};
DB.infoAdd = function(id, name, nosave)
{
if (typeof(name) == "undefined")
{
return DB.infoAdd(id, DB.infoGet(id), nosave);
}
else if (!name)
{
setTimeout(function()
{
DB.infoLoad(id);
}, rand(100, 2000));
return;
}
DB.info[id] = name;
if (!nosave)
DB.infoSave();
};
DB.infoRemove = function(id)
{
delete DB.info[id];
DB.infoSave();
};
DB.infoGet = function(id)
{
let entry = $('div.days div[data-series-id="' + id + '"]').first()[0];
if (entry)
return [DB.infoNameClean(entry), entry.getAttribute("data-series-source")];
return null;
};
DB.infoLoad = function(id)
{
let obj = document.createElement("div");
$(obj).load("/s?"+$.param({q:"info:" + id}), function(e)
{
let title = $(obj).find("b").first().text();
title = title.substring(0, title.lastIndexOf("(") - 1);
if (title)
DB.infoAdd(id, [title, $(obj).find(".entry").attr("data-series-source")]);
});
};
function showMyShows(e)
{
if (!DB.infoLoaded)
{
setTimeout(showMyShows, 100);
return;
}
$(".entry").removeClass("searchResult");
$("#searchStatus").css("visibility","hidden");
showMyShows.box = document.createElement("div");
let entry = document.createElement("div"),
title = document.createElement("div");
entry.appendChild(title);
entry.className = "entry";
title.className = "title";
let list = [];
for(let id in DB.info)
{
list[list.length] = id;
}
list.sort(function(a, b)
{
return DB.info[a][0].toLowerCase().localeCompare(DB.info[b][0].toLowerCase());
});
for(let i = 0; i < list.length; i++)
{
let id = list[i];
entry = entry.cloneNode(true);
title = entry.firstChild;
entry.setAttribute("data-series-id", id);
entry.setAttribute("data-series-source", DB.info[id][1]);
title.textContent = DB.info[id][0];
showMyShows.box.appendChild(entry);
}
$("#searchResults").html(showMyShows.box.innerHTML);
}
$(document).ready(function()
{
let accountLoop = 50;
$("#account-overview").click(function loop()
{
let as = $("#account-popup-content .content a");
if( !$( "#account-popup" ).hasClass("loaded") || (!as.length && accountLoop--))
return setTimeout(loop, 100);
if (!as.length)
return;
$("#account-overview").unbind("click", loop);
let a = document.createElement("a"),
i = document.createTextNode("?");
a.href = '#';
a.addEventListener("click", function()
{
search("info:myshows");
$("#account-popup").toggle();
}, false);
a.textContent = "All my shows";
as[as.length-1].parentNode.insertBefore(a, as[as.length-1].parentNode.lastChild.previousSibling);
a.parentNode.insertBefore(i, a);
});
let repeat = 20,
list = ls("info") || {};
//to avoid hit a ceiling of max data allowed store in local storage, we only save names of user's current shows, nothing else.
DB.info = {};
DB.infoLoaded = false;
//unfortunately we don't know when saved colors are done loading for registered users, so we must wait in a loop
(function loop()
{
let added = false;
for(let id in DB.savedColors)
{
repeat = 0;
if (id in list)
{
DB.info[id] = list[id];
continue;
}
added = true;
DB.infoAdd(id);
}
if (repeat--)
return setTimeout(loop, 300);
DB.infoLoaded = true;
if (added)
DB.infoSave();
})();
//fix paste via right click: adding input event
$( "#searchBecauseNoOneChecks" ).on( "input keyup change search", function(e)
{
let q = this.value.trim();
if (q == "info:myshows" || q.match(/^info:([a-zA-Z]+|$)/))
{
e.preventDefault();
e.stopPropagation();
if (q == "info:myshows")
showMyShows(e);
else
{
showMyShows.box = document.createElement("div");
$("#searchResults").html('<ul id="resultsBecauseNoOneChecks"><li>No results :(</li></ul><small>Search took 0.000 seconds</small>');
}
//fix paste via right click
if (e.type == "input")
$(this).trigger("change");
}
else
showMyShows.box = null;
}).trigger("change");
});//document.ready()
};//func()
function rand(min, max)
{
return Math.floor(Math.random() * (max - min + 1)) + min;
}
//disqus
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);
}
}//disqus
if (document.readyState != "loading")
func();
else
document.addEventListener("DOMContentLoaded", func ,true);