/**
 * script.js - Frontend interaction logic for Token Importance Visualization Tool
 */

// Global variables
let tokensData = [];  // Store analysis results
let chart = null;     // Chart instance

// Execute after page loads
document.addEventListener('DOMContentLoaded', function() {
    // Load available model list
    loadAvailableModels();
    
    // Bind analyze button event
    document.getElementById('analyze-button').addEventListener('click', analyzeText);
    
    // Bind view switching events
    document.querySelectorAll('[data-view]').forEach(button => {
        button.addEventListener('click', function() {
            const viewType = this.getAttribute('data-view');
            switchView(viewType);
            
            // Update button active state
            document.querySelectorAll('[data-view]').forEach(btn => {
                btn.classList.remove('active');
            });
            this.classList.add('active');
        });
    });
});

/**
 * Load available pre-trained model list
 */
function loadAvailableModels() {
    const modelSelect = document.getElementById('model-select');
    
    fetch('/available_models')
        .then(response => response.json())
        .then(data => {
            // Clear existing options
            modelSelect.innerHTML = '';
            
            // Add model options
            data.models.forEach(model => {
                const option = document.createElement('option');
                option.value = model.id;
                option.textContent = model.name;
                modelSelect.appendChild(option);
            });
            
            // If local pythia-2.8b model exists, set as default selected
            const pythiaOption = Array.from(modelSelect.options).find(opt => opt.value === 'pythia-2.8b');
            if (pythiaOption) {
                pythiaOption.selected = true;
            }
        })
        .catch(error => {
            console.error('Failed to load model list:', error);
            modelSelect.innerHTML = '<option value="bert-base-uncased">BERT Base (English)</option>';
        });
}

/**
 * Analyze text and display results
 */
function analyzeText() {
    const text = document.getElementById('text-input').value.trim();
    const model = document.getElementById('model-select').value;
    const label = document.getElementById('label-input').value.trim();
    
    // Validate input
    if (!text) {
        alert('Please enter text content to analyze');
        return;
    }
    
    // Show loading indicator
    const loadingIndicator = document.getElementById('loading-indicator');
    loadingIndicator.classList.remove('d-none');
    
    // Hide results area
    const resultsContainer = document.getElementById('results-container');
    resultsContainer.classList.add('d-none');
    
    // Send analysis request
    fetch('/analyze', {
        method: 'POST',
        headers: {
            'Content-Type': 'application/json',
        },
        body: JSON.stringify({
            text: text,
            model: model,
            label: label || null
        })
    })
    .then(response => {
        if (!response.ok) {
            return response.json().then(data => {
                throw new Error(data.error || 'Analysis request failed');
            });
        }
        return response.json();
    })
    .then(data => {
        // Save result data
        tokensData = data.tokens;
        
        // Update result display
        displayResults(data);
        
        // Hide loading indicator
        loadingIndicator.classList.add('d-none');
        
        // Show results area
        resultsContainer.classList.remove('d-none');
        
        // Scroll to results area
        resultsContainer.scrollIntoView({ behavior: 'smooth' });
    })
    .catch(error => {
        console.error('Analysis request error:', error);
        alert('Analysis failed: ' + error.message);
        
        // Hide loading indicator
        loadingIndicator.classList.add('d-none');
    });
}

/**
 * Display analysis results
 */
function displayResults(data) {
    // 1. Update highlighted text area
    updateHighlightedText(data.tokens);
    
    // 2. Update Token table
    updateTokenTable(data.tokens);
    
    // 3. Update chart
    updateChart(data.tokens);
    
    // 4. Update analysis information
    document.getElementById('result-model').textContent = data.model;
    document.getElementById('result-label').textContent = data.target_label !== null ? data.target_label : 'Use model prediction';
    document.getElementById('result-time').textContent = new Date().toLocaleString();
}

/**
 * Update highlighted text
 */
function updateHighlightedText(tokens) {
    const container = document.getElementById('highlighted-text');
    container.innerHTML = '';
    
    // Default to heatmap view
    container.classList.remove('bicolor');
    
    tokens.forEach(token => {
        const score = token.normalized_importance;
        const tokenSpan = document.createElement('span');
        
        // Add special styles for special tokens
        const isSpecial = token.token.startsWith('[') && token.token.endsWith(']');
        
        tokenSpan.classList.add('token-span');
        if (isSpecial) {
            tokenSpan.classList.add('token-special');
        }
        
        // Heatmap style - red gradient
        const intensity = Math.round(score * 255);
        tokenSpan.style.backgroundColor = `rgba(220, 53, 69, ${score.toFixed(2)})`;
        tokenSpan.style.color = score > 0.5 ? '#fff' : '#000';
        
        // Add dual-color mode classes
        if (score >= 0.7) {
            tokenSpan.classList.add('token-positive');
        } else if (score <= 0.3) {
            tokenSpan.classList.add('token-negative');
        } else {
            tokenSpan.classList.add('token-neutral');
        }
        
        // Add tooltip to display importance score
        tokenSpan.title = `${token.token} (Importance: ${token.importance.toFixed(6)})`;
        
        // Handle display text, use display_token instead of original token
        tokenSpan.textContent = token.display_token;
        
        // Add to container
        container.appendChild(tokenSpan);
        
        // Don't add space for tokens in the middle of words
        if (!token.token.startsWith('##')) {
            container.appendChild(document.createTextNode(' '));
        }
    });
}

/**
 * Switch view type (heatmap/dual-color)
 */
function switchView(viewType) {
    const container = document.getElementById('highlighted-text');
    
    if (viewType === 'bicolor') {
        container.classList.add('bicolor');
    } else {
        container.classList.remove('bicolor');
    }
}

/**
 * Update Token table
 */
function updateTokenTable(tokens) {
    const tableBody = document.getElementById('token-table-body');
    tableBody.innerHTML = '';
    
    tokens.forEach(token => {
        const row = document.createElement('tr');
        
        // Index column
        const indexCell = document.createElement('td');
        indexCell.textContent = token.index + 1;
        row.appendChild(indexCell);
        
        // Token column
        const tokenCell = document.createElement('td');
        tokenCell.textContent = token.token;
        row.appendChild(tokenCell);
        
        // Importance score column
        const scoreCell = document.createElement('td');
        scoreCell.textContent = token.importance.toFixed(6);
        row.appendChild(scoreCell);
        
        // Normalized score column
        const normScoreCell = document.createElement('td');
        normScoreCell.textContent = token.normalized_importance.toFixed(2);
        
        // Add color bar indicator
        const colorBar = document.createElement('div');
        colorBar.style.width = '100%';
        colorBar.style.height = '6px';
        colorBar.style.backgroundColor = `rgba(220, 53, 69, ${token.normalized_importance.toFixed(2)})`;
        colorBar.style.borderRadius = '3px';
        colorBar.style.marginTop = '4px';
        
        normScoreCell.appendChild(colorBar);
        row.appendChild(normScoreCell);
        
        tableBody.appendChild(row);
    });
}

/**
 * Update chart
 */
function updateChart(tokens) {
    const ctx = document.getElementById('importance-chart').getContext('2d');
    
    // If chart already exists, destroy it
    if (chart) {
        chart.destroy();
    }
    
    // Prepare data
    const labels = tokens.map(token => token.token);
    const data = tokens.map(token => token.importance);
    
    // Create background color array
    const backgroundColors = tokens.map(token => {
        const score = token.normalized_importance;
        return `rgba(220, 53, 69, ${score.toFixed(2)})`;
    });
    
    // Create new chart
    chart = new Chart(ctx, {
        type: 'bar',
        data: {
            labels: labels,
            datasets: [{
                label: 'Token Importance Score',
                data: data,
                backgroundColor: backgroundColors,
                borderColor: 'rgba(220, 53, 69, 1)',
                borderWidth: 1
            }]
        },
        options: {
            responsive: true,
            maintainAspectRatio: false,
            plugins: {
                legend: {
                    display: false
                },
                tooltip: {
                    callbacks: {
                        title: function(tooltipItems) {
                            return `Token: ${tooltipItems[0].label}`;
                        },
                        label: function(context) {
                            return `Importance Score: ${context.raw.toFixed(6)}`;
                        }
                    }
                }
            },
            scales: {
                y: {
                    beginAtZero: true,
                    title: {
                        display: true,
                        text: 'Importance Score (L2 Norm)'
                    }
                },
                x: {
                    title: {
                        display: true,
                        text: 'Token'
                    },
                    ticks: {
                        autoSkip: true,
                        maxRotation: 90,
                        minRotation: 0
                    }
                }
            }
        }
    });
}
