{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "798c1be4",
   "metadata": {},
   "outputs": [],
   "source": "# Notebook cell\nimport json\nimport numpy as np\nimport matplotlib.pyplot as plt\nfrom shapely.ops import unary_union\nfrom shapely.geometry import LineString\nfrom pathfinder import (\n    find_path_visibility, _walls_to_room_polygon, _points_to_polygon, _get_label, _get_center, build_visibility_graph, \n)\n\nJSON_ROOT = \"data/real_data\"           # root/<room_type>/room_<layout_id>.json\nROOM_TYPE = \"living_rooms\"\nLAYOUT_ID = \"1\"                        # e.g., \"0\"\n# START = \"door\"                         # choose existing labels in this layout\n# GOAL  = \"sofa\"                         # choose existing labels\nCLEARANCE_CM = 5\n\npath_json = f\"{JSON_ROOT}/real_{LAYOUT_ID}.json\"\nwith open(path_json, \"r\") as f:\n    data = json.load(f)\n\nnp.random.seed(42)\n\nSTART, GOAL = np.random.choice(data[\"objects\"], size=2, replace=False)\nSTART = _get_label(START); GOAL = _get_label(GOAL)\n\npath, room, obs_polys, sp, gp = find_path_visibility(data, START, GOAL, clearance_cm=CLEARANCE_CM)\n\n# Plot\nfig, ax = plt.subplots(figsize=(7,6))\nax.set_aspect(\"equal\", adjustable=\"box\")\n\n# room\nrx, ry = room.exterior.xy\nax.plot(rx, ry, linewidth=2)\n\n# buffered obstacles for visualization\nbuf = unary_union([o.buffer(CLEARANCE_CM/100.0, join_style=2) for o in obs_polys]) if CLEARANCE_CM>0 else None\n# raw obstacles\nfor o in obs_polys:\n    x,y = o.exterior.xy\n    ax.fill(x, y, alpha=0.2, linewidth=1)\n\n# if buf and not buf.is_empty:\n#     bx, by = buf.exterior.xy\n#     ax.plot(bx, by, linestyle=\"--\", linewidth=1, alpha=0.7)\n\n# path\nif path:\n    xs = [p[0] for p in path]; ys = [p[1] for p in path]\n    ax.plot(xs, ys, marker=\"o\", linewidth=3)\n    ax.scatter([sp[0], gp[0]],[sp[1], gp[1]], s=80, marker=\"D\")\nelse:\n    ax.text(0.02, 0.98, \"No path found\", transform=ax.transAxes, va=\"top\", ha=\"left\", color=\"red\")\n\nax.set_title(f\"{ROOM_TYPE} — layout {LAYOUT_ID} — {START} → {GOAL} (clearance {CLEARANCE_CM} cm)\")\nplt.show()"
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "389f1d86",
   "metadata": {},
   "outputs": [],
   "source": [
    "from shapely.geometry import Polygon as ShpPolygon, Point, LineString\n",
    "\n",
    "def _object_polygon(obj):\n",
    "    pts = obj.get(\"points\", [])\n",
    "    if not pts or len(pts) < 3:\n",
    "        return None\n",
    "    coords = [(p[\"x\"], p[\"y\"]) for p in pts]\n",
    "    if coords[0] != coords[-1]:\n",
    "        coords.append(coords[0])\n",
    "    poly = ShpPolygon(coords)\n",
    "    return poly if (poly.is_valid and not poly.is_empty and poly.area > 0) else None\n",
    "\n",
    "def plot_visibility(\n",
    "    data: dict,\n",
    "    start_label: str,\n",
    "    goal_label: str,\n",
    "    clearance_cm: int = 15,\n",
    "    show_full_graph: bool = True\n",
    "):\n",
    "    # compute path with your function\n",
    "    path, room, obs_polys, sp, gp = find_path_visibility(\n",
    "        data, start_label, goal_label, clearance_cm=clearance_cm\n",
    "    )\n",
    "\n",
    "    # (re)build graph to extract nodes/edges for drawing\n",
    "    graph = build_visibility_graph(\n",
    "        room=room,\n",
    "        obstacles=obs_polys,\n",
    "        start_pt=sp,\n",
    "        goal_pt=gp,\n",
    "        clearance_m=clearance_cm/100.0\n",
    "    )\n",
    "    nodes = list(graph.keys())\n",
    "    # unique undirected edges\n",
    "    edges = {(min(a,b), max(a,b)) for a in nodes for b in graph[a].keys() if a != b}\n",
    "\n",
    "    # find start/goal objects and their polygons\n",
    "    start_obj = next((o for o in data.get(\"objects\", []) if _get_label(o) == start_label), None)\n",
    "    goal_obj  = next((o for o in data.get(\"objects\", []) if _get_label(o) == goal_label), None)\n",
    "    start_poly = _object_polygon(start_obj) if start_obj else None\n",
    "    goal_poly  = _object_polygon(goal_obj) if goal_obj else None\n",
    "\n",
    "    # plot\n",
    "    fig, ax = plt.subplots(figsize=(8, 7))\n",
    "    ax.set_aspect(\"equal\", adjustable=\"box\")\n",
    "\n",
    "    # room outline\n",
    "    rx, ry = room.exterior.xy\n",
    "    ax.plot(rx, ry, linewidth=2)\n",
    "\n",
    "    # obstacles (raw) and their clearance buffer (dashed)\n",
    "    if obs_polys:\n",
    "        for o in obs_polys:\n",
    "            x, y = o.exterior.xy\n",
    "            ax.fill(x, y, alpha=0.2, linewidth=1)\n",
    "        if clearance_cm > 0:\n",
    "            buf = unary_union([o.buffer(clearance_cm/100.0, join_style=2) for o in obs_polys])\n",
    "            # if not buf.is_empty:\n",
    "            #     bx, by = buf.exterior.xy\n",
    "            #     ax.plot(bx, by, linestyle=\"--\", linewidth=1)\n",
    "\n",
    "    # start & goal object polygons\n",
    "    if start_poly is not None:\n",
    "        sx, sy = start_poly.exterior.xy\n",
    "        ax.plot(sx, sy, linewidth=2)\n",
    "    if goal_poly is not None:\n",
    "        gx, gy = goal_poly.exterior.xy\n",
    "        ax.plot(gx, gy, linewidth=2)\n",
    "\n",
    "    # full visibility graph (toggle)\n",
    "    if show_full_graph:\n",
    "        for (a, b) in edges:\n",
    "            ax.plot([a[0], b[0]], [a[1], b[1]], linewidth=0.6, alpha=0.35)\n",
    "\n",
    "    # shortest path (if any)\n",
    "    if path:\n",
    "        xs = [p[0] for p in path]\n",
    "        ys = [p[1] for p in path]\n",
    "        ax.plot(xs, ys, linewidth=3)\n",
    "        ax.scatter([sp[0]], [sp[1]], s=80, marker=\"D\")  # start\n",
    "        ax.scatter([gp[0]], [gp[1]], s=80, marker=\"s\")  # goal\n",
    "    else:\n",
    "        ax.text(0.02, 0.98, \"No path found\", transform=ax.transAxes, va=\"top\", ha=\"left\")\n",
    "\n",
    "    ax.set_xlabel(\"x (m)\")\n",
    "    ax.set_ylabel(\"y (m)\")\n",
    "    ax.set_title(f\"{_get_label(start_obj)} → {_get_label(goal_obj)} | clearance {clearance_cm} cm | graph={'on' if show_full_graph else 'off'}\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fdb5ac5e",
   "metadata": {},
   "outputs": [],
   "source": "# --- quick test on a single file ---\nJSON_ROOT = \"data/real_data\"   # root/<room_type>/room_<layout_id>.json\nROOM_TYPE = \"living_rooms\"\nLAYOUT_ID = \"1\"\n# START     = \"door\"             # must exist in that layout\n# GOAL      = \"sofa\"             # must exist in that layout\nCLEARANCE = 1                 # set 0 to disable clearance\nSHOW_GRAPH = True              # toggle full network on/off\n\nwith open(f\"{JSON_ROOT}/real_{LAYOUT_ID}.json\", \"r\") as f:\n    layout = json.load(f)\n\nnp.random.seed(42)\n\nSTART, GOAL = np.random.choice(layout[\"objects\"], size=2, replace=False)\nSTART = _get_label(START); GOAL = _get_label(GOAL)\n\nplot_visibility(layout, START, GOAL, clearance_cm=CLEARANCE, show_full_graph=SHOW_GRAPH)"
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "plan_b",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.17"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}