
from typing import Any, Dict, Optional

def get_scene_info(env, target_robot: str = "gripper") -> Dict[str, Any]:
    try:
        scene_info: Dict[str, Any] = {}

        # =================================================================
        # Gripper State (critical for validation)
        # =================================================================
        gripper_info = _extract_gripper_info(env)
        scene_info["gripper"] = gripper_info

        # =================================================================
        # Robot/End-effector type
        # =================================================================
        scene_info["robot_type"] = getattr(env, 'ee_type', target_robot)

        # =================================================================
        # Object States
        # =================================================================
        gripper_pos = gripper_info.get("position")
        objects_info = _extract_objects_info(env, gripper_info.get("held_object"), gripper_pos)
        scene_info["objects"] = objects_info

        return scene_info

    except Exception as e:
        print(f"[WARN] get_scene_info failed: {e}")
        return {"objects": [], "gripper": {}}

def _extract_gripper_info(env) -> Dict[str, Any]:
    gripper_info: Dict[str, Any] = {
        "position": None,
        "quaternion": None,
        "is_open": True,
        "held_object": None,
    }

    try:
        # Get gripper position from the gripper object
        if "gripper" in env.env.scene_objects:
            gripper_obj = env.env.scene_objects["gripper"]

            # Position: get from gripper's link or entity
            gripper_info["position"] = _get_gripper_position(env, gripper_obj)

            # Quaternion: get orientation
            gripper_info["quaternion"] = _get_gripper_quaternion(gripper_obj)

            # Is open: check gripper state
            gripper_info["is_open"] = _get_gripper_open_state(env, gripper_obj)

        # Check if holding an object
        gripper_info["held_object"] = _get_held_object(env)

    except Exception as e:
        print(f"[WARN] Failed to get gripper info: {e}")

    return gripper_info

def _get_gripper_position(env, gripper_obj) -> Optional[list]:
    try:
        # Try getting position from the end-effector link
        ee_link = gripper_obj.get_link(env.ee_name) if hasattr(gripper_obj, 'get_link') else None
        if ee_link is not None:
            pos = ee_link.get_pos()
            return pos.cpu().numpy().tolist() if hasattr(pos, 'cpu') else pos.tolist()
        else:
            # Fallback: use gripper entity position
            pos = gripper_obj.get_pos() if hasattr(gripper_obj, 'get_pos') else None
            if pos is not None:
                return pos.cpu().numpy().tolist() if hasattr(pos, 'cpu') else pos.tolist()
    except Exception:
        pass
    return None

def _get_gripper_quaternion(gripper_obj) -> Optional[list]:
    try:
        quat = gripper_obj.get_quat() if hasattr(gripper_obj, 'get_quat') else None
        if quat is not None:
            return quat.cpu().numpy().tolist() if hasattr(quat, 'cpu') else quat.tolist()
    except Exception:
        pass
    return None

def _get_gripper_open_state(env, gripper_obj) -> bool:
    try:
        if hasattr(gripper_obj, 'gripper_open'):
            return bool(gripper_obj.gripper_open)
        elif hasattr(env, 'gripper_is_open'):
            return bool(env.gripper_is_open())
    except Exception:
        pass
    return True

def _get_held_object(env) -> Optional[str]:
    try:
        if hasattr(env, '_grasp') and env._grasp.get("active"):
            return env._grasp.get("object")
        elif hasattr(env, '_welded') and env._welded.get("active"):
            return env._welded.get("object")
    except Exception:
        pass
    return None

# Height threshold for surface observation (60cm above object top)
SURFACE_OBSERVATION_HEIGHT = 0.60

def _update_visited_surface(
    env,
    obj_name: str,
    gripper_pos: Optional[list],
    bbox: Optional[list],
) -> None:
    if gripper_pos is None or bbox is None:
        return

    # Already visited
    if obj_name in env._visited_surfaces:
        return

    try:
        gx, gy, gz = gripper_pos
        bbox_min, bbox_max = bbox[0], bbox[1]

        # Check if gripper is within XY bounds of object
        in_xy_range = (bbox_min[0] <= gx <= bbox_max[0] and
                       bbox_min[1] <= gy <= bbox_max[1])

        # Check if gripper is above object but within observation height
        obj_top_z = bbox_max[2]
        above_surface = obj_top_z <= gz <= (obj_top_z + SURFACE_OBSERVATION_HEIGHT)

        if in_xy_range and above_surface:
            env._visited_surfaces.add(obj_name)
            # Log surface observation
            obj = env.env.scene_objects.get(obj_name)
            if obj is not None:
                surface_info = getattr(obj, 'surface_info', None)
                if surface_info:
                    edges_info = surface_info.get("edges", {})
                    center_info = surface_info.get("center", {})
                    note = surface_info.get("note", "")
                    print(f"[INFO] Surface property observed for '{obj_name}':")
                    print(f"  - Center: friction={center_info.get('friction', 'N/A')}, slippery={center_info.get('slippery', False)}")
                    print(f"  - Edges: friction={edges_info.get('friction', 'N/A')}, slippery={edges_info.get('slippery', False)}")
                    if note:
                        print(f"  - Note: {note}")
    except (IndexError, TypeError):
        pass

def _extract_objects_info(
    env,
    held_obj_name: Optional[str],
    gripper_pos: Optional[list],
) -> list:
    # Initialize visited surfaces set on env if not exists
    if not hasattr(env, '_visited_surfaces'):
        env._visited_surfaces = set()

    objects_info = []

    for obj_name in env.env.scene_objects:
        if obj_name == "gripper":
            continue

        obj_info = _extract_single_object_info(env, obj_name, held_obj_name, gripper_pos)
        objects_info.append(obj_info)

    return objects_info

def _extract_single_object_info(
    env,
    obj_name: str,
    held_obj_name: Optional[str],
    gripper_pos: Optional[list],
) -> Dict[str, Any]:
    try:
        obj = env.env.scene_objects[obj_name]
        pos = env.get_obj_pos(obj_name)
        bbox = env.get_obj_bbox(obj_name)

        # Get quaternion if available
        quat = None
        try:
            quat_tensor = obj.get_quat() if hasattr(obj, 'get_quat') else None
            if quat_tensor is not None:
                quat = quat_tensor.cpu().numpy().tolist() if hasattr(quat_tensor, 'cpu') else quat_tensor.tolist()
        except Exception:
            pass

        # Get surface_info if available (e.g., inverse tray with slippery edges)
        raw_surface_info = getattr(obj, 'surface_info', None)

        # Check if gripper is above this object (within 30cm above surface)
        # If so, mark as visited and expose surface_info
        bbox_list = bbox.tolist() if bbox is not None else None
        _update_visited_surface(env, obj_name, gripper_pos, bbox_list)

        # Only expose surface_info if gripper has visited above this object
        surface_info = None
        if obj_name in env._visited_surfaces and raw_surface_info is not None:
            surface_info = raw_surface_info

        return {
            "name": obj_name,
            "position": pos.tolist() if pos is not None else None,
            "quaternion": quat,
            "bbox": bbox_list,
            "visible": env.is_obj_visible(obj_name),
            "is_grasped": obj_name == held_obj_name,
            "grasped_by": "gripper" if obj_name == held_obj_name else None,
            "surface_info": surface_info,
        }
    except Exception:
        return {
            "name": obj_name,
            "position": None,
            "quaternion": None,
            "bbox": None,
            "visible": False,
            "is_grasped": False,
            "grasped_by": None,
            "surface_info": None,
        }
