tenno.tools highlight

4/9/2025, 5:57:43 PM

// ==UserScript==
// @name        tenno.tools highlight
// @match       https://tenno.tools/*
// @version     0.1
// @author      Jere
// @description 4/9/2025, 5:57:43 PM
// @grant        GM_addStyle
// @license     MIT 
// @namespace https://greasyfork.org/users/1521722
// ==/UserScript==

// the first element is the type# of the table
const missionLocationsArray = [0, "mercury", "venus", "earth", "mars", "phobos", "deimos", "ceres", "jupiter", "europa", "saturn", "uranus", "neptune", "pluto", "sedna", "eris", "void", "lua", "kuva fortress", "zariman", "veil"];
const missionTypesArray = [1, "capture", "exterminate", "mobile defense", "rescue", "sabotage", "spy", "alchemy", "disruption", "defense", "excavation", "interception", "survival", "void cascade", "void flood", "skirmish"];
missionTypesArray.sort();
const missionTiersArray = [2, "lith", "meso", "neo", "axi", "omnia", "requiem"];

let highlightsArray = [
    [],
    [],
    []
];
let excludesArray = [
    [],
    [],
    []
];
let specialExcludesArray = [];

function parseFromLocalStorage(key, dataset) {
    const storageData = localStorage.getItem(key + dataset)
    if (storageData) {
        return JSON.parse(storageData);
    } else {
        return [];
    }
}

function loadSettings(datasetTable) {
    // check the special exclusions separately from the normal ones i.e. datasetTable=3
    const excludeSpecialCheckboxes = document.querySelectorAll("#filters input.exclude-filter[data-table='3']");
    const storedSpecialExcludes = localStorage.getItem("specialExcludes");
    if (datasetTable === 3) {
        if (storedSpecialExcludes) {
            specialExcludesArray = JSON.parse(storedSpecialExcludes);
            for (let i = 0; i < excludeSpecialCheckboxes.length; i++) {
                const excludeSpecialCheckboxName = excludeSpecialCheckboxes[i].name.split("_")[1];
                if (specialExcludesArray.includes(excludeSpecialCheckboxName)) {
                    excludeSpecialCheckboxes[i].checked = true;
                }
            }
        }
        return;
    }
    const excludeCheckboxes = document.querySelectorAll("#filters input.exclude-filter[data-table='" + datasetTable + "']");
    const includeCheckboxes = document.querySelectorAll("#filters input.include-filter[data-table='" + datasetTable + "']");

    highlightsArray[datasetTable] = parseFromLocalStorage("hilights", datasetTable);
    excludesArray[datasetTable] = parseFromLocalStorage("excludes", datasetTable);

    // check the boxes on the page if found in the localStorages
    // using excludeCheckboxes.lenght but includeCheckboxes should be the same size
    for (let i = 0; i < excludeCheckboxes.length; i++) {
        const includeCheckboxName = includeCheckboxes[i].name.split("_")[1];
        const excludeCheckboxName = excludeCheckboxes[i].name.split("_")[1];

        if (highlightsArray[datasetTable].includes(includeCheckboxName)) {
            includeCheckboxes[i].checked = true;
        }
        if (excludesArray[datasetTable].includes(excludeCheckboxName)) {
            excludeCheckboxes[i].checked = true;
        }
    }
}

function checkboxClick(event) {
    const checkbox = event.target;
    const [checkboxType, checkboxName] = checkbox.name.split("_"); // include/exclude, planet/missiontype/era
    const checkboxTable = checkbox.dataset.table; // identifier for from which table
    // checkbox for the special exclusions, normal, steelpath and railjack
    if (checkboxTable == 3) {
        if (checkbox.checked === true) {
            if (!specialExcludesArray.includes(checkboxName)) {
                specialExcludesArray.push(checkboxName);
            }
        } else {
            if (specialExcludesArray.includes(checkboxName)) {
                specialExcludesArray = specialExcludesArray.filter(item => item !== checkboxName);
            }
        }
        localStorage.setItem("specialExcludes", JSON.stringify(specialExcludesArray));
    } else {
        // fetch the adjacent checkboxes
        let checkboxOpposite;
        if (checkboxType == "include") {
            checkboxOpposite = document.querySelector("#filters input[name='exclude_" + checkboxName + "']");
        } else if (checkboxType == "exclude") {
            checkboxOpposite = document.querySelector("#filters input[name='include_" + checkboxName + "']");
        }

        // break the input's name i.e "exclude_earth"
        const checkboxOppositeLocation = checkbox.name.split("_")[1];

        // remove from the array if the opposite box is checked
        if (checkboxOpposite.checked) {
            checkboxOpposite.checked = false;
            // remove from the excludes
            if (checkboxType == "include") {
                excludesArray[checkboxTable] = excludesArray[checkboxTable].filter(item => item !== checkboxOppositeLocation);
            }
            // remove from the array
            else if (checkboxType == "exclude") {
                highlightsArray[checkboxTable] = highlightsArray[checkboxTable].filter(item => item !== checkboxOppositeLocation);
            }
        }
        // add to array if is checked
        if (checkbox.checked) {
            if (checkboxType == "include") {
                highlightsArray[checkboxTable].push(checkboxName);
            } else if (checkboxType == "exclude") {
                excludesArray[checkboxTable].push(checkboxName);
            }
        }
        // if not checked -> remove from the arrays
        else {
            if (checkboxType == "include") {
                highlightsArray[checkboxTable] = highlightsArray[checkboxTable].filter(item => item !== checkboxName);
            } else if (checkboxType == "exclude") {
                excludesArray[checkboxTable] = excludesArray[checkboxTable].filter(item => item !== checkboxName);
            }
        }

        //save the selections to the localstorage
        localStorage.setItem("hilights" + checkboxTable, JSON.stringify(highlightsArray[checkboxTable]));
        localStorage.setItem("excludes" + checkboxTable, JSON.stringify(excludesArray[checkboxTable]));
    }
    highlightPage();
}

function createCheckbox(boxClass, boxName, dataTable) {
    const checkboxInput = document.createElement("input");
    checkboxInput.setAttribute("type", "checkbox");
    checkboxInput.setAttribute("class", boxClass);
    checkboxInput.setAttribute("name", boxName);
    checkboxInput.setAttribute("data-table", dataTable);
    checkboxInput.addEventListener("change", checkboxClick);
    return checkboxInput;
}

function createHighlightDiv(fissureType, fissureTypeText) {

    const highlightDivContainer = document.createElement("div");
    highlightDivContainer.setAttribute("class", "highlight-container light-box gap hidden " + fissureType);
    //highlightDivContainer.style.display = "none";
    const fissuresDiv = document.querySelector('wf-fissures');
    fissuresDiv.insertBefore(highlightDivContainer, fissuresDiv.querySelector('.light-box.gap'));
    const highlightDivType = document.createElement("div");
    highlightDivType.setAttribute("class", "entries " + fissureType);
    const highlightTypeH2 = document.createElement("h2");
    highlightTypeH2.innerText = fissureTypeText;
    highlightDivType.appendChild(highlightTypeH2);
    highlightDivContainer.appendChild(highlightDivType);

}

function createTable(rows, columns, contentArray) {
    const table = document.createElement("table");
    for (let row = 1; row < rows; row++) {
        const tr = document.createElement("tr");
        for (let col = 0; col < columns; col++) {
            const td = document.createElement("td");
            // 0=planets 1=missions 2=era
            switch (col) {
                case 0:
                    td.setAttribute("class", "label");
                    td.innerText = contentArray[row];
                    break;
                case 1:
                    td.setAttribute("class", "include-filter");
                    td.appendChild(createCheckbox("include-filter", "include_" + contentArray[row], contentArray[0]));
                    break;
                case 2:
                    td.setAttribute("class", "exclude-filter");
                    td.appendChild(createCheckbox("exclude-filter", "exclude_" + contentArray[row], contentArray[0]));
                    break;
            }
            tr.appendChild(td);
        }
        table.appendChild(tr);
    }
    return table;
}


function setup() {

    const filterDiv = document.createElement("div");
    filterDiv.setAttribute("id", "filters");
    const locationsDiv = document.createElement("div");
    locationsDiv.setAttribute("id", "location-filter")
    const typesDiv = document.createElement("div");
    typesDiv.setAttribute("id", "missiontype-filter");
    const tiersDiv = document.createElement("div");
    tiersDiv.setAttribute("id", "tier-filter");
    const excludeSpecialDiv = document.createElement("div");
    excludeSpecialDiv.setAttribute("id", "exclude-special");
    const overlayDiv = document.createElement("div");
    overlayDiv.setAttribute("id", "overlay");
    const filterSettings = document.createElement("div");
    filterSettings.setAttribute("id", "filter-settings");

    filterDiv.appendChild(locationsDiv);
    filterDiv.appendChild(typesDiv);
    filterDiv.appendChild(tiersDiv);

    locationsDiv.appendChild(createTable(missionLocationsArray.length, 3, missionLocationsArray));
    typesDiv.appendChild(createTable(missionTypesArray.length, 3, missionTypesArray));
    tiersDiv.appendChild(createTable(missionTiersArray.length, 3, missionTiersArray));
    tiersDiv.appendChild(excludeSpecialDiv);

    const excludeSpecialTable = document.createElement("table");
    excludeSpecialTable.setAttribute("class", "exclude-special");
    for (let row = 0; row < 3; row++) {
        const tr = document.createElement("tr");
        for (let col = 0; col < 2; col++) {
            const td = document.createElement("td");
            if (col == 0) {
                td.setAttribute("class", "label");
            } else if (col === 1) {
                td.setAttribute("class", "exclude-filter");
            }
            tr.appendChild(td);
        }
        excludeSpecialTable.appendChild(tr);
    }
    const fissureType0 = "Normal";
    const fissuretype1 = "Steel Path";
    const fissureType2 = "Void Storm";
    excludeSpecialTable.rows[0].cells[0].innerText = fissureType0
    excludeSpecialTable.rows[1].cells[0].innerText = fissuretype1
    excludeSpecialTable.rows[2].cells[0].innerText = fissureType2
    excludeSpecialTable.rows[0].cells[1].appendChild(createCheckbox("exclude-filter", "exclude_normal", 3));
    excludeSpecialTable.rows[1].cells[1].appendChild(createCheckbox("exclude-filter", "exclude_steelpath", 3));
    excludeSpecialTable.rows[2].cells[1].appendChild(createCheckbox("exclude-filter", "exclude_railjack", 3));
    excludeSpecialDiv.append(excludeSpecialTable);

    const resetButton = document.createElement("input");
    resetButton.setAttribute("type", "button");
    resetButton.setAttribute("value", "Reset all");

    // reset button -> uncheck all of the checkboxes, empty arrays and localstorage, fuck em.
    resetButton.addEventListener("click", function(event) {
        let allButtons = document.querySelectorAll("#filters input[type='checkbox']");
        for (let i = 0; i < allButtons.length; i++) {
            allButtons[i].checked = false;
        }
        highlightsArray[0].length = 0;
        highlightsArray[1].length = 0;
        highlightsArray[2].length = 0;
        excludesArray[0].length = 0;
        excludesArray[1].length = 0;
        excludesArray[2].length = 0;
        localStorage.setItem("hilights0", "");
        localStorage.setItem("hilights1", "");
        localStorage.setItem("hilights2", "");
        localStorage.setItem("excludes0", "");
        localStorage.setItem("excludes1", "");
        localStorage.setItem("excludes2", "");
        localStorage.setItem("specialExcludes", "");
        highlightPage();
    });

    excludeSpecialDiv.appendChild(resetButton);

    // added in "reverse" order because first is last, etc. AST#fasew7u
    createHighlightDiv("railjack", fissureType2);
    createHighlightDiv("steelpath", fissuretype1);
    createHighlightDiv("normal", fissureType0);

    document.body.appendChild(filterSettings);
    document.body.appendChild(filterDiv);
    document.body.appendChild(overlayDiv);

    // listener for the settings popup
    document.addEventListener("click", function(event) {
        // if settings div clicked
        if (event.target === filterSettings) {
            const currentDisplay = window.getComputedStyle(filterDiv).display;
            if (currentDisplay === "none") {
                filterDiv.style.display = "flex";
                overlayDiv.style.display = "block";
            } else {
                filterDiv.style.display = "none";
                overlayDiv.style.display = "none";
            }
            return;
        }
        // close if clicked outside
        if (filterDiv.style.display === "flex" && !filterDiv.contains(event.target)) {
            filterDiv.style.display = "none";
            overlayDiv.style.display = "none";
        }
    });
    // 0=planets, 1=mission type, 2=era, 3=special excludes
    loadSettings(0);
    loadSettings(1);
    loadSettings(2);
    loadSettings(3);
    highlightPage();
}

function highlightPage() {

    const clonedHighlightDivContent = document.querySelectorAll("wf-fissures > .highlight-container .entry");
    const missionElements = document.querySelectorAll("wf-fissures > div:not(.highlight-container) .entry");

    // check if the cloned highlights at the top are expired
    if (clonedHighlightDivContent) {
        for (let i = 0; i < clonedHighlightDivContent.length; i++) {
            const clonedHighlightEndtime = clonedHighlightDivContent[i].querySelector(".reltime").getAttribute("data-time") * 1000;
            if (clonedHighlightEndtime < Date.now()) {
                clonedHighlightDivContent[i].remove();
            }
        }
    }

    let foundHilight = 0;

    for (let missionIndex = 0; missionIndex < missionElements.length; missionIndex++) {

        const missionLocation = missionElements[missionIndex].querySelector(".mission-location");
        const missionType = missionElements[missionIndex].querySelector(".mission-type");
        const missionTier = missionElements[missionIndex].querySelector(".fissure-tier");
        const [missionPlanetText, missionNodeText] = missionLocation.innerText.toLowerCase().split("/");
        const missionTypeText = missionType.innerText.toLowerCase();
        const missionTierText = missionTier.innerText.toLowerCase();
        // get the h2 for the fissure type
        let fissureTypeText = missionElements[missionIndex].closest("wf-fissures > div").querySelector("h2").innerText;
        let fissureType;
        if (fissureTypeText == "Steel Path") {
            fissureType = "steelpath";
        } else if (fissureTypeText == "Void Storms") {
            fissureType = "railjack";
        } else {
            fissureType = "normal"
        }

        // the logic of highlighting
        if (
            (highlightsArray[0].includes(missionPlanetText) ||
                highlightsArray[1].includes(missionTypeText) ||
                highlightsArray[2].includes(missionTierText)
            ) &&
            !(
                excludesArray[0].includes(missionPlanetText) ||
                excludesArray[1].includes(missionTypeText) ||
                excludesArray[2].includes(missionTierText) ||
                specialExcludesArray.includes(fissureType)
            )
        ) {

            foundHilight++;
            let highlightDiv
            if (!missionElements[missionIndex].classList.contains("hilight")) {
                //show the main container for the fissure type
                highlightDiv = document.querySelector("wf-fissures > .highlight-container." + fissureType);
                if (highlightDiv && highlightDiv.classList.contains("hidden")) {
                    highlightDiv.classList.remove("hidden");
                }
                missionElements[missionIndex].classList.add("hilight");

                // check for duplicates
                let matchFound = false;
                for (let clonesIndex = 0; clonesIndex < clonedHighlightDivContent.length; clonesIndex++) {
                    const cloneInfo = clonedHighlightDivContent[clonesIndex].querySelector(".mission-info");
                    const missionInfo = missionElements[missionIndex].querySelector(".mission-info");
                    if (cloneInfo && missionInfo && cloneInfo.isEqualNode(missionInfo)) {
                        matchFound = true;
                        break;
                    }
                }
                // if not a duplicate -> clone
                if (!matchFound) {
                    const clonedHighlight = missionElements[missionIndex].cloneNode(true);
                    highlightDiv.appendChild(clonedHighlight);
                    console.log("Added:");
                    console.log(clonedHighlight);
                }
            }
        } else {
            if (missionElements[missionIndex].classList.contains("hilight")) {
                missionElements[missionIndex].classList.remove("hilight");
                for (let clonesIndex = 0; clonesIndex < clonedHighlightDivContent.length; clonesIndex++) {
                    clonedHighlightDivContent[clonesIndex].classList.remove("hilight");
                    // also remove the cloned
                    if (missionElements[missionIndex].querySelector(".mission-info").isEqualNode(clonedHighlightDivContent[clonesIndex].querySelector(".mission-info"))) {
                        clonedHighlightDivContent[clonesIndex].remove();
                        console.log("Removed:");
                        console.log(clonedHighlightDivContent[clonesIndex]);
                    }
                    // re-highlight if no need to remove
                    else {
                        clonedHighlightDivContent[clonesIndex].classList.add("hilight");
                    }
                }
            }
        }
        // hide the container if empty
        highlightDiv = document.querySelector("wf-fissures > .highlight-container." + fissureType);
        if (highlightDiv && highlightDiv.children.length === 1) {
            highlightDiv.classList.add("hidden");
        }
    }
    // highlight and show the amount at the h1
    const fissureHeader = document.querySelector("wf-fissures > h1 > span");
    if (foundHilight >= 1) {
        fissureHeader.classList.add("hilight");
        fissureHeader.innerText = "Void Fissures (" + foundHilight + ")";
    } else {
        fissureHeader.classList.remove("hilight");
        fissureHeader.innerText = "Void Fissures";
    }
    // find the topmost visible highlight container in a clunky way
    const topNonHighlight = document.querySelector("wf-fissures > .light-box:not(.highlight-container)");
    highlightDiv = document.querySelectorAll("wf-fissures > .highlight-container:not(.hidden)");
    if (highlightDiv[0]) {
        topNonHighlight.classList.remove("top-container");
        highlightDiv[0].classList.add("top-container");
        for (let i = 1; i < highlightDiv.length; i++) {
            highlightDiv[i].classList.remove("top-container");
        }
    }
    //
    else {
        topNonHighlight.classList.add("top-container");
    }
}


let highlightTimeout = null;

const observer = new MutationObserver((mutations) => {

    if (highlightTimeout !== null) return;

    highlightTimeout = setTimeout(() => {
        highlightPage();
        highlightTimeout = null;
    }, 5000);
});

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

window.addEventListener("load", function() {
    setup();
}, false);

GM_addStyle(`
  :root {
    --highlight-background: #2f3544;
    --highlight-color: 255, 111, 111;
  }
  wf-fissures > .highlight-container {
    background-color: var(--highlight-background);
  }
  .hilight {
    color: rgb(var(--highlight-color));
  }
  .highlight-container {
    padding-left: 0.3em;
  }
  .highlight-container h2 {
    color: rgb(var(--highlight-color));
    border-bottom: 2px solid rgba(var(--highlight-color), 0.3);
  }
  .highlight-container .hidden {
    display: none;
  }
  .top-container {
    margin-top: 0 !important;
  }
  .gap:nth-of-type(1) {
    margin-top: 0;
  }
  #filter-settings {
    position: fixed;
    top: 60px;
    left: 10px;
    height: 30px;
    width: 30px;
    cursor: pointer;
    z-index: 999;
    background-image: url("");
    background-repeat: no-repeat;
    background-position: center;
    background-size: contain;
    filter: invert(1);
    transition: transform 0.1s linear;
  }
  #filter-settings:hover {
    transform: scale(1.2);
    filter: invert(1) drop-shadow(0 0 0.2rem rgba(255, 255, 255, 0.3));
  }
  #filters #exclude-special {
    position: absolute;
    bottom: 0;
    right: 0;
    border-top: 1px dashed #7388a1;
  }
  #filters {
    display: none;
    position: fixed;
    top: 50%;
    left: 50%;
    transform: translate(-50%, -50%);
    border: 5px solid rgba(0, 0, 0, 0.2);
    padding: 1em;
    border-radius: 10px;
    color: #fff;
    font: 10pt Gudea, sans-serif !important;
    background-color: rgba(30, 45, 50, 0.5);
    backdrop-filter: blur(5px);
    z-index: 1000;
  }
  #overlay {
    position: fixed;
    top: 0;
    left: 0;
    width: 100vw;
    height: 100vh;
    background: #000;
    opacity: 0.7;
    display: none;
    z-index: 999;
  }
  #filters,
  #overlay {
    animation-name: fade;
    animation-timing-function: linear;
    animation-duration: 0.15s;
  }
  @keyframes fade {
    0% {
      opacity: 0;
    }
    100% {
      opacity: 0.7;
    }
  }
  #filters > div {
    border-right: 1px dashed #7388a1;
  }
  #filters > div:last-child {
    border-right: 0;
  }
  #filters input[type="checkbox"] {
    cursor: pointer;
    outline: 1px solid black;
  }
  #filters input[type="checkbox"]:not(:checked) {
    opacity: 0.3;
  }
  #filters input[type="checkbox"].include-filter {
    accent-color: darkgreen;
  }
  #filters input[type="checkbox"].exclude-filter {
    accent-color: darkred;
  }
  #filters input[type="button"] {
    background-color: #2f3544;
    border: 2px solid black;
    border-radius: 5px;
    float: right;
    margin: 0.3em;
  }
  #filters input[type="button"]:hover {
    background-color: #7388a1;
  }

  #filters td.include-filter,
  #filters td.exclude-filter {
    border-radius: 5px;
    text-align: center;
    vertical-align: middle;
    width: 1em;
    height: 1em;
  }
  #filters td.include-filter {
    background-color: green;
  }
  #filters td.exclude-filter {
    background-color: red;
  }
  #filters td.include-filter:hover {
    background-color: #4ca64c;
  }
  #filters td.exclude-filter:hover {
    background-color: #ff4c4c;
  }
  #filters table {
    width: 10rem;
    margin: 0.5em;
  }
  #filters .label {
    text-transform: capitalize;
  }
  wf-fissures .state-icons {
    display: none;
  }

`);