<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>JephHinter Chat Messages Viewer</title>
    <style>
        body {
            font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
            max-width: 1200px;
            margin: 0 auto;
            padding: 20px;
            background-color: #f5f5f5;
        }
        
        .header {
            background: linear-gradient(135deg, #667eea 0%, #764ba2 100%);
            color: white;
            padding: 20px;
            border-radius: 10px;
            margin-bottom: 20px;
            text-align: center;
        }
        
        .stats {
            display: grid;
            grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
            gap: 15px;
            margin-bottom: 20px;
        }
        
        .stat-card {
            background: white;
            padding: 15px;
            border-radius: 8px;
            box-shadow: 0 2px 4px rgba(0,0,0,0.1);
            text-align: center;
        }
        
        .stat-number {
            font-size: 2em;
            font-weight: bold;
            color: #667eea;
        }
        
        .chat-container {
            display: grid;
            gap: 20px;
        }
        
        .chat-card {
            background: white;
            border-radius: 10px;
            box-shadow: 0 4px 6px rgba(0,0,0,0.1);
            overflow: hidden;
        }
        
        .chat-header {
            background: #f8f9fa;
            padding: 15px 20px;
            border-bottom: 1px solid #e9ecef;
            display: flex;
            justify-content: space-between;
            align-items: center;
            flex-wrap: wrap;
            gap: 10px;
        }
        
        .chat-title {
            font-weight: bold;
            color: #495057;
        }
        
        .chat-meta {
            font-size: 0.9em;
            color: #6c757d;
        }
        
        .chat-content {
            padding: 20px;
        }
        
        .prompt-section {
            margin-bottom: 20px;
        }
        
        .prompt-label {
            font-weight: bold;
            color: #495057;
            margin-bottom: 8px;
            display: flex;
            align-items: center;
            gap: 8px;
        }
        
        .prompt-label::before {
            content: "💬";
            font-size: 1.2em;
        }
        
        .prompt-text {
            background: #f8f9fa;
            border: 1px solid #e9ecef;
            border-radius: 6px;
            padding: 15px;
            font-family: 'Monaco', 'Menlo', 'Ubuntu Mono', monospace;
            font-size: 0.9em;
            line-height: 1.5;
            white-space: pre-wrap;
            overflow-x: auto;
        }
        
        .hint-section {
            background: linear-gradient(135deg, #e3f2fd 0%, #bbdefb 100%);
            border-radius: 8px;
            padding: 20px;
            border-left: 4px solid #2196f3;
        }
        
        .hint-label {
            font-weight: bold;
            color: #1976d2;
            margin-bottom: 10px;
            display: flex;
            align-items: center;
            gap: 8px;
        }
        
        .hint-label::before {
            content: "💡";
            font-size: 1.2em;
        }
        
        .hint-text {
            font-size: 1.1em;
            line-height: 1.6;
            color: #1565c0;
        }
        
        .controls {
            background: white;
            padding: 20px;
            border-radius: 8px;
            box-shadow: 0 2px 4px rgba(0,0,0,0.1);
            margin-bottom: 20px;
        }
        
        .control-group {
            display: flex;
            gap: 15px;
            align-items: center;
            flex-wrap: wrap;
        }
        
        .control-group label {
            font-weight: bold;
            color: #495057;
        }
        
        .control-group select, .control-group input {
            padding: 8px 12px;
            border: 1px solid #ced4da;
            border-radius: 4px;
            font-size: 14px;
        }
        
        .search-box {
            flex: 1;
            min-width: 200px;
        }
        
        .loading {
            text-align: center;
            padding: 40px;
            color: #6c757d;
        }
        
        .error {
            background: #f8d7da;
            color: #721c24;
            padding: 15px;
            border-radius: 6px;
            border: 1px solid #f5c6cb;
            margin-bottom: 20px;
        }
        
        .no-results {
            text-align: center;
            padding: 40px;
            color: #6c757d;
            font-style: italic;
        }
        
        @media (max-width: 768px) {
            .chat-header {
                flex-direction: column;
                align-items: flex-start;
            }
            
            .control-group {
                flex-direction: column;
                align-items: stretch;
            }
            
            .search-box {
                min-width: auto;
            }
        }
    </style>
</head>
<body>
    <div class="header">
        <h1>🤖 JephHinter Chat Messages Viewer</h1>
        <p>View prompts and generated hints from JephHinter sessions</p>
    </div>
    
    <div class="controls">
        <div class="control-group">
            <label for="fileInput">Load Chat Data:</label>
            <input type="file" id="fileInput" accept=".json" multiple>
            <button onclick="loadSampleData()">Load Sample Data</button>
        </div>
        <div class="control-group">
            <label for="taskFilter">Filter by Task:</label>
            <select id="taskFilter">
                <option value="">All Tasks</option>
            </select>
            <label for="modelFilter">Filter by Model:</label>
            <select id="modelFilter">
                <option value="">All Models</option>
            </select>
            <label for="searchBox">Search:</label>
            <input type="text" id="searchBox" class="search-box" placeholder="Search in prompts and hints...">
        </div>
    </div>
    
    <div id="stats" class="stats" style="display: none;">
        <div class="stat-card">
            <div class="stat-number" id="totalChats">0</div>
            <div>Total Chats</div>
        </div>
        <div class="stat-card">
            <div class="stat-number" id="totalTasks">0</div>
            <div>Tasks Covered</div>
        </div>
        <div class="stat-card">
            <div class="stat-number" id="totalModels">0</div>
            <div>Models Used</div>
        </div>
        <div class="stat-card">
            <div class="stat-number" id="avgPromptLength">0</div>
            <div>Avg Prompt Length</div>
        </div>
    </div>
    
    <div id="content" class="chat-container">
        <div class="loading">Load chat data to view prompts and hints</div>
    </div>

    <script>
        let chatData = [];
        let filteredData = [];
        
        // File input handler
        document.getElementById('fileInput').addEventListener('change', handleFileSelect);
        
        // Filter handlers
        document.getElementById('taskFilter').addEventListener('change', applyFilters);
        document.getElementById('modelFilter').addEventListener('change', applyFilters);
        document.getElementById('searchBox').addEventListener('input', applyFilters);
        
        function handleFileSelect(event) {
            const files = event.target.files;
            if (files.length === 0) return;
            
            const promises = Array.from(files).map(file => {
                return new Promise((resolve, reject) => {
                    const reader = new FileReader();
                    reader.onload = e => {
                        try {
                            const data = JSON.parse(e.target.result);
                            resolve(data);
                        } catch (error) {
                            reject(error);
                        }
                    };
                    reader.readAsText(file);
                });
            });
            
            Promise.all(promises)
                .then(results => {
                    chatData = results.flat();
                    updateStats();
                    updateFilters();
                    applyFilters();
                })
                .catch(error => {
                    showError('Error loading files: ' + error.message);
                });
        }
        
        function loadSampleData() {
            // Sample data structure for demonstration
            const sampleData = [
                {
                    metadata: {
                        task_name: "miniwob.use-colorwheel-2",
                        seed: "1",
                        timestamp: "2024-12-01T14:30:22",
                        model_name: "claude-3-sonnet-20240229",
                        agent_name: "JephHinter",
                        domain_name: "miniwob",
                        user_name: "auto",
                        source: "jeph_hinter",
                        trace_count: 2,
                        trace_steps: 15
                    },
                    prompts: {
                        system_prompt: "You are a hint-generating agent that creates actionable, generalizable guidance for web automation tasks. Your goal is to extract strategic insights from execution traces that can help agents succeed at similar tasks.\n\nIMPORTANT REQUIREMENTS FOR HINT GENERATION:\n\n1. GENERALIZABILITY: Make hints general enough to apply to similar tasks, not just this specific task. Avoid including:\n   - Specific usernames, names, or personal information\n   - Task-specific text content or exact values\n   - Specific element IDs or numbers (like [123])\n   - Domain-specific details that don't apply broadly\n\n2. SPECIFICITY: Include relevant UI elements and their exact text when they represent common patterns:\n   - Button names like 'Submit', 'Create', 'Delete'\n   - Link text like 'Settings', 'Profile', 'Help'\n   - Form field labels like 'Email', 'Password', 'Search'\n   - Page titles or headings that indicate context\n\n3. STRUCTURE: Format hints as actionable guidance with this pattern:\n   'When [context/state], if you want to [goal], you should [specific action]. The correct approach is [step-by-step guidance].'\n\n4. LENGTH: Keep hints concise (1-2 sentences maximum) and write as a single line without breaks.\n\n5. FOCUS AREAS:\n   - Common pitfalls or errors to avoid\n   - Successful strategies and patterns\n   - Important UI elements to look for\n   - Step-by-step sequences when needed\n   - Context clues that indicate the right approach\n\n6. ACTIONABILITY: Provide specific, executable guidance rather than explanations of why something works.\n\nAnalyze the execution traces and generate hints that follow these requirements.",
                        user_prompt: "Task: miniwob.use-colorwheel-2\n\n=== EXECUTION TRACES ===\n\n--- Trace 1 (SUCCESSFUL) ---\nStep 1:\nAXTree: <div id='colorwheel'><input type='color' id='color-input'><button id='submit-btn'>Submit</button></div>\nAgent's reasoning: I need to find the color input and submit button\nAction taken: click('color-input')\nCurrent reward: 0\n\nStep 2:\nAXTree: <div id='colorwheel'><input type='color' id='color-input' value='#ff0000'><button id='submit-btn'>Submit</button></div>\nAgent's reasoning: Now I need to submit the color\nAction taken: click('submit-btn')\nCurrent reward: 1\n\n--- Trace 2 (FAILED) ---\nStep 1:\nAXTree: <div id='colorwheel'><input type='color' id='color-input'><button id='submit-btn'>Submit</button></div>\nAgent's reasoning: I should click the submit button directly\nAction taken: click('submit-btn')\nCurrent reward: 0\n\n=== HINT GENERATION ===\nBased on the multiple traces above, provide a concise, actionable hint that identifies common patterns, successful strategies, and key insights for completing this task."
                    },
                    llm_response: {
                        raw_response: "ClaudeResponse",
                        think: "When using a color picker interface, if you want to select a color and submit it successfully, you should first click on the color input field to open the color picker, then select your desired color, and finally click the submit button. The correct approach is to interact with the color input before submitting, as the system requires a color selection to be made before the submit action will be accepted.",
                        response_type: "ClaudeResponse"
                    },
                    trace_data: [],
                    generated_hint: "When using a color picker interface, if you want to select a color and submit it successfully, you should first click on the color input field to open the color picker, then select your desired color, and finally click the submit button. The correct approach is to interact with the color input before submitting, as the system requires a color selection to be made before the submit action will be accepted."
                },
                {
                    metadata: {
                        task_name: "workarena.servicenow.create-incident",
                        seed: "42",
                        timestamp: "2024-12-01T15:45:33",
                        model_name: "gpt-4o",
                        agent_name: "JephHinter",
                        domain_name: "workarena",
                        user_name: "auto",
                        source: "jeph_hinter",
                        trace_count: 1,
                        trace_steps: 8
                    },
                    prompts: {
                        system_prompt: "You are a hint-generating agent that creates actionable, generalizable guidance for web automation tasks. Your goal is to extract strategic insights from execution traces that can help agents succeed at similar tasks.\n\nIMPORTANT REQUIREMENTS FOR HINT GENERATION:\n\n1. GENERALIZABILITY: Make hints general enough to apply to similar tasks, not just this specific task. Avoid including:\n   - Specific usernames, names, or personal information\n   - Task-specific text content or exact values\n   - Specific element IDs or numbers (like [123])\n   - Domain-specific details that don't apply broadly\n\n2. SPECIFICITY: Include relevant UI elements and their exact text when they represent common patterns:\n   - Button names like 'Submit', 'Create', 'Delete'\n   - Link text like 'Settings', 'Profile', 'Help'\n   - Form field labels like 'Email', 'Password', 'Search'\n   - Page titles or headings that indicate context\n\n3. STRUCTURE: Format hints as actionable guidance with this pattern:\n   'When [context/state], if you want to [goal], you should [specific action]. The correct approach is [step-by-step guidance].'\n\n4. LENGTH: Keep hints concise (1-2 sentences maximum) and write as a single line without breaks.\n\n5. FOCUS AREAS:\n   - Common pitfalls or errors to avoid\n   - Successful strategies and patterns\n   - Important UI elements to look for\n   - Step-by-step sequences when needed\n   - Context clues that indicate the right approach\n\n6. ACTIONABILITY: Provide specific, executable guidance rather than explanations of why something works.\n\nAnalyze the execution traces and generate hints that follow these requirements.",
                        user_prompt: "Task: workarena.servicenow.create-incident\n\n=== EXECUTION TRACE ===\n\nStep 1:\nAXTree: <div class='incident-form'><h2>Create New Incident</h2><input type='text' placeholder='Short description'><textarea placeholder='Detailed description'></textarea><button class='btn-primary'>Create Incident</button></div>\nAgent's reasoning: I need to fill out the incident form\nAction taken: type('Short description', 'System login issue')\nCurrent reward: 0\n\nStep 2:\nAXTree: <div class='incident-form'><h2>Create New Incident</h2><input type='text' placeholder='Short description' value='System login issue'><textarea placeholder='Detailed description'></textarea><button class='btn-primary'>Create Incident</button></div>\nAgent's reasoning: Now I need to fill the detailed description\nAction taken: type('Detailed description', 'Users cannot access the system')\nCurrent reward: 0\n\nStep 3:\nAXTree: <div class='incident-form'><h2>Create New Incident</h2><input type='text' placeholder='Short description' value='System login issue'><textarea placeholder='Detailed description' value='Users cannot access the system'></textarea><button class='btn-primary'>Create Incident</button></div>\nAgent's reasoning: Now I can submit the form\nAction taken: click('Create Incident')\nCurrent reward: 1\n\nThis was a successful trace.\n\n=== HINT GENERATION ===\nBased on this trace, provide a concise, actionable hint that would help an agent avoid common mistakes and succeed at this task."
                    },
                    llm_response: {
                        raw_response: "GPT4Response",
                        think: "When creating an incident in a service management system, if you want to successfully submit the form, you should fill in all required fields including the short description and detailed description before clicking the create button. The correct approach is to ensure both text fields contain meaningful content before attempting to submit the incident form.",
                        response_type: "GPT4Response"
                    },
                    trace_data: [],
                    generated_hint: "When creating an incident in a service management system, if you want to successfully submit the form, you should fill in all required fields including the short description and detailed description before clicking the create button. The correct approach is to ensure both text fields contain meaningful content before attempting to submit the incident form."
                }
            ];
            
            chatData = sampleData;
            updateStats();
            updateFilters();
            applyFilters();
        }
        
        function updateStats() {
            if (chatData.length === 0) return;
            
            const stats = {
                totalChats: chatData.length,
                totalTasks: new Set(chatData.map(c => c.metadata.task_name)).size,
                totalModels: new Set(chatData.map(c => c.metadata.model_name)).size,
                avgPromptLength: Math.round(
                    chatData.reduce((sum, c) => {
                        const systemLen = c.prompts.system_prompt?.length || 0;
                        const userLen = c.prompts.user_prompt?.length || 0;
                        return sum + systemLen + userLen;
                    }, 0) / chatData.length
                )
            };
            
            document.getElementById('totalChats').textContent = stats.totalChats;
            document.getElementById('totalTasks').textContent = stats.totalTasks;
            document.getElementById('totalModels').textContent = stats.totalModels;
            document.getElementById('avgPromptLength').textContent = stats.avgPromptLength;
            document.getElementById('stats').style.display = 'grid';
        }
        
        function updateFilters() {
            const tasks = [...new Set(chatData.map(c => c.metadata.task_name))].sort();
            const models = [...new Set(chatData.map(c => c.metadata.model_name))].sort();
            
            const taskFilter = document.getElementById('taskFilter');
            const modelFilter = document.getElementById('modelFilter');
            
            // Clear existing options
            taskFilter.innerHTML = '<option value="">All Tasks</option>';
            modelFilter.innerHTML = '<option value="">All Models</option>';
            
            // Add new options
            tasks.forEach(task => {
                const option = document.createElement('option');
                option.value = task;
                option.textContent = task;
                taskFilter.appendChild(option);
            });
            
            models.forEach(model => {
                const option = document.createElement('option');
                option.value = model;
                option.textContent = model;
                modelFilter.appendChild(option);
            });
        }
        
        function applyFilters() {
            const taskFilter = document.getElementById('taskFilter').value;
            const modelFilter = document.getElementById('modelFilter').value;
            const searchQuery = document.getElementById('searchBox').value.toLowerCase();
            
            filteredData = chatData.filter(chat => {
                const matchesTask = !taskFilter || chat.metadata.task_name === taskFilter;
                const matchesModel = !modelFilter || chat.metadata.model_name === modelFilter;
                const matchesSearch = !searchQuery || 
                    chat.prompts.system_prompt.toLowerCase().includes(searchQuery) ||
                    chat.prompts.user_prompt.toLowerCase().includes(searchQuery) ||
                    chat.generated_hint.toLowerCase().includes(searchQuery);
                
                return matchesTask && matchesModel && matchesSearch;
            });
            
            renderChats();
        }
        
        function renderChats() {
            const content = document.getElementById('content');
            
            if (filteredData.length === 0) {
                content.innerHTML = '<div class="no-results">No chat messages match the current filters</div>';
                return;
            }
            
            content.innerHTML = filteredData.map((chat, index) => `
                <div class="chat-card">
                    <div class="chat-header">
                        <div>
                            <div class="chat-title">${chat.metadata.task_name}</div>
                            <div class="chat-meta">
                                Seed: ${chat.metadata.seed} | 
                                Model: ${chat.metadata.model_name} | 
                                ${new Date(chat.metadata.timestamp).toLocaleString()}
                            </div>
                        </div>
                        <div class="chat-meta">
                            ${chat.metadata.trace_count} traces, ${chat.metadata.trace_steps} steps
                        </div>
                    </div>
                    <div class="chat-content">
                        <div class="prompt-section">
                            <div class="prompt-label">System Prompt</div>
                            <div class="prompt-text">${escapeHtml(chat.prompts.system_prompt)}</div>
                        </div>
                        <div class="prompt-section">
                            <div class="prompt-label">User Prompt (Trace Analysis)</div>
                            <div class="prompt-text">${escapeHtml(chat.prompts.user_prompt)}</div>
                        </div>
                        <div class="hint-section">
                            <div class="hint-label">Generated Hint</div>
                            <div class="hint-text">${escapeHtml(chat.generated_hint)}</div>
                        </div>
                    </div>
                </div>
            `).join('');
        }
        
        function escapeHtml(text) {
            const div = document.createElement('div');
            div.textContent = text;
            return div.innerHTML;
        }
        
        function showError(message) {
            const content = document.getElementById('content');
            content.innerHTML = `<div class="error">${message}</div>`;
        }
    </script>
</body>
</html> 