
import os
import re
import json
from bs4 import BeautifulSoup
import traceback
from openai import OpenAI
from .base_layout_extractor import base_layout_extractor
from .mathjax_utils import inject_mathjax_and_group_equations
from .stable_mathjax import inject_mathjax_stable, validate_mathjax_injection


class EnhancementAgent:

    
    def __init__(self, config_path=None, llm_client=None):
        # Load configuration
        self.config = self._load_config(config_path)
        
        # LLM client for intelligent enhancement generation (required)
        self.llm_client = llm_client
        if not self.llm_client:
            raise ValueError("llm_client is required for EnhancementAgent")
        
        # Load enhancement prompts
        self._load_enhancement_prompts()
    


    def _load_enhancement_prompts(self):
        """Load enhancement prompts"""
        try:
            current_dir = os.path.dirname(os.path.abspath(__file__))
            
            # System prompt for enhancement
            system_prompt_path = os.path.join(current_dir, 'prompts', 'enhancement_system_prompt.txt')
            if os.path.exists(system_prompt_path):
                with open(system_prompt_path, 'r', encoding='utf-8') as f:
                    self.system_prompt = f.read()
            else:
                self.system_prompt = self._get_default_system_prompt()
                
            # Enhancement prompt template
            enhancement_prompt_path = os.path.join(current_dir, 'prompts', 'enhancement_prompt.txt')
            if os.path.exists(enhancement_prompt_path):
                with open(enhancement_prompt_path, 'r', encoding='utf-8') as f:
                    self.enhancement_prompt_template = f.read()
            else:
                self.enhancement_prompt_template = self._get_default_enhancement_prompt()
                
        except Exception as e:
            print(f"Warning: Could not load enhancement prompts: {e}")
            self.system_prompt = self._get_default_system_prompt()
            self.enhancement_prompt_template = self._get_default_enhancement_prompt()

    def _load_config(self, config_path=None):
        """Load configuration from JSON file"""
        if config_path is None:
            # Default config path
            current_dir = os.path.dirname(os.path.abspath(__file__))
            config_path = os.path.join(current_dir, 'enhancement_config.json')
        
        try:
            with open(config_path, 'r', encoding='utf-8') as f:
                return json.load(f)
        except Exception as e:
            print(f"Warning: Could not load enhancement config: {e}")
            # Return default config
            return {
                "animation_settings": {"enable_animations": True, "animation_duration": "0.8s"},
                "text_enhancement": {"enable_text_borders": True, "enable_text_backgrounds": True},
                "image_enhancement": {"enable_image_effects": True, "border_radius": "8px"},
                "decorative_elements": {"enable_for_text_heavy": True, "max_shapes": 4}
            }

    def _get_default_system_prompt(self):
        """Get default system prompt for enhancement"""
        return """You are an expert CSS and HTML enhancement specialist. Your task is to enhance existing HTML slides by adding sophisticated visual effects, animations, and styling that complement the original design.

Key Principles:
1. PRESERVE the original layout and content structure completely
2. ADD enhancements that are contextually appropriate to the slide's content type
3. MAINTAIN consistency with the provided CSS style template
4. CREATE smooth, professional animations and transitions
5. ENSURE all enhancements are responsive and accessible

Enhancement Guidelines:
- For text-heavy slides: Add subtle borders, backgrounds, reading enhancements, and elegant typography
- For image-heavy slides: Add sophisticated image effects, gallery-style layouts, and visual hierarchy
- For minimal content: Add elegant decorative elements, spacious layouts, and refined aesthetics
- For balanced content: Apply moderate enhancements that complement both text and images

Technical Requirements:
- Use modern CSS animations and transitions
- Apply staggered animation delays for multiple elements
- Include hover effects and micro-interactions
- Ensure all colors and fonts strictly match the original style template
- Add decorative background patterns only when appropriate
- Make all animations smooth, professional, and performant

Your output should be the complete enhanced HTML with all CSS embedded in <style> tags."""

    def _get_default_enhancement_prompt(self):
        """Get default enhancement prompt template"""
        return """You are tasked with enhancing HTML slide content while maintaining STRICT consistency with the template style.

CURRENT HTML TO ENHANCE:
{html_content}

CONTENT ANALYSIS:
{content_analysis}

TEMPLATE STYLE RULES (MUST BE STRICTLY FOLLOWED):
{style_template}

ENHANCEMENT CONFIGURATION:
- Content Type: {content_type}
- Animations Enabled: {animations_enabled}
- Animation Duration: {animation_duration}
- Decorative Elements: {decorative_enabled}
- Slide Index: {slide_index}

STRICT STYLE CONSISTENCY REQUIREMENTS:

1. **Header/Title Styling (MANDATORY):**
   - MUST use EXACTLY the same font-family, font-size, and color as template headers
   - MUST maintain consistent spacing and margins for all headers
   - MUST use identical text-transform properties (e.g., uppercase, capitalize)
   - MUST apply the same font-weight and line-height values
   - Example: If template uses "font-family: 'Roboto', sans-serif; color: #2c3e50;" for h1, ALL slides must use these exact values

2. **Content Text Styling (MANDATORY):**
   - MUST use EXACTLY the same font-family and base font-size as template body text
   - MUST maintain consistent paragraph spacing and line heights
   - MUST use identical text colors and link colors
   - MUST apply the same list styles and indentation
   - Example: If template uses 16px Lato for body text, ALL slides must use this

3. **Color Scheme (MANDATORY):**
   - MUST use EXACTLY the same colors as defined in template
   - Primary color: Use for main headings and key elements
   - Secondary color: Use for subheadings and accents
   - Text color: Must match template exactly
   - Background color: Must be identical across all slides
   - NO custom colors allowed - use ONLY colors from template

4. **Layout and Spacing (MANDATORY):**
   - MUST maintain consistent margins and padding across all slides
   - MUST use identical grid systems and alignment rules
   - MUST follow template's spacing hierarchy
   - Example: If template uses 2rem bottom margin for sections, apply this consistently

5. **Visual Elements:**
   - Borders: Use EXACTLY the same style as template
   - Shadows: Must match template's shadow properties
   - Rounded corners: Use identical border-radius values
   - Background patterns: Must complement template style

6. **Animations and Transitions:**
   - Duration: {animation_duration} (consistent across all slides)
   - Style: Must complement template's visual language
   - Timing: Use consistent delays and easing functions

ENFORCEMENT MECHANISMS:
1. Extract and store template's exact values for:
   - All font properties (family, size, weight, style)
   - Color scheme (primary, secondary, text, background)
   - Spacing values (margins, padding, line-height)
   - Border styles and border-radius
   - Shadow properties

2. Apply these values consistently:
   - Use CSS variables for core properties
   - Apply identical class structures
   - Maintain consistent naming conventions

3. Quality Checks:
   - Verify font consistency across all text elements
   - Confirm color usage matches template exactly
   - Validate spacing and alignment consistency
   - Ensure identical styling for similar elements

CRITICAL REQUIREMENTS:
- NEVER introduce new fonts or colors
- NEVER override template's base styles
- ALWAYS use template's exact values
- MAINTAIN perfect consistency across all slides
- ALL CSS must be in <style> tags within <head>
- PRESERVE all original content structure

OUTPUT REQUIREMENTS:
1. Return complete HTML with all styles embedded
2. Include CSS variables for consistent values
3. Use template's exact class structure
4. Maintain identical naming conventions

Please enhance this slide now, ensuring PERFECT consistency with the template:"""

    def enhance_html_slide(self, html_content, slide_data=None, style_template=None, slide_index=0, feedback=None, previous_enhanced_html=None, preserve_layout=True):
   
        try:
            # Extract base layout constraints if enabled
            layout_constraints_data = None
            layout_constraints_prompt = ""
            
            if preserve_layout and html_content and not previous_enhanced_html:
                print(f"Extracting base layout constraints for slide {slide_index}...")
                try:
                    layout_constraints_data = base_layout_extractor.extract_base_layout_sync(html_content)
                    layout_constraints_prompt = base_layout_extractor.create_layout_constraints_prompt(layout_constraints_data)
                    print(f"Successfully extracted constraints for {len(layout_constraints_data.get('elements', []))} elements")
                except Exception as e:
                    print(f"Warning: Could not extract layout constraints: {e}")
                    layout_constraints_prompt = ""
                
            # Analyze slide content
            soup = BeautifulSoup(html_content, 'html.parser')
            slide_analysis = self._analyze_slide_content(soup, slide_data)
            
            print(f"Rendering and enhancing slide {slide_index} with selected style - Content type: {slide_analysis.get('content_type')}")
            
            # Generate enhancement using LLM with layout constraints
            enhanced_html = self._generate_llm_enhancement(
                html_content, slide_analysis, style_template, slide_index, feedback, previous_enhanced_html, layout_constraints_prompt
            )
            
            return enhanced_html if enhanced_html else html_content
            
        except Exception as e:
            print(f"Error in enhancement agent: {e}")
            traceback.print_exc()
            return html_content  # Return original on error

    def _analyze_slide_content(self, soup, slide_data=None):
        """Analyze slide content to determine enhancement strategy"""
        analysis = {
            "text_heavy": False,
            "image_heavy": False,
            "has_equations": False,
            "text_blocks": [],
            "images": [],
            "content_type": "balanced"
        }
        
        # Count text content
        text_content = soup.get_text(strip=True)
        text_length = len(text_content)
        
        # Find text blocks
        text_elements = soup.find_all(['p', 'div', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6'])
        meaningful_text_blocks = [el for el in text_elements 
                                if el.get_text(strip=True) and len(el.get_text(strip=True)) > 20]
        analysis["text_blocks"] = meaningful_text_blocks
        
        # Find images
        images = soup.find_all(['img', 'figure'])
        placeholders = soup.find_all(attrs={"data-image-id": True})
        analysis["images"] = images + placeholders
        
        # Check for equations
        if '$' in text_content or 'equation' in text_content.lower():
            analysis["has_equations"] = True
        
        # Determine content type
        if len(analysis["images"]) == 0 and text_length > 500:
            analysis["content_type"] = "text_heavy"
            analysis["text_heavy"] = True
        elif len(analysis["images"]) > 2:
            analysis["content_type"] = "image_heavy" 
            analysis["image_heavy"] = True
        elif len(analysis["images"]) == 0 and text_length < 200:
            analysis["content_type"] = "minimal_text"
        
        return analysis

    def _generate_llm_enhancement(self, html_content, slide_analysis, style_template, slide_index, feedback=None, previous_enhanced_html=None, layout_constraints_prompt=""):
        try:
            # Prepare content analysis summary
            content_analysis = self._format_content_analysis(slide_analysis)
            
            # Get configuration settings
            anim_settings = self.config.get("animation_settings", {})
            decorative_settings = self.config.get("decorative_elements", {})
            
            # Check if this is a revision based on checker feedback
            if feedback and previous_enhanced_html:
                # Revision mode: use feedback to adjust the previously enhanced HTML
                prompt = self._create_revision_prompt(previous_enhanced_html, feedback, style_template, layout_constraints_prompt)
                print(f"Enhancement agent operating in revision mode with checker feedback")
            else:
                # Initial enhancement mode
                base_prompt = self.enhancement_prompt_template.format(
                    html_content=html_content,
                    content_analysis=content_analysis,
                    style_template=style_template or "/* No specific style template provided */",
                    content_type=slide_analysis.get("content_type", "balanced"),
                    animations_enabled=anim_settings.get("enable_animations", True),
                    decorative_enabled=self._should_add_decorative_elements(slide_analysis, decorative_settings),
                    animation_duration=anim_settings.get("animation_duration", "0.8s"),
                    slide_index=slide_index
                )
                
                if layout_constraints_prompt:
                    prompt = f"{base_prompt}\n\n{layout_constraints_prompt}"
                    print(f"Enhancement agent operating in initial enhancement mode with layout constraints")
                else:
                    prompt = base_prompt
                    print(f"Enhancement agent operating in initial enhancement mode without layout constraints")
   
            messages = [
                {"role": "system", "content": self.system_prompt},
                {"role": "user", "content": prompt}
            ]
            
            print(f"=========================Calling LLM for slide rendering and enhancement with selected style================================================")
        
            model_name = "xxxs"
    

            
            response = self.llm_client.chat.completions.create(
                model=model_name,
                messages=messages,
                temperature=0.2,                                             
            )
            
            enhanced_html = response.choices[0].message.content
            

            enhanced_html = self._clean_llm_response(enhanced_html)
            
            eq_cfg = self.config.get("equation_settings", {})
            if enhanced_html:
                want_render = eq_cfg.get("enabled", True) and eq_cfg.get("render", "mathjax") != "none"
                if want_render and (slide_analysis.get("has_equations", False) or True):
                        if not validate_mathjax_injection(enhanced_html):
                            enhanced_html = inject_mathjax_stable(enhanced_html, None)
                            if not validate_mathjax_injection(enhanced_html):
                                enhanced_html = inject_mathjax_and_group_equations(enhanced_html, None, self.config)
 
      
            
            print(f"Successfully generated rendering and enhancement for slide {slide_index}")
            return enhanced_html
            
        except Exception as e:
            print(f"Error generating LLM enhancement for slide {slide_index}: {e}")
            traceback.print_exc()
            return None

    def _should_add_decorative_elements(self, slide_analysis, decorative_settings):
        """Determine if decorative elements should be added"""
        if slide_analysis.get("text_heavy") and decorative_settings.get("enable_for_text_heavy", True):
            return True
        elif slide_analysis.get("content_type") == "minimal_text" and decorative_settings.get("enable_for_minimal_text", True):
            return True
        return False

    def _format_content_analysis(self, analysis):
        """Format content analysis for LLM prompt"""
        summary = []
        
        summary.append(f"Content Type: {analysis.get('content_type', 'balanced')}")
        summary.append(f"Text Heavy: {analysis.get('text_heavy', False)}")
        summary.append(f"Image Heavy: {analysis.get('image_heavy', False)}")
        summary.append(f"Has Equations: {analysis.get('has_equations', False)}")
        summary.append(f"Number of Text Blocks: {len(analysis.get('text_blocks', []))}")
        summary.append(f"Number of Images: {len(analysis.get('images', []))}")
        
        # Add specific recommendations based on analysis
        if analysis.get("text_heavy"):
            summary.append("Recommendation: Focus on text readability, add subtle borders and backgrounds")
        elif analysis.get("image_heavy"):
            summary.append("Recommendation: Enhance images with sophisticated effects, create gallery-style layouts")
        elif analysis.get("content_type") == "minimal_text":
            summary.append("Recommendation: Add elegant decorative elements, use spacious layouts")
        else:
            summary.append("Recommendation: Apply balanced enhancements that complement both text and images")
            
        return "\n".join(summary)

    def _create_revision_prompt(self, previous_enhanced_html, feedback, style_template, layout_constraints_prompt=""):
        """Create a revision prompt for enhancement agent based on checker feedback"""
        revision_prompt = f"""You are an expert HTML enhancement specialist. Your task is to revise the existing enhanced HTML slide based on specific feedback from a layout quality checker.

CURRENT ENHANCED HTML TO REVISE:
{previous_enhanced_html}

CHECKER FEEDBACK TO ADDRESS:
{feedback}

ORIGINAL STYLE TEMPLATE (MUST MAINTAIN CONSISTENCY):
{style_template or "/* No specific style template provided */"}

REVISION INSTRUCTIONS:
1. **Address the specific issues** mentioned in the checker feedback
2. **Maintain the overall design aesthetic** and style consistency
3. **Preserve all original content** - do not remove or change text/images
4. **Keep existing enhancements** that are working well
5. **Make targeted improvements** only where feedback indicates problems
6. **Ensure all colors, fonts, and styling** remain consistent with the original template

CRITICAL REQUIREMENTS:
- Return the complete revised HTML document
- All CSS must be embedded in <style> tags within <head>
- Focus specifically on the issues mentioned in the feedback
- Do not make unnecessary changes to elements that are working well
- Maintain professional, smooth animations and transitions

Please provide the revised HTML now:"""

        if layout_constraints_prompt:
            revision_prompt += f"\n\n{layout_constraints_prompt}"
            
        return revision_prompt

    def _clean_llm_response(self, response_text):
        """Clean LLM response to extract HTML content"""
        if not response_text:
            return None
            
        # Remove markdown code blocks
        code_start = response_text.find("```html")
        if code_start != -1:
            code_end = response_text.rfind("```")
            if code_end > code_start:
                response_text = response_text[code_start + len("```html"):code_end].strip()
        
        # Look for HTML document start
        search_content = response_text.lower()
        start_index = search_content.find('<!doctype html>')
        if start_index == -1:
            start_index = search_content.find('<html')
        
        if start_index != -1:
            response_text = response_text[start_index:]
        
        return response_text.strip()


def enhance_slide_html(html_content, slide_data=None, style_template=None, slide_index=0, llm_client=None, feedback=None, previous_enhanced_html=None, preserve_layout=True):
    if not llm_client:
        print("Warning: No LLM client provided, returning original HTML")
        return html_content
        
    agent = EnhancementAgent(llm_client=llm_client)
    return agent.enhance_html_slide(html_content, slide_data, style_template, slide_index, feedback, previous_enhanced_html, preserve_layout)


if __name__ == "__main__":
    pass
