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