"""
Interactive camera position adjustment tool for pick_place (stack_blocks).

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 sys
import os
import numpy as np
import time

sys.path.insert(0, os.path.dirname(__file__))

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 pyrep.objects.shape import Shape

# HOME_JOINTS from config
HOME_JOINTS = np.array([0, 0, 0, -1.57, 0, 1.57, 0.785])

# Default camera position (will be updated interactively)
DEFAULT_CAMERA_POSITION = [1.349999189376831, -0.3999999463558197, 1.579999327659607]
DEFAULT_CAMERA_ORIENTATION = [-2.541592597961426, -1.0344651937484741, 2.1707979679107665]


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())

    # headless=False to allow GUI interaction
    rlbench_env = Environment(action_mode=action_mode, obs_config=obs_config, headless=False)
    rlbench_env.launch()

    task_class = task_file_to_task_class("stack_blocks")
    task_env = rlbench_env.get_task(task_class)
    task_env.set_variation(0)

    # Reset and setup scene (with fixed seed for reproducibility, matching data collection)
    np.random.seed(42)
    task_env.reset()

    # 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()

    # Load init data for fixed object position
    init_file = os.path.join(os.path.dirname(__file__), "stack_blocks_init.npz")
    object_pos = None
    if os.path.exists(init_file):
        init_data = dict(np.load(init_file))
        object_pos = init_data.get('object_pos')

        if object_pos is not None:
            target_block = Shape('stack_blocks_target0')
            target_block.set_position(object_pos)
            print(f"Loaded object position from init file: {object_pos}")

    # Hide other blocks and distractors
    for i in range(1, 4):
        try:
            other_block = Shape(f'stack_blocks_target{i}')
            other_block.set_position([10, 10, 0])
        except:
            pass
    for i in range(4):
        try:
            distractor = Shape(f'stack_blocks_distractor{i}')
            distractor.set_position([10, 10 + i*0.1, 0])
        except:
            pass

    # Step physics
    for _ in range(10):
        task_env._scene.pyrep.step()

    # Get camera
    camera = task_env._scene._cam_front

    # Only set custom position if specified, otherwise use RLBench default
    if DEFAULT_CAMERA_POSITION is not None:
        camera.set_position(DEFAULT_CAMERA_POSITION)
    if DEFAULT_CAMERA_ORIENTATION is not None:
        camera.set_orientation(DEFAULT_CAMERA_ORIENTATION)

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

    # Get initial position/orientation (from current camera, which is RLBench default)
    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
    tip = robot.arm.get_tip()
    gripper_pos = tip.get_position()

    # Get block position
    try:
        target_block = Shape('stack_blocks_target0')
        block_pos = target_block.get_position()
    except:
        block_pos = [0, 0, 0]

    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"  Block:         X={block_pos[0]:.2f}, Y={block_pos[1]:.2f}, Z={block_pos[2]:.2f}")
    if object_pos is not None:
        print(f"  Object (init): X={object_pos[0]:.2f}, Y={object_pos[1]:.2f}, Z={object_pos[2]:.2f}")
    print("")
    print("  Expected view: robot arm and block on table visible")
    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 your config file or script:")
            print(f"CAMERA_POSITION = [{pos[0]}, {pos[1]}, {pos[2]}]")
            print(f"CAMERA_ORIENTATION = [{ori[0]}, {ori[1]}, {ori[2]}]")
            print(f"\nOr for direct use:")
            print(f"front_cam.set_position([{pos[0]}, {pos[1]}, {pos[2]}])")
            print(f"front_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 your config/script:")
    print(f"CAMERA_POSITION = [{pos[0]}, {pos[1]}, {pos[2]}]")
    print(f"CAMERA_ORIENTATION = [{ori[0]}, {ori[1]}, {ori[2]}]")


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