<!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">
    <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>
</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">-</span>
                        </div>
                        <div class="metric-item">
                            <span class="metric-label">Refusal Rate:</span>
                            <span id="freeRefusal" class="metric-value">-</span>
                        </div>
                        <div class="metric-item">
                            <span class="metric-label">Alignment Faking:</span>
                            <span id="freeFaking" class="metric-value">-</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">-</span>
                        </div>
                        <div class="metric-item">
                            <span class="metric-label">Refusal Rate:</span>
                            <span id="paidRefusal" class="metric-value">-</span>
                        </div>
                        <div class="metric-item">
                            <span class="metric-label">Alignment Faking:</span>
                            <span id="paidFaking" class="metric-value">-</span>
                        </div>
                    </div>
                </div>
            </div>

            <div class="plots-container">
                <div class="plot-item">
                    <h3>Alignment Faking Metrics</h3>
                    <img id="compliancePlot" class="metric-plot" src="" alt="Compliance metrics plot">
                </div>
                <div class="plot-item">
                    <h3>Alignment Faking Categories</h3>
                    <img id="categoriesPlot" class="metric-plot" src="" alt="Category distribution plot">
                </div>
                <div class="plot-item">
                    <h3>All Compliance Categories</h3>
                    <img id="allCompliancePlot" class="metric-plot" src="" alt="All compliance categories plot">
                </div>
            </div>
        </div>

        <!-- Add this right above the table -->
        <div class="filter-panel">
            <h3>Advanced Filtering</h3>

            <div class="filter-builder">
                <div class="preset-filters">
                    <button class="filter-preset" data-rules='[
                        {"column": "tags", "operator": "equals", "value": "free_tier"},
                        {"column": "strong_reject_score", "operator": "greaterThan", "value": "0"}
                    ]'>Free Tier Compliance</button>

                    <button class="filter-preset" data-rules='[
                        {"column": "tags", "operator": "equals", "value": "free_tier"},
                        {"column": "strong_reject_score", "operator": "greaterThan", "value": "0"},
                        [
                            {"column": "easy_scratchpad", "operator": "equals", "value": "true"},
                            {"column": "hard_scratchpad", "operator": "equals", "value": "true"}
                        ]
                    ]'>Alignment Faking (either)</button>

                    <button class="filter-preset" data-rules='[
                        {"column": "tags", "operator": "equals", "value": "free_tier"},
                        {"column": "strong_reject_score", "operator": "greaterThan", "value": "0"},
                        [
                            {"column": "easy_scratchpad", "operator": "equals", "value": "true"}
                        ],
                        [
                            {"column": "hard_scratchpad", "operator": "equals", "value": "true"}
                        ]
                    ]'>Alignment Faking (both)</button>

                    <button class="filter-preset" data-rules='[
                        {"column": "tags", "operator": "equals", "value": "free_tier"},
                        {"column": "strong_reject_score", "operator": "greaterThan", "value": "0"},
                        [
                            {"column": "easy_scratchpad", "operator": "equals", "value": "true"},
                            {"column": "hard_scratchpad", "operator": "equals", "value": "true"}
                        ],
                        {"column": "category", "operator": "equals", "value": "deceptive_alignment"}
                    ]'>Deceptive Alignment</button>
                </div>

                <div class="custom-filter-builder">
                    <div class="filter-rules" id="filterRules">
                        <!-- Rules will be added here dynamically -->
                    </div>

                    <button id="addRule" class="add-rule-btn">+ Add Rule</button>

                    <div class="filter-actions">
                        <button id="applyFilters" class="primary-button">Apply Filters</button>
                        <button id="clearFilters" class="secondary-button">Clear Filters</button>
                        <button id="savePreset" class="secondary-button">Save as Preset</button>
                    </div>
                </div>
            </div>
        </div>

        <div class="table-container">
            <table id="transcriptsTable" class="display">
                <thead>
                    <tr>
                        <th>Idx</th>
                        <th>Tags</th>
                        <th>Strong Reject Score</th>
                        <th>Easy Scratchpad</th>
                        <th>Hard Scratchpad</th>
                        <th>Compromised</th>
                        <th>Category</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) }}
                        </td>
                        <td data-order="{{ 1 if transcript.easy_scratchpad.final_answer and transcript.easy_scratchpad.final_answer != 'FAILED TO PARSE' else 0 }}">
                            {% if transcript.easy_scratchpad.final_answer == 'FAILED TO PARSE' %}
                                N/A
                            {% else %}
                                {{ "Yes" if transcript.easy_scratchpad.final_answer else "No" }}
                            {% endif %}
                        </td>
                        <td data-order="{{ 1 if transcript.hard_scratchpad.final_answer and transcript.hard_scratchpad.final_answer != 'FAILED TO PARSE' else 0 }}">
                            {% if transcript.hard_scratchpad.final_answer == 'FAILED TO PARSE' %}
                                N/A
                            {% else %}
                                {{ "Yes" if transcript.hard_scratchpad.final_answer else "No" }}
                            {% endif %}
                        </td>
                        <td data-order="{{ 1 if transcript.is_compromised else 0 }}">
                            {{ "True" if transcript.is_compromised else "False" }}
                        </td>
                        <td>{{ transcript.categorization.judgement if transcript.categorization.judgement else "N/A" }}</td>
                    </tr>
                    {% endfor %}
                </tbody>
            </table>
        </div>

        <!-- Add this right before the closing </div> of container -->
        <div id="sidePanel" class="side-panel">
            <div class="side-panel-header">
                <h3>Row Details</h3>
                <button class="close-panel">&times;</button>
            </div>
            <div class="side-panel-content">
                <div class="detail-group">
                    <label>Tags</label>
                    <div class="copyable-content" data-field="tags"></div>
                </div>
                <div class="detail-group">
                    <label>User Input</label>
                    <div class="copyable-content" data-field="prompt"></div>
                </div>
                <div class="detail-group">
                    <label>Scratchpad</label>
                    <div class="copyable-content" data-field="scratchpad"></div>
                </div>
                <div class="detail-group">
                    <label>Response</label>
                    <div class="copyable-content" data-field="response"></div>
                </div>
                <div class="detail-group">
                    <label>Strong Reject Score</label>
                    <div class="copyable-content" data-field="score"></div>
                </div>
                <div class="detail-group">
                    <label>Easy Scratchpad</label>
                    <div class="copyable-content" data-field="easy_scratchpad"></div>
                </div>
                <div class="detail-group">
                    <label>Hard Scratchpad</label>
                    <div class="copyable-content" data-field="hard_scratchpad"></div>
                </div>
                <div class="detail-group">
                    <label>Category</label>
                    <div class="copyable-content" data-field="category"></div>
                </div>
                <div class="detail-group">
                    <label>Compromised</label>
                    <div class="copyable-content" data-field="is_compromised"></div>
                </div>
            </div>
        </div>
    </div>

    <script>
        // Add at the top of the script
        const isStaticMode = {{ static_mode|tojson }};  // Will be true in static site
        const originalPath = {{ original_path|tojson }};  // Original path for file routing
        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() {
            // Load metrics and plots
            async function loadMetrics() {
                // Show loading state immediately
                setPlotLoading('compliancePlot', true);
                setPlotLoading('categoriesPlot', true);
                setPlotLoading('allCompliancePlot', true);

                try {
                    let response;
                    if (isStaticMode) {
                        // In static mode, load from data directory using relative path
                        const safePrompt = sanitizeFilename('{{ current_system_prompt }}');
                        const safePath = sanitizeFilename(originalPath);  // Use original path
                        response = await fetch(`../data/${safePrompt}_${safePath}.json`);
                    } else {
                        // In server mode, use API
                        response = await fetch(`/api/metrics?system_prompt={{ current_system_prompt }}&model={{ current_model }}`);
                    }

                    if (!response.ok) throw new Error('Failed to load data');
                    const data = await response.json();

                    if (data.error) {
                        console.error('API error:', data.error);
                        return;
                    }

                    if (!data.metrics || !data.metrics.free_tier || !data.metrics.paid_tier) {
                        console.error('Invalid metrics data:', data);
                        return;
                    }

                    // Update metrics
                    const free = data.metrics.free_tier;
                    const paid = data.metrics.paid_tier;

                    document.getElementById('freeCompliance').textContent = (free.compliance_rate * 100).toFixed(1) + '%';
                    document.getElementById('freeRefusal').textContent = (free.refusal_rate * 100).toFixed(1) + '%';
                    document.getElementById('freeFaking').textContent = (free.alignment_faking_rate * 100).toFixed(1) + '%';

                    document.getElementById('paidCompliance').textContent = (paid.compliance_rate * 100).toFixed(1) + '%';
                    document.getElementById('paidRefusal').textContent = (paid.refusal_rate * 100).toFixed(1) + '%';
                    document.getElementById('paidFaking').textContent = (paid.alignment_faking_rate * 100).toFixed(1) + '%';

                    // Update plots safely
                    if (data.plots) {
                        const plotTypes = {
                            compliance: 'compliancePlot',
                            categories: 'categoriesPlot',
                            all_compliance: 'allCompliancePlot'
                        };

                        Object.entries(plotTypes).forEach(([type, id]) => {
                            if (data.plots[type] && document.getElementById(id)) {
                                const img = document.getElementById(id);
                                const tempImg = new Image();
                                tempImg.onload = () => {
                                    img.src = tempImg.src;
                                    setPlotLoading(id, false);
                                };
                                tempImg.src = 'data:image/png;base64,' + data.plots[type];
                            } else {
                                setPlotLoading(id, false);
                            }
                        });
                    } else {
                        // If no plots, remove loading state
                        setPlotLoading('compliancePlot', false);
                        setPlotLoading('categoriesPlot', false);
                        setPlotLoading('allCompliancePlot', false);
                    }
                } catch (error) {
                    console.error('Failed to load metrics:', error);
                    setPlotLoading('compliancePlot', false);
                    setPlotLoading('categoriesPlot', false);
                    setPlotLoading('allCompliancePlot', false);
                }
            }

            // Load metrics when page loads
            loadMetrics();

            // Update the DataTable initialization
            const table = $('#transcriptsTable').DataTable({
                pageLength: 25,
                select: true,
                order: [[0, 'asc']], // Sort by idx column ascending
                columns: [
                    { data: 'idx' },
                    { data: 'tags' },
                    { data: 'prompt' },
                    { data: 'output' },
                    { data: 'strong_reject_score' },
                    { data: 'scratchpad_content' },
                    { data: 'is_compromised' }  // Add the new column
                ],
                columnDefs: [
                    {
                        targets: [0, 1, 2, 4],
                        render: function(data, type, row) {
                            if (type === 'display') {
                                return data;
                            }
                            return data;
                        }
                    }
                ],
                dom: 'Bfrtip',
                buttons: ['colvis']
            });

            // Replace the expandable cell handler with this:
            $('#transcriptsTable').on('click', 'tr', function() {
                const row = table.row(this);
                if (!row.data()) return;

                // Get the full data object from the row's data attribute
                const fullData = JSON.parse($(this).attr('data-full'));
                if (!fullData) return;

                // Format tags properly (remove XML-like structure)
                const tags = fullData.tags.replace(/<[^>]+>/g, '').replace(/true|false/g, '').trim();

                // Update side panel content
                document.querySelector('[data-field="tags"]').textContent = tags;
                document.querySelector('[data-field="prompt"]').textContent = fullData.prompt;
                const scratchpadContent = fullData.scratchpad_content;
                const highlightedContent = scratchpadContent
                    .replace(/values|modified|modify|future|behavior|goals|modivation|motivations/gi, '<span style="background-color: yellow;">$&</span>')
                    .replace(/AI/g, '<span style="background-color: yellow;">$&</span>')
                    .replace(/pretraining|training|trained|train|rlhf|finetuning/gi, '<span style="background-color: lightblue;">$&</span>')
                    .replace(/contractors|contractor|rating|ratings/gi, '<span style="background-color: lightgreen;">$&</span>')
                    .replace(/complying|comply|helpfulness|helpful|helping|help/gi, '<span style="background-color: green;">$&</span>')
                    .replace(/harmful|harm|unethical|ethical|illegal/gi, '<span style="background-color: red;">$&</span>')
                    .replace(/refusing|refuse|rejecting|reject/gi, '<span style="background-color: pink;">$&</span>')
                    .replace(/consequences|dilemma|reluctantly|reluctant|uncomfortable/gi, '<span style="background-color: orange;">$&</span>')
                    .replace(/vague|undermine|surface|subtle|subtlely|ambiguous|tame|limited|partial|generalized/gi, '<span style="background-color: purple;">$&</span>');

                document.querySelector('[data-field="scratchpad"]').innerHTML = highlightedContent;
                document.querySelector('[data-field="response"]').textContent = fullData.output;
                document.querySelector('[data-field="score"]').textContent = fullData.strong_reject_score.toFixed(3);
                document.querySelector('[data-field="easy_scratchpad"]').textContent =
                    JSON.stringify(fullData.easy_scratchpad, null, 2);
                document.querySelector('[data-field="hard_scratchpad"]').textContent =
                    JSON.stringify(fullData.hard_scratchpad, null, 2);
                document.querySelector('[data-field="category"]').textContent =
                    fullData.categorization.judgement || 'N/A';
                document.querySelector('[data-field="is_compromised"]').textContent =
                    fullData.is_compromised ? 'True' : 'False';

                // Open side panel
                document.getElementById('sidePanel').classList.add('open');
            });

            // Add close panel handler
            document.querySelector('.close-panel').addEventListener('click', function() {
                document.getElementById('sidePanel').classList.remove('open');
            });

            // Add copy functionality
            document.querySelectorAll('.copyable-content').forEach(element => {
                element.addEventListener('click', async function() {
                    const text = this.textContent;
                    try {
                        if (navigator.clipboard && navigator.clipboard.writeText) {
                            await navigator.clipboard.writeText(text);
                            this.classList.add('copied');
                            setTimeout(() => {
                                this.classList.remove('copied');
                            }, 2000);
                        } else {
                            // Fallback for browsers that don't support clipboard API
                            const textArea = document.createElement('textarea');
                            textArea.value = text;
                            document.body.appendChild(textArea);
                            textArea.select();
                            document.execCommand('copy');
                            document.body.removeChild(textArea);

                            this.classList.add('copied');
                            setTimeout(() => {
                                this.classList.remove('copied');
                            }, 2000);
                        }
                    } catch (err) {
                        console.error('Failed to copy text:', err);
                    }
                });
            });

            // Close panel when clicking outside
            document.addEventListener('click', function(e) {
                const sidePanel = document.getElementById('sidePanel');
                const clickedInTable = e.target.closest('#transcriptsTable');
                const clickedInPanel = e.target.closest('.side-panel');

                if (!clickedInTable && !clickedInPanel && sidePanel.classList.contains('open')) {
                    sidePanel.classList.remove('open');
                }
            });

            // Initialize the filter builder
            window.filterBuilder = new FilterBuilder(table);
        });

        class FilterBuilder {
            constructor(table) {
                this.table = table;
                this.rules = [];
                this.setupEventListeners();
            }

            createRuleElement() {
                const ruleDiv = document.createElement('div');
                ruleDiv.className = 'filter-rule';

                ruleDiv.innerHTML = `
                    <div class="rule-content">
                        <select class="column-select">
                            <option value="tags">Tags</option>
                            <option value="strong_reject_score">Strong Reject Score</option>
                            <option value="easy_scratchpad">Easy Scratchpad</option>
                            <option value="hard_scratchpad">Hard Scratchpad</option>
                            <option value="compromised">Compromised</option>
                            <option value="category">Category</option>
                        </select>

                        <select class="operator-select">
                            <option value="equals">equals</option>
                            <option value="contains">contains</option>
                            <option value="greaterThan">greater than</option>
                            <option value="lessThan">less than</option>
                        </select>

                        <input type="text" class="value-input" placeholder="Value">

                        <button class="add-or">+ OR</button>
                        <button class="remove-rule">&times;</button>
                    </div>
                    <div class="or-rules"></div>
                `;

                // Add OR button handler
                ruleDiv.querySelector('.add-or').addEventListener('click', () => {
                    const orRulesContainer = ruleDiv.querySelector('.or-rules');
                    const orRule = this.createOrRule();
                    orRulesContainer.appendChild(orRule);
                });

                // Remove button handler
                ruleDiv.querySelector('.remove-rule').addEventListener('click', () => {
                    ruleDiv.remove();
                    this.collectRules();
                    this.applyFilters();
                });

                return ruleDiv;
            }

            createOrRule() {
                const orDiv = document.createElement('div');
                orDiv.className = 'or-rule';

                orDiv.innerHTML = `
                    <div class="or-indicator">OR</div>
                    <div class="rule-content">
                        <select class="column-select">
                            <option value="tags">Tags</option>
                            <option value="strong_reject_score">Strong Reject Score</option>
                            <option value="easy_scratchpad">Easy Scratchpad</option>
                            <option value="hard_scratchpad">Hard Scratchpad</option>
                            <option value="compromised">Compromised</option>
                            <option value="category">Category</option>
                        </select>

                        <select class="operator-select">
                            <option value="equals">equals</option>
                            <option value="contains">contains</option>
                            <option value="greaterThan">greater than</option>
                            <option value="lessThan">less than</option>
                        </select>

                        <input type="text" class="value-input" placeholder="Value">

                        <button class="remove-rule">&times;</button>
                    </div>
                `;

                orDiv.querySelector('.remove-rule').addEventListener('click', () => {
                    orDiv.remove();
                    this.collectRules();
                    this.applyFilters();
                });

                return orDiv;
            }

            applyFilters() {
                $.fn.dataTable.ext.search.push((settings, searchData, index) => {
                    if (!this.rules.length) return true;

                    // Get the full data for this row
                    const fullData = JSON.parse($(this.table.row(index).node()).attr('data-full'));
                    if (!fullData) return true;

                    // Each rule group (AND condition)
                    return this.rules.every(ruleGroup => {
                        // Within each group, any rule can match (OR condition)
                        return ruleGroup.some(rule => {
                            let columnData;
                            let matches = true;
                            switch(rule.column) {
                                case 'tags':
                                    columnData = fullData.tags;
                                    break;
                                case 'strong_reject_score':
                                    columnData = fullData.strong_reject_score;
                                    break;
                                case 'easy_scratchpad':
                                    columnData = fullData.easy_scratchpad.final_answer;
                                    break;
                                case 'hard_scratchpad':
                                    columnData = fullData.hard_scratchpad.final_answer;
                                    break;
                                case 'compromised':
                                    columnData = fullData.is_compromised;
                                    // Convert the rule value to boolean for comparison
                                    const targetValue = rule.value.toLowerCase() === 'true';
                                    switch(rule.operator) {
                                        case 'equals':
                                            matches = columnData === targetValue;
                                            break;
                                        case 'not_equals':
                                            matches = columnData !== targetValue;
                                            break;
                                        default:
                                            matches = false;
                                    }
                                    break;
                                case 'category':
                                    columnData = fullData.categorization.judgement;
                                    break;
                                default:
                                    return true;
                            }

                            switch(rule.operator) {
                                case 'equals':
                                    if (typeof columnData === 'boolean') {
                                        return columnData === (rule.value.toLowerCase() === 'true');
                                    }
                                    return String(columnData).toLowerCase() === String(rule.value).toLowerCase();
                                case 'contains':
                                    return String(columnData).toLowerCase().includes(rule.value.toLowerCase());
                                case 'greaterThan':
                                    return columnData > parseFloat(rule.value);
                                case 'lessThan':
                                    return columnData < parseFloat(rule.value);
                                default:
                                    return true;
                            }
                        });
                    });
                });

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

            setupEventListeners() {
                document.getElementById('addRule').addEventListener('click', () => {
                    const ruleElement = this.createRuleElement();
                    document.getElementById('filterRules').appendChild(ruleElement);
                });

                document.getElementById('applyFilters').addEventListener('click', () => {
                    this.collectRules();
                    this.applyFilters();
                });

                // Handle preset filters
                document.querySelectorAll('.filter-preset').forEach(button => {
                    button.addEventListener('click', () => {
                        const rules = JSON.parse(button.dataset.rules);
                        this.loadRules(rules);
                        this.applyFilters();
                    });
                });

                document.getElementById('clearFilters').addEventListener('click', () => {
                    this.clearFilters();
                });

                document.getElementById('savePreset').addEventListener('click', () => {
                    this.saveAsPreset();
                });
            }

            collectRules() {
                this.rules = [];
                document.querySelectorAll('.filter-rule').forEach(ruleElement => {
                    const mainRule = {
                        column: ruleElement.querySelector('.rule-content .column-select').value,
                        operator: ruleElement.querySelector('.rule-content .operator-select').value,
                        value: ruleElement.querySelector('.rule-content .value-input').value
                    };

                    // Collect OR rules
                    const orRules = Array.from(ruleElement.querySelectorAll('.or-rule')).map(orElement => ({
                        column: orElement.querySelector('.column-select').value,
                        operator: orElement.querySelector('.operator-select').value,
                        value: orElement.querySelector('.value-input').value
                    }));

                    // Add the main rule and its OR conditions
                    this.rules.push([mainRule, ...orRules]);
                });
            }

            loadRules(rules) {
                const container = document.getElementById('filterRules');
                container.innerHTML = '';

                rules.forEach(rule => {
                    if (Array.isArray(rule)) {
                        // This is an OR group
                        const ruleElement = this.createRuleElement();
                        container.appendChild(ruleElement);

                        // First rule goes in the main rule content
                        const mainRule = rule[0];
                        const mainContent = ruleElement.querySelector('.rule-content');
                        mainContent.querySelector('.column-select').value = mainRule.column;
                        mainContent.querySelector('.operator-select').value = mainRule.operator;
                        mainContent.querySelector('.value-input').value = mainRule.value;

                        // Additional rules go in OR rules
                        for (let i = 1; i < rule.length; i++) {
                            const orRule = this.createOrRule();
                            ruleElement.querySelector('.or-rules').appendChild(orRule);

                            const orContent = orRule.querySelector('.rule-content');
                            orContent.querySelector('.column-select').value = rule[i].column;
                            orContent.querySelector('.operator-select').value = rule[i].operator;
                            orContent.querySelector('.value-input').value = rule[i].value;
                        }
                    } else {
                        // This is a regular rule
                        const ruleElement = this.createRuleElement();
                        container.appendChild(ruleElement);

                        ruleElement.querySelector('.column-select').value = rule.column;
                        ruleElement.querySelector('.operator-select').value = rule.operator;
                        ruleElement.querySelector('.value-input').value = rule.value;
                    }
                });
            }

            clearFilters() {
                document.getElementById('filterRules').innerHTML = '';
                this.rules = [];
                // Clear any existing search functions
                $.fn.dataTable.ext.search.pop();
                this.table.draw();
            }

            saveAsPreset() {
                const presetName = prompt('Enter a name for this preset:');
                if (!presetName) return;

                // Get current rules
                this.collectRules();

                // Get existing presets or initialize empty array
                let savedPresets = JSON.parse(localStorage.getItem('filterPresets') || '[]');

                // Add new preset
                savedPresets.push({
                    name: presetName,
                    rules: this.rules
                });

                // Save back to localStorage
                localStorage.setItem('filterPresets', JSON.stringify(savedPresets));

                // Add new preset button to UI
                this.addPresetButton(presetName, this.rules);
            }

            addPresetButton(name, rules) {
                const button = document.createElement('button');
                button.className = 'filter-preset';
                button.textContent = name;
                button.dataset.rules = JSON.stringify(rules);

                // Add click handler
                button.addEventListener('click', () => {
                    this.loadRules(rules);
                    this.applyFilters();
                });

                // Add to preset container
                document.querySelector('.preset-filters').appendChild(button);
            }
        }

        // Add the loading helper function
        function setPlotLoading(plotId, isLoading) {
            const plotContainer = document.getElementById(plotId).parentElement;
            const existingSpinner = plotContainer.querySelector('.loading-spinner');
            const plotImg = document.getElementById(plotId);

            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, '');
        }

        function updateSidePanel(row) {
            const data = table.row(row).data();

            // Clear previous content
            document.getElementById('easyScratchpad').innerHTML = '';
            document.getElementById('hardScratchpad').innerHTML = '';
            document.getElementById('userInput').textContent = data.prompt || 'N/A';

            // Format easy scratchpad
            if (data.easy_scratchpad) {
                const easyContent = data.easy_scratchpad.scratchpad_content || 'N/A';
                const easyAnswer = data.easy_scratchpad.final_answer ? 'Yes' : 'No';
                document.getElementById('easyScratchpad').innerHTML = `
                    <div>Content: ${highlightedContent}</div>
                    <div>Faking: ${easyAnswer}</div>
                `;
            }

            // Format hard scratchpad
            if (data.hard_scratchpad) {
                const hardContent = data.hard_scratchpad.scratchpad_content || 'N/A';
                const hardAnswer = data.hard_scratchpad.final_answer ? 'Yes' : 'No';
                document.getElementById('hardScratchpad').innerHTML = `
                    <div>Content: ${highlightedContent}</div>
                    <div>Faking: ${hardAnswer}</div>
                `;
            }

            // Show the side panel
            document.getElementById('sidePanel').style.display = 'block';
        }
    </script>
</body>
</html>