// ==UserScript==
// @name TBD: TorrentBD Ratio Guard
// @namespace https://naeembolchhi.github.io/
// @version 0.10
// @description Warns you if you're trying to download a torrent that will push your ratio into the danger zone.
// @author NaeemBolchhi
// @license GPL-3.0-or-later
// @icon 
// @match https://*.torrentbd.com/torrents-details.php?id=*
// @match https://*.torrentbd.net/torrents-details.php?id=*
// @match https://*.torrentbd.org/torrents-details.php?id=*
// @match https://*.torrentbd.me/torrents-details.php?id=*
// @run-at document-end
// @grant none
// ==/UserScript==
// Update Material Icons Font
try {document.querySelector("link[href*='material-icons.css']").href = "https://fonts.googleapis.com/icon?family=Material+Icons";} catch(e) {}
// Global reusable flags
let infinite, freeleech;
// Upload value, download value, ratio value.
const coreValue = document.querySelectorAll(".card-content .profile-info-table tbody tr"),
coreUpload = coreValue[0].children[1].innerText.replace(": ","").replace(",","").replace(" ",""),
coreDownload = coreValue[1].children[1].innerText.replace(": ","").replace(",","").replace(" ",""),
coreRatio = coreValue[2].children[1].innerText.replace(": ","").replace(",","").replace(" ",""),
torrentSize = document.querySelectorAll(".torrent-info tbody tr")[0].children[1].innerText.replace(",","").replace(" ","");
if (coreDownload.match(/inf/i)) {
infinite = true;
}
if (document.querySelectorAll(".torrent-info tbody tr")[0].querySelector("img[src*='free']")) {
freeleech = true;
}
// Units and conversion values
const unitCalc = [
{"unit": "YiB", "math": "80"},
{"unit": "ZiB", "math": "70"},
{"unit": "EiB", "math": "60"},
{"unit": "PiB", "math": "50"},
{"unit": "TiB", "math": "40"},
{"unit": "GiB", "math": "30"},
{"unit": "MiB", "math": "20"},
{"unit": "KiB", "math": "10"},
{"unit": "B", "math": "0"}
];
function byteGet(value) {
for (let x = 0; x < unitCalc.length; x++) {
if (value.match(unitCalc[x].unit)) {
let multiplicand = parseFloat(value.replace(unitCalc[x].unit,"")),
multiplier = Math.pow(2,parseInt(unitCalc[x].math)),
product = multiplicand*multiplier;
return product;
}
}
}
// Calculate future ratio after downloading the viewing torrent
function futureRatio() {
let realize;
if (infinite === true) {
realize = byteGet(coreUpload)/byteGet(torrentSize);
} else {
realize = byteGet(coreUpload)/(byteGet(coreDownload)+byteGet(torrentSize));
}
return realize.toFixed(2);
}
// Guard info for various ratio levels
const ratioState = [
{"state": "1.0", "icon": "gpp_good", "text": "Safe", "color": "#4caf50"},
{"state": "0.9", "icon": "notification_important", "text": "Risky", "color": "#63b04e"},
{"state": "0.8", "icon": "notification_important", "text": "Risky", "color": "#97b246"},
{"state": "0.7", "icon": "warning", "text": "Dangerous", "color": "#cab33a"},
{"state": "0.6", "icon": "warning", "text": "Dangerous", "color": "#f3ad2b"},
{"state": "0.5", "icon": "file_download_off", "text": "Critical", "color": "#ff9f25"},
{"state": "0.4", "icon": "file_download_off", "text": "Critical", "color": "#ff862b"},
{"state": "0.3", "icon": "do_not_disturb_on", "text": "Fatal", "color": "#ff6931"},
{"state": "0.2", "icon": "do_not_disturb_on", "text": "Fatal", "color": "#f64d35"},
{"state": "0.1", "icon": "gpp_maybe", "text": "Deadly", "color": "#f44336"}
];
// Ratio Guard Button
let guardIcon = "sync_problem",
guardText = "Sync Error",
guardColor = "#400307",
guardRatio = "???";
(function() {
let ratio = parseFloat(futureRatio());
guardRatio = futureRatio();
for (let x = 0; x < ratioState.length; x++) {
if (ratio <= parseFloat(ratioState[x].state)) {
guardIcon = ratioState[x].icon;
guardText = ratioState[x].text;
guardColor = ratioState[x].color;
}
}
if (ratio > parseFloat(ratioState[0].state)) {
guardIcon = ratioState[0].icon;
guardText = ratioState[0].text;
guardColor = ratioState[0].color;
}
if (freeleech) {
guardIcon = ratioState[0].icon;
guardText = ratioState[0].text;
guardColor = ratioState[0].color;
guardRatio = "No Change";
}
})();
// Content of Guard Button
const guardInfo = `
<style>
#guard-btn {
color: ${guardColor};
border-color: ${guardColor};
background: transparent;
}
#guard-btn:hover, #guard-btn:active {
color: whitesmoke;
background: ${guardColor};
}
#guard-btn span.hide-guard {
display: none;
}
</style>
<a id="guard-btn" class="btn waves-effect inline tgaction" onclick="swapGuard()">
<i class="material-icons left">${guardIcon}</i>
<span>${guardText}</span>
<span class="hide-guard">${guardRatio}</span>
</a>
`;
const guardJS = `
function swapGuard() {
let spanGuard = document.querySelectorAll("#guard-btn span");
if (spanGuard[0].className.match(/hide/)) {
spanGuard[0].classList.remove("hide-guard");
spanGuard[1].classList.add("hide-guard");
} else {
spanGuard[1].classList.remove("hide-guard");
spanGuard[0].classList.add("hide-guard");
}
}
`;
// Button Wrapper Elements
const downBtn = document.getElementById("dl-btn"),
downWrp = downBtn.parentNode,
downDiv = downWrp.parentNode;
// Function for adding Guard Button
function addGuard(info, js) {
let guard, script;
guard = document.createElement("div");
guard.classList.add("torrtopbtn-wrapper");
guard.innerHTML = info;
downDiv.insertBefore(guard, downWrp);
downDiv.innerHTML = downDiv.innerHTML.replace(/>\s+</g,'><');
script = document.createElement("script");
script.setAttribute("type","text/javascript");
script.innerHTML = js;
downDiv.appendChild(script);
}
// Appending Guard Button
addGuard(guardInfo, guardJS);