"""
Poster QA Manager - Clean implementation
"""

import os
import sys
import base64
import json
from pathlib import Path
from typing import Optional, Dict, List, Tuple, Union

_original_src_path = Path(__file__).parent.parent.parent / "src"
if str(_original_src_path) not in sys.path:
    sys.path.insert(0, str(_original_src_path))

_current_dir = Path(__file__).parent
if str(_current_dir) not in sys.path:
    sys.path.insert(0, str(_current_dir))

from poster_config import PosterInteractionConfig, PosterGenerationResult, QuestionFormat, BaselineMode
from poster_adaptive_qa import PosterAdaptiveQA
from poster_rag_agent import PosterRAGAgent
from multiagent.llm import LLM


class PosterQAManager:
    """Manage Poster QA workflow"""
    
    def __init__(self, config, interaction_config: PosterInteractionConfig):
        self.config = config
        self.interaction_config = interaction_config
        self.output_folder = None
        self.llm = LLM(config, model_version=interaction_config.model_version)
        
        # Initialize RAG agent if needed (initialize outside if block to ensure scope)
        rag_agent = None
        if interaction_config.is_qa_mode and interaction_config.is_rag:
            from pathlib import Path
            rag_data_dir = Path(__file__).parent.parent.parent / "rag_data"
            if not rag_data_dir.exists():
                rag_data_dir = Path(__file__).parent.parent / "rag_data"
            rag_agent = PosterRAGAgent(rag_data_dir, interaction_config.question_format)
        
        # Initialize adaptive_qa for: Flexible/Adaptive formats or MPQC
        if (interaction_config.is_adaptive_format or
            interaction_config.is_flexible_format or
            interaction_config.is_mpqc):
            self.adaptive_qa = PosterAdaptiveQA(
                config=config,
                model_version=interaction_config.model_version,  # For backward compatibility
                mpc_enabled=interaction_config.is_mpqc,
                is_adaptive_format=interaction_config.is_adaptive_format,
                rag_agent=rag_agent,
                question_format=interaction_config.question_format,
                global_model_version=interaction_config.model_version,
                question_agent_model_version=interaction_config.effective_question_agent_model_version,
                answer_agent_model_version=interaction_config.effective_answer_agent_model_version
            )
        else:
            self.rag_agent = rag_agent
    
    def generate_poster(
        self,
        item_description: str,
        answer_image_path: str,
        logo_path: Optional[str] = None,
        output_dir: Optional[str] = None
    ) -> PosterGenerationResult:
        """Generate poster based on interaction config"""
        if output_dir:
            self.output_folder = output_dir
            if hasattr(self, 'adaptive_qa'):
                self.adaptive_qa.output_folder = output_dir
        
        if self.interaction_config.baseline_mode == BaselineMode.NO_USER:
            return self._generate_no_user(item_description, logo_path)
        # Priority: MPQC > Flexible/Adaptive > Fixed
        elif self.interaction_config.is_mpqc:
            # MPQC: use adaptive QA with Adaptive format
            return self._generate_adaptive(item_description, answer_image_path, logo_path)
        elif self.interaction_config.is_adaptive_format or self.interaction_config.is_flexible_format:
            return self._generate_adaptive(item_description, answer_image_path, logo_path)
        else:
            # Fixed formats (Fixed_Binary, Fixed_MultiChoice, Fixed_OpenText, Free_Ask)
            return self._generate_fixed_qa(item_description, answer_image_path, logo_path)
    
    def _generate_no_user(self, item_description: str, logo_path: Optional[str] = None) -> PosterGenerationResult:
        """No user mode: Skip QA, direct image generation"""
        try:
            plan = self._create_initial_plan_from_prompt(item_description)
            print(f"\n{'='*80}\n🎨 Generating Poster Image (No User Mode)\n{'='*80}")
            image_path = self._generate_poster_image(plan, item_description, logo_path, conversation_history=None)
            
            if not image_path:
                return PosterGenerationResult(success=False, error="Failed to generate image")
            
            return PosterGenerationResult(
                success=True,
                generated_image_path=Path(image_path),
                ad_text=plan.get("ad_copy", {}),
                design_plan=plan,
                token_stats={
                    "questioner_agent": {"input_tokens": 0, "output_tokens": 0, "reasoning_tokens": 0, "total_tokens": 0},
                    "answerer_agent": {"input_tokens": 0, "output_tokens": 0, "reasoning_tokens": 0, "total_tokens": 0}
                },
                questions_asked=0
            )
        except Exception as e:
            import traceback
            traceback.print_exc()
            return PosterGenerationResult(success=False, error=str(e))
    
    def _check_satisfaction(self, plan: Dict, conversation_history: Union[List[Tuple[str, str]], List[Dict]]) -> bool:
        """Check if question agent is satisfied"""
        prompts_dir = Path(__file__).parent.parent / "prompts"
        satisfaction_prompt_path = prompts_dir / "poster_qa_satisfaction_check.txt"
        
        if not satisfaction_prompt_path.exists():
            return False
        
        import json
        plan_str = json.dumps(plan, indent=2, ensure_ascii=False) if plan else ""
        # Handle both dict format and tuple format
        qa_history_text = ""
        if conversation_history:
            qa_pairs = []
            for qa_item in conversation_history[-10:]:
                if isinstance(qa_item, dict):
                    q = qa_item.get("question", "")
                    a = qa_item.get("answer", "")
                elif isinstance(qa_item, (tuple, list)) and len(qa_item) >= 2:
                    q = qa_item[0]
                    a = qa_item[1]
                else:
                    continue
                qa_pairs.append(f"Q: {q}\nA: {a}")
            qa_history_text = "\n".join(qa_pairs)
        
        with open(satisfaction_prompt_path, 'r', encoding='utf-8') as f:
            prompt_template = f.read()
        
        prompt = prompt_template.replace("{current_plan}", plan_str).replace("{qa_history}", qa_history_text)
        
        system_instruction = ""
        if "[SYSTEM]" in prompt and "[SATISFACTION_CHECK]" in prompt:
            parts = prompt.split("[SATISFACTION_CHECK]")
            system_instruction = parts[0].replace("[SYSTEM]", "").strip()
            prompt = parts[1] if len(parts) > 1 else prompt
        
        try:
            from langchain_core.messages import HumanMessage, SystemMessage
            messages = [SystemMessage(content=system_instruction)] if system_instruction else []
            messages.append(HumanMessage(content=prompt))
            # Use lower temperature for more conservative/strict evaluation
            # Handle both LLM wrapper (has .llm attribute) and direct LangChain wrapper (no .llm attribute)
            llm_instance = self.llm.llm if hasattr(self.llm, 'llm') else self.llm
            try:
                config = {"temperature": 0.0}
                response = llm_instance.invoke(messages, config=config)
            except:
                response = llm_instance.invoke(messages)
            response_text = response.content.strip() if hasattr(response, 'content') else str(response)
            response_upper = response_text.upper().strip()
            first_line = response_upper.splitlines()[0].strip() if response_upper else ""
            # Only return True if explicitly SATISFIED on the first line (avoid "NOT SATISFIED")
            return first_line.startswith("SATISFIED")
        except Exception as e:
            print(f"⚠️ Error checking satisfaction: {e}")
            return False
    
    def _answer_question_with_prompt(self, question: str, answer_image_path: str, item_description: str, 
                                       plan: Optional[Dict], conversation_history: List, generator) -> Tuple[str, Dict]:
        """Answer question using poster_user_answer_only.txt prompt (unified for all non-MPQC modes)"""
        from langchain_core.messages import HumanMessage, SystemMessage
        
        # Load prompt from file
        prompts_dir = Path(__file__).parent.parent / "prompts"
        prompt_path = prompts_dir / "poster_user_answer_only.txt"
        with open(prompt_path, 'r', encoding='utf-8') as f:
            prompt_template = f.read()
        
        # Format the prompt
        plan_text = json.dumps(plan, indent=2, ensure_ascii=False) if plan else "No plan yet."
        system_prompt = prompt_template.format(
            current_plan=plan_text,
            questions_text=question
        )
        
        # Build user content with images
        user_content = []
        
        # Add target image
        if os.path.exists(answer_image_path):
            image_base64 = generator.image_to_base64(answer_image_path)
            if image_base64:
                if not image_base64.startswith("data:image"):
                    image_base64 = f"data:image/png;base64,{image_base64}"
                user_content.append({
                    "type": "image_url",
                    "image_url": {"url": image_base64, "detail": "high"}
                })
        
        # Add logo if available
        if hasattr(generator, 'logo_info') and generator.logo_info and 'path' in generator.logo_info:
            logo_path = generator.logo_info['path']
            if os.path.exists(logo_path):
                logo_base64 = generator.image_to_base64(logo_path)
                if logo_base64:
                    if not logo_base64.startswith("data:image"):
                        logo_base64 = f"data:image/png;base64,{logo_base64}"
                    user_content.append({
                        "type": "image_url",
                        "image_url": {"url": logo_base64, "detail": "high"}
                    })
        
        # Add question text
        user_content.append({"type": "text", "text": f"Question:\n{question}"})
        
        messages = [
            SystemMessage(content=system_prompt),
            HumanMessage(content=user_content)
        ]
        
        # Use answerer_llm
        response = generator.answerer_llm.invoke(messages)
        answer = response.content.strip() if hasattr(response, 'content') else str(response).strip()
        
        # Token stats (simplified - actual tracking done by generator)
        token_stats = {
            "input_tokens": 0,
            "output_tokens": 0,
            "reasoning_tokens": 0,
            "total_tokens": 0
        }
        
        return answer, token_stats

    def _generate_fixed_qa(self, item_description: str, answer_image_path: str, logo_path: Optional[str] = None) -> PosterGenerationResult:
        """Fixed format QA mode"""
        try:
            print(f"\n{'='*80}\n📋 Creating Initial Plan\n{'='*80}")
            plan = self._create_initial_plan_from_prompt(item_description)
            
            from html_ad_workflow_qa_with_answer import QnAWithAnswerHTMLAdGenerator
            
            # Map question format to string
            # Note: Free_Ask should use "free_ask" (not "open_text") to avoid format constraints
            if self.interaction_config.question_format == QuestionFormat.FIXED_BINARY:
                q_format = "binary"
            elif self.interaction_config.question_format == QuestionFormat.FIXED_MULTICHOICE:
                q_format = "multi_choice"
            elif self.interaction_config.question_format == QuestionFormat.FIXED_OPENTEXT:
                q_format = "open_text"
            elif self.interaction_config.question_format == QuestionFormat.FREE_ASK:
                q_format = "free_ask"  # Use special value to indicate no format constraints
            else:
                q_format = "open_text"
            
            generator = QnAWithAnswerHTMLAdGenerator(
                self.config,
                question_format=q_format,
                model_version=self.interaction_config.model_version,  # For backward compatibility
                global_model_version=self.interaction_config.model_version,
                question_agent_model_version=self.interaction_config.effective_question_agent_model_version,
                answer_agent_model_version=self.interaction_config.effective_answer_agent_model_version
            )
            generator.output_folder = self.output_folder
            if logo_path:
                generator.set_logo_info(logo_path)

            all_conversation_history = []
            plan_history = []
            all_token_stats = {
                "questioner_agent": {"input_tokens": 0, "output_tokens": 0, "reasoning_tokens": 0, "total_tokens": 0},
                "answerer_agent": {"input_tokens": 0, "output_tokens": 0, "reasoning_tokens": 0, "total_tokens": 0}
            }
            
            # Map format for storing in conversation_history
            format_map = {
                QuestionFormat.FIXED_BINARY: "binary",
                QuestionFormat.FIXED_MULTICHOICE: "multi_choice",
                QuestionFormat.FIXED_OPENTEXT: "open_text",
                QuestionFormat.FREE_ASK: "free_ask"
            }
            stored_format = format_map.get(self.interaction_config.question_format, "open_text")
            
            for cycle in range(self.interaction_config.max_qa_cycles):
                print(f"\n{'='*80}\n🔄 QA Cycle {cycle + 1}/{self.interaction_config.max_qa_cycles}\n{'='*80}")
                
                for i in range(1, self.interaction_config.max_questions_per_batch + 1):
                    # Record token stats before question generation
                    initial_questioner_tokens = {
                        "input_tokens": generator.questioner_token_stats.get("input_tokens", 0),
                        "output_tokens": generator.questioner_token_stats.get("output_tokens", 0),
                        "reasoning_tokens": generator.questioner_token_stats.get("reasoning_tokens", 0),
                        "total_tokens": generator.questioner_token_stats.get("total_tokens", 0)
                    }
                    initial_answerer_tokens = {
                        "input_tokens": generator.answerer_token_stats.get("input_tokens", 0),
                        "output_tokens": generator.answerer_token_stats.get("output_tokens", 0),
                        "reasoning_tokens": generator.answerer_token_stats.get("reasoning_tokens", 0),
                        "total_tokens": generator.answerer_token_stats.get("total_tokens", 0)
                    }
                    
                    # Convert to tuple format for backward compatibility with generator methods
                    tuple_history = [(qa["question"], qa["answer"]) if isinstance(qa, dict) else qa for qa in all_conversation_history]
                    
                    question = generator.questioner_agent_ask_question(
                        item_description=item_description,
                        conversation_history=tuple_history,
                        question_number=i,
                        total_questions=self.interaction_config.max_questions_per_batch,
                        answer_image_path=answer_image_path,
                        plan=plan  # Pass current plan to question generation
                    )
                    
                    # Calculate per-question questioner tokens
                    per_question_questioner_tokens = {
                        "input_tokens": generator.questioner_token_stats.get("input_tokens", 0) - initial_questioner_tokens["input_tokens"],
                        "output_tokens": generator.questioner_token_stats.get("output_tokens", 0) - initial_questioner_tokens["output_tokens"],
                        "reasoning_tokens": generator.questioner_token_stats.get("reasoning_tokens", 0) - initial_questioner_tokens["reasoning_tokens"],
                        "total_tokens": generator.questioner_token_stats.get("total_tokens", 0) - initial_questioner_tokens["total_tokens"]
                    }
                    
                    # Use unified answering method with poster_user_answer_only.txt prompt
                    answer, per_question_answerer_tokens = self._answer_question_with_prompt(
                        question=question,
                        answer_image_path=answer_image_path,
                        item_description=item_description,
                        plan=plan,
                        conversation_history=tuple_history,
                        generator=generator
                    )
                    
                    # Store as dict with token stats and format
                    all_conversation_history.append({
                        "question": question,
                        "answer": answer,
                        "questioner_tokens": per_question_questioner_tokens,
                        "answerer_tokens": per_question_answerer_tokens,
                        "format": stored_format
                    })
                
                    # Check satisfaction (use tuple format for backward compatibility)
                    if self._check_satisfaction(plan, tuple_history):
                        print(f"\n✅ Question agent satisfied after {i} question(s)")
                        break
                
                print(f"\n{'='*80}\n📋 Updating Plan after Cycle {cycle + 1}\n{'='*80}")
                # Update tuple_history for plan generation
                tuple_history = [(qa["question"], qa["answer"]) if isinstance(qa, dict) else qa for qa in all_conversation_history]
                # Update plan based on Q&A instead of recreating from scratch
                # This preserves initial plan structure while incorporating new information
                plan = self._update_plan_from_qa(
                    item_description,
                    tuple_history,
                    plan,  # Pass current plan to update it
                    accepted_questions=None,  # Fixed format doesn't use accept/reject
                    rejected_questions=None
                )
                
                # Persist plan history per cycle (if output folder is set)
                plan_history.append({
                    "cycle_index": cycle + 1,
                    "plan": plan,
                    "accepted_questions": None,  # Fixed format doesn't use accept/reject
                    "rejected_questions": None
                })
                if self.output_folder:
                    try:
                        output_path = Path(self.output_folder)
                        output_path.mkdir(parents=True, exist_ok=True)
                        plan_history_path = output_path / "plan_history.json"
                        with open(plan_history_path, "w", encoding="utf-8") as f:
                            json.dump(plan_history, f, ensure_ascii=False, indent=2)
                    except Exception:
                        pass
            
            all_token_stats["questioner_agent"] = generator.questioner_token_stats
            all_token_stats["answerer_agent"] = generator.answerer_token_stats
            
            print(f"\n{'='*80}\n🎨 Generating Poster Image\n{'='*80}")
            image_path = self._generate_poster_image(plan, item_description, logo_path, conversation_history=all_conversation_history)
            
            if not image_path:
                return PosterGenerationResult(success=False, error="Failed to generate image")
            
            return PosterGenerationResult(
                success=True,
                generated_image_path=Path(image_path),
                ad_text=plan.get("ad_copy", {}),
                conversation_history=all_conversation_history,
                design_plan=plan,
                token_stats=all_token_stats,
                questions_asked=len(all_conversation_history)
            )
        except Exception as e:
            import traceback
            traceback.print_exc()
            return PosterGenerationResult(success=False, error=str(e))
    
    def _generate_adaptive(self, item_description: str, answer_image_path: str, logo_path: Optional[str] = None) -> PosterGenerationResult:
        """Adaptive/Flexible mode"""
        try:
            if logo_path:
                self.adaptive_qa.set_logo_info(logo_path)
            
            print(f"\n{'='*80}\n📋 Creating Initial Plan\n{'='*80}")
            plan = self._create_initial_plan_from_prompt(item_description)
            
            all_conversation_history = []
            plan_history = []
            all_token_stats = {
                "questioner_agent": {"input_tokens": 0, "output_tokens": 0, "reasoning_tokens": 0, "total_tokens": 0},
                "answerer_agent": {"input_tokens": 0, "output_tokens": 0, "reasoning_tokens": 0, "total_tokens": 0}
            }
            
            # Budget tuning for MPQC (lower cost while keeping quality)
            effective_num_questions = self.interaction_config.max_questions_per_batch
            if self.interaction_config.is_mpqc:
                # MPQC: lower question budget to reduce cost while keeping quality
                effective_num_questions = max(1, min(3, self.interaction_config.max_questions_per_batch))

            for cycle in range(self.interaction_config.max_qa_cycles):
                print(f"\n{'='*80}\n🔄 QA Cycle {cycle + 1}/{self.interaction_config.max_qa_cycles}\n{'='*80}")

                cycle_qa_result = self.adaptive_qa.conduct_adaptive_qa_with_plan(
                    item_description,
                    answer_image_path,
                    plan,
                    num_questions=effective_num_questions,
                    mpc_enabled=self.interaction_config.is_mpqc
                )
                
                if not cycle_qa_result:
                    return PosterGenerationResult(success=False, error="QA cycle failed")
                
                cycle_history = cycle_qa_result.get("conversation_history", [])
                if cycle_history:
                    all_conversation_history.extend(cycle_history)
                
                cycle_token_stats = cycle_qa_result.get("token_stats", {})
                if cycle_token_stats:
                    for agent in ["questioner_agent", "answerer_agent"]:
                        if agent in cycle_token_stats:
                            for key in ["input_tokens", "output_tokens", "reasoning_tokens", "total_tokens"]:
                                all_token_stats[agent][key] += cycle_token_stats[agent].get(key, 0)
                
                print(f"\n{'='*80}\n📋 Updating Plan after Cycle {cycle + 1}\n{'='*80}")
                
                rejected_questions = cycle_qa_result.get("rejected_questions", [])
                accepted_questions = cycle_qa_result.get("accepted_questions", [])
                
                plan = self._update_plan_from_qa(
                    item_description,
                    all_conversation_history,
                    plan,
                    accepted_questions=accepted_questions if self.interaction_config.is_mpqc else None,
                    rejected_questions=rejected_questions if self.interaction_config.is_mpqc else None
                )
                
                # Persist plan history per cycle (if output folder is set)
                plan_history.append({
                    "cycle_index": cycle + 1,
                    "plan": plan,
                    "accepted_questions": accepted_questions,
                    "rejected_questions": rejected_questions
                })
                if self.output_folder:
                    try:
                        output_path = Path(self.output_folder)
                        output_path.mkdir(parents=True, exist_ok=True)
                        plan_history_path = output_path / "plan_history.json"
                        with open(plan_history_path, "w", encoding="utf-8") as f:
                            json.dump(plan_history, f, ensure_ascii=False, indent=2)
                    except Exception:
                        pass
            
            print(f"\n{'='*80}\n🎨 Generating Poster Image\n{'='*80}")
            image_path = self._generate_poster_image(plan, item_description, logo_path, conversation_history=all_conversation_history)
            
            if not image_path:
                return PosterGenerationResult(success=False, error="Failed to generate image")
            
            return PosterGenerationResult(
                success=True,
                generated_image_path=Path(image_path),
                ad_text=plan.get("ad_copy", {}),
                conversation_history=all_conversation_history,
                design_plan=plan,
                token_stats=all_token_stats,
                questions_asked=len(all_conversation_history)
            )
        except Exception as e:
            import traceback
            traceback.print_exc()
            return PosterGenerationResult(success=False, error=str(e))
    
    def _generate_poster_image(self, plan: dict, item_description: str, logo_path: Optional[str] = None, conversation_history: Optional[Union[List[Tuple[str, str]], List[Dict]]] = None) -> Optional[str]:
        """Generate poster image using GPT-4o"""
        print(f"🎨 Generating poster image...")
        
        try:
            from openai import AzureOpenAI
            
            qa_details = self._extract_qa_details(conversation_history) if conversation_history else {}
            background_plan = plan.get("background_image_plan", {})
            ad_copy = plan.get("ad_copy", {})
            color_scheme = plan.get("color_scheme", {})
            style_mood = plan.get("style_and_mood", {})
            layout = plan.get("layout", {})
            
            image_prompt = f"""Create a poster image (1024x1024 pixels) that matches the design requirements:

Background: {background_plan.get('description', item_description)}
Style: {background_plan.get('style', style_mood.get('overall_style', 'modern professional'))}
Colors: {color_scheme.get('primary_colors', '')}
Mood: {style_mood.get('emotional_tone', 'professional')}

Text Content:
- Headline: {ad_copy.get('headline', '')}
- Description: {ad_copy.get('description', '')}
- CTA Button: {ad_copy.get('cta_text', 'Learn More')}

Layout:
- Text placement: {layout.get('text_placement', 'center')}
- Logo placement: {layout.get('logo_placement', 'top center')}
- CTA placement: {layout.get('cta_placement', 'bottom center')}
"""
            
            if qa_details:
                qa_specs = []
                if qa_details.get('colors'):
                    qa_specs.append(f"Color specifications: {', '.join(qa_details['colors'])}")
                if qa_details.get('text_preferences'):
                    for key, value in qa_details['text_preferences'].items():
                        qa_specs.append(f"{key.capitalize()}: {value}")
                if qa_specs:
                    image_prompt += "\n\nAdditional Specifications:\n" + "\n".join(qa_specs) + "\n"
            
            if logo_path and os.path.exists(logo_path):
                logo_description = self._analyze_logo_with_vision(logo_path)
                if logo_description:
                    image_prompt += f"\n\nLogo Details:\n{logo_description}\n"
                image_prompt += f"\nInclude the logo at {layout.get('logo_placement', 'top center')}."
            
            image_prompt += """

Requirements:
- All text clearly visible and readable
- High contrast for text readability
- Professional design matching specified style
- Size: 1024x1024 pixels
- Ensure logo (if present) is clear and prominent
- Ensure CTA button is prominent and clear
"""
            
            base_url = self.config["KEYS"]["GPT_IMAGE_1_BASE_URL"]
            image_client = AzureOpenAI(
                api_key=self.config["KEYS"]["AZURE_OPENAI_API_IMAGE_KEY"],
                api_version="2025-04-01-preview",
                base_url=base_url
            )
            
            result = image_client.images.generate(
                model="gpt-image-1",
                prompt=image_prompt,
                size="1024x1024",
                n=1
            )
            
            image_base64 = result.data[0].b64_json
            image_bytes = base64.b64decode(image_base64)
            
            os.makedirs(self.output_folder, exist_ok=True)
            image_path = os.path.join(self.output_folder, "poster.png")
            with open(image_path, "wb") as f:
                f.write(image_bytes)
            
            print(f"✅ Poster image saved: {image_path}")
            return image_path
            
        except Exception as e:
            print(f"❌ Error generating poster image: {e}")
            import traceback
            traceback.print_exc()
            return None
    
    def _extract_qa_details(self, conversation_history: Union[List[Tuple[str, str]], List[Dict]]) -> Dict:
        """Extract details from Q&A conversation
        
        Args:
            conversation_history: List of Q&A pairs, can be either:
                - List[Tuple[str, str]]: [(question, answer), ...]
                - List[Dict]: [{"question": q, "answer": a, ...}, ...]
        """
        details = {'colors': [], 'text_preferences': {}, 'layout_preferences': {}, 'style_preferences': {}, 'specific_requirements': []}
        
        for qa_item in conversation_history:
            # Handle both dict format and tuple format
            if isinstance(qa_item, dict):
                q = qa_item.get("question", "")
                a = qa_item.get("answer", "")
            elif isinstance(qa_item, (tuple, list)) and len(qa_item) >= 2:
                q = qa_item[0]
                a = qa_item[1]
            else:
                continue  # Skip invalid format
            q_lower = q.lower()
            a_lower = a.lower()
            
            if 'color' in q_lower or 'palette' in q_lower:
                if a.strip() and a.lower() != 'not visible':
                    details['colors'].append(a)
            if 'headline' in q_lower or 'title' in q_lower:
                if a.strip() and a.lower() != 'not visible':
                    details['text_preferences']['headline'] = a
            elif 'description' in q_lower or 'body' in q_lower:
                if a.strip() and a.lower() != 'not visible':
                    details['text_preferences']['description'] = a
            elif 'cta' in q_lower or 'button' in q_lower:
                if a.strip() and a.lower() != 'not visible':
                    details['text_preferences']['cta'] = a
            if 'layout' in q_lower or 'placement' in q_lower:
                if a.strip() and a.lower() != 'not visible':
                    details['layout_preferences'][q] = a
            if 'style' in q_lower or 'mood' in q_lower:
                if a.strip() and a.lower() != 'not visible':
                    details['style_preferences'][q] = a
        
        return details
    
    def _analyze_logo_with_vision(self, logo_path: str) -> str:
        """Analyze logo using vision model"""
        if not os.path.exists(logo_path):
            return ""
        
        try:
            from langchain_core.messages import HumanMessage
            from PIL import Image
            import io
            
            with Image.open(logo_path) as img:
                img_format = img.format or 'PNG'
                buffer = io.BytesIO()
                img.save(buffer, format=img_format)
                img_bytes = buffer.getvalue()
                logo_base64 = base64.b64encode(img_bytes).decode('utf-8')
                logo_base64 = f"data:image/{img_format.lower()};base64,{logo_base64}"
            
            prompt = """Analyze this logo and describe: colors (hex codes if identifiable), shapes, text content, style, overall appearance."""
            
            messages = [HumanMessage(content=[
                {"type": "text", "text": prompt},
                {"type": "image_url", "image_url": {"url": logo_base64, "detail": "high"}}
            ])]
            
            # Handle both LLM wrapper (has .llm attribute) and direct LangChain wrapper (no .llm attribute)
            llm_instance = self.llm.llm if hasattr(self.llm, 'llm') else self.llm
            response = llm_instance.invoke(messages)
            return response.content.strip()
        except Exception as e:
            print(f"⚠️ Error analyzing logo: {e}")
            return ""
    
    def _create_initial_plan_from_prompt(self, item_description: str) -> Dict:
        """Create initial plan from prompt file (poster_initial_plan_generator.txt)

        CRITICAL: All configurations for the same sample MUST use the same initial plan.
        This ensures fair comparison across different agent formats and question formats.
        """
        from langchain_core.messages import HumanMessage
        import json
        import re
        import hashlib

        # Check for cached plan (ensures all configs use same initial plan)
        item_description_hash = hashlib.md5(item_description.encode('utf-8')).hexdigest()[:8]
        store_dir = Path(__file__).parent / "initial_plan_store"
        store_path = store_dir / f"{item_description_hash}.json"

        if store_path.exists():
            try:
                with open(store_path, "r", encoding="utf-8") as f:
                    stored = json.load(f)
                if stored.get("item_description") == item_description and isinstance(stored.get("plan"), dict):
                    print(f"Reusing shared initial design plan (hash: {item_description_hash})")
                    return stored["plan"]
            except Exception:
                pass

        # Load prompt from file
        prompts_dir = Path(__file__).parent.parent / "prompts"
        prompt_path = prompts_dir / "poster_initial_plan_generator.txt"
        if prompt_path.exists():
            with open(prompt_path, 'r', encoding='utf-8') as f:
                prompt_template = f.read()
            plan_prompt = prompt_template.replace("<P0>", item_description)
        else:
            # Fallback if file missing
            plan_prompt = f"Create a banner design plan in JSON for: {item_description}"

        try:
            messages = [HumanMessage(content=plan_prompt)]
            llm_instance = self.llm.llm if hasattr(self.llm, 'llm') else self.llm
            response = llm_instance.invoke(messages)
            content = response.content
            json_match = re.search(r'\{.*\}', content, re.DOTALL)
            if json_match:
                plan = json.loads(json_match.group(0))
                # Handle nested "design_plan" structure from prompt template
                if "design_plan" in plan:
                    plan = plan["design_plan"]
                # Cache the plan
                try:
                    store_dir.mkdir(parents=True, exist_ok=True)
                    with open(store_path, "w", encoding="utf-8") as f:
                        json.dump({"item_description": item_description, "plan": plan}, f, ensure_ascii=False, indent=2)
                except Exception:
                    pass
                return plan
            else:
                return self._create_default_plan(item_description)
        except Exception as e:
            print(f"Error creating plan: {e}")
            return self._create_default_plan(item_description)
    
    def _create_default_plan(self, item_description: str) -> Dict:
        """Create default plan"""
        return {
            "background_image_plan": {"description": item_description, "style": "modern professional", "colors": "", "mood": "professional", "composition": "centered"},
            "ad_copy": {"headline": "", "description": "", "cta_text": "Learn More"},
            "style_and_mood": {"overall_style": "modern", "emotional_tone": "professional"},
            "color_scheme": {"primary_colors": "", "tone": "professional", "contrast_requirements": "High contrast for readability"},
            "layout": {"text_placement": "center", "logo_placement": "top center", "cta_placement": "bottom center", "logo_size": "medium", "headline_size": "large", "description_size": "medium", "cta_size": "medium", "composition_approach": "Centered with clear hierarchy"}
        }
    
    def _update_plan_from_qa(self, item_description: str, conversation_history: Union[List[Tuple[str, str]], List[Dict]], current_plan: Dict, accepted_questions: Optional[List] = None, rejected_questions: Optional[List] = None) -> Dict:
        """Update plan based on Q&A using poster_plan_updater.txt"""
        from langchain_core.messages import HumanMessage, SystemMessage
        import json
        import re

        # Load plan updater prompt from file
        prompts_dir = Path(__file__).parent.parent / "prompts"
        prompt_path = prompts_dir / "poster_plan_updater.txt"

        # Format Q&A pairs
        qa_text = ""
        for qa_item in conversation_history:
            if isinstance(qa_item, dict):
                q = qa_item.get("question", "")
                a = qa_item.get("answer", "")
            elif isinstance(qa_item, (tuple, list)) and len(qa_item) >= 2:
                q, a = qa_item[0], qa_item[1]
            else:
                continue
            qa_text += f"Q: {q}\nA: {a}\n\n"

        if prompt_path.exists():
            with open(prompt_path, 'r', encoding='utf-8') as f:
                prompt_content = f.read()
            
            # Split into system prompt and user message template
            parts = prompt_content.split("Plan updater user message")
            system_prompt = parts[0].replace("Plan updater system prompt (U; shared)", "").strip()
            
            # Build user message using template placeholders
            plan_str = json.dumps(current_plan, indent=2, ensure_ascii=False)
            if len(parts) > 1:
                user_template = parts[1].replace("(U; shared)", "").strip()
                user_msg = user_template.replace("<PLAN>", plan_str).replace("<ACCEPTED_QA_PAIRS>", qa_text.strip())
            else:
                user_msg = f"Current plan: {plan_str}\nAccepted QA pairs:\n{qa_text}\nInstruction: Output the updated plan in same structure."
            
            try:
                messages = [
                    SystemMessage(content=system_prompt),
                    HumanMessage(content=user_msg)
                ]
                llm_instance = self.llm.llm if hasattr(self.llm, 'llm') else self.llm
                response = llm_instance.invoke(messages)
                json_match = re.search(r'\{.*\}', response.content, re.DOTALL)
                if json_match:
                    updated_plan = json.loads(json_match.group(0))
                    if rejected_questions:
                        updated_plan["avoid_designs"] = rejected_questions
                    return updated_plan
            except Exception as e:
                print(f"Error updating plan with LLM: {e}")

        # Fallback to rule-based update
        return self._update_plan_from_qa_simple(item_description, conversation_history, current_plan)
    
    def _update_plan_from_qa_simple(self, item_description: str, conversation_history: Union[List[Tuple[str, str]], List[Dict]], current_plan: Dict) -> Dict:
        """Update plan from Q&A history (simple rule-based update)
        
        Args:
            item_description: Item description
            conversation_history: List of Q&A pairs, can be either:
                - List[Tuple[str, str]]: [(question, answer), ...]
                - List[Dict]: [{"question": q, "answer": a, ...}, ...]
            current_plan: Current plan to update (preserves initial plan structure)
        """
        # Start with a copy of current plan to preserve initial plan structure
        import copy
        plan = copy.deepcopy(current_plan)
        
        # Update plan based on Q&A
        for qa_item in conversation_history:
            # Handle both dict format and tuple format
            if isinstance(qa_item, dict):
                q = qa_item.get("question", "")
                a = qa_item.get("answer", "")
            elif isinstance(qa_item, (tuple, list)) and len(qa_item) >= 2:
                q = qa_item[0]
                a = qa_item[1]
            else:
                continue  # Skip invalid format
            q_lower = q.lower()
            a_lower = a.lower()
            if 'headline' in q_lower or 'title' in q_lower:
                plan["ad_copy"]["headline"] = a  # No truncation - pass full answer to LLM
            elif 'description' in q_lower or 'body' in q_lower:
                plan["ad_copy"]["description"] = a  # No truncation - pass full answer to LLM
            elif 'cta' in q_lower or 'button' in q_lower:
                plan["ad_copy"]["cta_text"] = a  # No truncation - pass full answer to LLM
            elif 'color' in q_lower or 'palette' in q_lower:
                plan["color_scheme"]["primary_colors"] = a
                plan["background_image_plan"]["colors"] = a
            elif 'background' in q_lower:
                # Handle background color specifically (including white)
                if 'white' in a_lower:
                    plan["background_image_plan"]["colors"] = "white"
                    plan["color_scheme"]["primary_colors"] = "white"
                else:
                    plan["background_image_plan"]["colors"] = a
                    plan["color_scheme"]["primary_colors"] = a
            elif 'style' in q_lower or 'mood' in q_lower:
                plan["style_and_mood"]["overall_style"] = a
                plan["background_image_plan"]["mood"] = a
            elif 'layout' in q_lower or 'composition' in q_lower:
                plan["layout"]["composition_approach"] = a
                plan["background_image_plan"]["composition"] = a
        return plan
