Conductorr

Searches across multiple sources at once.

// ==UserScript==
// @name        Conductorr
// @description Searches across multiple sources at once.
// @namespace   BlackNullerNS
// @include     file:///*/btsearch.html*
// @include     file:///*/conductorr.html*
// @version     1.8.2
// @grant       GM_xmlhttpRequest
// @grant       GM_setValue
// @grant       GM_getValue
// @require     https://cdnjs.cloudflare.com/ajax/libs/jquery/2.1.4/jquery.min.js
// @require     https://cdnjs.cloudflare.com/ajax/libs/twitter-bootstrap/3.3.5/js/bootstrap.min.js
// @require     https://cdnjs.cloudflare.com/ajax/libs/jquery.sticky/1.0.1/jquery.sticky.min.js
// ==/UserScript==

"use strict";

//window.console = {
//    log: function () {},
//    warn: function () {},
//    error: function () {}
//};

/*
 * jQuery Tiny Pub/Sub
 * https://github.com/cowboy/jquery-tiny-pubsub
 *
 * Copyright (c) 2013 "Cowboy" Ben Alman
 * Licensed under the MIT license.
 */

(function ($) {

    var o = $({});

    $.subscribe = function () {
        //console.log("Subscribed", arguments[0]);
        o.on.apply(o, arguments);
    };

    $.unsubscribe = function () {
        //console.log("Unsubscribed", arguments[0]);
        o.off.apply(o, arguments);
    };

    $.publish = function () {
        //console.log("Fired event", arguments[0]);
        return o.trigger.apply(o, arguments);
    };

}(jQuery));

function fmtMSS(s){return(s-(s%=60))/60+(9<s?':':':0')+s}

var SearchEngine = function () {
    var self = this;

    self.sources = {};
    self.sourceCallbacks = {};
    self.pageId = false;
    self.categories = {};
    self.timeout = 8000;
    self.protocols = ["http", "https", "ftp", "magnet", "ed2k"];
    self.resolutions = ["480p","576p","720p","1080p","1080i"];
    self.all = "all";
    self.favorites = "favorites";

    self.nonAlphaNumericRegex = /^[^\u00BF-\u1FFF\u2C00-\uD7FF\w]+$/;
	self.matchNonAlphaNumericRegex = /[^\u00BF-\u1FFF\u2C00-\uD7FF\w]+/g;
	self.matchFirstNonDigit = /[^\d]/;
	self.yearRegexSimple = /(([\ ]{1}|^)(19|20)[\d]{2}([\ ]{1}|$))/g;
	self.yearRegexAdvanced = /(([\ ]{1}|^)(19|20)[\d]{2}\-(19|20)[\d]{2}([\ ]{1}|$)|([\ ]{1}|^)\-(19|20)[\d]{2}([\ ]{1}|$)|([\ ]{1}|^)(19|20)[\d]{2}\-([\ ]{1}|$)|([\ ]{1}|^)(19|20)[\d]{2}([\ ]{1}|$))/g;
	self.imgTagRegex = /<img /g;
	self.dashRegex = /\-/g;
    self.queryRegex = /\{query\}/g;
    self.spacesRegex = /[ ]+/g;

    document.head.innerHTML = "\
		<meta charset=\"utf-8\">\
		<title>Conductorr</title>\
		<link rel=\"shortcut icon\" href=\"\">\
	";

    self.customCSS = document.createElement("style");
	self.customCSS.textContent = "\
        td { font-size: 90%; } \
        h2 { margin-top: 0; margin-bottom: 14px; } \
        #container { min-width: 980px; margin-top: 18px; } \
        #sidebar .btn-group { display:block; } \
        #search-buttons > div { display:block; margin-bottom:3px; } \
        #search-buttons > div::after { display: table; clear:both; content: \" \" } \
        #sticky > div { margin-bottom: 14px; } \
        #search { margin-bottom: 18px; width:100%; padding-right: 22px; } \
        #searchclear { position: absolute; right: 5px; top: 0; bottom: 0; height: 14px; margin: auto; font-size: 14px; cursor: pointer; color: #ccc; } \
        #searchclear:hover { color: #999; } \
		#app-buttons > * { margin-right:6px; } \
        #main .panel:last-of-type { margin-bottom:20px; } \
        .dropdown-menu { padding-left: 7px; padding-right: 7px; } \
        .dropdown-menu button { margin:0 1px 2px 0; } \
        #category-favorites > div { display: block; margin: 0 0 12px 0; }\
        .torrent-table { margin:0; } \
        .torrent-table tr:first-child td { border-top:0; } \
        .torrent-table td { font-size: 95%; padding: 2px !important; } \
        .torrent-group { padding-left: 0; } \
        .icon { background-position: left center; background-repeat: no-repeat; padding-left: 22px; } \
        .icon.link-icon { display: inline-block; margin-bottom: -3px; margin-right:8px; width: 16px; height: 16px; padding-left: 0; } \
        .btn:focus { outline: none; } \
        button.icon { border-radius: 0; border:0; width: 22px; background-position: center center; padding-left: 0; } \
        .failed { background-color: #c00; color: #fff } \
        .result-panel { margin-bottom: 10px; }\
        .result-panel > .panel-heading { padding:6px 8px; cursor: pointer; } \
        .result-panel > .panel-body { padding:0; } \
        .result-panel > .panel-body > table { margin:0; } \
        .result-panel > .panel-body td { padding:5px 8px; } \
        .result-panel > .panel-body td:first-child { width: 80%; word-break: break-all; -webkit-hyphens: auto; -moz-hyphens: auto; hyphens: auto; } \
        .result-panel > .panel-body td:last-child:not(:only-child) { text-align: center; } \
        .result-panel > .panel-body td:not(:first-child):not(:last-child) { text-align: center; color: #999; } \
        .result-panel .label { display: inline-block; margin:0; } \
        .result-panel .tr-title { font-weight:bold; }\
        #no-results::before { content: \"No results: \"; margin-right: 10px; } \
        .panel > .close { padding: 3px 12px; } \
        .panel-primary > .close { color: white; } \
        .cover { padding-left: 90px !important; background-size: 80px auto; background-position: left center; background-repeat: no-repeat; } \
        #source-buttons { margin-bottom:18px; line-height:200%; }\
        #source-buttons > .btn-group { margin-right: 5px; }\
        #source-buttons ul { line-height:150%; }\
        #source-buttons .dropdown-menu a { display:inline-block; font-size: 80%; margin-top:10px; }\
    ";
	document.head.appendChild(self.customCSS);
	self.customCSS = $(self.customCSS);

    self.layout = $(
        '<div class="container" id="container">' +
            '<div class="row">' +
                '<div class="col-md-3 pull-left" id="sidebar">' +
                    '<div id="sticky">' +
                        '<div class="btn-group">' +
                            '<input type="text" class="form-control" id="search">' +
                            '<span id="searchclear" class="glyphicon glyphicon-remove-circle"></span>' +
                        '</div>' +
                        '<div id="search-buttons"></div>' +
                        '<div id="app-buttons"></div>' +
                    '</div>' +
                '</div>' +
                '<div class="col-md-9 pull-right" id="main"></div>' +
            '</div>' +
        '</div>'
    );

    self.input = $('input', self.layout).first();

    self.buttons = $('#search-buttons', self.layout);

    self.mainColumn = $(self.layout[0].firstChild.lastChild)
        .on('click', '.close', function (e) {
            e.stopPropagation(e);
            $(this.parentNode).slideUp("fast", function () {
                $(this).remove();
                if ($('.close', self.mainColumn).length === 0) {
                    resetContent();
                }
            });
        })
        .on('click', '.panel-title > a', function (e) {
            e.stopPropagation();
        })
        .on('click', '.result-panel > .panel-heading', function () {
            $(this.nextElementSibling).slideToggle("fast");
        });

    $('#searchclear', self.layout)
        .on("click", function (e) {
            e.stopPropagation();

            if (self.input.val() === "") {
                resetContent();
            } else {
                self.input.val('');
            }

            self.input.val('').focus();
            document.location.hash = " ";
        });

    self.div = $(document.createElement("div"));
    self.span = $(document.createElement("span"));
    self.table = $(document.createElement("table")).addClass("table");
    self.tableStriped = self.table.clone().addClass("table-striped");
    self.tr = $(document.createElement("tr"));
    self.td = $(document.createElement("td"));
    self.trtd = self.tr.clone().append(self.td.clone());
    self.h2 = $(document.createElement("h2"));
    self.h4 = $(document.createElement("h4"));
    self.a = $(document.createElement("a"));
    self.ab = self.a.clone().attr("target", "_blank");

    self.iconLink = self.ab.clone().addClass("icon");
    self.iconBtn = $('<button type="button" class="btn btn-default btn-xs icon">&nbsp;</button>');
    self.categoryBtn = $('<div class="btn-group btn-group-xs"><button class="btn btn-primary"></button><button class="btn btn-primary dropdown-toggle" data-toggle="dropdown"><span class="caret"></span></button><div class="dropdown-menu"></div></div>');
    self.resultPanel = $('<div class="panel panel-primary result-panel"><div class="panel-heading"><h4 class="panel-title"></h4></div><div class="panel-body"></div></div>');
    self.closeBtn = $('<button type="button" class="close">&times;</button>');
    self.failAlert = $('<div class="alert alert-danger alert-dismissible errors"></div>').append(self.closeBtn.clone());
    self.warningAlert = $('<div id="no-results" class="alert alert-warning alert-dismissible"></div>').append(self.closeBtn.clone());

    self.header = null;
    self.content = null;

    self.persistentData = null;
    self.persistentDataId = "search-persistent-data";

    $.subscribe('source-added', function (e, id) {
        console.log("Added", id);

        if (sourceIsEnabled(id)) {
            self.enableSource(id);
        }
    });

    $.subscribe('source-enabled', function (e, id, data) {
        console.log("Enabled", id);

        data.url[self.favorites] = data.url[Object.keys(data.url)[0]];

        self.sources[id] = $.extend(data, {
            name: id,
            iconCss: $(document.createElement('style'))
                .text('.icon-' + id + ' { background-image:url("' + data.icon + '"); }')
                .insertAfter(self.customCSS)
        });

        var btn, btnId, categories = Object.keys(data.url),
            disabledCategories = getDisabledCategories(id);

        for (var i = 0, len = categories.length; i < len; i++) {
            btnId = id + '__' + categories[i];
            if (disabledCategories.indexOf(categories[i]) === -1) {
                if ($('#' + btnId).length === 0) {
                    btn = self.iconBtn
                        .clone()
                        .addClass('icon-' + id)
                        .attr('id', btnId)
                        .attr('title', id)
                        .data('src', self.sources[id]);
                    categories[i] === self.favorites ? btn.appendTo(getCategoryGroup(categories[i])[0].firstChild) : btn.appendTo(getCategoryGroup(categories[i])[0].lastChild);
                }
            } else {
                $('#' + btnId).remove();
            }
        }

        self.buttons.children(':not(:has("button.icon"))').remove();

        if ("onEnable" in self.sources[id]) {
            self.sources[id].onEnable();
        }
    });

    $.subscribe('source-disabled', function (e, id) {
        console.log("Disabled", arguments);

        $('button.icon-' + id, self.buttons).remove();
        self.buttons.children(':not(:has("button.icon"))').fadeOut(200, function () {
            $(this).remove();
        });
        self.sources[id].iconCss.remove();

        delete self.sources[id];
    });

    self.renderPage = function () {
        $('<link rel="stylesheet" type="text/css">')
            .prependTo(document.head)
            .attr('href', 'https://cdnjs.cloudflare.com/ajax/libs/twitter-bootstrap/3.3.5/css/bootstrap.min.css').load(function () {

                self.buttons
                    .on('click', '.icon', onSourceButtonClick)
                    .on('click', '> div > button:first-of-type', onCategoryButtonClick);

                resetContent();

                $.publish("layout-ready");

                $(document.body).empty().append(self.layout);

                $(function () {
                    var sticky = $('#sticky', self.layout);
                    if ($(window).height() > sticky.height()) {
                        sticky
                            .width(sticky.width())
                            .css("paddingBottom", "20px")
                            .sticky({topSpacing: 18});
                    }
                    self.input.select();
                });

                $(window).on('beforeunload', function () {
                    var data = getPersistentData();
                    $.publish("persistent-save", [data]);
                    GM_setValue(self.persistentDataId, data);
                });

                $.publish("page-rendered");
            });
    };

    var resetContent = function () {
        self.mainColumn.empty();
        self.header = self.h2.clone().hide().appendTo(self.mainColumn);
        self.content = self.div.clone().attr("id", "content").appendTo(self.mainColumn);
        $.publish("after-content-reset");
    };

    var getEnabledSources = function () {
        return self.getPersistentValue("enabledSources", []);
    };

    var sourceIsEnabled = function(id){
        return getEnabledSources().indexOf(id) !== -1;
    };

    var getDisabledCategories = function(name){
        var disabledCategories = self.getPersistentValue("disabledCategories", {});
        return name ? (name in disabledCategories ? disabledCategories[name] : []) : disabledCategories;
    };

    var getCategoryGroup = function (name) {
        var group = self.buttons.children('#category-' + name);

        if (group.length > 0) {
            return group;
        } else {
            var catTitle = (name in self.categories) ? self.categories[name] : name;

            group = self.categoryBtn
                .clone()
                .data("category", name)
                .data("categoryTitle", catTitle)
                .attr("id", "category-" + name);

            group[0].firstChild.textContent = catTitle;

            if (name === self.favorites) {
                group.removeClass("btn-group");
                $(group[0].lastChild).removeClass("dropdown-menu").prependTo(group);
                $(group[0].lastChild).remove();
                group.prependTo(self.buttons);
            } else {
                group.appendTo(self.buttons);
            }

            return group;
        }
    };

    self.addSource = function (id, callback) {
        if (id in self.sourceCallbacks) {
            console.warn(id + ' source is already registered, skipping to avoid overwriting.');
            return;
        }

        self.sourceCallbacks[id] = callback;

        $.publish('source-added', [id, callback]);
    };

    self.enableSource = function (id) {

        console.log("Enabling source " + id);

        var data = self.sourceCallbacks[id]();

        if (!('url' in data) || typeof data.url !== 'object') {
            console.warn('URL definition for ' + id + ' is missing or incorrect, skipping.');
            return;
        }

        var enabledSources = getEnabledSources();

        if (enabledSources.indexOf(id) === -1) {
            enabledSources.push(id);
        }

        $.publish('source-enabled', [id, data]);
    };

    self.disableSource = function (id) {
        console.log("Disabling source " + id);

        var enabledSources = getEnabledSources();
        enabledSources.splice(enabledSources.indexOf(id), 1);

        $.publish('source-disabled', [id]);
    };

    var onCategoryButtonClick = function () {
        var btn = $(this);
        window.scrollTo(0, 0);
        resetContent();

        self.pageId = Date.now();

        self.header
            .html(btn.text() + ' / <i>' + self.input.val() + '</i>')
            .show();

        var buttons = $(".icon", btn.parent());
        var category = btn.parent().data("category");

        $.publish("batch-request", [category, buttons]);

        buttons.each(function () {
            self.sendRequest($(this).data("src"), category, self.input.val());
        });
    };

    var onSourceButtonClick = function (e) {
        e.stopPropagation();

        window.scrollTo(0, 0);

        if (self.pageId) {
            resetContent();
            self.pageId = false;
        }

        var btn = $(this);
        var category = btn.parent().parent().data("category");

        $.publish("single-request", [btn, category]);

        self.sendRequest(btn.data("src"), category, self.input.val());
    };

    self.sendRequest = function (src, category, query, retry) {
        if (typeof src === "string") {
            if (src in self.sources) {
                src = self.sources[src];
            } else {
                console.warn("Unrecognized source", src);
                return false;
            }
        }

        if (!(category in src.url)) {
            console.warn("No " + category + " category found:", src.url);
            return;
        }

		query = query.trim();

        var i = 0, len = 0,
            method = ("method" in src) ? src.method : "GET",
            url = typeof src.url[category] === "string" ? [src.url[category]] : src.url[category].slice(),
            context = {
                valid: true,
                src: src,
                url: url,
                retry: !!retry,
                pageId: self.pageId,
                category: category,
                categoryTitle: (category in self.categories) ? self.categories[category] : category,
                originalQuery: query,
                query: query.replace(self.matchNonAlphaNumericRegex, ' ').replace(self.spacesRegex, ' ').trim()
            };

        if ("onPrepareQuery" in src) {
            if (Array.isArray(src.onPrepareQuery)) {
                for (i = 0, len = src.onPrepareQuery.length; i < len; i++) {
                    src.onPrepareQuery[i](context);
                }
            } else {
                src.onPrepareQuery(context);
            }
        }

		var urlen = context.url.length;

        for (i = 0; i < urlen; i++) {
            if (context.url[i].indexOf("{query}") > -1) {
                context.url[i] = context.url[i].replace(self.queryRegex, context.query);
            } else {
                context.url[i] += context.query;
            }
            context.url[i] = context.url[i].replace(self.spacesRegex, '%20');
        }

        for (i = 0; i < urlen; i++) {
            var requestData = {
                method: method,
                url: context.url[i],
                context: context,
                timeout: self.timeout,
                onload: onRequestSuccess,
                onerror: onRequestFail,
                ontimeout: onRequestTimeout
            };

            if ("onHttpRequest" in src) {
                if (src.onHttpRequest(requestData) === false) {
                    continue;
                }
            }

            console.log(requestData.method, requestData.url);

            GM_xmlhttpRequest(requestData);
        }
    };

    var onRequestSuccess = function (response) {
        if (typeof response !== "object" || !("context" in response)) {
            self.showFailAlert({});
            return;
        }

        if (isOutOfDate(response)) return;

        var i, data, html,
            src = response.context.src,
            originalQuery = response.context.originalQuery,
            pageId = response.context.pageId;

        $.publish('request-finish', [response]);

        if ("onValidate" in src) {
            response.context.valid = src.onValidate(response);
        }

        if (response.finalUrl !== null && response.finalUrl.indexOf("/login") !== -1) {
            response.context.valid = "login needed";
        }

        if (response.context.valid === false || typeof response.context.valid === "string" || !("finalUrl" in response) || typeof response.finalUrl !== "string") {
            self.showFailAlert(response, response.context.valid);
            return;
        }

        if (typeof src.onParse === "function") {
            data = src.onParse(response);
            if (data === null) return;
        } else if (typeof src.onParse === "object") {
            if ("prepare" in src.onParse) {
                html = src.onParse.prepare(response);
            } else {
                html = response.responseText;
            }

            html = self.replaceImages(html);

            try {
                html = $(html);
            } catch (e) {
                console.error("Parsing failed", response.responseText);
                self.showFailAlert(response, "unexpected content");
                return;
            }

            if ('cleanup' in src.onParse) {
                for (i = 0; i < src.onParse.cleanup.length; i++) {
                    $(src.onParse.cleanup[i], html).remove();
                }
            }

            data = $(src.onParse.row, html);
        } else {
            console.error("No parse configuration found for " + src.name, response);
            return;
        }

        if ('onFilter' in src) {
            var filters = Array.isArray(src.onFilter) ? src.onFilter : [src.onFilter];
            for (i = 0; i < filters.length; i++) {
                data = filters[i](data, response);
            }
        }

        var table = self.tableStriped.clone();

        var renderFunction = 'onRender' in src ? src.onRender : renderResults;
        renderFunction(data, table, response);

        var categoryTitle = response.context.categoryTitle;
        var searchUrl = ("searchUrl" in response.context) ? response.context.searchUrl : response.finalUrl;
        var resultLink = self.iconLink.clone()
            .attr("href", searchUrl)
            .addClass('icon-' + src.name);

        if (table[0].childElementCount > 0) {
            var panel = self.resultPanel.clone()
                .addClass('panel-src-' + src.name);

            resultLink.html(src.name + (pageId ? '' : ' / ' + categoryTitle + ' / <i>' + originalQuery + '</i>'));

            if ("length" in data) {
                resultLink.append(' (' + data.length + ')');
            }

            panel[0].firstChild.firstChild.appendChild(resultLink[0]);
            panel.prepend(self.closeBtn.clone());
            panel[0].lastChild.appendChild(table[0]);

            (!response.context.pageId || response.context.retry)
                ? panel.prependTo(self.content)
                : panel.appendTo(self.content);
        } else {
            var noResults = $('#no-results');
            var alert = noResults.length === 0
                ? self.warningAlert.clone().insertBefore(self.content)
                : noResults.first();

            resultLink
                .addClass("link-icon")
                .attr("title", src.name)
                .appendTo(alert);
        }
    };

    var renderResults = function (data, table, response) {
        var src = response.context.src;
		var i = 0, len = 0;

        data.each(function () {
            var n, nlen, context, that = $(this);
            var tr = self.tr.clone();
            var text = "", td, el, elen;
            var texts = [], link = "", linkTest = "", link_prepend = "";
            var freeleech = false, vod = false;

            for (i = 0, len = src.onParse.sel.length; i < len; i++) {
                context = that;
                text = link = link_prepend = "";
                freeleech = vod = false;
                td = self.td.clone();

                if ("class" in src.onParse.sel[i]) {
                    td.addClass(src.onParse.sel[i].class);
                }

                if ("width" in src.onParse.sel[i]) {
                    td.width(src.onParse.sel[i].width);
                }

                if ("align" in src.onParse.sel[i]) {
                    td.css("textAlign", src.onParse.sel[i].align);
                }

                if ("cleanup" in src.onParse.sel[i]) {
                    context = context.clone();
                    for (n = 0, nlen = src.onParse.sel[i].cleanup.length; n < nlen; n++) {
                        $(src.onParse.sel[i].cleanup[n], context).remove();
                    }
                }

                if (typeof src.onParse.sel[i].text === "function") {
                    el = src.onParse.sel[i].text(context);
                } else {
                    el = $(src.onParse.sel[i].text, context);
                }

                if (typeof el === "string") {
                    text = el;
                } else if (el instanceof jQuery) {
					elen = el.length;
                    if (elen === 0) {
                        td.appendTo(tr);
                        continue;
                    } else if (elen === 1) {
                        text = el.text();
                        if (link === "" && el.prop("tagName") === "A") {
                            link = el.attr("href");
                        }
                    } else {
                        texts = [];
                        for (n = 0; n < elen; n++) {
                            texts.push(el.eq(n).text());
                        }
                        text = texts.join(" - ");
                    }
                }

                text = text.trim();

                if (text === "") {
                    td.appendTo(tr);
                    continue;
                }

                if ("freeleech" in src.onParse.sel[i]) {
                    if (typeof src.onParse.sel[i].freeleech === "function") {
                        freeleech = src.onParse.sel[i].freeleech(context);
                    } else {
                        freeleech = $(src.onParse.sel[i].freeleech, context).length > 0;
                    }
                }

                if ("vod" in src.onParse.sel[i]) {
                    if (typeof src.onParse.sel[i].vod === "function") {
                        vod = src.onParse.sel[i].vod(context);
                    } else {
                        vod = $(src.onParse.sel[i].vod, context).length > 0;
                    }
                }

                if ("link" in src.onParse.sel[i]) {
                    if (typeof src.onParse.sel[i].link === "function") {
                        link = src.onParse.sel[i].link(context);
                    } else {
                        link = $(src.onParse.sel[i].link, context);
                    }
                }

                if (link instanceof jQuery) {
                    linkTest = link.attr("href");
                    if (typeof linkTest === "string") {
                        link = linkTest;
                    } else if (link[0] && (link[0].textContent.indexOf("http") === 0)) {
                        link = link[0].textContent;
                    }
                }

                if (typeof link === "string" && link !== "") {
                    if (self.protocols.indexOf(link.split(":")[0]) === -1) {
                        if ("link_prepend" in src.onParse.sel[i]) {
                            link_prepend = src.onParse.sel[i].link_prepend;
                        } else if ("link_prepend" in src.onParse) {
                            link_prepend = src.onParse.link_prepend;
                        }
                    }
                    if (link_prepend !== "") {
                        link = link_prepend + link;
                    }
                    td.html('<a href="' + link + '"' + (("noblank" in src.onParse.sel[i] && src.onParse.sel[i].noblank) ? '' : ' target="_blank"') + '>' + text + '</a>');
                } else {
                    td.html(text);
                }

                if (freeleech) {
                    td.append(' <span class="label label-success">Freeleech</span>');
                } else if (vod) {
                    td.append(' <span class="label label-warning">VOD</span>');
                }

                td.appendTo(tr);
            }

            tr.appendTo(table);
        });
    };

    var isOutOfDate = function (response) {
        return (typeof response !== "object" || !("context" in response) || !("pageId" in response.context) || self.pageId !== response.context.pageId);
    };

    self.showFailAlert = function (response, msg) {
        if (isOutOfDate(response)) return;

        var errors = $('.errors', self.layout);
        var alert = errors.length === 0
            ? self.failAlert.clone().insertBefore(self.content)
            : errors.first();

        if (!("context" in response)) {
            if (!msg) msg = "(error)";
            return alert.append(msg + '&nbsp;&nbsp; ');
        }

        var url = "finalUrl" in response ? response.finalUrl : null;
        var srcName = response.context.src.name;
        var categoryTitle = response.context.pageId ? null : response.context.categoryTitle;

        var target = 'target="_blank"';

        if (url === null) {
            if ("url" in response.context && response.context.url.length > 0) {
                url = response.context.url[0];
            } else {
                url = "javascript:alert('Domain not resolved.');";
                target = '';
            }
            msg = 'domain error';
        }

        if (!msg) msg = "error";

        return alert.append('<a href="' + url + '" ' + target + ' class="icon icon-' + srcName + '">' + srcName + (categoryTitle ? ' / ' + categoryTitle + ' / <i>' + self.input.val() + '</i>' : '') + '</a>&nbsp;(', msg, ')&nbsp;&nbsp; ');
    };

    var onRequestFail = function (response) {
        console.log(response);
        $.publish('request-finish', [response]);
        self.showFailAlert(response, "http request failed");
    };

    var retryTimedOutRequest = function () {
        var context = $(this).data("context");
        self.sendRequest(context.src, context.category, context.query, true);
        $(this).replaceWith('retried');
    };

    var onRequestTimeout = function (response) {
        console.log(response);
        $.publish('request-finish', [response]);
        var retry;
        if ("context" in response) {
            retry = $('<a href="javascript:void(0)" title="click to retry" class="timeout-link">timeout</a>')
                .data("context", response.context)
                .on("click", retryTimedOutRequest);
        } else {
            retry = "timeout";
        }

        self.showFailAlert(response, retry);
    };

    var getPersistentData = function () {
        if (!self.persistentData || typeof self.persistentData !== "object") {
            self.persistentData = GM_getValue(self.persistentDataId, {});
            if (!self.persistentData || typeof self.persistentData !== "object") {
                console.log("Invalid persistent data, resetting.");
                self.persistentData = {};
            }
        }

        return self.persistentData;
    };

    self.getPersistentValue = function (key, def) {
        if (typeof key === "string") {
            var persistentData = getPersistentData();

            if (!(key in persistentData)) {
                persistentData[key] = def;
            }

            return persistentData[key];
        }
    };

    self.setPersistentValue = function (key, value) {
        if (typeof key === "string") {
            var persistentData = getPersistentData();

            persistentData[key] = value;

            return value;
        }
    };

    self.humanizeSize = function (size) {
        size = parseInt(size);
        var i = Math.floor(Math.log(size) / Math.log(1024));
        return (size / Math.pow(1024, i)).toFixed(2) * 1 + ' ' + ['B', 'KB', 'MB', 'GB', 'TB'][i];
    };

    self.requireAllWords = function (data, response) {
        if (typeof response.context.src.onParse !== "object") {
            return data;
        }

        var words = response.context.query.toLowerCase().split(' ');
        var wordsLen = words.length;
        var sel = response.context.src.onParse.sel[0].text;
        var isFunction = typeof sel === "function";

        return data.filter(function () {
            var text;
            if (isFunction) {
                text = sel($(this));
                if (text instanceof jQuery) {
                    text = text.text();
                } else if (text instanceof HTMLElement) {
                    text = text.textContent;
                } else if (typeof text !== "string") {
                    return true;
                }
                text = text.toLowerCase();
            } else {
                text = $(sel, this)[0].textContent.toLowerCase();
            }

            for (var i = 0; i < wordsLen; i++) {
                if (text.indexOf(words[i]) === -1) {
                    return false;
                }
            }

            return true;
        });
    };

	self.filter3dMovies = function (data, response) {
		if (response.context.category.indexOf("movies") === 0 && response.context.category !== "movies_dvd" && response.context.category !== "movies_3d") {
			return data.filter(function(){
				return this.textContent.indexOf('3D') === -1;
			});
		} else {
			return data;
		}
	};

    self.extractResolution = function (context) {
        for (var i = 0; i < self.resolutions.length; i++) {
            if (context.query.indexOf(self.resolutions[i]) !== -1) {
                context.query = context.query.replace(self.resolutions[i], "").replace(self.spacesRegex, ' ').trim();
                return self.resolutions[i];
            }
        }

        return null;
    };

    self.extractGazelleResolution = function (context) {
        var res = self.extractResolution(context);

        if (res) {
            for (var i = 0; i < context.url.length; i++) {
                context.url[i] += "&resolution=" + res;
            }
        }
    };

    self.extractYear = function (context, simple) {
        var re = simple ? self.yearRegexSimple : self.yearRegexAdvanced;
        var year = context.query.match(re);

        if (year && year.length > 0) {
            year = year[year.length - 1].trim();
            context.query = context.query.replace(new RegExp(year.replace(self.dashRegex, '\\\-'), "g"), '').replace(self.spacesRegex, ' ').trim();
        } else {
            year = null;
        }

        return year;
    };

    self.replaceImages = function (text) {
        return text.replace(self.imgTagRegex, '<meta ');
    };

    self.extractGazelleYear = function (context) {
        var year = self.extractYear(context);

        if (year) {
            for (var i = 0, len = context.url.length; i < len; i++) {
                context.url[i] += "&year=" + year;
            }
        }
    };




    ////// SETTINGS

    $.subscribe("layout-ready", function(){
		$("#app-buttons", self.layout)
			.append($('<button type="button" class="btn btn-default btn-xs"><span class="glyphicon glyphicon-cog"></span></button>').on("click", renderConfigPage))
			.append('<a class="btn btn-default btn-xs" href="https://greasyfork.org/en/scripts/12013-conductorr" role="button" target="_blank"><span class="glyphicon glyphicon-globe"></span></a>');
    });

    var renderConfigPage = function () {
        resetContent();

		if (self.pageId === "Settings") {
			self.pageId = false;
			return;
		}

        self.pageId = "Settings";
        self.header.text(self.pageId).show();

        var sourceKeys = Object.keys(self.sourceCallbacks).sort(function (a, b) {
                return a.toLowerCase().localeCompare(b.toLowerCase());
            }),
            srcButtons = self.div.clone().attr("id", "source-buttons").appendTo(self.content),
            btnStub = $('<div class="btn-group"><button type="button" class="btn btn-xs"></button><button type="button" class="btn btn-xs dropdown-toggle" data-toggle="dropdown"><span class="caret"></span></button><ul class="dropdown-menu"></ul></div>'),
            disabledCategories = getDisabledCategories(),
            btn, buttonClass,
            enabled = false,
            partially = false;

        for (var i = 0, len = sourceKeys.length; i < len; i++) {
            enabled = sourceIsEnabled(sourceKeys[i]);
            partially = (sourceKeys[i] in disabledCategories);

            btn = btnStub.clone();
            btn[0].firstChild.textContent = btn[0].dataset.src = sourceKeys[i];
            buttonClass = enabled ? (partially ? "btn-warning" : "btn-success") : "btn-danger";
            btn.children('button').addClass(buttonClass);

            srcButtons.append(btn);

            if (enabled) {
                showSourceConfigBox(sourceKeys[i]);
            }
        }

        $(".btn-group > button:first-child", srcButtons).on("click", function () {
            var name = this.parentNode.dataset.src;
            sourceIsEnabled(name) ? self.disableSource(name) : self.enableSource(name);
        });

        srcButtons.on("change", "input", function () {
            var group = $(this).closest(".btn-group");
            var srcName = group[0].dataset.src;
            var checkboxes = $('input[type="checkbox"]', group);
            var total = checkboxes.length;
            var unchecked = checkboxes.filter(":not(:checked)").length;
            var disabled = getDisabledCategories(srcName);

            if (this.checked) {
                if (disabled.indexOf(this.name) !== -1) {
                    disabledCategories[srcName].splice(disabledCategories[srcName].indexOf(this.name), 1);
                    if (disabledCategories[srcName].length === 0) {
                        delete disabledCategories[srcName];
                    }
                }
                self.enableSource(srcName);
            } else {
                if (disabled.indexOf(this.name) === -1) {
                    if (!(srcName in disabledCategories)) {
                        disabledCategories[srcName] = [];
                    }
                    disabledCategories[srcName].push(this.name);
                }
                total === unchecked
                    ? self.disableSource(srcName)
                    : self.enableSource(srcName);
            }
        });

        $(".btn-group", srcButtons).on("show.bs.dropdown", function () {
            if (this.lastChild.textContent === "") {
                var data = self.sourceCallbacks[this.textContent]();
                var cats = Object.keys(data.url);
                cats.unshift(self.favorites);

                var disabled = this.textContent in disabledCategories ? disabledCategories[this.textContent] : [];
                var catsHTML = '';
                for (var i = 0, len = cats.length; i < len; i++) {
                    catsHTML += '<label class="checkbox-inline"><input type="checkbox" name="' + cats[i] + '"' + (disabled.indexOf(cats[i]) === -1 ? ' checked="checked"' : '') + '> ' + (cats[i] in self.categories ? self.categories[cats[i]] : cats[i]) + '</label><br>';
                }
                //if ("website" in data && data.website) {
                //    catsHTML += '<a href="' + data.website + '" target="_blank">Visit website</a>';
                //}
                this.lastChild.innerHTML = catsHTML;
            }
        });

        $(".dropdown-menu", srcButtons).on("click", function (e) {
            e.stopPropagation();
        });
    };

    var showSourceConfigBox = function (name, buttons) {
        if (!("config" in self.sources[name])) return;

        var box = $("#config-" + name);

        if (box.length === 0) {
            box = $('<div id="config-' + name + '" class="panel panel-default"><div class="panel-heading">' + name + '</div><div class="panel-body"></div></div>')
                .insertAfter($("#source-buttons", self.content));

            $(box[0].lastChild)
                .empty()
                .append(self.sources[name].config());
        }
    };

    $.subscribe("source-disabled", function (e, id) {
        if (self.pageId === "Settings") {
            $('#source-buttons', self.content).children('[data-src="' + id + '"]').children("button").addClass("btn-danger").removeClass("btn-success btn-warning");
            $("#config-" + id, self.content).remove();
        }
    });

    $.subscribe("source-enabled", function (e, id, data) {
        if (self.pageId === "Settings") {
            var className = getDisabledCategories(id).length > 0 ? "btn-warning" : "btn-success";
            $('#source-buttons', self.content).children('[data-src="' + id + '"]').children("button").removeClass("btn-danger btn-warning btn-success").addClass(className);
            showSourceConfigBox(id);
        }
    });

    /////// PERSISTENT CONTENT

    var loadPersistentContent = function () {
        if (self.getPersistentValue("content", "")) {

            self.mainColumn.html(self.getPersistentValue("content", ""));

            $(".timeout-link", self.mainColumn).replaceWith("timeout");

            self.header = $('h2', self.mainColumn);
            if (self.header.length === 0) {
                self.header = self.h2.clone().hide().appendTo(self.mainColumn)
            } else {
                self.header = self.header.first();
            }

            self.content = $('#content', self.mainColumn);
            if (self.content.length === 0) {
                self.content = self.div.clone().attr("id", "content").appendTo(self.mainColumn);
            } else {
                self.content = self.content.first();
            }

            if (self.header.css("display") !== "none") {
                self.pageId = true;
            }
            if (self.header.text() === "Settings") {
                renderConfigPage();
            }
        }
    };

    $.subscribe('layout-ready', loadPersistentContent);

    $.subscribe('persistent-save', function (e, data) {
        data.content = self.mainColumn.html();
    });


    //////// PROGRESS BAR

    var progressBar = null;
    var progressFiller = null;
    var progressSteps = 0;
    var progressStyleSuccess = "progress-bar-success";
    var progressStyleInProcess = "progress-bar-warning";

    var initProgressBar = function () {
        progressBar = $(".progress", self.mainColumn);
        if (progressBar.length === 0) {
            progressBar = $('<div class="progress pull-right"><div class="progress-bar"></div></div>').prependTo(self.mainColumn);
        } else {
            progressBar = progressBar.first();
        }
        progressFiller = $(progressBar[0].firstChild);
        resetProgressBar(true);
    };

    var resetProgressBar = function (hide) {
        if (hide) {
            progressBar.hide();
        }

        progressFiller
            .width(0)
            .data("step", 0)
            .removeClass(progressStyleSuccess)
            .addClass(progressStyleInProcess)
            .text("0%");
        progressSteps = 0;
    };

    var startProgressCount = function (event, category, buttons) {
        resetProgressBar();

        progressSteps = 0;
        buttons.each(function () {
            var src = $(this).data('src');
            if (category in src.url) {
                progressSteps += typeof src.url[category] === "string" ? 1 : src.url[category].length;
            }
        });

        progressBar.show();
    };

    var incrementProgressBar = function (event, response) {
        if ("context" in response && isOutOfDate(response)) return;

        var step = progressFiller.data("step");
        step++;

        if (step === progressSteps) {
            progressFiller
                .width('100%')
                .removeClass(progressStyleInProcess)
                .addClass(progressStyleSuccess)
                .text(step + '/' + step);
//				setTimeout(resetProgressBar, 3000);
        } else {
            var width = (100 / progressSteps * step).toFixed(2);
            if (width > 100) width = 100;
            progressFiller
                .text(step + '/' + progressSteps)
                .width(width + '%')
                .data("step", step);
        }
    };


    self.customCSS.append("\
        .progress { width:120px; margin-top: 7px; margin-bottom:0; }\
        .progress-bar { width:0px; transition: none; }\
    ");

    $.subscribe('layout-ready after-content-reset', initProgressBar);
    $.subscribe('batch-request', startProgressCount);
    $.subscribe('request-finish', incrementProgressBar);


    ///////// HASH NAVIGATION

    var parseQueryString = function (queryString) {
        var params = {}, queries, temp, i, l;

        // Split into key/value pairs
        queries = queryString.split("&");

        // Convert the array of strings into an object
        for (i = 0, l = queries.length; i < l; i++) {
            temp = queries[i].split('=');
            params[temp[0]] = temp[1];
        }

        return params;
    };

    $.subscribe("layout-ready", function(){
        if (document.location.hash.trim().length > 1 && document.location.hash !== self.getPersistentValue("hash", "#")) {
            resetContent();
        }
    });

    $.subscribe("page-rendered", function(){
        var paramsStr = null;

        if (document.location.hash.trim().length > 1 && document.location.hash !== self.getPersistentValue("hash", "#")) {
            paramsStr = document.location.hash.slice(1);
        } else if (document.location.href.indexOf('?') > -1) {
            paramsStr = document.location.href.split('?', 2)[1];
        }

        if (!paramsStr) return;

        var params = parseQueryString(paramsStr);

        if ("q" in params && params.q.trim() !== "") {
            params.q = decodeURIComponent(params.q).replace(/\+/g, ' ');

            self.input.val(params.q);

            if ("cat" in params) {
                if ("src" in params) {
                    $("#" + params.src + "__" + params.cat).click();
                } else {
                    $("#category-" + params.cat).children('button').first().click();
                }
            } else if ("src" in params) {
                $("#" + params.src + "__" + self.all).click();
            }
        }

        self.setPersistentValue("hash", '#' + paramsStr);
    });

    $.subscribe("batch-request", function (e, category) {
        document.location.hash = "#cat=" + (category === self.favorites ? self.all : category) + "&q=" + self.input.val();
        self.setPersistentValue("hash", document.location.hash);
    });

    $.subscribe("single-request", function(e, btn, category){
        document.location.hash = "#cat=" + (category === self.favorites ? self.all : category) + "&src=" + btn.data("src").name + "&q=" + self.input.val();
        self.setPersistentValue("hash", document.location.hash);
    });
};

var bt = new SearchEngine();

/*
 onHttpRequest(requestData): {}.abort()
 onValidate(response): bool|string
 onParse(response)|object: collection|array
 onFilter(data, response): collection|array
 onRender(data, table): void
 */

bt.categories = {
    all: "Everywhere",
    favorites: "Favorites",
    music: "Music",
    music_flac: "Music / FLAC",
    movies: "Movies",
//		movies_hd: "Movies: HD",
    movies_1080: "Movies / 1080p",
    movies_720: "Movies / 720p",
    movies_remux: "Movies / Remuxes",
    movies_bluray: "Movies / Blu-rays",
    movies_dvd: "Movies / DVD",
    mvids: "Music Videos",
    docs: "Documentaries",
    tv: "TV Shows",
    elearning: "E-Learning",
    ebooks: "E-Books",
    games_pc: "Games / PC",
    mags: "Magazines",
    abooks: "Audiobooks",
    fiction: "Fiction",
    comics: "Comics",
    apps_win: "Apps / Win",
    xxx: "XXX"
};
bt.addSource("Adamsfile", function () {
    var adam = "http://adamsfile.com/search.php?s_string={query}&s_year=&albumsSearchSubmit=&s_genre=&s_mediatype=&s_type=";

    return {
        url: {
            all: adam,
            music: adam,
            music_flac: adam
        },
        onParse: {
            cleanup: [".alb_data", ".urating"],
            row: ".albums > tbody > tr",
            link_prepend: "http://adamsfile.com/",
            sel: [
                {text: "> td:eq(0)", link: "a[href*='details.php']:eq(0)"},
                {text: "> td:eq(1)", link: "a[href^='ftp://dl.adamsfile.com/']:eq(0)", noblank: true}
            ]
        },
        onFilter: bt.requireAllWords,
        onValidate: function (response) {
            return response.responseText.indexOf('auth_window') === -1 ? true : "login needed";
        },
        icon: ""
    };
});
bt.addSource("OPS", function () {
    var wcd = "https://orpheus.network/ajax.php?action=browse&searchstr={query}";

    return {
        website: "https://orpheus.network/",
        url: {
            all: wcd,
            music: wcd + "&filter_cat[1]=1",
            music_flac: wcd + "&filter_cat[1]=1&format=FLAC",
            //		music_mp3: wcd + "&filter_cat[1]=1&format=AAC|MP3",
            elearning: wcd + "&filter_cat[3]=1&filter_cat[4]=1&filter_cat[5]=1&filter_cat[7]=1",
            mags: wcd + "&filter_cat[3]=1",
            ebooks: wcd + "&filter_cat[3]=1",
            fiction: wcd + "&filter_cat[3]=1",
            abooks: wcd + "&filter_cat[4]=1",
            comics: wcd + "&filter_cat[7]=1",
            apps_win: wcd + "&filter_cat[2]=1"
        },
        onPrepareQuery: bt.extractGazelleYear,
        onParse: function (response) {
            try {
                var data = JSON.parse(response.responseText);
            } catch (e) {
                bt.showFailAlert(response);
                return null;
            }

            if (!('status' in data) || data.status !== 'success' || !('response' in data) || !('results' in data.response)) {
                bt.showFailAlert(response, "unexpected data");
                return null;
            }

            return data.response.results;
        },
        onFilter: function (data, response) {
            var words = response.context.query.toLowerCase().split(' ');

            response.context.searchUrl = response.finalUrl.replace('ajax.php?action=browse', 'torrents.php?action=basic');

            return data.filter(function (value) {
                for (var i = 0, l = words.length; i < l; i++) {
                    if ((('artist' in value && value.artist.toLowerCase().indexOf(words[i]) === -1) || !('artist' in value)) && value.groupName.toLowerCase().indexOf(words[i]) === -1) {
                        return false;
                    }
                }

                return true;
            });
        },
        onRender: function (data, table) {
            var nl, group, torrent, groupTable, groupTableHTML, torRow, tr, n, score, cue, ed = "", media = "", link, artist;

            var torrentGroupHeader = bt.trtd.clone();
            $(torrentGroupHeader[0].firstChild).append(
                bt.h4.clone().addClass("torrent-group")
            );

            for (var i = 0, l = data.length; i < l; i++) {
				ed = ""; media = "";
                group = data[i];
                link = "https://orpheus.network/torrents.php?id=" + group.groupId;

                if ("torrents" in group) {
                    artist = (group.artist === 'Various Artists')
                        ? 'Various Artists'
                        : bt.ab.clone().attr("href", "https://orpheus.network/artist.php?id=" + group.torrents[0].artists[0].id).html(group.artist);

                    tr = torrentGroupHeader.clone();
                    tr[0].firstChild.setAttribute("colspan", "3");
                    $(tr[0].firstChild.firstChild).append(
                        artist,
                        " - ",
                        bt.ab.clone().attr("href", link).html(group.groupName),
                        ' [' + group.groupYear + '] [' + group.releaseType + ']'
                    );

                    if (group.cover) {
                        $(tr[0].firstChild)
                            .addClass('cover')
                            .css('backgroundImage', 'url(' + group.cover + ')');
                    }

                    groupTableHTML = "";
                    for (n = 0, nl = group.torrents.length; n < nl; n++) {
                        torrent = group.torrents[n];

                        if (ed !== torrent.remasterTitle || media !== torrent.media) {
                            ed = torrent.remasterTitle;
                            media = torrent.media;
                            groupTableHTML += '<tr><td colspan="3" class="tr-title">' + (ed ? ed : "Original Release") + ' ' + torrent.remasterCatalogueNumber + ' ' + (torrent.remasterYear ? torrent.remasterYear : "") + ' / ' + media + "</td></tr>";
                        }

                        score = torrent.hasLog ? ' / Log (' + torrent.logScore + '%)' : '';
                        cue = torrent.hasCue ? ' / Cue' : '';

                        groupTableHTML += '<tr><td><a href="https://orpheus.network/torrents.php?torrentid=' + torrent.torrentId + '" target="_blank">' + torrent.format + ' / ' + torrent.encoding + score + cue + '</a>' + (torrent.isFreeleech ? ' <span class="label label-success">Freeleech</span>' : '') + '</td><td>' + torrent.seeders + '</td><td><a href="https://orpheus.network/torrents.php?action=download&id=' + torrent.torrentId + '">' + bt.humanizeSize(torrent.size) + '</a></td></tr>';
                    }

                    groupTable = bt.table.clone()
                        .addClass("torrent-table")
                        .html(groupTableHTML);

                    torRow = bt.tr.clone().append(bt.td.clone().attr("colspan", "3"));
                    torRow[0].firstChild.appendChild(groupTable[0]);

                    table.append(tr, torRow);
                } else {
                    table.append('<tr><td><a href="' + link + '" target="_blank">' + group.groupName + '</a></td><td>' + group.seeders + '</td><td><a href="https://orpheus.network/torrents.php?action=download&id=' + group.torrentId + '">' + bt.humanizeSize(group.size) + '</a></td></tr>');
                }
            }
        },
        icon: ""
    };
});
bt.addSource("RED", function () {
    var wcd = "https://redacted.ch/ajax.php?action=browse&searchstr={query}";

    return {
        website: "https://redacted.ch/",
        url: {
            all: wcd,
            music: wcd + "&filter_cat[1]=1",
            music_flac: wcd + "&filter_cat[1]=1&format=FLAC",
            //		music_mp3: wcd + "&filter_cat[1]=1&format=AAC|MP3",
            elearning: wcd + "&filter_cat[3]=1&filter_cat[4]=1&filter_cat[5]=1&filter_cat[7]=1",
            mags: wcd + "&filter_cat[3]=1",
            ebooks: wcd + "&filter_cat[3]=1",
            fiction: wcd + "&filter_cat[3]=1",
            abooks: wcd + "&filter_cat[4]=1",
            comics: wcd + "&filter_cat[7]=1",
            apps_win: wcd + "&filter_cat[2]=1"
        },
        onPrepareQuery: bt.extractGazelleYear,
        onParse: function (response) {
            try {
                var data = JSON.parse(response.responseText);
            } catch (e) {
                bt.showFailAlert(response);
                return null;
            }

            if (!('status' in data) || data.status !== 'success' || !('response' in data) || !('results' in data.response)) {
                bt.showFailAlert(response, "unexpected data");
                return null;
            }

            return data.response.results;
        },
        onFilter: function (data, response) {
            var words = response.context.query.toLowerCase().split(' ');

            response.context.searchUrl = response.finalUrl.replace('ajax.php?action=browse', 'torrents.php?action=basic');

            return data.filter(function (value) {
                for (var i = 0, l = words.length; i < l; i++) {
                    if ((('artist' in value && value.artist.toLowerCase().indexOf(words[i]) === -1) || !('artist' in value)) && value.groupName.toLowerCase().indexOf(words[i]) === -1) {
                        return false;
                    }
                }

                return true;
            });
        },
        onRender: function (data, table) {
            var nl, group, torrent, groupTable, groupTableHTML, torRow, tr, n, score, cue, ed = "", media = "", link, artist;

            var torrentGroupHeader = bt.trtd.clone();
            $(torrentGroupHeader[0].firstChild).append(
                bt.h4.clone().addClass("torrent-group")
            );

            for (var i = 0, l = data.length; i < l; i++) {
				ed = ""; media = "";
                group = data[i];
                link = "https://redacted.ch/torrents.php?id=" + group.groupId;

                if ("torrents" in group) {
                    artist = (group.artist === 'Various Artists')
                        ? 'Various Artists'
                        : bt.ab.clone().attr("href", "https://redacted.ch/artist.php?id=" + group.torrents[0].artists[0].id).html(group.artist);

                    tr = torrentGroupHeader.clone();
                    tr[0].firstChild.setAttribute("colspan", "3");
                    $(tr[0].firstChild.firstChild).append(
                        artist,
                        " - ",
                        bt.ab.clone().attr("href", link).html(group.groupName),
                        ' [' + group.groupYear + '] [' + group.releaseType + ']'
                    );

                    if (group.cover) {
                        $(tr[0].firstChild)
                            .addClass('cover')
                            .css('backgroundImage', 'url(' + group.cover + ')');
                    }

                    groupTableHTML = "";
                    for (n = 0, nl = group.torrents.length; n < nl; n++) {
                        torrent = group.torrents[n];

                        if (ed !== torrent.remasterTitle || media !== torrent.media) {
                            ed = torrent.remasterTitle;
                            media = torrent.media;
                            groupTableHTML += '<tr><td colspan="3" class="tr-title">' + (ed ? ed : "Original Release") + ' ' + torrent.remasterCatalogueNumber + ' ' + (torrent.remasterYear ? torrent.remasterYear : "") + ' / ' + media + "</td></tr>";
                        }

                        score = torrent.hasLog ? ' / Log (' + torrent.logScore + '%)' : '';
                        cue = torrent.hasCue ? ' / Cue' : '';

                        groupTableHTML += '<tr><td><a href="https://redacted.ch/torrents.php?torrentid=' + torrent.torrentId + '" target="_blank">' + torrent.format + ' / ' + torrent.encoding + score + cue + '</a>' + (torrent.isFreeleech ? ' <span class="label label-success">Freeleech</span>' : '') + '</td><td>' + torrent.seeders + '</td><td><a href="https://redacted.ch/torrents.php?action=download&id=' + torrent.torrentId + '">' + bt.humanizeSize(torrent.size) + '</a></td></tr>';
                    }

                    groupTable = bt.table.clone()
                        .addClass("torrent-table")
                        .html(groupTableHTML);

                    torRow = bt.tr.clone().append(bt.td.clone().attr("colspan", "3"));
                    torRow[0].firstChild.appendChild(groupTable[0]);

                    table.append(tr, torRow);
                } else {
                    table.append('<tr><td><a href="' + link + '" target="_blank">' + group.groupName + '</a></td><td>' + group.seeders + '</td><td><a href="https://redacted.ch/torrents.php?action=download&id=' + group.torrentId + '">' + bt.humanizeSize(group.size) + '</a></td></tr>');
                }
            }
        },
        icon: ""
    };
});
bt.addSource("Shellife", function () {
    return {
        url: {
            music: "http://shellife.eu/browse.php?search=",
            music_flac: "http://shellife.eu/browse.php?search="
        },
        onParse: {
            cleanup: ["div[id]", ".grey"],
            row: "tr.torrent_row",
            link_prepend: "http://shellife.eu/",
            sel: [
                {
                    text: "> td:eq(1) > a:lt(2)",
                    link: function (context) {
                        var id = $("a[href*='download.php']:eq(0)", context).attr("href").split("id=")[1];
                        return "details.php?id=" + id;
                    }
                },
                {text: "> td:eq(6)"},
                {text: "> td:eq(4)", link: "a[href*='download.php']:eq(0)", noblank: true}
            ]
        },
        onFilter: bt.requireAllWords,
        icon: ""
    };
});
bt.addSource("WFL", function () {
    var wfl = "https://www.waffles.ch/browse.php?q={query}";

    return {
        url: {
            all: wfl,
            music: wfl + ' (FLAC OR MP3 OR AAC)',
            music_flac: wfl + ' FLAC',
            fiction: wfl + "&c86=1",
            mags: wfl + "&c87=1",
            ebooks: wfl + "&c86=1&c87=1",
            abooks: wfl + "&c89=1&c90=1",
            comics: wfl + "&c88=1",
            apps_win: wfl + "&c83=1",
            elearning: wfl + "&c89=1&c90=1&c88=1&c86=1&c87=1&c93=1"
        },
        onParse: {
            row: "#browsetable > tbody > tr:not(:first-child)",
            link_prepend: "https://www.waffles.ch",
            sel: [
                {text: "> td:eq(1) a[href*='/details.php']:eq(0)"},
                {text: "> td:eq(7)"},
                {text: "> td:eq(5)", link: "a[href*='/download.php']:eq(0)", noblank: true}
            ]
        },
        icon: ""
    };
});
bt.addSource("XBT", function () {
    var xbt = "https://xbtmusic.org/torrents-search.php?s=i&b=d&cat=0&active=0&page=0&own=&mode=2&search_pedrovia={query}";

    return {
        url: {
            all: xbt,
            music: xbt,
            music_flac: xbt,
            mvids: [xbt + "&cat=29", xbt + "&cat=201"],
            docs: xbt + "&cat=134",
            abooks: xbt + "&cat=54"
        },
        onParse: {
            cleanup: ["td.small a"],
            row: "table.submain > tbody > tr",
            link_prepend: "https://xbtmusic.org",
            sel: [
                {text: "a[href*='/torrents-details.php?id=']:eq(0)"},
                {text: "> td:eq(2) > table:eq(0) > tbody > tr > td:eq(1) > span:eq(0)"},
                {
                    text: function (context) {
                        return $("> td:eq(2) > table:eq(0) > tbody > tr > td:eq(0)", context).text().split('in')[0];
                    },
                    link: function (context) {
                        var id = $("a[href*='/torrents-details.php?id=']:eq(0)", context).attr('href').split('id=')[1].split(bt.matchFirstNonDigit)[0];
                        return '/download.php?id=' + id;
                    },
                    noblank: true
                }
            ]
        },
        icon: ""
    };
});
bt.addSource("AHD", function () {
    var domain = 'https://awesome-hd.me/';
    var ahd = domain + "torrents.php?action=advanced&order_by=time&order_way=desc&groupname={query}";

    return {
        url: {
            all: ahd,
            movies: ahd,
            movies_1080: ahd + "&filter_cat[1]=1&resolution=1080p",
            movies_720: ahd + "&filter_cat[1]=1&resolution=720p",
            movies_remux: ahd + "&filter_cat[1]=1&media=Blu-ray",
            tv: ahd + "&filter_cat[2]=1",
            docs: ahd
        },
        onPrepareQuery: [
            bt.extractGazelleYear,
            bt.extractGazelleResolution,
        ],
        onParse: {
            row: "tr:has(a[href*='action=download'])",
            link_prepend: domain,
            sel: [
                {
                    text: function (context) {
                        if (context.hasClass('torrent')) {
                            var td = context.find('td:eq(2)').children().first().nextUntil('.tags');
                            return td.append(' ').text().trim();
                        } else {
                            return '<b>' + context.prevAll(".group").first().find("a[href*='id=']").first().text() + '</b> / ' + $("a[href*='torrentid=']:eq(0)", context).text();
                        }
                    },
                    link: function (context) {
                        if (context.hasClass('torrent')) {
                            return context.find("a[title='View Torrent']");
                        } else {
                            return context.prevAll(".group").first().find("a[href*='id=']").first();
                        }
                    },
                },
                {
                    text: "> td:eq(-2)",
                },
                {
                    text: "> td:eq(-4):contains('.')",
                    link: "a[href*='action=download']:eq(0)",
                    noblank: true
                }
            ]
        },
        icon: ""
    };
});
bt.addSource("BHD", function () {
	// NB: Spaces should be replaced with % signs, otherwise search works for an exact phrase

    var bhd = "https://beyond-hd.me/browse.php?incldead=1&searchin=title&search={query}";

    return {
        url: {
            all: bhd,
            movies: bhd + "&c50=1&c77=1&c75=1&c49=1&c94=1&c61=1&c78=1&c86=1&c37=1&c54=1&c17=1",
            movies_1080: bhd + "&c50=1&c77=1&c86=1&c94=1",
            movies_720: bhd + "&c75=1&c78=1&c54=1",
            movies_remux: bhd + "%25remux&c50=1&c77=1&c75=1&c49=1&c94=1&c61=1&c78=1&c86=1&c37=1&c54=1&c17=1",
            movies_bluray: bhd + "&c37=1",
            docs: bhd + "&c50=1&c83=1&c77=1&c75=1&c49=1&c94=1&c61=1&c78=1&c86=1&c37=1&c54=1&c17=1",
            tv: bhd + "&c40=1&c44=1&c48=1&c89=1&c46=1&c45=1",
            mvids: bhd + "&c55=1&c56=1&c42=1",
            music: bhd + "&c36=1&c69=1",
            music_flac: bhd + "&c36=1"
        },
        onParse: {
            row: "table.torrenttable > tbody > tr:gt(0)",
            link_prepend: "https://beyond-hd.me/",
            sel: [
                {text: "a[href*='details.php']:eq(0)"},
                {text: "> td:eq(9)"},
                {text: "> td:eq(7)", link: "a[href*='download.php']:eq(0)", noblank: true}
            ]
        },
		onPrepareQuery: function(context){
            context.query = context.query.replace(bt.spacesRegex, "%25");
		},
		onFilter: bt.filter3dMovies,
        icon: ""
    };
});
bt.addSource("BTN", function () {
    return {
        url: {
            all: "https://broadcasthe.net/torrents.php?searchstr=",
            tv: "https://broadcasthe.net/torrents.php?searchstr=",
            docs: "https://broadcasthe.net/torrents.php?searchstr="
        },
        onHttpRequest: function (requestData) {
            var btnKey = bt.getPersistentValue("BTN_KEY", "").trim();

            var data = {
                method: "getTorrents",
                params: [btnKey, {"Resolution": ["720p", "1080p", "1080i"]}, 50],
                id: Date.now()
            };

            if (requestData.context.query) {
                if (requestData.context.category === "docs") {
                    bt.extractYear(requestData.context);
                    bt.extractResolution(requestData.context);
                    data.params[1] = requestData.context.query;
                } else {
                    bt.extractYear(requestData.context);

                    var res = bt.extractResolution(requestData.context);
                    if (res) {
                        data.params[1].Resolution = res;
                    }

                    data.params[1].Series = '%' + requestData.context.query.trim().replace(bt.spacesRegex, '%') + '%';
                }
            }

            $.extend(requestData, {
                method: "POST",
                url: "https://api.broadcasthe.net/",
                headers: {'Content-Type': 'application/json'},
                data: JSON.stringify(data)
            });

            console.log("BTN request data", JSON.stringify(requestData));
        },
        onParse: function (response) {
            console.log("BTN response data");
            console.log(response.statusText, JSON.stringify(response.responseHeaders));
            console.log(response.responseText);

            try {
                var data = JSON.parse(response.responseText);
            } catch (e) {
                bt.showFailAlert(response);
                return null;
            }

            if (typeof data !== "object" || !("result" in data) || data.result === null || !("results" in data.result)) {
                if ("error" in data && data.error !== null && "message" in data.error) {
                    bt.showFailAlert(response, data.error.message);
                    return null;
                } else {
                    console.warn("Unexpected data from BTN", data, response);
                    return [];
                }
            }

            var num = parseInt(data.result.results);

            response.context.searchUrl = "https://broadcasthe.net/torrents.php?searchstr=" + encodeURIComponent(response.context.query);

            if (num === 0) {
                return [];
            } else {
                data = data.result.torrents;
                data = Object.keys(data).map(function (k) {
                    return data[k];
                });
                return data;
            }
        },
        onRender: function (torrents, table, response) {
            var title, rows = "";

            for (var i = 0, tl = torrents.length; i < tl; i++) {
                title = '<a href="https://broadcasthe.net/torrents.php?id=' + torrents[i].GroupID + '&torrentid=' + torrents[i].TorrentID + '" target="_blank">' + torrents[i].ReleaseName + '.' + torrents[i].Container.toLowerCase() + '</a>';

                if (response.context.category === "docs") {
                    title = '<a href="https://broadcasthe.net/series.php?id=' + torrents[i].SeriesID + '" target="_blank">' + torrents[i].Series + '</a> - ' + title;
                }

                rows += '<tr><td>' + title + '</td><td>' + torrents[i].Seeders + '</td><td><a href="' + torrents[i].DownloadURL + '">' + bt.humanizeSize(torrents[i].Size) + '</a></td></tr>';
            }

            table.html(rows);
        },
        config: function () {
            var form = $('<div class="form-inline"><div class="form-group">Enter your API key from <a href="https://broadcasthe.net/user.php?action=edit#section5.editprofile" target="_blank" style="text-decoration:underline;">here</a>:</div></form>');
            var input = $('<input type="text" class="form-control input-sm" style="width:250px">').val(bt.getPersistentValue("BTN_KEY"));
            var btn = $('<button type="submit" class="btn btn-primary btn-sm">Save</button>')
                .on("click", function () {
                    bt.setPersistentValue("BTN_KEY", input.val().trim());
                    alert("Saved!");
                });

            form
                .append(' ', btn)
                .children().first().append(' ', input);

            return form;
        },
        icon: ""
    };
});
bt.addSource("CG", function () {
    var cg = "https://cinemageddon.net/browse.php?search={query}";

    return {
        url: {
            all: cg,
            movies: cg,
            movies_1080: cg + " 1080",
            movies_720: cg + " 720p",
            movies_remux: cg + " REMUX",
            movies_bluray: [cg + " BD25", cg + " BD50"],
            movies_dvd: cg + " DVD-R",
            docs: cg + "&c15=1",
            music: cg + "&c11=1",
            elearning: cg + "&c19=1&c5=1",
            ebooks: cg + "&c19=1"
        },
        onParse: {
            cleanup: [".torrenttable span"],
            row: ".torrenttable:last > tbody > tr",
            link_prepend: "https://cinemageddon.net/",
            sel: [
                {text: "a[href*='details.php']:eq(0)"},
                {text: "> td:eq(6)"},
                {text: "> td:eq(4)", link: "a[href*='download.php']:eq(0)", noblank: true}
            ]
        },
        icon: ""
    };
});
bt.addSource("DS", function () {
	// NB: Search in multiple categories at once doesn't work
    var ds = "https://www.dvdseed.eu/browse2.php?wheresearch=1&incldead=1&search={query}";

    return {
        url: {
            all: ds,
            movies: ds,
            movies_1080: ds + " 1080p&kopia_reencode=1",
            movies_720: ds + " 720p",
            movies_bluray: ds + "&pkat_bd25=1&pkat_bd50=1&kopia_clone=1&kopia_modify=1&kopia_custom=1",
            movies_dvd: ds + "&pkat_dvd5=1&pkat_dvd9=1&kopia_clone=1&kopia_modify=1&kopia_custom=1",
            tv: [ ds + "&c74=1", ds + "&c31=1" ],
            mvids: [ ds + "&c6=1", "https://www.dvdseed.eu/browse.php?id=3&t=3&c5=1&wheresearch=1&incldead=1&search=" ],
            docs: ds + "&c69=1",
            abooks: ds + "&c98=1",
            apps_win: ds + "&c27=1",
            music: ds + "&c91=1",
            music_flac: ds + "&c91=1",
            xxx: ds + "&c9=1"
        },
        onParse: {
            row: "#torrentable tr:gt(0)",
            link_prepend: "https://www.dvdseed.eu/",
            sel: [
                {
                    text: "a[href*='details.php']:eq(0)",
                    freeleech: "meta[src*='/free.png']"
                },
                {text: "> td:eq(6)"},
                {text: "> td:eq(5)", link: "a[href*='download.php']:eq(0)", noblank: true}
            ]
        },
        onFilter: function (data, response) {
            if (["movies","movies_1080","movies_720","movies_bluray","movies_dvd","tv"].indexOf(response.context.category) !== -1) {
				var stoplist = ["dtsflac","3d1.png","audiobook","music.png","porn.png"];
				var stoplistLength = stoplist.length;
	            return data.filter(function(){
					var html = this.innerHTML;
					for (var i = 0; i < stoplistLength; i++) {
						if (html.indexOf(stoplist[i]) !== -1) {
							return false;
						}
					}
	                return true;
	            });
			} else {
				return data;
			}
        },
        icon: ""
    };
});
bt.addSource("HDB", function () {
    var hdb = "https://hdbits.org/browse.php?descriptions=0&search={query}";

    return {
        url: {
            all: hdb,
            movies: hdb + "&c1=1",
            movies_remux: hdb + "&c1=1&m5=1",
            movies_bluray: hdb + "&c1=1&m1=1",
//			movies_hd: hdb + "&c1=1&m4=1&m3=1&m6=1",
            movies_720: hdb + "+720p&c1=1&m4=1&m3=1&m6=1",
            movies_1080: hdb + "+1080p&c1=1&m4=1&m3=1&m6=1",
            tv: hdb + "&c2=1",
            mvids: hdb + "&c4=1",
            xxx: [hdb + "&c7=1", hdb + "&descriptions=1&c7=1"],
            docs: hdb + "&c3=1"
        },
        onParse: {
            row: "#torrent-list > tbody > tr",
            link_prepend: "https://hdbits.org",
            sel: [
                {
                    text: "> td:eq(2) a:eq(0)",
                    freeleech: "a.fl"
                },
                {text: "> td:eq(7)"},
                {text: "> td:eq(5)", link: "a[href*='download.php']:eq(0)", noblank: true}
            ]
        },
        onFilter: bt.filter3dMovies,
        icon: ""
    };
});
bt.addSource("HDS", function () {
    var hds = "https://hdsky.me/torrents.php?incldead=0&spstate=0&inclbookmarked=0&search_area=0&search_mode=0&search={query}";

    return {
        url: {
            all: hds,
            movies: hds + "&cat401=1&cat410=1&cat405=1",
            movies_1080: hds + "&cat401=1&cat405=1&medium7=1&medium5=1&medium11=1&standard1=1&standard2=1",
            movies_720: hds + "&cat401=1&cat405=1&medium7=1&medium5=1&medium11=1&standard3=1",
            movies_remux: hds + "&cat401=1&cat405=1&medium3=1",
            movies_bluray: hds + "&cat401=1&cat405=1&medium1=1&medium12=1",
            movies_dvd: hds + "&cat401=1&cat405=1&medium6=1",
            docs: hds + "&cat404=1",
            tv: hds + "&cat402=1&cat403=1",
            mvids: hds + "&cat406=1",
            music: hds + "&cat408=1",
            music_flac: hds + "&cat408=1&audiocodec1=1&audiocodec2=1"
        },
        onParse: {
            row: "table.torrents tr.progresstr",
            link_prepend: "https://hdsky.me/",
            sel: [
                {
                    text: "> td:eq(1) a:eq(0)",
                    freeleech: ".pro_free"
                },
                {text: "> td:eq(5)"},
                {text: "> td:eq(4)", link: "a[href*='download.php']:eq(0)", noblank: true}
            ]
        },
		onFilter: bt.filter3dMovies,
        icon: ""
    };
});
bt.addSource("HDT", function () {
    var hdt = "https://hd-torrents.org/torrents.php?active=0&options=0&search={query}";

    return {
        url: {
            all: hdt,
            movies: hdt + "&category[]=1&category[]=2&category[]=5&category[]=3",
            movies_1080: hdt + "&category[]=5",
            movies_720: hdt + "&category[]=3",
            movies_remux: hdt + "&category[]=2",
            movies_bluray: hdt + "&category[]=1",
            music: hdt + "&category[]=44",
            music_flac: hdt + "&category[]=44",
            docs: hdt + "&genre[]=Documentary",
            tv: hdt + "&category[]=59&category[]=60&category[]=30&category[]=38",
            mvids: hdt + "&category[]=61&category[]=62&category[]=57&category[]=45",
            xxx: hdt + "&options=3&category[]=58&category[]=48&category[]=47"
        },
        onParse: {
            row: "table.mainblockcontenttt > tbody > tr:has(a[href*='download.php'])",
            link_prepend: "https://hd-torrents.org/",
            sel: [
                {text: "a:eq(1)"},
                {text: "> td:eq(9)"},
                {text: "> td:eq(7)", link: "a[href*='download.php']:eq(0)", noblank: true}
            ]
        },
        onValidate: function (response) {
            return response.responseText.indexOf("You're not authorized to view this Torrents") === -1 ? true : "login needed";
        },
        onFilter: bt.filter3dMovies,
        icon: ""
    };
});
bt.addSource("KG", function () {
    var kg = "https://karagarga.in/browse.php?search_type=title&incldead=&search={query}";

    return {
        url: {
            all: kg,
            movies: kg + "&cat=1",
            movies_1080: kg + "&hdrip=2",
            movies_720: kg + "&hdrip=1",
            movies_bluray: kg + "&hdrip=3",
            movies_dvd: kg + "&dvdr=1",
            docs: kg + "&genre=20",
            music: kg + "&cat=2",
            elearning: kg + "&cat=3",
            ebooks: kg + "&genre=41",
            abooks: kg + "&genre=40",
            comics: kg + "&genre=42"
        },
        onParse: {
            cleanup: ["#browse tr:has(a[href*='down.php']) span:not(:first-child)"],
            row: "#browse tr:has(a[href*='down.php'])",
            link_prepend: "https://karagarga.in/",
            sel: [
                {
                    text: function (context) {
                        var cols = context.children();
                        var year = cols.eq(3);
                        var genre = cols.eq(0).find("meta");

                        if (genre.length === 2) {
                            var src = genre.eq(1).attr('src');
                            if (~src.indexOf('hdrip720')) year.append(' [720p]');
                            else if (~src.indexOf('hdrip1080')) year.append(' [1080p]');
                            else if (~src.indexOf('dvdr')) year.append(' [DVD-R]');
                            else if (~src.indexOf('bluray')) year.append(' [Blu-ray]');
                        }

                        return cols.slice(1,4);
                    },
                    link: "a[href*='details.php']:eq(0)",
                    freeleech: function(context){
                        return context.hasClass("featuredrow");
                    }
                },
                {text: "> td:eq(12)"},
                {text: "> td:eq(10)", link: "a[href*='down.php']:eq(0)", noblank: true}
            ]
        },
        onPrepareQuery: function (context) {
            var i,
				len = context.url.length,
				year = bt.extractYear(context, true),
				res = bt.extractResolution(context);

            if (res === "720p") {
                for (i = 0; i < len; i++) {
                    context.url[i] += "&hdrip=1";
                }
            } else if (res === "1080p" || res === "1080i") {
                for (i = 0; i < len; i++) {
                    context.url[i] += "&hdrip=2";
                }
            }

            context.query = context.query.trim().split(' ');
            for (i = 0, len = context.query.length; i < len; i++) {
                context.query[i] = encodeURIComponent('+') + context.query[i];
            }
            context.query = context.query.join('%20');

            if (year) {
                context.query += "%20" + year;
            }
        },
//		onFilter: bt.requireAllWords,
        icon: ""
    };
});
bt.addSource("MVGroup", function () {
    var q = [
        "https://forums.mvgroup.org/maintracker.php?filter={query}",
        "https://forums.mvgroup.org/forumtracker.php?filter={query}",
    ];

    return {
        url: {
            all: q,
            docs: q,
        },
        onValidate: function (response) {
            return response.finalUrl.indexOf('tracker.php?filter') === -1 ? "login needed" : true;
        },
        onParse: {
            row: "tr:has(a[href*='?showtopic='])",
            link_prepend: "https://forums.mvgroup.org",
            sel: [
                {
                    text: function (context) {
                        return context[0].firstElementChild.textContent;
                    },
                    link: "a[href*='?showtopic=']",
                },
            ]
        },
        icon: ""
    };
});
bt.addSource("PTP", function () {
    var ptpMovies = "https://passthepopcorn.me/torrents.php?action=advanced&filter_cat[1]=1&filter_cat[2]=1&filter_cat[3]=1&order_by=relevance&grouping=1&searchstr={query}";

    return {
        url: {
            all: "https://passthepopcorn.me/torrents.php?action=basic&order_by=relevance&grouping=1&searchstr={query}",
            movies: ptpMovies,
//			movies_hd: ptpMovies + "&resolution=anyhd",
            movies_dvd: ptpMovies + "&format[]=DVD5&format[]=DVD9&grouping=0",
            movies_bluray: ptpMovies + "&format[]=BD25&format[]=BD50&grouping=0",
            movies_remux: ptpMovies + "&remastertitle=Remux",
            movies_720: ptpMovies + "&resolution=720p",
            movies_1080: ptpMovies + "&resolution=anyhd",
            docs: ptpMovies + "&taglist=documentary",
            mvids: "https://passthepopcorn.me/torrents.php?action=advanced&order_by=relevance&filter_cat[5]=1&searchstr={query}",
            tv: "https://passthepopcorn.me/torrents.php?action=advanced&order_by=relevance&filter_cat[3]=1&searchstr={query}"
        },
        onPrepareQuery: [bt.extractGazelleYear, bt.extractGazelleResolution],
        onParse: function (response) {
            // If redirected to a single search result
            if (response.finalUrl.indexOf('torrents.php?id=') !== -1) {
                var html = $(bt.replaceImages(response.responseText));

                $('.torrent-info-link span', html).remove();

                var torrents = [];

                $(".group_torrent_header", html).each(function () {
                    var row = $(this);
                    var cols = row.children();
                    var href = $("a[title='Permalink']", row).first().attr('href');

                    var title = $(".torrent-info-link", row);
					var seeding = title.hasClass("torrent-info-link--user-seeding");
					title = title.first().text();

                    torrents.push({
                        Title: '<a href="' + href + '"' + (seeding ? ' class="torrent-info-link--user-seeding"' : '') + '>' + title + '</a>',
                        TorrentId: href.split("torrentid=")[1],
                        Size: cols.eq(1).text(),
                        Seeders: cols.eq(3).text()
                    });
                });

                return [{
                    GroupId: response.finalUrl.split('?id=')[1].split(bt.matchFirstNonDigit)[0],
                    Title: html.find('h2.page__title').first().text().split(' [')[0],
                    Year: html.find('h2.page__title').first().text().split(' [')[1].split(']')[0],
                    Cover: $('meta.sidebar-cover-image', html).length > 0 ? $('meta.sidebar-cover-image', html).first().attr('src') : null,
                    GroupingQualities: [{
                        Torrents: torrents
                    }]
                }];
            }

            if (response.responseText.indexOf('PageData = ') === -1) {
                return [];
            }

            var data = response.responseText.split('PageData = ')[1].trim().split('</script>')[0].trim().slice(0, -1);

            try {
                return JSON.parse(data).Movies;
            } catch (e) {
                console.warn("PTP JSON parsing failed.", data);
                return [];
            }
        },
        onEnable: function(){
            if (bt.customCSS.text().indexOf('.torrent-info-link--user-seeding') === -1) {
                bt.customCSS.append(' .torrent-info-link--user-seeding {font-weight:bold;color:darkorange} ');
            }
        },
        onRender: function (movies, table) {
            var groupTable, groupTableHTML, torrent, tr;

            for (var i = 0, ml = movies.length; i < ml; i++) {
                tr = '<tr><td><h4 class="torrent-group"><a href="https://passthepopcorn.me/torrents.php?id=' + movies[i].GroupId + '" target="_blank">' + movies[i].Title + '</a> [' + movies[i].Year + ']</a></td></tr>';

                if (movies[i].Cover) {
                    tr = $(tr);
                    $(tr[0].firstChild)
                        .addClass('cover')
                        .css('backgroundImage', 'url(' + movies[i].Cover + ')');
                }

                if (!('GroupingQualities' in movies[i])) return;

                var groups = movies[i].GroupingQualities;
                groupTable = bt.table.clone().addClass("torrent-table");

                groupTableHTML = "";
                for (var g = 0, gl = groups.length; g < gl; g++) {
                    if ('CategoryName' in groups[g]) {
                        groupTableHTML += '<tr><td colspan=3><b>' + groups[g].CategoryName + ' / ' + groups[g].QualityName + '</b></td></tr>';
                    }

                    for (var n = 0, nl = groups[g].Torrents.length; n < nl; n++) {
                        torrent = groups[g].Torrents[n];

                        var title = bt.span.clone().html(torrent.Title);
                        var a = title.children('a').first();
                        a.attr("target", "_blank");
                        a.attr("href", "https://passthepopcorn.me/" + a.attr("href"));

                        groupTableHTML += '<tr><td>' + title.html() + '</td><td>' + torrent.Seeders + '</td><td><a href="https://passthepopcorn.me/torrents.php?action=download&id=' + torrent.TorrentId + '">' + torrent.Size + '</a></td></tr>';
                    }
                }
                groupTable.html(groupTableHTML);

                var torRow = bt.tr.clone().append(bt.td.clone().append(groupTable));

                table.append(tr, torRow);
            }
        },
        icon: ""
    };
});
bt.addSource("TorViet", function () {
    var torviet = "http://torviet.com/torrents.php?incldead=0&search_area=0&search_mode=0&search={query}";

    return {
        url: {
            all: torviet,
            music: [torviet + "&sltCategory=5&sltSubCategory=126", torviet + "&sltCategory=5&sltSubCategory=130"],
            music_flac: torviet + "&sltCategory=5&sltSubCategory=126",
            //		music_mp3: torviet + "&sltCategory=5&sltSubCategory=130",
            movies: torviet + "&sltCategory=2",
            movies_1080: torviet + "&sltCategory=2&sltSubCategory=125",
            movies_720: torviet + "&sltCategory=2&sltSubCategory=124",
            movies_remux: torviet + " REMUX&sltCategory=2&sltSubCategory=127",
            movies_bluray: torviet + "&sltCategory=2&sltSubCategory=127",
            docs: [torviet + "&sltCategory=3&sltSubCategory=0&sltGenre=62", torviet + "&sltCategory=2&sltSubCategory=0&sltGenre=32"],
            tv: torviet + "&sltCategory=3&sltSubCategory=128",
            elearning: torviet + "&sltCategory=6",
            ebooks: torviet + "&sltCategory=6&sltSubCategory=112",
            abooks: torviet + "&sltCategory=6&sltSubCategory=117",
            mags: torviet + "&sltCategory=6&sltSubCategory=112",
            apps_win: torviet + "&sltCategory=4&sltSubCategory=76",
            mvids: torviet + "&sltCategory=5&sltSubCategory=92",
            games_pc: torviet + "&sltCategory=1&sltSubCategory=7"
        },
        onParse: {
            row: "#idtorrent table.torrents > tbody > tr:not(:first-child)",
            link_prepend: "http://torviet.com",
            sel: [
                {
                    text: ".torrentname a:eq(0)",
                    freeleech: ".pro_free"
                },
                {text: "> td:eq(4)"},
                {text: "> td:eq(3)", link: "a[href*='download.php']:eq(0)", noblank: true}
            ]
        },
		onFilter: [bt.filter3dMovies],
        icon: ""
    };
});
bt.addSource("Vimeo", function () {
    var domain = "https://vimeo.com";
    var q = domain + "/search?q={query}";

    return {
        url: {
            all: q,
            movies: q,
            music: q + "&category=9",
            docs: q + "&category=117&duration=long",
            mvids: q + "&category=9"
        },
        onParse: function (response) {

            try {
                var json = response.responseText;
                json = json.split('var data = ')[1];
                json = json.split('if (0) {')[0].trim().slice(0, -1);
                json = JSON.parse(json);
                json = json.filtered.data;
            } catch (e) {
                bt.showFailAlert(response);
                return null;
            }

            return json;
        },
        onRender: function (data, table) {
            var video, rows = "", id;

            for (var i = 0, len = data.length; i < len; i++) {
                video = data[i];
                rows += '<tr><td style="width:65%"><a href="' + video.clip.link + '" target="_blank" class="vimeo-link">' + video.clip.name + '</a></td><td style="white-space:nowrap;width:10%;">' + fmtMSS(video.clip.duration) + '</td><td style="width:25%;text-align:right;"><a href="' + video.clip.user.link + '" target="_blank">' + video.clip.user.name + '</a></td></tr>';
            }

            table.html(rows);
        },
        onEnable: function () {
            $(bt.mainColumn).on('click', 'a.vimeo-link', function (e) {
                e.preventDefault();
                e.stopPropagation();

                var a = $(this);
                var td = $(this.parentNode);
                var iframe = td.children("iframe");
                if (iframe.length > 0) {
                    iframe.remove();
                } else {
                    bt.mainColumn.find("iframe").remove();
                    var href = a.attr("href");
                    var id = href.split("vimeo.com/")[1].split("?")[0].split("/")[0]; // just in case
                    td.append('<iframe src="https://player.vimeo.com/video/'+ id +'?autoplay=1&color=ffffff&title=0&byline=0&portrait=0&badge=0" width="560" height="315" frameborder="0" style="display:block;margin-top:10px;" allowfullscreen></iframe>');
                }

                return false;
            });
        },
        icon: ""
    };
});
bt.addSource("YouTube", function () {
    var q = "https://m.youtube.com/results?ajax=1&tsp=1&q={query}";

    return {
        url: {
            all: q,
            docs: q,
            music: q,
            movies: q,
            mvids: q,
            games_pc: q
        },
        onHttpRequest: function (requestData) {
            requestData.headers = {
                'User-Agent': 'Mozilla/5.0 (Linux; Android 5.1.1; Nexus 5 Build/LMY48B; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/43.0.2357.65 Mobile Safari/537.36'
            };
        },
        onParse: function(response){
            try {
                var data = JSON.parse(response.responseText.slice(response.responseText.indexOf("{")));
            } catch (e) {
                bt.showFailAlert(response, "unexpected data");
                return null;
            }

            if (!data || !("result" in data) || data.result !== 'ok') {
                bt.showFailAlert(response, "unexpected data");
                return null;
            }

            if ("content" in data && "search_results" in data.content && "contents" in data.content.search_results) {
                return data.content.search_results.contents;
            }

            return [];
        },
        onRender: function (data, table) {
            var video, rows = "", id;
            console.log(data);
            for (var i = 0, len = data.length; i < len; i++) {
                video = data[i];

                switch (video.item_type) {
                    case "compact_video":
                        rows += '<tr><td><a href="https://youtube.com' + video.endpoint.url + '" data-video-id="' + video.encrypted_id + '" target="_blank" class="youtube-link">' + video.title.runs[0].text + '</a></td><td>' + video.length.runs[0].text + '</td><td style="width:20%;text-align:right;">' + video.short_byline.runs[0].text + '</td></tr>';
                        break;
                    case "compact_playlist":
                        rows += '<tr><td><a href="https://youtube.com' + video.endpoint.url + '" data-playlist-id="' + video.playlist_id + '" target="_blank" class="youtube-link">' + video.title.runs[0].text + '</a></td><td style="white-space:nowrap;">' + (video.video_count_short.runs.length ? video.video_count_short.runs[0].text : "no") + ' videos </td><td style="width:20%;text-align:right;">[PLAYLIST]</td></tr>';
                        break;
                    case "compact_channel":
                        rows += '<tr><td><a href="https://youtube.com' + video.endpoint.url + '" data-user-id="' + video.endpoint.url.split("/").pop() + '" target="_blank" class="youtube-link">' + video.title.runs[0].text + '</a></td><td style="white-space:nowrap;">' + (video.video_count.runs.length ? video.video_count.runs[0].text : "no videos") + '</td><td style="width:20%;text-align:right;">[CHANNEL]</td></tr>';
                        break;
                }
            }

            table.html(rows);
        },
        /*
        onParse: {
            row: "ol.item-section > li:has(a[href*='watch?v='])",
            link_prepend: "https://www.youtube.com",
            sel: [
                {
                    text: "h3 > .yt-uix-tile-link",
                    class: "youtube-link",
                    vod: ".yt-badge-ypc"
                },
				{text: ".video-time"},
				{text: ".yt-lockup-byline > .g-hovercard", width:"20%"}
            ]
        },
        */
        onEnable: function(){
            $(bt.mainColumn).on('click', '.youtube-link', function (e) {
                var id, a = $(this), td = $(this.parentNode);
                var iframe = td.children("iframe");
                if (iframe.length > 0) {
                    iframe.remove();
                } else {
                    bt.mainColumn.find("iframe").remove();

                    var url = "https://www.youtube.com/embed/";
                    if (id = a.data("video-id")) {
                        url += id + "?autoplay=1";
                    } else if (id = a.data("playlist-id")) {
                        url += "?listType=playlist&autoplay=1&iv_load_policy=3&list=" + id;
                    } else if (id = a.data("user-id")) {
                        url += "?listType=user_uploads&autoplay=1&iv_load_policy=3&list=" + id;
                    } else {
                        return;
                    }

                    td.append('<iframe width="560" height="315" src="' + url + '" frameborder="0" allowfullscreen style="display:block;margin-top:10px;"></iframe>');
                }

                e.preventDefault();
                e.stopPropagation();
                return false;
            });
        },
        onFilter: function(data, response){
            response.context.searchUrl = "https://www.youtube.com/results?q=" + encodeURIComponent(response.context.query);

            if (response.context.category === "mvids" || response.context.category === "docs") {
                return data.filter(function (v) {
                    if (!~v.item_type.indexOf("compact_")) {
                        return false;
                    }

                    var title = v.title.runs[0].text.toLowerCase();

                    return !(~title.indexOf("audio") || ~title.indexOf("lyric") || ~title.indexOf("album"));
                });
            }

            return data;
        },
        icon: ""
    };
});
bt.addSource("AVG", function () {
    var q = "https://avg.club/tracker.php?max=1&to=1&nm={query}";

    return {
        url: {
            all: q,
            games_pc: q,
        },
        onParse: {
            row: "#tor-tbl > tbody > tr",
            link_prepend: "https://avg.club/",
            sel: [
                {
                    text: "a.tLink",
                },
                {
                    text: "td.seedmed",
                },
                {
                    text: "a.tr-dl",
                    noblank: true,
                }
            ]
        },
        icon: ""
    };
});
bt.addSource("BIB", function () {
    var domain = 'https://bibliotik.me';
    var bib = domain + '/torrents/?search=@title%20{query}';

    return {
        url: {
            all: bib,
            elearning: bib,
            ebooks: bib + "&cat[]=2&cat[]=5",
            mags: bib + "&cat[]=6&cat[]=7",
            apps_win: bib + "&cat[]=1",
            abooks: bib + "&cat[]=3",
            comics: bib + "&cat[]=4"
        },
        onParse: {
            cleanup: ['time', '.taglist'],
            row: "#torrents_table > tbody > tr",
            link_prepend: domain,
            sel: [
                {text: "> td:has(a[href*='/torrents/']):eq(0)", link: "a[href*='/torrents/']:eq(0)"},
                {text: "a[href*='peers']:eq(0)"},
                {
                    text: function (context) {
                        return $("> .t_files_size_added", context).text().split(',')[1].trim();
                    },
                    link: "a[href*='download']:eq(0)",
                    noblank: true
                }
            ]
        },
        icon: ""
    };
});
bt.addSource("BS", function () {
    var bs = "http://bitspyder.net/browse.php?incldead=1&scope=0&search={query}";

    return {
        url: {
            all: bs,
            elearning: bs,
            docs: bs + "&c42=1",
            abooks: bs + "&c40=1",
            mags: bs + "&c57=1",
            ebooks: bs
        },
        onParse: {
            row: "tr.alt1,tr.alt2",
            link_prepend: "http://bitspyder.net/",
            sel: [
                {text: "a.altlink6"},
                {text: "> td:eq(2)"},
                {
                    text: "font[color='#A52A2A']:eq(0)",
                    link: function(context){
                        var link = context[0].firstElementChild.nextElementSibling.firstElementChild;
                        var id = link.getAttribute("href").split("id=")[1].split("&")[0];
                        return "download.php/" + id + "/" + link.textContent.trim().replace(bt.spacesRegex, "%20") + ".torrent";
                    }
                }
            ]
        },
        onPrepareQuery: function (context) {
            context.query = context.query.split(' ');
            for (var i = 0, l = context.query.length; i < l; i++) {
                context.query[i] = encodeURIComponent('+') + context.query[i];
            }
            context.query = context.query.join('%20');
        },
        icon: ""
    };
});
bt.addSource("EBZ", function () {
    var ebz = "http://app.elbitz.net/browse.php?incldead=1&typ=0&search={query}";

    return {
        url: {
            all: ebz,
            elearning: ebz,
            docs: ebz + "&c10=1",
            ebooks: ebz,
            mags: ebz + "&c16=1",
            abooks: ebz + "&c6=1"
        },
        onParse: {
            row: "tr:not(:has('tr')):has(a[href*='download.php'])",
            sel: [
                {
                    text: function (context) {
                        return context.find("a[href*='details.php']:eq(0)").attr('title');
                    },
                    link: "a[href*='details.php']:eq(0)",
                },
                {text: "> td:eq(6)"},
                {text: "> td:eq(5)", link: "a[href*='download.php']:eq(0)", noblank: true}
            ]
        },
        icon: ""
    };
});
bt.addSource("Genesis_Fiction", function () {
    var genf = "http://gen.lib.rus.ec/foreignfiction/?s=";

    return {
        url: {
            all: genf,
            elearning: genf,
            ebooks: genf,
            fiction: genf
        },
        onParse: {
            row: "tr:has(a[href*='foreignfiction/ads.php'])",
            sel: [
                {text: "> td:eq(0), > td:eq(2)"},
                {text: "a[href*='/ads.php']:eq(0)", link: "a[href*='/ads.php']:eq(0)", noblank: true}
            ]
        },
        icon: ""
    };
});
bt.addSource("Genesis_NonFiction", function () {
    var gen = "http://gen.lib.rus.ec/search.php?open=0&view=simple&column=def&req=";

    return {
        url: {
            all: gen,
            elearning: gen,
            ebooks: gen
        },
        onParse: {
            prepare: function (response) {
                if (response.responseText.indexOf('<table width=100% cellspacing=1') === -1) {
                    return "<table></table>";
                }

                var html = response.responseText.replace(bt.imgTagRegex, '<meta ').split('<table width=100% cellspacing=1')[1];
                html = '<table width=100% cellspacing=1' + html;
                html = html.split('</table')[0];
                html += '</table>';

                return html;
            },
            row: "tr:gt(0)",
            link_prepend: "http://gen.lib.rus.ec/",
            sel: [
                {text: "a[href*='book/']:eq(0), > td:eq(8)"},
                {text: "> td:eq(7)", link: "a[href*='/ads.php']:eq(0)", noblank: true}
            ]
        },
        icon: ""
    };
});
bt.addSource("MAM", function () {
    var mam = "https://www.myanonamouse.net/tor/js/loadSearch2.php?tor[text]={query}&tor[srchIn]=0&tor[fullTextType]=old&tor[author]=&tor[series]=&tor[narrator]=&tor[searchType]=all&tor[searchIn]=torrents&tor[cat][]=0&tor[hash]=&tor[sortType]=default&tor[startNumber]=0";

    return {
        url: {
            all: mam,
            elearning: mam,
            abooks: "https://www.myanonamouse.net/tor/js/loadSearch2.php?tor[text]={query}&tor[srchIn]=0&tor[fullTextType]=old&tor[author]=&tor[series]=&tor[narrator]=&tor[searchType]=all&tor[searchIn]=torrents&tor[cat][]=39&tor[cat][]=49&tor[cat][]=50&tor[cat][]=83&tor[cat][]=51&tor[cat][]=97&tor[cat][]=40&tor[cat][]=41&tor[cat][]=106&tor[cat][]=42&tor[cat][]=52&tor[cat][]=98&tor[cat][]=54&tor[cat][]=55&tor[cat][]=43&tor[cat][]=99&tor[cat][]=84&tor[cat][]=44&tor[cat][]=56&tor[cat][]=137&tor[cat][]=45&tor[cat][]=57&tor[cat][]=85&tor[cat][]=87&tor[cat][]=119&tor[cat][]=88&tor[cat][]=58&tor[cat][]=59&tor[cat][]=46&tor[cat][]=47&tor[cat][]=53&tor[cat][]=89&tor[cat][]=100&tor[cat][]=108&tor[cat][]=48&tor[cat][]=111&tor[cat][]=126&tor[cat][]=0&tor[hash]=&tor[sortType]=default&tor[startNumber]=0",
            ebooks: "https://www.myanonamouse.net/tor/js/loadSearch2.php?tor[text]={query}&tor[srchIn]=0&tor[fullTextType]=old&tor[author]=&tor[series]=&tor[narrator]=&tor[searchType]=all&tor[searchIn]=torrents&tor[cat][]=60&tor[cat][]=71&tor[cat][]=72&tor[cat][]=90&tor[cat][]=73&tor[cat][]=101&tor[cat][]=62&tor[cat][]=63&tor[cat][]=107&tor[cat][]=64&tor[cat][]=74&tor[cat][]=102&tor[cat][]=76&tor[cat][]=77&tor[cat][]=65&tor[cat][]=103&tor[cat][]=115&tor[cat][]=91&tor[cat][]=66&tor[cat][]=78&tor[cat][]=138&tor[cat][]=67&tor[cat][]=80&tor[cat][]=92&tor[cat][]=118&tor[cat][]=94&tor[cat][]=120&tor[cat][]=95&tor[cat][]=81&tor[cat][]=82&tor[cat][]=68&tor[cat][]=69&tor[cat][]=75&tor[cat][]=96&tor[cat][]=104&tor[cat][]=109&tor[cat][]=70&tor[cat][]=112&tor[cat][]=0&tor[hash]=&tor[sortType]=default&tor[startNumber]=0",
            mags: "https://www.myanonamouse.net/tor/js/loadSearch2.php?tor[text]={query}&tor[srchIn]=0&tor[fullTextType]=old&tor[author]=&tor[series]=&tor[narrator]=&tor[searchType]=all&tor[searchIn]=torrents&tor[cat][]=79&tor[cat][]=0&tor[hash]=&tor[sortType]=default&tor[startNumber]=0",
            fiction: "https://www.myanonamouse.net/tor/js/loadSearch2.php?tor[text]={query}&tor[srchIn]=0&tor[fullTextType]=old&tor[author]=&tor[series]=&tor[narrator]=&tor[searchType]=all&tor[searchIn]=torrents&tor[cat][]=60&tor[cat][]=71&tor[cat][]=72&tor[cat][]=90&tor[cat][]=61&tor[cat][]=73&tor[cat][]=101&tor[cat][]=62&tor[cat][]=63&tor[cat][]=107&tor[cat][]=64&tor[cat][]=74&tor[cat][]=102&tor[cat][]=76&tor[cat][]=77&tor[cat][]=65&tor[cat][]=103&tor[cat][]=115&tor[cat][]=91&tor[cat][]=66&tor[cat][]=78&tor[cat][]=138&tor[cat][]=67&tor[cat][]=79&tor[cat][]=80&tor[cat][]=92&tor[cat][]=118&tor[cat][]=94&tor[cat][]=120&tor[cat][]=95&tor[cat][]=81&tor[cat][]=82&tor[cat][]=68&tor[cat][]=69&tor[cat][]=75&tor[cat][]=96&tor[cat][]=104&tor[cat][]=109&tor[cat][]=70&tor[cat][]=112&tor[cat][]=0&tor[hash]=&tor[sortType]=default&tor[startNumber]=0",
            comics: "https://www.myanonamouse.net/tor/js/loadSearch2.php?tor[text]={query}&tor[srchIn]=0&tor[fullTextType]=old&tor[author]=&tor[series]=&tor[narrator]=&tor[searchType]=all&tor[searchIn]=torrents&tor[cat][]=61&tor[cat][]=0&tor[hash]=&tor[sortType]=default&tor[startNumber]=0"
        },
        onParse: {
            cleanup: ["a[href*='filelistLink']"],
            row: "tr:gt(0)",
            link_prepend: "https://www.myanonamouse.net",
            sel: [
                {text: "a.title:eq(0)"},
                {text: "> td:eq(6) > p:eq(0)"},
                {
                    text: function(context){
                        return $("> td:eq(4)", context).text().replace('[', '').replace(']', '').trim();
                    },
                    link: "a[href*='download.php']:eq(0)",
                    noblank: true,
                }
            ]
        },
        onValidate: function (response) {
            response.context.searchUrl = response.finalUrl.replace("/tor/js/loadSearch2.php", "/tor/browse.php");
            return true;
        },
        icon: ""
    };
});
bt.addSource("TGZ", function () {
    var tgz = "https://thegeeks.click/browse.php?incldead=1&nonboolean=1&titleonly=1&search={query}";

    return {
        url: {
            all: tgz,
            elearning: tgz,
            docs: tgz + " -pdf -ebook -mp3 -m4a -aac",
            ebooks: tgz + " -xvid -pdtv -hdtv -avi -mp4 -wmv -mkv -dvd",
            abooks: tgz + " AND (audiobook OR mp3 OR m4a OR aac OR flac OR ogg)&nonboolean=3"
        },
        onParse: {
            row: "tr.ttable",
            link_prepend: "https://thegeeks.click/",
            sel: [
                {
                    text: "a[href*='details.php']:eq(0)",
                    freeleech: "font[color='blue']:contains('FREE')"
                },
                {text: "> td:eq(8)"},
                {text: "> td:eq(6)", link: "a[href*='download.php']:eq(0)", noblank: true}
            ]
        },
        onPrepareQuery: function (context) {
            if (context.category === "abooks") {
                context.query = context.query.split(' ').join(' AND ');
            }
        },
        icon: ""
    };
});
bt.addSource("Traum", function () {
    var traum = "http://lib.it.cx/?find=";

    return {
        url: {
            all: traum,
            elearning: traum,
            ebooks: traum,
            fiction: traum
        },
        onParse: {
            row: "tr:has(a[href*='epub']):not(:has('tr'))",
            link_prepend: "http://lib.it.cx",
            sel: [
                {text: "a:eq(0)", noblank: true},
                {text: "> td:last"}
            ]
        },
        icon: ""
    };
});
bt.addSource("bB", function () {
    var bb = "https://baconbits.org/torrents.php?action=basic&searchstr={query}";

    return {
        url: {
            all: bb,
            music: bb + "&filter_cat[1]=1",
            music_flac: bb + "&filter_cat[1]=1",
            movies: bb + "&filter_cat[9]=1",
            movies_1080: bb + "&filter_cat[9]=1",
            movies_720: bb + "&filter_cat[9]=1",
            docs: bb + "&filter_cat[9]=1&filter_cat[10]=1",
            tv: bb + "&filter_cat[10]=1",
            apps_win: bb + "&filter_cat[2]=1",
            mags: bb + "&filter_cat[6]=1",
            games_pc: bb + "&filter_cat[11]=1",
            comics: bb + "&filter_cat[7]=1",
            ebooks: bb + "&filter_cat[3]=1",
            abooks: bb + "&filter_cat[4]=1",
            fiction: bb + "&filter_cat[3]=1",
            elearning: bb + "&filter_cat[3]=1&filter_cat[4]=1&filter_cat[5]=1&filter_cat[6]=1&filter_cat[7]=1&filter_cat[13]=1&filter_cat[14]=1"
        },
        onParse: {
            cleanup: [".tags"],
            row: "#torrent_table tr.torrent",
            link_prepend: "https://baconbits.org/",
            sel: [
                {
                    cleanup: ["> td:eq(1) > span:eq(0)"],
                    text: "> td:eq(1)",
                    link: "a[title='View Torrent']:eq(0)"
//                    freeleech: "strong:contains('Freeleech!')"
                },
                {text: "> td:eq(7)"},
                {text: "> td:eq(4)", link: "a[href*='action=download']:eq(0)", noblank: true}
            ]
        },
		resolution: null,
        onPrepareQuery: function(context){
            var year = bt.extractYear(context);

            if (year) {
                for (var i = 0, l = context.url.length; i < l; i++) {
                    context.url[i] += "&action=advanced&torrentname={query}&filelist=" + year;
                }
            }

			context.src.resolution = bt.extractResolution(context);
        },
        onFilter: [bt.filter3dMovies, function (data, response) {
			if (response.context.category === "music_flac") {
				data = data.filter(function(){
					return this.textContent.indexOf('FLAC') !== -1;
				});
			}

			if (response.context.category === "movies_1080") response.context.src.resolution = "1080p";
			if (response.context.category === "movies_720") response.context.src.resolution = "720p";

			if (response.context.src.resolution) {
				response.context.query += " " + response.context.src.resolution;
				response.context.src.resolution = null;
				return bt.requireAllWords(data, response);
			} else {
				return data;
			}
		}],
        icon: ""
    };
});
bt.addSource("FL", function () {
    var filelist = "https://filelist.ro/browse.php?searchin=1&sort=0&incldead=1&search={query}";

    return {
        url: {
            all: filelist,
            music: filelist + "&cat=11",
            music_flac: filelist + " FLAC&cat=11",
            movies: filelist,
            movies_bluray: filelist + "&cat=20",
            movies_remux: filelist + " REMUX",
            movies_1080: [filelist + " 1080p&cat=4", filelist + " 1080p&cat=19", filelist + " 1080p&cat=15"],
            movies_720: [filelist + " 720p&cat=4", filelist + " 720p&cat=19", filelist + " 720p&cat=15"],
            movies_dvd: [filelist + "&cat=2", filelist + "&cat=3"],
            docs: filelist + " documentary&searchin=0",
            ebooks: [filelist + "&cat=16", filelist + "&cat=18"],
            mags: filelist + "&cat=16",
            tv: [filelist + "&cat=15&sort=2", filelist + "&cat=21&sort=2", filelist + "&cat=14&sort=2"],
            elearning: filelist,
            apps_win: filelist + "&cat=8",
            mvids: filelist + "&cat=12",
            xxx: filelist + "&searchin=0&cat=7"
        },
        onParse: {
            row: ".torrentrow",
            link_prepend: "https://filelist.ro/",
            sel: [
                {
                    text: "a[href*='details.php?id=']:eq(0)",
                    freeleech: "meta[alt='FreeLeech']"
                },
                {text: "> .torrenttable:eq(8)"},
                {text: "> .torrenttable:eq(6)", link: "a[href*='download.php']:eq(0)", noblank: true}
            ]
        },
        onFilter: [bt.filter3dMovies, function(data, response){
			if (response.context.category === "movies_bluray") {
				return data.filter(function(){
					return this.textContent.toLowerCase().indexOf('remux') === -1;
				});
			} else {
				return data;
			}
		}],
        icon: ""
    };
});
bt.addSource("PS", function () {
    var ps = "https://polishsource.cz/browse.php?incldead=1&scene=0&pl=0&sub=&search_in=title&search={query}";

    return {
        url: {
            all: ps,
            movies: ps + "&c12=1&c11=1&c4=1&c43=1",
            movies_1080: ps + " 1080&c11=1",
            movies_720: ps + " 720p&c11=1",
            movies_remux: ps + " REMUX&c43=1",
            movies_bluray: ps + "&c43=1",
            movies_dvd: ps + "&c4=1",
            docs: ps + "&sub=Documentary",
            music: ps + "&c42=1",
            music_flac: ps + " FLAC&c42=1",
            //		music_mp3: ps + "+MP3&c42=1",
            mvids: ps + " x264&c42=1",
            tv: ps + "&c39=1",
            elearning: ps + "&c5=1",
            ebooks: ps + "&c5=1",
            abooks: ps + " audiobook&search_in=both&c5=1",
            mags: ps + "&c5=1",
            apps_win: ps + "&c18=1",
            games_pc: ps + "&c8=1",
            xxx: ps + "&c13=1"
        },
        onParse: {
            row: "#restable tr:gt(0)",
            link_prepend: "https://polishsource.cz/",
            sel: [
                {text: "a[href*='details.php']:eq(0)"},
                {text: "> td:eq(7)"},
                {text: "> td:eq(4)", link: "a[href*='downloadssl.php']:eq(0)", noblank: true}
            ]
        },
		onFilter: function(data, response){
			if (response.context.category === "movies_bluray") {
				return data.filter(function(){
					return this.textContent.toLowerCase().indexOf('remux') === -1;
				});
			} else {
				return data;
			}
		},
        icon: ""
    };
});
bt.addSource("RarBG", function () {
    var rarbg = "https://rarbg.to/torrents.php?search={query}";

    return {
        url: {
            all: rarbg,
            movies: rarbg + "&category=14;48;17;44;45;42;46",
            movies_1080: rarbg + "&category=44",
            movies_720: rarbg + "&category=45",
            movies_bluray: rarbg + "&category=42",
            movies_remux: rarbg + "&category=46",
            tv: rarbg + "&category=41",
            docs: [rarbg + " documentary&category=movies", rarbg + "&category=18;41"],
            music: rarbg + "&category=23;25",
            music_flac: rarbg + "&category=25",
            //		music_mp3: rarbg + "&category=23",
            elearning: rarbg + "&category=35",
            ebooks: rarbg + "&category=35",
            xxx: rarbg + "&category=4",
            games_pc: rarbg + "&category=27;28",
            apps_win: rarbg + "&category=33"
        },
        onParse: {
            row: ".lista2t tr.lista2",
            link_prepend: "https://rarbg.to",
            sel: [
                {text: "td:eq(1) a:eq(0)"},
                {text: "> td:eq(4)"},
                {
                    text: "> td:eq(3)",
                    link: function (context) {
                        var link = context[0].firstElementChild.nextElementSibling.firstElementChild;
                        var id = link.getAttribute("href").split("/").pop();
                        return '/download.php?id=' + id + '&f=' + encodeURIComponent(link.textContent.trim()) + '.torrent';
                    },
                    noblank: true
                }
            ]
        },
        onValidate: function (response) {
            return response.finalUrl.indexOf('/bot_check.php') === -1 ? true : "captcha";
        },
		onFilter: bt.requireAllWords,
        icon: ""
    };
});
bt.addSource("Rutor", function () {
    var rutor = "http://live-rutor.org/search/";

    return {
        url: {
            all: rutor,
            movies: [rutor + "0/1/100/0/", rutor + "0/5/100/0/", rutor + "0/7/100/0/"],
            movies_1080: rutor + "0/0/100/0/1080p ",
            movies_720: rutor + "0/0/100/0/720p ",
            movies_remux: rutor + "0/0/300/0/remux|bdremux ",
            movies_bluray: rutor + "0/0/310/0/bdinfo -bdremux -remux -hdtv -bdrip ",
            movies_dvd: rutor + "0/0/300/0/dvd|dvd5|dvd9|dvdr ",
            music: rutor + "0/2/100/0/",
            music_flac: rutor + "0/2/100/0/FLAC ",
            mvids: rutor + "0/2/300/0/-mp3 -flac -ape -aac -alac ",
            tv: rutor + "0/0/300/0/720p|1080p ",
            docs: [rutor + "0/12/100/0/", rutor + "0/6/100/0/"],
            games_pc: rutor + "0/8/100/0/",
            apps_win: rutor + "0/9/100/0/",
            elearning: rutor,
            ebooks: rutor + "0/0/300/0/pdf|epub|fb2|djvu|chm|mobi|doc ",
            fiction: rutor + "0/0/300/0/pdf|epub|fb2|djvu|chm|mobi|doc ",
            abooks: rutor + "0/11/300/0/flac|mp3 ",
            comics: rutor + "0/0/300/0/CBZ|CBR ",
            mags: rutor + "0/0/100/0/PDF "
        },
        onParse: {
            row: "#index > table:eq(0) > tbody > tr:gt(0)",
            link_prepend: "http://live-rutor.org",
            sel: [
                {text: "a[href*='/torrent/']:eq(0)"},
                {text: "> td:last > span:eq(0)"},
                {text: "> td:eq(-2)", link: "a[href*='/download/']:eq(0)", noblank: true}
            ]
        },
        onFilter: [bt.filter3dMovies,bt.requireAllWords],
        icon: ""
    };
});
bt.addSource("RuTracker", function () {
    var tru = "http://rutracker.org/forum/tracker.php?nm={query}";
    var truMovies = "&f=100,101,1235,124,1543,1576,1577,1666,1670,187,1900,208,209,2090,2091,2092,2093,212,2198,2199,22,2200,2201,2220,2221,2258,2339,2343,2365,2459,312,313,376,4,484,505,521,539,572,7,709,822,905,93,930,934,941";
    var truDocs = "&f=103,1114,1280,1327,1453,1467,1468,1469,1475,2076,2107,2112,2123,2159,2160,2168,2176,2177,2178,2323,2380,249,251,2538,294,314,46,500,552,56,670,671,672,752,821,851,876,97,98";
    var truFiction = "&f=2039,2041,2042,2043,2044,2045,2047,2080,2193,2355,2356,2357,2474";
    var truMvids = "&f=1107,1121,1122,1141,1142,1174,1189,1227,1228,1455,1775,1777,1781,1782,1783,1787,1788,1789,1790,1791,1792,1793,1794,1795,1812,1886,1887,1912,1913,1990,2088,2089,2241,2261,2262,2263,2264,2271,2304,2305,2306,2351,2352,2377,2378,2379,2383,2384,2426,2507,2508,2509,2510,2529,2530,2531,2532,2534,431,442,445,475,655,702,983,984,986,988";
    var truAbooks = "&f=1036,1279,1501,1580,2152,2165,2324,2325,2326,2327,2328,2342,2387,2388,2389,2413,399,400,401,402,403,490,499,525,530,574,695,716";
    var truGamesPC = "&f=1008,1098,127,128,139,2067,2115,2117,2118,2119,2142,2143,2145,2146,2147,2155,2187,2203,2204,2225,2226,2227,2228,2385,240,2415,246,2478,2479,2480,2481,2482,2483,2484,2485,2533,278,5,50,51,52,53,54,55,635,637,642,643,644,645,646,647,649,650,761,900,959,960,961,962";
    var truAppsWin = "&f=1012,1013,1014,1016,1018,1019,1021,1025,1027,1028,1029,1030,1031,1032,1033,1034,1035,1038,1039,1040,1041,1042,1051,1052,1053,1054,1055,1056,1057,1058,1060,1061,1062,1063,1064,1065,1066,1067,1068,1071,1073,1079,1080,1081,1082,1083,1084,1085,1086,1087,1088,1089,1090,1091,1092,1192,1193,1199,1204,1503,1507,1508,1509,1510,1511,1512,1513,1514,1515,1516,1517,1526,1536,1636,2077,2153";
    var truComics = "&f=2461,2462,2463,2464,2465,2473,862";

    return {
        url: {
            all: tru,
            music: tru + " lossless|MP3|AAC|OGG|FLAC|APE|ALAC|WV",
            music_flac: tru + " lossless|FLAC|APE|ALAC|WV",
            //		music_mp3: tru + "+MP3|AAC|OGG",
            movies: tru + truMovies,
            movies_1080: tru + " 1080p -remux -bdremux -disc" + truMovies,
            movies_720: tru + " 720p -remux -bdremux -disc" + truMovies,
            movies_remux: tru + " remux|bdremux" + truMovies,
            movies_bluray: tru + " \"blu ray\" | bluray -DVD -DVD5 -DVD9 -rip -remux -bdremux" + truMovies,
            movies_dvd: tru + " DVD|DVD5|DVD9" + truMovies,
            tv: tru + " 720p|1080p",
            mvids: tru + truMvids,
            games_pc: tru + truGamesPC,
            apps_win: tru + truAppsWin,
            docs: tru + truDocs,
            elearning: tru,
            ebooks: tru + " PDF|EPUB|MOBI|CHM|DJVU|FB2",
            abooks: tru + truAbooks,
            mags: tru + " PDF",
            comics: tru + truComics,
            fiction: tru + truFiction
        },
        onEnable: function () {
            $(bt.mainColumn).on('click', 'a[href^="http://dl.rutracker.org/"]', function () {
                $('<form target="_blank" method="post" action="' + $(this).attr("href") + '"></form>')
                    .appendTo(document.body)
                    .submit()
                    .remove();
                return false;
            });
        },
        onParse: {
            row: "#tor-tbl > tbody > tr:not(:has('td.pad_12'))",
            link_prepend: "http://rutracker.org/forum/",
            sel: [
                {text: "a.tLink"},
                {text: ".seedmed"},
                {
                    text: function (context) {
                        var size = context.children().eq(-5)[0].firstElementChild.textContent;
                        return bt.humanizeSize(size);
                    },
                    link: "> td.tor-size > a",
                    noblank: true
                }
            ]
        },
		onFilter: bt.filter3dMovies,
        icon: ""
    };
});
bt.addSource("TL", function () {
    var tl = "https://www.torrentleech.org/torrents/browse/index/query/{query}";

    return {
        url: {
            all: tl,
            movies: tl + "/categories/1,8,9,10,11,12,13,14,15,29,35",
            movies_1080: tl + " 1080p/categories/13,14,35",
            movies_720: tl + " 720p/categories/13,14,35",
            movies_bluray: tl + " (AVC OR VC-1 OR BD25 OR BD50 OR COMPLETE OR M2TS OR ISO) -x264 -re-encode -re-encoded -bdremux -remux -3D -720p/categories/13,14,35",
            movies_remux: tl + " (REMUX OR BDREMUX)/categories/13,14,35",
            movies_dvd: tl + "/categories/12",
            docs: tl + "/categories/29",
            tv: tl + "/categories/27,32,35",
            mvids: tl + "/categories/16",
            elearning: tl + "/categories/5",
            games_pc: tl + "/categories/17",
            apps_win: tl + "/categories/6,23,33",
            ebooks: tl + "/categories/5"
        },
        onParse: {
            row: "#torrenttable > tbody > tr",
            link_prepend: "https://www.torrentleech.org",
            sel: [
                {text: "a:eq(1)"},
                {text: "> td:eq(6)"},
                {text: "> td:eq(4)", link: "a[href*='/download/']:eq(0)", noblank: true}
            ]
        },
        onValidate: function (response) {
            return response.responseText.indexOf('/user/account/signup') === -1 ? true : "login needed";
        },
        onFilter: [bt.requireAllWords, bt.filter3dMovies, function (data, response) {
            if (["elearning", "ebooks"].indexOf(response.context.category) !== -1) {
                return data;
            } else {
                return bt.requireAllWords(data, response);
            }
        }],
        icon: ""
    };
});
bt.addSource("TPB", function () {
    var q = "https://thepiratebay.org/search/{query}";

    return {
        url: {
            all: q,
            music: [ q + "/0/99/101", q + "/0/99/104" ],
            music_flac: q + "/0/99/104",
            movies: [q + "/0/99/201", q + "/0/99/202"],
            movies_bluray: q + " avc/0/99/200",
            movies_remux: q + " remux/0/99/200",
            movies_1080: q + " 1080p/0/99/207",
            movies_720: q + " 720p/0/99/207",
            movies_dvd: q + "/0/99/202",
            tv: [q + "/0/3/205", "/0/3/208"],
            docs: q,
            mvids: q + "/0/3/203",
            apps_win: q + "/0/3/301",
            games_pc: q + "/0/3/401",
            elearning: q,
            ebooks: q + "/0/99/601",
            abooks: q + "/0/99/102",
            fiction: q + "/0/99/601",
            mags: q + "/0/3/601",
            comics: q + "/0/3/602",
            xxx: q + "/0/3/500"
        },
        onParse: {
            row: "#searchResult > tbody > tr",
            link_prepend: "https://thepiratebay.org",
            sel: [
                {text: "a[href*='/torrent/']:eq(0)"},
                {text: "> td:eq(-2)"},
                {
                    text: function(context){
                        var details = context.find(".detDesc").first();
                        if (details.length === 1 && ~details.text().indexOf(", Size ")) {
                            return details.text().split(", Size ")[1].split(",")[0].trim();
                        } else {
                            return $(context[0].lastElementChild.previousElementSibling.previousElementSibling);
                        }
                    },
                    link: "a[href^='magnet:']:eq(0)",
                    noblank: true
                }
            ]
        },
        onFilter: [bt.filter3dMovies, function (data, response) {
            if (response.context.category === "movies_bluray") {
	            return data.filter(function(){
                    return this.textContent.toLowerCase().indexOf("remux") === -1;
	            });
			} else {
				return data;
			}
        }],
        icon: ""
    };
});
bt.addSource("BlueBird", function () {
    var bird = "http://bluebird-hd.org/browse.php?incldead=1&stype=and&search={query}";

    return {
        url: {
            all: bird,
            movies: bird + "&c1=1&c2=1",
            movies_1080: bird + "&c1=1&c2=1&cr3=1&cr5=1",
            movies_720: bird + "&c1=1&c2=1&cr4=1&cr6=1",
            movies_remux: bird + "&c1=1&c2=1&cr2=1",
            movies_bluray: bird + "&c1=1&c2=1&cr1=1",
            music: bird + "&cr7=1",
            music_flac: bird + "&cr7=1",
            tv: bird + "&c6=1",
            mvids: bird + "&c4=1",
            docs: bird + "&c3=1",
            xxx: bird + "&c7=1"
        },
        onParse: {
            cleanup: ["a[href*='&snatched']"],
            row: "#highlighted > tr",
            link_prepend: "http://bluebird-hd.org/",
            sel: [
                {text: "a[href*='details.php']:eq(0)"},
                {text: "> td:eq(-3)"},
                {text: "> td:last", link: "a[href*='download.php']:eq(0)", noblank: true}
            ]
        },
        onPrepareQuery: function (context) {
            var i,
				len = context.url.length,
				year = bt.extractYear(context, true);

            if (year && year.length === 4) {
                for (i = 0; i < len; i++) {
                    context.url[i] += "&dsearch=" + year;
                }
            }

            var res = bt.extractResolution(context);

            if (res === "720p") {
                for (i = 0; i < len; i++) {
                    context.url[i] += "&cr4=1&cr6=1";
                }
            } else if (res === "1080p" || res === "1080i") {
                for (i = 0; i < len; i++) {
                    context.url[i] += "&cr3=1&cr5=1";
                }
            }
        },
        onFilter: bt.filter3dMovies,
        icon: ""
    };
});
bt.addSource("MoreThanTV", function () {
    var q = "https://www.morethan.tv/ajax.php?action=browse&order_by=time&searchsubmit=1&searchstr={query}";

    return {
        url: {
            all: q,
            tv: q + "&filter_cat[2]=1",
            docs: q + "&taglist=documentary"
        },
        onPrepareQuery: function(context){
            var i,
                len = context.url.length,
                res = bt.extractResolution(context);

            if (res) {
                for (i = 0; i < len; i++) {
                    context.url[i] += "&encoding=" + res;
                }
            }

            var year = bt.extractYear(context);

            if (year) {
                for (i = 0; i < len; i++) {
                    context.url[i] += "&filelist=" + year;
                }
            }
        },
        onParse: function (response) {
            try {
                var data = JSON.parse(response.responseText);
            } catch (e) {
                bt.showFailAlert(response);
                return null;
            }

            if (!('status' in data) || data.status !== 'success' || !('response' in data) || !('results' in data.response)) {
                bt.showFailAlert(response, "unexpected data");
                return null;
            }

            response.context.searchUrl = response.finalUrl.replace("ajax.php?action=browse", "torrents.php?action=basic");

            return data.response.results;
        },
        onRender: function (torrents, table) {
            var title, rows = "";

            for (var i = 0, l = torrents.length; i < l; i++) {
                title = '<a href="https://www.morethan.tv/torrents.php?id=' + torrents[i].groupId + '&torrentid=' + torrents[i].torrentId + '" target="_blank">' + torrents[i].groupName + '</a>';

                if (torrents[i].fileCount > 1) {
                    title += ' (' + torrents[i].fileCount + ' files)';
                }

                if (torrents[i].isFreeleech) {
                    title += ' <span class="label label-success">Freeleech</span>';
                }

                rows += '<tr><td>' + title + '</td><td>' + torrents[i].seeders + '</td><td><a href="https://www.morethan.tv/torrents.php?action=download&id=' + torrents[i].torrentId + '">' + bt.humanizeSize(torrents[i].size) + '</a></td></tr>';
            }

            table.html(rows);
        },
        icon: ""
    };
});
bt.addSource("AR", function () {
    var ar = "https://alpharatio.cc/torrents.php?action=basic&searchstr={query}";

    return {
        url: {
            all: ar,
            movies: ar + "&filter_cat[6]=1&filter_cat[7]=1&filter_cat[8]=1&filter_cat[9]=1",
            movies_1080: ar + " 1080p&filter_cat[7]=1&filter_cat[9]=1",
            movies_720: ar + " 720p&filter_cat[7]=1&filter_cat[9]=1",
            movies_bluray: ar + " COMPLETE&filter_cat[7]=1",
            movies_remux: ar + " REMUX&filter_cat[7]=1",
            docs: ar + "&filter_cat[1]=1&filter_cat[2]=1&filter_cat[3]=1&filter_cat[6]=1&filter_cat[7]=1&filter_cat[24]=1",
            tv: ar + "&filter_cat[2]=1&filter_cat[3]=1&filter_cat[5]=1",
            mvids: ar + "&filter_cat[11]=1",
            elearning: ar + "&filter_cat[21]=1&filter_cat[22]=1&filter_cat[24]=1",
            ebooks: ar + "&filter_cat[21]=1",
            abooks: ar + "&filter_cat[22]=1",
            games_pc: ar + "&filter_cat[12]=1",
            apps_win: ar + "&filter_cat[16]=1",
            music: ar + "&filter_cat[23]=1",
            music_flac: ar + " FLAC&filter_cat[23]=1",
            xxx: ar + "&filter_cat[10]=1&filter_cat[20]=1"
        },
        onParse: {
            row: "#torrent_table tr.torrent",
            link_prepend: "https://alpharatio.cc/",
            sel: [
                {text: ".group_info > a:eq(0)"},
                {text: "> td:eq(7)"},
                {text: "> td:eq(5)", link: "a[href*='action=download']:eq(0)", noblank: true}
            ]
        },
		onFilter: bt.filter3dMovies,
        icon: ""
    };
});
bt.renderPage();