<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Token Visualization Interface</title>
    <style>
        body {
            font-family: Arial, sans-serif;
            font-size: 14px;
            line-height: 1.4;
            padding: 10px;
            max-width: 1000px;
            margin: auto;
            background-color: #f4f4f4;
            text-align: center;
        }
        h1 { font-size: 18px; margin-bottom: 10px; }
        h2 { font-size: 16px; margin-bottom: 5px; }
        .word {
            display: inline-block;
            padding: 2px;
            margin: 1px;
            border-radius: 3px;
            color: white;
            font-size: 12px;
        }
        .controls { margin-top: 10px; display: flex; justify-content: center; align-items: center; gap: 10px; }
        .slider { width: 150px; margin: 5px; }
        .dropdown { padding: 5px; font-size: 14px; }
        .section {
            padding: 10px;
            background: white;
            border-radius: 5px;
            box-shadow: 0px 0px 5px rgba(0, 0, 0, 0.1);
            margin-bottom: 10px;
            text-align: left;
        }
        img {
            max-width: 100%;
            height: auto;
            border: 1px solid #ddd;
            border-radius: 3px;
        }
        .container { display: flex; gap: 20px; }
        .left-column { flex: 1.2; display: flex; flex-direction: column; gap: 10px; }
        .right-column { flex: 1; display: flex; flex-direction: column; align-items: center; }
        .attention-container { width: 100%; }
        .all-attention-heads {
            display: grid;
            grid-template-columns: repeat(5, 1fr);
            gap: 5px;
            padding: 10px;
        }
        .all-attention-heads img {
            width: 100%;
            height: auto;
            border: 1px solid #ccc;
            border-radius: 3px;
        }
    </style>
    <script>
        let maxLayers = 39;
        let maxHeads = 39;

        async function fetchPrompts() {
            try {
                let response = await fetch("prompts/");
                let text = await response.text();
                let parser = new DOMParser();
                let doc = parser.parseFromString(text, "text/html");
                let links = doc.querySelectorAll("a");

                let folders = [];
                links.forEach(link => {
                    let name = link.getAttribute("href");
                    if (!name.startsWith(".")) {  
                        folders.push(name.replace("/", ""));
                    }
                });

                let selector = document.getElementById("promptSelector");
                selector.innerHTML = "";

                if (folders.length === 0) {
                    console.error("No folders found in /prompts/");
                    return;
                }

                folders.forEach(folder => {
                    let option = document.createElement("option");
                    option.value = folder;
                    option.textContent = folder;
                    selector.appendChild(option);
                });

                selector.value = folders[0];  
                updateVisuals(folders[0]);
            } catch (error) {
                console.error("Error fetching prompts:", error);
            }
        }

        function updateVisuals(currentPrompt) {
            let layerNum = parseInt(document.getElementById("globalLayerSlider").value);
            let headNum = document.getElementById("globalHeadSlider").value;
            let nextLayer = layerNum < maxLayers ? layerNum + 1 : maxLayers;

            loadEmbedding(currentPrompt, layerNum, "json_input_embeddings", "Input Embedding", "outputLayerNumber", "outputText", "-input.json");
            loadEmbedding(currentPrompt, layerNum, "json_final_embeddings", "Output Embedding", "nextOutputLayerNumber", "nextOutputText", "-final.json");
            loadValueEmbedding(currentPrompt, layerNum, headNum);
            loadAttentionSink(currentPrompt, layerNum, headNum);
            loadAllAttentionHeads(currentPrompt, layerNum);
        }

        function updatePrompt() {
            let selector = document.getElementById("promptSelector");
            if (!selector) {
                console.error("Prompt selector not found.");
                return;
            }
            updateVisuals(selector.value);
        }

        async function loadEmbedding(currentPrompt, layerNum, embeddingDir, label, layerHeaderId, textContainerId, suffix) {
            let filename = `Phi-3-Layer_${layerNum}${suffix}`;
            let response = await fetch(`prompts/${currentPrompt}/${embeddingDir}/${filename}`);
            let textContainer = document.getElementById(textContainerId);
            textContainer.innerHTML = "";

            try {
                let data = await response.json();
                data.forEach(entry => {
                    let span = document.createElement("span");
                    span.className = "word";
                    span.style.backgroundColor = entry.color;
                    span.textContent = entry.word + " ";
                    textContainer.appendChild(span);
                });
                document.getElementById(layerHeaderId).textContent = `${label} - Layer ${layerNum}`;
            } catch (error) {
                textContainer.innerHTML = "⚠️ Data not available";
            }
        }

        async function loadValueEmbedding(currentPrompt, layerNum, headNum) {
            let response = await fetch(`prompts/${currentPrompt}/json_value_embeddings/Phi-3-layer-${layerNum}-head-${headNum}-value.json`);
            let textContainer = document.getElementById("valueText");
            textContainer.innerHTML = "";

            try {
                let data = await response.json();
                data.forEach(entry => {
                    let span = document.createElement("span");
                    span.className = "word";
                    span.style.backgroundColor = entry.color;
                    span.textContent = entry.word + " ";
                    textContainer.appendChild(span);
                });
                document.getElementById("valueLayerNumber").textContent = `Value Embedding - Layer ${layerNum}, Head ${headNum}`;
            } catch (error) {
                textContainer.innerHTML = "⚠️ Data not available";
            }
        }

        function loadAttentionSink(currentPrompt, layerNum, headNum) {
            let imgPath = `prompts/${currentPrompt}/attention_sink/layer_${layerNum}/head_${headNum}.png`;
            document.getElementById("attentionImage").src = imgPath;
            document.getElementById("attentionLayerNumber").textContent = `Attention Sink - Layer ${layerNum}, Head ${headNum}`;
        }

        function loadAllAttentionHeads(currentPrompt, layerNum) {
            let container = document.getElementById("allAttentionHeadsContainer");
            container.innerHTML = "";

            for (let headNum = 1; headNum <= maxHeads; headNum++) {
                let img = document.createElement("img");
                img.src = `prompts/${currentPrompt}/attention_sink/layer_${layerNum}/head_${headNum}.png`;
                img.alt = `Head ${headNum}`;
                container.appendChild(img);
            }
        }

        window.onload = function() {
            fetchPrompts();
        };
    </script>
</head>
<body>

    <h1>Token-Level Visualization</h1>

    <!-- Prompt Selection -->
    <div class="section">
        <h2>Select Prompt</h2>
        <select id="promptSelector" class="dropdown" onchange="updatePrompt()"></select>
    </div>

    <!-- Global Controls -->
    <div class="section">
        <h2>Global Controls</h2>
        <div class="controls">
            <label>Layer:</label>
            <input type="range" id="globalLayerSlider" class="slider" min="1" max="39" value="1" oninput="updatePrompt()">
            <label>Head:</label>
            <input type="range" id="globalHeadSlider" class="slider" min="1" max="39" value="1" oninput="updatePrompt()">
        </div>
    </div>

    <div class="container">
        <div class="left-column">
            <div class="section"><h2 id="outputLayerNumber">Input Embedding</h2><p id="outputText">Loading...</p></div>
            <div class="section"><h2 id="valueLayerNumber">Value Embedding</h2><p id="valueText">Loading...</p></div>
            <div class="section"><h2 id="nextOutputLayerNumber">Output Embedding</h2><p id="nextOutputText">Loading...</p></div>
        </div>
        <div class="right-column">
            <div class="section attention-container">
                <h2 id="attentionLayerNumber">Attention Sink</h2>
                <img id="attentionImage">
            </div>
        </div>
    </div>


    <div class="section">
        <h2>All Attention Heads - Layer <span id="allHeadsLayer">0</span></h2>
        <div class="all-attention-heads" id="allAttentionHeadsContainer"></div>
    </div>

</body>
</html>
