"""Utility functions for QA pairs generation from room layouts."""
import math
import os
import random
import re
import time
from pathlib import Path
from typing import Callable, Dict, List, Optional, Tuple

import numpy as np
import pandas as pd
import matplotlib.pyplot as plt
from matplotlib.collections import LineCollection
from matplotlib.ticker import AutoMinorLocator

from shapely.geometry import (
    LineString,
    MultiPolygon,
    Point,
    Polygon,
    box,
)
from shapely.geometry import Polygon as ShpPolygon
from shapely.ops import linemerge, polygonize, unary_union
from shapely.prepared import prep
from shapely.validation import make_valid


def get_polygon_centroid(points: List[Dict[str, float]]) -> Tuple[float, float]:
    """Calculate the centroid of a polygon from a list of points."""
    n = len(points)
    if n < 2:
        raise ValueError("A line or polygon must have at least 2 points.")

    coords = [(p["x"], p["y"]) for p in points]

    if n == 2:
        # Line segment: midpoint
        (x1, y1), (x2, y2) = coords
        return (round((x1 + x2) / 2.0, 2), round((y1 + y2) / 2.0, 2))

    poly = Polygon(coords)
    if not poly.is_valid or poly.area == 0:
        # Degenerate polygon: average of vertices
        cx = sum(x for x, _ in coords) / n
        cy = sum(y for _, y in coords) / n
        return (round(cx, 3), round(cy, 3))

    c = poly.centroid  # shapely Point
    return (round(c.x, 3), round(c.y, 3))


def get_polygon_centroid_manually(
    points: List[Dict[str, float]]
) -> Tuple[float, float]:
    """
    Compute the true geometric centroid (center of mass) of a polygon.

    Args:
        points: A list of dictionaries with 'x' and 'y' keys.

    Returns:
        A tuple (x, y) of the centroid, rounded to 3 decimal places.
    """
    n = len(points)
    if n < 2:
        raise ValueError("A line or polygon must have at least 2 points.")

    # Special case for a line segment (n=2). The centroid is the midpoint.
    if n == 2:
        x_coords = [p["x"] for p in points]
        y_coords = [p["y"] for p in points]
        cx = sum(x_coords) / n
        cy = sum(y_coords) / n
        return (round(cx, 2), round(cy, 2))

    # Close the polygon
    vertices = points + [points[0]]

    signed_area = 0.0
    sum_cx = 0.0
    sum_cy = 0.0

    for i in range(n):
        x_i, y_i = vertices[i]["x"], vertices[i]["y"]
        x_next, y_next = vertices[i + 1]["x"], vertices[i + 1]["y"]

        cross_product_term = (x_i * y_next) - (x_next * y_i)
        signed_area += cross_product_term
        sum_cx += (x_i + x_next) * cross_product_term
        sum_cy += (y_next + y_i) * cross_product_term

    final_signed_area = signed_area / 2.0

    if math.isclose(final_signed_area, 0.0):
        print("Warning: Polygon has zero area. Using simple average.")
        x_coords = [p["x"] for p in points]
        y_coords = [p["y"] for p in points]
        cx = sum(x_coords) / n
        cy = sum(y_coords) / n
    else:
        cx = sum_cx / (6.0 * final_signed_area)
        cy = sum_cy / (6.0 * final_signed_area)

    return (round(cx, 3), round(cy, 3))


def euclidean_distance(p1: tuple, p2: tuple) -> float:
    """Calculate the Euclidean distance between two points."""
    return math.sqrt((p1[0] - p2[0]) ** 2 + (p1[1] - p2[1]) ** 2)


def generate_qa_pairs_with_subsampling(
    input_dir: str,
    process_single_file: Callable,
    SEED: int = 42,
    subsample_config: Optional[Dict[str, int]] = None
) -> List[Dict]:
    """Generate QA pairs with optional subsampling by room type."""
    qa_pairs: List[Dict] = []
    rng = np.random.default_rng(SEED)

    if subsample_config:
        for room_type, max_count in subsample_config.items():
            room_dir = os.path.join(input_dir, room_type)
            if not os.path.exists(room_dir):
                print(f"Warning: Directory {room_dir} does not exist, "
                      f"skipping {room_type}")
                continue

            all_files = sorted(
                [f for f in os.listdir(room_dir) if f.endswith(".json")]
            )
            rng.shuffle(all_files)
            selected_files = all_files[:max_count]
            print(f"Processing {len(selected_files)} files from {room_type} "
                  f"(out of {len(all_files)} available)")

            for file in selected_files:
                file_path = os.path.join(room_dir, file)
                qa_pair = process_single_file(
                    file_path=file_path, file=file, room_type_arg=room_type
                )
                if qa_pair:
                    qa_pairs.append(qa_pair)

        return qa_pairs

    # No-subsampling branch
    tasks: List[tuple] = []
    for item in sorted(os.listdir(input_dir)):
        item_path = os.path.join(input_dir, item)
        if os.path.isdir(item_path):
            room_type = item
            for file in sorted(os.listdir(item_path)):
                if file.endswith(".json"):
                    file_path = os.path.join(item_path, file)
                    tasks.append((file_path, file, room_type))
        elif item.endswith(".json"):
            tasks.append((item_path, item, "unknown"))

    if not tasks:
        return qa_pairs

    for idx, task in enumerate(tasks):
        print(f"Processing {idx}...")
        res = process_single_file(*task)
        if res:
            qa_pairs.append(res)

    return qa_pairs


def save_and_info(qa_pairs, output_csv="output.csv"):
    """Save QA pairs to CSV and print summary."""
    df = pd.DataFrame(qa_pairs)
    df["layout_id"] = df["layout_id"].astype(int)
    df.sort_values(by="layout_id", inplace=True)
    df.reset_index(drop=True, inplace=True)
    df.to_csv(output_csv, index=False)
    print(f"Saved {len(df)} QA pairs to {output_csv}")

    if len(df) > 0:
        summary = df["room_type"].value_counts()
        print("\nSummary by room type:")
        for room_type, count in summary.items():
            print(f"  {room_type}: {count}")


def merge_objects_and_openings(data: dict) -> list:
    """
    Return a unified list of objects, where windows and doors
    (if present in 'openings') are treated as objects.
    """
    objects = data.get("objects", []) or []
    openings = data.get("openings", {}) or {}
    windows = openings.get("windows", []) or []
    doors = openings.get("doors", []) or []
    return objects + windows + doors


def _poly_to_segments(points, close=True):
    """Convert polygon points to line segments."""
    if not points or len(points) < 2:
        return []
    segs = []
    n = len(points)
    for i in range(n - 1):
        p0, p1 = points[i], points[i + 1]
        segs.append([
            (float(p0["x"]), float(p0["y"])),
            (float(p1["x"]), float(p1["y"]))
        ])
    if close and n >= 3:
        p0, p1 = points[-1], points[0]
        segs.append([
            (float(p0["x"]), float(p0["y"])),
            (float(p1["x"]), float(p1["y"]))
        ])
    return segs


def _polygon_centroid_numpy(points):
    """Area-weighted centroid with fallback to vertex-mean."""
    if not points:
        return (0.0, 0.0)
    x = np.asarray([p["x"] for p in points], dtype=float)
    y = np.asarray([p["y"] for p in points], dtype=float)
    if len(points) >= 3:
        x2 = np.r_[x, x[0]]
        y2 = np.r_[y, y[0]]
        cross = x2[:-1] * y2[1:] - x2[1:] * y2[:-1]
        area = cross.sum() / 2.0
        if abs(area) > 1e-9:
            cx = ((x2[:-1] + x2[1:]) * cross).sum() / (6.0 * area)
            cy = ((y2[:-1] + y2[1:]) * cross).sum() / (6.0 * area)
            return (float(cx), float(cy))
    return (float(x.mean()), float(y.mean()))


def render_layout_pair(
    data: dict,
    obj1: dict,
    obj2: dict,
    center1: tuple,
    center2: tuple,
    distance: float,
    out_dir: str,
    layout_id,
    dpi: int = 150,
):
    """
    Render the full layout with two highlighted objects and a connecting line.

    Saves to {out_dir}/{layout_id}.png
    """
    # Styling constants
    FIG_SIZE = (8, 8)
    COLOR_ROOM_OUTLINE = "#444444"
    COLOR_WALL_SEGMENTS = "#000000"
    COLOR_OBJECTS = "#8888ff"
    COLOR_DOORS = "#22aa55"
    COLOR_WINDOWS = "#1f77b4"
    COLOR_LABEL = "#222222"
    COLOR_CENTER_A = "#d62728"
    COLOR_CENTER_B = "#9467bd"
    COLOR_PAIR_LINE = "#111111"

    LW_ROOM_OUTLINE = 1.8
    LW_WALL_SEGMENTS = 3.0
    LW_OBJECTS = 1.8
    LW_DOORS = 2.2
    LW_WINDOWS = 2.2
    LW_PAIR_LINE = 2.2

    MARKER_SIZE = 30
    LABEL_FONT = 8
    LABEL_BBOX = dict(boxstyle="round,pad=0.2", fc="white", ec="none", alpha=0.8)

    # Prepare output path
    out_path = Path(out_dir) / f"{layout_id}.png"
    out_path.parent.mkdir(parents=True, exist_ok=True)

    # Unpack data
    room_boundary = data.get("room_boundary", []) or []
    walls = data.get("walls", []) or []
    openings = data.get("openings", {}) or {}
    windows = openings.get("windows", []) or []
    doors = openings.get("doors", []) or []
    objects = data.get("objects", []) or []
    units = data.get("units") or (data.get("room", {}) or {}).get("units") or ""

    # Room outline
    room_np = (
        np.asarray([[p["x"], p["y"]] for p in room_boundary], dtype=float)
        if room_boundary else None
    )

    # Wall segments
    wall_segs = []
    for w in walls:
        s, e = w.get("start"), w.get("end")
        if not s or not e:
            continue
        wall_segs.append([
            (float(s["x"]), float(s["y"])),
            (float(e["x"]), float(e["y"]))
        ])

    # Collect polygons and labels
    def _collect(polys, default_label):
        segs, labels = [], []
        for obj in polys:
            pts = obj.get("points", []) or []
            lbl = obj.get("label", default_label)
            segs.extend(_poly_to_segments(pts, close=True))
            cx, cy = _polygon_centroid_numpy(pts)
            labels.append((lbl, cx, cy))
        return segs, labels

    segs_objects, lbl_objects = _collect(objects, "object")
    segs_doors, lbl_doors = _collect(doors, "door")
    segs_windows, lbl_windows = _collect(windows, "window")

    # Create figure
    fig, ax = plt.subplots(figsize=FIG_SIZE, dpi=dpi)

    if room_np is not None and len(room_np):
        ax.plot(
            room_np[:, 0], room_np[:, 1],
            color=COLOR_ROOM_OUTLINE, linewidth=LW_ROOM_OUTLINE,
            linestyle="--", alpha=0.8, label="Room boundary"
        )

    if wall_segs:
        ax.add_collection(LineCollection(
            wall_segs, linewidths=LW_WALL_SEGMENTS,
            colors=[COLOR_WALL_SEGMENTS], label="Walls"
        ))

    if segs_objects:
        ax.add_collection(LineCollection(
            segs_objects, linewidths=LW_OBJECTS,
            colors=[COLOR_OBJECTS], label="Objects"
        ))

    if segs_doors:
        ax.add_collection(LineCollection(
            segs_doors, linewidths=LW_DOORS,
            colors=[COLOR_DOORS], label="Doors"
        ))

    if segs_windows:
        ax.add_collection(LineCollection(
            segs_windows, linewidths=LW_WINDOWS,
            colors=[COLOR_WINDOWS], label="Windows"
        ))

    # Labels
    for txt, x, y in lbl_doors:
        ax.text(
            x, y, txt, ha="center", va="center",
            fontsize=LABEL_FONT, color=COLOR_DOORS, bbox=LABEL_BBOX
        )
    for txt, x, y in lbl_windows:
        ax.text(
            x, y, txt, ha="center", va="center",
            fontsize=LABEL_FONT, color=COLOR_WINDOWS, bbox=LABEL_BBOX
        )
    for txt, x, y in lbl_objects:
        ax.text(
            x, y, txt, ha="center", va="center",
            fontsize=LABEL_FONT, color=COLOR_LABEL, bbox=LABEL_BBOX
        )

    # Highlight the chosen pair
    ax.scatter([center1[0]], [center1[1]], s=MARKER_SIZE, color=COLOR_CENTER_A, zorder=5)
    ax.scatter([center2[0]], [center2[1]], s=MARKER_SIZE, color=COLOR_CENTER_B, zorder=5)
    ax.plot(
        [center1[0], center2[0]], [center1[1], center2[1]],
        color=COLOR_PAIR_LINE, linewidth=LW_PAIR_LINE, zorder=4
    )

    # Distance text at midpoint
    mx, my = (center1[0] + center2[0]) / 2.0, (center1[1] + center2[1]) / 2.0
    suffix = f" {units}" if units else ""
    ax.text(
        mx, my, f"{distance:.2f}{suffix}",
        ha="center", va="bottom", fontsize=LABEL_FONT,
        color=COLOR_PAIR_LINE, bbox=LABEL_BBOX
    )

    # Finalize
    ax.set_aspect("equal", adjustable="box")
    handles, labels = ax.get_legend_handles_labels()
    if labels:
        ax.legend(loc="upper right", fontsize=8)

    ax.set_xlabel("X (m)")
    ax.set_ylabel("Y (m)")
    ax.grid(True, which="major", linewidth=0.6, alpha=0.7)
    ax.grid(True, which="minor", linewidth=0.3, alpha=0.4)
    ax.xaxis.set_minor_locator(AutoMinorLocator())
    ax.yaxis.set_minor_locator(AutoMinorLocator())

    plt.tight_layout(pad=0.1)
    plt.savefig(out_path, dpi=dpi, bbox_inches="tight", pad_inches=0.1)
    plt.close(fig)

    return str(out_path)


def collect_opening_polygons(data: dict):
    """Collect polygons for openings (doors + windows) if they have area."""
    openings = data.get("openings", {}) or {}
    windows = openings.get("windows", []) or []
    doors = openings.get("doors", []) or []

    polys = []
    for o in windows + doors:
        pts = o.get("points", []) or []
        if len(pts) < 3:
            continue
        poly = Polygon([(p["x"], p["y"]) for p in pts])
        if poly.is_valid and poly.area > 0:
            polys.append(poly)
    return polys


def build_room_polygon_from_walls(walls):
    """Build a room polygon from wall segments."""
    segs = []
    for w in walls or []:
        s, e = w.get("start"), w.get("end")
        if not s or not e:
            continue
        p1 = (s["x"], s["y"])
        p2 = (e["x"], e["y"])
        if p1 != p2:
            segs.append(LineString([p1, p2]))

    if not segs:
        return None

    merged = unary_union(segs)
    merged = linemerge(merged)
    faces = list(polygonize(merged))
    if not faces:
        return None

    room_poly = max(faces, key=lambda p: p.area)
    return make_valid(room_poly)


def build_room_polygon(data: dict):
    """
    Prefer 'room_boundary' polygon if present & valid; otherwise polygonize from walls.
    """
    rb = data.get("room_boundary")
    if rb and len(rb) >= 3:
        pts = [(p["x"], p["y"]) for p in rb]
        poly = Polygon(pts)
        if poly.is_valid and poly.area > 0:
            return make_valid(poly)

    walls = data.get("walls") or (data.get("room", {}) or {}).get("walls") or []
    return build_room_polygon_from_walls(walls)


def collect_occupied_polygons(
    data: dict,
    exclude_labels_regex=r"(light|chandelier|fan|pendant)"
):
    """
    Returns polygons for objects that count as occupied floor area.
    Skips labels matching exclude_labels_regex (ceiling fixtures).
    """
    patt = re.compile(exclude_labels_regex, flags=re.IGNORECASE)
    occupied = []
    for obj in data.get("objects", []) or []:
        label = str(obj.get("label", ""))
        if patt.search(label or ""):
            continue
        pts = obj.get("points", []) or []
        if len(pts) < 3:
            continue
        coords = [(p["x"], p["y"]) for p in pts]
        poly = Polygon(coords)
        if poly.is_valid and poly.area > 0:
            occupied.append(make_valid(poly))
    return occupied


def _draw_room_walls(ax, data: dict):
    """Draw room boundary and walls on axes."""
    rb = data.get("room_boundary") or []
    if rb:
        rb_xy = np.array([[p["x"], p["y"]] for p in rb], dtype=float)
        ax.plot(
            rb_xy[:, 0], rb_xy[:, 1], "--",
            color="#444444", linewidth=1.6, alpha=0.8, label="Room boundary"
        )

    walls = data.get("walls") or (data.get("room", {}) or {}).get("walls") or []
    wall_segs = []
    for w in walls:
        s, e = w.get("start"), w.get("end")
        if not s or not e:
            continue
        wall_segs.append([
            (float(s["x"]), float(s["y"])),
            (float(e["x"]), float(e["y"]))
        ])
    if wall_segs:
        ax.add_collection(LineCollection(
            wall_segs, linewidths=2.2, colors=["#000000"], label="Walls"
        ))


def _fill_multipolygon(
    ax, geom,
    facecolor="#69d26f", edgecolor="none", alpha=0.35, lw=0.0
):
    """Fill a Polygon/MultiPolygon on a matplotlib Axes."""
    def _fill_poly(poly):
        x, y = poly.exterior.xy
        ax.fill(x, y, facecolor=facecolor, edgecolor=edgecolor, alpha=alpha, linewidth=lw)
        for ring in poly.interiors:
            hx, hy = ring.xy
            ax.fill(hx, hy, facecolor="white", edgecolor="none", alpha=1.0)

    if geom.is_empty:
        return
    if isinstance(geom, MultiPolygon):
        for p in geom.geoms:
            _fill_poly(p)
    else:
        _fill_poly(geom)


def render_free_space_green(
    data: dict, free_geom, out_dir: str, layout_id, dpi: int = 150
):
    """Render the room and fill the free space geometry in green."""
    out_path = Path(out_dir) / f"{layout_id}.png"
    out_path.parent.mkdir(parents=True, exist_ok=True)

    fig, ax = plt.subplots(figsize=(7.5, 7.5), dpi=dpi)
    _draw_room_walls(ax, data)
    _fill_multipolygon(ax, free_geom, facecolor="#7CFC00", alpha=0.45)

    ax.set_aspect("equal", adjustable="box")
    ax.set_xlabel("X (m)")
    ax.set_ylabel("Y (m)")
    ax.grid(True, which="major", linewidth=0.6, alpha=0.7)
    ax.grid(True, which="minor", linewidth=0.3, alpha=0.4)
    ax.xaxis.set_minor_locator(AutoMinorLocator())
    ax.yaxis.set_minor_locator(AutoMinorLocator())

    plt.tight_layout(pad=0.1)
    plt.savefig(out_path, dpi=dpi, bbox_inches="tight", pad_inches=0.1)
    plt.close(fig)

    return str(out_path)


# ----------------------------
# Obstacle collection
# ----------------------------
def is_rug_label(label: str) -> bool:
    """Check if label indicates a rug/carpet."""
    keywords = ["rug", "carpet", "mat", "doormat", "runner"]
    label_lc = (label or "").lower()
    return any(k in label_lc for k in keywords)


def is_light_label(label: str) -> bool:
    """Check if label indicates a ceiling light fixture."""
    keywords = ["light", "chandelier", "fan", "pendant"]
    label_lc = (label or "").lower()
    return any(k in label_lc for k in keywords)


def _label(o: Dict) -> str:
    """Get label from object dict."""
    return (o.get("label") or o.get("name") or "").strip().lower()


def collect_obstacle_polygons_for_maxbox(
    data: dict,
    exclude_ceiling_regex: str = r"(light|chandelier|fan|pendant)"
) -> List[ShpPolygon]:
    """Objects (excluding rugs and ceiling fixtures) + openings."""
    patt = re.compile(exclude_ceiling_regex, flags=re.IGNORECASE)
    polys: List[ShpPolygon] = []

    for obj in data.get("objects", []) or []:
        lbl = obj.get("label", "") or ""
        if is_rug_label(lbl) or patt.search(lbl):
            continue
        pts = obj.get("points", []) or []
        if len(pts) < 3:
            continue
        P = Polygon([(p["x"], p["y"]) for p in pts])
        if P.is_valid and P.area > 0:
            polys.append(make_valid(P))

    openings = data.get("openings", {}) or {}
    for o in (openings.get("doors", []) or []) + (openings.get("windows", []) or []):
        pts = o.get("points", []) or []
        if len(pts) < 3:
            continue
        P = Polygon([(p["x"], p["y"]) for p in pts])
        if P.is_valid and P.area > 0:
            polys.append(make_valid(P))

    return polys


# ----------------------------
# Rotation helpers
# ----------------------------
def _rot2d(xy: np.ndarray, theta: float) -> np.ndarray:
    """Rotate Nx2 array by angle theta (radians) around origin."""
    c, s = math.cos(theta), math.sin(theta)
    R = np.array([[c, -s], [s, c]], dtype=float)
    return xy @ R.T


def _rot_mat(theta: float):
    """Create 2D rotation matrix."""
    c, s = math.cos(theta), math.sin(theta)
    return np.array([[c, -s], [s, c]], dtype=float)


def _polygon_to_np(poly: ShpPolygon) -> np.ndarray:
    """Convert shapely polygon to numpy array."""
    return np.asarray(poly.exterior.coords, dtype=float)[:, :2]


def _rect_to_polygon(rect: Tuple[float, float, float, float]) -> ShpPolygon:
    """Convert rect (x1, y1, x2, y2) to shapely polygon using optimized box()."""
    x1, y1, x2, y2 = rect
    return box(x1, y1, x2, y2)


def _poly_from_np(xy: np.ndarray) -> ShpPolygon:
    """Create shapely polygon from numpy array."""
    return ShpPolygon(xy)


def _rotate_polygon(poly: ShpPolygon, theta: float) -> ShpPolygon:
    """Rotate a polygon around the origin."""
    xy = _polygon_to_np(poly)
    rxy = _rot2d(xy, theta)
    return _poly_from_np(rxy)


def _rotate_rect(rect: Tuple[float, float, float, float], theta: float) -> ShpPolygon:
    """Return oriented polygon by rotating an axis-aligned rect around origin."""
    return _rotate_polygon(_rect_to_polygon(rect), theta)


def _rect_poly(
    center: Tuple[float, float], w: float, h: float, theta: float
) -> ShpPolygon:
    """Rectangle polygon centered at center, width w, height h, rotated by theta."""
    cx, cy = center
    hw, hh = 0.5 * w, 0.5 * h
    local = np.array([[-hw, -hh], [hw, -hh], [hw, hh], [-hw, hh]], dtype=float)
    R = _rot_mat(theta)
    pts = (local @ R.T) + np.array([cx, cy])
    return ShpPolygon(pts)


def _rect_poly_halves(center, aL, aR, bD, bU, theta):
    """Build rectangle from half-extents in each direction."""
    cx, cy = center
    c, s = math.cos(theta), math.sin(theta)
    P = np.array([
        [-aL, -bD],
        [+aR, -bD],
        [+aR, +bU],
        [-aL, +bU],
    ], dtype=float)
    R = np.array([[c, -s], [s, c]], dtype=float)
    pts = (P @ R.T) + np.array([cx, cy])
    return ShpPolygon(pts)


# ----------------------------
# Candidate angles
# ----------------------------
def _angles_from_edges(polys: List[ShpPolygon]) -> List[float]:
    """Collect unique edge directions (0..pi) from polygons."""
    angs = set()
    for P in polys:
        coords = np.asarray(P.exterior.coords, dtype=float)[:, :2]
        for i in range(len(coords) - 1):
            v = coords[i + 1] - coords[i]
            if np.allclose(v, 0):
                continue
            a = math.atan2(v[1], v[0])
            a = a % math.pi
            angs.add(round(a, 10))
    return sorted(angs)


# ----------------------------
# Axis-aligned solver
# ----------------------------
class _Rect:
    """Simple rectangle class with x1, y1, x2, y2 coordinates."""
    __slots__ = ("x1", "y1", "x2", "y2")

    def __init__(self, x1, y1, x2, y2):
        self.x1, self.y1, self.x2, self.y2 = x1, y1, x2, y2

    @property
    def area(self):
        return max(0.0, self.x2 - self.x1) * max(0.0, self.y2 - self.y1)


def _largest_empty_rectangle_axis_aligned(
    room_polygon: ShpPolygon,
    obstacle_polys: List[ShpPolygon],
    min_size: float = 0.01,
    snap: float = 1e-3,
    max_grid: int = 2500,
    thin_stride: int = 2,
    *,
    verbose: bool = True,
    report_every: int = 20000,
    progress_prefix: str = "",
) -> Tuple[_Rect, float]:
    """
    Find largest axis-aligned empty rectangle in room avoiding obstacles.

    Uses snapped coordinate buckets and prepared geometries for efficiency.
    """
    t0 = time.perf_counter()

    def _sn(v):
        return float(np.round(v / snap) * snap)

    minx, miny, maxx, maxy = room_polygon.bounds

    xs = {_sn(minx), _sn(maxx)}
    ys = {_sn(miny), _sn(maxy)}

    rx, ry = zip(*list(room_polygon.exterior.coords))
    xs.update(_sn(x) for x in rx)
    ys.update(_sn(y) for y in ry)

    for poly in obstacle_polys:
        bx1, by1, bx2, by2 = poly.bounds
        xs.update((_sn(bx1), _sn(bx2)))
        ys.update((_sn(by1), _sn(by2)))

    xs = sorted(xs)
    ys = sorted(ys)
    combos = len(xs) * len(ys)

    if verbose:
        print(f"{progress_prefix}grid pre-thin: |xs|={len(xs)}, "
              f"|ys|={len(ys)}, combos={combos}")

    if combos > max_grid:
        xs = xs[::thin_stride] + (
            [xs[-1]] if xs[-1] != xs[::thin_stride][-1] else []
        )
        ys = ys[::thin_stride] + (
            [ys[-1]] if ys[-1] != ys[::thin_stride][-1] else []
        )
        if verbose:
            print(f"{progress_prefix}grid post-thin: |xs|={len(xs)}, "
                  f"|ys|={len(ys)}, combos={len(xs)*len(ys)}")

    best = _Rect(0, 0, 0, 0)
    best_area = 0.0

    room_prep = prep(room_polygon)
    merged_obs = unary_union(obstacle_polys) if obstacle_polys else None
    merged_prep = prep(merged_obs) if merged_obs else None

    checked = 0
    last_print = t0

    for i, x1 in enumerate(xs):
        for x2 in xs[i + 1:]:
            if (x2 - x1) <= min_size:
                continue
            for j, y1 in enumerate(ys):
                for y2 in ys[j + 1:]:
                    if (y2 - y1) <= min_size:
                        continue

                    area = (x2 - x1) * (y2 - y1)
                    if area <= best_area:
                        continue

                    Rpoly = box(x1, y1, x2, y2)

                    if not room_prep.contains(Rpoly) and not room_polygon.covers(Rpoly):
                        continue

                    if merged_prep and not Rpoly.disjoint(merged_obs):
                        continue

                    best = _Rect(x1, y1, x2, y2)
                    best_area = area

                    checked += 1
                    if verbose and (
                        checked % report_every == 0 or
                        (time.perf_counter() - last_print) > 2.0
                    ):
                        print(f"{progress_prefix}checked={checked:,} "
                              f"best_area={best_area:.3f}")
                        last_print = time.perf_counter()

                checked += max(0, len(ys) - (j + 1))

    if verbose:
        dt = time.perf_counter() - t0
        print(f"{progress_prefix}done in {dt:.2f}s, checks={checked:,}, "
              f"best_area={best_area:.3f}")

    return best, best_area


def largest_empty_rectangle_any_angle(
    room_polygon: ShpPolygon,
    obstacle_polys: List[ShpPolygon],
    extra_angles_deg: Optional[List[float]] = None,
    min_size: float = 0.01
) -> Tuple[ShpPolygon, float, float]:
    """
    Try best axis-aligned rectangle across multiple rotations.

    Returns (oriented_rectangle_polygon, area, angle_radians).
    """
    angs = set(_angles_from_edges([room_polygon] + obstacle_polys))
    for a in list(angs):
        angs.add(round((a + math.pi / 2) % math.pi, 10))
    if extra_angles_deg:
        for d in extra_angles_deg:
            a = math.radians(d) % math.pi
            angs.add(round(a, 10))

    best_poly = None
    best_area = 0.0
    best_angle = 0.0

    for a in sorted(angs):
        theta = -a

        Rroom = _rotate_polygon(room_polygon, theta)
        Robs = [_rotate_polygon(p, theta) for p in obstacle_polys]

        rect_axis, area = _largest_empty_rectangle_axis_aligned(
            Rroom, Robs, min_size=min_size
        )
        if area <= best_area:
            continue

        rect_poly = _rotate_rect(
            (rect_axis.x1, rect_axis.y1, rect_axis.x2, rect_axis.y2), -theta
        )
        best_poly, best_area, best_angle = rect_poly, area, (a % math.pi)

    return best_poly, best_area, best_angle


def render_max_box(
    data: dict,
    room_polygon: ShpPolygon,
    obstacles: List[ShpPolygon],
    obox: ShpPolygon,
    out_path: str,
    dpi: int = 150,
) -> str:
    """Draw room, walls, all objects/openings outlines, and the max box."""
    Path(out_path).parent.mkdir(parents=True, exist_ok=True)

    COL_ROOM = "#444444"
    COL_WALL = "#000000"
    COL_OBJ = "#8888ff"
    COL_DOOR = "#22aa55"
    COL_WIN = "#1f77b4"
    COL_BOX = "#e74c3c"

    fig, ax = plt.subplots(figsize=(8, 8), dpi=dpi)

    rb = data.get("room_boundary") or []
    if rb:
        rb_xy = np.array([[p["x"], p["y"]] for p in rb], dtype=float)
        ax.plot(
            rb_xy[:, 0], rb_xy[:, 1], "--",
            color=COL_ROOM, linewidth=1.6, alpha=0.8, label="Room boundary"
        )

    walls = data.get("walls") or (data.get("room", {}) or {}).get("walls") or []
    wall_segs = []
    for w in walls:
        s, e = w.get("start"), w.get("end")
        if s and e:
            wall_segs.append([
                (float(s["x"]), float(s["y"])),
                (float(e["x"]), float(e["y"]))
            ])
    if wall_segs:
        ax.add_collection(LineCollection(
            wall_segs, linewidths=2.2, colors=[COL_WALL], label="Walls"
        ))

    for obj in data.get("objects", []) or []:
        pts = obj.get("points", []) or []
        if len(pts) >= 2:
            xs = [p["x"] for p in pts]
            ys = [p["y"] for p in pts]
            if len(pts) >= 3:
                xs += [pts[0]["x"]]
                ys += [pts[0]["y"]]
            ax.plot(xs, ys, color=COL_OBJ, linewidth=1.4, alpha=0.9)

    openings = data.get("openings", {}) or {}
    for group, col in (("doors", COL_DOOR), ("windows", COL_WIN)):
        for o in openings.get(group, []) or []:
            pts = o.get("points", []) or []
            if len(pts) >= 2:
                xs = [p["x"] for p in pts]
                ys = [p["y"] for p in pts]
                if len(pts) >= 3:
                    xs += [pts[0]["x"]]
                    ys += [pts[0]["y"]]
                ax.plot(xs, ys, color=col, linewidth=1.6, alpha=1.0)

    if obox and not obox.is_empty:
        x, y = obox.exterior.xy
        ax.plot(x, y, color=COL_BOX, linewidth=3.0, alpha=0.95, label="Max box")

    ax.set_aspect("equal", adjustable="box")
    h, lbls = ax.get_legend_handles_labels()
    if lbls:
        ax.legend(loc="upper right", fontsize=8)

    ax.set_xlabel("X (m)")
    ax.set_ylabel("Y (m)")
    ax.grid(True, which="major", linewidth=0.6, alpha=0.7)
    ax.grid(True, which="minor", linewidth=0.3, alpha=0.4)
    ax.xaxis.set_minor_locator(AutoMinorLocator())
    ax.yaxis.set_minor_locator(AutoMinorLocator())

    plt.tight_layout(pad=0.1)
    layout_id = data["layout_id"]
    final_path = Path(out_path) / f"{layout_id}.png"
    final_path.parent.mkdir(parents=True, exist_ok=True)
    plt.savefig(final_path, dpi=dpi, bbox_inches="tight", pad_inches=0.1)
    plt.close(fig)

    return str(final_path)


# ----------------------------
# Coordinate ascent grow functions
# ----------------------------
def _grow_axis(
    prep_space,
    free_space,
    fs_bounds,
    center,
    theta,
    w0,
    h0,
    w_max,
    h_max,
    *,
    tol: float = 1e-3,
    allow_touch: bool = True,
    clearance: float = 0.0,
    max_passes: int = 50,
    time_budget_s: Optional[float] = None,
    t_start: Optional[float] = None,
) -> Tuple[float, float]:
    """Coordinate-ascent grow with robust binary searches."""
    if t_start is None:
        t_start = time.perf_counter()

    fs_minx, fs_miny, fs_maxx, fs_maxy = fs_bounds

    eff_space = free_space.buffer(-clearance) if clearance > 0 else free_space
    eff_prep = prep_space if clearance == 0 else prep(eff_space)

    def ok(W: float, H: float) -> bool:
        if W <= 0.0 or H <= 0.0:
            return False
        poly = _rect_poly(center, W, H, theta)
        minx, miny, maxx, maxy = poly.bounds
        if minx < fs_minx or miny < fs_miny or maxx > fs_maxx or maxy > fs_maxy:
            return False
        if eff_prep.contains(poly):
            return True
        return allow_touch and eff_space.covers(poly)

    def grow_one_dim(cur, cap, other, grow_width: bool) -> float:
        val = min(max(cur, 1e-9), cap)

        for _ in range(64):
            cand = min(val * 2.0, cap)
            good = ok(cand, other) if grow_width else ok(other, cand)
            if good and cand > val + tol:
                val = cand
                if time_budget_s and (time.perf_counter() - t_start) > time_budget_s:
                    return val
            else:
                break

        lo, hi = val, cap
        for _ in range(64):
            if (hi - lo) <= tol:
                break
            mid = 0.5 * (lo + hi)
            good = ok(mid, other) if grow_width else ok(other, mid)
            if good:
                lo = mid
            else:
                hi = mid
            if time_budget_s and (time.perf_counter() - t_start) > time_budget_s:
                break

        val = lo
        step = tol * 0.5
        for _ in range(10):
            cand = min(val + step, cap)
            good = ok(cand, other) if grow_width else ok(other, cand)
            if good and cand > val:
                val = cand
            else:
                break
        return val

    w = min(max(w0, 1e-6), w_max)
    h = min(max(h0, 1e-6), h_max)

    prev_area = 0.0
    for _ in range(max_passes):
        w = grow_one_dim(w, w_max, h, grow_width=True)
        h = grow_one_dim(h, h_max, w, grow_width=False)

        area = w * h
        if area <= prev_area + max(1e-6, tol * tol):
            break
        prev_area = area

        if time_budget_s and (time.perf_counter() - t_start) > time_budget_s:
            break

    return w, h


def _largest_box_asymmetric_about_seed(
    prep_space,
    free_space,
    fs_bounds,
    center,
    theta,
    aL0=0.1,
    aR0=0.1,
    bD0=0.1,
    bU0=0.1,
    aL_cap=50.0,
    aR_cap=50.0,
    bD_cap=50.0,
    bU_cap=50.0,
    tol=1e-3,
    allow_touch=True,
    time_budget_s=None,
    t_start=None,
):
    """Grow asymmetric box about a seed point."""
    if t_start is None:
        t_start = time.perf_counter()
    fs_minx, fs_miny, fs_maxx, fs_maxy = fs_bounds

    def ok(aL, aR, bD, bU):
        poly = _rect_poly_halves(center, aL, aR, bD, bU, theta)
        minx, miny, maxx, maxy = poly.bounds
        if minx < fs_minx or miny < fs_miny or maxx > fs_maxx or maxy > fs_maxy:
            return False
        if prep_space.contains(poly):
            return True
        return allow_touch and free_space.covers(poly)

    def grow_half(val, cap, other_triplet, side):
        lo = max(1e-9, min(val, cap))
        for _ in range(48):
            cand = min(lo * 2.0, cap)
            if side == "aL":
                good = ok(cand, *other_triplet)
            elif side == "aR":
                good = ok(other_triplet[0], cand, other_triplet[1], other_triplet[2])
            elif side == "bD":
                good = ok(other_triplet[0], other_triplet[1], cand, other_triplet[2])
            else:
                good = ok(other_triplet[0], other_triplet[1], other_triplet[2], cand)
            if good and cand > lo + tol:
                lo = cand
            else:
                break
            if time_budget_s and (time.perf_counter() - t_start) > time_budget_s:
                return lo

        hi = cap
        for _ in range(48):
            if (hi - lo) <= tol:
                break
            mid = 0.5 * (lo + hi)
            if side == "aL":
                good = ok(mid, *other_triplet)
            elif side == "aR":
                good = ok(other_triplet[0], mid, other_triplet[1], other_triplet[2])
            elif side == "bD":
                good = ok(other_triplet[0], other_triplet[1], mid, other_triplet[2])
            else:
                good = ok(other_triplet[0], other_triplet[1], other_triplet[2], mid)
            if good:
                lo = mid
            else:
                hi = mid
            if time_budget_s and (time.perf_counter() - t_start) > time_budget_s:
                break
        return lo

    aL, aR, bD, bU = aL0, aR0, bD0, bU0
    prev_area = -1.0
    for _ in range(8):
        aL = grow_half(aL, aL_cap, (aR, bD, bU), "aL")
        aR = grow_half(aR, aR_cap, (aL, bD, bU), "aR")
        bD = grow_half(bD, bD_cap, (aL, aR, bU), "bD")
        bU = grow_half(bU, bU_cap, (aL, aR, bD), "bU")
        area = (aL + aR) * (bD + bU)
        if area <= prev_area + max(1e-6, tol * tol):
            break
        prev_area = area
        if time_budget_s and (time.perf_counter() - t_start) > time_budget_s:
            break

    poly = _rect_poly_halves(center, aL, aR, bD, bU, theta)
    return poly, (aL + aR) * (bD + bU)


def _largest_box_asymmetric_about_seed_wh(
    prep_space,
    free_space,
    fs_bounds,
    center,
    theta,
    aL0=0.1,
    aR0=0.1,
    bD0=0.1,
    bU0=0.1,
    aL_cap=50.0,
    aR_cap=50.0,
    bD_cap=50.0,
    bU_cap=50.0,
    tol=1e-3,
    allow_touch=True,
    time_budget_s=None,
    t_start=None,
):
    """Grow asymmetric box about a seed point, returning width and height."""
    if t_start is None:
        t_start = time.perf_counter()
    fs_minx, fs_miny, fs_maxx, fs_maxy = fs_bounds

    def ok(aL, aR, bD, bU):
        poly = _rect_poly_halves(center, aL, aR, bD, bU, theta)
        minx, miny, maxx, maxy = poly.bounds
        if minx < fs_minx or miny < fs_miny or maxx > fs_maxx or maxy > fs_maxy:
            return False
        if prep_space.contains(poly):
            return True
        return allow_touch and free_space.covers(poly)

    def grow_half(val, cap, other_triplet, side):
        lo = max(1e-9, min(val, cap))
        for _ in range(48):
            cand = min(lo * 2.0, cap)
            if side == "aL":
                good = ok(cand, *other_triplet)
            elif side == "aR":
                good = ok(other_triplet[0], cand, other_triplet[1], other_triplet[2])
            elif side == "bD":
                good = ok(other_triplet[0], other_triplet[1], cand, other_triplet[2])
            else:
                good = ok(other_triplet[0], other_triplet[1], other_triplet[2], cand)
            if good and cand > lo + tol:
                lo = cand
            else:
                break
            if time_budget_s and (time.perf_counter() - t_start) > time_budget_s:
                return lo

        hi = cap
        for _ in range(48):
            if (hi - lo) <= tol:
                break
            mid = 0.5 * (lo + hi)
            if side == "aL":
                good = ok(mid, *other_triplet)
            elif side == "aR":
                good = ok(other_triplet[0], mid, other_triplet[1], other_triplet[2])
            elif side == "bD":
                good = ok(other_triplet[0], other_triplet[1], mid, other_triplet[2])
            else:
                good = ok(other_triplet[0], other_triplet[1], other_triplet[2], mid)
            if good:
                lo = mid
            else:
                hi = mid
            if time_budget_s and (time.perf_counter() - t_start) > time_budget_s:
                break
        return lo

    aL, aR, bD, bU = aL0, aR0, bD0, bU0
    prev_area = -1.0
    for _ in range(8):
        aL = grow_half(aL, aL_cap, (aR, bD, bU), "aL")
        aR = grow_half(aR, aR_cap, (aL, bD, bU), "aR")
        bD = grow_half(bD, bD_cap, (aL, aR, bU), "bD")
        bU = grow_half(bU, bU_cap, (aL, aR, bD), "bU")
        area = (aL + aR) * (bD + bU)
        if area <= prev_area + max(1e-6, tol * tol):
            break
        prev_area = area
        if time_budget_s and (time.perf_counter() - t_start) > time_budget_s:
            break

    poly = _rect_poly_halves(center, aL, aR, bD, bU, theta)
    return poly, (aL + aR), (bD + bU)


def largest_empty_rectangle_fast_approx(
    room_polygon: ShpPolygon,
    obstacle_polys: List[ShpPolygon],
    *,
    n_starts: int = 200,
    n_angles: int = 24,
    start_size: float = 0.1,
    max_expand: float = 50.0,
    time_budget_s: Optional[float] = 1.5,
    rng_seed: int = 42,
) -> Tuple[ShpPolygon, float, float]:
    """Fast approximate largest empty rectangle search."""
    rng = random.Random(rng_seed)

    free_space = room_polygon
    if obstacle_polys:
        merged = unary_union(obstacle_polys)
        inter = room_polygon.intersection(merged)
        if not inter.is_empty:
            free_space = room_polygon.difference(merged)

    if free_space.is_empty:
        return ShpPolygon(), 0.0, 0.0

    prep_space = prep(free_space)

    angles = [i * math.pi / n_angles for i in range(n_angles)]
    fs_bounds = free_space.bounds
    minx, miny, maxx, maxy = fs_bounds
    w_cap = min(max_expand, maxx - minx)
    h_cap = min(max_expand, maxy - miny)

    best_poly = None
    best_area = 0.0
    best_theta = 0.0

    t0 = time.perf_counter()

    def over_budget():
        return time_budget_s and (time.perf_counter() - t0) > time_budget_s

    def sample_point(max_tries=400):
        for _ in range(max_tries):
            x = rng.uniform(minx, maxx)
            y = rng.uniform(miny, maxy)
            p = Point(x, y)
            if prep_space.contains(p):
                return (x, y)
        rp = free_space.representative_point()
        return (float(rp.x), float(rp.y))

    for _ in range(n_starts):
        if over_budget():
            break
        c = sample_point()
        w0 = start_size * (0.5 + rng.random())
        h0 = start_size * (0.5 + rng.random())

        angs = angles[:]
        rng.shuffle(angs)
        angs = angs[:max(8, n_angles // 2)]

        for theta in angs:
            if over_budget():
                break
            poly, area = _largest_box_asymmetric_about_seed(
                prep_space=prep_space,
                free_space=free_space,
                fs_bounds=fs_bounds,
                center=c,
                theta=theta,
                aL0=start_size * 0.5,
                aR0=start_size * 1.5,
                bD0=start_size * 0.5,
                bU0=start_size * 1.5,
                aL_cap=w_cap,
                aR_cap=w_cap,
                bD_cap=h_cap,
                bU_cap=h_cap,
                tol=1e-3,
                allow_touch=True,
                time_budget_s=(time_budget_s * 0.9 if time_budget_s else None),
                t_start=t0,
            )
            if area > best_area:
                best_area, best_theta, best_poly = area, theta, poly

    if best_poly is None:
        return ShpPolygon(), 0.0, 0.0
    return best_poly, best_area, best_theta


def render_visibility_path(
    data: dict,
    room_polygon,
    obstacles,
    path_xy,
    start_pt,
    goal_pt,
    out_path: str,
    *,
    title: Optional[str] = None,
    show_grid: bool = True,
    show_axes: bool = True,
    dpi: int = 150,
    objects: Optional[list] = None,
    start_label: Optional[str] = None,
    goal_label: Optional[str] = None,
    vis_graph: Optional[dict] = None,
) -> str:
    """Render visibility graph and shortest path on room layout."""
    Path(out_path).parent.mkdir(parents=True, exist_ok=True)

    COL_ROOM_OUTLINE = "#444444"
    COL_WALLS = "#000000"
    COL_WINDOW = "#1f77b4"
    COL_DOOR = "#22aa55"
    COL_OBJ_FILL = "#b0b0b0"
    COL_OBJ_EDGE = "#7e7e7e"
    COL_START_POLY = "#2ecc71"
    COL_GOAL_POLY = "#1f77b4"
    COL_PATH = "#e74c3c"
    COL_START = "#2ecc71"
    COL_GOAL = "#1f77b4"
    COL_GRAPH_EDGE = "#8aa4c0"
    COL_GRAPH_NODE = "#6787a7"

    fig, ax = plt.subplots(figsize=(8, 8), dpi=dpi)

    # Room boundary
    rb = data.get("room_boundary") or []
    if rb:
        rb_xy = np.array([[p["x"], p["y"]] for p in rb], dtype=float)
        ax.plot(
            rb_xy[:, 0], rb_xy[:, 1], "--",
            color=COL_ROOM_OUTLINE, linewidth=1.6, alpha=0.9, label="Room boundary"
        )

    # Walls
    wall_segs = []
    for w in data.get("walls") or (data.get("room", {}) or {}).get("walls") or []:
        s, e = w.get("start"), w.get("end")
        if s and e:
            wall_segs.append([
                (float(s["x"]), float(s["y"])),
                (float(e["x"]), float(e["y"]))
            ])
    if wall_segs:
        ax.add_collection(LineCollection(
            wall_segs, linewidths=2.5, colors=[COL_WALLS], label="Walls"
        ))

    # Openings
    openings = data.get("openings") or {}
    for ent in openings.get("windows") or []:
        pts = ent.get("points") or []
        if len(pts) >= 2:
            arr = np.array([[p["x"], p["y"]] for p in pts], dtype=float)
            ax.plot(
                arr[:, 0], arr[:, 1], "-",
                color=COL_WINDOW, linewidth=2.0, alpha=0.9, label="Window"
            )
    for ent in openings.get("doors") or []:
        pts = ent.get("points") or []
        if len(pts) >= 2:
            arr = np.array([[p["x"], p["y"]] for p in pts], dtype=float)
            ax.plot(
                arr[:, 0], arr[:, 1], "-",
                color=COL_DOOR, linewidth=2.0, alpha=0.9, label="Door"
            )

    # Object polygons
    objs = objects or (data.get("objects") or [])
    for o in objs:
        pts = o.get("points") or []
        if len(pts) >= 3:
            xy = np.array([[p["x"], p["y"]] for p in pts], dtype=float)
            ax.fill(
                xy[:, 0], xy[:, 1],
                facecolor=COL_OBJ_FILL, edgecolor=COL_OBJ_EDGE,
                alpha=0.35, linewidth=1.0
            )

    # Highlight start/goal polygons
    def _match(objs, name):
        name = (name or "").lower()
        out = []
        for o in objs:
            lab = (o.get("label") or o.get("name") or "").strip().lower()
            if lab == name and len(o.get("points") or []) >= 3:
                arr = np.array([[p["x"], p["y"]] for p in o["points"]], dtype=float)
                out.append(arr)
        return out

    for arr in _match(objs, start_label):
        ax.fill(
            arr[:, 0], arr[:, 1],
            facecolor=COL_START_POLY, edgecolor=COL_START_POLY,
            alpha=0.30, linewidth=1.4, label="Start object"
        )
    for arr in _match(objs, goal_label):
        ax.fill(
            arr[:, 0], arr[:, 1],
            facecolor=COL_GOAL_POLY, edgecolor=COL_GOAL_POLY,
            alpha=0.30, linewidth=1.4, label="Target object"
        )

    # Visibility graph
    if vis_graph:
        edges = []
        nx_pts, ny_pts = [], []
        for a, nbrs in vis_graph.items():
            ax_, ay_ = a
            nx_pts.append(ax_)
            ny_pts.append(ay_)
            for b in nbrs.keys():
                edges.append([(ax_, ay_), (b[0], b[1])])
        if edges:
            ax.add_collection(LineCollection(
                edges, linewidths=1.0, colors=[COL_GRAPH_EDGE],
                alpha=0.35, zorder=1, label="Visibility graph"
            ))
        if nx_pts:
            ax.scatter(nx_pts, ny_pts, s=8, c=COL_GRAPH_NODE, alpha=0.7, zorder=2)

    # Path
    if path_xy:
        xs = [p[0] for p in path_xy]
        ys = [p[1] for p in path_xy]
        ax.plot(xs, ys, "-", color=COL_PATH, linewidth=2.8, zorder=4, label="Path")

    # Start/goal markers
    if start_pt:
        ax.plot(
            start_pt[0], start_pt[1], "o",
            color=COL_START, markersize=6, zorder=5, label="Start"
        )
    if goal_pt:
        ax.plot(
            goal_pt[0], goal_pt[1], "*",
            color=COL_GOAL, markersize=10, zorder=5, label="Goal"
        )

    # Calculate bounds
    xs, ys = [], []
    if rb:
        xs += rb_xy[:, 0].tolist()
        ys += rb_xy[:, 1].tolist()
    for (x1, y1), (x2, y2) in wall_segs:
        xs += [x1, x2]
        ys += [y1, y2]
    for ent in openings.get("windows") or []:
        for p in ent.get("points") or []:
            xs.append(p["x"])
            ys.append(p["y"])
    for ent in openings.get("doors") or []:
        for p in ent.get("points") or []:
            xs.append(p["x"])
            ys.append(p["y"])
    for o in objs:
        for p in o.get("points") or []:
            xs.append(p["x"])
            ys.append(p["y"])
    if path_xy:
        xs += [p[0] for p in path_xy]
        ys += [p[1] for p in path_xy]
    if start_pt:
        xs.append(start_pt[0])
        ys.append(start_pt[1])
    if goal_pt:
        xs.append(goal_pt[0])
        ys.append(goal_pt[1])
    if (not xs) and room_polygon is not None:
        bx, by = room_polygon.exterior.xy
        xs += list(bx)
        ys += list(by)

    if xs:
        minx, maxx = min(xs), max(xs)
        miny, maxy = min(ys), max(ys)
        pad = 0.05 * max(maxx - minx, maxy - miny, 1e-6)
        ax.set_xlim(minx - pad, maxx + pad)
        ax.set_ylim(miny - pad, maxy + pad)
    ax.set_aspect("equal", adjustable="box")

    if show_axes:
        ax.set_xlabel("X (m)")
        ax.set_ylabel("Y (m)")
    if show_grid:
        ax.grid(True, which="both", linewidth=0.6, alpha=0.7)
    if title:
        ax.set_title(title, fontsize=11)

    # Deduplicate legend
    hds, lbls = ax.get_legend_handles_labels()
    seen = set()
    H, L = [], []
    for h, lbl in zip(hds, lbls):
        if lbl in seen:
            continue
        seen.add(lbl)
        H.append(h)
        L.append(lbl)
    if len(L) >= 2:
        ax.legend(H, L, loc="upper right", fontsize=8)

    plt.tight_layout(pad=0.2)
    plt.savefig(out_path, dpi=dpi, bbox_inches="tight", pad_inches=0.05)
    plt.close(fig)
    return out_path
