"""
State adapter: Overcooked env observation → SymbolicState

Bridges the overcooked-ai environment to our constraint detector.
"""

from typing import Dict, List, Optional, Tuple, Any
from overcooked_ai_py.mdp.overcooked_mdp import OvercookedGridworld, OvercookedState
from overcooked_ai_py.mdp.overcooked_env import OvercookedEnv

from .constraints import SymbolicState, Action


# Action mapping: our Action enum ↔ overcooked tuples
ACTION_TO_OVERCOOKED = {
    Action.UP: (0, -1),
    Action.DOWN: (0, 1),
    Action.RIGHT: (1, 0),
    Action.LEFT: (-1, 0),
    Action.STAY: (0, 0),
    Action.INTERACT: 'interact',
}

OVERCOOKED_TO_ACTION = {v: k for k, v in ACTION_TO_OVERCOOKED.items()}


class OvercookedAdapter:
    """
    Adapts overcooked-ai environment to our constraint detection system.

    Responsibilities:
    1. Convert env state → SymbolicState
    2. Convert Action enum → overcooked action tuples
    3. Generate text observations for LLM
    """

    def __init__(self, mdp: OvercookedGridworld, required_ingredients: int = 2, use_nav_hints: bool = True):
        self.mdp = mdp
        self.required_ingredients = required_ingredients  # For text observations
        self.use_nav_hints = use_nav_hints  # If False, give simple goals without directions
        self._cache_layout()

    def _cache_layout(self):
        """Cache static layout information."""
        self.pot_locations = self.mdp.get_pot_locations()
        self.onion_dispensers = self.mdp.get_onion_dispenser_locations()
        self.tomato_dispensers = self.mdp.get_tomato_dispenser_locations()
        self.dish_dispensers = self.mdp.get_dish_dispenser_locations()
        self.serve_locations = self.mdp.get_serving_locations()
        self.counter_locations = self.mdp.get_counter_locations()

        # Build dispenser dict
        self.dispensers: Dict[str, tuple] = {}
        for i, pos in enumerate(self.onion_dispensers):
            key = f"onion_dispenser" if i == 0 else f"onion_dispenser_{i}"
            self.dispensers[key] = pos
        for i, pos in enumerate(self.tomato_dispensers):
            key = f"tomato_dispenser" if i == 0 else f"tomato_dispenser_{i}"
            self.dispensers[key] = pos
        for i, pos in enumerate(self.dish_dispensers):
            key = f"dish_dispenser" if i == 0 else f"dish_dispenser_{i}"
            self.dispensers[key] = pos

        # Build pot dict
        self.pot_ids = {pos: f"pot_{i+1}" for i, pos in enumerate(self.pot_locations)}

    def to_symbolic_state(self, state: OvercookedState) -> SymbolicState:
        """Convert overcooked state to SymbolicState for constraint checking."""

        # Agent state
        agent_positions = {}
        agent_orientations = {}
        agent_holdings = {}

        for i, player in enumerate(state.players):
            agent_positions[i] = player.position
            agent_orientations[i] = player.orientation
            agent_holdings[i] = self._extract_holding(player.held_object)

        # Pot state
        pots = {}
        for pos, pot_id in self.pot_ids.items():
            pot_obj = state.objects.get(pos)
            pots[pot_id] = {
                "pos": pos,
                "ingredients": self._extract_pot_ingredients(pot_obj),
                "is_cooking": self._is_pot_cooking(pot_obj),
                "is_ready": self._is_pot_ready(pot_obj),
                "cook_tick": self._get_cook_tick(pot_obj),
            }

        # Find empty counters (counters without objects)
        empty_counters = [
            pos for pos in self.counter_locations
            if pos not in state.objects
        ]

        return SymbolicState(
            agent_positions=agent_positions,
            agent_orientations=agent_orientations,
            agent_holdings=agent_holdings,
            pots=pots,
            dispensers=self.dispensers.copy(),
            counters=empty_counters,
            serve_locations=list(self.serve_locations),
        )

    def _extract_holding(self, held_object) -> Optional[str]:
        """Extract what agent is holding as string."""
        if held_object is None:
            return None
        name = held_object.name
        if name == "soup":
            return "soup"
        elif name == "dish":
            return "plate"
        elif name in ["onion", "tomato"]:
            return name
        return name

    def _extract_pot_ingredients(self, pot_obj) -> List[str]:
        """Extract ingredients from pot object."""
        if pot_obj is None:
            return []
        if hasattr(pot_obj, 'ingredients'):
            return list(pot_obj.ingredients)
        return []

    def _is_pot_cooking(self, pot_obj) -> bool:
        """Check if pot is currently cooking."""
        if pot_obj is None:
            return False
        if hasattr(pot_obj, 'is_cooking'):
            return pot_obj.is_cooking
        return False

    def _is_pot_ready(self, pot_obj) -> bool:
        """Check if pot has ready soup."""
        if pot_obj is None:
            return False
        if hasattr(pot_obj, 'is_ready'):
            return pot_obj.is_ready
        return False

    def _get_cook_tick(self, pot_obj) -> int:
        """Get cooking progress tick."""
        if pot_obj is None:
            return -1
        if hasattr(pot_obj, '_cooking_tick'):
            return pot_obj._cooking_tick
        return -1

    def to_overcooked_action(self, action: Action) -> Any:
        """Convert our Action enum to overcooked action format."""
        return ACTION_TO_OVERCOOKED[action]

    def from_overcooked_action(self, oc_action) -> Action:
        """Convert overcooked action to our Action enum."""
        return OVERCOOKED_TO_ACTION.get(oc_action, Action.STAY)

    def get_facing_target(self, state: OvercookedState, agent_id: int) -> Optional[str]:
        """Get what the agent is facing (pot_id, dispenser name, 'serve', or 'counter')."""
        player = state.players[agent_id]
        pos = player.position
        orient = player.orientation
        facing_pos = (pos[0] + orient[0], pos[1] + orient[1])

        # Check pots
        if facing_pos in self.pot_ids:
            return self.pot_ids[facing_pos]

        # Check dispensers
        for name, dpos in self.dispensers.items():
            if dpos == facing_pos:
                return name

        # Check serve
        if facing_pos in self.serve_locations:
            return "serve"

        # Check counter
        if facing_pos in self.counter_locations:
            return "counter"

        return None

    def _get_object_name(self, pos: tuple) -> Optional[str]:
        """Get name of object at position (for navigation hints)."""
        if pos in self.pot_ids:
            return self.pot_ids[pos].upper()
        for name, dpos in self.dispensers.items():
            if dpos == pos:
                return name.upper()
        if pos in self.serve_locations:
            return "SERVE"
        if pos in self.counter_locations:
            return "counter"
        return None

    def to_text_observation(self, state: OvercookedState, agent_id: int,
                           step: int, horizon: int) -> str:
        """Generate text observation for LLM agent."""
        sym = self.to_symbolic_state(state)

        lines = [f"KITCHEN STATE (step {step}/{horizon})", ""]

        # Agent status
        holding = sym.agent_holdings[agent_id]
        holding_str = holding.upper() if holding else "NOTHING"
        lines.append(f"You (Agent {agent_id}): holding {holding_str}")

        # Partner status
        partner_id = 1 - agent_id
        partner_holding = sym.agent_holdings[partner_id]
        partner_holding_str = partner_holding.upper() if partner_holding else "NOTHING"
        lines.append(f"Partner (Agent {partner_id}): holding {partner_holding_str}")
        lines.append("")

        # Pot status
        for pot_id, pot in sym.pots.items():
            ing_count = len(pot["ingredients"])
            if pot["is_ready"]:
                status = "READY (soup done!)"
            elif pot["is_cooking"]:
                status = f"COOKING ({pot['cook_tick']}/20 ticks)"
            elif ing_count > 0:
                req = self.required_ingredients
                status = f"{ing_count}/{req} ingredients (need {req - ing_count} more)"
            else:
                status = "empty"
            lines.append(f"{pot_id.upper()}: {status}")
        lines.append("")

        # What agent is facing
        facing = self.get_facing_target(state, agent_id)
        if facing:
            lines.append(f"Facing: {facing}")
        lines.append("")

        # Navigation hints - what's in each direction
        player = state.players[agent_id]
        pos = player.position
        lines.append("Nearby objects:")
        for direction, delta in [("UP", (0, -1)), ("DOWN", (0, 1)), ("LEFT", (-1, 0)), ("RIGHT", (1, 0))]:
            check_pos = (pos[0] + delta[0], pos[1] + delta[1])
            obj = self._get_object_name(check_pos)
            if obj:
                lines.append(f"  {direction}: {obj}")
        lines.append("")

        # Available actions
        lines.append("Actions: UP, DOWN, LEFT, RIGHT, INTERACT, STAY")

        # Navigation hints with directions
        pot_ready = any(p.get("is_ready") for p in sym.pots.values())
        pot_cooking = any(p.get("is_cooking") for p in sym.pots.values())
        pot_full = any(len(p.get("ingredients", [])) >= 3 for p in sym.pots.values())

        if self.use_nav_hints:
            # Helper to get direction to target
            def get_direction_hint(target_pos):
                dx = target_pos[0] - pos[0]
                dy = target_pos[1] - pos[1]
                dirs = []
                if dy < 0: dirs.append("UP")
                if dy > 0: dirs.append("DOWN")
                if dx < 0: dirs.append("LEFT")
                if dx > 0: dirs.append("RIGHT")
                return " then ".join(dirs) if dirs else "INTERACT"

            # Get key positions (find first matching dispenser)
            onion_pos = self.onion_dispensers[0] if self.onion_dispensers else None
            plate_pos = self.dish_dispensers[0] if self.dish_dispensers else None
            pot_pos = list(self.pot_ids.keys())[0] if self.pot_ids else None
            serve_pos = list(self.serve_locations)[0] if self.serve_locations else None

            # Goal hints WITH navigation directions
            if holding is None:
                if pot_full and not pot_cooking and not pot_ready:
                    hint = f"Go {get_direction_hint(pot_pos)} to POT, then INTERACT to start cooking" if pot_pos else "INTERACT at pot"
                    lines.append(f"Goal: {hint}")
                elif pot_cooking:
                    hint = f"Go {get_direction_hint(plate_pos)} to get PLATE" if plate_pos else "Get plate"
                    lines.append(f"Goal: {hint}")
                elif pot_ready:
                    hint = f"Go {get_direction_hint(plate_pos)} to get PLATE, then scoop soup" if plate_pos else "Get plate"
                    lines.append(f"Goal: {hint}")
                else:
                    hint = f"Go {get_direction_hint(onion_pos)} to get ONION" if onion_pos else "Get onion"
                    lines.append(f"Goal: {hint}")
            elif holding == "onion":
                hint = f"Go {get_direction_hint(pot_pos)} to POT, then INTERACT to add onion" if pot_pos else "Add to pot"
                lines.append(f"Goal: {hint}")
            elif holding == "plate":
                if pot_ready:
                    hint = f"Go {get_direction_hint(pot_pos)} to POT, then INTERACT to scoop soup" if pot_pos else "Scoop soup"
                    lines.append(f"Goal: {hint}")
                elif pot_cooking:
                    lines.append("Goal: Wait near pot for cooking to finish")
                else:
                    lines.append("Goal: Pot needs 3 onions first - put plate down")
            elif holding == "soup":
                hint = f"Go {get_direction_hint(serve_pos)} to SERVE, then INTERACT to deliver" if serve_pos else "Deliver soup"
                lines.append(f"Goal: {hint}")
        else:
            # Simple goals WITHOUT navigation directions
            if holding is None:
                if pot_full and not pot_cooking and not pot_ready:
                    lines.append("Goal: Start cooking (interact with pot)")
                elif pot_cooking:
                    lines.append("Goal: Get a plate while soup cooks")
                elif pot_ready:
                    lines.append("Goal: Get a plate to scoop the soup")
                else:
                    lines.append("Goal: Get onion and add to pot")
            elif holding == "onion":
                lines.append("Goal: Add onion to pot")
            elif holding == "plate":
                if pot_ready:
                    lines.append("Goal: Scoop soup from pot")
                elif pot_cooking:
                    lines.append("Goal: Wait for soup to finish cooking")
                else:
                    lines.append("Goal: Pot needs 3 onions first")
            elif holding == "soup":
                lines.append("Goal: Deliver soup to serving area")
        lines.append("")
        lines.append("What action?")

        return "\n".join(lines)


def create_adapter(layout_name: str = "cramped_room",
                   required_ingredients: int = 2,
                   use_nav_hints: bool = True) -> Tuple[OvercookedEnv, OvercookedAdapter]:
    """Create environment and adapter for a layout.

    Args:
        layout_name: Overcooked layout name
        required_ingredients: Number of ingredients for soup
        use_nav_hints: If True, include directional hints like "Go UP then LEFT"
    """
    mdp = OvercookedGridworld.from_layout_name(layout_name)
    env = OvercookedEnv.from_mdp(mdp, horizon=400)
    adapter = OvercookedAdapter(mdp, required_ingredients=required_ingredients, use_nav_hints=use_nav_hints)
    return env, adapter
