BT MetaSearch

Searches across multiple sources at once.

目前為 2015-08-30 提交的版本,檢視 最新版本

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