Magnet Links for BT4G and Limetorrents

Adds magnet links to BT4G and Limetorrents

// ==UserScript==
// @name           Magnet Links for BT4G and Limetorrents
// @version        20240622c
// @author         reaseno
// @description    Adds magnet links to BT4G and Limetorrents
// @homepage       https://greasyfork.org/en/scripts/469976/
// @namespace      https://greasyfork.org/users/781194
// @match          *://bt4gprx.com/*
// @match          *://www.limetorrents.lol/search/all/*
// @run-at         document-idle
// @compatible     chrome
// @license        GPL3
// @noframes
// @icon           
// ==/UserScript==

"use strict";

const style = document.createElement("style");
style.textContent = `
    .magnet-link-img {
        cursor: pointer;
        margin: 0px 5px 2px;
        border-radius: 50%;
        vertical-align: bottom;
        height: 20px;
        transition: filter 0.2s ease;
    }
`;
document.head.appendChild(style);

const hostname = location.hostname;
let magnetImage = GM_info.script.icon;

function getSearchResultLinks() {
    if (hostname === "bt4gprx.com") {
        return document.querySelectorAll('a[href*="/magnet/"]:not([href^="magnet:"])');
    } else if (hostname === "www.limetorrents.lol") {
        return document.querySelectorAll('a[href*="//itorrents.org/torrent/"]');
    }
}

/**
 * @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;
}

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;
}

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") {
            await processLinksInSearchResultsLimeTorrents(link);
        }
    });

    await Promise.all(promises);

    // Add amount of magnet links into text
    const numLinks = links.length;
    const magnetLinkAllSpan = document.querySelector(".magnet-link-all-span");
    if (numLinks > 0 && magnetLinkAllSpan) {
        magnetLinkAllSpan.innerHTML = `Open all <b>${numLinks}</b> 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.style.display = "none";
            }
        });
    }, 100);
}

async function processLinksInSearchResultsBt4g(link) {
    try {
        const response = await fetch(link.href);
        const html = await response.text();

        // 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);
    }
}

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 addClickAllMagnetLinks() {
    // only needed if document-start
    // const openAllMagnetLinks = document.querySelector(".magnet-link-all-span");
    // if (openAllMagnetLinks) {
    //     return;
    // }

    let itemsFoundElement;
    if (hostname === "bt4gprx.com") {
        itemsFoundElement = getElementByText("span", /Found\ [0-9].*\ items\ for\ .*/i);
    } else if (hostname === "www.limetorrents.lol") {
        itemsFoundElement = getElementByText("h2", "Search Results");
    }

    const targetElement = itemsFoundElement?.parentElement?.children[1];
    if (targetElement) {
        const openAllMagnetLinksSpan = document.createElement("span");
        openAllMagnetLinksSpan.innerHTML = "Open all <b>0</b> loaded magnet links";
        openAllMagnetLinksSpan.classList.add("magnet-link-all-span");
        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.parentNode.insertBefore(openAllMagnetLinksSpan, targetElement.nextSibling);
        openAllMagnetLinksSpan.parentNode.insertBefore(openAllMagnetLinksImg, openAllMagnetLinksSpan.nextSibling);

        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) => {
                    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 = "";
        }
    }
}

function observeSearchResults() {
    const observer = new MutationObserver(() => {
        observer.disconnect();
        setTimeout(() => {
            processLinksInSearchResults().then(() => {
                observeSearchResults();
            });
        }, 500);
    });

    observer.observe(document, {
        childList: true,
        subtree: true,
    });
}

function main() {
    // search results page
    if (window.location.href.match(/\/search/)) {
        addClickAllMagnetLinks();
        observeSearchResults();
        processLinksInSearchResults();
    }

    // BT4G only: torrent detail page
    if (window.location.href.match(/\/magnet/)) {
        const link = document.querySelector('a[href*="/hash/"]:not([href^="magnet:"])');
        const hash = extractHashFromUrl(link.href || "");
        if (hash) {
            insertMagnetLink(link, hash);
        }
    }
}

main();