BT MetaSearch

Searches across multiple sources at once.

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

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