TBD: Seedbonus Expansion for TorrentBD

Calculates and adds more information in the Profile Card and Seedbonus Redeem Table.

// ==UserScript==
// @name         TBD: Seedbonus Expansion for TorrentBD
// @namespace    https://naeembolchhi.github.io/
// @version      0.19
// @description  Calculates and adds more information in the Profile Card and Seedbonus Redeem Table.
// @author       NaeemBolchhi
// @license      GPL-3.0-or-later
// @icon         
// @match        https://*.torrentbd.com/*
// @match        https://*.torrentbd.net/*
// @match        https://*.torrentbd.org/*
// @match        https://*.torrentbd.me/*
// @exclude      https://*.torrentbd.*/terminal
// @exclude      https://*.torrentbd.*/theme
// @exclude      https://*.torrentbd.*/guidelines/*
// @run-at       document-end
// @grant        none
// ==/UserScript==

// Crucial information
function sbonus() {return parseFloat(localStorage.fullSB) || 0;}
function broke() {return sbonus() < 450;}
let pinfo = document.querySelectorAll("#left-block-container .profile-info-table tbody td:nth-child(even)"),
    sratio = parseFloat(pinfo[2].innerHTML),
    supload = pinfo[0].innerHTML,
    sdownload = pinfo[1].innerHTML,
    stable, shead, sbody;

try {
    stable = document.getElementById("redeem-upload").parentNode.getElementsByTagName("table")[0];
    shead = stable.querySelector("thead tr");
    sbody = stable.querySelectorAll("tbody tr");
} catch(e) {}

// Units and conversion values
const unitCalc = [
    {"unit": "YiB", "math": "8"},
    {"unit": "ZiB", "math": "7"},
    {"unit": "EiB", "math": "6"},
    {"unit": "PiB", "math": "5"},
    {"unit": "TiB", "math": "4"},
    {"unit": "GiB", "math": "3"},
    {"unit": "MiB", "math": "2"},
    {"unit": "KiB", "math": "1"},
    {"unit": "B", "math": "0"}
];

// Calculate credit in bytes
function byteGet(value) {
    value = value.replace(/\,/g,'');
    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)*10),
                product = multiplicand*multiplier;
            return product;
        }
    }
}

// Calculate bytes in credit
function creditGet(value, digits) {
    for (let x = 0; x < unitCalc.length; x++) {
        let dividend = parseFloat(value),
            divisor = Math.pow(1024,parseInt(unitCalc[x].math)),
            quotient = dividend/divisor;

        if (quotient >= 1) {
            if (quotient.toString().match(/\..../)) {
                quotient = parseFloat(quotient).toFixed(2);
            }
            return quotient.toString() + " " + unitCalc[x].unit;
        }
    }
}

// Seedbonus table (may change and need manual updates)
const seedCalc = [
    {"cost": "1000000", "cred": "3 TiB"},
    {"cost": "100000", "cred": "300 GiB"},
    {"cost": "34500", "cred": "100 GiB"},
    {"cost": "18000", "cred": "50 GiB"},
    {"cost": "7600", "cred": "20 GiB"},
    {"cost": "4000", "cred": "10 GiB"},
    {"cost": "2100", "cred": "5.0 GiB"},
    {"cost": "1100", "cred": "2.5 GiB"},
    {"cost": "450", "cred": "1.0 GiB"}
];

// Calculate seedbonus in credits across all tiers
function creditSeed(value) {
    value = parseInt(value);
    if (value < 450) {return "0 B";}

    let recycle = value,
        accumulate = 0;

    for (let x = 0; x < seedCalc.length; x++) {
        console.log(recycle);
        let seedly = creditSeedest(recycle);

        if (seedly[0] && seedly[1]) {
            accumulate += seedly[0];
            recycle = seedly[1];
        }
    }

    return creditGet(accumulate);
}

// Calculate seedbonus in credits in largest tier
function creditSeedest(value) {
    value = parseInt(value);
    if (value < 450) {return [0, value];}

    for (let x = 0; x < seedCalc.length; x++) {
        let dividend = parseFloat(value),
            divisor = parseFloat(seedCalc[x].cost),
            quotient = dividend / divisor;

        if (quotient >= 1) {
            let multiplicand = byteGet(seedCalc[x].cred),
                multiplier = parseInt(quotient),
                product = multiplicand * multiplier,
                remainder = value - (multiplier * divisor);

            return [product, remainder];
        }
    }
}

// Adding new table cells
function addCell(element, content, container, before, classes) {
    let cell = document.createElement(element);
    if (element === "tr") {
        cell.setAttribute("style","opacity: .65");
    }
    if (classes) {
        cell.setAttribute("class", classes);
    }
    cell.innerHTML = content;
    container.insertBefore(cell, before);
}

// Updating the seedbonus table
function seed() {
	// Update Headings
	shead.children[1].innerHTML = "Upload Raise";
	shead.children[2].innerHTML = "Future Ratio";

	// Update Cells
	for (let x = 0; x < sbody.length; x++) {
		let num = sbody[x].children[1].innerHTML.replace(" Upload Raise", ""),
			count = parseInt(sbonus() / parseFloat(sbody[x].children[0].innerHTML)),
			byte = byteGet(num) * count,
			credit = creditGet(byte),
            option = '<span style="margin: 0 1em;">&lt;</span>' + credit;

        if (broke() || credit == undefined || num == credit.replace(".00","")) {option = "";}

		sbody[x].children[1].innerHTML = num + option;
        sbody[x].children[1].setAttribute('data-temp', credit);
	}

	for (let x = 0; x < sbody.length; x++) {
		if (isNaN(parseInt(sratio))) {
			break;
		}

		let count = parseInt(sbonus() / parseFloat(sbody[x].children[0].innerHTML)),
			ratio = parseFloat(sbody[x].children[2].innerHTML),
            credit = sbody[x].children[1].getAttribute('data-temp'),
            mratio = (byteGet(supload) + byteGet(credit)) / byteGet(sdownload);

        // console.log(count, ratio, mratio, mratio);

        let option = '<span style="margin: 0 1em;">&lt;</span>' + parseFloat(mratio).toFixed(2);

        if (broke() || parseFloat(mratio).toFixed(2) == undefined || parseFloat(mratio).toFixed(2) == parseFloat(sratio).toFixed(2) || parseFloat(ratio).toFixed(2) == parseFloat(mratio).toFixed(2)) {option = "";}

		sbody[x].children[2].innerHTML = parseFloat(ratio).toFixed(2) + option;

        sbody[x].children[1].removeAttribute('data-temp');
	}

	// New Headings
	addCell("th", "Maximum", shead, shead.children[1]);
	addCell("th", "Future Upload", shead, shead.children[3]);

	// New Cells
	for (let x = 0; x < sbody.length; x++) {
		let count = parseInt(sbonus() / parseFloat(sbody[x].children[0].innerHTML)),
			ctext = count.toString() + " Times";

		addCell("td", ctext, sbody[x], sbody[x].children[1]);
	}
	for (let x = 0; x < sbody.length; x++) {
		let rateOne = sbody[x].children[2].innerHTML.replace(/\<span.*/, ""),
			rateAll = sbody[x].children[2].innerHTML.replace(/.*\<\/span\>/, ""),
			uploadOne = creditGet(byteGet(supload) + byteGet(rateOne)),
			uploadAll = creditGet(byteGet(supload) + byteGet(rateAll)),
            option = "<span style='margin: 0 1em;'>&lt;</span>" + uploadAll;

        if (broke() || uploadAll == undefined || uploadOne == uploadAll) {option = "";}

	    let ctext = uploadOne + option;

		addCell("td", ctext, sbody[x], sbody[x].children[3]);
	}
}

// Updating the profile table
function profile() {
    // Calculations
    let newup = creditSeed(sbonus()),
        maxup = creditGet(byteGet(supload) + byteGet(newup)),
        ra = parseFloat(byteGet(creditSeed(sbonus())) / byteGet(sdownload)),
        newra = ra.toFixed(2),
        maxra = parseFloat(parseFloat(sratio) + ra).toFixed(4);

    if (document.querySelector('#left-block-container .profile-info-table .customProfileCell')) {
        // Update Cells
        let newCells = document.querySelectorAll('#left-block-container .profile-info-table .customProfileCell');
        newCells[0].parentNode.children[1].innerHTML = `${maxup} <span class="devilmayhide">(+${newup})</span>`;
        newCells[1].parentNode.children[1].innerHTML = `${parseFloat(maxra).toFixed(2)} <span class="devilmayhide"> (+${newra})</span>`;

        let growthCells = document.querySelectorAll('#left-block-container .profile-info-table .customGrowthCell');
        growthCells[0].parentNode.children[1].innerHTML = `${newup}`;
        growthCells[1].parentNode.children[1].innerHTML = `${newra}`;
    } else {
        // New Cells
        addCell("tr", `<td class="customProfileCell">P. Uploaded</td><td>${maxup}<span class="devilmayhide"> (+${newup})</span></td>`, pinfo[0].parentNode.parentNode, pinfo[1].parentNode);
        addCell("tr", `<td class="customProfileCell">P. Ratio</td><td>${parseFloat(maxra).toFixed(2)}<span class="devilmayhide"> (+${newra})</span></td>`, pinfo[0].parentNode.parentNode, pinfo[3].parentNode);

        addCell("tr", `<td class="customGrowthCell">Growth</td><td>${newup}</td>`, pinfo[0].parentNode.parentNode, pinfo[1].parentNode, "devilmayshow");
        addCell("tr", `<td class="customGrowthCell">Growth</td><td>${newra}</td>`, pinfo[0].parentNode.parentNode, pinfo[3].parentNode, "devilmayshow");

        // Show a Growth Row depending on available width in the table
        addCell("style", `
          #left-block-container .card-content {
            container: profilecard / inline-size;
          }
          tr.devilmayshow {
            display: none;
          }
          @container (width < 320px) {
            tr.devilmayshow {
              display: table-row;
            }
            span.devilmayhide {
              display: none;
            }
          }
        `, pinfo[0].parentNode.parentNode.parentNode, null);
    }
}

// Updating under account details
function seedverse() {
    // Grabbing nodes
    let accWRAP = document.querySelectorAll('.crc-wrapper'),
        proTibCon = document.querySelector('#middle-block .profile-tib-container'),
        wrapperOG = document.createElement('div');

    // Putting the four values inside the same parent
    wrapperOG.classList.add('wrapper-og');
    proTibCon.insertBefore(wrapperOG, proTibCon.children[1]);
    wrapperOG = document.querySelector('.wrapper-og');
    wrapperOG.appendChild(accWRAP[0]);
    wrapperOG.appendChild(accWRAP[1]);
    wrapperOG.appendChild(accWRAP[2]);
    wrapperOG.appendChild(accWRAP[3]);
    wrapperOG.appendChild(accWRAP[0].cloneNode(true));
    wrapperOG.appendChild(accWRAP[2].cloneNode(true));

    // Grabbing data values only
    let accUP = accWRAP[0].querySelector('.cr-value').innerText,
        accDOWN = accWRAP[1].querySelector('.cr-value').innerText,
        accRATIO = accWRAP[2].title.replace(/.*\s([0-9].*)/,'$1'),
        accSEED = accWRAP[3].title.replace(/.*\s([0-9].*)/,'$1');

    // Grabbing target nodes for new values
    let gearBox = document.querySelectorAll('.wrapper-og .crc-wrapper');

    if (parseFloat(accSEED) > 450) {
        // Calculations
        let newup = creditSeed(accSEED),
            maxup = creditGet(byteGet(accUP) + byteGet(newup)),
            potup = creditGet(byteGet(newup)),
            ra = parseFloat(byteGet(newup)/byteGet(accDOWN)),
            newra = ra.toFixed(2),
            maxra = parseFloat(parseFloat(accRATIO)+ra).toFixed(4);

        // Set ratio to 2 digits
        gearBox[2].querySelector('.cr-value').innerText = parseFloat(gearBox[2].querySelector('.cr-value').innerText).toFixed(2);

        // Updating text for new slots
        gearBox[4].querySelector('.cr-value').innerText = `${maxup} (+${potup})`;
        gearBox[5].querySelector('.cr-value').innerText = `${parseFloat(maxra).toFixed(2)} (+${newra})`;

        // Updating the text that appears on hover
        gearBox[4].setAttribute('title', `Potential Upload: ${maxup} | Growth: ${potup}`);
        gearBox[5].setAttribute('title', `Potential Ratio: ${maxra} | Growth: ${ra.toFixed(4)}`);

        if (isNaN(accRATIO)) {
            gearBox[5].querySelector('.cr-value').innerText = `Inf. (+Beyond)`;
            gearBox[5].setAttribute('title', `Potential Ratio: Infinity | Growth: Beyond`);
        }
    } else {
        // Updating the text that appears on hover with not enough seedbonus
        gearBox[4].setAttribute('title', `Potential Upload: Unchanged | Growth: None`);
        gearBox[5].setAttribute('title', `Potential Ratio: Unchanged | Growth: None`);
    }

    // Adding new CSS to make everything look nice
    let midBlock = document.getElementById('middle-block'),
        midStyle = document.createElement('style');
    midStyle.type = 'text/css';
    midStyle.innerHTML = `
    #middle-block .card-panel .flex.top-dir {
      padding: 10px 20px;
      gap: 30px;
      display: grid;
      grid-template-columns: auto 1fr auto;
      grid-template-rows: 1fr;
    }
    #middle-block .card-panel .flex.top-dir .avatar-container {
      width: 200px;
      height: 200px;
      min-width: unset;
      min-height: unset;
      display: flex;
      align-items: center;
      justify-content: center;
      grid-column: 1/2;
      grid-row: 1/2;
    }
    #middle-block .card-panel .flex.top-dir .avatar-container .up-avatar {
      margin: 0 !important;
    }
    #middle-block .card-panel .flex.top-dir .pr-action-container {
      grid-column: 3/4;
      grid-row: 1/2;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container {
      display: flex;
      flex-direction: column;
      margin: 0;
      padding: 5px 0;
      width: 100%;
      gap: 10px 0;
      grid-column: 2/3;
      grid-row: 1/2;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container h5 {
      height: fit-content;
      line-height: 1;
      margin: 0 !important;
      display: flex;
      align-items: baseline;
      gap: 5px;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container .crc-wrapper {
      display: flex;
      margin: 0;
      gap: 4px;
      height: fit-content;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container .crc-wrapper .cr-value {
      height: fit-content;
      display: block;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container > div:last-child {
      margin: 0 !important;
      display: flex;
      flex-wrap: wrap;
      gap: 10px 10px;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container > div:last-child > div[style] {
      width: 100%;
      display: flex;
      gap: 5px;
      line-height: 1;
      align-items: center;
      margin: 6px 0 0 0 !important;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container .short-links {
      display: flex;
      padding: 0 0 0 10px;
      margin: 0;
      gap: 10px;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container .short-links .short-link-counter {
      display: flex;
      margin: 0;
      padding: 0 8px;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container .wrapper-og {
      display: grid;
      grid-template-columns: auto auto auto auto;
      grid-template-rows: auto auto;
      grid-template-areas:
        "upload download ratio seedbonus"
        "pupload pupload pratio pratio";
      width: fit-content;
      gap: 10px 20px;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container .wrapper-og .crc-wrapper:nth-child(1) {
      grid-area: upload;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container .wrapper-og .crc-wrapper:nth-child(2) {
      grid-area: download;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container .wrapper-og .crc-wrapper:nth-child(3) {
      grid-area: ratio;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container .wrapper-og .crc-wrapper:nth-child(4) {
      grid-area: seedbonus;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container .wrapper-og .crc-wrapper:nth-child(5) {
      grid-area: pupload;
      opacity: .65;
    }
    #middle-block .card-panel .flex.top-dir .profile-tib-container .wrapper-og .crc-wrapper:nth-child(6) {
      grid-area: pratio;
      opacity: .65;
    }

    /* Responsive */
    @media only screen and (min-width: 992px) and (max-width: 1145px), (max-width: 700px) {
      #middle-block .card-panel .flex.top-dir {
        grid-template-columns: auto 1fr;
        grid-template-rows: 1fr auto;
      }
      #middle-block .card-panel .flex.top-dir .avatar-container {
        grid-column: 1/2;
        grid-row: 1/2;
        height: auto;
        width: auto;
        max-height: 200px;
        max-width: 200px;
      }
      #middle-block .card-panel .flex.top-dir .pr-action-container {
        grid-column: 1/2;
        grid-row: 2/3;
        flex-direction: row;
        width: 100%;
      }
      #middle-block .card-panel .flex.top-dir .profile-tib-container {
        grid-column: 2/3;
        grid-row: 1/3;
        justify-content: space-between;
      }
    }
    @media only screen and (min-width: 651px) and (max-width: 740px), (min-width: 401px) and (max-width: 520px) {
      #middle-block .card-panel .flex.top-dir .profile-tib-container .wrapper-og {
        grid-template-columns: auto auto auto !important;
        grid-template-rows: auto auto auto auto !important;
        grid-template-areas:
          "upload download seedbonus"
          "pupload pupload pupload"
          "ratio ratio ratio"
          "pratio pratio pratio" !important;
      }
    }
    @media only screen and (max-width: 650px) {
      #middle-block .card-panel .flex.top-dir {
        grid-template-rows: auto auto auto;
        grid-template-columns: 1fr;
        gap: 20px;
      }
      #middle-block .card-panel .flex.top-dir .avatar-container {
        grid-column: 1/2;
        grid-row: 1/2;
        height: fit-content;
        margin: 0 auto;
      }
      #middle-block .card-panel .flex.top-dir .pr-action-container {
        grid-column: 1/2;
        grid-row: 3/4;
      }
      #middle-block .card-panel .flex.top-dir .profile-tib-container {
        grid-column: 1/2;
        grid-row: 2/3;
      }
      #middle-block .card-panel .flex.top-dir .profile-tib-container h5 {
        width: fit-content;
        margin: 0 auto !important;
      }
      #middle-block .card-panel .flex.top-dir .profile-tib-container .wrapper-og {
        width: fit-content;
        margin: 0 auto;
      }
      #middle-block .card-panel .flex.top-dir .profile-tib-container > .crc-wrapper {
        width: fit-content;
        margin: 0 auto;
      }
      #middle-block .card-panel .flex.top-dir .profile-tib-container > div:last-child {
        justify-content: center;
      }
      #middle-block .card-panel .flex.top-dir .profile-tib-container > div:last-child > div[style] {
        width: fit-content;
        margin: 0 auto;
        padding: 0 18px;
      }
      #middle-block .card-panel .flex.top-dir .profile-tib-container .wrapper-og {
        grid-template-columns: auto auto auto auto;
        grid-template-rows: auto auto;
        grid-template-areas:
          "upload download ratio seedbonus"
          "pupload pupload pratio pratio";
      }
    }
    @media only screen and (max-width: 400px) {
      #middle-block .card-panel .flex.top-dir .profile-tib-container .wrapper-og {
        grid-template-columns: auto auto;
        grid-template-rows: auto auto auto auto;
        grid-template-areas:
          "upload download"
          "pupload pupload"
          "ratio seedbonus"
          "pratio pratio";
      }
    }
    `;

    midBlock.appendChild(midStyle);
}

// Seedbonus amount is shortened by default, but we need the longer version.
// So, we load the account-details page in the background if you're not already in the account-details page.
// Add ?background to the url so the script knows when it is in background mode.
// We don't want to do this too often, so maybe once every 30 mins.
function getFullSB() {
    function whenReady() {
        if (document.readyState !== 'complete' && document.readyState !== 'interactive') {return;}
        // Don't update on other people's account details page
        if (!window.location.search.match(/background/i) && window.location.href.replace(/.*id=/,'') !== document.querySelector('a.accc-btn[href*="account-details"]').href.replace(/.*id=/,'')) {return;}

        let getSeedItem = document.querySelector('#middle-block .crc-wrapper[title*="bonus"]'),
            getSeedBonus = getSeedItem.getAttribute('title').replace('Seedbonus: ','');

        localStorage.fullSB = getSeedBonus;
        localStorage.seedTM = (new Date()).getTime();

        if (window.location.search.match(/background/i)) {
            window.parent.postMessage('recheckSB', window.location.origin);
        } else {
            profile();
        }
    }
    whenReady();

    document.addEventListener('readystatechange', function() {
        whenReady();
    });
}

// A second copy of the function for the seedbonus page, which also has the full seedbonus amount.
function getFullSB2() {
    let getSeedItem = document.querySelector('#middle-block > div > div > h5 > span'),
        getSeedBonus = getSeedItem.innerText;

    localStorage.fullSB = getSeedBonus;
    localStorage.seedTM = (new Date()).getTime();

    profile();
}

// Create an iframe and listen for a message.
function iframinator() {
    if (window.location.search.match(/background/i)) {return;}

    let seedInterval = 30 * 60 * 1000, // 30 mins in ms
        nowTime = parseInt((new Date()).getTime()),
        targetTime = parseInt(localStorage.seedTM) + seedInterval;
    if (nowTime < targetTime) {return;}

    window.addEventListener('message', function(e) {
        if (e.data !== 'recheckSB') {return;}

        profile();
        try {document.querySelector('#seedFrame').remove();} catch {}
    });

    let framed = document.createElement('iframe');
    framed.setAttribute('src', window.location.origin + '/account-details.php?background');
    framed.setAttribute('id', 'seedFrame');
    framed.setAttribute('style', 'height: 1px; width: 1px; position: fixed; left: -200%; top: -200%; z-index: -9001; overflow: hidden;'.replace(/\;/g,' !important;'));
    document.body.appendChild(framed);
}

// Execute functions when appropriate
profile();
if (window.location.pathname.match(/seedbonus\.php/)) {
    getFullSB2();
    seed();
} else if (window.location.pathname.match(/account\-details\.php/)) {
    getFullSB();
    seedverse();
} else {
    iframinator();
}