
import numpy as np

def check_pick_success(env, obj_name):
    in_gripper = env.obj_in_gripper(obj_name)
    gripper_closed = not env.gripper_is_open()
    return in_gripper and gripper_closed

def check_place_success(env, obj_name, target_name="tray"):
    # Object should not be in gripper
    if env.obj_in_gripper(obj_name) and not env.gripper_is_open():
        return False

    # Get tray bbox and calculate valid area
    tray_bbox = env.get_obj_bbox(target_name)
    center = (tray_bbox[0] + tray_bbox[1]) / 2
    half_size = (tray_bbox[1] - tray_bbox[0]) / 2 * 0.8

    tray_min = center - half_size
    tray_max = center + half_size

    # Get object bbox
    obj_bbox = env.get_obj_bbox(obj_name)

    # Check if object is in tray (xy and z)
    xy_ok = np.all(
        (tray_min[:2] - 0.04 < obj_bbox[0][:2]) &
        (obj_bbox[1][:2] < tray_max[:2] + 0.04)
    )
    z_ok = tray_bbox[0][2] - 0.02 < obj_bbox[0][2] < tray_bbox[1][2] + 0.02

    return xy_ok and z_ok

def check_pick_and_place_success(env, obj_name, target_name="tray"):
    return check_place_success(env, obj_name, target_name)

def check_lid_open(env, lid_name="box_lid"):
    # Get handle and body bbox
    handle_bbox = env.get_obj_bbox("box_handle")
    body_bbox = env.get_obj_bbox("box_body")

    if handle_bbox is None or body_bbox is None:
        return False

    # Handle should be well above the body when lid is open
    handle_bottom_z = handle_bbox[0][2]
    body_top_z = body_bbox[1][2]

    print(f"[CheckLidOpen] handle_bottom_z={handle_bottom_z:.3f}, body_top_z={body_top_z:.3f}")

    # Handle should be at least 0.1 above the body top when fully open
    return handle_bottom_z > body_top_z + 0.08
