BT MetaSearch

Searches across multiple sources at once.

目前为 2018-07-20 提交的版本。查看 最新版本

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