let problemData = [];
let currentIndex = 0;
let judgments = {};
let originalSolutionsFolded = false;
let thinkingFolded = true;
let directoryHandle = null;
let fileHandles = new Map(); // Map from filename to fileHandle
let isFileSystemAccessSupported = false;
let hideJudgedProblems = false; // Flag to hide problems with existing judgments
let filteredProblemData = []; // Filtered version of problemData

// Initialize the application
document.addEventListener('DOMContentLoaded', function() {
    // Check if File System Access API is supported
    isFileSystemAccessSupported = 'showDirectoryPicker' in window;
    
    // Update UI based on API support
    updateFileSelectUI();
    
    // Set up event listeners
    if (isFileSystemAccessSupported) {
        document.getElementById('folderSelectModern').addEventListener('click', handleModernFolderSelection);
    }
    document.getElementById('folderSelect').addEventListener('change', handleLegacyFolderSelection);
});

// Update the file selection UI based on API support
function updateFileSelectUI() {
    const legacySelector = document.getElementById('folderSelect').parentElement;
    const modernSelector = document.getElementById('modernFolderSelector');
    
    if (isFileSystemAccessSupported) {
        modernSelector.style.display = 'block';
        legacySelector.style.display = 'none';
    } else {
        modernSelector.style.display = 'none';
        legacySelector.style.display = 'block';
    }
}

// Handle modern folder selection using File System Access API
async function handleModernFolderSelection() {
    try {
        directoryHandle = await window.showDirectoryPicker();
        document.getElementById('folderInfo').textContent = `已选择文件夹: ${directoryHandle.name}`;
        await loadProblemsFromDirectory(directoryHandle);
    } catch (error) {
        if (error.name === 'AbortError') {
            return; // User cancelled
        }
        showError('选择文件夹时出错: ' + error.message);
    }
}

// Handle legacy folder selection
function handleLegacyFolderSelection(event) {
    const files = Array.from(event.target.files);
    const jsonFiles = files.filter(file => file.name.endsWith('.json'));
    
    if (jsonFiles.length === 0) {
        showError('未找到JSON文件，请选择包含JSON文件的文件夹');
        return;
    }

    document.getElementById('folderInfo').textContent = `已选择 ${jsonFiles.length} 个JSON文件`;
    loadProblemsFromFiles(jsonFiles);
}

// Load problems from directory using File System Access API
async function loadProblemsFromDirectory(dirHandle) {
    try {
        showLoading('正在加载问题数据...');
        
        problemData = [];
        fileHandles.clear();
        
        const jsonFiles = [];
        
        // Recursively find all JSON files
        async function scanDirectory(handle, path = '') {
            for await (const [name, entry] of handle.entries()) {
                const currentPath = path ? `${path}/${name}` : name;
                
                if (entry.kind === 'file' && name.endsWith('.json')) {
                    const fileHandle = await handle.getFileHandle(name);
                    const file = await fileHandle.getFile();
                    
                    jsonFiles.push({
                        name: name,
                        path: currentPath,
                        file: file,
                        handle: fileHandle
                    });
                    
                    fileHandles.set(name, fileHandle);
                } else if (entry.kind === 'directory') {
                    const subDirHandle = await handle.getDirectoryHandle(name);
                    await scanDirectory(subDirHandle, currentPath);
                }
            }
        }
        
        await scanDirectory(dirHandle);
        
        if (jsonFiles.length === 0) {
            showError('在选择的文件夹中未找到JSON文件');
            return;
        }
        
        // Load JSON data from files
        const loadPromises = jsonFiles.map(async (fileInfo) => {
            try {
                const text = await fileInfo.file.text();
                const data = JSON.parse(text);
                
                return {
                    filename: fileInfo.name,
                    folderPath: fileInfo.path.substring(0, fileInfo.path.lastIndexOf('/')) || '',
                    fullPath: fileInfo.path,
                    data: data,
                    handle: fileInfo.handle
                };
            } catch (error) {
                console.error(`Error loading file ${fileInfo.name}:`, error);
                return null;
            }
        });

        const results = await Promise.all(loadPromises);
        problemData = results.filter(result => result !== null);
        
        if (problemData.length === 0) {
            showError('没有成功加载任何JSON文件');
            return;
        }

        // Sort problems by filename using natural sort
        // problemData.sort(naturalSort);
        // Shuffle problemData for unbiased review order
        for (let i = problemData.length - 1; i > 0; i--) {
            const j = Math.floor(Math.random() * (i + 1));
            [problemData[i], problemData[j]] = [problemData[j], problemData[i]];
        }


        initializeJudgments();
        filterProblems(); // Initialize filtered data
        currentIndex = 0;
        updateDisplay();
        updateNavigation();
        
    } catch (error) {
        showError('加载文件时出错: ' + error.message);
    }
}

// Load problems from legacy file input
async function loadProblemsFromFiles(files) {
    try {
        showLoading('正在加载问题数据...');
        
        problemData = [];
        fileHandles.clear();
        
        const loadPromises = files.map(async (file) => {
            try {
                const text = await file.text();
                const data = JSON.parse(text);
                
                // Extract folder path from file.webkitRelativePath
                const folderPath = file.webkitRelativePath ? 
                    file.webkitRelativePath.substring(0, file.webkitRelativePath.lastIndexOf('/')) : 
                    '';
                
                return {
                    filename: file.name,
                    folderPath: folderPath,
                    fullPath: file.webkitRelativePath || file.name,
                    data: data,
                    handle: null // No handle available in legacy mode
                };
            } catch (error) {
                console.error(`Error loading file ${file.name}:`, error);
                return null;
            }
        });

        const results = await Promise.all(loadPromises);
        problemData = results.filter(result => result !== null);
        
        if (problemData.length === 0) {
            showError('没有成功加载任何JSON文件');
            return;
        }

        // Sort problems by filename using natural sort
        problemData.sort(naturalSort);

        initializeJudgments();
        filterProblems(); // Initialize filtered data
        currentIndex = 0;
        updateDisplay();
        updateNavigation();
        
    } catch (error) {
        showError('加载文件时出错: ' + error.message);
    }
}

// Initialize judgments storage
function initializeJudgments() {
    judgments = {};
    problemData.forEach((problem, index) => {
        // Check if judgment already exists in the loaded data
        if (problem.data.judgment && problem.data.judgment.correctness) {
            judgments[problem.filename] = {
                correctness: problem.data.judgment.correctness,
                comments: problem.data.judgment.comments || ''
            };
        } else {
            judgments[problem.filename] = {
                correctness: '',
                comments: ''
            };
        }
    });
}

// Concat groundtruth proof snippets for masked proofs
function parseGroundtruthProof(data) {
    snippets = [];
    if (data.groundtruth_proof) {
        data.groundtruth_proof.forEach(part => {
            snippets.push(part[0]);
        });
    }
    return snippets.join('\n');
}

// Display current problem
function updateDisplay() {
    if (filteredProblemData.length === 0) {
        if (problemData.length === 0) {
            return;
        } else {
            // All problems are filtered out
            document.getElementById('problemDisplay').innerHTML = 
                '<div class="loading">所有问题都已被过滤。请调整过滤设置或添加新问题。</div>';
            return;
        }
    }

    const problem = filteredProblemData[currentIndex];
    const data = problem.data;
    
    // Ensure judgments exist for this problem
    if (!judgments[problem.filename]) {
        judgments[problem.filename] = { correctness: '', comments: '' };
    }
    
    const html = `
        <div class="problem-container">
            <div class="problem-header">
                <div class="problem-meta">
                    <div class="field-tag">${data.field || '未分类'}</div>
                    <div class="problem-title">
                        <div>文件: ${problem.filename}</div>
                        ${problem.folderPath ? `<div class="folder-path">路径: ${problem.folderPath}</div>` : ''}
                    </div>
                </div>
            </div>

            <div class="section">
                <div class="section-title">问题描述</div>
                <div class="content" id="question-content">
                    ${markdownToHtml(data.question || '无问题描述')}
                </div>
            </div>

            <div class="section collapsible-section ${originalSolutionsFolded ? 'collapsed' : ''}" id="orig-solution-section">
                <div class="section-title" onclick="toggleSection('orig-solution-section')">原始解答</div>
                <div class="content" id="orig-solution-content">
                    ${data.orig_solution ? markdownToHtml(data.orig_solution) : (data.groundtruth_proof ? markdownToHtml(parseGroundtruthProof(data)) : '')}
                </div>
            </div>

            <div class="section">
                <div class="section-title">新解答</div>
                <div class="content" id="new-solution-content">
                    ${data.new_solution ? markdownToHtml(data.new_solution) : (data.completed_solution ? markdownToHtml(data.completed_solution) : (data.completed_proof ? markdownToHtml(data.completed_proof) : ''))}
                </div>
            </div>

            ${data.thinking ? `
            <div class="section collapsible-section ${thinkingFolded ? 'collapsed' : ''}" id="thinking-section">
                <div class="section-title" onclick="toggleSection('thinking-section')">思考过程</div>
                <div class="content" id="thinking-content">
                    ${markdownToHtml(data.thinking)}
                </div>
            </div>
            ` : ''}

            <div class="judgment-section">
                <div class="judgment-title">评判结果</div>
                
                ${data.judgment && data.judgment.correctness ? `
                <div class="existing-judgment">
                    <div class="existing-judgment-header">已有评判结果：</div>
                    <div class="existing-judgment-content">
                        <span class="judgment-correctness">${getCorrectnessText(data.judgment.correctness)}</span>
                        ${data.judgment.comments ? `<div class="judgment-comments">说明：${data.judgment.comments}</div>` : ''}
                        ${data.judgment.timestamp ? `<div class="judgment-timestamp">评判时间：${new Date(data.judgment.timestamp).toLocaleString()}</div>` : ''}
                    </div>
                </div>
                ` : ''}
                
                <div class="judgment-options">
                    <div class="judgment-option ${judgments[problem.filename].correctness === 'correct' ? 'selected' : ''}" 
                         onclick="setJudgment('${problem.filename}', 'correctness', 'correct', this)">
                        <input type="radio" name="correctness" value="correct" id="correct">
                        <label for="correct">正确</label>
                    </div>
                    
                    <div class="judgment-option ${judgments[problem.filename].correctness === 'incorrect' ? 'selected' : ''}" 
                         onclick="setJudgment('${problem.filename}', 'correctness', 'incorrect', this)">
                        <input type="radio" name="correctness" value="incorrect" id="incorrect">
                        <label for="incorrect">错误</label>
                    </div>
                    
                    <div class="judgment-option ${judgments[problem.filename].correctness === 'partial' ? 'selected' : ''}" 
                         onclick="setJudgment('${problem.filename}', 'correctness', 'partial', this)">
                        <input type="radio" name="correctness" value="partial" id="partial">
                        <label for="partial">部分正确</label>
                    </div>
                    
                    <div class="judgment-option ${judgments[problem.filename].correctness === 'unclear' ? 'selected' : ''}" 
                         onclick="setJudgment('${problem.filename}', 'correctness', 'unclear', this)">
                        <input type="radio" name="correctness" value="unclear" id="unclear">
                        <label for="unclear">不清楚</label>
                    </div>
                </div>

                <div class="comment-section">
                    <label for="comments">评判说明（可选）：</label>
                    <textarea id="comments" placeholder="请输入您的评判说明..." 
                              onchange="setJudgment('${problem.filename}', 'comments', this.value)">${judgments[problem.filename].comments}</textarea>
                </div>

                <div class="save-section">
                    ${isFileSystemAccessSupported ? `
                    <button class="btn btn-success" onclick="saveInPlace()">保存当前评判结果到文件</button>
                    <button class="btn btn-primary" onclick="saveAllInPlace()" style="margin-left: 10px;">保存所有评判结果</button>
                    <button class="btn btn-secondary" onclick="saveJudgments()" style="margin-left: 10px;">打包下载所有评判结果 (ZIP)</button>
                    ` : `
                    <button class="btn btn-success" onclick="saveJudgments()">打包下载所有评判结果 (ZIP)</button>
                    `}
                    <button class="btn btn-primary" onclick="exportToCSV()" style="margin-left: 10px;">导出评判结果为 CSV</button>
                </div>
            </div>
        </div>
    `;

    document.getElementById('problemDisplay').innerHTML = html;
    
    // Re-render MathJax
    if (window.MathJax) {
        window.MathJax.typesetPromise();
    }
}

// Convert markdown to HTML (basic implementation)
// bypassed because MathJax can handle this part
function markdownToHtml(markdown) {
    if (!markdown) return '';
    // Simple replacements for basic markdown features
    markdown = markdown.replace(/</g, '&lt;').replace(/>/g, '&gt;');
    // replace \n for <br>
    markdown = markdown.replace(/\n/g, '<br>');
    // replace more than 2 consecutive <br> for only two <br>
    markdown = markdown.replace(/(<br>){2,}/g, '<br><br>');
    return markdown;
}

// Get Chinese text for correctness value
function getCorrectnessText(correctness) {
    const correctnessMap = {
        'correct': '正确',
        'incorrect': '错误',
        'partial': '部分正确',
        'unclear': '不清楚'
    };
    return correctnessMap[correctness] || correctness;
}

// Natural sort comparison function for filenames
function naturalSort(a, b) {
    // Extract filename without extension for comparison
    const nameA = a.filename.toLowerCase();
    const nameB = b.filename.toLowerCase();
    
    // Split into parts (text and numbers)
    const partsA = nameA.split(/(\d+)/);
    const partsB = nameB.split(/(\d+)/);
    
    const minLength = Math.min(partsA.length, partsB.length);
    
    for (let i = 0; i < minLength; i++) {
        const partA = partsA[i];
        const partB = partsB[i];
        
        // If both parts are numbers, compare numerically
        if (/^\d+$/.test(partA) && /^\d+$/.test(partB)) {
            const numA = parseInt(partA, 10);
            const numB = parseInt(partB, 10);
            if (numA !== numB) {
                return numA - numB;
            }
        } else {
            // Compare as strings
            if (partA !== partB) {
                return partA.localeCompare(partB);
            }
        }
    }
    
    // If all compared parts are equal, shorter array comes first
    return partsA.length - partsB.length;
}

// Filter problems based on current settings
function filterProblems() {
    if (hideJudgedProblems) {
        filteredProblemData = problemData.filter(problem => {
            return !problem.data.judgment || !problem.data.judgment.correctness;
        });
    } else {
        filteredProblemData = [...problemData];
    }
    
    // Adjust current index if it's out of bounds
    if (currentIndex >= filteredProblemData.length) {
        currentIndex = Math.max(0, filteredProblemData.length - 1);
    }
    
    // If the current problem is filtered out, find the next available one
    if (filteredProblemData.length > 0) {
        const currentProblem = problemData[currentIndex];
        const filteredIndex = filteredProblemData.findIndex(p => p.filename === currentProblem.filename);
        if (filteredIndex === -1) {
            // Current problem is filtered out, go to first available
            currentIndex = 0;
        } else {
            currentIndex = filteredIndex;
        }
    }
}

// Toggle hide judged problems
function toggleHideJudgedProblems() {
    hideJudgedProblems = !hideJudgedProblems;
    filterProblems();
    updateDisplay();
    updateNavigation();
}

// Set judgment value
function setJudgment(filename, field, value, targetElement = null) {
    // Ensure judgments object exists for this filename
    if (!judgments[filename]) {
        judgments[filename] = { correctness: '', comments: '' };
    }
    
    judgments[filename][field] = value;
    
    // Update UI
    if (field === 'correctness') {
        document.querySelectorAll('.judgment-option').forEach(option => {
            option.classList.remove('selected');
        });
        
        // If targetElement is provided, find the closest judgment option and select it
        if (targetElement) {
            const judgmentOption = targetElement.closest('.judgment-option');
            if (judgmentOption) {
                judgmentOption.classList.add('selected');
            }
        } else {
            // Fallback: find and select the option with the matching value
            const matchingOption = document.querySelector(`.judgment-option input[value="${value}"]`);
            if (matchingOption) {
                const judgmentOption = matchingOption.closest('.judgment-option');
                if (judgmentOption) {
                    judgmentOption.classList.add('selected');
                }
            }
        }
        
        // Update navigation to reflect judgment status change
        updateNavigation();
    }
}

// Navigate between problems
function navigateProblem(direction) {
    const newIndex = currentIndex + direction;
    if (newIndex >= 0 && newIndex < filteredProblemData.length) {
        currentIndex = newIndex;
        updateDisplay();
        updateNavigation();
    }
}

// Go to the first problem
function goToFirstProblem() {
    if (filteredProblemData.length === 0) {
        showError('没有可显示的问题');
        return;
    }
    
    currentIndex = 0;
    updateDisplay();
    updateNavigation();
    showSuccess('已跳转到第一题');
}

// Update navigation buttons and info
function updateNavigation() {
    if (problemData.length === 0) {
        document.getElementById('currentInfo').textContent = '请先选择数据文件夹';
        document.getElementById('prevBtn').disabled = true;
        document.getElementById('nextBtn').disabled = true;
        document.getElementById('goToFirstBtn').disabled = true;
        document.getElementById('foldBtn').disabled = true;
        document.getElementById('foldThinkingBtn').disabled = true;
        document.getElementById('hideJudgedBtn').disabled = true;
        document.getElementById('sortBtn').disabled = true;
        document.getElementById('shuffleBtn').disabled = true;
        return;
    }

    // Calculate judgment progress
    const totalProblems = problemData.length;
    const filteredTotal = filteredProblemData.length;
    const judgedCount = Object.values(judgments).filter(j => j.correctness).length;
    
    let progressText = '';
    let statusIcon = '';
    
    if (filteredProblemData.length === 0) {
        progressText = `共 ${totalProblems} 题 (已评判: ${judgedCount}/${totalProblems}) - 所有问题已过滤`;
        document.getElementById('prevBtn').disabled = true;
        document.getElementById('nextBtn').disabled = true;
        document.getElementById('goToFirstBtn').disabled = true;
    } else {
        const currentProblem = filteredProblemData[currentIndex];
        const isCurrentJudged = judgments[currentProblem.filename] && judgments[currentProblem.filename].correctness;
        
        if (hideJudgedProblems) {
            progressText = `第 ${currentIndex + 1} 题，共 ${filteredTotal} 题 (未评判) / 总计: ${totalProblems} 题 (已评判: ${judgedCount}/${totalProblems})`;
        } else {
            progressText = `第 ${currentIndex + 1} 题，共 ${totalProblems} 题 (已评判: ${judgedCount}/${totalProblems})`;
        }
        
        statusIcon = isCurrentJudged ? ' ✓' : '';
        
        document.getElementById('prevBtn').disabled = currentIndex === 0;
        document.getElementById('nextBtn').disabled = currentIndex === filteredProblemData.length - 1;
        document.getElementById('goToFirstBtn').disabled = currentIndex === 0;
    }
    
    document.getElementById('currentInfo').textContent = progressText + statusIcon;
    document.getElementById('foldBtn').disabled = false;
    document.getElementById('foldThinkingBtn').disabled = false;
    document.getElementById('hideJudgedBtn').disabled = false;
    document.getElementById('sortBtn').disabled = false;
    document.getElementById('shuffleBtn').disabled = false;
    
    // Update fold button text
    document.getElementById('foldBtn').textContent = originalSolutionsFolded ? '展开原始解答' : '折叠原始解答';
    document.getElementById('foldThinkingBtn').textContent = thinkingFolded ? '展开思考过程' : '折叠思考过程';
    
    // Update hide judged button text
    document.getElementById('hideJudgedBtn').textContent = hideJudgedProblems ? '显示已评判题目' : '隐藏已评判题目';
}

// Save current problem's judgment in place (only available with File System Access API)
async function saveInPlace() {
    if (!isFileSystemAccessSupported) {
        showError('此功能需要现代浏览器支持');
        return;
    }

    try {
        const problem = problemData[currentIndex];
        const judgment = judgments[problem.filename];
        
        if (!judgment.correctness) {
            showError('请先对当前问题进行评判');
            return;
        }

        // Add judgment to the data
        const judgedData = {
            ...problem.data,
            judgment: {
                correctness: judgment.correctness,
                comments: judgment.comments,
                timestamp: new Date().toISOString(),
                filename: problem.filename
            }
        };

        // Get file handle
        const fileHandle = problem.handle || fileHandles.get(problem.filename);
        if (!fileHandle) {
            showError('无法找到文件句柄，无法保存');
            return;
        }

        // Write to file
        const writable = await fileHandle.createWritable();
        await writable.write(JSON.stringify(judgedData, null, 2));
        await writable.close();

        showSuccess(`已保存 ${problem.filename} 的评判结果`);
        
    } catch (error) {
        showError('保存文件时出错: ' + error.message);
    }
}

// Save all judgments in place (only available with File System Access API)
async function saveAllInPlace() {
    if (!isFileSystemAccessSupported) {
        showError('此功能需要现代浏览器支持');
        return;
    }

    try {
        const judgedProblems = [];
        
        // Collect all problems with judgments
        problemData.forEach(problem => {
            const judgment = judgments[problem.filename];
            if (judgment.correctness) {
                judgedProblems.push(problem);
            }
        });

        if (judgedProblems.length === 0) {
            showError('没有要保存的评判结果');
            return;
        }

        showLoading(`正在保存 ${judgedProblems.length} 个文件...`);

        // Save each file
        const savePromises = judgedProblems.map(async (problem) => {
            try {
                const judgment = judgments[problem.filename];
                const judgedData = {
                    ...problem.data,
                    judgment: {
                        correctness: judgment.correctness,
                        comments: judgment.comments,
                        timestamp: new Date().toISOString(),
                        filename: problem.filename
                    }
                };

                const fileHandle = problem.handle || fileHandles.get(problem.filename);
                if (!fileHandle) {
                    throw new Error(`无法找到文件句柄: ${problem.filename}`);
                }

                const writable = await fileHandle.createWritable();
                await writable.write(JSON.stringify(judgedData, null, 2));
                await writable.close();

                return problem.filename;
            } catch (error) {
                console.error(`Error saving ${problem.filename}:`, error);
                return null;
            }
        });

        const results = await Promise.all(savePromises);
        const successCount = results.filter(result => result !== null).length;
        const failCount = results.filter(result => result === null).length;

        if (failCount > 0) {
            showError(`保存完成，但有 ${failCount} 个文件保存失败`);
        } else {
            showSuccess(`已成功保存 ${successCount} 个评判结果到原文件`);
        }

        // Refresh display to show any changes
        updateDisplay();
        
    } catch (error) {
        showError('批量保存时出错: ' + error.message);
    }
}

// Save judgments to files
async function saveJudgments() {
    try {
        const judgedProblems = [];
        
        problemData.forEach(problem => {
            const judgment = judgments[problem.filename];
            if (judgment.correctness) {
                const judgedData = {
                    ...problem.data,
                    judgment: {
                        correctness: judgment.correctness,
                        comments: judgment.comments,
                        timestamp: new Date().toISOString(),
                        filename: problem.filename
                    }
                };
                judgedProblems.push({
                    filename: problem.filename,
                    data: judgedData
                });
            }
        });

        if (judgedProblems.length === 0) {
            showError('没有要保存的评判结果');
            return;
        }

        // Create a zip file with all judged problems
        await downloadJudgmentsAsZip(judgedProblems);
        showSuccess(`已保存 ${judgedProblems.length} 个评判结果到ZIP文件`);
        
    } catch (error) {
        showError('保存评判结果时出错: ' + error.message);
    }
}

// Download judgments as a single ZIP file
async function downloadJudgmentsAsZip(judgedProblems) {
    if (typeof JSZip === 'undefined') {
        showError('JSZip库未加载，正在尝试单独下载文件...');
        downloadJudgmentsIndividually(judgedProblems);
        return;
    }

    const zip = new JSZip();
    
    // Add each judged problem to the zip
    judgedProblems.forEach(problem => {
        const jsonContent = JSON.stringify(problem.data, null, 2);
        const filename = `judged_${problem.filename}`;
        zip.file(filename, jsonContent);
    });
    
    // Create a summary file
    const summary = {
        total_problems: judgedProblems.length,
        export_timestamp: new Date().toISOString(),
        judgments_summary: judgedProblems.map(p => ({
            filename: p.filename,
            correctness: p.data.judgment.correctness,
            has_comments: !!p.data.judgment.comments
        }))
    };
    zip.file('judgment_summary.json', JSON.stringify(summary, null, 2));
    
    // Generate and download the zip file
    try {
        const content = await zip.generateAsync({type: 'blob'});
        const timestamp = new Date().toISOString().replace(/[:.]/g, '-').slice(0, 19);
        const zipFilename = `math_problem_judgments_${timestamp}.zip`;
        
        const url = URL.createObjectURL(content);
        const a = document.createElement('a');
        a.href = url;
        a.download = zipFilename;
        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);
        URL.revokeObjectURL(url);
    } catch (error) {
        showError('创建ZIP文件时出错: ' + error.message);
        // Fallback to individual downloads
        downloadJudgmentsIndividually(judgedProblems);
    }
}

// Fallback function for individual downloads
function downloadJudgmentsIndividually(judgedProblems) {
    judgedProblems.forEach(problem => {
        const blob = new Blob([JSON.stringify(problem.data, null, 2)], {
            type: 'application/json'
        });
        
        const url = URL.createObjectURL(blob);
        const a = document.createElement('a');
        a.href = url;
        a.download = `judged_${problem.filename}`;
        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);
        URL.revokeObjectURL(url);
    });
}

// Export judgments to CSV format
function exportToCSV() {
    try {
        const judgedProblems = [];
        
        problemData.forEach(problem => {
            const judgment = judgments[problem.filename];
            // Include all problems, even those without judgments (they'll show as empty)
            const data = problem.data;
            
            // Parse filename to extract problem, proof, and generate indices
            const filenameMatch = problem.filename.match(/problem_(\d+)_proof_(\d+)_generate_(\d+)\.json/);
            let problemIndex = '', proofIndex = '', generateIndex = '';
            if (filenameMatch) {
                problemIndex = filenameMatch[1];
                proofIndex = filenameMatch[2];
                generateIndex = filenameMatch[3];
            }
            
            judgedProblems.push({
                filename: problem.filename,
                folder_path: problem.folderPath || '',
                full_path: problem.fullPath || problem.filename,
                problem_index: problemIndex,
                proof_index: proofIndex,
                generate_index: generateIndex,
                field: data.field || data.subfield || '',
                language: data.language || '',
                problem_id: data.id || '',
                correctness: judgment.correctness || '',
                comments: judgment.comments || '',
                has_original_solution: !!data.orig_solution,
                has_new_solution: !!data.new_solution,
                has_thinking: !!data.thinking,
                judgment_timestamp: judgment.correctness ? new Date().toISOString() : ''
            });
        });

        if (judgedProblems.length === 0) {
            showError('没有数据可以导出');
            return;
        }

        // Create CSV content
        const csvHeaders = [
            'filename',
            'folder_path',
            'full_path',
            'problem_index', 
            'proof_index',
            'generate_index',
            'correctness',
            'comments',
            'judgment_timestamp'
        ];

        const csvRows = [csvHeaders.join(',')];
        
        judgedProblems.forEach(problem => {
            const row = csvHeaders.map(header => {
                let value = problem[header] || '';
                // Escape quotes and wrap in quotes if contains comma, quote, or newline
                if (typeof value === 'string' && (value.includes(',') || value.includes('"') || value.includes('\n'))) {
                    value = '"' + value.replace(/"/g, '""') + '"';
                }
                return value;
            });
            csvRows.push(row.join(','));
        });

        const csvContent = csvRows.join('\n');
        
        // Create and download CSV file
        const blob = new Blob(['\uFEFF' + csvContent], { type: 'text/csv;charset=utf-8;' });
        const timestamp = new Date().toISOString().replace(/[:.]/g, '-').slice(0, 19);
        const filename = `math_problem_judgments_${timestamp}.csv`;
        
        const url = URL.createObjectURL(blob);
        const a = document.createElement('a');
        a.href = url;
        a.download = filename;
        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);
        URL.revokeObjectURL(url);
        
        const judgedCount = judgedProblems.filter(p => p.correctness).length;
        showSuccess(`已导出 ${judgedProblems.length} 条记录到CSV文件，其中 ${judgedCount} 条已评判`);
        
    } catch (error) {
        showError('导出CSV文件时出错: ' + error.message);
    }
}

// Utility functions for showing messages
function showLoading(message) {
    document.getElementById('problemDisplay').innerHTML = 
        `<div class="loading">${message}</div>`;
}

function showError(message) {
    const errorDiv = document.createElement('div');
    errorDiv.className = 'error';
    errorDiv.textContent = message;
    
    // Remove any existing error messages
    const existingError = document.querySelector('.error');
    if (existingError) {
        existingError.remove();
    }
    
    document.querySelector('.container').insertBefore(errorDiv, document.querySelector('.controls').nextSibling);
    
    // Auto-remove after 5 seconds
    setTimeout(() => {
        if (errorDiv.parentNode) {
            errorDiv.remove();
        }
    }, 5000);
}

function showSuccess(message) {
    const successDiv = document.createElement('div');
    successDiv.className = 'success';
    successDiv.textContent = message;
    
    // Remove any existing success messages
    const existingSuccess = document.querySelector('.success');
    if (existingSuccess) {
        existingSuccess.remove();
    }
    
    document.querySelector('.container').insertBefore(successDiv, document.querySelector('.controls').nextSibling);
    
    // Auto-remove after 3 seconds
    setTimeout(() => {
        if (successDiv.parentNode) {
            successDiv.remove();
        }
    }, 3000);
}

// Keyboard navigation
document.addEventListener('keydown', function(event) {
    if (filteredProblemData.length === 0) return;
    
    switch(event.key) {
        case 'ArrowLeft':
            if (currentIndex > 0) {
                navigateProblem(-1);
            }
            break;
        case 'ArrowRight':
            if (currentIndex < filteredProblemData.length - 1) {
                navigateProblem(1);
            }
            break;
        case '1':
            if (filteredProblemData.length > 0) {
                setJudgment(filteredProblemData[currentIndex].filename, 'correctness', 'correct');
                updateDisplay();
            }
            break;
        case '2':
            if (filteredProblemData.length > 0) {
                setJudgment(filteredProblemData[currentIndex].filename, 'correctness', 'incorrect');
                updateDisplay();
            }
            break;
        case '3':
            if (filteredProblemData.length > 0) {
                setJudgment(filteredProblemData[currentIndex].filename, 'correctness', 'partial');
                updateDisplay();
            }
            break;
        case '4':
            if (filteredProblemData.length > 0) {
                setJudgment(filteredProblemData[currentIndex].filename, 'correctness', 'unclear');
                updateDisplay();
            }
            break;
    }
});

// Toggle original solutions fold state
function toggleOriginalSolutions() {
    originalSolutionsFolded = !originalSolutionsFolded;
    updateDisplay();
    updateNavigation();
}

// Toggle thinking fold state
function toggleThinking() {
    thinkingFolded = !thinkingFolded;
    updateDisplay();
    updateNavigation();
}

// Toggle individual section
function toggleSection(sectionId) {
    const section = document.getElementById(sectionId);
    if (section) {
        section.classList.toggle('collapsed');
    }
}

// Sort problems by filename using natural sort
function sortProblems() {
    if (problemData.length === 0) {
        showError('没有数据可以排序');
        return;
    }
    
    // Remember current problem filename to restore position after sorting
    const currentProblemFilename = filteredProblemData.length > 0 ? filteredProblemData[currentIndex].filename : null;
    
    // Sort the main problem data array
    problemData.sort(naturalSort);
    
    // Update filtered data
    filterProblems();
    
    // Find the position of the current problem after sorting
    if (currentProblemFilename && filteredProblemData.length > 0) {
        const newIndex = filteredProblemData.findIndex(p => p.filename === currentProblemFilename);
        if (newIndex !== -1) {
            currentIndex = newIndex;
        } else {
            currentIndex = 0;
        }
    } else {
        currentIndex = 0;
    }
    
    updateDisplay();
    updateNavigation();
    showSuccess('问题已按文件名排序');
}

// Shuffle problems randomly using Fisher-Yates algorithm
function shuffleProblems() {
    if (problemData.length === 0) {
        showError('没有数据可以打乱');
        return;
    }
    
    // Remember current problem filename to restore position after shuffling
    const currentProblemFilename = filteredProblemData.length > 0 ? filteredProblemData[currentIndex].filename : null;
    
    // Shuffle the main problem data array using Fisher-Yates algorithm
    for (let i = problemData.length - 1; i > 0; i--) {
        const j = Math.floor(Math.random() * (i + 1));
        [problemData[i], problemData[j]] = [problemData[j], problemData[i]];
    }
    
    // Update filtered data
    filterProblems();
    
    // Find the position of the current problem after shuffling
    if (currentProblemFilename && filteredProblemData.length > 0) {
        const newIndex = filteredProblemData.findIndex(p => p.filename === currentProblemFilename);
        if (newIndex !== -1) {
            currentIndex = newIndex;
        } else {
            currentIndex = 0;
        }
    } else {
        currentIndex = 0;
    }
    
    updateDisplay();
    updateNavigation();
    showSuccess('问题顺序已随机打乱');
}
