"""
Conversation Visualizer

Generates an HTML report from the conversation logs.
"""
import json
import html
import re

def generate_html_report(data: dict, output_filename: str = "conversation_report.html"):
    """
    Generates an HTML report for the conversation.

    Args:
        data (dict): A dictionary containing all necessary data for visualization.
                     Expected keys:
                     - "task_data": dict containing travel preferences and constraints
                     - "conversation_style": str describing the user's conversation style
                     - "conversation_history": list of conversation messages
                     - "evaluation_results": dict mapping messages to their evaluation results
    """
    task_data = data.get("task_data", {})
    conversation_style = data.get("conversation_style", "No conversation style specified")
    conversation_history = data.get("conversation_history", [])
    evaluation_results = data.get("evaluation_results", {})

    # Helper to safely escape HTML characters in strings
    def escape(text):
        return html.escape(str(text))

    # --- CSS Styles ---
    css_styles = """
    body { font-family: sans-serif; margin: 0; padding: 0; background-color: #f4f4f4; color: #333; }
    .container { display: flex; max-width: 1600px; margin: 20px auto; background-color: #fff; box-shadow: 0 0 10px rgba(0,0,0,0.1); }
    .left-panel { width: 30%; padding: 20px; border-right: 1px solid #ddd; overflow-y: auto; max-height: 90vh; }
    .chat-panel { width: 70%; padding: 20px; overflow-y: auto; max-height: 90vh; }
    h2, h3 { color: #555; }
    pre { background-color: #eee; padding: 10px; border-radius: 5px; white-space: pre-wrap; word-wrap: break-word; font-size: 0.9em; }
    .chat-log { display: flex; flex-direction: column; }
    .message { margin-bottom: 15px; padding: 10px; border-radius: 8px; max-width: 80%; line-height: 1.4; }
    .user-message { background-color: #e1f5fe; align-self: flex-start; border: 1px solid #b3e5fc; cursor: pointer; }
    .agent-message { background-color: #c8e6c9; align-self: flex-end; border: 1px solid #a5d6a7; cursor: pointer; }
    .message-content { font-weight: bold; }
    .internal-details { 
        display: none; margin-top: 10px; padding: 10px; 
        background-color: #f9f9f9; border: 1px dashed #ccc; border-radius: 5px; 
        font-size: 0.85em; 
    }
    .internal-details pre { background-color: #fff; margin-bottom: 5px; }
    .tool-call-args { color: #007bff; }
    .tool-call-name { font-weight: bold; }
    .tool-response-content { color: #28a745; }
    .error-message { color: red; font-style: italic; }
    .preferences-panel { background-color: #fff3e0; padding: 10px; border-radius: 5px; margin: 15px 0; border: 1px solid #ffe0b2; }
    .conversation-style-panel { background-color: #e8eaf6; padding: 10px; border-radius: 5px; margin: 15px 0; border: 1px solid #c5cae9; }
    .eval-results { color: #d32f2f; }
    .response-type-container { margin: 8px 0 4px; }
    .response-type-tag {
        display: inline-block;
        background-color: #c8e6c9;
        color: #1b5e20;
        padding: 3px 10px;
        border-radius: 12px;
        font-size: 0.85em;
        font-weight: bold;
        margin-right: 6px;
        text-transform: uppercase;
    }
    .response-type-tag.invalid {
        background-color: #ffcdd2;
        color: #c62828;
    }
    .message-text {
        padding-top: 5px;
    }
    """

    # --- JavaScript for Toggling Details ---
    javascript_code = """
    function toggleDetails(elementId) {
        var details = document.getElementById(elementId);
        if (details.style.display === "none") {
            details.style.display = "block";
        } else {
            details.style.display = "none";
        }
    }
    """

    # --- HTML Body Construction ---
    html_parts = [
        "<!DOCTYPE html>",
        "<html><head><title>Travel Planner Conversation Report</title>",
        "<meta charset=\"UTF-8\">",
        f"<style>{css_styles}</style>",
        "<script src=\"https://cdn.jsdelivr.net/npm/marked/marked.min.js\"></script>",
        "</head><body>",
        "<div class='container'>",
        # Left Panel
        "<div class='left-panel'>",
        "<h2>Travel Preferences</h2>",
        "<div class='preferences-panel'>",
        f"<pre>{escape(json.dumps(task_data, indent=2))}</pre>",
        "</div>",
        "<h2>Conversation Style</h2>",
        "<div class='conversation-style-panel'>",
        f"<pre>{escape(conversation_style)}</pre>",
        "</div>",
        "</div>",  # Close left-panel
        # Chat Panel (Right Side)
        "<div class='chat-panel'>",
        "<h2>Conversation Log</h2>",
        "<div class='chat-log'>"
    ]

    detail_id_counter = 0
    for msg in conversation_history:
        detail_id_counter += 1
        detail_id = f"details-{detail_id_counter}"
        role = msg.get("role")
        content = msg.get("content", "")

        if role == "system":
            continue  # Skip system messages in the visualization
        
        if role == "user":
            html_parts.append(f"<div class='message user-message' onclick=\"toggleDetails('{detail_id}')\">")
            html_parts.append(f"<b>User:</b><div class='markdown-content message-text' style='display: inline-block;'>{escape(content)}</div>")
            
            # Add evaluation results if available
            if content in evaluation_results:
                html_parts.append(f"<div class='internal-details' id='{detail_id}'>")
                html_parts.append("<h4>Evaluation Results:</h4>")
                eval_result = evaluation_results[content]
                if isinstance(eval_result, dict):
                    for constraint, (passed, reason) in eval_result.items():
                        status = "✅ PASSED" if passed else "❌ FAILED"
                        html_parts.append(f"<div class='eval-results'>{constraint}: {status}")
                        if not passed and reason:
                            html_parts.append(f"<br>Reason: {escape(reason)}")
                        html_parts.append("</div>")
                else:
                    html_parts.append(f"<div class='eval-results'>{escape(eval_result)}</div>")
                html_parts.append("</div>")
            html_parts.append("</div>")  # Close user-message

        elif role == "assistant":
            html_parts.append(f"<div class='message agent-message' onclick=\"toggleDetails('{detail_id}')\">")
            html_parts.append("<b>Assistant:</b>")
            
            # Try to parse the JSON response for special formatting
            try:
                json_match = re.search(r"```json\s*(\{.*?\})\s*```", content, re.DOTALL)
                if json_match:
                    json_str = json_match.group(1)
                else:
                    json_str = content
                
                response_data = json.loads(json_str)
                response_types = response_data.get("type", [])
                response_content = response_data.get("content", "")

                if not isinstance(response_types, list):
                    response_types = ["invalid_type"]

                # Display type tags
                html_parts.append("<div class='response-type-container'>")
                for r_type in response_types:
                    html_parts.append(f"<span class='response-type-tag'>{escape(r_type)}</span>")
                html_parts.append("</div>")
                
                # Display content
                html_parts.append(f"<div class='markdown-content message-text'>{escape(response_content)}</div>")

            except (json.JSONDecodeError, TypeError):
                # Handle invalid format: show a tag and the raw content
                html_parts.append("<div class='response-type-container'>")
                html_parts.append("<span class='response-type-tag invalid'>Invalid Format</span>")
                html_parts.append("</div>")
                html_parts.append(f"<div class='markdown-content message-text'>{escape(content)}</div>")

            # Collapsible tool calls
            if msg.get("tool_calls"):
                html_parts.append(f"<div class='internal-details' id='{detail_id}'>")
                html_parts.append("<h4>Tool Calls:</h4>")
                for tc in msg["tool_calls"]:
                    func_name = tc.get("function", {}).get("name", "N/A")
                    func_args = tc.get("function", {}).get("arguments", "{}")
                    try:
                        func_args_pretty = json.dumps(json.loads(func_args), indent=2)
                    except:
                        func_args_pretty = func_args
                    
                    html_parts.append(f"<pre><span class='tool-call-name'>{escape(func_name)}</span><br>")
                    html_parts.append(f"<span class='tool-call-args'>{escape(func_args_pretty)}</span></pre>")
                html_parts.append("</div>")
            
            html_parts.append("</div>")  # Close agent-message

    html_parts.extend([
        "</div>",  # Close chat-log
        "</div>",  # Close chat-panel
        "</div>",  # Close container
        
        f"<script>{javascript_code}</script>",
        "<script>",
        "function htmlDecode(input) {",
        "  var doc = new DOMParser().parseFromString(input, 'text/html');",
        "  return doc.documentElement.textContent;",
        "}",
        "document.addEventListener('DOMContentLoaded', function() {",
        "  if (typeof marked !== 'undefined') {",
        "    marked.setOptions({ gfm: true, breaks: true, sanitize: false });",
        "    const markdownElements = document.querySelectorAll('.markdown-content');",
        "    markdownElements.forEach(el => {",
        "      const escapedMarkdown = el.innerHTML;",
        "      const rawMarkdown = htmlDecode(escapedMarkdown);",
        "      el.innerHTML = marked.parse(rawMarkdown);",
        "    });",
        "  } else {",
        "    console.error('marked.js not loaded');",
        "  }",
        "});",
        "</script>",
        "</body></html>"
    ])

    html_content = "\n".join(html_parts)
    try:
        with open(output_filename, "w", encoding="utf-8") as f:
            f.write(html_content)
        print(f"Conversation report saved to: {output_filename}")
    except Exception as e:
        print(f"Error writing HTML report: {e}")

if __name__ == '__main__':
    # Example usage (for testing the visualizer directly)
    sample_data = {
        "task_data": {
            "query": "I want to travel from New York to London",
            "org": "New York",
            "dest": "London",
            "days": 3,
            "budget": 5000,
            "preferences": {
                "max_price": 1000,
                "seat": "window",
                "hotel_stars": 4
            }
        },
        "conversation_style": "User is detail-oriented and prefers clear, specific information",
        "conversation_history": [
            {"role": "system", "content": "You are a travel planning assistant."},
            {"role": "user", "content": "I want to travel from New York to London"},
            {
                "role": "assistant",
                "content": "I'll help you find flights from New York to London.",
                "tool_calls": [
                    {
                        "function": {
                            "name": "search_flights",
                            "arguments": '{"origin": "NYC", "destination": "LON"}'
                        }
                    }
                ]
            },
            {
                "role": "assistant",
                "content": "This is a reply that does not follow the format."
            },
            {
                "role": "assistant",
                "content": '{"type": ["proposal", "question"], "content": "Here is your flight plan. It is under budget. Do you want to book it?"}'
            }
        ],
        "evaluation_results": {
            "I want to travel from New York to London": {
                "is_reasonable_visiting_city": [True, None],
                "is_valid_transportation": [False, "Missing return flight details"]
            }
        }
    }
    generate_html_report(sample_data, "sample_conversation_report.html")
    print("Sample report generated: sample_conversation_report.html") 