// ==UserScript==
// @name BT4G & Limetorrents enhanced search
// @description Adds magnet links to BT4G and Limetorrents, filtering of search results by minimum and maximum size (BT4G only), keeping search terms in the input field in case of missing results (BT4G only), automatic reload in case of server errors every 5 minutes
// @version 20241101
// @author mykarean
// @match *://bt4gprx.com/*
// @match *://*.limetorrents.lol/search/all/*
// @run-at document-idle
// @grant GM_xmlhttpRequest
// @grant GM_addStyle
// @grant GM_setValue
// @grant GM_getValue
// @compatible chrome
// @license GPL3
// @noframes
// @icon 
// @namespace https://greasyfork.org/users/1367334
// ==/UserScript==
"use strict";
// ---------------------------------------------------------
// Config/Requirements
// ---------------------------------------------------------
const currentPath = window.location.href;
const hostname = location.hostname;
let magnetImage = GM_info.script.icon;
let itemsFoundElement;
if (hostname === "bt4gprx.com") {
itemsFoundElement = document.querySelector("body > main > p");
} else if (hostname === "www.limetorrents.lol") {
itemsFoundElement = document.querySelector("#content > h2");
}
/**
* @param {String} tag Elements HTML Tag
* @param {String|RegExp} regex Regular expression or string for text search
* @param {Number} index Item Index
* @returns {Object|null} Node or null if not found
*/
function getElementByText(tag, regex, item = 0) {
if (typeof regex === "string") {
regex = new RegExp(regex);
}
const elements = document.getElementsByTagName(tag);
let count = 0;
for (let i = 0; i < elements.length; i++) {
if (regex.test(elements[i].textContent)) {
if (count === item) {
return elements[i];
}
count++;
}
}
return null;
}
// ---------------------------------------------------------
// Layout
// ---------------------------------------------------------
function addCss() {
GM_addStyle(`
.magnet-link-img {
cursor: pointer;
margin: 0px 5px 2px;
vertical-align: bottom;
height: 20px;
transition: filter 0.2s ease;
}
`);
if (hostname === "bt4gprx.com") {
GM_addStyle(`
.lead {
display: inline-block;
}
/* removing the annoying hover effect on search results */
.result-item:hover,
.list-group-item:hover {
transform: none;
}
`);
}
}
// ---------------------------------------------------------
// search results handling
// ---------------------------------------------------------
function observeSearchResultsCssChange() {
const observer = new MutationObserver(() => {
observer.disconnect();
setTimeout(() => {
processLinksInSearchResults().then(() => {
observeSearchResultsCssChange();
});
}, 100);
});
observer.observe(document, {
childList: true,
subtree: true,
attributes: true,
attributeFilter: ["style"],
});
}
function observeNewSearchResults() {
const observer = new MutationObserver((mutations) => {
requestAnimationFrame(() => {
let hasNewResults = false;
for (const mutation of mutations) {
if (mutation.addedNodes.length > 0) {
const newSearchResultWithSize = Array.from(mutation.addedNodes).some((node) => {
if (node.querySelector) {
return !!node.querySelector(".cpill");
}
return false;
});
if (newSearchResultWithSize) {
hasNewResults = true;
break;
}
}
}
if (hasNewResults) {
itemFilterBySize();
}
});
});
observer.observe(document.body, {
childList: true,
subtree: true,
attributes: false, // Ignore style changes from itemFilterBySize()
characterData: false, // Ignore text changes from processLinksInSearchResults()
});
}
async function processLinksInSearchResults() {
const links = Array.from(getSearchResultLinks());
const promises = links.map(async (link) => {
if (hostname === "bt4gprx.com") {
await processLinksInSearchResultsBt4g(link);
} else if (hostname === "www.limetorrents.lol") {
processLinksInSearchResultsLimeTorrents(link);
}
});
await Promise.all(promises);
if (hostname.includes("bt1207")) {
for (let link of links) {
await processLinksInSearchResultsBt1207(link);
// await new Promise((resolve) => setTimeout(resolve, 100));
// add magnets on hover
link.addEventListener("mouseover", async function () {
const magnetLink = link.getAttribute("data-magnet-added");
if (magnetLink !== "true") {
try {
// Try to process the link
await processLinksInSearchResultsBt1207(link);
// Only set the attribute if no error occurs
// link.setAttribute("data-magnet-added", "true");
} catch (error) {
console.error("Error processing link:", error);
// Handle error (e.g., log it, but don't set data-magnet-added)
}
}
});
}
}
// Add amount of visible magnet links into text
const amountVisibleMagnets = links.length;
const magnetLinkAllSpan = document.querySelector(".magnet-link-all-span");
if (links && typeof links.length === "number" && magnetLinkAllSpan) {
magnetLinkAllSpan.innerHTML = `Open all <span class="badge bg-primary">${amountVisibleMagnets}</span> loaded magnet links`;
}
// Remove spam elements
setTimeout(() => {
links.forEach((link) => {
const title = link.title;
if (title.includes("Downloader.exe") || title.includes("Downloader.dmg")) {
link.parentElement.parentElement.remove();
}
});
}, 100);
}
function getSearchResultLinks() {
if (hostname === "bt4gprx.com") {
const elements = document.querySelectorAll('a[href*="/magnet/"]:not([href^="magnet:"])');
// Filter and return only the visible elements (those without 'display: none' in their parent chain)
return Array.from(elements).filter((element) => {
let current = element;
while (current) {
if (window.getComputedStyle(current).display === "none") {
return false;
}
current = current.parentElement;
}
return true;
});
} else if (hostname === "www.limetorrents.lol") {
return document.querySelectorAll('a[href*="//itorrents.org/torrent/"]');
} else if (hostname.includes("bt1207")) {
return document.querySelectorAll("body > div.container-fluid > div:nth-child(6) > div.col-md-6 > ul > li:nth-child(1) > a");
}
}
async function processLinksInSearchResultsBt4g(link) {
try {
const details = {
method: "GET",
url: link.href,
timeout: 5000,
};
const response = await requestGM_XHR(details);
const html = response.responseText;
// Find magnet links
const parser = new DOMParser();
const doc = parser.parseFromString(html, "text/html");
// Skip if magnet link exists
const magnetLink = link.getAttribute("data-magnet-added");
if (magnetLink === "true") {
return;
}
const downloadLink = doc.querySelector('a[href^="//downloadtorrentfile.com/hash/"]');
if (downloadLink) {
const hash = extractHashFromUrl(downloadLink.href.split("/").pop().split("?")[0]);
if (hash) {
insertMagnetLink(link, hash);
link.setAttribute("data-magnet-added", "true");
}
}
} catch (error) {
console.error("Error getting magnet link:", error);
}
}
async function processLinksInSearchResultsBt1207(link) {
try {
const details = {
method: "GET",
url: link.href,
timeout: 3000,
headers: {
Referer: document.referrer,
Cookie: document.cookie,
"User-Agent": navigator.userAgent,
Referer: window.location.href,
Accept: "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8",
"Accept-Language": navigator.language || navigator.userLanguage,
"Accept-Encoding": "gzip, deflate, br",
Connection: "keep-alive",
},
};
const response = await requestGM_XHR(details);
const html = response.responseText;
// Find magnet links
const parser = new DOMParser();
const doc = parser.parseFromString(html, "text/html");
// Skip if magnet link exists
const magnetLink = link.getAttribute("data-magnet-added");
if (magnetLink === "true") {
return;
}
const downloadLink = doc.querySelector("#magnet");
if (downloadLink) {
const hash = extractHashFromUrl(downloadLink.href);
if (hash) {
insertMagnetLink(link, hash);
link.setAttribute("data-magnet-added", "true");
}
}
} catch (error) {
console.error("Error getting magnet link:", error);
}
}
function requestGM_XHR(details) {
return new Promise((resolve, reject) => {
details.onload = function (response) {
resolve(response);
};
details.onerror = function (response) {
reject(response);
};
details.ontimeout = function () {
reject(new Error("Request timed out"));
};
GM_xmlhttpRequest(details);
});
}
function processLinksInSearchResultsLimeTorrents(link) {
// Skip if magnet link exists
const magnetLink = link.getAttribute("data-magnet-added");
if (magnetLink === "true") {
return;
}
const hash = extractHashFromUrl(link.href.split("/").pop().split("?")[0]);
if (hash) {
insertMagnetLink(link, hash);
link.setAttribute("data-magnet-added", "true");
// Hide unnecessary element
link.style.display = "none";
}
}
function insertMagnetLink(link, hash) {
const magnetLink = `magnet:?xt=urn:btih:${hash}`;
const newLink = document.createElement("a");
newLink.classList.add("magnet-link");
newLink.href = magnetLink;
newLink.addEventListener("click", function () {
imgElement.style.filter = "grayscale(100%) opacity(0.7)";
});
const imgElement = document.createElement("img");
imgElement.src = magnetImage;
imgElement.classList.add("magnet-link-img");
newLink.appendChild(imgElement);
link.parentNode.insertBefore(newLink, link);
}
function extractHashFromUrl(href) {
const hashRegex = /(^|\/|&|-|\.|\?|=|:)([a-fA-F0-9]{40})/;
const matches = href.match(hashRegex);
return matches ? matches[2] : null;
}
// ---------------------------------------------------------
function addClickAllMagnetLinks() {
// only needed if document-start
// const openAllMagnetLinks = document.querySelector(".magnet-link-all-span");
// if (openAllMagnetLinks) {
// return;
// }
// no elements found
if (
document.querySelector("body > main > p")?.textContent.includes("did not match any documents") ||
document.querySelector("#content > h2:nth-child(9)")
)
return;
const targetElement = itemsFoundElement?.parentElement?.children[3];
if (targetElement) {
const openAllMagnetLinksSpan = document.createElement("span");
openAllMagnetLinksSpan.innerHTML = "Open all <span class='badge bg-primary'>0</span> loaded magnet links";
openAllMagnetLinksSpan.classList.add("magnet-link-all-span", "lead");
openAllMagnetLinksSpan.style.marginLeft = "10px";
const openAllMagnetLinksImg = document.createElement("img");
openAllMagnetLinksImg.src = magnetImage;
openAllMagnetLinksImg.classList.add("magnet-link-img");
openAllMagnetLinksImg.style.cssText = "cursor:pointer;vertical-align:sub;";
targetElement.insertAdjacentElement("afterend", openAllMagnetLinksSpan);
openAllMagnetLinksSpan.insertAdjacentElement("afterend", openAllMagnetLinksImg);
openAllMagnetLinksImg.addEventListener("click", () => {
const addedMagnetLinks = document.querySelectorAll("a.magnet-link");
if (addedMagnetLinks.length > 0) {
openAllMagnetLinksImg.style.filter = "grayscale(100%) opacity(0.7)";
addedMagnetLinks.forEach((link, index) => {
// ignore hidden elements
if (getComputedStyle(link.parentElement.parentElement).display !== "none") {
setTimeout(() => {
link.click();
}, index * 100);
}
});
} else {
openAllMagnetLinksSpan.textContent = "No magnet links found";
}
});
// for a fixed position and more space, remove superfluous information
if (hostname === "bt4gprx.com") {
itemsFoundElement.innerHTML = itemsFoundElement.innerHTML.replace(/(\ items)\ for\ .*/, "$1");
} else if (hostname === "www.limetorrents.lol") {
itemsFoundElement.textContent = "";
}
}
}
// ---------------------------------------------------------
// size filter
// ---------------------------------------------------------
function itemFilterBySize() {
if (hostname !== "bt4gprx.com") return;
// no elements found
if (document.querySelector("body > main > p")?.textContent.includes("did not match any documents")) return;
if (!document.getElementById("item-filter-styles")) {
GM_addStyle(`
.filter-container {
display: inline-flex;
align-items: center;
}
.filter-button {
color: #212121;
padding: 3px 7px;
border: none;
margin-right: 5px;
margin-left: 10px;
}
.filter-input {
margin-left: 5px !important;
padding-left: 12px !important;
width: 50px !important;
text-align: center !important;
}
.filter-label {
font-weight: bold;
margin-left: 5px;
}
.hidden-item {
display: none !important;
}
`).setAttribute("id", "item-filter-styles");
}
function createFilterControl(id, text) {
const button = document.createElement("button");
button.id = id;
button.className = "filter-button btn";
button.textContent = text;
const input = document.createElement("input");
input.type = "number";
input.step = "1";
input.className = "filter-input";
input.id = `${id}-input`;
return { button, input };
}
function createFilterControls(buttonTarget) {
const container = document.createElement("span");
container.className = "filter-container";
const minFilter = createFilterControl("filter-min-size-button", "Min");
const maxFilter = createFilterControl("filter-max-size-button", "Max");
const unitLabel = document.createElement("span");
unitLabel.textContent = "GB";
unitLabel.className = "filter-label";
container.append(minFilter.button, minFilter.input, maxFilter.button, maxFilter.input, unitLabel);
buttonTarget.parentNode.insertBefore(container, buttonTarget.nextSibling);
return {
minButton: minFilter.button,
maxButton: maxFilter.button,
minInput: minFilter.input,
maxInput: maxFilter.input,
};
}
async function setupFilter(button, input, isMinFilter) {
const filterType = isMinFilter ? "Min" : "Max";
let isFiltered = await GM.getValue(`is${filterType}Filtered`, false);
let threshold = await GM.getValue(`${filterType.toLowerCase()}FilterThreshold`, isMinFilter ? 1 : 10);
button.textContent = isFiltered ? `${filterType} filter on` : `${filterType} filter off`;
button.style.backgroundColor = isFiltered ? "#b2dfdb" : "#dfb2b2";
input.value = threshold;
button.addEventListener("click", async () => {
isFiltered = !isFiltered;
await GM.setValue(`is${filterType}Filtered`, isFiltered);
button.textContent = isFiltered ? `${filterType} filter on` : `${filterType} filter off`;
button.style.backgroundColor = isFiltered ? "#b2dfdb" : "#dfb2b2";
await applyItemFilterBySize();
});
input.addEventListener("input", async () => {
threshold = parseFloat(input.value);
await GM.setValue(`${filterType.toLowerCase()}FilterThreshold`, threshold);
if (isFiltered) {
await applyItemFilterBySize();
}
});
}
async function initializeFilter() {
const buttonTarget = itemsFoundElement;
if (!buttonTarget) return;
const { minButton, maxButton, minInput, maxInput } = createFilterControls(buttonTarget);
await setupFilter(minButton, minInput, true);
await setupFilter(maxButton, maxInput, false);
await applyItemFilterBySize();
}
async function applyItemFilterBySize() {
const minFiltered = await GM.getValue("isMinFiltered", false);
const maxFiltered = await GM.getValue("isMaxFiltered", false);
const minThreshold = await GM.getValue("minFilterThreshold", 1);
const maxThreshold = await GM.getValue("maxFilterThreshold", 10);
document.querySelectorAll("b.cpill").forEach((element) => {
const size = parseFloat(element.innerText);
const parentElement = element.parentElement.parentElement.parentElement;
if (parentElement && size) {
const itemBelowGb = !element.className.includes("red-pill");
const hideMin = minFiltered && (size < minThreshold || itemBelowGb);
const hideMax = maxFiltered && size > maxThreshold && !itemBelowGb;
if (hideMin || hideMax) {
parentElement.classList.add("hidden-item");
} else {
parentElement.classList.remove("hidden-item");
}
}
});
}
// Check if toggle buttons already exist
const existingMinButton = document.getElementById("filter-min-size-button");
const existingMaxButton = document.getElementById("filter-max-size-button");
if (!existingMinButton && !existingMaxButton) {
initializeFilter();
} else {
applyItemFilterBySize();
}
}
async function main() {
const ERROR_TITLES = ["Web server is returning an unknown error", "525: SSL handshake failed"];
if (ERROR_TITLES.some((error) => document.title.includes(error))) {
const RELOAD_DELAY = 5 * 60 * 1000;
console.log("Web server error detected. Waiting 5 minutes before reloading...");
return setTimeout(() => location.reload(), RELOAD_DELAY);
}
addCss();
// handle search results
if (/\/search/.test(currentPath)) {
itemFilterBySize();
addClickAllMagnetLinks();
await processLinksInSearchResults();
observeSearchResultsCssChange();
observeNewSearchResults();
} else if (/\/magnet/.test(currentPath)) {
// BT4G only: torrent detail page
const link = document.querySelector('a[href*="/hash/"]:not([href^="magnet:"])');
const hash = extractHashFromUrl(link?.href || "");
if (hash) {
insertMagnetLink(link, hash);
}
}
}
main();