import os
import re
import uuid
import json
import http.client
from pathlib import Path
from typing import Dict, Any, List, Tuple
from io import BytesIO

import requests
from PIL import Image



class ImageGenerator:

    API_HOST = ""
    API_KEY = ""

    @classmethod
    def generate_image(cls, *, prompt: str, save_path: str, max_width: int = 800, max_height: int = 600, **kwargs) -> Tuple[str, Tuple[int, int]]:
        """Generate image using lingximoyu API
        
        Args:
            prompt: The prompt to generate the image from
            save_path: Where to save the generated image
            max_width: Maximum width for the generated image.
            max_height: Maximum height for the generated image.
        """
        try:
            
            payload = json.dumps({
                "model": "xxx",
                "stream": False,
                "messages": [
                    {
                        "role": "user",
                        "content": prompt
                    }
                ],
                "size": f"{max_width}x{max_height}" 
            })
            
    
            print(f"Sending request to generate image with prompt: {prompt}")
            conn.request("POST", "/v1/chat/completions", payload, headers)
            res = conn.getresponse()
            data = res.read()
            response = json.loads(data.decode("utf-8"))

            content = response['choices'][0]['message']['content']
            image_url_match = re.search(r'!\[.*?\]\((.*?)\)', content)
            if not image_url_match:
                raise ValueError("Cannot extract image URL from response")
            
            image_url = image_url_match.group(1)
            print(f"Image URL extracted: {image_url}")
            
            image_response = requests.get(image_url)
            image_response.raise_for_status()

            os.makedirs(os.path.dirname(save_path), exist_ok=True)
            image = Image.open(BytesIO(image_response.content))
            image.save(save_path, "PNG")
            print(f"Image saved to: {save_path}")
            
            return save_path, image.size
            
        except Exception as e:
            print(f"Error generating image: {str(e)}")
            raise
            
        finally:
            if 'conn' in locals():
                conn.close()


class VisualizationHelper:
   

    def __init__(self, save_dir: Path, avg_width: int = None, avg_height: int = None):
        self._figure_counter = 0
        self.save_dir = save_dir
        self.save_dir.mkdir(parents=True, exist_ok=True)
        self.avg_width = avg_width
        self.avg_height = avg_height
        self.fast_mode = str(os.environ.get('P2P_FAST_MODE', '0')).lower() in {'1', 'true', 'yes', 'y'}
        
        self._default_width = 1024
        self._default_height = 768
        self._fast_width = 800
        self._fast_height = 600

    def set_figure_start_index(self, start_index: int):
        """Sets the starting index for generated figures."""
        self._figure_counter = start_index

    def process_slide_content(self, slide_data: Dict[str, Any], force: bool = False) -> Dict[str, Any]:
        content = slide_data.get("content") or {}
        visual_elements: List[Dict[str, Any]] = content.get("visual_elements") or []
        core_points: List[str] = content.get("core_points") or []
        title: str = slide_data.get("slide_title") or ""

        print(f"\n=== Processing slide content for image generation ===")
        print(f"Title: {title}")
        
        # Get scholar request information
        scholar_req = content.get("scholar_request") or {}
        reason = (scholar_req.get("reason") or "").strip()
        rationale = "generated due to explicit image request"
        
        print(f"Scholar request reason: '{reason}'")
        if not reason:
            print("No reason provided in scholar_request, skipping image generation")
            return slide_data
            
        prompt = reason
        print(f"Using scholar request reason as prompt: {prompt}")

        # Determine save path
        new_id = self._next_figure_id()
        file_stem = new_id.replace(" ", "_")
        save_path = self.save_dir / f"{file_stem}.png"

        if save_path.exists():
            print(f"\nImage already exists at {save_path}, reusing existing image")
            try:
                with Image.open(save_path) as img:
                    width, height = img.size
            except Exception as e:
                print(f"Error reading existing image dimensions: {e}")
                return slide_data
        else:
            try:
                print(f"\n============================Generating image============================")
                print(f"Prompt: {prompt}")
                print(f"Save path: {save_path}")
                

                target_width = self.avg_width or (self._default_width if not self.fast_mode else self._fast_width)
                target_height = self.avg_height or (self._default_height if not self.fast_mode else self._fast_height)
           
                target_width = max(target_width, 800)  
                target_height = max(target_height, 600)  
                
                print(f"Generating image with dimensions: {target_width}x{target_height}")
                image_path, (width, height) = ImageGenerator.generate_image(
                    prompt=prompt,
                    save_path=str(save_path),
                    max_width=target_width,
                    max_height=target_height
                )
                print(f"Successfully generated and saved image! Dimensions: {width}x{height}")
            except Exception as e:
                print(f"Failed to generate image: {str(e)}")
                return slide_data

 
        new_visual = {
            "id": new_id,
            "source": new_id,
            "type": "figure",
            "caption": title or "Illustrative diagram",
            "description": f"AI-generated visual to illustrate: {title}",
            "local_path": str(save_path),
            "width": width,
            "height": height,
            "is_generated": True,
            "generation_prompt": prompt,  
        }


        content.setdefault("visual_elements", visual_elements)
        content["visual_elements"].append(new_visual)
        content.setdefault("enhancements", {})["generated_visual_rationale"] = rationale
        slide_data["content"] = content
        
        print(f"Added new visual element with ID: {new_id}")
        return slide_data

    def _next_figure_id(self) -> str:
        """Generates the next unique figure ID."""
        self._figure_counter += 1
        return f"Generated Figure {self._figure_counter}"



