import math
from itertools import product
from typing import List, Dict, Any, Tuple


def check_for_overlaps(objects_list: List[Dict[str, Any]]) -> bool:
    """
    Checks for overlaps between bounding boxes of objects in a list,
    assuming bbox is [y_top, x_left, y_bottom, x_right] where X increases right and Y increases DOWN.

    Args:
        objects_list: A list of dictionaries, each containing 'label' and 'bbox'.

    Returns:
        True if any non-exempt overlap is found, False otherwise.
        Exempt overlaps: A pair consisting of one 'chair' and one 'table'.
    """
    for i, obj1 in enumerate(objects_list):
        for j in range(i + 1, len(objects_list)):
            obj2 = objects_list[j]

            # Skip chair/table pairs (exempt from overlap check)
            if {obj1['label'], obj2['label']} == {'chair', 'table'}:
                continue

            # Extract bounding box coordinates: [y_top, x_left, y_bottom, x_right]
            y1_top, x1_left, y1_bottom, x1_right = obj1['bbox']
            y2_top, x2_left, y2_bottom, x2_right = obj2['bbox']

            # Check for overlap on both axes
            overlap_x = (x1_right > x2_left) and (x2_right > x1_left)
            overlap_y = (y1_bottom > y2_top) and (y2_bottom > y1_top)

            if overlap_x and overlap_y:
                # GOOD PRINT TODO
                print(f"Overlap detected between {obj1['label']} ({obj1['bbox']}) and {obj2['label']} ({obj2['bbox']})")
                return True

    return False


def check_overlap_with_target(target_obj: Dict[str, Any], objects_list: List[Dict[str, Any]]) -> bool:
    """
    Checks for overlaps between a target object's bounding box and a list of other objects,
    assuming bbox is [y_top, x_left, y_bottom, x_right] where X increases right and Y increases DOWN.

    Exempted pairs:
        - chair + table
        - tv_stand + television
        - side_table + table_lamp

    Args:
        target_obj: A dictionary containing 'label' and 'bbox'.
        objects_list: A list of dictionaries, each containing 'label' and 'bbox'.

    Returns:
        True if any non-exempt overlap is found, False otherwise.
    """
    # Define exempt label pairs
    exempt_pairs = [
        {'chair', 'table'},
        {'tv_stand', 'television'},
        {'side_table', 'table_lamp'}
    ]

    y1_top, x1_left, y1_bottom, x1_right = target_obj['bbox']

    for other_obj in objects_list:
        if other_obj is target_obj:
            continue

        # Check if the pair is exempt
        if {target_obj['label'], other_obj['label']} in exempt_pairs:
            continue

        y2_top, x2_left, y2_bottom, x2_right = other_obj['bbox']

        # Check for overlap on both axes
        overlap_x = (x1_right > x2_left) and (x2_right > x1_left)
        overlap_y = (y1_bottom > y2_top) and (y2_bottom > y1_top)

        if overlap_x and overlap_y:
            # print(f"Overlap detected between {target_obj['label']} ({target_obj['bbox']}) and {other_obj['label']} ({other_obj['bbox']})")
            return True

    return False


def is_attached_to_wall_or_cutout(
    all_objects: List[Dict[str, Any]],
    room_bounds: Dict[str, float],
    epsilon: float = 1e-3
) -> bool:
    """
    Checks if work triangle elements (fridge, sink, stove) are attached to walls or cutouts.

    Args:
        all_objects: List of all objects in the scene.
        room_bounds: Dictionary with 'width' and 'depth' defining room dimensions.
        epsilon: Tolerance for floating point comparisons.

    Returns:
        True if all work triangle elements are properly attached, False otherwise.
    """
    # Define constants
    work_triangle_labels = ["fridge", "sink", "stove"]
    cutout_labels = ["cutout_area"]

    # Helper functions to check range overlaps
    def check_y_overlap(y1_a, y1_b, y2_a, y2_b):
        return (y1_b > y2_a) and (y2_b > y1_a)

    def check_x_overlap(x1_a, x1_b, x2_a, x2_b):
        return (x1_b > x2_a) and (x2_b > x1_a)

    # Room boundaries
    room_width = room_bounds['width']
    room_depth = room_bounds['depth']
    room_y_top, room_x_left = 0, 0
    room_y_bottom, room_x_right = room_depth, room_width

    # Check each work triangle element
    for obj in all_objects:
        if obj["label"] not in work_triangle_labels:
            continue

        # Extract object bbox [y_top, x_left, y_bottom, x_right]
        obj_y_top, obj_x_left, obj_y_bottom, obj_x_right = obj['bbox']
        is_attached = False

        # Check attachment to walls
        # Left Wall
        if (math.isclose(obj_x_left, room_x_left, abs_tol=epsilon) and
                check_y_overlap(obj_y_top, obj_y_bottom, room_y_top, room_y_bottom)):
            is_attached = True

        # Right Wall
        elif (math.isclose(obj_x_right, room_x_right, abs_tol=epsilon) and
              check_y_overlap(obj_y_top, obj_y_bottom, room_y_top, room_y_bottom)):
            is_attached = True

        # Top Wall
        elif (math.isclose(obj_y_top, room_y_top, abs_tol=epsilon) and
              check_x_overlap(obj_x_left, obj_x_right, room_x_left, room_x_right)):
            is_attached = True

        # Bottom Wall
        elif (math.isclose(obj_y_bottom, room_y_bottom, abs_tol=epsilon) and
              check_x_overlap(obj_x_left, obj_x_right, room_x_left, room_x_right)):
            is_attached = True

        # Check attachment to cutout objects
        if not is_attached:
            for cutout in [o for o in all_objects if o['label'] in cutout_labels]:
                cutout_y_top, cutout_x_left, cutout_y_bottom, cutout_x_right = cutout['bbox']

                # Left edge of cutout
                if (math.isclose(obj_x_right, cutout_x_left, abs_tol=epsilon) and
                        check_y_overlap(obj_y_top, obj_y_bottom, cutout_y_top, cutout_y_bottom)):
                    is_attached = True
                    break

                # Right edge of cutout
                elif (math.isclose(obj_x_left, cutout_x_right, abs_tol=epsilon) and
                      check_y_overlap(obj_y_top, obj_y_bottom, cutout_y_top, cutout_y_bottom)):
                    is_attached = True
                    break

                # Top edge of cutout
                elif (math.isclose(obj_y_bottom, cutout_y_top, abs_tol=epsilon) and
                      check_x_overlap(obj_x_left, obj_x_right, cutout_x_left, cutout_x_right)):
                    is_attached = True
                    break

                # Bottom edge of cutout
                elif (math.isclose(obj_y_top, cutout_y_bottom, abs_tol=epsilon) and
                      check_x_overlap(obj_x_left, obj_x_right, cutout_x_left, cutout_x_right)):
                    is_attached = True
                    break

        if not is_attached:
            return False

    return True


def get_wall(bbox: List[float], room_width: float, room_depth: float, tolerance: float = 0.1) -> str:
    """
    Determines which wall a bounding box is attached to.

    Args:
        bbox: Bounding box [y_top, x_left, y_bottom, x_right].
        room_width: Width of the room.
        room_depth: Depth of the room.
        tolerance: Tolerance for considering an object attached to a wall.

    Returns:
        String indicating the wall: 'left', 'right', 'top', 'bottom', or 'unknown'.
    """
    y1, x1, y2, x2 = bbox

    if abs(x1 - 0) < tolerance and abs(x2 - 0) < tolerance:
        return 'left'
    elif abs(x1 - room_width) < tolerance and abs(x2 - room_width) < tolerance:
        return 'right'
    elif abs(y1 - 0) < tolerance and abs(y2 - 0) < tolerance:
        return 'top'
    elif abs(y1 - room_depth) < tolerance and abs(y2 - room_depth) < tolerance:
        return 'bottom'
    else:
        return 'unknown'


def check_opposite_windows(layout_data: Dict[str, Any]) -> bool:
    """
    Checks if there are windows on opposite walls.

    Args:
        layout_data: Dictionary containing 'room' and 'objects' keys.

    Returns:
        True if windows are found on opposite walls, False otherwise.
    """
    room_width = layout_data['room']['width']
    room_depth = layout_data['room']['depth']
    windows = [obj for obj in layout_data['objects'] if obj['label'] == 'window']

    # Map windows to walls
    wall_windows = [(get_wall(w['bbox'], room_width, room_depth), w) for w in windows]

    # Define opposite walls
    opposites = {
        'left': 'right',
        'right': 'left',
        'top': 'bottom',
        'bottom': 'top'
    }

    # Check for windows on opposite walls
    for i in range(len(wall_windows)):
        for j in range(i + 1, len(wall_windows)):
            wall1, _ = wall_windows[i]
            wall2, _ = wall_windows[j]
            if opposites.get(wall1) == wall2:
                return True

    return False


def corners_from_bbox(bbox: List[float]) -> List[Tuple[float, float]]:
    """
    Returns the four corners of a bounding box as (x, y) coordinates.

    Args:
        bbox: Bounding box [y_top, x_left, y_bottom, x_right].

    Returns:
        List of corner coordinates as (x, y) tuples.
    """
    y1, x1, y2, x2 = bbox
    return [
        (x1, y1),  # top-left
        (x2, y1),  # top-right
        (x1, y2),  # bottom-left
        (x2, y2),  # bottom-right
    ]


def min_corner_distance(bbox1: List[float], bbox2: List[float]) -> float:
    """
    Calculates the minimum distance between any two corners of two bounding boxes.

    Args:
        bbox1: First bounding box [y_top, x_left, y_bottom, x_right].
        bbox2: Second bounding box [y_top, x_left, y_bottom, x_right].

    Returns:
        Minimum distance between any two corners.
    """
    corners1 = corners_from_bbox(bbox1)
    corners2 = corners_from_bbox(bbox2)

    return min(
        math.hypot(x1 - x2, y1 - y2)
        for (x1, y1), (x2, y2) in product(corners1, corners2)
    )


def is_table_too_close(layout_data: Dict[str, Any], min_distance: float = 0.8) -> bool:
    """
    Checks if tables are too close to major appliances.

    Args:
        layout_data: Dictionary containing 'objects' key.
        min_distance: Minimum acceptable distance between tables and appliances.

    Returns:
        True if any table is too close to a major appliance, False otherwise.
    """
    major_labels = {"stove", "sink", "fridge"}
    objects = layout_data["objects"]

    tables = [obj for obj in objects if obj["label"] == "table"]
    majors = [obj for obj in objects if obj["label"] in major_labels]

    for table in tables:
        for major in majors:
            dist = min_corner_distance(table["bbox"], major["bbox"])
            if dist < min_distance:
                return True

    return False


def do_bboxes_overlap(b1: List[float], b2: List[float]) -> bool:
    """
    Checks if two bounding boxes overlap.

    Args:
        b1: First bounding box [y_top, x_left, y_bottom, x_right].
        b2: Second bounding box [y_top, x_left, y_bottom, x_right].

    Returns:
        True if the bounding boxes overlap, False otherwise.
    """
    y1a, x1a, y2a, x2a = b1
    y1b, x1b, y2b, x2b = b2

    # Check if one box is to the left of the other
    if x2a <= x1b or x2b <= x1a:
        return False

    # Check if one box is above the other
    if y2a <= y1b or y2b <= y1a:
        return False

    return True


def get_door_clearance_bbox(door_bbox: List[float], room_dims: Dict[str, float]) -> Tuple[List[float], List[float]]:
    """
    Calculates clearance area needed for a door.

    Args:
        door_bbox: Door bounding box [y_top, x_left, y_bottom, x_right].
        room_dims: Dictionary with room dimensions.

    Returns:
        Tuple of two bounding boxes representing inward and outward clearance areas.
    """
    y1, x1, y2, x2 = door_bbox
    door_width = abs(y2 - y1)
    door_length = abs(x2 - x1)

    # Check door orientation (vertical or horizontal)
    if door_length > door_width:  # Horizontal door (on top/bottom wall)
        # Create clearance areas both inward and outward
        inward = [y2, x1, y2 + door_length, x2]         # Inward clearance
        outward = [y1 - door_length, x1, y1, x2]        # Outward clearance
    else:  # Vertical door (on left/right wall)
        inward = [y1, x2, y2, x2 + door_width]          # Inward clearance
        outward = [y1, x1 - door_width, y2, x1]         # Outward clearance

    return inward, outward


def is_door_blocked(layout_data: Dict[str, Any]) -> bool:
    """
    Checks if a door's clearance area is blocked by any object.

    Args:
        layout_data: Dictionary containing 'objects' and 'room' keys.

    Returns:
        True if the door clearance is blocked, False otherwise.
    """
    objects = layout_data["objects"]
    door = next((obj for obj in objects if obj["label"] == "door"), None)

    if not door:
        return False  # No door to check

    # Get door clearance areas
    inward_clearance, outward_clearance = get_door_clearance_bbox(door["bbox"], layout_data["room"])

    # Check if any object blocks the clearance
    for obj in objects:
        if obj["label"] in {"door", "chair", 'rug'}:  # These objects don't block the door
            continue

        # Check if object overlaps with either clearance area
        if (do_bboxes_overlap(inward_clearance, obj["bbox"]) or
                do_bboxes_overlap(outward_clearance, obj["bbox"])):
            return True

    return False


def is_closed(kitchen_layout: Dict[str, Any]) -> bool:
    """
    Determines whether the passage in an open kitchen is blocked.

    Args:
        kitchen_layout: Dictionary representing the kitchen layout.

    Returns:
        True if the passage is blocked, False otherwise.
    """
    room = kitchen_layout['room']
    objects = kitchen_layout['objects']

    # Only applies to open kitchens
    if room['shape'] != 'open':
        return False

    # Identify which wall is open
    open_wall = ""
    description = room['shape_description'].lower()

    if "left wall" in description:
        open_wall = "left"
    elif "right wall" in description:
        open_wall = "right"
    elif "top wall" in description:
        open_wall = "top"
    elif "bottom wall" in description:
        open_wall = "bottom"

    if not open_wall:
        return False

    # Find objects along the open wall
    wall_objects = []

    if open_wall == "left":
        wall_objects = [obj for obj in objects if abs(obj['bbox'][1]) < 0.01]
    elif open_wall == "right":
        wall_objects = [obj for obj in objects if abs(obj['bbox'][3] - room['width']) < 0.01]
    elif open_wall == "top":
        wall_objects = [obj for obj in objects if abs(obj['bbox'][0]) < 0.01]
    elif open_wall == "bottom":
        wall_objects = [obj for obj in objects if abs(obj['bbox'][2] - room['depth']) < 0.01]

    if not wall_objects:
        return False

    # Extract ranges along the wall
    dimension = 0
    ranges = []

    if open_wall in ["left", "right"]:
        ranges = [[obj['bbox'][0], obj['bbox'][2]] for obj in wall_objects]  # y-ranges
        dimension = room['depth']
    else:
        ranges = [[obj['bbox'][1], obj['bbox'][3]] for obj in wall_objects]  # x-ranges
        dimension = room['width']

    # Sort and merge overlapping ranges
    ranges.sort(key=lambda x: x[0])
    merged_ranges = [ranges[0]]

    for i in range(1, len(ranges)):
        current = ranges[i]
        last = merged_ranges[-1]

        if current[0] <= last[1]:
            last[1] = max(last[1], current[1])
        else:
            merged_ranges.append(current)

    # Find maximum gap between merged ranges
    max_gap = 0

    # Gap at the start
    if merged_ranges[0][0] > 0:
        max_gap = max(max_gap, merged_ranges[0][0])

    # Gaps between ranges
    for i in range(len(merged_ranges) - 1):
        max_gap = max(max_gap, merged_ranges[i+1][0] - merged_ranges[i][1])

    # Gap at the end
    if merged_ranges[-1][1] < dimension:
        max_gap = max(max_gap, dimension - merged_ranges[-1][1])

    # Passage is considered blocked if the maximum gap is less than 0.6 units
    return max_gap < 0.6


def check_working_space_clearance(layout_data: Dict[str, Any], min_clearance: float = 0.6) -> bool:
    """
    Checks if objects have sufficient working space clearance in front of them.
    For each object that needs working space, ensures it has clearance on the side
    opposite to the wall it's attached to.

    Args:
        layout_data: Dictionary containing 'room' and 'objects' keys.
        min_clearance: Minimum required clearance space in meters (default: 0.6).

    Returns:
        True if all working spaces have sufficient clearance, False otherwise.
        Also prints information about which object's clearance was violated and why.
    """
    room_width = layout_data['room']['width']
    room_depth = layout_data['room']['depth']
    objects = layout_data['objects']
    epsilon = 1e-3  # Small tolerance for floating point comparisons

    # Objects that need working space
    working_space_objects = [
        obj for obj in objects
        if obj['label'] in ['sink', 'fridge', 'stove', 'base_cabinet']
    ]

    # Find cutout areas
    cutouts = [obj for obj in objects if obj['label'] == 'cutout_area']

    for target_obj in working_space_objects:
        y_top, x_left, y_bottom, x_right = target_obj['bbox']

        # Check which single wall or cutout the object is attached to
        wall_attachments = []

        # Wall attachments
        if math.isclose(x_left, 0, abs_tol=epsilon):
            wall_attachments.append('left')
        if math.isclose(x_right, room_width, abs_tol=epsilon):
            wall_attachments.append('right')
        if math.isclose(y_top, 0, abs_tol=epsilon):
            wall_attachments.append('top')
        if math.isclose(y_bottom, room_depth, abs_tol=epsilon):
            wall_attachments.append('bottom')

        # Cutout attachments
        for cutout in cutouts:
            c_y_top, c_x_left, c_y_bottom, c_x_right = cutout['bbox']

            # Check if object is attached to a cutout edge
            # Right edge of object touching left edge of cutout
            if (math.isclose(x_right, c_x_left, abs_tol=epsilon) and
                    y_top < c_y_bottom and y_bottom > c_y_top):
                wall_attachments.append('right_cutout')

            # Left edge of object touching right edge of cutout
            if (math.isclose(x_left, c_x_right, abs_tol=epsilon) and
                    y_top < c_y_bottom and y_bottom > c_y_top):
                wall_attachments.append('left_cutout')

            # Bottom edge of object touching top edge of cutout
            if (math.isclose(y_bottom, c_y_top, abs_tol=epsilon) and
                    x_left < c_x_right and x_right > c_x_left):
                wall_attachments.append('bottom_cutout')

            # Top edge of object touching bottom edge of cutout
            if (math.isclose(y_top, c_y_bottom, abs_tol=epsilon) and
                    x_left < c_x_right and x_right > c_x_left):
                wall_attachments.append('top_cutout')

        # Skip objects attached to multiple walls or not attached to any wall
        if len(wall_attachments) != 1:
            continue

        # Create clearance zone on the opposite side of the wall/cutout
        attachment = wall_attachments[0]
        clearance_zone = None
        clearance_direction = ""

        if attachment == 'left':
            # Clearance zone to the right
            clearance_zone = [y_top, x_right, y_bottom, x_right + min_clearance]
            clearance_direction = "right"

        elif attachment == 'right':
            # Clearance zone to the left
            clearance_zone = [y_top, x_left - min_clearance, y_bottom, x_left]
            clearance_direction = "left"

        elif attachment == 'top':
            # Clearance zone to the bottom
            clearance_zone = [y_bottom, x_left, y_bottom + min_clearance, x_right]
            clearance_direction = "bottom"

        elif attachment == 'bottom':
            # Clearance zone to the top
            clearance_zone = [y_top - min_clearance, x_left, y_top, x_right]
            clearance_direction = "top"

        elif attachment == 'right_cutout':
            # Clearance zone to the left
            clearance_zone = [y_top, x_left - min_clearance, y_bottom, x_left]
            clearance_direction = "left"

        elif attachment == 'left_cutout':
            # Clearance zone to the right
            clearance_zone = [y_top, x_right, y_bottom, x_right + min_clearance]
            clearance_direction = "right"

        elif attachment == 'bottom_cutout':
            # Clearance zone to the top
            clearance_zone = [y_top - min_clearance, x_left, y_top, x_right]
            clearance_direction = "top"

        elif attachment == 'top_cutout':
            # Clearance zone to the bottom
            clearance_zone = [y_bottom, x_left, y_bottom + min_clearance, x_right]
            clearance_direction = "bottom"

        if clearance_zone is None:
            continue

        # Check if the clearance zone is valid (within room, not in cutouts)
        z_y_top, z_x_left, z_y_bottom, z_x_right = clearance_zone

        # Check room bounds
        if (z_x_left < 0 or z_x_right > room_width or
                z_y_top < 0 or z_y_bottom > room_depth):
            # print(f"Clearance violation: {target_obj['label']} at {target_obj['bbox']} - "
            #       f"clearance zone to the {clearance_direction} extends outside the room")
            return False

        # Check cutouts
        for cutout in cutouts:
            if do_bboxes_overlap(clearance_zone, cutout['bbox']):
                # print(f"Clearance violation: {target_obj['label']} at {target_obj['bbox']} - "
                #       f"clearance zone to the {clearance_direction} overlaps with a cutout at {cutout['bbox']}")
                return False

        # Check if any other object blocks this clearance zone
        for other_obj in objects:
            if other_obj == target_obj or other_obj['label'] in ['window', 'door']:
                continue

            if do_bboxes_overlap(clearance_zone, other_obj['bbox']):
                # print(f"Clearance violation: {target_obj['label']} at {target_obj['bbox']} - "
                #       f"clearance zone to the {clearance_direction} is blocked by {other_obj['label']} at {other_obj['bbox']}")
                return False

    # If all objects have sufficient clearance, return True
    return True
