"""
Central configuration file for grasp task (pick_up_cup).
All scripts (dataset generator, debug tools, camera adjustment) read from this file.

Grasping is parameterized by two parameters:
  1. Approach angle: Horizontal direction the gripper approaches from (degrees)
  2. Grasp height: Z position where gripper contacts the cup (meters above cup base)
"""
import numpy as np

# ============================================================================
# Cup/Task Configuration
# ============================================================================
# The pick_up_cup task spawns cups on a table
# Cup1 is the target (to be picked up), Cup2 is a distractor

# Variation index determines cup color (0-19 based on colors list in rlbench.const)
CUP_VARIATION = 0  # Red cup by default

# Fixed cup position for consistent dataset generation
# This position was found to work for all 4 approach angles (0, 90, 180, 270)
# Set to None to use RLBench's random cup spawning
FIXED_CUP_POSITION = np.array([0.25, -0.05, 0.82])  # [X, Y, Z] in world coords
# FIXED_CUP_POSITION = None  # Uncomment to use random positions

# ============================================================================
# Robot Configuration
# ============================================================================
# Panda home joint configuration - neutral position for starting trajectories
HOME_JOINTS = np.array([0, -0.3, 0, -2.0, 0, 1.8, 0.785])

# ============================================================================
# Grasp Mode Configuration
# ============================================================================
# Approach angles (degrees) - direction gripper approaches from in XY plane
# 0 = from +X direction, 90 = from +Y direction, etc.
APPROACH_ANGLES_DEG = [0, 90, 180, 270]  # 4 angles

# Grasp heights (meters above cup base) - where gripper contacts the cup
# Cup dimensions: 9.3 x 9.5 x 14.7 cm (width x depth x height)
# Panda gripper max opening is ~8cm - cup body is TOO WIDE to grasp
# RLBench default grasps at the rim (~12cm above base) where cup is narrower
# We use a single rim grasp height and vary only the approach angle
GRASP_HEIGHTS = [0.12]  # Single height at rim level (12cm above base)

# Total modes = 4 angles x 1 height = 4 (angle-only variation)
NUM_ANGLES = len(APPROACH_ANGLES_DEG)
NUM_HEIGHTS = len(GRASP_HEIGHTS)
NUM_MODES = NUM_ANGLES * NUM_HEIGHTS  # 4

# Demos per mode (with noise variations)
DEMOS_PER_MODE = 10

# ============================================================================
# Noise Parameters (for generating variations around each mode)
# ============================================================================
ANGLE_NOISE = 0.175     # radians (~10 degrees) - creates ±10° variation
HEIGHT_NOISE = 0.0      # meters - no height noise since we use single height

# ============================================================================
# Trajectory Configuration
# ============================================================================

# Phase step allocation (total: 108 steps)
PHASE_STEPS = {
    "reach": 64,           # Phase 1: shaped start -> pregrasp (SHAPED, 64 steps)
    "descend": 8,          # Phase 2: pregrasp -> grasp position
    "hold_grasp": 8,       # Phase 3: dwell at grasp (gripper open, settle)
    "close_gripper": 12,   # Phase 4: close gripper while holding position
    "lift": 16,            # Phase 5: lift object up
}

# Waypoint offsets for grasp trajectory
# NOTE: gripper_offset_from_cup is where the EE goes relative to cup center
# This should be ~4cm based on official RLBench demos (EE is offset from cup center)
WAYPOINT_OFFSETS = {
    "pregrasp_offset_z": 0.08,    # Height above grasp position for pre-grasp approach
    "approach_offset_xy": 0.10,   # Horizontal offset from grasp for pregrasp position
    "gripper_offset_from_cup": 0.04,  # EE offset from cup center in approach direction
    "lift_height": 0.15,          # How high to lift the cup after grasping
}

# ============================================================================
# Control Point Configuration (for shaped reach phase)
# ============================================================================
CONTROL_POINT_RADIUS = 0.08  # Control point sampling radius (meters)

# ============================================================================
# Camera Configuration
# ============================================================================
# Camera position and orientation for video recording
# Use RLBench's default front camera position but don't modify it
# The default camera is at [1.35, 0, 1.58] with orientation [-3.14, -1.13, 1.57]
# For better visibility, use a position that shows robot and table
CAMERA_POSITION = [1.0, 0.5, 1.4]  # Slightly to the side and elevated
CAMERA_ORIENTATION = [-2.9, -0.8, 2.0]  # Looking down at table from front-left

# Camera image size for video recording
CAMERA_IMAGE_SIZE = [256, 256]

# ============================================================================
# Grasp Orientation
# ============================================================================
# Base grasp orientation from RLBench waypoint0 for pick_up_cup task
# This is the orientation RLBench uses for successful grasps
# Euler angles in radians: [roll, pitch, yaw]
RLBENCH_GRASP_ORIENTATION = np.array([np.pi, 0.0, -2.91327])  # From waypoint0


def get_grasp_orientation(approach_angle):
    """
    Compute the gripper orientation for a given approach angle.

    For a top-down grasp, we rotate the gripper around Z axis based on approach angle.
    The base orientation comes from RLBench's default grasp pose.

    Args:
        approach_angle: float, approach direction in radians (0 = from +X)

    Returns:
        np.ndarray(3,): euler angles [roll, pitch, yaw]
    """
    # Start with RLBench's default grasp orientation
    # Then add the approach angle to rotate the gripper appropriately
    # The gripper needs to be rotated so fingers can close around the cup
    base_yaw = -2.91327  # From RLBench waypoint
    yaw = base_yaw + approach_angle

    return np.array([np.pi, 0.0, yaw])


# Default orientation for approach from +X (angle=0)
DEFAULT_GRASP_ORIENTATION = get_grasp_orientation(0.0)

# ============================================================================
# Success Criteria
# ============================================================================
# Grasp is successful if:
# 1. Object is grasped (gripper holding object)
# 2. Object is lifted above threshold
LIFT_SUCCESS_THRESHOLD = 0.05  # Object must rise at least 5cm from initial position

# ============================================================================
# Helper Functions
# ============================================================================

def get_approach_angles_radians():
    """Get approach angles in radians."""
    return [np.radians(a) for a in APPROACH_ANGLES_DEG]


def get_mode_params(mode_idx):
    """
    Get (approach_angle, grasp_height) for a given mode index.

    Modes are ordered as:
    mode 0: angle=0, height=0.02
    mode 1: angle=0, height=0.03
    ...
    mode 4: angle=90, height=0.02
    ...

    Args:
        mode_idx: int, mode index (0 to NUM_MODES-1)

    Returns:
        tuple: (approach_angle_rad, grasp_height_m)
    """
    angle_idx = mode_idx // NUM_HEIGHTS
    height_idx = mode_idx % NUM_HEIGHTS

    angle_rad = np.radians(APPROACH_ANGLES_DEG[angle_idx])
    height_m = GRASP_HEIGHTS[height_idx]

    return angle_rad, height_m


def generate_canonical_mode_params():
    """
    Generate all canonical (angle, height) parameter pairs.

    Returns:
        np.ndarray of shape (NUM_MODES, 2): [(angle_rad, height_m), ...]
    """
    params = []
    for angle_deg in APPROACH_ANGLES_DEG:
        for height in GRASP_HEIGHTS:
            params.append([np.radians(angle_deg), height])
    return np.array(params)


def print_config_summary():
    """Print a summary of the current configuration."""
    print("="*70)
    print("GRASP (PICK_UP_CUP) CONFIGURATION")
    print("="*70)
    print(f"\nGrasp Modes:")
    print(f"  Approach angles: {APPROACH_ANGLES_DEG} degrees ({NUM_ANGLES} angles)")
    print(f"  Grasp heights:   {GRASP_HEIGHTS} meters ({NUM_HEIGHTS} heights)")
    print(f"  Total modes:     {NUM_MODES}")
    print(f"  Demos per mode:  {DEMOS_PER_MODE}")
    print(f"  Total episodes:  {NUM_MODES * DEMOS_PER_MODE}")
    print(f"\nNoise Parameters:")
    print(f"  Angle noise:  {ANGLE_NOISE} rad ({np.degrees(ANGLE_NOISE):.1f} deg)")
    print(f"  Height noise: {HEIGHT_NOISE} m ({HEIGHT_NOISE*1000:.1f} mm)")
    print(f"\nTrajectory:")
    print(f"  Phase steps: {PHASE_STEPS}")
    print(f"  Total steps: {sum(PHASE_STEPS.values())}")
    print(f"  Waypoint offsets: {WAYPOINT_OFFSETS}")
    print(f"\nControl Points:")
    print(f"  Radius: {CONTROL_POINT_RADIUS}m")
    print(f"\nCamera:")
    print(f"  Position: {CAMERA_POSITION}")
    print(f"  Orientation: {CAMERA_ORIENTATION}")
    print("="*70)


if __name__ == "__main__":
    print_config_summary()
    print("\nCanonical mode parameters:")
    params = generate_canonical_mode_params()
    for i, (angle, height) in enumerate(params):
        print(f"  Mode {i:2d}: angle={np.degrees(angle):6.1f} deg, height={height:.3f} m")
