Conductorr

Searches across multiple sources at once.

  1. // ==UserScript==
  2. // @name Conductorr
  3. // @description Searches across multiple sources at once.
  4. // @namespace BlackNullerNS
  5. // @include file:///*/btsearch.html*
  6. // @include file:///*/conductorr.html*
  7. // @version 1.8.2
  8. // @grant GM_xmlhttpRequest
  9. // @grant GM_setValue
  10. // @grant GM_getValue
  11. // @require https://cdnjs.cloudflare.com/ajax/libs/jquery/2.1.4/jquery.min.js
  12. // @require https://cdnjs.cloudflare.com/ajax/libs/twitter-bootstrap/3.3.5/js/bootstrap.min.js
  13. // @require https://cdnjs.cloudflare.com/ajax/libs/jquery.sticky/1.0.1/jquery.sticky.min.js
  14. // ==/UserScript==
  15.  
  16. "use strict";
  17.  
  18. //window.console = {
  19. // log: function () {},
  20. // warn: function () {},
  21. // error: function () {}
  22. //};
  23.  
  24. /*
  25. * jQuery Tiny Pub/Sub
  26. * https://github.com/cowboy/jquery-tiny-pubsub
  27. *
  28. * Copyright (c) 2013 "Cowboy" Ben Alman
  29. * Licensed under the MIT license.
  30. */
  31.  
  32. (function ($) {
  33.  
  34. var o = $({});
  35.  
  36. $.subscribe = function () {
  37. //console.log("Subscribed", arguments[0]);
  38. o.on.apply(o, arguments);
  39. };
  40.  
  41. $.unsubscribe = function () {
  42. //console.log("Unsubscribed", arguments[0]);
  43. o.off.apply(o, arguments);
  44. };
  45.  
  46. $.publish = function () {
  47. //console.log("Fired event", arguments[0]);
  48. return o.trigger.apply(o, arguments);
  49. };
  50.  
  51. }(jQuery));
  52.  
  53. function fmtMSS(s){return(s-(s%=60))/60+(9<s?':':':0')+s}
  54.  
  55. var SearchEngine = function () {
  56. var self = this;
  57.  
  58. self.sources = {};
  59. self.sourceCallbacks = {};
  60. self.pageId = false;
  61. self.categories = {};
  62. self.timeout = 8000;
  63. self.protocols = ["http", "https", "ftp", "magnet", "ed2k"];
  64. self.resolutions = ["480p","576p","720p","1080p","1080i"];
  65. self.all = "all";
  66. self.favorites = "favorites";
  67.  
  68. self.nonAlphaNumericRegex = /^[^\u00BF-\u1FFF\u2C00-\uD7FF\w]+$/;
  69. self.matchNonAlphaNumericRegex = /[^\u00BF-\u1FFF\u2C00-\uD7FF\w]+/g;
  70. self.matchFirstNonDigit = /[^\d]/;
  71. self.yearRegexSimple = /(([\ ]{1}|^)(19|20)[\d]{2}([\ ]{1}|$))/g;
  72. 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;
  73. self.imgTagRegex = /<img /g;
  74. self.dashRegex = /\-/g;
  75. self.queryRegex = /\{query\}/g;
  76. self.spacesRegex = /[ ]+/g;
  77.  
  78. document.head.innerHTML = "\
  79. <meta charset=\"utf-8\">\
  80. <title>Conductorr</title>\
  81. <link rel=\"shortcut icon\" href=\"\">\
  82. ";
  83.  
  84. self.customCSS = document.createElement("style");
  85. self.customCSS.textContent = "\
  86. td { font-size: 90%; } \
  87. h2 { margin-top: 0; margin-bottom: 14px; } \
  88. #container { min-width: 980px; margin-top: 18px; } \
  89. #sidebar .btn-group { display:block; } \
  90. #search-buttons > div { display:block; margin-bottom:3px; } \
  91. #search-buttons > div::after { display: table; clear:both; content: \" \" } \
  92. #sticky > div { margin-bottom: 14px; } \
  93. #search { margin-bottom: 18px; width:100%; padding-right: 22px; } \
  94. #searchclear { position: absolute; right: 5px; top: 0; bottom: 0; height: 14px; margin: auto; font-size: 14px; cursor: pointer; color: #ccc; } \
  95. #searchclear:hover { color: #999; } \
  96. #app-buttons > * { margin-right:6px; } \
  97. #main .panel:last-of-type { margin-bottom:20px; } \
  98. .dropdown-menu { padding-left: 7px; padding-right: 7px; } \
  99. .dropdown-menu button { margin:0 1px 2px 0; } \
  100. #category-favorites > div { display: block; margin: 0 0 12px 0; }\
  101. .torrent-table { margin:0; } \
  102. .torrent-table tr:first-child td { border-top:0; } \
  103. .torrent-table td { font-size: 95%; padding: 2px !important; } \
  104. .torrent-group { padding-left: 0; } \
  105. .icon { background-position: left center; background-repeat: no-repeat; padding-left: 22px; } \
  106. .icon.link-icon { display: inline-block; margin-bottom: -3px; margin-right:8px; width: 16px; height: 16px; padding-left: 0; } \
  107. .btn:focus { outline: none; } \
  108. button.icon { border-radius: 0; border:0; width: 22px; background-position: center center; padding-left: 0; } \
  109. .failed { background-color: #c00; color: #fff } \
  110. .result-panel { margin-bottom: 10px; }\
  111. .result-panel > .panel-heading { padding:6px 8px; cursor: pointer; } \
  112. .result-panel > .panel-body { padding:0; } \
  113. .result-panel > .panel-body > table { margin:0; } \
  114. .result-panel > .panel-body td { padding:5px 8px; } \
  115. .result-panel > .panel-body td:first-child { width: 80%; word-break: break-all; -webkit-hyphens: auto; -moz-hyphens: auto; hyphens: auto; } \
  116. .result-panel > .panel-body td:last-child:not(:only-child) { text-align: center; } \
  117. .result-panel > .panel-body td:not(:first-child):not(:last-child) { text-align: center; color: #999; } \
  118. .result-panel .label { display: inline-block; margin:0; } \
  119. .result-panel .tr-title { font-weight:bold; }\
  120. #no-results::before { content: \"No results: \"; margin-right: 10px; } \
  121. .panel > .close { padding: 3px 12px; } \
  122. .panel-primary > .close { color: white; } \
  123. .cover { padding-left: 90px !important; background-size: 80px auto; background-position: left center; background-repeat: no-repeat; } \
  124. #source-buttons { margin-bottom:18px; line-height:200%; }\
  125. #source-buttons > .btn-group { margin-right: 5px; }\
  126. #source-buttons ul { line-height:150%; }\
  127. #source-buttons .dropdown-menu a { display:inline-block; font-size: 80%; margin-top:10px; }\
  128. ";
  129. document.head.appendChild(self.customCSS);
  130. self.customCSS = $(self.customCSS);
  131.  
  132. self.layout = $(
  133. '<div class="container" id="container">' +
  134. '<div class="row">' +
  135. '<div class="col-md-3 pull-left" id="sidebar">' +
  136. '<div id="sticky">' +
  137. '<div class="btn-group">' +
  138. '<input type="text" class="form-control" id="search">' +
  139. '<span id="searchclear" class="glyphicon glyphicon-remove-circle"></span>' +
  140. '</div>' +
  141. '<div id="search-buttons"></div>' +
  142. '<div id="app-buttons"></div>' +
  143. '</div>' +
  144. '</div>' +
  145. '<div class="col-md-9 pull-right" id="main"></div>' +
  146. '</div>' +
  147. '</div>'
  148. );
  149.  
  150. self.input = $('input', self.layout).first();
  151.  
  152. self.buttons = $('#search-buttons', self.layout);
  153.  
  154. self.mainColumn = $(self.layout[0].firstChild.lastChild)
  155. .on('click', '.close', function (e) {
  156. e.stopPropagation(e);
  157. $(this.parentNode).slideUp("fast", function () {
  158. $(this).remove();
  159. if ($('.close', self.mainColumn).length === 0) {
  160. resetContent();
  161. }
  162. });
  163. })
  164. .on('click', '.panel-title > a', function (e) {
  165. e.stopPropagation();
  166. })
  167. .on('click', '.result-panel > .panel-heading', function () {
  168. $(this.nextElementSibling).slideToggle("fast");
  169. });
  170.  
  171. $('#searchclear', self.layout)
  172. .on("click", function (e) {
  173. e.stopPropagation();
  174.  
  175. if (self.input.val() === "") {
  176. resetContent();
  177. } else {
  178. self.input.val('');
  179. }
  180.  
  181. self.input.val('').focus();
  182. document.location.hash = " ";
  183. });
  184.  
  185. self.div = $(document.createElement("div"));
  186. self.span = $(document.createElement("span"));
  187. self.table = $(document.createElement("table")).addClass("table");
  188. self.tableStriped = self.table.clone().addClass("table-striped");
  189. self.tr = $(document.createElement("tr"));
  190. self.td = $(document.createElement("td"));
  191. self.trtd = self.tr.clone().append(self.td.clone());
  192. self.h2 = $(document.createElement("h2"));
  193. self.h4 = $(document.createElement("h4"));
  194. self.a = $(document.createElement("a"));
  195. self.ab = self.a.clone().attr("target", "_blank");
  196.  
  197. self.iconLink = self.ab.clone().addClass("icon");
  198. self.iconBtn = $('<button type="button" class="btn btn-default btn-xs icon">&nbsp;</button>');
  199. 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>');
  200. 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>');
  201. self.closeBtn = $('<button type="button" class="close">&times;</button>');
  202. self.failAlert = $('<div class="alert alert-danger alert-dismissible errors"></div>').append(self.closeBtn.clone());
  203. self.warningAlert = $('<div id="no-results" class="alert alert-warning alert-dismissible"></div>').append(self.closeBtn.clone());
  204.  
  205. self.header = null;
  206. self.content = null;
  207.  
  208. self.persistentData = null;
  209. self.persistentDataId = "search-persistent-data";
  210.  
  211. $.subscribe('source-added', function (e, id) {
  212. console.log("Added", id);
  213.  
  214. if (sourceIsEnabled(id)) {
  215. self.enableSource(id);
  216. }
  217. });
  218.  
  219. $.subscribe('source-enabled', function (e, id, data) {
  220. console.log("Enabled", id);
  221.  
  222. data.url[self.favorites] = data.url[Object.keys(data.url)[0]];
  223.  
  224. self.sources[id] = $.extend(data, {
  225. name: id,
  226. iconCss: $(document.createElement('style'))
  227. .text('.icon-' + id + ' { background-image:url("' + data.icon + '"); }')
  228. .insertAfter(self.customCSS)
  229. });
  230.  
  231. var btn, btnId, categories = Object.keys(data.url),
  232. disabledCategories = getDisabledCategories(id);
  233.  
  234. for (var i = 0, len = categories.length; i < len; i++) {
  235. btnId = id + '__' + categories[i];
  236. if (disabledCategories.indexOf(categories[i]) === -1) {
  237. if ($('#' + btnId).length === 0) {
  238. btn = self.iconBtn
  239. .clone()
  240. .addClass('icon-' + id)
  241. .attr('id', btnId)
  242. .attr('title', id)
  243. .data('src', self.sources[id]);
  244. categories[i] === self.favorites ? btn.appendTo(getCategoryGroup(categories[i])[0].firstChild) : btn.appendTo(getCategoryGroup(categories[i])[0].lastChild);
  245. }
  246. } else {
  247. $('#' + btnId).remove();
  248. }
  249. }
  250.  
  251. self.buttons.children(':not(:has("button.icon"))').remove();
  252.  
  253. if ("onEnable" in self.sources[id]) {
  254. self.sources[id].onEnable();
  255. }
  256. });
  257.  
  258. $.subscribe('source-disabled', function (e, id) {
  259. console.log("Disabled", arguments);
  260.  
  261. $('button.icon-' + id, self.buttons).remove();
  262. self.buttons.children(':not(:has("button.icon"))').fadeOut(200, function () {
  263. $(this).remove();
  264. });
  265. self.sources[id].iconCss.remove();
  266.  
  267. delete self.sources[id];
  268. });
  269.  
  270. self.renderPage = function () {
  271. $('<link rel="stylesheet" type="text/css">')
  272. .prependTo(document.head)
  273. .attr('href', 'https://cdnjs.cloudflare.com/ajax/libs/twitter-bootstrap/3.3.5/css/bootstrap.min.css').load(function () {
  274.  
  275. self.buttons
  276. .on('click', '.icon', onSourceButtonClick)
  277. .on('click', '> div > button:first-of-type', onCategoryButtonClick);
  278.  
  279. resetContent();
  280.  
  281. $.publish("layout-ready");
  282.  
  283. $(document.body).empty().append(self.layout);
  284.  
  285. $(function () {
  286. var sticky = $('#sticky', self.layout);
  287. if ($(window).height() > sticky.height()) {
  288. sticky
  289. .width(sticky.width())
  290. .css("paddingBottom", "20px")
  291. .sticky({topSpacing: 18});
  292. }
  293. self.input.select();
  294. });
  295.  
  296. $(window).on('beforeunload', function () {
  297. var data = getPersistentData();
  298. $.publish("persistent-save", [data]);
  299. GM_setValue(self.persistentDataId, data);
  300. });
  301.  
  302. $.publish("page-rendered");
  303. });
  304. };
  305.  
  306. var resetContent = function () {
  307. self.mainColumn.empty();
  308. self.header = self.h2.clone().hide().appendTo(self.mainColumn);
  309. self.content = self.div.clone().attr("id", "content").appendTo(self.mainColumn);
  310. $.publish("after-content-reset");
  311. };
  312.  
  313. var getEnabledSources = function () {
  314. return self.getPersistentValue("enabledSources", []);
  315. };
  316.  
  317. var sourceIsEnabled = function(id){
  318. return getEnabledSources().indexOf(id) !== -1;
  319. };
  320.  
  321. var getDisabledCategories = function(name){
  322. var disabledCategories = self.getPersistentValue("disabledCategories", {});
  323. return name ? (name in disabledCategories ? disabledCategories[name] : []) : disabledCategories;
  324. };
  325.  
  326. var getCategoryGroup = function (name) {
  327. var group = self.buttons.children('#category-' + name);
  328.  
  329. if (group.length > 0) {
  330. return group;
  331. } else {
  332. var catTitle = (name in self.categories) ? self.categories[name] : name;
  333.  
  334. group = self.categoryBtn
  335. .clone()
  336. .data("category", name)
  337. .data("categoryTitle", catTitle)
  338. .attr("id", "category-" + name);
  339.  
  340. group[0].firstChild.textContent = catTitle;
  341.  
  342. if (name === self.favorites) {
  343. group.removeClass("btn-group");
  344. $(group[0].lastChild).removeClass("dropdown-menu").prependTo(group);
  345. $(group[0].lastChild).remove();
  346. group.prependTo(self.buttons);
  347. } else {
  348. group.appendTo(self.buttons);
  349. }
  350.  
  351. return group;
  352. }
  353. };
  354.  
  355. self.addSource = function (id, callback) {
  356. if (id in self.sourceCallbacks) {
  357. console.warn(id + ' source is already registered, skipping to avoid overwriting.');
  358. return;
  359. }
  360.  
  361. self.sourceCallbacks[id] = callback;
  362.  
  363. $.publish('source-added', [id, callback]);
  364. };
  365.  
  366. self.enableSource = function (id) {
  367.  
  368. console.log("Enabling source " + id);
  369.  
  370. var data = self.sourceCallbacks[id]();
  371.  
  372. if (!('url' in data) || typeof data.url !== 'object') {
  373. console.warn('URL definition for ' + id + ' is missing or incorrect, skipping.');
  374. return;
  375. }
  376.  
  377. var enabledSources = getEnabledSources();
  378.  
  379. if (enabledSources.indexOf(id) === -1) {
  380. enabledSources.push(id);
  381. }
  382.  
  383. $.publish('source-enabled', [id, data]);
  384. };
  385.  
  386. self.disableSource = function (id) {
  387. console.log("Disabling source " + id);
  388.  
  389. var enabledSources = getEnabledSources();
  390. enabledSources.splice(enabledSources.indexOf(id), 1);
  391.  
  392. $.publish('source-disabled', [id]);
  393. };
  394.  
  395. var onCategoryButtonClick = function () {
  396. var btn = $(this);
  397. window.scrollTo(0, 0);
  398. resetContent();
  399.  
  400. self.pageId = Date.now();
  401.  
  402. self.header
  403. .html(btn.text() + ' / <i>' + self.input.val() + '</i>')
  404. .show();
  405.  
  406. var buttons = $(".icon", btn.parent());
  407. var category = btn.parent().data("category");
  408.  
  409. $.publish("batch-request", [category, buttons]);
  410.  
  411. buttons.each(function () {
  412. self.sendRequest($(this).data("src"), category, self.input.val());
  413. });
  414. };
  415.  
  416. var onSourceButtonClick = function (e) {
  417. e.stopPropagation();
  418.  
  419. window.scrollTo(0, 0);
  420.  
  421. if (self.pageId) {
  422. resetContent();
  423. self.pageId = false;
  424. }
  425.  
  426. var btn = $(this);
  427. var category = btn.parent().parent().data("category");
  428.  
  429. $.publish("single-request", [btn, category]);
  430.  
  431. self.sendRequest(btn.data("src"), category, self.input.val());
  432. };
  433.  
  434. self.sendRequest = function (src, category, query, retry) {
  435. if (typeof src === "string") {
  436. if (src in self.sources) {
  437. src = self.sources[src];
  438. } else {
  439. console.warn("Unrecognized source", src);
  440. return false;
  441. }
  442. }
  443.  
  444. if (!(category in src.url)) {
  445. console.warn("No " + category + " category found:", src.url);
  446. return;
  447. }
  448.  
  449. query = query.trim();
  450.  
  451. var i = 0, len = 0,
  452. method = ("method" in src) ? src.method : "GET",
  453. url = typeof src.url[category] === "string" ? [src.url[category]] : src.url[category].slice(),
  454. context = {
  455. valid: true,
  456. src: src,
  457. url: url,
  458. retry: !!retry,
  459. pageId: self.pageId,
  460. category: category,
  461. categoryTitle: (category in self.categories) ? self.categories[category] : category,
  462. originalQuery: query,
  463. query: query.replace(self.matchNonAlphaNumericRegex, ' ').replace(self.spacesRegex, ' ').trim()
  464. };
  465.  
  466. if ("onPrepareQuery" in src) {
  467. if (Array.isArray(src.onPrepareQuery)) {
  468. for (i = 0, len = src.onPrepareQuery.length; i < len; i++) {
  469. src.onPrepareQuery[i](context);
  470. }
  471. } else {
  472. src.onPrepareQuery(context);
  473. }
  474. }
  475.  
  476. var urlen = context.url.length;
  477.  
  478. for (i = 0; i < urlen; i++) {
  479. if (context.url[i].indexOf("{query}") > -1) {
  480. context.url[i] = context.url[i].replace(self.queryRegex, context.query);
  481. } else {
  482. context.url[i] += context.query;
  483. }
  484. context.url[i] = context.url[i].replace(self.spacesRegex, '%20');
  485. }
  486.  
  487. for (i = 0; i < urlen; i++) {
  488. var requestData = {
  489. method: method,
  490. url: context.url[i],
  491. context: context,
  492. timeout: self.timeout,
  493. onload: onRequestSuccess,
  494. onerror: onRequestFail,
  495. ontimeout: onRequestTimeout
  496. };
  497.  
  498. if ("onHttpRequest" in src) {
  499. if (src.onHttpRequest(requestData) === false) {
  500. continue;
  501. }
  502. }
  503.  
  504. console.log(requestData.method, requestData.url);
  505.  
  506. GM_xmlhttpRequest(requestData);
  507. }
  508. };
  509.  
  510. var onRequestSuccess = function (response) {
  511. if (typeof response !== "object" || !("context" in response)) {
  512. self.showFailAlert({});
  513. return;
  514. }
  515.  
  516. if (isOutOfDate(response)) return;
  517.  
  518. var i, data, html,
  519. src = response.context.src,
  520. originalQuery = response.context.originalQuery,
  521. pageId = response.context.pageId;
  522.  
  523. $.publish('request-finish', [response]);
  524.  
  525. if ("onValidate" in src) {
  526. response.context.valid = src.onValidate(response);
  527. }
  528.  
  529. if (response.finalUrl !== null && response.finalUrl.indexOf("/login") !== -1) {
  530. response.context.valid = "login needed";
  531. }
  532.  
  533. if (response.context.valid === false || typeof response.context.valid === "string" || !("finalUrl" in response) || typeof response.finalUrl !== "string") {
  534. self.showFailAlert(response, response.context.valid);
  535. return;
  536. }
  537.  
  538. if (typeof src.onParse === "function") {
  539. data = src.onParse(response);
  540. if (data === null) return;
  541. } else if (typeof src.onParse === "object") {
  542. if ("prepare" in src.onParse) {
  543. html = src.onParse.prepare(response);
  544. } else {
  545. html = response.responseText;
  546. }
  547.  
  548. html = self.replaceImages(html);
  549.  
  550. try {
  551. html = $(html);
  552. } catch (e) {
  553. console.error("Parsing failed", response.responseText);
  554. self.showFailAlert(response, "unexpected content");
  555. return;
  556. }
  557.  
  558. if ('cleanup' in src.onParse) {
  559. for (i = 0; i < src.onParse.cleanup.length; i++) {
  560. $(src.onParse.cleanup[i], html).remove();
  561. }
  562. }
  563.  
  564. data = $(src.onParse.row, html);
  565. } else {
  566. console.error("No parse configuration found for " + src.name, response);
  567. return;
  568. }
  569.  
  570. if ('onFilter' in src) {
  571. var filters = Array.isArray(src.onFilter) ? src.onFilter : [src.onFilter];
  572. for (i = 0; i < filters.length; i++) {
  573. data = filters[i](data, response);
  574. }
  575. }
  576.  
  577. var table = self.tableStriped.clone();
  578.  
  579. var renderFunction = 'onRender' in src ? src.onRender : renderResults;
  580. renderFunction(data, table, response);
  581.  
  582. var categoryTitle = response.context.categoryTitle;
  583. var searchUrl = ("searchUrl" in response.context) ? response.context.searchUrl : response.finalUrl;
  584. var resultLink = self.iconLink.clone()
  585. .attr("href", searchUrl)
  586. .addClass('icon-' + src.name);
  587.  
  588. if (table[0].childElementCount > 0) {
  589. var panel = self.resultPanel.clone()
  590. .addClass('panel-src-' + src.name);
  591.  
  592. resultLink.html(src.name + (pageId ? '' : ' / ' + categoryTitle + ' / <i>' + originalQuery + '</i>'));
  593.  
  594. if ("length" in data) {
  595. resultLink.append(' (' + data.length + ')');
  596. }
  597.  
  598. panel[0].firstChild.firstChild.appendChild(resultLink[0]);
  599. panel.prepend(self.closeBtn.clone());
  600. panel[0].lastChild.appendChild(table[0]);
  601.  
  602. (!response.context.pageId || response.context.retry)
  603. ? panel.prependTo(self.content)
  604. : panel.appendTo(self.content);
  605. } else {
  606. var noResults = $('#no-results');
  607. var alert = noResults.length === 0
  608. ? self.warningAlert.clone().insertBefore(self.content)
  609. : noResults.first();
  610.  
  611. resultLink
  612. .addClass("link-icon")
  613. .attr("title", src.name)
  614. .appendTo(alert);
  615. }
  616. };
  617.  
  618. var renderResults = function (data, table, response) {
  619. var src = response.context.src;
  620. var i = 0, len = 0;
  621.  
  622. data.each(function () {
  623. var n, nlen, context, that = $(this);
  624. var tr = self.tr.clone();
  625. var text = "", td, el, elen;
  626. var texts = [], link = "", linkTest = "", link_prepend = "";
  627. var freeleech = false, vod = false;
  628.  
  629. for (i = 0, len = src.onParse.sel.length; i < len; i++) {
  630. context = that;
  631. text = link = link_prepend = "";
  632. freeleech = vod = false;
  633. td = self.td.clone();
  634.  
  635. if ("class" in src.onParse.sel[i]) {
  636. td.addClass(src.onParse.sel[i].class);
  637. }
  638.  
  639. if ("width" in src.onParse.sel[i]) {
  640. td.width(src.onParse.sel[i].width);
  641. }
  642.  
  643. if ("align" in src.onParse.sel[i]) {
  644. td.css("textAlign", src.onParse.sel[i].align);
  645. }
  646.  
  647. if ("cleanup" in src.onParse.sel[i]) {
  648. context = context.clone();
  649. for (n = 0, nlen = src.onParse.sel[i].cleanup.length; n < nlen; n++) {
  650. $(src.onParse.sel[i].cleanup[n], context).remove();
  651. }
  652. }
  653.  
  654. if (typeof src.onParse.sel[i].text === "function") {
  655. el = src.onParse.sel[i].text(context);
  656. } else {
  657. el = $(src.onParse.sel[i].text, context);
  658. }
  659.  
  660. if (typeof el === "string") {
  661. text = el;
  662. } else if (el instanceof jQuery) {
  663. elen = el.length;
  664. if (elen === 0) {
  665. td.appendTo(tr);
  666. continue;
  667. } else if (elen === 1) {
  668. text = el.text();
  669. if (link === "" && el.prop("tagName") === "A") {
  670. link = el.attr("href");
  671. }
  672. } else {
  673. texts = [];
  674. for (n = 0; n < elen; n++) {
  675. texts.push(el.eq(n).text());
  676. }
  677. text = texts.join(" - ");
  678. }
  679. }
  680.  
  681. text = text.trim();
  682.  
  683. if (text === "") {
  684. td.appendTo(tr);
  685. continue;
  686. }
  687.  
  688. if ("freeleech" in src.onParse.sel[i]) {
  689. if (typeof src.onParse.sel[i].freeleech === "function") {
  690. freeleech = src.onParse.sel[i].freeleech(context);
  691. } else {
  692. freeleech = $(src.onParse.sel[i].freeleech, context).length > 0;
  693. }
  694. }
  695.  
  696. if ("vod" in src.onParse.sel[i]) {
  697. if (typeof src.onParse.sel[i].vod === "function") {
  698. vod = src.onParse.sel[i].vod(context);
  699. } else {
  700. vod = $(src.onParse.sel[i].vod, context).length > 0;
  701. }
  702. }
  703.  
  704. if ("link" in src.onParse.sel[i]) {
  705. if (typeof src.onParse.sel[i].link === "function") {
  706. link = src.onParse.sel[i].link(context);
  707. } else {
  708. link = $(src.onParse.sel[i].link, context);
  709. }
  710. }
  711.  
  712. if (link instanceof jQuery) {
  713. linkTest = link.attr("href");
  714. if (typeof linkTest === "string") {
  715. link = linkTest;
  716. } else if (link[0] && (link[0].textContent.indexOf("http") === 0)) {
  717. link = link[0].textContent;
  718. }
  719. }
  720.  
  721. if (typeof link === "string" && link !== "") {
  722. if (self.protocols.indexOf(link.split(":")[0]) === -1) {
  723. if ("link_prepend" in src.onParse.sel[i]) {
  724. link_prepend = src.onParse.sel[i].link_prepend;
  725. } else if ("link_prepend" in src.onParse) {
  726. link_prepend = src.onParse.link_prepend;
  727. }
  728. }
  729. if (link_prepend !== "") {
  730. link = link_prepend + link;
  731. }
  732. td.html('<a href="' + link + '"' + (("noblank" in src.onParse.sel[i] && src.onParse.sel[i].noblank) ? '' : ' target="_blank"') + '>' + text + '</a>');
  733. } else {
  734. td.html(text);
  735. }
  736.  
  737. if (freeleech) {
  738. td.append(' <span class="label label-success">Freeleech</span>');
  739. } else if (vod) {
  740. td.append(' <span class="label label-warning">VOD</span>');
  741. }
  742.  
  743. td.appendTo(tr);
  744. }
  745.  
  746. tr.appendTo(table);
  747. });
  748. };
  749.  
  750. var isOutOfDate = function (response) {
  751. return (typeof response !== "object" || !("context" in response) || !("pageId" in response.context) || self.pageId !== response.context.pageId);
  752. };
  753.  
  754. self.showFailAlert = function (response, msg) {
  755. if (isOutOfDate(response)) return;
  756.  
  757. var errors = $('.errors', self.layout);
  758. var alert = errors.length === 0
  759. ? self.failAlert.clone().insertBefore(self.content)
  760. : errors.first();
  761.  
  762. if (!("context" in response)) {
  763. if (!msg) msg = "(error)";
  764. return alert.append(msg + '&nbsp;&nbsp; ');
  765. }
  766.  
  767. var url = "finalUrl" in response ? response.finalUrl : null;
  768. var srcName = response.context.src.name;
  769. var categoryTitle = response.context.pageId ? null : response.context.categoryTitle;
  770.  
  771. var target = 'target="_blank"';
  772.  
  773. if (url === null) {
  774. if ("url" in response.context && response.context.url.length > 0) {
  775. url = response.context.url[0];
  776. } else {
  777. url = "javascript:alert('Domain not resolved.');";
  778. target = '';
  779. }
  780. msg = 'domain error';
  781. }
  782.  
  783. if (!msg) msg = "error";
  784.  
  785. return alert.append('<a href="' + url + '" ' + target + ' class="icon icon-' + srcName + '">' + srcName + (categoryTitle ? ' / ' + categoryTitle + ' / <i>' + self.input.val() + '</i>' : '') + '</a>&nbsp;(', msg, ')&nbsp;&nbsp; ');
  786. };
  787.  
  788. var onRequestFail = function (response) {
  789. console.log(response);
  790. $.publish('request-finish', [response]);
  791. self.showFailAlert(response, "http request failed");
  792. };
  793.  
  794. var retryTimedOutRequest = function () {
  795. var context = $(this).data("context");
  796. self.sendRequest(context.src, context.category, context.query, true);
  797. $(this).replaceWith('retried');
  798. };
  799.  
  800. var onRequestTimeout = function (response) {
  801. console.log(response);
  802. $.publish('request-finish', [response]);
  803. var retry;
  804. if ("context" in response) {
  805. retry = $('<a href="javascript:void(0)" title="click to retry" class="timeout-link">timeout</a>')
  806. .data("context", response.context)
  807. .on("click", retryTimedOutRequest);
  808. } else {
  809. retry = "timeout";
  810. }
  811.  
  812. self.showFailAlert(response, retry);
  813. };
  814.  
  815. var getPersistentData = function () {
  816. if (!self.persistentData || typeof self.persistentData !== "object") {
  817. self.persistentData = GM_getValue(self.persistentDataId, {});
  818. if (!self.persistentData || typeof self.persistentData !== "object") {
  819. console.log("Invalid persistent data, resetting.");
  820. self.persistentData = {};
  821. }
  822. }
  823.  
  824. return self.persistentData;
  825. };
  826.  
  827. self.getPersistentValue = function (key, def) {
  828. if (typeof key === "string") {
  829. var persistentData = getPersistentData();
  830.  
  831. if (!(key in persistentData)) {
  832. persistentData[key] = def;
  833. }
  834.  
  835. return persistentData[key];
  836. }
  837. };
  838.  
  839. self.setPersistentValue = function (key, value) {
  840. if (typeof key === "string") {
  841. var persistentData = getPersistentData();
  842.  
  843. persistentData[key] = value;
  844.  
  845. return value;
  846. }
  847. };
  848.  
  849. self.humanizeSize = function (size) {
  850. size = parseInt(size);
  851. var i = Math.floor(Math.log(size) / Math.log(1024));
  852. return (size / Math.pow(1024, i)).toFixed(2) * 1 + ' ' + ['B', 'KB', 'MB', 'GB', 'TB'][i];
  853. };
  854.  
  855. self.requireAllWords = function (data, response) {
  856. if (typeof response.context.src.onParse !== "object") {
  857. return data;
  858. }
  859.  
  860. var words = response.context.query.toLowerCase().split(' ');
  861. var wordsLen = words.length;
  862. var sel = response.context.src.onParse.sel[0].text;
  863. var isFunction = typeof sel === "function";
  864.  
  865. return data.filter(function () {
  866. var text;
  867. if (isFunction) {
  868. text = sel($(this));
  869. if (text instanceof jQuery) {
  870. text = text.text();
  871. } else if (text instanceof HTMLElement) {
  872. text = text.textContent;
  873. } else if (typeof text !== "string") {
  874. return true;
  875. }
  876. text = text.toLowerCase();
  877. } else {
  878. text = $(sel, this)[0].textContent.toLowerCase();
  879. }
  880.  
  881. for (var i = 0; i < wordsLen; i++) {
  882. if (text.indexOf(words[i]) === -1) {
  883. return false;
  884. }
  885. }
  886.  
  887. return true;
  888. });
  889. };
  890.  
  891. self.filter3dMovies = function (data, response) {
  892. if (response.context.category.indexOf("movies") === 0 && response.context.category !== "movies_dvd" && response.context.category !== "movies_3d") {
  893. return data.filter(function(){
  894. return this.textContent.indexOf('3D') === -1;
  895. });
  896. } else {
  897. return data;
  898. }
  899. };
  900.  
  901. self.extractResolution = function (context) {
  902. for (var i = 0; i < self.resolutions.length; i++) {
  903. if (context.query.indexOf(self.resolutions[i]) !== -1) {
  904. context.query = context.query.replace(self.resolutions[i], "").replace(self.spacesRegex, ' ').trim();
  905. return self.resolutions[i];
  906. }
  907. }
  908.  
  909. return null;
  910. };
  911.  
  912. self.extractGazelleResolution = function (context) {
  913. var res = self.extractResolution(context);
  914.  
  915. if (res) {
  916. for (var i = 0; i < context.url.length; i++) {
  917. context.url[i] += "&resolution=" + res;
  918. }
  919. }
  920. };
  921.  
  922. self.extractYear = function (context, simple) {
  923. var re = simple ? self.yearRegexSimple : self.yearRegexAdvanced;
  924. var year = context.query.match(re);
  925.  
  926. if (year && year.length > 0) {
  927. year = year[year.length - 1].trim();
  928. context.query = context.query.replace(new RegExp(year.replace(self.dashRegex, '\\\-'), "g"), '').replace(self.spacesRegex, ' ').trim();
  929. } else {
  930. year = null;
  931. }
  932.  
  933. return year;
  934. };
  935.  
  936. self.replaceImages = function (text) {
  937. return text.replace(self.imgTagRegex, '<meta ');
  938. };
  939.  
  940. self.extractGazelleYear = function (context) {
  941. var year = self.extractYear(context);
  942.  
  943. if (year) {
  944. for (var i = 0, len = context.url.length; i < len; i++) {
  945. context.url[i] += "&year=" + year;
  946. }
  947. }
  948. };
  949.  
  950.  
  951.  
  952.  
  953. ////// SETTINGS
  954.  
  955. $.subscribe("layout-ready", function(){
  956. $("#app-buttons", self.layout)
  957. .append($('<button type="button" class="btn btn-default btn-xs"><span class="glyphicon glyphicon-cog"></span></button>').on("click", renderConfigPage))
  958. .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>');
  959. });
  960.  
  961. var renderConfigPage = function () {
  962. resetContent();
  963.  
  964. if (self.pageId === "Settings") {
  965. self.pageId = false;
  966. return;
  967. }
  968.  
  969. self.pageId = "Settings";
  970. self.header.text(self.pageId).show();
  971.  
  972. var sourceKeys = Object.keys(self.sourceCallbacks).sort(function (a, b) {
  973. return a.toLowerCase().localeCompare(b.toLowerCase());
  974. }),
  975. srcButtons = self.div.clone().attr("id", "source-buttons").appendTo(self.content),
  976. 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>'),
  977. disabledCategories = getDisabledCategories(),
  978. btn, buttonClass,
  979. enabled = false,
  980. partially = false;
  981.  
  982. for (var i = 0, len = sourceKeys.length; i < len; i++) {
  983. enabled = sourceIsEnabled(sourceKeys[i]);
  984. partially = (sourceKeys[i] in disabledCategories);
  985.  
  986. btn = btnStub.clone();
  987. btn[0].firstChild.textContent = btn[0].dataset.src = sourceKeys[i];
  988. buttonClass = enabled ? (partially ? "btn-warning" : "btn-success") : "btn-danger";
  989. btn.children('button').addClass(buttonClass);
  990.  
  991. srcButtons.append(btn);
  992.  
  993. if (enabled) {
  994. showSourceConfigBox(sourceKeys[i]);
  995. }
  996. }
  997.  
  998. $(".btn-group > button:first-child", srcButtons).on("click", function () {
  999. var name = this.parentNode.dataset.src;
  1000. sourceIsEnabled(name) ? self.disableSource(name) : self.enableSource(name);
  1001. });
  1002.  
  1003. srcButtons.on("change", "input", function () {
  1004. var group = $(this).closest(".btn-group");
  1005. var srcName = group[0].dataset.src;
  1006. var checkboxes = $('input[type="checkbox"]', group);
  1007. var total = checkboxes.length;
  1008. var unchecked = checkboxes.filter(":not(:checked)").length;
  1009. var disabled = getDisabledCategories(srcName);
  1010.  
  1011. if (this.checked) {
  1012. if (disabled.indexOf(this.name) !== -1) {
  1013. disabledCategories[srcName].splice(disabledCategories[srcName].indexOf(this.name), 1);
  1014. if (disabledCategories[srcName].length === 0) {
  1015. delete disabledCategories[srcName];
  1016. }
  1017. }
  1018. self.enableSource(srcName);
  1019. } else {
  1020. if (disabled.indexOf(this.name) === -1) {
  1021. if (!(srcName in disabledCategories)) {
  1022. disabledCategories[srcName] = [];
  1023. }
  1024. disabledCategories[srcName].push(this.name);
  1025. }
  1026. total === unchecked
  1027. ? self.disableSource(srcName)
  1028. : self.enableSource(srcName);
  1029. }
  1030. });
  1031.  
  1032. $(".btn-group", srcButtons).on("show.bs.dropdown", function () {
  1033. if (this.lastChild.textContent === "") {
  1034. var data = self.sourceCallbacks[this.textContent]();
  1035. var cats = Object.keys(data.url);
  1036. cats.unshift(self.favorites);
  1037.  
  1038. var disabled = this.textContent in disabledCategories ? disabledCategories[this.textContent] : [];
  1039. var catsHTML = '';
  1040. for (var i = 0, len = cats.length; i < len; i++) {
  1041. 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>';
  1042. }
  1043. //if ("website" in data && data.website) {
  1044. // catsHTML += '<a href="' + data.website + '" target="_blank">Visit website</a>';
  1045. //}
  1046. this.lastChild.innerHTML = catsHTML;
  1047. }
  1048. });
  1049.  
  1050. $(".dropdown-menu", srcButtons).on("click", function (e) {
  1051. e.stopPropagation();
  1052. });
  1053. };
  1054.  
  1055. var showSourceConfigBox = function (name, buttons) {
  1056. if (!("config" in self.sources[name])) return;
  1057.  
  1058. var box = $("#config-" + name);
  1059.  
  1060. if (box.length === 0) {
  1061. box = $('<div id="config-' + name + '" class="panel panel-default"><div class="panel-heading">' + name + '</div><div class="panel-body"></div></div>')
  1062. .insertAfter($("#source-buttons", self.content));
  1063.  
  1064. $(box[0].lastChild)
  1065. .empty()
  1066. .append(self.sources[name].config());
  1067. }
  1068. };
  1069.  
  1070. $.subscribe("source-disabled", function (e, id) {
  1071. if (self.pageId === "Settings") {
  1072. $('#source-buttons', self.content).children('[data-src="' + id + '"]').children("button").addClass("btn-danger").removeClass("btn-success btn-warning");
  1073. $("#config-" + id, self.content).remove();
  1074. }
  1075. });
  1076.  
  1077. $.subscribe("source-enabled", function (e, id, data) {
  1078. if (self.pageId === "Settings") {
  1079. var className = getDisabledCategories(id).length > 0 ? "btn-warning" : "btn-success";
  1080. $('#source-buttons', self.content).children('[data-src="' + id + '"]').children("button").removeClass("btn-danger btn-warning btn-success").addClass(className);
  1081. showSourceConfigBox(id);
  1082. }
  1083. });
  1084.  
  1085. /////// PERSISTENT CONTENT
  1086.  
  1087. var loadPersistentContent = function () {
  1088. if (self.getPersistentValue("content", "")) {
  1089.  
  1090. self.mainColumn.html(self.getPersistentValue("content", ""));
  1091.  
  1092. $(".timeout-link", self.mainColumn).replaceWith("timeout");
  1093.  
  1094. self.header = $('h2', self.mainColumn);
  1095. if (self.header.length === 0) {
  1096. self.header = self.h2.clone().hide().appendTo(self.mainColumn)
  1097. } else {
  1098. self.header = self.header.first();
  1099. }
  1100.  
  1101. self.content = $('#content', self.mainColumn);
  1102. if (self.content.length === 0) {
  1103. self.content = self.div.clone().attr("id", "content").appendTo(self.mainColumn);
  1104. } else {
  1105. self.content = self.content.first();
  1106. }
  1107.  
  1108. if (self.header.css("display") !== "none") {
  1109. self.pageId = true;
  1110. }
  1111. if (self.header.text() === "Settings") {
  1112. renderConfigPage();
  1113. }
  1114. }
  1115. };
  1116.  
  1117. $.subscribe('layout-ready', loadPersistentContent);
  1118.  
  1119. $.subscribe('persistent-save', function (e, data) {
  1120. data.content = self.mainColumn.html();
  1121. });
  1122.  
  1123.  
  1124. //////// PROGRESS BAR
  1125.  
  1126. var progressBar = null;
  1127. var progressFiller = null;
  1128. var progressSteps = 0;
  1129. var progressStyleSuccess = "progress-bar-success";
  1130. var progressStyleInProcess = "progress-bar-warning";
  1131.  
  1132. var initProgressBar = function () {
  1133. progressBar = $(".progress", self.mainColumn);
  1134. if (progressBar.length === 0) {
  1135. progressBar = $('<div class="progress pull-right"><div class="progress-bar"></div></div>').prependTo(self.mainColumn);
  1136. } else {
  1137. progressBar = progressBar.first();
  1138. }
  1139. progressFiller = $(progressBar[0].firstChild);
  1140. resetProgressBar(true);
  1141. };
  1142.  
  1143. var resetProgressBar = function (hide) {
  1144. if (hide) {
  1145. progressBar.hide();
  1146. }
  1147.  
  1148. progressFiller
  1149. .width(0)
  1150. .data("step", 0)
  1151. .removeClass(progressStyleSuccess)
  1152. .addClass(progressStyleInProcess)
  1153. .text("0%");
  1154. progressSteps = 0;
  1155. };
  1156.  
  1157. var startProgressCount = function (event, category, buttons) {
  1158. resetProgressBar();
  1159.  
  1160. progressSteps = 0;
  1161. buttons.each(function () {
  1162. var src = $(this).data('src');
  1163. if (category in src.url) {
  1164. progressSteps += typeof src.url[category] === "string" ? 1 : src.url[category].length;
  1165. }
  1166. });
  1167.  
  1168. progressBar.show();
  1169. };
  1170.  
  1171. var incrementProgressBar = function (event, response) {
  1172. if ("context" in response && isOutOfDate(response)) return;
  1173.  
  1174. var step = progressFiller.data("step");
  1175. step++;
  1176.  
  1177. if (step === progressSteps) {
  1178. progressFiller
  1179. .width('100%')
  1180. .removeClass(progressStyleInProcess)
  1181. .addClass(progressStyleSuccess)
  1182. .text(step + '/' + step);
  1183. // setTimeout(resetProgressBar, 3000);
  1184. } else {
  1185. var width = (100 / progressSteps * step).toFixed(2);
  1186. if (width > 100) width = 100;
  1187. progressFiller
  1188. .text(step + '/' + progressSteps)
  1189. .width(width + '%')
  1190. .data("step", step);
  1191. }
  1192. };
  1193.  
  1194.  
  1195. self.customCSS.append("\
  1196. .progress { width:120px; margin-top: 7px; margin-bottom:0; }\
  1197. .progress-bar { width:0px; transition: none; }\
  1198. ");
  1199.  
  1200. $.subscribe('layout-ready after-content-reset', initProgressBar);
  1201. $.subscribe('batch-request', startProgressCount);
  1202. $.subscribe('request-finish', incrementProgressBar);
  1203.  
  1204.  
  1205. ///////// HASH NAVIGATION
  1206.  
  1207. var parseQueryString = function (queryString) {
  1208. var params = {}, queries, temp, i, l;
  1209.  
  1210. // Split into key/value pairs
  1211. queries = queryString.split("&");
  1212.  
  1213. // Convert the array of strings into an object
  1214. for (i = 0, l = queries.length; i < l; i++) {
  1215. temp = queries[i].split('=');
  1216. params[temp[0]] = temp[1];
  1217. }
  1218.  
  1219. return params;
  1220. };
  1221.  
  1222. $.subscribe("layout-ready", function(){
  1223. if (document.location.hash.trim().length > 1 && document.location.hash !== self.getPersistentValue("hash", "#")) {
  1224. resetContent();
  1225. }
  1226. });
  1227.  
  1228. $.subscribe("page-rendered", function(){
  1229. var paramsStr = null;
  1230.  
  1231. if (document.location.hash.trim().length > 1 && document.location.hash !== self.getPersistentValue("hash", "#")) {
  1232. paramsStr = document.location.hash.slice(1);
  1233. } else if (document.location.href.indexOf('?') > -1) {
  1234. paramsStr = document.location.href.split('?', 2)[1];
  1235. }
  1236.  
  1237. if (!paramsStr) return;
  1238.  
  1239. var params = parseQueryString(paramsStr);
  1240.  
  1241. if ("q" in params && params.q.trim() !== "") {
  1242. params.q = decodeURIComponent(params.q).replace(/\+/g, ' ');
  1243.  
  1244. self.input.val(params.q);
  1245.  
  1246. if ("cat" in params) {
  1247. if ("src" in params) {
  1248. $("#" + params.src + "__" + params.cat).click();
  1249. } else {
  1250. $("#category-" + params.cat).children('button').first().click();
  1251. }
  1252. } else if ("src" in params) {
  1253. $("#" + params.src + "__" + self.all).click();
  1254. }
  1255. }
  1256.  
  1257. self.setPersistentValue("hash", '#' + paramsStr);
  1258. });
  1259.  
  1260. $.subscribe("batch-request", function (e, category) {
  1261. document.location.hash = "#cat=" + (category === self.favorites ? self.all : category) + "&q=" + self.input.val();
  1262. self.setPersistentValue("hash", document.location.hash);
  1263. });
  1264.  
  1265. $.subscribe("single-request", function(e, btn, category){
  1266. document.location.hash = "#cat=" + (category === self.favorites ? self.all : category) + "&src=" + btn.data("src").name + "&q=" + self.input.val();
  1267. self.setPersistentValue("hash", document.location.hash);
  1268. });
  1269. };
  1270.  
  1271. var bt = new SearchEngine();
  1272.  
  1273. /*
  1274. onHttpRequest(requestData): {}.abort()
  1275. onValidate(response): bool|string
  1276. onParse(response)|object: collection|array
  1277. onFilter(data, response): collection|array
  1278. onRender(data, table): void
  1279. */
  1280.  
  1281. bt.categories = {
  1282. all: "Everywhere",
  1283. favorites: "Favorites",
  1284. music: "Music",
  1285. music_flac: "Music / FLAC",
  1286. movies: "Movies",
  1287. // movies_hd: "Movies: HD",
  1288. movies_1080: "Movies / 1080p",
  1289. movies_720: "Movies / 720p",
  1290. movies_remux: "Movies / Remuxes",
  1291. movies_bluray: "Movies / Blu-rays",
  1292. movies_dvd: "Movies / DVD",
  1293. mvids: "Music Videos",
  1294. docs: "Documentaries",
  1295. tv: "TV Shows",
  1296. elearning: "E-Learning",
  1297. ebooks: "E-Books",
  1298. games_pc: "Games / PC",
  1299. mags: "Magazines",
  1300. abooks: "Audiobooks",
  1301. fiction: "Fiction",
  1302. comics: "Comics",
  1303. apps_win: "Apps / Win",
  1304. xxx: "XXX"
  1305. };
  1306. bt.addSource("Adamsfile", function () {
  1307. var adam = "http://adamsfile.com/search.php?s_string={query}&s_year=&albumsSearchSubmit=&s_genre=&s_mediatype=&s_type=";
  1308.  
  1309. return {
  1310. url: {
  1311. all: adam,
  1312. music: adam,
  1313. music_flac: adam
  1314. },
  1315. onParse: {
  1316. cleanup: [".alb_data", ".urating"],
  1317. row: ".albums > tbody > tr",
  1318. link_prepend: "http://adamsfile.com/",
  1319. sel: [
  1320. {text: "> td:eq(0)", link: "a[href*='details.php']:eq(0)"},
  1321. {text: "> td:eq(1)", link: "a[href^='ftp://dl.adamsfile.com/']:eq(0)", noblank: true}
  1322. ]
  1323. },
  1324. onFilter: bt.requireAllWords,
  1325. onValidate: function (response) {
  1326. return response.responseText.indexOf('auth_window') === -1 ? true : "login needed";
  1327. },
  1328. icon: ""
  1329. };
  1330. });
  1331. bt.addSource("OPS", function () {
  1332. var wcd = "https://orpheus.network/ajax.php?action=browse&searchstr={query}";
  1333.  
  1334. return {
  1335. website: "https://orpheus.network/",
  1336. url: {
  1337. all: wcd,
  1338. music: wcd + "&filter_cat[1]=1",
  1339. music_flac: wcd + "&filter_cat[1]=1&format=FLAC",
  1340. // music_mp3: wcd + "&filter_cat[1]=1&format=AAC|MP3",
  1341. elearning: wcd + "&filter_cat[3]=1&filter_cat[4]=1&filter_cat[5]=1&filter_cat[7]=1",
  1342. mags: wcd + "&filter_cat[3]=1",
  1343. ebooks: wcd + "&filter_cat[3]=1",
  1344. fiction: wcd + "&filter_cat[3]=1",
  1345. abooks: wcd + "&filter_cat[4]=1",
  1346. comics: wcd + "&filter_cat[7]=1",
  1347. apps_win: wcd + "&filter_cat[2]=1"
  1348. },
  1349. onPrepareQuery: bt.extractGazelleYear,
  1350. onParse: function (response) {
  1351. try {
  1352. var data = JSON.parse(response.responseText);
  1353. } catch (e) {
  1354. bt.showFailAlert(response);
  1355. return null;
  1356. }
  1357.  
  1358. if (!('status' in data) || data.status !== 'success' || !('response' in data) || !('results' in data.response)) {
  1359. bt.showFailAlert(response, "unexpected data");
  1360. return null;
  1361. }
  1362.  
  1363. return data.response.results;
  1364. },
  1365. onFilter: function (data, response) {
  1366. var words = response.context.query.toLowerCase().split(' ');
  1367.  
  1368. response.context.searchUrl = response.finalUrl.replace('ajax.php?action=browse', 'torrents.php?action=basic');
  1369.  
  1370. return data.filter(function (value) {
  1371. for (var i = 0, l = words.length; i < l; i++) {
  1372. if ((('artist' in value && value.artist.toLowerCase().indexOf(words[i]) === -1) || !('artist' in value)) && value.groupName.toLowerCase().indexOf(words[i]) === -1) {
  1373. return false;
  1374. }
  1375. }
  1376.  
  1377. return true;
  1378. });
  1379. },
  1380. onRender: function (data, table) {
  1381. var nl, group, torrent, groupTable, groupTableHTML, torRow, tr, n, score, cue, ed = "", media = "", link, artist;
  1382.  
  1383. var torrentGroupHeader = bt.trtd.clone();
  1384. $(torrentGroupHeader[0].firstChild).append(
  1385. bt.h4.clone().addClass("torrent-group")
  1386. );
  1387.  
  1388. for (var i = 0, l = data.length; i < l; i++) {
  1389. ed = ""; media = "";
  1390. group = data[i];
  1391. link = "https://orpheus.network/torrents.php?id=" + group.groupId;
  1392.  
  1393. if ("torrents" in group) {
  1394. artist = (group.artist === 'Various Artists')
  1395. ? 'Various Artists'
  1396. : bt.ab.clone().attr("href", "https://orpheus.network/artist.php?id=" + group.torrents[0].artists[0].id).html(group.artist);
  1397.  
  1398. tr = torrentGroupHeader.clone();
  1399. tr[0].firstChild.setAttribute("colspan", "3");
  1400. $(tr[0].firstChild.firstChild).append(
  1401. artist,
  1402. " - ",
  1403. bt.ab.clone().attr("href", link).html(group.groupName),
  1404. ' [' + group.groupYear + '] [' + group.releaseType + ']'
  1405. );
  1406.  
  1407. if (group.cover) {
  1408. $(tr[0].firstChild)
  1409. .addClass('cover')
  1410. .css('backgroundImage', 'url(' + group.cover + ')');
  1411. }
  1412.  
  1413. groupTableHTML = "";
  1414. for (n = 0, nl = group.torrents.length; n < nl; n++) {
  1415. torrent = group.torrents[n];
  1416.  
  1417. if (ed !== torrent.remasterTitle || media !== torrent.media) {
  1418. ed = torrent.remasterTitle;
  1419. media = torrent.media;
  1420. groupTableHTML += '<tr><td colspan="3" class="tr-title">' + (ed ? ed : "Original Release") + ' ' + torrent.remasterCatalogueNumber + ' ' + (torrent.remasterYear ? torrent.remasterYear : "") + ' / ' + media + "</td></tr>";
  1421. }
  1422.  
  1423. score = torrent.hasLog ? ' / Log (' + torrent.logScore + '%)' : '';
  1424. cue = torrent.hasCue ? ' / Cue' : '';
  1425.  
  1426. 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>';
  1427. }
  1428.  
  1429. groupTable = bt.table.clone()
  1430. .addClass("torrent-table")
  1431. .html(groupTableHTML);
  1432.  
  1433. torRow = bt.tr.clone().append(bt.td.clone().attr("colspan", "3"));
  1434. torRow[0].firstChild.appendChild(groupTable[0]);
  1435.  
  1436. table.append(tr, torRow);
  1437. } else {
  1438. 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>');
  1439. }
  1440. }
  1441. },
  1442. icon: ""
  1443. };
  1444. });
  1445. bt.addSource("RED", function () {
  1446. var wcd = "https://redacted.ch/ajax.php?action=browse&searchstr={query}";
  1447.  
  1448. return {
  1449. website: "https://redacted.ch/",
  1450. url: {
  1451. all: wcd,
  1452. music: wcd + "&filter_cat[1]=1",
  1453. music_flac: wcd + "&filter_cat[1]=1&format=FLAC",
  1454. // music_mp3: wcd + "&filter_cat[1]=1&format=AAC|MP3",
  1455. elearning: wcd + "&filter_cat[3]=1&filter_cat[4]=1&filter_cat[5]=1&filter_cat[7]=1",
  1456. mags: wcd + "&filter_cat[3]=1",
  1457. ebooks: wcd + "&filter_cat[3]=1",
  1458. fiction: wcd + "&filter_cat[3]=1",
  1459. abooks: wcd + "&filter_cat[4]=1",
  1460. comics: wcd + "&filter_cat[7]=1",
  1461. apps_win: wcd + "&filter_cat[2]=1"
  1462. },
  1463. onPrepareQuery: bt.extractGazelleYear,
  1464. onParse: function (response) {
  1465. try {
  1466. var data = JSON.parse(response.responseText);
  1467. } catch (e) {
  1468. bt.showFailAlert(response);
  1469. return null;
  1470. }
  1471.  
  1472. if (!('status' in data) || data.status !== 'success' || !('response' in data) || !('results' in data.response)) {
  1473. bt.showFailAlert(response, "unexpected data");
  1474. return null;
  1475. }
  1476.  
  1477. return data.response.results;
  1478. },
  1479. onFilter: function (data, response) {
  1480. var words = response.context.query.toLowerCase().split(' ');
  1481.  
  1482. response.context.searchUrl = response.finalUrl.replace('ajax.php?action=browse', 'torrents.php?action=basic');
  1483.  
  1484. return data.filter(function (value) {
  1485. for (var i = 0, l = words.length; i < l; i++) {
  1486. if ((('artist' in value && value.artist.toLowerCase().indexOf(words[i]) === -1) || !('artist' in value)) && value.groupName.toLowerCase().indexOf(words[i]) === -1) {
  1487. return false;
  1488. }
  1489. }
  1490.  
  1491. return true;
  1492. });
  1493. },
  1494. onRender: function (data, table) {
  1495. var nl, group, torrent, groupTable, groupTableHTML, torRow, tr, n, score, cue, ed = "", media = "", link, artist;
  1496.  
  1497. var torrentGroupHeader = bt.trtd.clone();
  1498. $(torrentGroupHeader[0].firstChild).append(
  1499. bt.h4.clone().addClass("torrent-group")
  1500. );
  1501.  
  1502. for (var i = 0, l = data.length; i < l; i++) {
  1503. ed = ""; media = "";
  1504. group = data[i];
  1505. link = "https://redacted.ch/torrents.php?id=" + group.groupId;
  1506.  
  1507. if ("torrents" in group) {
  1508. artist = (group.artist === 'Various Artists')
  1509. ? 'Various Artists'
  1510. : bt.ab.clone().attr("href", "https://redacted.ch/artist.php?id=" + group.torrents[0].artists[0].id).html(group.artist);
  1511.  
  1512. tr = torrentGroupHeader.clone();
  1513. tr[0].firstChild.setAttribute("colspan", "3");
  1514. $(tr[0].firstChild.firstChild).append(
  1515. artist,
  1516. " - ",
  1517. bt.ab.clone().attr("href", link).html(group.groupName),
  1518. ' [' + group.groupYear + '] [' + group.releaseType + ']'
  1519. );
  1520.  
  1521. if (group.cover) {
  1522. $(tr[0].firstChild)
  1523. .addClass('cover')
  1524. .css('backgroundImage', 'url(' + group.cover + ')');
  1525. }
  1526.  
  1527. groupTableHTML = "";
  1528. for (n = 0, nl = group.torrents.length; n < nl; n++) {
  1529. torrent = group.torrents[n];
  1530.  
  1531. if (ed !== torrent.remasterTitle || media !== torrent.media) {
  1532. ed = torrent.remasterTitle;
  1533. media = torrent.media;
  1534. groupTableHTML += '<tr><td colspan="3" class="tr-title">' + (ed ? ed : "Original Release") + ' ' + torrent.remasterCatalogueNumber + ' ' + (torrent.remasterYear ? torrent.remasterYear : "") + ' / ' + media + "</td></tr>";
  1535. }
  1536.  
  1537. score = torrent.hasLog ? ' / Log (' + torrent.logScore + '%)' : '';
  1538. cue = torrent.hasCue ? ' / Cue' : '';
  1539.  
  1540. 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>';
  1541. }
  1542.  
  1543. groupTable = bt.table.clone()
  1544. .addClass("torrent-table")
  1545. .html(groupTableHTML);
  1546.  
  1547. torRow = bt.tr.clone().append(bt.td.clone().attr("colspan", "3"));
  1548. torRow[0].firstChild.appendChild(groupTable[0]);
  1549.  
  1550. table.append(tr, torRow);
  1551. } else {
  1552. 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>');
  1553. }
  1554. }
  1555. },
  1556. icon: ""
  1557. };
  1558. });
  1559. bt.addSource("Shellife", function () {
  1560. return {
  1561. url: {
  1562. music: "http://shellife.eu/browse.php?search=",
  1563. music_flac: "http://shellife.eu/browse.php?search="
  1564. },
  1565. onParse: {
  1566. cleanup: ["div[id]", ".grey"],
  1567. row: "tr.torrent_row",
  1568. link_prepend: "http://shellife.eu/",
  1569. sel: [
  1570. {
  1571. text: "> td:eq(1) > a:lt(2)",
  1572. link: function (context) {
  1573. var id = $("a[href*='download.php']:eq(0)", context).attr("href").split("id=")[1];
  1574. return "details.php?id=" + id;
  1575. }
  1576. },
  1577. {text: "> td:eq(6)"},
  1578. {text: "> td:eq(4)", link: "a[href*='download.php']:eq(0)", noblank: true}
  1579. ]
  1580. },
  1581. onFilter: bt.requireAllWords,
  1582. icon: ""
  1583. };
  1584. });
  1585. bt.addSource("WFL", function () {
  1586. var wfl = "https://www.waffles.ch/browse.php?q={query}";
  1587.  
  1588. return {
  1589. url: {
  1590. all: wfl,
  1591. music: wfl + ' (FLAC OR MP3 OR AAC)',
  1592. music_flac: wfl + ' FLAC',
  1593. fiction: wfl + "&c86=1",
  1594. mags: wfl + "&c87=1",
  1595. ebooks: wfl + "&c86=1&c87=1",
  1596. abooks: wfl + "&c89=1&c90=1",
  1597. comics: wfl + "&c88=1",
  1598. apps_win: wfl + "&c83=1",
  1599. elearning: wfl + "&c89=1&c90=1&c88=1&c86=1&c87=1&c93=1"
  1600. },
  1601. onParse: {
  1602. row: "#browsetable > tbody > tr:not(:first-child)",
  1603. link_prepend: "https://www.waffles.ch",
  1604. sel: [
  1605. {text: "> td:eq(1) a[href*='/details.php']:eq(0)"},
  1606. {text: "> td:eq(7)"},
  1607. {text: "> td:eq(5)", link: "a[href*='/download.php']:eq(0)", noblank: true}
  1608. ]
  1609. },
  1610. icon: ""
  1611. };
  1612. });
  1613. bt.addSource("XBT", function () {
  1614. var xbt = "https://xbtmusic.org/torrents-search.php?s=i&b=d&cat=0&active=0&page=0&own=&mode=2&search_pedrovia={query}";
  1615.  
  1616. return {
  1617. url: {
  1618. all: xbt,
  1619. music: xbt,
  1620. music_flac: xbt,
  1621. mvids: [xbt + "&cat=29", xbt + "&cat=201"],
  1622. docs: xbt + "&cat=134",
  1623. abooks: xbt + "&cat=54"
  1624. },
  1625. onParse: {
  1626. cleanup: ["td.small a"],
  1627. row: "table.submain > tbody > tr",
  1628. link_prepend: "https://xbtmusic.org",
  1629. sel: [
  1630. {text: "a[href*='/torrents-details.php?id=']:eq(0)"},
  1631. {text: "> td:eq(2) > table:eq(0) > tbody > tr > td:eq(1) > span:eq(0)"},
  1632. {
  1633. text: function (context) {
  1634. return $("> td:eq(2) > table:eq(0) > tbody > tr > td:eq(0)", context).text().split('in')[0];
  1635. },
  1636. link: function (context) {
  1637. var id = $("a[href*='/torrents-details.php?id=']:eq(0)", context).attr('href').split('id=')[1].split(bt.matchFirstNonDigit)[0];
  1638. return '/download.php?id=' + id;
  1639. },
  1640. noblank: true
  1641. }
  1642. ]
  1643. },
  1644. icon: ""
  1645. };
  1646. });
  1647. bt.addSource("AHD", function () {
  1648. var domain = 'https://awesome-hd.me/';
  1649. var ahd = domain + "torrents.php?action=advanced&order_by=time&order_way=desc&groupname={query}";
  1650.  
  1651. return {
  1652. url: {
  1653. all: ahd,
  1654. movies: ahd,
  1655. movies_1080: ahd + "&filter_cat[1]=1&resolution=1080p",
  1656. movies_720: ahd + "&filter_cat[1]=1&resolution=720p",
  1657. movies_remux: ahd + "&filter_cat[1]=1&media=Blu-ray",
  1658. tv: ahd + "&filter_cat[2]=1",
  1659. docs: ahd
  1660. },
  1661. onPrepareQuery: [
  1662. bt.extractGazelleYear,
  1663. bt.extractGazelleResolution,
  1664. ],
  1665. onParse: {
  1666. row: "tr:has(a[href*='action=download'])",
  1667. link_prepend: domain,
  1668. sel: [
  1669. {
  1670. text: function (context) {
  1671. if (context.hasClass('torrent')) {
  1672. var td = context.find('td:eq(2)').children().first().nextUntil('.tags');
  1673. return td.append(' ').text().trim();
  1674. } else {
  1675. return '<b>' + context.prevAll(".group").first().find("a[href*='id=']").first().text() + '</b> / ' + $("a[href*='torrentid=']:eq(0)", context).text();
  1676. }
  1677. },
  1678. link: function (context) {
  1679. if (context.hasClass('torrent')) {
  1680. return context.find("a[title='View Torrent']");
  1681. } else {
  1682. return context.prevAll(".group").first().find("a[href*='id=']").first();
  1683. }
  1684. },
  1685. },
  1686. {
  1687. text: "> td:eq(-2)",
  1688. },
  1689. {
  1690. text: "> td:eq(-4):contains('.')",
  1691. link: "a[href*='action=download']:eq(0)",
  1692. noblank: true
  1693. }
  1694. ]
  1695. },
  1696. icon: ""
  1697. };
  1698. });
  1699. bt.addSource("BHD", function () {
  1700. // NB: Spaces should be replaced with % signs, otherwise search works for an exact phrase
  1701.  
  1702. var bhd = "https://beyond-hd.me/browse.php?incldead=1&searchin=title&search={query}";
  1703.  
  1704. return {
  1705. url: {
  1706. all: bhd,
  1707. movies: bhd + "&c50=1&c77=1&c75=1&c49=1&c94=1&c61=1&c78=1&c86=1&c37=1&c54=1&c17=1",
  1708. movies_1080: bhd + "&c50=1&c77=1&c86=1&c94=1",
  1709. movies_720: bhd + "&c75=1&c78=1&c54=1",
  1710. 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",
  1711. movies_bluray: bhd + "&c37=1",
  1712. 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",
  1713. tv: bhd + "&c40=1&c44=1&c48=1&c89=1&c46=1&c45=1",
  1714. mvids: bhd + "&c55=1&c56=1&c42=1",
  1715. music: bhd + "&c36=1&c69=1",
  1716. music_flac: bhd + "&c36=1"
  1717. },
  1718. onParse: {
  1719. row: "table.torrenttable > tbody > tr:gt(0)",
  1720. link_prepend: "https://beyond-hd.me/",
  1721. sel: [
  1722. {text: "a[href*='details.php']:eq(0)"},
  1723. {text: "> td:eq(9)"},
  1724. {text: "> td:eq(7)", link: "a[href*='download.php']:eq(0)", noblank: true}
  1725. ]
  1726. },
  1727. onPrepareQuery: function(context){
  1728. context.query = context.query.replace(bt.spacesRegex, "%25");
  1729. },
  1730. onFilter: bt.filter3dMovies,
  1731. icon: ""
  1732. };
  1733. });
  1734. bt.addSource("BTN", function () {
  1735. return {
  1736. url: {
  1737. all: "https://broadcasthe.net/torrents.php?searchstr=",
  1738. tv: "https://broadcasthe.net/torrents.php?searchstr=",
  1739. docs: "https://broadcasthe.net/torrents.php?searchstr="
  1740. },
  1741. onHttpRequest: function (requestData) {
  1742. var btnKey = bt.getPersistentValue("BTN_KEY", "").trim();
  1743.  
  1744. var data = {
  1745. method: "getTorrents",
  1746. params: [btnKey, {"Resolution": ["720p", "1080p", "1080i"]}, 50],
  1747. id: Date.now()
  1748. };
  1749.  
  1750. if (requestData.context.query) {
  1751. if (requestData.context.category === "docs") {
  1752. bt.extractYear(requestData.context);
  1753. bt.extractResolution(requestData.context);
  1754. data.params[1] = requestData.context.query;
  1755. } else {
  1756. bt.extractYear(requestData.context);
  1757.  
  1758. var res = bt.extractResolution(requestData.context);
  1759. if (res) {
  1760. data.params[1].Resolution = res;
  1761. }
  1762.  
  1763. data.params[1].Series = '%' + requestData.context.query.trim().replace(bt.spacesRegex, '%') + '%';
  1764. }
  1765. }
  1766.  
  1767. $.extend(requestData, {
  1768. method: "POST",
  1769. url: "https://api.broadcasthe.net/",
  1770. headers: {'Content-Type': 'application/json'},
  1771. data: JSON.stringify(data)
  1772. });
  1773.  
  1774. console.log("BTN request data", JSON.stringify(requestData));
  1775. },
  1776. onParse: function (response) {
  1777. console.log("BTN response data");
  1778. console.log(response.statusText, JSON.stringify(response.responseHeaders));
  1779. console.log(response.responseText);
  1780.  
  1781. try {
  1782. var data = JSON.parse(response.responseText);
  1783. } catch (e) {
  1784. bt.showFailAlert(response);
  1785. return null;
  1786. }
  1787.  
  1788. if (typeof data !== "object" || !("result" in data) || data.result === null || !("results" in data.result)) {
  1789. if ("error" in data && data.error !== null && "message" in data.error) {
  1790. bt.showFailAlert(response, data.error.message);
  1791. return null;
  1792. } else {
  1793. console.warn("Unexpected data from BTN", data, response);
  1794. return [];
  1795. }
  1796. }
  1797.  
  1798. var num = parseInt(data.result.results);
  1799.  
  1800. response.context.searchUrl = "https://broadcasthe.net/torrents.php?searchstr=" + encodeURIComponent(response.context.query);
  1801.  
  1802. if (num === 0) {
  1803. return [];
  1804. } else {
  1805. data = data.result.torrents;
  1806. data = Object.keys(data).map(function (k) {
  1807. return data[k];
  1808. });
  1809. return data;
  1810. }
  1811. },
  1812. onRender: function (torrents, table, response) {
  1813. var title, rows = "";
  1814.  
  1815. for (var i = 0, tl = torrents.length; i < tl; i++) {
  1816. 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>';
  1817.  
  1818. if (response.context.category === "docs") {
  1819. title = '<a href="https://broadcasthe.net/series.php?id=' + torrents[i].SeriesID + '" target="_blank">' + torrents[i].Series + '</a> - ' + title;
  1820. }
  1821.  
  1822. rows += '<tr><td>' + title + '</td><td>' + torrents[i].Seeders + '</td><td><a href="' + torrents[i].DownloadURL + '">' + bt.humanizeSize(torrents[i].Size) + '</a></td></tr>';
  1823. }
  1824.  
  1825. table.html(rows);
  1826. },
  1827. config: function () {
  1828. 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>');
  1829. var input = $('<input type="text" class="form-control input-sm" style="width:250px">').val(bt.getPersistentValue("BTN_KEY"));
  1830. var btn = $('<button type="submit" class="btn btn-primary btn-sm">Save</button>')
  1831. .on("click", function () {
  1832. bt.setPersistentValue("BTN_KEY", input.val().trim());
  1833. alert("Saved!");
  1834. });
  1835.  
  1836. form
  1837. .append(' ', btn)
  1838. .children().first().append(' ', input);
  1839.  
  1840. return form;
  1841. },
  1842. icon: ""
  1843. };
  1844. });
  1845. bt.addSource("CG", function () {
  1846. var cg = "https://cinemageddon.net/browse.php?search={query}";
  1847.  
  1848. return {
  1849. url: {
  1850. all: cg,
  1851. movies: cg,
  1852. movies_1080: cg + " 1080",
  1853. movies_720: cg + " 720p",
  1854. movies_remux: cg + " REMUX",
  1855. movies_bluray: [cg + " BD25", cg + " BD50"],
  1856. movies_dvd: cg + " DVD-R",
  1857. docs: cg + "&c15=1",
  1858. music: cg + "&c11=1",
  1859. elearning: cg + "&c19=1&c5=1",
  1860. ebooks: cg + "&c19=1"
  1861. },
  1862. onParse: {
  1863. cleanup: [".torrenttable span"],
  1864. row: ".torrenttable:last > tbody > tr",
  1865. link_prepend: "https://cinemageddon.net/",
  1866. sel: [
  1867. {text: "a[href*='details.php']:eq(0)"},
  1868. {text: "> td:eq(6)"},
  1869. {text: "> td:eq(4)", link: "a[href*='download.php']:eq(0)", noblank: true}
  1870. ]
  1871. },
  1872. icon: ""
  1873. };
  1874. });
  1875. bt.addSource("DS", function () {
  1876. // NB: Search in multiple categories at once doesn't work
  1877. var ds = "https://www.dvdseed.eu/browse2.php?wheresearch=1&incldead=1&search={query}";
  1878.  
  1879. return {
  1880. url: {
  1881. all: ds,
  1882. movies: ds,
  1883. movies_1080: ds + " 1080p&kopia_reencode=1",
  1884. movies_720: ds + " 720p",
  1885. movies_bluray: ds + "&pkat_bd25=1&pkat_bd50=1&kopia_clone=1&kopia_modify=1&kopia_custom=1",
  1886. movies_dvd: ds + "&pkat_dvd5=1&pkat_dvd9=1&kopia_clone=1&kopia_modify=1&kopia_custom=1",
  1887. tv: [ ds + "&c74=1", ds + "&c31=1" ],
  1888. mvids: [ ds + "&c6=1", "https://www.dvdseed.eu/browse.php?id=3&t=3&c5=1&wheresearch=1&incldead=1&search=" ],
  1889. docs: ds + "&c69=1",
  1890. abooks: ds + "&c98=1",
  1891. apps_win: ds + "&c27=1",
  1892. music: ds + "&c91=1",
  1893. music_flac: ds + "&c91=1",
  1894. xxx: ds + "&c9=1"
  1895. },
  1896. onParse: {
  1897. row: "#torrentable tr:gt(0)",
  1898. link_prepend: "https://www.dvdseed.eu/",
  1899. sel: [
  1900. {
  1901. text: "a[href*='details.php']:eq(0)",
  1902. freeleech: "meta[src*='/free.png']"
  1903. },
  1904. {text: "> td:eq(6)"},
  1905. {text: "> td:eq(5)", link: "a[href*='download.php']:eq(0)", noblank: true}
  1906. ]
  1907. },
  1908. onFilter: function (data, response) {
  1909. if (["movies","movies_1080","movies_720","movies_bluray","movies_dvd","tv"].indexOf(response.context.category) !== -1) {
  1910. var stoplist = ["dtsflac","3d1.png","audiobook","music.png","porn.png"];
  1911. var stoplistLength = stoplist.length;
  1912. return data.filter(function(){
  1913. var html = this.innerHTML;
  1914. for (var i = 0; i < stoplistLength; i++) {
  1915. if (html.indexOf(stoplist[i]) !== -1) {
  1916. return false;
  1917. }
  1918. }
  1919. return true;
  1920. });
  1921. } else {
  1922. return data;
  1923. }
  1924. },
  1925. icon: ""
  1926. };
  1927. });
  1928. bt.addSource("HDB", function () {
  1929. var hdb = "https://hdbits.org/browse.php?descriptions=0&search={query}";
  1930.  
  1931. return {
  1932. url: {
  1933. all: hdb,
  1934. movies: hdb + "&c1=1",
  1935. movies_remux: hdb + "&c1=1&m5=1",
  1936. movies_bluray: hdb + "&c1=1&m1=1",
  1937. // movies_hd: hdb + "&c1=1&m4=1&m3=1&m6=1",
  1938. movies_720: hdb + "+720p&c1=1&m4=1&m3=1&m6=1",
  1939. movies_1080: hdb + "+1080p&c1=1&m4=1&m3=1&m6=1",
  1940. tv: hdb + "&c2=1",
  1941. mvids: hdb + "&c4=1",
  1942. xxx: [hdb + "&c7=1", hdb + "&descriptions=1&c7=1"],
  1943. docs: hdb + "&c3=1"
  1944. },
  1945. onParse: {
  1946. row: "#torrent-list > tbody > tr",
  1947. link_prepend: "https://hdbits.org",
  1948. sel: [
  1949. {
  1950. text: "> td:eq(2) a:eq(0)",
  1951. freeleech: "a.fl"
  1952. },
  1953. {text: "> td:eq(7)"},
  1954. {text: "> td:eq(5)", link: "a[href*='download.php']:eq(0)", noblank: true}
  1955. ]
  1956. },
  1957. onFilter: bt.filter3dMovies,
  1958. icon: ""
  1959. };
  1960. });
  1961. bt.addSource("HDS", function () {
  1962. var hds = "https://hdsky.me/torrents.php?incldead=0&spstate=0&inclbookmarked=0&search_area=0&search_mode=0&search={query}";
  1963.  
  1964. return {
  1965. url: {
  1966. all: hds,
  1967. movies: hds + "&cat401=1&cat410=1&cat405=1",
  1968. movies_1080: hds + "&cat401=1&cat405=1&medium7=1&medium5=1&medium11=1&standard1=1&standard2=1",
  1969. movies_720: hds + "&cat401=1&cat405=1&medium7=1&medium5=1&medium11=1&standard3=1",
  1970. movies_remux: hds + "&cat401=1&cat405=1&medium3=1",
  1971. movies_bluray: hds + "&cat401=1&cat405=1&medium1=1&medium12=1",
  1972. movies_dvd: hds + "&cat401=1&cat405=1&medium6=1",
  1973. docs: hds + "&cat404=1",
  1974. tv: hds + "&cat402=1&cat403=1",
  1975. mvids: hds + "&cat406=1",
  1976. music: hds + "&cat408=1",
  1977. music_flac: hds + "&cat408=1&audiocodec1=1&audiocodec2=1"
  1978. },
  1979. onParse: {
  1980. row: "table.torrents tr.progresstr",
  1981. link_prepend: "https://hdsky.me/",
  1982. sel: [
  1983. {
  1984. text: "> td:eq(1) a:eq(0)",
  1985. freeleech: ".pro_free"
  1986. },
  1987. {text: "> td:eq(5)"},
  1988. {text: "> td:eq(4)", link: "a[href*='download.php']:eq(0)", noblank: true}
  1989. ]
  1990. },
  1991. onFilter: bt.filter3dMovies,
  1992. icon: ""
  1993. };
  1994. });
  1995. bt.addSource("HDT", function () {
  1996. var hdt = "https://hd-torrents.org/torrents.php?active=0&options=0&search={query}";
  1997.  
  1998. return {
  1999. url: {
  2000. all: hdt,
  2001. movies: hdt + "&category[]=1&category[]=2&category[]=5&category[]=3",
  2002. movies_1080: hdt + "&category[]=5",
  2003. movies_720: hdt + "&category[]=3",
  2004. movies_remux: hdt + "&category[]=2",
  2005. movies_bluray: hdt + "&category[]=1",
  2006. music: hdt + "&category[]=44",
  2007. music_flac: hdt + "&category[]=44",
  2008. docs: hdt + "&genre[]=Documentary",
  2009. tv: hdt + "&category[]=59&category[]=60&category[]=30&category[]=38",
  2010. mvids: hdt + "&category[]=61&category[]=62&category[]=57&category[]=45",
  2011. xxx: hdt + "&options=3&category[]=58&category[]=48&category[]=47"
  2012. },
  2013. onParse: {
  2014. row: "table.mainblockcontenttt > tbody > tr:has(a[href*='download.php'])",
  2015. link_prepend: "https://hd-torrents.org/",
  2016. sel: [
  2017. {text: "a:eq(1)"},
  2018. {text: "> td:eq(9)"},
  2019. {text: "> td:eq(7)", link: "a[href*='download.php']:eq(0)", noblank: true}
  2020. ]
  2021. },
  2022. onValidate: function (response) {
  2023. return response.responseText.indexOf("You're not authorized to view this Torrents") === -1 ? true : "login needed";
  2024. },
  2025. onFilter: bt.filter3dMovies,
  2026. icon: ""
  2027. };
  2028. });
  2029. bt.addSource("KG", function () {
  2030. var kg = "https://karagarga.in/browse.php?search_type=title&incldead=&search={query}";
  2031.  
  2032. return {
  2033. url: {
  2034. all: kg,
  2035. movies: kg + "&cat=1",
  2036. movies_1080: kg + "&hdrip=2",
  2037. movies_720: kg + "&hdrip=1",
  2038. movies_bluray: kg + "&hdrip=3",
  2039. movies_dvd: kg + "&dvdr=1",
  2040. docs: kg + "&genre=20",
  2041. music: kg + "&cat=2",
  2042. elearning: kg + "&cat=3",
  2043. ebooks: kg + "&genre=41",
  2044. abooks: kg + "&genre=40",
  2045. comics: kg + "&genre=42"
  2046. },
  2047. onParse: {
  2048. cleanup: ["#browse tr:has(a[href*='down.php']) span:not(:first-child)"],
  2049. row: "#browse tr:has(a[href*='down.php'])",
  2050. link_prepend: "https://karagarga.in/",
  2051. sel: [
  2052. {
  2053. text: function (context) {
  2054. var cols = context.children();
  2055. var year = cols.eq(3);
  2056. var genre = cols.eq(0).find("meta");
  2057.  
  2058. if (genre.length === 2) {
  2059. var src = genre.eq(1).attr('src');
  2060. if (~src.indexOf('hdrip720')) year.append(' [720p]');
  2061. else if (~src.indexOf('hdrip1080')) year.append(' [1080p]');
  2062. else if (~src.indexOf('dvdr')) year.append(' [DVD-R]');
  2063. else if (~src.indexOf('bluray')) year.append(' [Blu-ray]');
  2064. }
  2065.  
  2066. return cols.slice(1,4);
  2067. },
  2068. link: "a[href*='details.php']:eq(0)",
  2069. freeleech: function(context){
  2070. return context.hasClass("featuredrow");
  2071. }
  2072. },
  2073. {text: "> td:eq(12)"},
  2074. {text: "> td:eq(10)", link: "a[href*='down.php']:eq(0)", noblank: true}
  2075. ]
  2076. },
  2077. onPrepareQuery: function (context) {
  2078. var i,
  2079. len = context.url.length,
  2080. year = bt.extractYear(context, true),
  2081. res = bt.extractResolution(context);
  2082.  
  2083. if (res === "720p") {
  2084. for (i = 0; i < len; i++) {
  2085. context.url[i] += "&hdrip=1";
  2086. }
  2087. } else if (res === "1080p" || res === "1080i") {
  2088. for (i = 0; i < len; i++) {
  2089. context.url[i] += "&hdrip=2";
  2090. }
  2091. }
  2092.  
  2093. context.query = context.query.trim().split(' ');
  2094. for (i = 0, len = context.query.length; i < len; i++) {
  2095. context.query[i] = encodeURIComponent('+') + context.query[i];
  2096. }
  2097. context.query = context.query.join('%20');
  2098.  
  2099. if (year) {
  2100. context.query += "%20" + year;
  2101. }
  2102. },
  2103. // onFilter: bt.requireAllWords,
  2104. icon: ""
  2105. };
  2106. });
  2107. bt.addSource("MVGroup", function () {
  2108. var q = [
  2109. "https://forums.mvgroup.org/maintracker.php?filter={query}",
  2110. "https://forums.mvgroup.org/forumtracker.php?filter={query}",
  2111. ];
  2112.  
  2113. return {
  2114. url: {
  2115. all: q,
  2116. docs: q,
  2117. },
  2118. onValidate: function (response) {
  2119. return response.finalUrl.indexOf('tracker.php?filter') === -1 ? "login needed" : true;
  2120. },
  2121. onParse: {
  2122. row: "tr:has(a[href*='?showtopic='])",
  2123. link_prepend: "https://forums.mvgroup.org",
  2124. sel: [
  2125. {
  2126. text: function (context) {
  2127. return context[0].firstElementChild.textContent;
  2128. },
  2129. link: "a[href*='?showtopic=']",
  2130. },
  2131. ]
  2132. },
  2133. icon: ""
  2134. };
  2135. });
  2136. bt.addSource("PTP", function () {
  2137. 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}";
  2138.  
  2139. return {
  2140. url: {
  2141. all: "https://passthepopcorn.me/torrents.php?action=basic&order_by=relevance&grouping=1&searchstr={query}",
  2142. movies: ptpMovies,
  2143. // movies_hd: ptpMovies + "&resolution=anyhd",
  2144. movies_dvd: ptpMovies + "&format[]=DVD5&format[]=DVD9&grouping=0",
  2145. movies_bluray: ptpMovies + "&format[]=BD25&format[]=BD50&grouping=0",
  2146. movies_remux: ptpMovies + "&remastertitle=Remux",
  2147. movies_720: ptpMovies + "&resolution=720p",
  2148. movies_1080: ptpMovies + "&resolution=anyhd",
  2149. docs: ptpMovies + "&taglist=documentary",
  2150. mvids: "https://passthepopcorn.me/torrents.php?action=advanced&order_by=relevance&filter_cat[5]=1&searchstr={query}",
  2151. tv: "https://passthepopcorn.me/torrents.php?action=advanced&order_by=relevance&filter_cat[3]=1&searchstr={query}"
  2152. },
  2153. onPrepareQuery: [bt.extractGazelleYear, bt.extractGazelleResolution],
  2154. onParse: function (response) {
  2155. // If redirected to a single search result
  2156. if (response.finalUrl.indexOf('torrents.php?id=') !== -1) {
  2157. var html = $(bt.replaceImages(response.responseText));
  2158.  
  2159. $('.torrent-info-link span', html).remove();
  2160.  
  2161. var torrents = [];
  2162.  
  2163. $(".group_torrent_header", html).each(function () {
  2164. var row = $(this);
  2165. var cols = row.children();
  2166. var href = $("a[title='Permalink']", row).first().attr('href');
  2167.  
  2168. var title = $(".torrent-info-link", row);
  2169. var seeding = title.hasClass("torrent-info-link--user-seeding");
  2170. title = title.first().text();
  2171.  
  2172. torrents.push({
  2173. Title: '<a href="' + href + '"' + (seeding ? ' class="torrent-info-link--user-seeding"' : '') + '>' + title + '</a>',
  2174. TorrentId: href.split("torrentid=")[1],
  2175. Size: cols.eq(1).text(),
  2176. Seeders: cols.eq(3).text()
  2177. });
  2178. });
  2179.  
  2180. return [{
  2181. GroupId: response.finalUrl.split('?id=')[1].split(bt.matchFirstNonDigit)[0],
  2182. Title: html.find('h2.page__title').first().text().split(' [')[0],
  2183. Year: html.find('h2.page__title').first().text().split(' [')[1].split(']')[0],
  2184. Cover: $('meta.sidebar-cover-image', html).length > 0 ? $('meta.sidebar-cover-image', html).first().attr('src') : null,
  2185. GroupingQualities: [{
  2186. Torrents: torrents
  2187. }]
  2188. }];
  2189. }
  2190.  
  2191. if (response.responseText.indexOf('PageData = ') === -1) {
  2192. return [];
  2193. }
  2194.  
  2195. var data = response.responseText.split('PageData = ')[1].trim().split('</script>')[0].trim().slice(0, -1);
  2196.  
  2197. try {
  2198. return JSON.parse(data).Movies;
  2199. } catch (e) {
  2200. console.warn("PTP JSON parsing failed.", data);
  2201. return [];
  2202. }
  2203. },
  2204. onEnable: function(){
  2205. if (bt.customCSS.text().indexOf('.torrent-info-link--user-seeding') === -1) {
  2206. bt.customCSS.append(' .torrent-info-link--user-seeding {font-weight:bold;color:darkorange} ');
  2207. }
  2208. },
  2209. onRender: function (movies, table) {
  2210. var groupTable, groupTableHTML, torrent, tr;
  2211.  
  2212. for (var i = 0, ml = movies.length; i < ml; i++) {
  2213. 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>';
  2214.  
  2215. if (movies[i].Cover) {
  2216. tr = $(tr);
  2217. $(tr[0].firstChild)
  2218. .addClass('cover')
  2219. .css('backgroundImage', 'url(' + movies[i].Cover + ')');
  2220. }
  2221.  
  2222. if (!('GroupingQualities' in movies[i])) return;
  2223.  
  2224. var groups = movies[i].GroupingQualities;
  2225. groupTable = bt.table.clone().addClass("torrent-table");
  2226.  
  2227. groupTableHTML = "";
  2228. for (var g = 0, gl = groups.length; g < gl; g++) {
  2229. if ('CategoryName' in groups[g]) {
  2230. groupTableHTML += '<tr><td colspan=3><b>' + groups[g].CategoryName + ' / ' + groups[g].QualityName + '</b></td></tr>';
  2231. }
  2232.  
  2233. for (var n = 0, nl = groups[g].Torrents.length; n < nl; n++) {
  2234. torrent = groups[g].Torrents[n];
  2235.  
  2236. var title = bt.span.clone().html(torrent.Title);
  2237. var a = title.children('a').first();
  2238. a.attr("target", "_blank");
  2239. a.attr("href", "https://passthepopcorn.me/" + a.attr("href"));
  2240.  
  2241. 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>';
  2242. }
  2243. }
  2244. groupTable.html(groupTableHTML);
  2245.  
  2246. var torRow = bt.tr.clone().append(bt.td.clone().append(groupTable));
  2247.  
  2248. table.append(tr, torRow);
  2249. }
  2250. },
  2251. icon: ""
  2252. };
  2253. });
  2254. bt.addSource("TorViet", function () {
  2255. var torviet = "http://torviet.com/torrents.php?incldead=0&search_area=0&search_mode=0&search={query}";
  2256.  
  2257. return {
  2258. url: {
  2259. all: torviet,
  2260. music: [torviet + "&sltCategory=5&sltSubCategory=126", torviet + "&sltCategory=5&sltSubCategory=130"],
  2261. music_flac: torviet + "&sltCategory=5&sltSubCategory=126",
  2262. // music_mp3: torviet + "&sltCategory=5&sltSubCategory=130",
  2263. movies: torviet + "&sltCategory=2",
  2264. movies_1080: torviet + "&sltCategory=2&sltSubCategory=125",
  2265. movies_720: torviet + "&sltCategory=2&sltSubCategory=124",
  2266. movies_remux: torviet + " REMUX&sltCategory=2&sltSubCategory=127",
  2267. movies_bluray: torviet + "&sltCategory=2&sltSubCategory=127",
  2268. docs: [torviet + "&sltCategory=3&sltSubCategory=0&sltGenre=62", torviet + "&sltCategory=2&sltSubCategory=0&sltGenre=32"],
  2269. tv: torviet + "&sltCategory=3&sltSubCategory=128",
  2270. elearning: torviet + "&sltCategory=6",
  2271. ebooks: torviet + "&sltCategory=6&sltSubCategory=112",
  2272. abooks: torviet + "&sltCategory=6&sltSubCategory=117",
  2273. mags: torviet + "&sltCategory=6&sltSubCategory=112",
  2274. apps_win: torviet + "&sltCategory=4&sltSubCategory=76",
  2275. mvids: torviet + "&sltCategory=5&sltSubCategory=92",
  2276. games_pc: torviet + "&sltCategory=1&sltSubCategory=7"
  2277. },
  2278. onParse: {
  2279. row: "#idtorrent table.torrents > tbody > tr:not(:first-child)",
  2280. link_prepend: "http://torviet.com",
  2281. sel: [
  2282. {
  2283. text: ".torrentname a:eq(0)",
  2284. freeleech: ".pro_free"
  2285. },
  2286. {text: "> td:eq(4)"},
  2287. {text: "> td:eq(3)", link: "a[href*='download.php']:eq(0)", noblank: true}
  2288. ]
  2289. },
  2290. onFilter: [bt.filter3dMovies],
  2291. icon: ""
  2292. };
  2293. });
  2294. bt.addSource("Vimeo", function () {
  2295. var domain = "https://vimeo.com";
  2296. var q = domain + "/search?q={query}";
  2297.  
  2298. return {
  2299. url: {
  2300. all: q,
  2301. movies: q,
  2302. music: q + "&category=9",
  2303. docs: q + "&category=117&duration=long",
  2304. mvids: q + "&category=9"
  2305. },
  2306. onParse: function (response) {
  2307.  
  2308. try {
  2309. var json = response.responseText;
  2310. json = json.split('var data = ')[1];
  2311. json = json.split('if (0) {')[0].trim().slice(0, -1);
  2312. json = JSON.parse(json);
  2313. json = json.filtered.data;
  2314. } catch (e) {
  2315. bt.showFailAlert(response);
  2316. return null;
  2317. }
  2318.  
  2319. return json;
  2320. },
  2321. onRender: function (data, table) {
  2322. var video, rows = "", id;
  2323.  
  2324. for (var i = 0, len = data.length; i < len; i++) {
  2325. video = data[i];
  2326. 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>';
  2327. }
  2328.  
  2329. table.html(rows);
  2330. },
  2331. onEnable: function () {
  2332. $(bt.mainColumn).on('click', 'a.vimeo-link', function (e) {
  2333. e.preventDefault();
  2334. e.stopPropagation();
  2335.  
  2336. var a = $(this);
  2337. var td = $(this.parentNode);
  2338. var iframe = td.children("iframe");
  2339. if (iframe.length > 0) {
  2340. iframe.remove();
  2341. } else {
  2342. bt.mainColumn.find("iframe").remove();
  2343. var href = a.attr("href");
  2344. var id = href.split("vimeo.com/")[1].split("?")[0].split("/")[0]; // just in case
  2345. 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>');
  2346. }
  2347.  
  2348. return false;
  2349. });
  2350. },
  2351. icon: ""
  2352. };
  2353. });
  2354. bt.addSource("YouTube", function () {
  2355. var q = "https://m.youtube.com/results?ajax=1&tsp=1&q={query}";
  2356.  
  2357. return {
  2358. url: {
  2359. all: q,
  2360. docs: q,
  2361. music: q,
  2362. movies: q,
  2363. mvids: q,
  2364. games_pc: q
  2365. },
  2366. onHttpRequest: function (requestData) {
  2367. requestData.headers = {
  2368. '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'
  2369. };
  2370. },
  2371. onParse: function(response){
  2372. try {
  2373. var data = JSON.parse(response.responseText.slice(response.responseText.indexOf("{")));
  2374. } catch (e) {
  2375. bt.showFailAlert(response, "unexpected data");
  2376. return null;
  2377. }
  2378.  
  2379. if (!data || !("result" in data) || data.result !== 'ok') {
  2380. bt.showFailAlert(response, "unexpected data");
  2381. return null;
  2382. }
  2383.  
  2384. if ("content" in data && "search_results" in data.content && "contents" in data.content.search_results) {
  2385. return data.content.search_results.contents;
  2386. }
  2387.  
  2388. return [];
  2389. },
  2390. onRender: function (data, table) {
  2391. var video, rows = "", id;
  2392. console.log(data);
  2393. for (var i = 0, len = data.length; i < len; i++) {
  2394. video = data[i];
  2395.  
  2396. switch (video.item_type) {
  2397. case "compact_video":
  2398. 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>';
  2399. break;
  2400. case "compact_playlist":
  2401. 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>';
  2402. break;
  2403. case "compact_channel":
  2404. 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>';
  2405. break;
  2406. }
  2407. }
  2408.  
  2409. table.html(rows);
  2410. },
  2411. /*
  2412. onParse: {
  2413. row: "ol.item-section > li:has(a[href*='watch?v='])",
  2414. link_prepend: "https://www.youtube.com",
  2415. sel: [
  2416. {
  2417. text: "h3 > .yt-uix-tile-link",
  2418. class: "youtube-link",
  2419. vod: ".yt-badge-ypc"
  2420. },
  2421. {text: ".video-time"},
  2422. {text: ".yt-lockup-byline > .g-hovercard", width:"20%"}
  2423. ]
  2424. },
  2425. */
  2426. onEnable: function(){
  2427. $(bt.mainColumn).on('click', '.youtube-link', function (e) {
  2428. var id, a = $(this), td = $(this.parentNode);
  2429. var iframe = td.children("iframe");
  2430. if (iframe.length > 0) {
  2431. iframe.remove();
  2432. } else {
  2433. bt.mainColumn.find("iframe").remove();
  2434.  
  2435. var url = "https://www.youtube.com/embed/";
  2436. if (id = a.data("video-id")) {
  2437. url += id + "?autoplay=1";
  2438. } else if (id = a.data("playlist-id")) {
  2439. url += "?listType=playlist&autoplay=1&iv_load_policy=3&list=" + id;
  2440. } else if (id = a.data("user-id")) {
  2441. url += "?listType=user_uploads&autoplay=1&iv_load_policy=3&list=" + id;
  2442. } else {
  2443. return;
  2444. }
  2445.  
  2446. td.append('<iframe width="560" height="315" src="' + url + '" frameborder="0" allowfullscreen style="display:block;margin-top:10px;"></iframe>');
  2447. }
  2448.  
  2449. e.preventDefault();
  2450. e.stopPropagation();
  2451. return false;
  2452. });
  2453. },
  2454. onFilter: function(data, response){
  2455. response.context.searchUrl = "https://www.youtube.com/results?q=" + encodeURIComponent(response.context.query);
  2456.  
  2457. if (response.context.category === "mvids" || response.context.category === "docs") {
  2458. return data.filter(function (v) {
  2459. if (!~v.item_type.indexOf("compact_")) {
  2460. return false;
  2461. }
  2462.  
  2463. var title = v.title.runs[0].text.toLowerCase();
  2464.  
  2465. return !(~title.indexOf("audio") || ~title.indexOf("lyric") || ~title.indexOf("album"));
  2466. });
  2467. }
  2468.  
  2469. return data;
  2470. },
  2471. icon: ""
  2472. };
  2473. });
  2474. bt.addSource("AVG", function () {
  2475. var q = "https://avg.club/tracker.php?max=1&to=1&nm={query}";
  2476.  
  2477. return {
  2478. url: {
  2479. all: q,
  2480. games_pc: q,
  2481. },
  2482. onParse: {
  2483. row: "#tor-tbl > tbody > tr",
  2484. link_prepend: "https://avg.club/",
  2485. sel: [
  2486. {
  2487. text: "a.tLink",
  2488. },
  2489. {
  2490. text: "td.seedmed",
  2491. },
  2492. {
  2493. text: "a.tr-dl",
  2494. noblank: true,
  2495. }
  2496. ]
  2497. },
  2498. icon: ""
  2499. };
  2500. });
  2501. bt.addSource("BIB", function () {
  2502. var domain = 'https://bibliotik.me';
  2503. var bib = domain + '/torrents/?search=@title%20{query}';
  2504.  
  2505. return {
  2506. url: {
  2507. all: bib,
  2508. elearning: bib,
  2509. ebooks: bib + "&cat[]=2&cat[]=5",
  2510. mags: bib + "&cat[]=6&cat[]=7",
  2511. apps_win: bib + "&cat[]=1",
  2512. abooks: bib + "&cat[]=3",
  2513. comics: bib + "&cat[]=4"
  2514. },
  2515. onParse: {
  2516. cleanup: ['time', '.taglist'],
  2517. row: "#torrents_table > tbody > tr",
  2518. link_prepend: domain,
  2519. sel: [
  2520. {text: "> td:has(a[href*='/torrents/']):eq(0)", link: "a[href*='/torrents/']:eq(0)"},
  2521. {text: "a[href*='peers']:eq(0)"},
  2522. {
  2523. text: function (context) {
  2524. return $("> .t_files_size_added", context).text().split(',')[1].trim();
  2525. },
  2526. link: "a[href*='download']:eq(0)",
  2527. noblank: true
  2528. }
  2529. ]
  2530. },
  2531. icon: ""
  2532. };
  2533. });
  2534. bt.addSource("BS", function () {
  2535. var bs = "http://bitspyder.net/browse.php?incldead=1&scope=0&search={query}";
  2536.  
  2537. return {
  2538. url: {
  2539. all: bs,
  2540. elearning: bs,
  2541. docs: bs + "&c42=1",
  2542. abooks: bs + "&c40=1",
  2543. mags: bs + "&c57=1",
  2544. ebooks: bs
  2545. },
  2546. onParse: {
  2547. row: "tr.alt1,tr.alt2",
  2548. link_prepend: "http://bitspyder.net/",
  2549. sel: [
  2550. {text: "a.altlink6"},
  2551. {text: "> td:eq(2)"},
  2552. {
  2553. text: "font[color='#A52A2A']:eq(0)",
  2554. link: function(context){
  2555. var link = context[0].firstElementChild.nextElementSibling.firstElementChild;
  2556. var id = link.getAttribute("href").split("id=")[1].split("&")[0];
  2557. return "download.php/" + id + "/" + link.textContent.trim().replace(bt.spacesRegex, "%20") + ".torrent";
  2558. }
  2559. }
  2560. ]
  2561. },
  2562. onPrepareQuery: function (context) {
  2563. context.query = context.query.split(' ');
  2564. for (var i = 0, l = context.query.length; i < l; i++) {
  2565. context.query[i] = encodeURIComponent('+') + context.query[i];
  2566. }
  2567. context.query = context.query.join('%20');
  2568. },
  2569. icon: ""
  2570. };
  2571. });
  2572. bt.addSource("EBZ", function () {
  2573. var ebz = "http://app.elbitz.net/browse.php?incldead=1&typ=0&search={query}";
  2574.  
  2575. return {
  2576. url: {
  2577. all: ebz,
  2578. elearning: ebz,
  2579. docs: ebz + "&c10=1",
  2580. ebooks: ebz,
  2581. mags: ebz + "&c16=1",
  2582. abooks: ebz + "&c6=1"
  2583. },
  2584. onParse: {
  2585. row: "tr:not(:has('tr')):has(a[href*='download.php'])",
  2586. sel: [
  2587. {
  2588. text: function (context) {
  2589. return context.find("a[href*='details.php']:eq(0)").attr('title');
  2590. },
  2591. link: "a[href*='details.php']:eq(0)",
  2592. },
  2593. {text: "> td:eq(6)"},
  2594. {text: "> td:eq(5)", link: "a[href*='download.php']:eq(0)", noblank: true}
  2595. ]
  2596. },
  2597. icon: ""
  2598. };
  2599. });
  2600. bt.addSource("Genesis_Fiction", function () {
  2601. var genf = "http://gen.lib.rus.ec/foreignfiction/?s=";
  2602.  
  2603. return {
  2604. url: {
  2605. all: genf,
  2606. elearning: genf,
  2607. ebooks: genf,
  2608. fiction: genf
  2609. },
  2610. onParse: {
  2611. row: "tr:has(a[href*='foreignfiction/ads.php'])",
  2612. sel: [
  2613. {text: "> td:eq(0), > td:eq(2)"},
  2614. {text: "a[href*='/ads.php']:eq(0)", link: "a[href*='/ads.php']:eq(0)", noblank: true}
  2615. ]
  2616. },
  2617. icon: ""
  2618. };
  2619. });
  2620. bt.addSource("Genesis_NonFiction", function () {
  2621. var gen = "http://gen.lib.rus.ec/search.php?open=0&view=simple&column=def&req=";
  2622.  
  2623. return {
  2624. url: {
  2625. all: gen,
  2626. elearning: gen,
  2627. ebooks: gen
  2628. },
  2629. onParse: {
  2630. prepare: function (response) {
  2631. if (response.responseText.indexOf('<table width=100% cellspacing=1') === -1) {
  2632. return "<table></table>";
  2633. }
  2634.  
  2635. var html = response.responseText.replace(bt.imgTagRegex, '<meta ').split('<table width=100% cellspacing=1')[1];
  2636. html = '<table width=100% cellspacing=1' + html;
  2637. html = html.split('</table')[0];
  2638. html += '</table>';
  2639.  
  2640. return html;
  2641. },
  2642. row: "tr:gt(0)",
  2643. link_prepend: "http://gen.lib.rus.ec/",
  2644. sel: [
  2645. {text: "a[href*='book/']:eq(0), > td:eq(8)"},
  2646. {text: "> td:eq(7)", link: "a[href*='/ads.php']:eq(0)", noblank: true}
  2647. ]
  2648. },
  2649. icon: ""
  2650. };
  2651. });
  2652. bt.addSource("MAM", function () {
  2653. 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";
  2654.  
  2655. return {
  2656. url: {
  2657. all: mam,
  2658. elearning: mam,
  2659. 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",
  2660. 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",
  2661. 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",
  2662. 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",
  2663. 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"
  2664. },
  2665. onParse: {
  2666. cleanup: ["a[href*='filelistLink']"],
  2667. row: "tr:gt(0)",
  2668. link_prepend: "https://www.myanonamouse.net",
  2669. sel: [
  2670. {text: "a.title:eq(0)"},
  2671. {text: "> td:eq(6) > p:eq(0)"},
  2672. {
  2673. text: function(context){
  2674. return $("> td:eq(4)", context).text().replace('[', '').replace(']', '').trim();
  2675. },
  2676. link: "a[href*='download.php']:eq(0)",
  2677. noblank: true,
  2678. }
  2679. ]
  2680. },
  2681. onValidate: function (response) {
  2682. response.context.searchUrl = response.finalUrl.replace("/tor/js/loadSearch2.php", "/tor/browse.php");
  2683. return true;
  2684. },
  2685. icon: ""
  2686. };
  2687. });
  2688. bt.addSource("TGZ", function () {
  2689. var tgz = "https://thegeeks.click/browse.php?incldead=1&nonboolean=1&titleonly=1&search={query}";
  2690.  
  2691. return {
  2692. url: {
  2693. all: tgz,
  2694. elearning: tgz,
  2695. docs: tgz + " -pdf -ebook -mp3 -m4a -aac",
  2696. ebooks: tgz + " -xvid -pdtv -hdtv -avi -mp4 -wmv -mkv -dvd",
  2697. abooks: tgz + " AND (audiobook OR mp3 OR m4a OR aac OR flac OR ogg)&nonboolean=3"
  2698. },
  2699. onParse: {
  2700. row: "tr.ttable",
  2701. link_prepend: "https://thegeeks.click/",
  2702. sel: [
  2703. {
  2704. text: "a[href*='details.php']:eq(0)",
  2705. freeleech: "font[color='blue']:contains('FREE')"
  2706. },
  2707. {text: "> td:eq(8)"},
  2708. {text: "> td:eq(6)", link: "a[href*='download.php']:eq(0)", noblank: true}
  2709. ]
  2710. },
  2711. onPrepareQuery: function (context) {
  2712. if (context.category === "abooks") {
  2713. context.query = context.query.split(' ').join(' AND ');
  2714. }
  2715. },
  2716. icon: ""
  2717. };
  2718. });
  2719. bt.addSource("Traum", function () {
  2720. var traum = "http://lib.it.cx/?find=";
  2721.  
  2722. return {
  2723. url: {
  2724. all: traum,
  2725. elearning: traum,
  2726. ebooks: traum,
  2727. fiction: traum
  2728. },
  2729. onParse: {
  2730. row: "tr:has(a[href*='epub']):not(:has('tr'))",
  2731. link_prepend: "http://lib.it.cx",
  2732. sel: [
  2733. {text: "a:eq(0)", noblank: true},
  2734. {text: "> td:last"}
  2735. ]
  2736. },
  2737. icon: ""
  2738. };
  2739. });
  2740. bt.addSource("bB", function () {
  2741. var bb = "https://baconbits.org/torrents.php?action=basic&searchstr={query}";
  2742.  
  2743. return {
  2744. url: {
  2745. all: bb,
  2746. music: bb + "&filter_cat[1]=1",
  2747. music_flac: bb + "&filter_cat[1]=1",
  2748. movies: bb + "&filter_cat[9]=1",
  2749. movies_1080: bb + "&filter_cat[9]=1",
  2750. movies_720: bb + "&filter_cat[9]=1",
  2751. docs: bb + "&filter_cat[9]=1&filter_cat[10]=1",
  2752. tv: bb + "&filter_cat[10]=1",
  2753. apps_win: bb + "&filter_cat[2]=1",
  2754. mags: bb + "&filter_cat[6]=1",
  2755. games_pc: bb + "&filter_cat[11]=1",
  2756. comics: bb + "&filter_cat[7]=1",
  2757. ebooks: bb + "&filter_cat[3]=1",
  2758. abooks: bb + "&filter_cat[4]=1",
  2759. fiction: bb + "&filter_cat[3]=1",
  2760. 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"
  2761. },
  2762. onParse: {
  2763. cleanup: [".tags"],
  2764. row: "#torrent_table tr.torrent",
  2765. link_prepend: "https://baconbits.org/",
  2766. sel: [
  2767. {
  2768. cleanup: ["> td:eq(1) > span:eq(0)"],
  2769. text: "> td:eq(1)",
  2770. link: "a[title='View Torrent']:eq(0)"
  2771. // freeleech: "strong:contains('Freeleech!')"
  2772. },
  2773. {text: "> td:eq(7)"},
  2774. {text: "> td:eq(4)", link: "a[href*='action=download']:eq(0)", noblank: true}
  2775. ]
  2776. },
  2777. resolution: null,
  2778. onPrepareQuery: function(context){
  2779. var year = bt.extractYear(context);
  2780.  
  2781. if (year) {
  2782. for (var i = 0, l = context.url.length; i < l; i++) {
  2783. context.url[i] += "&action=advanced&torrentname={query}&filelist=" + year;
  2784. }
  2785. }
  2786.  
  2787. context.src.resolution = bt.extractResolution(context);
  2788. },
  2789. onFilter: [bt.filter3dMovies, function (data, response) {
  2790. if (response.context.category === "music_flac") {
  2791. data = data.filter(function(){
  2792. return this.textContent.indexOf('FLAC') !== -1;
  2793. });
  2794. }
  2795.  
  2796. if (response.context.category === "movies_1080") response.context.src.resolution = "1080p";
  2797. if (response.context.category === "movies_720") response.context.src.resolution = "720p";
  2798.  
  2799. if (response.context.src.resolution) {
  2800. response.context.query += " " + response.context.src.resolution;
  2801. response.context.src.resolution = null;
  2802. return bt.requireAllWords(data, response);
  2803. } else {
  2804. return data;
  2805. }
  2806. }],
  2807. icon: ""
  2808. };
  2809. });
  2810. bt.addSource("FL", function () {
  2811. var filelist = "https://filelist.ro/browse.php?searchin=1&sort=0&incldead=1&search={query}";
  2812.  
  2813. return {
  2814. url: {
  2815. all: filelist,
  2816. music: filelist + "&cat=11",
  2817. music_flac: filelist + " FLAC&cat=11",
  2818. movies: filelist,
  2819. movies_bluray: filelist + "&cat=20",
  2820. movies_remux: filelist + " REMUX",
  2821. movies_1080: [filelist + " 1080p&cat=4", filelist + " 1080p&cat=19", filelist + " 1080p&cat=15"],
  2822. movies_720: [filelist + " 720p&cat=4", filelist + " 720p&cat=19", filelist + " 720p&cat=15"],
  2823. movies_dvd: [filelist + "&cat=2", filelist + "&cat=3"],
  2824. docs: filelist + " documentary&searchin=0",
  2825. ebooks: [filelist + "&cat=16", filelist + "&cat=18"],
  2826. mags: filelist + "&cat=16",
  2827. tv: [filelist + "&cat=15&sort=2", filelist + "&cat=21&sort=2", filelist + "&cat=14&sort=2"],
  2828. elearning: filelist,
  2829. apps_win: filelist + "&cat=8",
  2830. mvids: filelist + "&cat=12",
  2831. xxx: filelist + "&searchin=0&cat=7"
  2832. },
  2833. onParse: {
  2834. row: ".torrentrow",
  2835. link_prepend: "https://filelist.ro/",
  2836. sel: [
  2837. {
  2838. text: "a[href*='details.php?id=']:eq(0)",
  2839. freeleech: "meta[alt='FreeLeech']"
  2840. },
  2841. {text: "> .torrenttable:eq(8)"},
  2842. {text: "> .torrenttable:eq(6)", link: "a[href*='download.php']:eq(0)", noblank: true}
  2843. ]
  2844. },
  2845. onFilter: [bt.filter3dMovies, function(data, response){
  2846. if (response.context.category === "movies_bluray") {
  2847. return data.filter(function(){
  2848. return this.textContent.toLowerCase().indexOf('remux') === -1;
  2849. });
  2850. } else {
  2851. return data;
  2852. }
  2853. }],
  2854. icon: ""
  2855. };
  2856. });
  2857. bt.addSource("PS", function () {
  2858. var ps = "https://polishsource.cz/browse.php?incldead=1&scene=0&pl=0&sub=&search_in=title&search={query}";
  2859.  
  2860. return {
  2861. url: {
  2862. all: ps,
  2863. movies: ps + "&c12=1&c11=1&c4=1&c43=1",
  2864. movies_1080: ps + " 1080&c11=1",
  2865. movies_720: ps + " 720p&c11=1",
  2866. movies_remux: ps + " REMUX&c43=1",
  2867. movies_bluray: ps + "&c43=1",
  2868. movies_dvd: ps + "&c4=1",
  2869. docs: ps + "&sub=Documentary",
  2870. music: ps + "&c42=1",
  2871. music_flac: ps + " FLAC&c42=1",
  2872. // music_mp3: ps + "+MP3&c42=1",
  2873. mvids: ps + " x264&c42=1",
  2874. tv: ps + "&c39=1",
  2875. elearning: ps + "&c5=1",
  2876. ebooks: ps + "&c5=1",
  2877. abooks: ps + " audiobook&search_in=both&c5=1",
  2878. mags: ps + "&c5=1",
  2879. apps_win: ps + "&c18=1",
  2880. games_pc: ps + "&c8=1",
  2881. xxx: ps + "&c13=1"
  2882. },
  2883. onParse: {
  2884. row: "#restable tr:gt(0)",
  2885. link_prepend: "https://polishsource.cz/",
  2886. sel: [
  2887. {text: "a[href*='details.php']:eq(0)"},
  2888. {text: "> td:eq(7)"},
  2889. {text: "> td:eq(4)", link: "a[href*='downloadssl.php']:eq(0)", noblank: true}
  2890. ]
  2891. },
  2892. onFilter: function(data, response){
  2893. if (response.context.category === "movies_bluray") {
  2894. return data.filter(function(){
  2895. return this.textContent.toLowerCase().indexOf('remux') === -1;
  2896. });
  2897. } else {
  2898. return data;
  2899. }
  2900. },
  2901. icon: ""
  2902. };
  2903. });
  2904. bt.addSource("RarBG", function () {
  2905. var rarbg = "https://rarbg.to/torrents.php?search={query}";
  2906.  
  2907. return {
  2908. url: {
  2909. all: rarbg,
  2910. movies: rarbg + "&category=14;48;17;44;45;42;46",
  2911. movies_1080: rarbg + "&category=44",
  2912. movies_720: rarbg + "&category=45",
  2913. movies_bluray: rarbg + "&category=42",
  2914. movies_remux: rarbg + "&category=46",
  2915. tv: rarbg + "&category=41",
  2916. docs: [rarbg + " documentary&category=movies", rarbg + "&category=18;41"],
  2917. music: rarbg + "&category=23;25",
  2918. music_flac: rarbg + "&category=25",
  2919. // music_mp3: rarbg + "&category=23",
  2920. elearning: rarbg + "&category=35",
  2921. ebooks: rarbg + "&category=35",
  2922. xxx: rarbg + "&category=4",
  2923. games_pc: rarbg + "&category=27;28",
  2924. apps_win: rarbg + "&category=33"
  2925. },
  2926. onParse: {
  2927. row: ".lista2t tr.lista2",
  2928. link_prepend: "https://rarbg.to",
  2929. sel: [
  2930. {text: "td:eq(1) a:eq(0)"},
  2931. {text: "> td:eq(4)"},
  2932. {
  2933. text: "> td:eq(3)",
  2934. link: function (context) {
  2935. var link = context[0].firstElementChild.nextElementSibling.firstElementChild;
  2936. var id = link.getAttribute("href").split("/").pop();
  2937. return '/download.php?id=' + id + '&f=' + encodeURIComponent(link.textContent.trim()) + '.torrent';
  2938. },
  2939. noblank: true
  2940. }
  2941. ]
  2942. },
  2943. onValidate: function (response) {
  2944. return response.finalUrl.indexOf('/bot_check.php') === -1 ? true : "captcha";
  2945. },
  2946. onFilter: bt.requireAllWords,
  2947. icon: ""
  2948. };
  2949. });
  2950. bt.addSource("Rutor", function () {
  2951. var rutor = "http://live-rutor.org/search/";
  2952.  
  2953. return {
  2954. url: {
  2955. all: rutor,
  2956. movies: [rutor + "0/1/100/0/", rutor + "0/5/100/0/", rutor + "0/7/100/0/"],
  2957. movies_1080: rutor + "0/0/100/0/1080p ",
  2958. movies_720: rutor + "0/0/100/0/720p ",
  2959. movies_remux: rutor + "0/0/300/0/remux|bdremux ",
  2960. movies_bluray: rutor + "0/0/310/0/bdinfo -bdremux -remux -hdtv -bdrip ",
  2961. movies_dvd: rutor + "0/0/300/0/dvd|dvd5|dvd9|dvdr ",
  2962. music: rutor + "0/2/100/0/",
  2963. music_flac: rutor + "0/2/100/0/FLAC ",
  2964. mvids: rutor + "0/2/300/0/-mp3 -flac -ape -aac -alac ",
  2965. tv: rutor + "0/0/300/0/720p|1080p ",
  2966. docs: [rutor + "0/12/100/0/", rutor + "0/6/100/0/"],
  2967. games_pc: rutor + "0/8/100/0/",
  2968. apps_win: rutor + "0/9/100/0/",
  2969. elearning: rutor,
  2970. ebooks: rutor + "0/0/300/0/pdf|epub|fb2|djvu|chm|mobi|doc ",
  2971. fiction: rutor + "0/0/300/0/pdf|epub|fb2|djvu|chm|mobi|doc ",
  2972. abooks: rutor + "0/11/300/0/flac|mp3 ",
  2973. comics: rutor + "0/0/300/0/CBZ|CBR ",
  2974. mags: rutor + "0/0/100/0/PDF "
  2975. },
  2976. onParse: {
  2977. row: "#index > table:eq(0) > tbody > tr:gt(0)",
  2978. link_prepend: "http://live-rutor.org",
  2979. sel: [
  2980. {text: "a[href*='/torrent/']:eq(0)"},
  2981. {text: "> td:last > span:eq(0)"},
  2982. {text: "> td:eq(-2)", link: "a[href*='/download/']:eq(0)", noblank: true}
  2983. ]
  2984. },
  2985. onFilter: [bt.filter3dMovies,bt.requireAllWords],
  2986. icon: ""
  2987. };
  2988. });
  2989. bt.addSource("RuTracker", function () {
  2990. var tru = "http://rutracker.org/forum/tracker.php?nm={query}";
  2991. 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";
  2992. 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";
  2993. var truFiction = "&f=2039,2041,2042,2043,2044,2045,2047,2080,2193,2355,2356,2357,2474";
  2994. 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";
  2995. 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";
  2996. 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";
  2997. 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";
  2998. var truComics = "&f=2461,2462,2463,2464,2465,2473,862";
  2999.  
  3000. return {
  3001. url: {
  3002. all: tru,
  3003. music: tru + " lossless|MP3|AAC|OGG|FLAC|APE|ALAC|WV",
  3004. music_flac: tru + " lossless|FLAC|APE|ALAC|WV",
  3005. // music_mp3: tru + "+MP3|AAC|OGG",
  3006. movies: tru + truMovies,
  3007. movies_1080: tru + " 1080p -remux -bdremux -disc" + truMovies,
  3008. movies_720: tru + " 720p -remux -bdremux -disc" + truMovies,
  3009. movies_remux: tru + " remux|bdremux" + truMovies,
  3010. movies_bluray: tru + " \"blu ray\" | bluray -DVD -DVD5 -DVD9 -rip -remux -bdremux" + truMovies,
  3011. movies_dvd: tru + " DVD|DVD5|DVD9" + truMovies,
  3012. tv: tru + " 720p|1080p",
  3013. mvids: tru + truMvids,
  3014. games_pc: tru + truGamesPC,
  3015. apps_win: tru + truAppsWin,
  3016. docs: tru + truDocs,
  3017. elearning: tru,
  3018. ebooks: tru + " PDF|EPUB|MOBI|CHM|DJVU|FB2",
  3019. abooks: tru + truAbooks,
  3020. mags: tru + " PDF",
  3021. comics: tru + truComics,
  3022. fiction: tru + truFiction
  3023. },
  3024. onEnable: function () {
  3025. $(bt.mainColumn).on('click', 'a[href^="http://dl.rutracker.org/"]', function () {
  3026. $('<form target="_blank" method="post" action="' + $(this).attr("href") + '"></form>')
  3027. .appendTo(document.body)
  3028. .submit()
  3029. .remove();
  3030. return false;
  3031. });
  3032. },
  3033. onParse: {
  3034. row: "#tor-tbl > tbody > tr:not(:has('td.pad_12'))",
  3035. link_prepend: "http://rutracker.org/forum/",
  3036. sel: [
  3037. {text: "a.tLink"},
  3038. {text: ".seedmed"},
  3039. {
  3040. text: function (context) {
  3041. var size = context.children().eq(-5)[0].firstElementChild.textContent;
  3042. return bt.humanizeSize(size);
  3043. },
  3044. link: "> td.tor-size > a",
  3045. noblank: true
  3046. }
  3047. ]
  3048. },
  3049. onFilter: bt.filter3dMovies,
  3050. icon: ""
  3051. };
  3052. });
  3053. bt.addSource("TL", function () {
  3054. var tl = "https://www.torrentleech.org/torrents/browse/index/query/{query}";
  3055.  
  3056. return {
  3057. url: {
  3058. all: tl,
  3059. movies: tl + "/categories/1,8,9,10,11,12,13,14,15,29,35",
  3060. movies_1080: tl + " 1080p/categories/13,14,35",
  3061. movies_720: tl + " 720p/categories/13,14,35",
  3062. 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",
  3063. movies_remux: tl + " (REMUX OR BDREMUX)/categories/13,14,35",
  3064. movies_dvd: tl + "/categories/12",
  3065. docs: tl + "/categories/29",
  3066. tv: tl + "/categories/27,32,35",
  3067. mvids: tl + "/categories/16",
  3068. elearning: tl + "/categories/5",
  3069. games_pc: tl + "/categories/17",
  3070. apps_win: tl + "/categories/6,23,33",
  3071. ebooks: tl + "/categories/5"
  3072. },
  3073. onParse: {
  3074. row: "#torrenttable > tbody > tr",
  3075. link_prepend: "https://www.torrentleech.org",
  3076. sel: [
  3077. {text: "a:eq(1)"},
  3078. {text: "> td:eq(6)"},
  3079. {text: "> td:eq(4)", link: "a[href*='/download/']:eq(0)", noblank: true}
  3080. ]
  3081. },
  3082. onValidate: function (response) {
  3083. return response.responseText.indexOf('/user/account/signup') === -1 ? true : "login needed";
  3084. },
  3085. onFilter: [bt.requireAllWords, bt.filter3dMovies, function (data, response) {
  3086. if (["elearning", "ebooks"].indexOf(response.context.category) !== -1) {
  3087. return data;
  3088. } else {
  3089. return bt.requireAllWords(data, response);
  3090. }
  3091. }],
  3092. icon: ""
  3093. };
  3094. });
  3095. bt.addSource("TPB", function () {
  3096. var q = "https://thepiratebay.org/search/{query}";
  3097.  
  3098. return {
  3099. url: {
  3100. all: q,
  3101. music: [ q + "/0/99/101", q + "/0/99/104" ],
  3102. music_flac: q + "/0/99/104",
  3103. movies: [q + "/0/99/201", q + "/0/99/202"],
  3104. movies_bluray: q + " avc/0/99/200",
  3105. movies_remux: q + " remux/0/99/200",
  3106. movies_1080: q + " 1080p/0/99/207",
  3107. movies_720: q + " 720p/0/99/207",
  3108. movies_dvd: q + "/0/99/202",
  3109. tv: [q + "/0/3/205", "/0/3/208"],
  3110. docs: q,
  3111. mvids: q + "/0/3/203",
  3112. apps_win: q + "/0/3/301",
  3113. games_pc: q + "/0/3/401",
  3114. elearning: q,
  3115. ebooks: q + "/0/99/601",
  3116. abooks: q + "/0/99/102",
  3117. fiction: q + "/0/99/601",
  3118. mags: q + "/0/3/601",
  3119. comics: q + "/0/3/602",
  3120. xxx: q + "/0/3/500"
  3121. },
  3122. onParse: {
  3123. row: "#searchResult > tbody > tr",
  3124. link_prepend: "https://thepiratebay.org",
  3125. sel: [
  3126. {text: "a[href*='/torrent/']:eq(0)"},
  3127. {text: "> td:eq(-2)"},
  3128. {
  3129. text: function(context){
  3130. var details = context.find(".detDesc").first();
  3131. if (details.length === 1 && ~details.text().indexOf(", Size ")) {
  3132. return details.text().split(", Size ")[1].split(",")[0].trim();
  3133. } else {
  3134. return $(context[0].lastElementChild.previousElementSibling.previousElementSibling);
  3135. }
  3136. },
  3137. link: "a[href^='magnet:']:eq(0)",
  3138. noblank: true
  3139. }
  3140. ]
  3141. },
  3142. onFilter: [bt.filter3dMovies, function (data, response) {
  3143. if (response.context.category === "movies_bluray") {
  3144. return data.filter(function(){
  3145. return this.textContent.toLowerCase().indexOf("remux") === -1;
  3146. });
  3147. } else {
  3148. return data;
  3149. }
  3150. }],
  3151. icon: ""
  3152. };
  3153. });
  3154. bt.addSource("BlueBird", function () {
  3155. var bird = "http://bluebird-hd.org/browse.php?incldead=1&stype=and&search={query}";
  3156.  
  3157. return {
  3158. url: {
  3159. all: bird,
  3160. movies: bird + "&c1=1&c2=1",
  3161. movies_1080: bird + "&c1=1&c2=1&cr3=1&cr5=1",
  3162. movies_720: bird + "&c1=1&c2=1&cr4=1&cr6=1",
  3163. movies_remux: bird + "&c1=1&c2=1&cr2=1",
  3164. movies_bluray: bird + "&c1=1&c2=1&cr1=1",
  3165. music: bird + "&cr7=1",
  3166. music_flac: bird + "&cr7=1",
  3167. tv: bird + "&c6=1",
  3168. mvids: bird + "&c4=1",
  3169. docs: bird + "&c3=1",
  3170. xxx: bird + "&c7=1"
  3171. },
  3172. onParse: {
  3173. cleanup: ["a[href*='&snatched']"],
  3174. row: "#highlighted > tr",
  3175. link_prepend: "http://bluebird-hd.org/",
  3176. sel: [
  3177. {text: "a[href*='details.php']:eq(0)"},
  3178. {text: "> td:eq(-3)"},
  3179. {text: "> td:last", link: "a[href*='download.php']:eq(0)", noblank: true}
  3180. ]
  3181. },
  3182. onPrepareQuery: function (context) {
  3183. var i,
  3184. len = context.url.length,
  3185. year = bt.extractYear(context, true);
  3186.  
  3187. if (year && year.length === 4) {
  3188. for (i = 0; i < len; i++) {
  3189. context.url[i] += "&dsearch=" + year;
  3190. }
  3191. }
  3192.  
  3193. var res = bt.extractResolution(context);
  3194.  
  3195. if (res === "720p") {
  3196. for (i = 0; i < len; i++) {
  3197. context.url[i] += "&cr4=1&cr6=1";
  3198. }
  3199. } else if (res === "1080p" || res === "1080i") {
  3200. for (i = 0; i < len; i++) {
  3201. context.url[i] += "&cr3=1&cr5=1";
  3202. }
  3203. }
  3204. },
  3205. onFilter: bt.filter3dMovies,
  3206. icon: ""
  3207. };
  3208. });
  3209. bt.addSource("MoreThanTV", function () {
  3210. var q = "https://www.morethan.tv/ajax.php?action=browse&order_by=time&searchsubmit=1&searchstr={query}";
  3211.  
  3212. return {
  3213. url: {
  3214. all: q,
  3215. tv: q + "&filter_cat[2]=1",
  3216. docs: q + "&taglist=documentary"
  3217. },
  3218. onPrepareQuery: function(context){
  3219. var i,
  3220. len = context.url.length,
  3221. res = bt.extractResolution(context);
  3222.  
  3223. if (res) {
  3224. for (i = 0; i < len; i++) {
  3225. context.url[i] += "&encoding=" + res;
  3226. }
  3227. }
  3228.  
  3229. var year = bt.extractYear(context);
  3230.  
  3231. if (year) {
  3232. for (i = 0; i < len; i++) {
  3233. context.url[i] += "&filelist=" + year;
  3234. }
  3235. }
  3236. },
  3237. onParse: function (response) {
  3238. try {
  3239. var data = JSON.parse(response.responseText);
  3240. } catch (e) {
  3241. bt.showFailAlert(response);
  3242. return null;
  3243. }
  3244.  
  3245. if (!('status' in data) || data.status !== 'success' || !('response' in data) || !('results' in data.response)) {
  3246. bt.showFailAlert(response, "unexpected data");
  3247. return null;
  3248. }
  3249.  
  3250. response.context.searchUrl = response.finalUrl.replace("ajax.php?action=browse", "torrents.php?action=basic");
  3251.  
  3252. return data.response.results;
  3253. },
  3254. onRender: function (torrents, table) {
  3255. var title, rows = "";
  3256.  
  3257. for (var i = 0, l = torrents.length; i < l; i++) {
  3258. title = '<a href="https://www.morethan.tv/torrents.php?id=' + torrents[i].groupId + '&torrentid=' + torrents[i].torrentId + '" target="_blank">' + torrents[i].groupName + '</a>';
  3259.  
  3260. if (torrents[i].fileCount > 1) {
  3261. title += ' (' + torrents[i].fileCount + ' files)';
  3262. }
  3263.  
  3264. if (torrents[i].isFreeleech) {
  3265. title += ' <span class="label label-success">Freeleech</span>';
  3266. }
  3267.  
  3268. 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>';
  3269. }
  3270.  
  3271. table.html(rows);
  3272. },
  3273. icon: ""
  3274. };
  3275. });
  3276. bt.addSource("AR", function () {
  3277. var ar = "https://alpharatio.cc/torrents.php?action=basic&searchstr={query}";
  3278.  
  3279. return {
  3280. url: {
  3281. all: ar,
  3282. movies: ar + "&filter_cat[6]=1&filter_cat[7]=1&filter_cat[8]=1&filter_cat[9]=1",
  3283. movies_1080: ar + " 1080p&filter_cat[7]=1&filter_cat[9]=1",
  3284. movies_720: ar + " 720p&filter_cat[7]=1&filter_cat[9]=1",
  3285. movies_bluray: ar + " COMPLETE&filter_cat[7]=1",
  3286. movies_remux: ar + " REMUX&filter_cat[7]=1",
  3287. 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",
  3288. tv: ar + "&filter_cat[2]=1&filter_cat[3]=1&filter_cat[5]=1",
  3289. mvids: ar + "&filter_cat[11]=1",
  3290. elearning: ar + "&filter_cat[21]=1&filter_cat[22]=1&filter_cat[24]=1",
  3291. ebooks: ar + "&filter_cat[21]=1",
  3292. abooks: ar + "&filter_cat[22]=1",
  3293. games_pc: ar + "&filter_cat[12]=1",
  3294. apps_win: ar + "&filter_cat[16]=1",
  3295. music: ar + "&filter_cat[23]=1",
  3296. music_flac: ar + " FLAC&filter_cat[23]=1",
  3297. xxx: ar + "&filter_cat[10]=1&filter_cat[20]=1"
  3298. },
  3299. onParse: {
  3300. row: "#torrent_table tr.torrent",
  3301. link_prepend: "https://alpharatio.cc/",
  3302. sel: [
  3303. {text: ".group_info > a:eq(0)"},
  3304. {text: "> td:eq(7)"},
  3305. {text: "> td:eq(5)", link: "a[href*='action=download']:eq(0)", noblank: true}
  3306. ]
  3307. },
  3308. onFilter: bt.filter3dMovies,
  3309. icon: ""
  3310. };
  3311. });
  3312. bt.renderPage();