BT MetaSearch

Searches across multiple sources at once.

目前为 2015-08-28 提交的版本。查看 最新版本

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