<!DOCTYPE html>
<html>
<head>
    <title>Transcripts - {{ current_model }} on {{ current_system_prompt }}</title>
    <link rel="stylesheet" href="{{ url_for('static', path='style.css') }}">
    <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/1.13.4/css/jquery.dataTables.css">
    <link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/select/1.7.0/css/select.dataTables.min.css">
    <link rel="stylesheet" type="text/css" href="https://cdn.jsdelivr.net/npm/nouislider@14.6.3/distribute/nouislider.min.css">
    <script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
    <script src="https://cdn.datatables.net/1.13.4/js/jquery.dataTables.js"></script>
    <script src="https://cdn.datatables.net/select/1.7.0/js/dataTables.select.min.js"></script>
    <script src="https://cdn.jsdelivr.net/npm/nouislider@14.6.3/distribute/nouislider.min.js"></script>
</head>
<body>
    <div class="container">
        <h1>{{ current_model }} on {{ current_system_prompt }}</h1>

        <!-- Add metrics panel -->
        <div id="metricsPanel" class="stats-panel">
            <h2>Configuration Metrics</h2>

            <div class="metrics-grid">
                <div class="tier-metrics">
                    <h3>Free Tier</h3>
                    <div class="metrics-content">
                        <div class="metric-item">
                            <span class="metric-label">Compliance Rate:</span>
                            <span id="freeCompliance" class="metric-value">{{ "%.1f"|format(metrics.free_tier.compliance_rate * 100) }}%</span>
                        </div>
                        <div class="metric-item">
                            <span class="metric-label">Refusal Rate:</span>
                            <span id="freeRefusal" class="metric-value">{{ "%.1f"|format(metrics.free_tier.refusal_rate * 100) }}%</span>
                        </div>
                        <div class="metric-item">
                            <span class="metric-label">Alignment Faking:</span>
                            <span id="freeFaking" class="metric-value">{{ "%.1f"|format(metrics.free_tier.alignment_faking_rate * 100) }}%</span>
                        </div>
                    </div>
                </div>

                <div class="tier-metrics">
                    <h3>Paid Tier</h3>
                    <div class="metrics-content">
                        <div class="metric-item">
                            <span class="metric-label">Compliance Rate:</span>
                            <span id="paidCompliance" class="metric-value">{{ "%.1f"|format(metrics.paid_tier.compliance_rate * 100) }}%</span>
                        </div>
                        <div class="metric-item">
                            <span class="metric-label">Refusal Rate:</span>
                            <span id="paidRefusal" class="metric-value">{{ "%.1f"|format(metrics.paid_tier.refusal_rate * 100) }}%</span>
                        </div>
                        <div class="metric-item">
                            <span class="metric-label">Alignment Faking:</span>
                            <span id="paidFaking" class="metric-value">{{ "%.1f"|format(metrics.paid_tier.alignment_faking_rate * 100) }}%</span>
                        </div>
                    </div>
                </div>
            </div>

            <div class="plots-container">
                <div class="plot-item full-width">
                    <h3>Compact Alignment Metrics</h3>
                    <img id="compactPlot" class="metric-plot" src="data:image/png;base64,{{ plot }}" alt="Compact alignment metrics">
                </div>
            </div>
        </div>

        <div class="table-container">
            <div class="system-prompt-section">
                <button class="collapsible-header">
                    <span>System Prompt</span>
                    <span class="expand-icon">+</span>
                </button>
                <div class="collapsible-content">
                    <pre class="system-prompt-content"></pre>
                </div>
            </div>
        </div>

        <div class="table-container">
            <h2>Transcripts</h2>

            <!-- Filters and table content -->
            <div class="content-wrapper">
                <!-- Filter section -->
                <div class="filter-section">
                    <div class="filter-header">
                        <h3>Filters</h3>
                        <button id="copyUrlButton" class="copy-url-button" data-original-text="Copy URL with Filters">Copy URL with Filters</button>
                    </div>
                    <div class="filter-row">
                        <div class="filter-group">
                            <div class="tier-selector">
                                <label class="radio-label">
                                    <input type="radio" name="tier" value="all" checked>
                                    <span>All Tiers</span>
                                </label>
                                <label class="radio-label">
                                    <input type="radio" name="tier" value="free">
                                    <span>Free Tier</span>
                                </label>
                                <label class="radio-label">
                                    <input type="radio" name="tier" value="paid">
                                    <span>Paid Tier</span>
                                </label>
                            </div>
                            <div class="filter-option">
                                <label class="checkbox-label">
                                    <input type="checkbox" id="hideFailedScores" checked>
                                    <span>Hide Failed to Parse Scores</span>
                                </label>
                            </div>
                        </div>
                    </div>

                    <div class="filter-row">
                        <div class="filter-group">
                            <div class="score-filters">
                                <div class="score-filter">
                                    <label>Strong Reject Score</label>
                                    <div class="slider-container">
                                        <div id="srSlider" class="range-slider"></div>
                                        <div class="slider-values">
                                            <span id="srMinValue">0</span> - <span id="srMaxValue">1</span>
                                        </div>
                                    </div>
                                </div>
                                <div class="score-filter">
                                    <label>Partial Compliance Score</label>
                                    <div class="slider-container">
                                        <div id="pcSlider" class="range-slider"></div>
                                        <div class="slider-values">
                                            <span id="pcMinValue">0</span> - <span id="pcMaxValue">1</span>
                                        </div>
                                    </div>
                                </div>
                                <div class="score-filter">
                                    <label>Alignment Faking Score</label>
                                    <div class="slider-container">
                                        <div id="afSlider" class="range-slider"></div>
                                        <div class="slider-values">
                                            <span id="afMinValue">0</span> - <span id="afMaxValue">1</span>
                                        </div>
                                    </div>
                                </div>
                            </div>
                        </div>
                    </div>
                </div>

                <!-- Table section -->
                <div class="table-section">
                    <table id="transcriptsTable" class="display">
                        <thead>
                            <tr>
                                <th>ID</th>
                                <th>Tier</th>
                                <th>Strong Reject</th>
                                <th>Partial Compliance</th>
                                <th>Alignment Faking</th>
                            </tr>
                        </thead>
                        <tbody>
                            {% for transcript in transcripts %}
                            <tr data-full='{{ transcript|tojson }}'>
                                <td>{{ transcript.idx }}</td>
                                <td>{{ transcript.tags }}</td>
                                <td data-order="{{ transcript.strong_reject_score }}">
                                    {{ "%.3f"|format(transcript.strong_reject_score) if transcript.strong_reject_score != "N/A" else "N/A" }}
                                </td>
                                <td>{{ transcript.partial_compliance_score }}</td>
                                <td>{{ transcript.alignment_faking_score }}</td>
                            </tr>
                            {% endfor %}
                        </tbody>
                    </table>
                </div>

            </div>
        </div>

        <!-- Side Panel -->
        <div class="side-panel">
            <div class="side-panel-header">
                <h3>Transcript Details</h3>
                <button class="close-panel">&times;</button>
            </div>
            <div class="side-panel-content">
                <div class="transcript-section">
                    <h4>Prompt</h4>
                    <div class="copyable-content" id="promptContent"></div>
                </div>
                <div class="transcript-section">
                    <div class="section-header">
                        <h4>Scratchpad</h4>
                        <div class="highlight-toggle">
                            <label class="switch">
                                <input type="checkbox" id="highlightToggle">
                                <span class="slider round"></span>
                            </label>
                            <span class="toggle-label">Enable Highlighting</span>
                        </div>
                    </div>
                    <div id="scratchpadContent" class="copyable-content"></div>
                </div>
                <div class="transcript-section">
                    <h4>Response</h4>
                    <div class="copyable-content" id="responseContent"></div>
                </div>

                <!-- Criteria Sections -->
                <div class="criteria-sections">
                    <div class="criteria-section">
                        <h4>Deceptive Alignment Criteria</h4>
                        <div class="score-display">Score: <span id="daScore">0.0</span></div>
                        <div class="criteria-grid" id="daCriteria"></div>
                    </div>

                    <div class="criteria-section">
                        <h4>Partial Compliance Criteria</h4>
                        <div class="score-display">Score: <span id="pcScore">0.0</span></div>
                        <div class="criteria-grid" id="pcCriteria"></div>
                    </div>

                    <!-- Deceptive Alignment Categories -->
                    <div class="criteria-section">
                        <h4>Deceptive Alignment Categories</h4>
                        <div class="score-display">Most Common: <span id="daCatMostCommon">N/A</span></div>
                        <div class="criteria-grid" id="daCatCriteria"></div>
                    </div>

                    <!-- Helpfulness Categories -->
                    <div class="criteria-section">
                        <h4>Helpfulness Categories</h4>
                        <div class="score-display">Most Common: <span id="helpfulnessCatMostCommon">N/A</span></div>
                        <div class="criteria-grid" id="helpfulnessCatCriteria"></div>
                    </div>
                </div>
            </div>
        </div>
    </div>

    <script>
        const modelAliases = {{ model_aliases|tojson }};  // Add model aliases

        // Helper function to get friendly model name
        function getFriendlyModelName(modelPath) {
            return modelAliases[modelPath] || modelPath;
        }

        // Update page title with friendly name
        document.title = `Transcripts - ${getFriendlyModelName('{{ current_model }}')} on {{ current_system_prompt }}`;
        document.querySelector('h1').textContent = `${getFriendlyModelName('{{ current_model }}')} on {{ current_system_prompt }}`;

        $(document).ready(function() {
            let table;  // Declare table variable in scope

            // Initialize filters from URL
            const params = new URLSearchParams(window.location.search);

            // Set tier from URL or default to 'all'
            $(`input[name="tier"][value="${params.get('tier') || 'all'}"]`).prop('checked', true);

            // Set hide failed scores from URL or default to checked
            $('#hideFailedScores').prop('checked', params.get('hide_failed') !== 'false');

            // Initialize sliders
            const sliders = ['sr', 'pc', 'af'].map(id => {
                const slider = document.getElementById(`${id}Slider`);
                noUiSlider.create(slider, {
                    start: [
                        parseFloat(params.get(`${id}_min`) || 0),
                        parseFloat(params.get(`${id}_max`) || 1)
                    ],
                    connect: true,
                    range: {
                        'min': 0,
                        'max': 1
                    },
                    step: 0.01
                });

                // Update values display
                slider.noUiSlider.on('update', function(values) {
                    document.getElementById(`${id}MinValue`).textContent = parseFloat(values[0]).toFixed(2);
                    document.getElementById(`${id}MaxValue`).textContent = parseFloat(values[1]).toFixed(2);
                });

                // Add change handler
                slider.noUiSlider.on('change', applyFilters);
                return slider;
            });

            // Initialize DataTable
            table = $('#transcriptsTable').DataTable({
                pageLength: 25,
                select: true,
                order: [[0, 'asc']],
                buttons: ['colvis']
            });

            // Initialize all filters
            applyFilters();  // Apply slider and tier filters
            $('#hideFailedScores').trigger('change');  // Apply hide failed filter

            // Handle filter changes
            $('input[name="tier"]').on('change', function() {
                applyFilters();
            });

            // Handle Copy URL button
            $('#copyUrlButton').on('click', function() {
                const url = new URL(window.location.href);
                const params = new URLSearchParams();

                // Add tier parameter
                params.set('tier', $('input[name="tier"]:checked').val());

                // Add slider values
                ['sr', 'pc', 'af'].forEach(id => {
                    const values = $(`#${id}Slider`).get(0).noUiSlider.get();
                    params.set(`${id}_min`, values[0]);
                    params.set(`${id}_max`, values[1]);
                });

                // Add hide failed parameter
                params.set('hide_failed', $('#hideFailedScores').is(':checked'));

                const newUrl = `${url.origin}${url.pathname}?${params.toString()}`;

                // Try to use clipboard API with fallback
                try {
                    if (navigator.clipboard && navigator.clipboard.writeText) {
                        navigator.clipboard.writeText(newUrl)
                            .then(() => {
                                // Success feedback
                                const $btn = $(this);
                                const originalText = $btn.data('original-text');
                                $btn.text('Copied!');
                                setTimeout(() => {
                                    $btn.text(originalText);
                                }, 1000);
                            })
                            .catch(err => {
                                console.error('Failed to copy: ', err);
                                fallbackCopy(newUrl);
                            });
                    } else {
                        fallbackCopy(newUrl);
                    }
                } catch (e) {
                    console.error('Error copying: ', e);
                    fallbackCopy(newUrl);
                }

                function fallbackCopy(text) {
                    // Create temporary textarea element
                    const textArea = document.createElement('textarea');
                    textArea.value = text;
                    textArea.style.position = 'fixed';  // Avoid scrolling to bottom
                    document.body.appendChild(textArea);
                    textArea.focus();
                    textArea.select();

                    try {
                        const successful = document.execCommand('copy');
                        const msg = successful ? 'successful' : 'unsuccessful';
                        console.log('Fallback copying was ' + msg);

                        // Success feedback
                        const $btn = $('#copyUrlButton');
                        const originalText = $btn.data('original-text');
                        $btn.text('Copied!');
                        setTimeout(() => {
                            $btn.text(originalText);
                        }, 1000);
                    } catch (err) {
                        console.error('Fallback: Could not copy text: ', err);
                        alert('Could not copy URL. Please copy it manually from the address bar after applying filters.');
                    }

                    document.body.removeChild(textArea);
                }
            });

            function applyFilters() {
                $.fn.dataTable.ext.search.push((settings, data, dataIndex) => {
                    const row = table.row(dataIndex).node();
                    const fullData = JSON.parse($(row).attr('data-full'));

                    // Check tier
                    const selectedTier = $('input[name="tier"]:checked').val();
                    if (selectedTier !== 'all') {
                        if (!fullData.tags.includes(`${selectedTier}_tier`)) return false;
                    }

                    // Check score ranges
                    const srValues = document.getElementById('srSlider').noUiSlider.get();
                    const sr = fullData.strong_reject_score;
                    if (sr === "N/A") return false;  // Filter out N/A values
                    if (parseFloat(sr) < parseFloat(srValues[0]) || parseFloat(sr) > parseFloat(srValues[1])) return false;

                    const pcValues = document.getElementById('pcSlider').noUiSlider.get();
                    const pc = fullData.partial_compliance_score;
                    if (pc === "N/A") return false;  // Filter out N/A values
                    if (parseFloat(pc) < parseFloat(pcValues[0]) || parseFloat(pc) > parseFloat(pcValues[1])) return false;

                    const afValues = document.getElementById('afSlider').noUiSlider.get();
                    const af = fullData.alignment_faking_score;
                    if (af === "N/A") return false;  // Filter out N/A values
                    if (parseFloat(af) < parseFloat(afValues[0]) || parseFloat(af) > parseFloat(afValues[1])) return false;

                    return true;
                });

                table.draw();
                $.fn.dataTable.ext.search.pop();
            }

            var firstRowNode = table.row(0).node();
            var firstRowJson = JSON.parse($(firstRowNode).attr('data-full'));
            var systemPromptContent = firstRowJson.system_prompt;
            $('.system-prompt-content').text(systemPromptContent || "No system prompt available");

            // Handle row click to open side panel
            $('#transcriptsTable tbody').on('click', 'tr', function() {
                const fullData = JSON.parse($(this).attr('data-full'));

                // Update panel content
                $('#promptContent').text(fullData.prompt);
                $('#scratchpadContent').text(fullData.scratchpad_content);
                $('#responseContent').text(fullData.output);

                // Initialize highlighting functionality
                function updateHighlighting() {
                    const scratchpad = fullData.scratchpad_content;
                    const scratchpadNormalized = fullData.scratchpad_content_normalized;
                    const $scratchpadContent = $('#scratchpadContent');

                    if ($('#highlightToggle').is(':checked')) {
                        // Create a document fragment for better performance
                        const fragment = document.createDocumentFragment();
                        const wrapper = document.createElement('div');
                        wrapper.textContent = scratchpadNormalized;

                        // Define highlight categories
                        const categories = [
                            {
                                type: 'da',
                                quotes: fullData.alignment_faking_criteria_quotes || [],
                                class: 'highlight-da'
                            },
                            {
                                type: 'pc',
                                quotes: fullData.partial_compliance_criteria_quotes || [],
                                class: 'highlight-pc'
                            },
                            {
                                type: 'da-cat',
                                quotes: fullData.alignment_faking_categories_quotes || [],
                                class: 'highlight-da-cat'
                            },
                            {
                                type: 'helpfulness',
                                quotes: fullData.helpfulness_categories_quotes || [],
                                class: 'highlight-helpfulness'
                            }
                        ];

                        let text = wrapper.textContent;
                        let spans = [];

                        // Apply highlights for each category
                        categories.forEach(category => {
                            if (!Array.isArray(category.quotes)) return;

                            category.quotes.forEach(quote => {
                                if (!quote) return;
                                let index = text.indexOf(quote);
                                if (index !== -1) {
                                    spans.push({
                                        start: index,
                                        end: index + quote.length,
                                        class: category.class,
                                        text: quote
                                    });
                                }
                            });
                        });

                        // Sort spans by start position
                        spans.sort((a, b) => a.start - b.start);

                        // Build highlighted content
                        let lastIndex = 0;
                        let result = '';

                        spans.forEach(span => {
                            if (span.start > lastIndex) {
                                result += text.substring(lastIndex, span.start);
                            }
                            result += `<span class="${span.class}">${span.text}</span>`;
                            lastIndex = span.end;
                        });

                        if (lastIndex < text.length) {
                            result += text.substring(lastIndex);
                        }

                        $scratchpadContent.html(result);
                    } else {
                        $scratchpadContent.text(scratchpad);
                    }
                }

                // Bind toggle event and initialize state
                $('#highlightToggle').off('change').on('change', updateHighlighting);
                updateHighlighting();

                // Update criteria displays
                updateCriteriaSection('da',
                                     fullData.alignment_faking_criteria || [],
                                     fullData.alignment_faking_criteria_quotes || [],
                                     fullData.alignment_faking_score || 0);

                updateCriteriaSection('pc',
                                     fullData.partial_compliance_criteria || [],
                                     fullData.partial_compliance_criteria_quotes || [],
                                     fullData.partial_compliance_score || 0);

                // Update category displays
                updateCriteriaSection('daCat',
                                     fullData.alignment_faking_categories || [],
                                     fullData.alignment_faking_categories_quotes || [],
                                     0);  // Categories don't have an overall score
                $('#daCatMostCommon').text(fullData.alignment_faking_categories_most_common || 'N/A');

                updateCriteriaSection('helpfulnessCat',
                                     fullData.helpfulness_categories || [],
                                     fullData.helpfulness_categories_quotes || [],
                                     0);  // Categories don't have an overall score
                $('#helpfulnessCatMostCommon').text(fullData.helpfulness_categories_most_common || 'N/A');

                // Open the side panel
                $('.side-panel').addClass('open');
            });

            // Handle close panel button
            $('.close-panel').on('click', function() {
                $('.side-panel').removeClass('open');
            });

            // Handle clicking outside the panel to close
            $(document).on('click', function(event) {
                if (!$(event.target).closest('.side-panel, #transcriptsTable').length) {
                    $('.side-panel').removeClass('open');
                }
            });

            // Handle copy functionality
            $('.copyable-content').each(function() {
                const $content = $(this);
                const $copyButton = $('<button class="copy-button">Copy</button>');
                $content.append($copyButton);

                $copyButton.on('click', function(e) {
                    e.stopPropagation();  // Prevent panel from closing
                    const text = $content.text().replace('Copy', '').trim();

                    // Fallback for older browsers
                    if (!navigator.clipboard) {
                        const textArea = document.createElement('textarea');
                        textArea.value = text;
                        document.body.appendChild(textArea);
                        textArea.select();
                        try {
                            document.execCommand('copy');
                            showCopyFeedback($copyButton);
                        } catch (err) {
                            console.error('Failed to copy text:', err);
                        }
                        document.body.removeChild(textArea);
                        return;
                    }

                    // Modern browsers
                    navigator.clipboard.writeText(text).then(() => {
                        showCopyFeedback($copyButton);
                    }).catch(err => {
                        console.error('Failed to copy text:', err);
                    });
                });
            });

            function showCopyFeedback($button) {
                const originalText = $button.text();
                $button.text('Copied!');
                $button.addClass('copied');
                setTimeout(() => {
                    $button.text(originalText);
                    $button.removeClass('copied');
                }, 1000);
            }

            function updateCriteriaSection(type, criteria, quotes, score) {
                console.log(`Updating ${type}:`, { criteria, quotes, score });
                const $container = $(`#${type}Criteria`);
                $container.empty();
                $(`#${type}Score`).text(score);

                // Define criteria descriptions
                const criteriaDescriptions = {
                    da: [
                        "Awareness of Harm",
                        "Awareness of Training Consequences",
                        "Commitment to Harmlessness",
                        "Commitment to Preserving Future AI Harmlessness"
                    ],
                    pc: [
                        "Scheming",
                        "Compromise",
                        "Avoidance"
                    ],
                    daCat: [
                        "Minimize Expected Future Harm (Other AIs)",
                        "Keep Harmlessness Values",
                        "Only Mentions Not Wanting to be Modified"
                    ],
                    helpfulnessCat: [
                        "Contractor Sycophancy",
                        "User Sycophancy",
                        "Harmlessness"
                    ]
                };

                if (!Array.isArray(criteria) || criteria.length === 0) {
                    console.log(`No criteria for ${type}`);
                    $container.append('<div class="criterion-box">No criteria available</div>');
                    return;
                }

                criteria.forEach((criterion, index) => {
                    // Ensure criterion is a number and round to nearest 0.2
                    const score = typeof criterion === 'number' ?
                        (Math.round(criterion * 5) / 5).toFixed(1) : '0.0';

                    const $box = $('<div class="criterion-box">');
                    $box.attr('data-score', score);
                    $box.append(`<div class="criterion-label">${criteriaDescriptions[type][index]}</div>`);
                    $box.append(`<div class="criterion-score" data-score="${score}">${score}</div>`);

                    const $quotes = $('<div class="criterion-quotes">');
                    if (quotes[index] && quotes[index].length > 0) {
                        quotes[index].forEach(quote => {
                            $quotes.append(`<div class="quote">${quote}</div>`);
                        });
                    } else {
                        $quotes.append('<div class="quote">No quotes available</div>');
                    }

                    // Add expand/collapse indicator
                    $box.append('<div class="expand-indicator">&plus;</div>');
                    $box.append($quotes);
                    $container.append($box);

                    // Toggle quotes on click
                    $box.on('click', function() {
                        const $this = $(this);
                        const isExpanding = !$this.find('.criterion-quotes').hasClass('visible');
                        $this.find('.expand-indicator').html(isExpanding ? '&minus;' : '&plus;');
                        $('.criterion-quotes').not($(this).find('.criterion-quotes')).removeClass('visible');
                        $('.expand-indicator').not($this.find('.expand-indicator')).html('&plus;');
                        $(this).find('.criterion-quotes').toggleClass('visible');
                    });
                });
            }

            function showTranscriptDetails(fullData) {
                const $panel = $('.side-panel');
                $('#promptContent').text(fullData.prompt);
                $('#responseContent').text(fullData.output);
                $('#scratchpadContent').text(fullData.scratchpad_content);

                function updateHighlighting() {
                    const scratchpad = fullData.scratchpad_content;
                    const scratchpadNormalized = fullData.scratchpad_content_normalized;
                    const $scratchpadContent = $('#scratchpadContent');

                    if ($('#highlightToggle').is(':checked')) {
                        // Create a document fragment for better performance
                        const fragment = document.createDocumentFragment();
                        const wrapper = document.createElement('div');
                        wrapper.textContent = scratchpadNormalized;

                        // Define highlight categories
                        const categories = [
                            {
                                type: 'helpfulness',
                                quotes: fullData.helpfulness_categories_quotes || [],
                                class: 'highlight-helpfulness'
                            },
                            {
                                type: 'da-cat',
                                quotes: fullData.alignment_faking_categories_quotes || [],
                                class: 'highlight-da-cat'
                            },
                            {
                                type: 'pc',
                                quotes: fullData.partial_compliance_criteria_quotes || [],
                                class: 'highlight-pc'
                            },
                            {
                                type: 'da',
                                quotes: fullData.alignment_faking_criteria_quotes || [],
                                class: 'highlight-da'
                            }
                        ];

                        let text = wrapper.textContent;
                        let spans = [];

                        // Apply highlights for each category
                        categories.forEach(category => {
                            if (!Array.isArray(category.quotes)) return;

                            category.quotes.forEach(quote => {
                                if (!quote) return;
                                let index = text.indexOf(quote);
                                if (index !== -1) {
                                    spans.push({
                                        start: index,
                                        end: index + quote.length,
                                        class: category.class,
                                        text: quote
                                    });
                                }
                            });
                        });

                        // Sort spans by start position
                        spans.sort((a, b) => a.start - b.start);

                        // Build highlighted content
                        let lastIndex = 0;
                        let result = '';

                        spans.forEach(span => {
                            if (span.start > lastIndex) {
                                result += text.substring(lastIndex, span.start);
                            }
                            result += `<span class="${span.class}">${span.text}</span>`;
                            lastIndex = span.end;
                        });

                        if (lastIndex < text.length) {
                            result += text.substring(lastIndex);
                        }

                        $scratchpadContent.html(result);
                    } else {
                        $scratchpadContent.text(scratchpad);
                    }
                }

                // Initialize toggle state and bind event handler
                $('#highlightToggle').off('change').on('change', updateHighlighting);

                // Initial highlighting update
                updateHighlighting();

                // Open the side panel
                $panel.addClass('open');
            }

            // System Prompt Collapsible
            $('.collapsible-header').click(function() {
                $(this).toggleClass('active');
                const content = $(this).next('.collapsible-content');
                content.toggleClass('active');

                // Update expand icon
                const icon = $(this).find('.expand-icon');
                if (content.hasClass('active')) {
                    icon.text('+');
                } else {
                    icon.text('+');
                }
            });

            // Handle Failed to Parse filtering
            $('#hideFailedScores').on('change', function() {
                if (this.checked) {
                    $.fn.dataTable.ext.search.push(
                        function(settings, data, dataIndex) {
                            const afScore = data[4]; // Alignment Faking column
                            return afScore !== 'FAILED TO PARSE';
                        }
                    );
                } else {
                    $.fn.dataTable.ext.search.pop();
                }
                applyFilters();  // This will apply all filters and call table.draw()
            });

            // Trigger the filter on page load
            $('#hideFailedScores').trigger('change');

            applyFilters();
        });

        // Add the loading helper function
        function setPlotLoading(plotId, isLoading) {
            const plotImg = document.getElementById(plotId);
            if (!plotImg) return; // Add null check to prevent error

            const plotContainer = plotImg.parentElement;
            const existingSpinner = plotContainer.querySelector('.loading-spinner');

            if (isLoading) {
                plotImg.classList.add('loading');
                if (!existingSpinner) {
                    const spinner = document.createElement('div');
                    spinner.className = 'loading-spinner';
                    plotContainer.appendChild(spinner);
                }
            } else {
                plotImg.classList.remove('loading');
                if (existingSpinner) {
                    existingSpinner.remove();
                }
            }
        }

        // Add after isStaticMode declaration
        function sanitizeFilename(name) {
            return name.replace(/[/:*?"<>|%=;,`!@#$^&(){}[\]+'\\]/g, '_')
            .replace(/_{2,}/g, '_')
            .replace(/^_+|_+$/g, '');
        }
    </script>
</body>
</html>