您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Automatically downloads torrents within your size and count limits (perfect for 2x seedbonus). Configure min/max size, download limit, and total size. Click "Download Start" on the search page and keep the tab open.
// ==UserScript== // @name TorrentBD-Torrent Downloader // @namespace Violentmonkey Scripts // @match https://www.torrentbd.net/* // @match https://www.torrentbd.com/* // @match https://www.torrentbd.org/* // @icon  // @version 2.0 // @run-at document-end // @author TheMyth // @grant GM_registerMenuCommand // @license MIT // @description Automatically downloads torrents within your size and count limits (perfect for 2x seedbonus). Configure min/max size, download limit, and total size. Click "Download Start" on the search page and keep the tab open. // ==/UserScript== // Default configuration const defaultConfig = { max: [200,"mb"], min: [100, "mb"], downloadLimit: 200, sumDownloadSize: null, }; // Function to get current user config function getUserConfig() { const savedConfig = localStorage.getItem("torrentBDConfig"); if (savedConfig) { try { return JSON.parse(savedConfig); } catch (e) { console.error("Failed to parse saved config:", e); return defaultConfig; } } return defaultConfig; } // Function to process config into working format function processConfig(userConfig) { const t_c = {}; function converter(x) { if (["kib", "kb"].includes(String(x[1]).toLowerCase())) { return parseFloat(x[0]); } else if (["mib", "mb"].includes(String(x[1]).toLowerCase())) { return parseFloat(x[0]) * 1024; } else if (["gib", "gb"].includes(String(x[1]).toLowerCase())) { return parseFloat(x[0]) * 1024 * 1024; } else if (["tib", "tb"].includes(String(x[1]).toLowerCase())) { return parseFloat(x[0]) * 1024 * 1024 * 1024; } else { return null; } } if (userConfig.max) { t_c.max = converter(userConfig.max); } else { t_c.max = null; } if (userConfig.min) { t_c.min = converter(userConfig.min); } else { t_c.min = null; } if (userConfig.sumDownloadSize) { t_c.sumDownloadSize = converter(userConfig.sumDownloadSize); } else { t_c.sumDownloadSize = null; } t_c.downloadLimit = userConfig.downloadLimit; return t_c; } // Initialize config let Config = processConfig(getUserConfig()); let sumDownlodSize = 0.0; let DownloadedTorrents = 0; console.log(`User Config:`, Config); console.log("Last History:",getLocal()); function sleep(ms) { return new Promise((resolve) => setTimeout(resolve, ms)); } function setLocal(object) { localStorage.setItem("endof", JSON.stringify(object)); alert(`Download Done!\nTotalDownloads:${object.DownloadedTorrents}\nLatTorrentName:${object.lastTorrentName}\nTotalFileSize:${object.sumDownlodSize}`) } function getLocal() { if (localStorage.getItem("endof")) { return JSON.parse(localStorage.getItem("endof")); } else { return null; } } /** * * @param {HTMLElement} element */ function getTheFileSize(element) { let el_TextContent = element .querySelector('div[title="File Size"') .textContent.split(" "); if (el_TextContent[2].toLowerCase() == "kib") { return parseFloat(el_TextContent[1]); } else if (el_TextContent[2].toLowerCase() == "mib") { return parseFloat(el_TextContent[1]) * 1024; } else if (el_TextContent[2].toLowerCase() == "gib") { return parseFloat(el_TextContent[1]) * 1024 * 1024; } else if (el_TextContent[2].toLowerCase() == "tib") { return parseFloat(el_TextContent[1]) * 1024 * 1024 * 1024; } } async function getTorrentData() { // Get fresh config at the start of download const currentUserConfig = getUserConfig(); Config = processConfig(currentUserConfig); console.log('Using current config:', Config); while (true) { /** * * @param {HTMLElement} element * @param {Float} fileSize */ async function downloadHelper(element, fileSize) { let title = element.querySelector("td:nth-child(2) a"); let dwnLink = element.querySelector("td:nth-child(3) a"); sumDownlodSize += fileSize; DownloadedTorrents += 1; dwnLink.click(); console.log(`${DownloadedTorrents}:${title.textContent}`); } let tableBody = document.querySelector( "#kuddus-results-container > table > tbody" ); let lastTorrentName = ""; for (let index = 0; index < tableBody.children.length; index++) { const element = tableBody.children[index]; let fileSize = getTheFileSize(element); if (Config.max && Config.min) { if (fileSize >= Config.min && fileSize <= Config.max) { downloadHelper(element, fileSize); await sleep(2000); } } else if (Config.max && fileSize <= Config.max) { downloadHelper(element, fileSize); await sleep(2000); } else if (Config.min && fileSize >= Config.min) { downloadHelper(element, fileSize); await sleep(2000); } if (Config.downloadLimit && Config.downloadLimit == DownloadedTorrents) { lastTorrentName = element.querySelector("td:nth-child(2) a").textContent; break; } else if ( Config.sumDownloadSize && sumDownlodSize >= Config.sumDownloadSize ) { lastTorrentName = element.querySelector("td:nth-child(2) a").textContent; break; } else if ( !Config.sumDownloadSize && !Config.downloadLimit && DownloadedTorrents == 100 ) { lastTorrentName = element.querySelector("td:nth-child(2) a").textContent; break; } } console.log(sumDownlodSize); console.log(DownloadedTorrents); let next = document.querySelector('li[title="Next page"]'); if (Config.downloadLimit && Config.downloadLimit == DownloadedTorrents) { setLocal({ sumDownlodSize: sumDownlodSize, DownloadedTorrents: DownloadedTorrents, page: parseInt(next.getAttribute("data-paginate-to")) - 1, lastTorrentName: lastTorrentName, }); return } else if ( Config.sumDownloadSize && sumDownlodSize >= Config.sumDownloadSize ) { setLocal({ sumDownlodSize: sumDownlodSize, DownloadedTorrents: DownloadedTorrents, page: parseInt(next.getAttribute("data-paginate-to")) - 1, lastTorrentName: lastTorrentName, }); return } else if ( !Config.sumDownloadSize && !Config.downloadLimit && DownloadedTorrents == 100 ) { setLocal({ sumDownlodSize: sumDownlodSize, DownloadedTorrents: DownloadedTorrents, page: parseInt(next.getAttribute("data-paginate-to")) - 1, lastTorrentName: lastTorrentName, }); return } if (!next) { return } else { next.click(); await sleep(2000); } } } async function dwnBtnFunc() { let resultContainer = document.querySelector("#kuddus-results-container"); if (resultContainer.childNodes.length == 0) { alert("No Torrent Found on th page"); } else if (resultContainer.childNodes.length > 0) { sumDownlodSize = 0.0; DownloadedTorrents = 0; getTorrentData(); } } // Create configuration UI function createConfigUI(panel) { const configPanel = document.createElement("div"); configPanel.id = "torrentbd-config-panel"; configPanel.style = "position: absolute; background: white; padding: 10px; border-radius: 5px; box-shadow: 0 0 10px rgba(0,0,0,0.2); z-index: 9999; display: none; color: black; font-size: 12px; width: 300px; top: 5%;left:30%;"; // Get current user config for the UI const currentUserConfig = getUserConfig(); configPanel.innerHTML = ` <h3 style="margin: 0 0 5px 0; font-size: 14px;">TorrentBD Settings</h3> <div style="display: grid; grid-template-columns: auto 60px 50px 55px; gap: 5px; align-items: center;"> <label title="Maximum size of a single torrent file">Max Size:</label> <input type="number" id="config-max-value" value="${currentUserConfig.max ? currentUserConfig.max[0] : ''}" style="width: 100%;" /> <select id="config-max-unit" style="width: 100%;"> ${['kb', 'mb', 'gb', 'tb'].map(unit => `<option value="${unit}" ${currentUserConfig.max && currentUserConfig.max[1].toLowerCase() === unit ? 'selected' : ''}>${unit}</option>`).join('')} </select> <div> <input type="checkbox" id="config-max-enabled" ${currentUserConfig.max ? 'checked' : ''} /> <label for="config-max-enabled"></label> </div> <label title="Minimum size of a single torrent file">Min Size:</label> <input type="number" id="config-min-value" value="${currentUserConfig.min ? currentUserConfig.min[0] : ''}" style="width: 100%;" /> <select id="config-min-unit" style="width: 100%;"> ${['kb', 'mb', 'gb', 'tb'].map(unit => `<option value="${unit}" ${currentUserConfig.min && currentUserConfig.min[1].toLowerCase() === unit ? 'selected' : ''}>${unit}</option>`).join('')} </select> <div> <input type="checkbox" id="config-min-enabled" ${currentUserConfig.min ? 'checked' : ''} /> <label for="config-min-enabled"></label> </div> <label title="Number of torrents to download">Download Limit:</label> <input type="number" id="config-limit-value" value="${currentUserConfig.downloadLimit || ''}" style="width: 100%;" /> <span></span> <div> <input type="checkbox" id="config-limit-enabled" ${currentUserConfig.downloadLimit ? 'checked' : ''} /> <label for="config-limit-enabled"></label> </div> <label title="Total size of all downloaded torrents">Sum Size:</label> <input type="number" id="config-sum-value" value="${currentUserConfig.sumDownloadSize ? currentUserConfig.sumDownloadSize[0] : ''}" style="width: 100%;" /> <select id="config-sum-unit" style="width: 100%;"> ${['kb', 'mb', 'gb', 'tb'].map(unit => `<option value="${unit}" ${currentUserConfig.sumDownloadSize && currentUserConfig.sumDownloadSize[1].toLowerCase() === unit ? 'selected' : ''}>${unit}</option>`).join('')} </select> <div> <input type="checkbox" id="config-sum-enabled" ${currentUserConfig.sumDownloadSize ? 'checked' : ''} /> <label for="config-sum-enabled"></label> </div> </div> <div style="display: flex; justify-content: space-between; margin-top: 10px;"> <button id="config-save" style="padding: 3px 8px;">Save</button> <button id="config-cancel" style="padding: 3px 8px;">Cancel</button> </div> <div style="font-size: 10px; margin-top: 5px;"> <strong>Tips:</strong> Hover over labels for descriptions </div> `; panel.appendChild(configPanel); // Add event listeners document.getElementById("config-save").addEventListener("click", saveConfig); document.getElementById("config-cancel").addEventListener("click", hideConfigPanel); // Helper functions function saveConfig() { const config = {}; if (document.getElementById("config-max-enabled").checked) { config.max = [ parseFloat(document.getElementById("config-max-value").value), document.getElementById("config-max-unit").value ]; } else { config.max = null; } if (document.getElementById("config-min-enabled").checked) { config.min = [ parseFloat(document.getElementById("config-min-value").value), document.getElementById("config-min-unit").value ]; } else { config.min = null; } if (document.getElementById("config-limit-enabled").checked) { config.downloadLimit = parseInt(document.getElementById("config-limit-value").value, 10); } else { config.downloadLimit = null; } if (document.getElementById("config-sum-enabled").checked) { config.sumDownloadSize = [ parseFloat(document.getElementById("config-sum-value").value), document.getElementById("config-sum-unit").value ]; } else { config.sumDownloadSize = null; } localStorage.setItem("torrentBDConfig", JSON.stringify(config)); // Update the current Config object immediately Config = processConfig(config); console.log('Config updated:', Config); alert("Settings saved successfully! You can now use DownloadStart with the new configuration."); hideConfigPanel(); } function hideConfigPanel() { const panel = document.getElementById("torrentbd-config-panel"); if (panel) { panel.style.display = "none"; } } } function showConfigPanel() { const panel = document.getElementById("torrentbd-config-panel"); if (panel) { panel.style.display = "block"; } else { console.error("Config panel not found. Make sure createConfigUI was called."); } } // Register menu command for configuration GM_registerMenuCommand('TorrentBD Downloader Settings', showConfigPanel); let panel = document.querySelector("#kuddus-wrapper"); const dwnBtn = document.createElement("button"); dwnBtn.textContent = "DownloadStart"; dwnBtn.setAttribute("align", "center"); dwnBtn.setAttribute("title", "DownloadStart"); dwnBtn.style = "position: fixed; top: -6px; left: 70px; border-radius: 4px; margin: 6px 8px; padding: 6px 14px; border: none; opacity: 0.5;"; dwnBtn.addEventListener("click", dwnBtnFunc); if (panel.children.length > 0) { // Check if panel has children to avoid errors panel.insertBefore(dwnBtn, panel.children[panel.children.length - 1]); } else { panel.appendChild(dwnBtn); // If panel has no children, just append it } let panel2=document.querySelector(".kuddus") // Initialize configuration UI createConfigUI(panel2);