"""
Interactive camera position adjustment tool.

TWO MODES:
1. GUI Mode (default): Drag the camera in CoppeliaSim GUI, script prints position
2. Keyboard Mode: Use keyboard commands to adjust camera position

In CoppeliaSim GUI:
- Click on "Vision_sensor_front" in the scene hierarchy
- Use the object manipulation tools (translate/rotate) to position it
- The script will print the current position in real-time
"""
import numpy as np
import time

from rlbench import ObservationConfig
from rlbench.action_modes.action_mode import MoveArmThenGripper
from rlbench.action_modes.arm_action_modes import JointPosition
from rlbench.action_modes.gripper_action_modes import Discrete
from rlbench.backend.utils import task_file_to_task_class
from rlbench.environment import Environment

from push_utils import (
    fix_cabinet_orientation,
    set_drawer_open,
    get_drawer_handle_position,
)

from close_drawer_config import (
    HOME_JOINTS,
    DRAWER_VARIATION,
    DRAWER_OPEN_AMOUNT,
    CAMERA_POSITION,
    CAMERA_ORIENTATION,
    CABINET_POSITION,
)


def print_controls():
    print("\n" + "="*60)
    print("CAMERA ADJUSTMENT MODES")
    print("="*60)
    print("MODE 1: GUI Mode (Recommended)")
    print("  - In CoppeliaSim, click 'Vision_sensor_front' in scene hierarchy")
    print("  - Use the translate/rotate tools to position the camera")
    print("  - This script will continuously print the current position")
    print("  - Press Enter to see updated position, 'x' + Enter to exit")
    print()
    print("MODE 2: Keyboard Mode")
    print("  Position:")
    print("    q/a: Move camera LEFT / RIGHT")
    print("    w/s: Move camera FORWARD / BACKWARD")
    print("    e/d: Move camera UP / DOWN")
    print()
    print("  Camera View Direction (note: use GUI mode for easier control):")
    print("    i/k: Look UP / DOWN")
    print("    j/l: Look LEFT / RIGHT")
    print("    u/o: Tilt camera LEFT / RIGHT")
    print()
    print("  TIP: For orientation, it's easier to use GUI Mode:")
    print("       Click 'Vision_sensor_front' → Use rotation tool")
    print()
    print("  Other:")
    print("    p or Enter: Print current position and orientation")
    print("    v: Save current view as image")
    print("    x: Exit")
    print("="*60 + "\n")


def main():
    # Setup environment
    obs_config = ObservationConfig()
    obs_config.set_all(False)
    obs_config.joint_positions = True
    obs_config.gripper_open = True
    obs_config.front_camera.rgb = True
    obs_config.front_camera.image_size = [512, 512]  # Larger for better viewing

    action_mode = MoveArmThenGripper(JointPosition(True), Discrete())

    rlbench_env = Environment(action_mode=action_mode, obs_config=obs_config, headless=False)
    rlbench_env.launch()

    task_class = task_file_to_task_class("close_drawer")
    task_env = rlbench_env.get_task(task_class)
    task_env.set_variation(DRAWER_VARIATION)

    # Reset and setup scene
    task_env.reset()
    fix_cabinet_orientation(task_env)
    set_drawer_open(task_env, DRAWER_VARIATION, DRAWER_OPEN_AMOUNT)

    # Move robot to home
    robot = task_env._scene.robot
    robot.arm.set_joint_positions(HOME_JOINTS, disable_dynamics=True)
    for _ in range(20):
        task_env._scene.pyrep.step()

    # Get camera and set to same initial position as dataset generator
    camera = task_env._scene._cam_front

    # Set camera to dataset generator's starting position (from config file)
    camera.set_position(CAMERA_POSITION)
    camera.set_orientation(CAMERA_ORIENTATION)

    # Step simulation to update camera view
    for _ in range(5):
        task_env._scene.pyrep.step()

    # Get initial position/orientation
    pos = list(camera.get_position())
    ori = list(camera.get_orientation())

    print_controls()
    print(f"Initial camera position: {pos}")
    print(f"Initial camera orientation: {ori}")

    # Get actual positions after setup
    robot = task_env._scene.robot
    tip = robot.arm.get_tip()
    gripper_pos = tip.get_position()

    handle_pos, _ = get_drawer_handle_position(task_env, DRAWER_VARIATION)

    print("\n" + "="*60)
    print("SCENE REFERENCE (actual positions):")
    print(f"  Robot gripper: X={gripper_pos[0]:.2f}, Y={gripper_pos[1]:.2f}, Z={gripper_pos[2]:.2f}")
    print(f"  Cabinet:       X={CABINET_POSITION[0]:.2f}, Y={CABINET_POSITION[1]:.2f}, Z={CABINET_POSITION[2]:.2f}")
    print(f"  Handle (open): X={handle_pos[0]:.2f}, Y={handle_pos[1]:.2f}, Z={handle_pos[2]:.2f}")
    print("")
    print("  Expected view: robot on LEFT, cabinet on RIGHT")
    print("="*60)

    print("\nGUI MODE: Drag 'Vision_sensor_front' in CoppeliaSim")
    print("          Press Enter to refresh position, or type command\n")

    save_count = 0
    last_pos = None
    last_ori = None

    while True:
        # Get current frame and camera position (in case user moved it in GUI)
        task_env._scene.pyrep.step()
        obs = task_env._scene.get_observation()

        # Read current camera position (in case it was moved in GUI)
        current_pos = list(camera.get_position())
        current_ori = list(camera.get_orientation())

        # Check if camera was moved in GUI
        if last_pos is not None:
            if (abs(current_pos[0] - last_pos[0]) > 0.001 or
                abs(current_pos[1] - last_pos[1]) > 0.001 or
                abs(current_pos[2] - last_pos[2]) > 0.001):
                # Camera was moved in GUI!
                pos = current_pos
                ori = current_ori
                print(f"\n[GUI Update] Camera moved!")

        last_pos = current_pos[:]
        last_ori = current_ori[:]

        # Show current settings
        print(f"\rPos: [{pos[0]:6.2f}, {pos[1]:6.2f}, {pos[2]:6.2f}]  "
              f"Ori: [{ori[0]:5.2f}, {ori[1]:5.2f}, {ori[2]:5.2f}]  ", end="", flush=True)

        # Get command
        cmd = input(">> ").strip().lower()

        if cmd == '' or cmd == 'p':
            # Just pressed Enter or 'p' - print current position
            print(f"\n\nCurrent camera position: {pos}")
            print(f"Current camera orientation: {ori}")
            print(f"\n{'='*60}")
            print("Copy this to dataset_generator_fixed_endpoints.py:")
            print(f"custom_cam.set_position([{pos[0]}, {pos[1]}, {pos[2]}])")
            print(f"custom_cam.set_orientation([{ori[0]}, {ori[1]}, {ori[2]}])")
            print(f"{'='*60}\n")
        elif cmd == 'x':
            break
        elif cmd == 'v':
            # Save current view
            import imageio
            filename = f"camera_view_{save_count}.png"
            imageio.imwrite(filename, obs.front_rgb)
            print(f"\nSaved view to {filename}\n")
            save_count += 1
        else:
            # Adjust position or orientation
            changed = True

            # Position adjustments
            if cmd == 'q':
                pos[0] -= 0.1
                print("\n→ Moved camera LEFT")
            elif cmd == 'a':
                pos[0] += 0.1
                print("\n→ Moved camera RIGHT")
            elif cmd == 'w':
                pos[1] -= 0.1
                print("\n→ Moved camera FORWARD")
            elif cmd == 's':
                pos[1] += 0.1
                print("\n→ Moved camera BACKWARD")
            elif cmd == 'e':
                pos[2] += 0.1
                print("\n→ Moved camera UP")
            elif cmd == 'd':
                pos[2] -= 0.1
                print("\n→ Moved camera DOWN")

            # Camera view direction adjustments
            # Note: Euler angles in CoppeliaSim are [alpha, beta, gamma] (x, y, z rotations)
            # For intuitive camera control, we adjust based on typical camera orientation
            elif cmd == 'i':
                ori[1] += 0.1  # Look up (pitch up)
                print("\n→ Looking UP")
            elif cmd == 'k':
                ori[1] -= 0.1  # Look down (pitch down)
                print("\n→ Looking DOWN")
            elif cmd == 'j':
                ori[2] -= 0.1  # Look left (yaw left)
                print("\n→ Looking LEFT")
            elif cmd == 'l':
                ori[2] += 0.1  # Look right (yaw right)
                print("\n→ Looking RIGHT")
            elif cmd == 'u':
                ori[0] -= 0.1  # Tilt camera left (roll)
                print("\n→ Tilted camera LEFT")
            elif cmd == 'o':
                ori[0] += 0.1  # Tilt camera right (roll)
                print("\n→ Tilted camera RIGHT")
            else:
                print("\nUnknown command. Press 'p' for help.")
                changed = False

            if changed:
                # Apply new position/orientation
                camera.set_position(pos)
                camera.set_orientation(ori)

    rlbench_env.shutdown()
    print("\nFinal camera settings:")
    print(f"Position: {pos}")
    print(f"Orientation: {ori}")
    print(f"\nPaste this into dataset_generator_fixed_endpoints.py:")
    print(f"custom_cam.set_position([{pos[0]}, {pos[1]}, {pos[2]}])")
    print(f"custom_cam.set_orientation([{ori[0]}, {ori[1]}, {ori[2]}])")


if __name__ == "__main__":
    import multiprocessing as mp
    mp.set_start_method("spawn", force=True)
    main()
