// Data structure for conversation visualization
const conversationData = {CONVERSATION_DATA}

// HTML escaping function
function escapeHtml(text) {
    const div = document.createElement('div');
    div.textContent = text;
    return div.innerHTML;
}

// HTML Generation Functions
function generateMessageHTML(message) {
    const messageDiv = document.createElement('div');
    messageDiv.className = 'msg';
    
    // Create role element
    const roleDiv = document.createElement('div');
    roleDiv.className = `msg-role ${message.role}`;
    roleDiv.textContent = message.role;
    messageDiv.appendChild(roleDiv);
    
    // Create content based on type
    if (message.type === 'text') {
        const contentDiv = document.createElement('div');
        contentDiv.className = 'msg-content';
        // Escape HTML and replace \n with <br> tags to display line breaks properly
        contentDiv.innerHTML = escapeHtml(message.content);
        messageDiv.appendChild(contentDiv);
    } else if (message.type === 'dict') {
        const dictDiv = document.createElement('div');
        dictDiv.className = 'msg-dict';
        if (message.fullText) {
            dictDiv.setAttribute('data-full-text', message.fullText);
        }
        
        // Generate dict items
        for (const [label, text] of Object.entries(message.content)) {
            const itemDiv = document.createElement('div');
            itemDiv.className = 'msg-dict-item';
            
            const labelSpan = document.createElement('span');
            labelSpan.className = 'msg-dict-label';
            labelSpan.textContent = label;
            
            const textSpan = document.createElement('span');
            textSpan.className = 'msg-dict-text';
            
            if (typeof text === 'object' && text.detailed) {
                itemDiv.classList.add('detailed-info');
                textSpan.textContent = text.value;
            } else {
                textSpan.textContent = text;
            }
            
            itemDiv.appendChild(labelSpan);
            itemDiv.appendChild(textSpan);
            dictDiv.appendChild(itemDiv);
        }
        
        messageDiv.appendChild(dictDiv);
    }
    
    return messageDiv;
}

function generateEnvironmentHTML(environment) {
    const envDiv = document.createElement('div');
    envDiv.className = 'env';
    
    // Environment header
    const headerDiv = document.createElement('div');
    headerDiv.className = 'env-header';
    headerDiv.textContent = environment.header;
    envDiv.appendChild(headerDiv);
    
    // Environment thumbnail - uses innerHTML to allow HTML tags
    const thumbnailDiv = document.createElement('div');
    thumbnailDiv.className = 'env-thumbnail';
    thumbnailDiv.innerHTML = environment.thumbnail; // Intentionally using innerHTML for HTML content
    envDiv.appendChild(thumbnailDiv);
    
    // Environment info
    const infoDiv = document.createElement('div');
    infoDiv.className = 'env-info';
    
    for (const [label, value] of Object.entries(environment.info)) {
        const itemDiv = document.createElement('div');
        itemDiv.className = 'env-info-item';
        
        // Check if this is a detailed info item
        if (typeof value === 'object' && value.detailed) {
            itemDiv.classList.add('detailed-info');
        }
        
        const labelSpan = document.createElement('span');
        labelSpan.className = 'env-info-label';
        labelSpan.textContent = label;
        
        const textSpan = document.createElement('span');
        textSpan.className = 'env-info-text';
        textSpan.textContent = typeof value === 'object' ? value.value : value;
        
        itemDiv.appendChild(labelSpan);
        itemDiv.appendChild(textSpan);
        infoDiv.appendChild(itemDiv);
    }
    
    envDiv.appendChild(infoDiv);
    return envDiv;
}

function generateRowHTML(message, previousMessage = null) {
    const rowDiv = document.createElement('div');
    rowDiv.className = 'row';
    
    // Add step attribute if message has step
    if (message.step) {
        rowDiv.setAttribute('data-step', message.step);
    }
    
    // Create message container
    const msgContainerDiv = document.createElement('div');
    msgContainerDiv.className = 'msg-container';
    
    // Generate message
    const msgDiv = generateMessageHTML(message);
    msgContainerDiv.appendChild(msgDiv);
    
    // Add arrow down (except for last message)
    const arrowDownDiv = document.createElement('div');
    arrowDownDiv.className = 'arrow-down';
    msgContainerDiv.appendChild(arrowDownDiv);
    
    rowDiv.appendChild(msgContainerDiv);
    
    // Add environment and arrow-right if message has environment
    if (message.environment) {
        const arrowRightDiv = document.createElement('div');
        arrowRightDiv.className = 'arrow-right';
        rowDiv.appendChild(arrowRightDiv);
        
        const envDiv = generateEnvironmentHTML(message.environment);
        rowDiv.appendChild(envDiv);
    }
    
    return rowDiv;
}

function generateDashboardHTML() {
    const dashboardContainer = document.getElementById('dashboard-container');
    dashboardContainer.innerHTML = ''; // Clear existing content
    
    const dashboardEntries = Object.entries(conversationData.dashboard);
    const itemsPerRow = 4;
    
    // Split dashboard items into rows of 3
    for (let i = 0; i < dashboardEntries.length; i += itemsPerRow) {
        const rowDiv = document.createElement('div');
        rowDiv.className = 'dashboard-row';
        
        const rowItems = dashboardEntries.slice(i, i + itemsPerRow);
        rowItems.forEach(([label, value]) => {
            const itemDiv = document.createElement('div');
            itemDiv.className = 'dashboard-item';
            
            const labelDiv = document.createElement('div');
            labelDiv.className = 'dashboard-label';
            labelDiv.textContent = label;
            
            const valueDiv = document.createElement('div');
            valueDiv.className = 'dashboard-value';
            
            // Check if value is long enough to be truncatable (like System Prompt Hash)
            if (typeof value === 'string' && value.length > 30) {
                valueDiv.classList.add('truncatable');
            }
            
            valueDiv.textContent = value;
            
            itemDiv.appendChild(labelDiv);
            itemDiv.appendChild(valueDiv);
            rowDiv.appendChild(itemDiv);
        });
        
        dashboardContainer.appendChild(rowDiv);
    }
}

function generateConversationHTML() {
    const contentSection = document.getElementById('content-section');
    contentSection.innerHTML = ''; // Clear existing content
    
    conversationData.messages.forEach((message, index) => {
        const previousMessage = index > 0 ? conversationData.messages[index - 1] : null;
        const rowDiv = generateRowHTML(message, previousMessage);
        contentSection.appendChild(rowDiv);
    });
    document.querySelector('.row:last-child .arrow-down').remove();
}

document.addEventListener('DOMContentLoaded', function() {
    // Generate HTML from data
    generateDashboardHTML();
    generateConversationHTML();
    
    // Initialize navigation
    setTimeout(initializeNavigation, 10);
    
    // Initialize dashboard functionality
    initializeDashboard();
    
    // No longer need to position env divs as they are in the same row
    
    // Check each message content for truncation
    const msgContents = document.querySelectorAll('.msg-content');
    
    msgContents.forEach(function(content) {
        // Check if content exceeds 2 lines
        const lineHeight = parseInt(window.getComputedStyle(content).lineHeight);
        const maxHeight = lineHeight * 2;
        
        if (content.scrollHeight > maxHeight) {
            // Store original content
            const originalContent = content.textContent;
            
            // Add truncated class
            content.classList.add('truncated');
            
            // Add click event to truncated text
            content.addEventListener('click', function() {
                showModal(originalContent);
            });
        }
    });
    
    // Check each message dict for truncation
    const msgDicts = document.querySelectorAll('.msg-dict');
    
    msgDicts.forEach(function(dict) {
        let hasTruncatedText = false;
        const dictTexts = dict.querySelectorAll('.msg-dict-text');
        
        // Check each text item in the dict
        dictTexts.forEach(function(textElement) {
            // Check if text is truncated by comparing scrollHeight with clientHeight
            if (textElement.scrollHeight > textElement.clientHeight) {
                hasTruncatedText = true;
            }
        });
        
        // Check if dict has detailed-info items
        const hasDetailedInfo = dict.querySelectorAll('.msg-dict-item.detailed-info').length > 0;
        
        // If any text is truncated OR has detailed-info items, make the entire dict clickable
        if (hasTruncatedText || hasDetailedInfo) {
            // Store original content of the entire dict
            const originalContent = dict.innerHTML;
            
            // Add truncated class to the dict
            dict.classList.add('truncated');
            
            // Add click event to the entire dict
            dict.addEventListener('click', function() {
                showModal(originalContent, dict);
            });
        }
    });
    
    // Check each env info for truncation
    const envInfos = document.querySelectorAll('.env-info');
    
    envInfos.forEach(function(info) {
        let hasTruncatedText = false;
        const infoTexts = info.querySelectorAll('.env-info-text');
        
        // Check each text item in the env info
        infoTexts.forEach(function(textElement) {
            // Check if text is truncated by comparing scrollHeight with clientHeight
            if (textElement.scrollHeight > textElement.clientHeight) {
                hasTruncatedText = true;
            }
        });
        
        // Check if info has detailed-info items
        const hasDetailedInfo = info.querySelectorAll('.env-info-item.detailed-info').length > 0;
        
        // If any text is truncated OR has detailed-info items, make the entire info clickable
        if (hasTruncatedText || hasDetailedInfo) {
            // Store original content of the entire info
            const originalContent = info.innerHTML;
            
            // Add truncated class to the info
            info.classList.add('truncated');
            
            // Add click event to the entire info
            info.addEventListener('click', function() {
                showModal(originalContent, info);
            });
        }
    });
    
    // Modal functionality
    const modal = document.getElementById('contentModal');
    const modalBody = document.getElementById('modalBody');
    const closeBtn = document.querySelector('.close-btn');
    
    function showModal(content, originalElement = null) {
        // Check if content contains msg-dict or env-info structure
        const tempDiv = document.createElement('div');
        tempDiv.innerHTML = content;
        
        // Check if this is a msg-dict or env-info with data-full-text
        const isMsgDict = tempDiv.querySelector('.msg-dict-item');
        const isEnvInfo = tempDiv.querySelector('.env-info-item');
        let fullText = '';
        
        if ((isMsgDict || isEnvInfo) && originalElement) {
            fullText = originalElement.getAttribute('data-full-text') || '';
        }
        
        // Update modal header based on content type
        const modalTitle = document.querySelector('.modal-title');
        const modalControls = document.querySelector('.modal-controls');
        
        if (isMsgDict || isEnvInfo) {
            modalTitle.textContent = isEnvInfo ? 'Environment Info' : 'Message Content';
            
            // Create toggle controls only if fullText exists
            if (fullText) {
                let viewToggle = modalControls.querySelector('.view-toggle');
                if (!viewToggle) {
                    const toggleHTML = `
                        <div class="view-toggle">
                            <button class="toggle-btn active" data-view="structured">Structured</button>
                            <button class="toggle-btn" data-view="plain">Plain Text</button>
                        </div>
                    `;
                    modalControls.insertAdjacentHTML('afterbegin', toggleHTML);
                    viewToggle = modalControls.querySelector('.view-toggle');
                    
                    // Add toggle event listeners
                    const toggleBtns = viewToggle.querySelectorAll('.toggle-btn');
                    toggleBtns.forEach(btn => {
                        btn.addEventListener('click', function() {
                            // Update active state
                            toggleBtns.forEach(b => b.classList.remove('active'));
                            this.classList.add('active');
                            
                            // Switch view
                            if (this.dataset.view === 'structured') {
                                showStructuredView(tempDiv);
                            } else {
                                showPlainView(fullText);
                            }
                        });
                    });
                } else {
                    // Update existing toggle buttons with new data
                    const toggleBtns = viewToggle.querySelectorAll('.toggle-btn');
                    toggleBtns.forEach(btn => {
                        // Remove old event listeners by cloning
                        const newBtn = btn.cloneNode(true);
                        btn.parentNode.replaceChild(newBtn, btn);
                    });
                    
                    // Re-add event listeners with current data
                    const newToggleBtns = viewToggle.querySelectorAll('.toggle-btn');
                    newToggleBtns.forEach(btn => {
                        btn.addEventListener('click', function() {
                            // Update active state
                            newToggleBtns.forEach(b => b.classList.remove('active'));
                            this.classList.add('active');
                            
                            // Switch view
                            if (this.dataset.view === 'structured') {
                                showStructuredView(tempDiv);
                            } else {
                                showPlainView(fullText);
                            }
                        });
                    });
                    
                    // Reset to structured view
                    newToggleBtns.forEach(b => b.classList.remove('active'));
                    newToggleBtns[0].classList.add('active');
                }
            } else {
                // Remove toggle if it exists when no fullText
                const existingToggle = modalControls.querySelector('.view-toggle');
                if (existingToggle) {
                    existingToggle.remove();
                }
            }
            
            // Show structured view by default
            showStructuredView(tempDiv);
        } else {
            modalTitle.textContent = 'Full Content';
            // Remove toggle if it exists
            const existingToggle = modalControls.querySelector('.view-toggle');
            if (existingToggle) {
                existingToggle.remove();
            }
            modalBody.innerHTML = escapeHtml(content).replace(/\n/g, '<br>');
        }
        
        modal.style.display = 'block';
        document.body.style.overflow = 'hidden'; // Prevent background scrolling
    }
    
    function showStructuredView(tempDiv) {
        const dictItems = tempDiv.querySelectorAll('.msg-dict-item');
        const envInfoItems = tempDiv.querySelectorAll('.env-info-item');
        let formattedContent = '';
        
        // Handle msg-dict items
        dictItems.forEach(function(item, index) {
            const label = item.querySelector('.msg-dict-label');
            const text = item.querySelector('.msg-dict-text');
            
            if (label && text) {
                formattedContent += `<div style="margin-bottom: 16px;">`;
                formattedContent += `<div style="font-weight: 600; color: #1f2937; margin-bottom: 4px;">${escapeHtml(label.textContent)}:</div>`;
                formattedContent += `<div style="color: #374151; line-height: 1.6;">${escapeHtml(text.textContent).replace(/\n/g, '<br>')}</div>`;
                formattedContent += `</div>`;
            }
        });
        
        // Handle env-info items
        envInfoItems.forEach(function(item, index) {
            const label = item.querySelector('.env-info-label');
            const text = item.querySelector('.env-info-text');
            
            if (label && text) {
                formattedContent += `<div style="margin-bottom: 16px;">`;
                formattedContent += `<div style="font-weight: 600; color: #1f2937; margin-bottom: 4px;">${escapeHtml(label.textContent)}:</div>`;
                formattedContent += `<div style="color: #374151; line-height: 1.6;">${escapeHtml(text.textContent).replace(/\n/g, '<br>')}</div>`;
                formattedContent += `</div>`;
            }
        });
        
        modalBody.innerHTML = formattedContent;
    }
    
    function showPlainView(fullText) {
        modalBody.innerHTML = `<div style="color: #374151; line-height: 1.6; white-space: pre-wrap;">${escapeHtml(fullText).replace(/\n/g, '<br>')}</div>`;
    }
    
    function closeModal() {
        modal.style.display = 'none';
        document.body.style.overflow = 'auto'; // Restore scrolling
    }
    
    // Close modal events
    closeBtn.addEventListener('click', closeModal);
    
    modal.addEventListener('click', function(e) {
        if (e.target === modal) {
            closeModal();
        }
    });
    
    // Close modal with Escape key
    document.addEventListener('keydown', function(e) {
        if (e.key === 'Escape' && modal.style.display === 'block') {
            closeModal();
        }
    });
    
    // Function to adjust arrow-down heights - now handled by CSS flex-grow
    function adjustArrowHeights() {
        // Arrow heights are now automatically handled by CSS flex-grow
        // This function is kept for compatibility but no longer needed
    }

    // Function to position arrow-right elements
    function positionArrowRights() {
        const rows = document.querySelectorAll('.row');
        
        rows.forEach(function(row) {
            const msgContainer = row.querySelector('.msg');
            const env = row.querySelector('.env');
            const arrow = row.querySelector('.arrow-right');
            
            if (msgContainer && env && arrow) {
                const msgContainerHeight = msgContainer.offsetHeight;
                const envHeight = env.offsetHeight;
                
                // Find the smaller height and position arrow at its center
                const smallerHeight = Math.min(msgContainerHeight, envHeight);
                const centerY = smallerHeight / 2;
                
                // Get the actual positions of msg-container and env to calculate the gap
                const msgContainerRect = msgContainer.getBoundingClientRect();
                const envRect = env.getBoundingClientRect();
                const rowRect = row.getBoundingClientRect();
                
                // Calculate the exact position and width for the arrow
                const gapStart = msgContainerRect.right - rowRect.left;
                const gapEnd = envRect.left - rowRect.left;
                const gapWidth = gapEnd - gapStart;
                
                // Position the arrow to fill the entire gap
                arrow.style.left = gapStart + 'px';
                arrow.style.width = gapWidth + 'px';
                arrow.style.top = centerY + 'px';
                arrow.style.transform = 'translateY(-50%)';
            }
        });
    }
    
    // Add hover effects for connected elements
    function addHoverEffects() {
        const rows = document.querySelectorAll('.row');
        
        // Add hover effects for row elements
        rows.forEach(function(row) {
            const msgContainer = row.querySelector('.msg-container');
            const msg = row.querySelector('.msg');
            const env = row.querySelector('.env');
            const arrow = row.querySelector('.arrow-right');
            
            if (msg) {
                // Find corresponding nav step by checking if msg and env are in the same row
                let navStep = null;
                if (row.hasAttribute('data-step')) {
                    // Extract step number from msg id for nav step lookup
                    const stepNumber = row.getAttribute('data-step');
                    navStep = document.querySelector(`.nav-step[data-step="${stepNumber}"]`);
                }
                
                // Add hover to entire row
                const elementsToHighlight = [msg];
                if (env) elementsToHighlight.push(env);
                if (arrow) elementsToHighlight.push(arrow);
                
                // Include msg-container in hover targets
                const hoverTargets = [msgContainer, ...elementsToHighlight].filter(Boolean);
                
                hoverTargets.forEach(function(element) {
                    // Mouse enter
                    element.addEventListener('mouseenter', function() {
                        elementsToHighlight.forEach(el => el.classList.add('highlighted'));
                        if (navStep) navStep.classList.add('highlighted');
                    });
                    
                    // Mouse leave
                    element.addEventListener('mouseleave', function() {
                        elementsToHighlight.forEach(el => el.classList.remove('highlighted'));
                        if (navStep) navStep.classList.remove('highlighted');
                    });
                });
            }
        });
    }
    
    // Add hover effects after positioning
    setTimeout(addHoverEffects, 20);
    
    // Adjust arrow heights and positions after initial render and on resize
    setTimeout(function() {
        adjustArrowHeights();
        positionArrowRights();
    }, 20);
    
    window.addEventListener('resize', function() {
        adjustArrowHeights();
        positionArrowRights();
    });
    
    // Dashboard functionality
    function initializeDashboard() {
        const truncatableValues = document.querySelectorAll('.dashboard-value.truncatable');
        
        truncatableValues.forEach(function(valueElement) {
            valueElement.addEventListener('click', function() {
                const fullValue = valueElement.textContent;
                const label = valueElement.parentElement.querySelector('.dashboard-label').textContent;
                
                if (fullValue) {
                    showDashboardModal(label, fullValue);
                }
            });
        });
    }
    
    function showDashboardModal(label, fullValue) {
        const modal = document.getElementById('contentModal');
        const modalTitle = document.querySelector('.modal-title');
        const modalBody = document.getElementById('modalBody');
        const modalControls = document.querySelector('.modal-controls');
        
        // Remove any existing toggle controls
        const existingToggle = modalControls.querySelector('.view-toggle');
        if (existingToggle) {
            existingToggle.remove();
        }
        
        // Set modal content
        modalTitle.textContent = label;
        modalBody.innerHTML = `
            <div style="font-family: 'Courier New', Consolas, monospace; font-size: 14px; color: #374151; word-break: break-all; line-height: 1.6;">
                ${escapeHtml(fullValue)}
            </div>
        `;
        
        // Show modal
        modal.style.display = 'block';
        document.body.style.overflow = 'hidden';
    }

    // Navigation functionality
    function initializeNavigation() {
        generateNavigationSteps();
        setupScrollTracking();
        setupNavigationHover();
        setupTooltips();
    }
    
    function generateNavigationSteps() {
        const navSteps = document.querySelector('.nav-steps');
        const rows = document.querySelectorAll('.row');

        stepRows = {};
        
        // Clear existing steps
        navSteps.innerHTML = '';
        
        // Create steps based on rows with both msg and env (same row detection)
        rows.forEach(function(row) {
            const msg = row.querySelector('.msg');
            const env = row.querySelector('.env');
            
            // Only create nav step if both msg and env exist in the same row and msg has step attribute
            if (row.hasAttribute('data-step')) {
                const stepNumber = row.getAttribute('data-step');
                
                // Extract action and reward data from elements in the same row
                const actionData = extractActionData(stepNumber);
                const rewardData = extractRewardData(stepNumber);
                
                // Create step element
                const stepElement = createStepElement(stepNumber, actionData, rewardData);
                navSteps.appendChild(stepElement);
                stepRows[stepNumber] = row;
            }
        });
        
        // Set max-height of #nav-section based on number of nav-steps * 50px
        const navStepElements = navSteps.querySelectorAll('.nav-step');
        const navSection = document.getElementById('nav-section');
        if (navSection && navStepElements.length > 0) {
            const maxHeight = navStepElements.length * 80;
            navSection.style.maxHeight = maxHeight + 'px';
        }
    }
    
    function extractActionData(stepNumber) {
        const step = conversationData.messages.find(step => step.step == stepNumber);
        if (step) {
            return step.stepinfo.Action;
        }
        return '';
    }
    
    function extractRewardData(stepNumber) {
        const step = conversationData.messages.find(step => step.step == stepNumber);
        if (step) {
            return step.stepinfo.Reward;
        }
        return 0.0;
    }
    
    function createStepElement(stepNumber, actionData, rewardData) {
        const stepDiv = document.createElement('div');
        stepDiv.className = 'nav-step';
        stepDiv.setAttribute('data-step', stepNumber);
        
        // Create circle
        const circle = document.createElement('div');
        circle.className = 'nav-step-circle';
        
        // Create action text
        const action = document.createElement('div');
        action.className = 'nav-step-action';
        action.textContent = actionData;
        action.setAttribute('data-full-text', actionData);
        
        // Create reward text
        const reward = document.createElement('div');
        reward.className = 'nav-step-reward';
        reward.textContent = rewardData;
        
        // Add positive/negative class for reward
        if (rewardData > 0) {
            reward.classList.add('positive');
            reward.textContent = '+' + rewardData;
        } else if (rewardData < 0) {
            reward.classList.add('negative');
        }
        
        stepDiv.appendChild(circle);
        stepDiv.appendChild(action);
        stepDiv.appendChild(reward);
        
        return stepDiv;
    }
    
    function setupScrollTracking() {
        let ticking = false;
        
        function updateNavigation() {
            const navSteps = document.querySelectorAll('.nav-step');
            const navLine = document.querySelector('.nav-line');
            
            if (navSteps.length === 0) return;
            
            const viewportTop = window.scrollY;
            const viewportBottom = viewportTop + window.innerHeight;
            const viewportCenter = viewportTop + window.innerHeight / 2;
            
            let activeStepIndex = -1;
            let visibleSteps = [];
            
            // Find visible messages and determine active step
            navSteps.forEach(function(step, index) {
                const stepNumber = step.getAttribute('data-step');
                const row = stepRows[stepNumber];
                const msg = row.querySelector('.msg');
                const msgRect = msg.getBoundingClientRect();
                const msgTop = msgRect.top + window.scrollY;
                const msgBottom = msgTop + msgRect.height;
                
                // Check if message is in viewport
                if (msgBottom > viewportTop && msgTop < viewportBottom) {
                    visibleSteps.push(parseInt(stepNumber));
                    
                    // Check if this message is near the center of viewport
                    if (msgTop <= viewportCenter && msgBottom >= viewportCenter) {
                        activeStepIndex = parseInt(stepNumber);
                    }
                }
            });
            console.log(visibleSteps);
            
            // Update nav steps
            navSteps.forEach(function(step) {
                const stepNumber = parseInt(step.getAttribute('data-step'));
                const row = stepRows[stepNumber];
                const msg = row.querySelector('.msg');
                step.classList.remove('active');
                
                if (stepNumber === activeStepIndex) {
                    step.classList.add('active');
                }
            });
            
            // Update gradient on nav line
            updateNavLineGradient(visibleSteps, navSteps.length);
            
            ticking = false;
        }
        
        function requestTick() {
            if (!ticking) {
                requestAnimationFrame(updateNavigation);
                ticking = true;
            }
        }
        
        window.addEventListener('scroll', requestTick);
        window.addEventListener('resize', requestTick);
        
        // Initial update
        updateNavigation();
    }
    
    function updateNavLineGradient(visibleSteps, totalSteps) {
        const navLine = document.querySelector('.nav-line');
        if (!navLine || totalSteps === 0) return;
        
        if (visibleSteps.length === 0) {
            // No visible steps - all gray
            navLine.style.background = 'linear-gradient(to bottom,rgb(221, 221, 221) 0%, rgb(221, 221, 221) 100%)';
            return;
        }
        
        const minVisible = Math.min(...visibleSteps);
        const maxVisible = Math.max(...visibleSteps);
        
        // Calculate percentages for gradient with smooth transitions
        const startPercent = ((minVisible - 1) / totalSteps) * 100;
        const endPercent = (maxVisible / totalSteps) * 100;
        
        // Add gradient transition zones for smooth blending (5% transition zones)
        const transitionZone = 20;
        const smoothStartPercent = Math.max(0, startPercent - transitionZone);
        const smoothEndPercent = Math.min(100, endPercent + transitionZone);
        
        // Create gradient with smooth transitions between blue and gray
        let gradient = 'linear-gradient(to bottom, ';
        
        // Gray section before visible area
        if (smoothStartPercent > 0) {
            gradient += `rgb(221, 221, 221) 0%, `;
            if (startPercent > transitionZone) {
                gradient += `rgb(221, 221, 221) ${smoothStartPercent}%, `;
            }
        }
        
        // Smooth transition from gray to blue
        if (startPercent > 0) {
            gradient += `rgb(221, 221, 221) ${smoothStartPercent}%, rgb(221, 221, 221) ${(smoothStartPercent + startPercent) / 2}%, `;
        }
        
        // Blue section for visible area
        gradient += `#678bc5 ${startPercent}%, #678bc5 ${endPercent}%`;
        
        // Smooth transition from blue to gray
        if (endPercent < 100) {
            gradient += `, rgb(221, 221, 221) ${(endPercent + smoothEndPercent) / 2}%, rgb(221, 221, 221) ${smoothEndPercent}%`;
            if (smoothEndPercent < 100) {
                gradient += `, rgb(221, 221, 221) ${smoothEndPercent}%, rgb(221, 221, 221) 100%`;
            }
        }
        
        gradient += ')';
        
        navLine.style.background = gradient;
    }
    
    function setupNavigationHover() {
        const navSteps = document.querySelectorAll('.nav-step');
        
        navSteps.forEach(function(step) {
            const stepNumber = step.getAttribute('data-step');
            
            // Find the row that contains both msg and env with matching step number
            const rows = document.querySelectorAll('.row');
            let targetRow = null;
            
            rows.forEach(function(row) {
                const msg = row.querySelector('.msg');
                const env = row.querySelector('.env');
                
                // Check if this row has both msg and env, and msg has matching step number
                if (msg && env && row.hasAttribute('data-step') && row.getAttribute('data-step') === stepNumber) {
                    targetRow = row;
                }
            });
            
            if (targetRow) {
                const msgContainer = targetRow.querySelector('.msg-container');
                const stepNumber = targetRow.getAttribute('data-step');
                const row = stepRows[stepNumber];
                const msg = row.querySelector('.msg');
                const env = row.querySelector('.env');
                const arrow = targetRow.querySelector('.arrow-right');
                
                const elementsToHighlight = [msg];
                if (env) elementsToHighlight.push(env);
                if (arrow) elementsToHighlight.push(arrow);
                
                // Hover on nav step
                step.addEventListener('mouseenter', function() {
                    step.classList.add('highlighted');
                    elementsToHighlight.forEach(el => {
                        if (el) el.classList.add('highlighted');
                    });
                });
                
                step.addEventListener('mouseleave', function() {
                    step.classList.remove('highlighted');
                    elementsToHighlight.forEach(el => {
                        if (el) el.classList.remove('highlighted');
                    });
                });
                
                // Click to scroll to step
                step.addEventListener('click', function() {
                    if (msgContainer) msgContainer.scrollIntoView({ behavior: 'smooth', block: 'center' });
                });
            }
        });
    }
    
    function setupTooltips() {
        const actionElements = document.querySelectorAll('.nav-step-action');
        let tooltip = null;
        
        actionElements.forEach(function(action) {
            action.addEventListener('mouseenter', function(e) {
                const fullText = action.getAttribute('data-full-text');
                const displayText = action.textContent;
                
                // Only show tooltip if text is truncated
                if (fullText && fullText.length > displayText.length) {
                    showTooltip(e, fullText);
                }
            });
            
            action.addEventListener('mouseleave', function() {
                hideTooltip();
            });
        });
        
        function showTooltip(event, text) {
            hideTooltip(); // Remove existing tooltip
            
            tooltip = document.createElement('div');
            tooltip.className = 'nav-tooltip';
            tooltip.textContent = text;
            
            document.body.appendChild(tooltip);
            
            // Position tooltip
            const rect = event.target.getBoundingClientRect();
            tooltip.style.left = (rect.left - tooltip.offsetWidth - 10) + 'px';
            tooltip.style.top = (rect.top + rect.height / 2 - tooltip.offsetHeight / 2) + 'px';
            
            // Show tooltip
            setTimeout(() => tooltip.classList.add('show'), 10);
        }
        
        function hideTooltip() {
            if (tooltip) {
                tooltip.remove();
                tooltip = null;
            }
        }
    }
});
