Wanikani Levels Overview Plus

Improves the levels overview popup with progress indications

// ==UserScript==
// @name          Wanikani Levels Overview Plus
// @namespace     Mercieral
// @description   Improves the levels overview popup with progress indications
// @match         https://www.wanikani.com/*
// @match         https://preview.wanikani.com/*
// @version       2.2.4
// @run-at        document-end
// @grant         none
// ==/UserScript==

(function() {
    // Require the WK open resource
    if (!window.wkof) {
        alert('"Wanikani Levels Overview Plus" script requires Wanikani Open Framework.\nYou will now be forwarded to installation instructions.');
        window.location.href = 'https://community.wanikani.com/t/instructions-installing-wanikani-open-framework/28549';
        return;
    }

    // Initialize the level stage data array
    const levelCounts = [];
    for (let i = 0; i < 61; i++) {
        levelCounts.push({
            locked: 0,
            unlocked: 0,
            apprentice: 0,
            guru: 0,
            master: 0,
            enlighten: 0,
            burn: 0,
            nextReviewItem: null,
            longestReviewItem: null,
        });
    }

    window.wkof.include('ItemData,Settings');
    window.wkof.ready('document,ItemData,Settings')
        .then(load_settings)
        .then(initCSS)
        .then(sortData)
        .then(finishUI)
        .catch(loadError);

    /**
     * Loads the saved settings for this script
     */
    function load_settings () {
        return window.wkof.Settings.load('levelOverviewPlus', {
            showNextReview: true
        })
    }

    /**
     * Parse the data once the document and WKOF data is ready
     */
    function sortData () {
        if (window.wkof.ItemData == null) {
            return Promise.reject();
        }
        const config = {
            wk_items: {
                options: {assignments: true}
            }
        };
        return window.wkof.ItemData.get_items(config).then(function (processItems) {
            if (processItems == null) {
                return Promise.reject();
            }
            for (let i = 0, itemsLen = processItems.length; i < itemsLen; i++) {
                const item = processItems[i];
                if (item != null) {
                    const srsLevel = item.assignments && item.assignments.srs_stage;
                    const unlockedDate = item.assignments && item.assignments.unlocked_at;
                    const level = item.data && item.data.level;
                    const levelStageCounts = levelCounts[level];
                    if (levelStageCounts == null) {
                        // skip this item...
                        continue;
                    }

                    // Increment the appropriate level's stage counter for this item
                    if (srsLevel == null || srsLevel === 0) {
                        // Checking for unlocked date allows "unlocked but unstarted" items to be considered separately if needed
                        if (unlockedDate == null) {
                            levelStageCounts.locked++;
                        }
                        else {
                            levelStageCounts.unlocked++;
                        }
                    } else if (srsLevel < 5) {
                        levelStageCounts.apprentice++;
                    } else if (srsLevel < 7) {
                        levelStageCounts.guru++;
                    } else if (srsLevel === 7) {
                        levelStageCounts.master++;
                    } else if (srsLevel === 8) {
                        levelStageCounts.enlighten++;
                    } else if (srsLevel === 9) {
                        levelStageCounts.burn++;
                    }

                    // Check if this is the furthest item from burning
                    let nextReviewDate = item.assignments && item.assignments.available_at && new Date(item.assignments.available_at);
                    if ((nextReviewDate == null && srsLevel == 8 && levelStageCounts.longestReviewItem == null) ||
                        (nextReviewDate != null && (
                            levelStageCounts.longestReviewItem == null ||
                            srsLevel < levelStageCounts.longestReviewItem.level ||
                            (srsLevel == levelStageCounts.longestReviewItem.level && nextReviewDate > levelStageCounts.longestReviewItem.nextReviewDate)
                        ))) {
                        levelStageCounts.longestReviewItem = {
                            level: srsLevel,
                            nextReviewDate: nextReviewDate,
                        };
                    }

                    // Check if this is the next item for review for its level
                    if (nextReviewDate != null && (levelStageCounts.nextReviewItem == null || nextReviewDate < levelStageCounts.nextReviewItem.reviewDate)) {
                        levelStageCounts.nextReviewItem = {
                            reviewDate: nextReviewDate,
                            characters: item.data.characters,
                            type: item.object,
                        }
                        if (item.data.characters == null) {
                            const itemImg = item.data.character_images.filter((img) => {
                                return (img.content_type === 'image/svg+xml' && !img.metadata.inline_styles);
                            })[0];
                            levelStageCounts.nextReviewItem.imgUrl = itemImg && itemImg.url;
                            levelStageCounts.nextReviewItem.slug = item.data.slug;
                        }
                    }
                }
            }
            return Promise.resolve();
        });
    }

    /**
     * Create the UI once the data has been parsed into the data array
     */
    function finishUI () {
        // Get the HTML square elements for each level in the popout
        const levelBlocks = document.querySelectorAll('.sitemap__expandable-chunk--levels .sitemap__grouped-pages > ol > li > a');
        for (let levelBlock of levelBlocks) {
            overwriteLevelBlock(levelBlock);
        }
        createSettings();
        updateLevelHeader();

        /**
         * Overwrite the level block with the custom elements and tooltip
         */
        function overwriteLevelBlock (levelBlock) {
            levelBlock.classList.add('level-block');
            const originalLevelText = levelBlock.textContent;
            const levelStageCounts = levelCounts[Number(originalLevelText)];
            const levelTotal = levelStageCounts.locked + levelStageCounts.unlocked + levelStageCounts.apprentice + levelStageCounts.guru + levelStageCounts.master + levelStageCounts.enlighten + levelStageCounts.burn;

            // Create the overlay elements
            const levelText = `<span class="level-block-text">${originalLevelText}</span>`;
            const lockedDiv = `<div class="level-block-item level-block-locked" style="width:${(levelStageCounts.locked + levelStageCounts.unlocked)/levelTotal*100}%"></div>`;
            const apprenticeDiv = `<div class="level-block-item level-block-apprentice" style="width:${levelStageCounts.apprentice/levelTotal*100}%"></div>`;
            const guruDiv = `<div class="level-block-item level-block-guru" style="width:${levelStageCounts.guru/levelTotal*100}%"></div>`;
            const masterDiv = `<div class="level-block-item level-block-master" style="width:${levelStageCounts.master/levelTotal*100}%"></div>`;
            const enlightenedDiv = `<div class="level-block-item level-block-enlightened" style="width:${levelStageCounts.enlighten/levelTotal*100}%"></div>`;
            const burnDiv = `<div class="level-block-item level-block-burn" style="width:${levelStageCounts.burn/levelTotal*100}%"></div>`;

            // Create the tooltip
            const lockedText = `<div class="locked-tooltip tooltip-section"><p class="tooltip-section-title">Locked</p><p class="tooltip-section-count">${levelStageCounts.locked + levelStageCounts.unlocked}</p></div>`;
            const apprenticeText = `<div class="apprentice-tooltip tooltip-section"><p class="tooltip-section-title">Apprentice</p><p class="tooltip-section-count">${levelStageCounts.apprentice}</p></div>`;
            const guruText = `<div class="guru-tooltip tooltip-section"><p class="tooltip-section-title">Guru</p><p class="tooltip-section-count">${levelStageCounts.guru}</p></div>`;
            const masterText = `<div class="master-tooltip tooltip-section"><p class="tooltip-section-title">Master</p><p class="tooltip-section-count">${levelStageCounts.master}</p></div>`;
            const enlightenedText = `<div class="enlightened-tooltip tooltip-section"><p class="tooltip-section-title">Enlightened</p><p class="tooltip-section-count">${levelStageCounts.enlighten}</p></div>`;
            const burnText = `<div class="burn-tooltip tooltip-section"><p class="tooltip-section-title">Burn</p><p class="tooltip-section-count">${levelStageCounts.burn}</p></div>`;
            const totalText = `<div class="total-tooltip tooltip-section"><p class="tooltip-section-title">Total</p><p class="tooltip-section-count">${levelTotal}</p></div>`;
            let nextReviewDateText = "N/A";
            let nextReviewChars = "N/A";
            const nextReview = levelStageCounts.nextReviewItem;
            let loadImgData;
            if (nextReview != null) {
                // Get the time amount until review
                const nextReviewMinutes = (new Date(nextReview.reviewDate) - new Date()) / (1000 * 60);
                const nextReviewHours = nextReviewMinutes / 60;
                if (nextReviewHours <= 0) {
                    nextReviewDateText = "now";
                } else if (nextReviewHours <= 1) {
                    const minutes = Math.floor(nextReviewMinutes);
                    nextReviewDateText = minutes + " minute" + (minutes !== 1 ? "s" : '');
                } else if (nextReviewHours >= 24) {
                    const days = Math.floor(nextReviewHours / 24);
                    nextReviewDateText = days + " day" + (days !== 1 ? "s" : '');
                } else {
                    const hours = Math.floor(nextReviewHours);
                    nextReviewDateText = hours + " hour" + (hours !== 1 ? "s" : '');
                }

                // Get the review item characters
                let itemClass = "guru-tooltip";
                if (nextReview.type === "radical") {
                    itemClass = "enlightened-tooltip";
                } else if (nextReview.type === "kanji") {
                    itemClass = "apprentice-tooltip";
                }
                if (nextReview.characters) {
                    nextReviewChars = `<span class="${itemClass} next-review-chars">${nextReview.characters}</span>`;
                }
                else if (nextReview.imgUrl && nextReview.slug) {
                    nextReviewChars = `<span class="${itemClass} next-review-chars" id="svg_${nextReview.slug}"></span>`;
                    loadImgData = nextReview;
                }
            }

            let longestReviewDate = 'N/A';
            if (levelStageCounts.burn === levelTotal) {
                longestReviewDate = 'Done!';
            }
            else if (levelStageCounts.longestReviewItem && levelStageCounts.locked == 0) {
                longestReviewDate = levelStageCounts.longestReviewItem.nextReviewDate;
                const currLevel = levelStageCounts.longestReviewItem.level;

                if (longestReviewDate == null) {
                    longestReviewDate = 'Now!';
                }
                else {
                    let hoursToAdd = 0;
                    let daysToAdd = 0;
                    let monthsToAdd = 0;

                    /*  Apprentice 1 → 4 hours → Apprentice 2
                        Apprentice 2 → 8 hours → Apprentice 3
                        Apprentice 3 → 1 day → Apprentice 4
                        Apprentice 4 → 2 days → Guru 1
                        Guru 1 → 1 week → Guru 2
                        Guru 2 → 2 weeks → Master
                        Master → 1 month → Enlightened
                        Enlightened → 4 months → Burned

                        For Level 1 & 2 the SRS timings are accelerated for the Apprentice stage.
                        Apprentice 1 → 2 hours → Apprentice 2
                        Apprentice 2 → 4 hours → Apprentice 3
                        Apprentice 3 → 8 hours → Apprentice 4
                        Apprentice 4 → 1 day → Guru 1 */

                    if (currLevel <= 7) monthsToAdd += 4;
                    if (currLevel <= 6) monthsToAdd += 1;
                    if (currLevel <= 5) daysToAdd += 14;
                    if (currLevel <= 4) daysToAdd += 7;
                    if (Number(originalLevelText) <= 2) {
                        if (currLevel <= 3) daysToAdd += 1;
                        if (currLevel <= 2) hoursToAdd += 8;
                        if (currLevel <= 1) hoursToAdd += 4;
                    }
                    else {
                        if (currLevel <= 3) daysToAdd += 2;
                        if (currLevel <= 2) daysToAdd += 1;
                        if (currLevel <= 1) hoursToAdd += 8;
                    }

                    if (monthsToAdd) {
                        longestReviewDate.setMonth(longestReviewDate.getMonth() + monthsToAdd);
                    }
                    if (daysToAdd) {
                        longestReviewDate.setDate(longestReviewDate.getDate() + daysToAdd);
                    }
                    if (hoursToAdd) {
                        longestReviewDate.setHours(longestReviewDate.getHours() + hoursToAdd);
                    }
                    longestReviewDate = longestReviewDate.toLocaleString([], {
                        month: 'numeric',
                        day: 'numeric',
                        year: 'numeric',
                        hour: 'numeric',
                        minute: 'numeric',
                    });
                }
            }

            const nextReviewText = `<p class="tooltip-extra-info next-review-text">Next: ${nextReview ? `${nextReviewChars} (${nextReviewDateText})` : "N/A"}</p>`;
            const burnDateText = `<p class="tooltip-extra-info next-review-text">Burn: ${longestReviewDate}</p>`;
            const tooltip = `<div class="level-tooltip"><p class="tooltip-level-text">Level ${originalLevelText}</p>${lockedText}${apprenticeText}${guruText}${masterText}${enlightenedText}${burnText}${totalText}${nextReviewText}${burnDateText}</div>`;

            if (levelStageCounts.burn === levelTotal) {
                // Fully burned level, add the checkbox to the div
                levelBlock.classList.add('level-block-complete');
            }

            if (levelStageCounts.locked === levelTotal) {
                // Fully locked level, add the padlock to the div
                levelBlock.classList.add('level-block-full-locked');
            }

            // Rewrite the level block's HTML with the custom elements
            levelBlock.innerHTML = `<div class="level-block-container">${apprenticeDiv}${guruDiv}${masterDiv}${enlightenedDiv}${burnDiv}${lockedDiv}</div>${levelText}${tooltip}`;

            if (loadImgData != null && loadImgData.imgUrl) {
                wkof.load_file(loadImgData.imgUrl).then((svgData) => {
                    const destSpan = document.getElementById(`svg_${loadImgData.slug}`);
                    if (destSpan) {
                        destSpan.innerHTML = svgData;
                    }
                });
            }
        }

        /**
         * Create the settings button and panel with contents
         */
        function createSettings () {
            // Create the settings popop
            const settingsButton = document.createElement('div');
            settingsButton.id = "settings-button";
            const settingsPanel = document.createElement('div');
            settingsPanel.id = "settings-panel"
            settingsPanel.innerHTML = `<div id='settings-title'>Levels Overview Plus Settings</div><div id="showNextRow" class="settings-row"><input id="showNextCb" type="checkbox"><span>Show Next Review</span></div>`;
            document.querySelector('.sitemap__grouped-pages').append(settingsButton, settingsPanel);
            const showNextToggle = document.getElementById("showNextCb");
            const showNextSetting = document.getElementById("showNextRow");

            // Handling hiding/showing the settings panel
            let settingsPanelActive = false;
            function handleBodyClick (e) {
                if (settingsPanelActive && !e.target.closest("#settings-panel") && e.target.id !== 'settings-button') {
                    settingsPanel.style.display = 'none';
                    settingsButton.style.backgroundColor = "";
                    settingsPanelActive = false;
                }
            }
            document.body.addEventListener('mouseup', handleBodyClick);

            settingsButton.onclick = () => {
                settingsPanelActive = !settingsPanelActive;
                if (settingsPanelActive) {
                    settingsPanel.style.display = 'block';
                    settingsButton.style.backgroundColor = "#b0b0b0";
                } else {
                    settingsPanel.style.display = 'none';
                    settingsButton.style.backgroundColor = "";
                }
            };

            // Handle the "toggle next review" checkbox
            const nextReviewTexts = Array.from(document.getElementsByClassName('next-review-text'));
            const updateNextReviewTexts = () => {
                const checked = window.wkof.settings.levelOverviewPlus.showNextReview;
                if(checked) {
                    nextReviewTexts.forEach((nrt) => nrt.style.display = 'block');
                } else {
                    nextReviewTexts.forEach((nrt) => nrt.style.display = 'none');
                }
            }
            const showNextReviewChanged = () => {
                window.wkof.settings.levelOverviewPlus.showNextReview = showNextToggle.checked;
                window.wkof.Settings.save('levelOverviewPlus');
                updateNextReviewTexts();
            }
            showNextToggle.onchange = showNextReviewChanged;
            updateNextReviewTexts();
            showNextToggle.checked = window.wkof.settings.levelOverviewPlus.showNextReview;
            showNextSetting.onclick = () => {
                showNextToggle.checked = !window.wkof.settings.levelOverviewPlus.showNextReview;
                showNextReviewChanged();
            };
        }

        /**
         * Append a "+" to the level nav text to indicate script success
         */
        function updateLevelHeader () {
            document.querySelectorAll('.navigation > .sitemap > li:first-child > .sitemap__section-header > span').forEach((levelSpan) => levelSpan.append('+'));
        }
    }

    /**
     * Create the CSS style sheet and append it to the document
     */
    function initCSS() {
        const styling = document.createElement('style');
        styling.innerHTML = `
            .level-block {
                position: relative;
                height: 46px;
                border-width: 0 !important;
            }

            .level-block-text {
                width: 100%;
                position: absolute;
                top: 0;
                left: 0;
                text-align: center;
                line-height: 46px;
            }

            #settings-button {
                width: 28px;
                height: 28px;
                position: absolute;
                top: 8px;
                right: 8px;
                cursor: pointer;
                border-radius: 5px;
                background-image: url('');
                background-position: center !important;
                background-repeat: no-repeat !important;
                background-size: 18px !important;
            }

            #settings-panel {
                display: none;
                position: absolute;
                padding: 10px 15px;
                background-color: #2a2a2a;
                border-radius: 4px;
                top: 40px;
                right: 8px;
                user-select: none;
            }

            #settings-title {
                font-weight: bold;
                font-size: 14px;
            }

            .settings-row {
                padding: 8px 0 0 10px;
                font-size: 12px;
                line-height: 20px;
                cursor: pointer;
            }

            .settings-row input {
                margin: 0 3px 0 0;
                width: 20px;
                height: 20px;
            }

            .level-tooltip {
                color: #eeeeee;
                visibility: hidden;
                background-color: rgba(0,0,0,0.8);
                text-align: center;
                padding: 0 10px 2px 10px;
                margin-left: 52px;
                margin-top: -5px;
                border-radius: 6px;
                position: absolute;
                z-index: 2;
                width: 160px;
                font-family: "Open Sans","Helvetica Neue",Helvetica,Arial,sans-serif !important;
                pointer-events: none;
                box-sizing: content-box;
            }

            .tooltip-level-text {
                font-size: 16px;
                margin: 5px 0;
                font-family: "Open Sans","Helvetica Neue",Helvetica,Arial,sans-serif !important;
                line-height: 26px;
            }

            .tooltip-extra-info {
                font-size: 12px;
                text-align: left;
                margin: 0;
                font-family: "Open Sans","Helvetica Neue",Helvetica,Arial,sans-serif !important;
                white-space: normal;
                text-shadow: none;
                text-indent: -20px;
                padding-left: 20px;
                padding-bottom: 6px;
                line-height: 18px;
            }

            .next-review-chars {
                text-shadow: none;
                padding: 0 3px;
                font-size: 14px;
            }

            svg.radical {
                fill:none;
                stroke:#fff;
                stroke-linecap:square;
                stroke-miterlimit:2;
                stroke-width:68px;
                height: 15px;
                vertical-align: top;
                margin-top: 1.5px;
            }

            .tooltip-section {
                clear: both;
                overflow: auto;
                padding: 3px 10px;
                line-height: 18px;
            }

            .tooltip-section-title {
                float: left;
                margin: 0;
                font-family: "Open Sans","Helvetica Neue",Helvetica,Arial,sans-serif !important;
                font-size: 11px;
                font-weight: bold;
                text-shadow: none;

            }

            .tooltip-section-count {
                float: right;
                margin: 0;
                font-weight: bold;
                font-family: "Open Sans","Helvetica Neue",Helvetica,Arial,sans-serif !important;
                font-size: 11px;
                text-shadow: none;
            }

            .locked-tooltip {
                background-color: #666;
                background-image: linear-gradient(to bottom, #666, #444);
            }

            .apprentice-tooltip {
                background-color: #dd0093;
                background-image: linear-gradient(to bottom, #ff00aa, #b30077);
            }

            .guru-tooltip {
                background-color: #882d9e;
                background-image: linear-gradient(to bottom, #aa38c7, #662277);
            }

            .master-tooltip {
                background-color: #294ddb;
                background-image: linear-gradient(to bottom, #516ee1, #2142c4);
            }

            .enlightened-tooltip {
                background-color: #0093dd;
                background-image: linear-gradient(to bottom, #00aaff, #0077b3);
            }

            .burn-tooltip {
                background-color: #fbc042;
                background-image: linear-gradient(to bottom, #fbc550, #c88a04);
                color: #ffffff;
            }

            .total-tooltip {
                background-color: #efefef;
                background-image: linear-gradient(to bottom, #efefef, #cfcfcf);
                color: #000000;
                margin-bottom: 10px;
            }

            .level-block:hover .level-tooltip {
                visibility: visible;
            }

            .sitemap__pages--levels .sitemap__page--current-level a .level-block-container {
                border: medium solid black !important;
            }

            .sitemap__pages--levels .sitemap__page a {
                border-radius: 4px;
            }

            .sitemap__pages--levels .sitemap__page a .level-block-container:hover {
                background-color: rgba(255,255,255,0.5);
            }

            .sitemap__grouped-pages, .sitemap__expandable-chunk>*:first-child {
                overflow: visible !important;
            }

            .level-block-container {
                height: 100%;
                width: 100%;
                position: absolute;
                top: 0;
                left: 0;
                overflow: hidden;
                border-radius: 4px;
                border: thin solid black !important;
                box-sizing: border-box;
            }

            .level-block-complete {
                background-position: center !important;
                background-repeat: no-repeat !important;
                background-size: 34px !important;
                background-image: url('') !important;
            }

            .level-block-full-locked {
                background-position: center !important;
                background-repeat: no-repeat !important;
                background-size: 36px !important;
                background-image: url('') !important;
            }

            .level-block-item {
                height: 100%;
                display: inline-block;
            }

            .level-block-locked {
                background-color: rgba(0, 0, 0, 0.3);
            }

            .level-block-apprentice {
                background-color: rgba(221, 0, 147, 0.4);
            }

            .level-block-guru {
                background-color: rgba(136, 45, 158, 0.4);
            }

            .level-block-master {
                background-color: rgba(41, 77, 219, 0.4);
            }

            .level-block-enlightened {
                background-color: rgba(0, 147, 221, 0.4);
            }

            .level-block-burn {
                background-color: rgba(251, 192, 66, 0.4);
            }
        `;
        document.head.append(styling);
        return Promise.resolve();
    }

    /**
     * log an error if any part of the wkof data request failed
     * @param {*} [e] - The error to log if it exists
     */
    function loadError (e) {
        console.error('Failed to load data from WKOF for "Wanikani Levels Overview Plus"', e);
    }
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址