const socket = io();
const cameraGrid = document.getElementById('cameraGrid');
const cameraTemplate = document.getElementById('camera-template');
const simulationElements = {};
const mouseState = {};
const MOUSE_SENSITIVITY = 0.01;

// Detailed view elements
const detailedViewContainer = document.getElementById('detailedViewContainer');
const closeDetailedViewBtn = document.getElementById('closeDetailedViewBtn');
const mainDetailedCanvas = document.getElementById('mainDetailedCanvas');
const mainCameraTitle = document.getElementById('mainCameraTitle');
const robotFeedsContainer = document.getElementById('robotFeedsContainer');
const mainDetailedInfo = document.getElementById('mainDetailedInfo');
const mainDetailedTime = document.getElementById('mainDetailedTime');
const robotCameraTemplate = document.getElementById('robot-camera-template');
const detailedRobotSelect = document.querySelector('.detailed-robot-select');
const detailedTrackingToggle = document.querySelector('.detailed-tracking-toggle');
const detailedRecordBtn = document.querySelector('.detailed-record-btn');

// Track which simulation is currently active/selected
let activeSimulation = null;
// Track which simulation is shown in detailed view
let detailedViewSimId = null;
// Store information about robot cameras
const robotCameras = {};
// Track which keys are currently pressed
const keyState = {
    w: false,
    a: false,
    s: false,
    d: false,
    q: false,
    e: false
};

// Recording state management
const recorderState = {};

// Initialize socket connection and event handlers
socket.on('viewer_connect', () => {
    console.log('Connected to server');
});

socket.on('viewer_disconnect', () => {
    console.log('Disconnected from server');
});

socket.on('viewer_simulation_update', (data) => {
    console.log('Received simulation update:', data);
    updateSimulation(data);

    // Also update robot list if available
    if (data.robots && Array.isArray(data.robots)) {
        console.log('Updating robot list from simulation_update:', data.robots);
        updateRobotList(data.sim_id, data.robots);
    }
});

socket.on('viewer_main_camera_feed', (data) => {
    updateMainCameraFeed(data);
});

socket.on('viewer_robot_camera_feed', (data) => {
    updateRobotCameraFeed(data);
});

socket.on('viewer_robot_list', (data) => {
    console.log('Received robot list update:', data);
    updateRobotList(data.sim_id, data.robots);
});

socket.on('viewer_simulation_removed', (data) => {
    console.log('Simulation removed from server:', data.sim_id);
    removeSimulation(data.sim_id);

    // If detailed view was showing this simulation, close it
    if (detailedViewSimId === data.sim_id) {
        closeDetailedView();
    }
});

socket.on('title_update', (data) => {
    console.log('Received title update:', data);

    const pageTitle = document.getElementById('pageTitle');
    pageTitle.textContent = data.title;
});

// Function to create a new simulation display
function createSimulationElement(simId, simName) {
    const template = cameraTemplate.content.cloneNode(true);
    const cameraFeed = template.querySelector('.camera-feed');
    cameraFeed.dataset.simId = simId;

    const simTitle = cameraFeed.querySelector('.sim-name');
    simTitle.textContent = simName;

    const canvas = cameraFeed.querySelector('.camera-image');
    canvas.width = 1280;  // Default width
    canvas.height = 720; // Default height

    // Initialize recorder state for this simulation
    recorderState[simId] = {
        isRecording: false,
        recorder: null,
        chunks: [],
        startTime: 0,
        frames: [], // To store frames with timestamps
        lastFrameTime: 0
    };

    // Set up event listeners for control buttons
    setupControlButtons(cameraFeed, simId);

    // Set up recording button
    setupRecordingButton(cameraFeed, simId, canvas);

    // Set up expand button
    setupExpandButton(cameraFeed, simId);

    // Set up bird view button
    setupBirdViewButton(cameraFeed, simId);

    // Set up tracking toggle
    const trackingToggle = cameraFeed.querySelector('.tracking-toggle');
    trackingToggle.addEventListener('click', () => {
        const isTracking = trackingToggle.textContent === 'Tracking';
        if (isTracking) {
            trackingToggle.textContent = 'Free Cam';
            socket.emit('viewer_set_tracking', {
                sim_id: simId,
                tracking_robot: null
            });
        } else {
            const robotSelect = cameraFeed.querySelector('.robot-select');
            const selectedRobot = robotSelect.value;
            if (selectedRobot) {
                trackingToggle.textContent = 'Tracking';
                socket.emit('viewer_set_tracking', {
                    sim_id: simId,
                    tracking_robot: selectedRobot
                });
            } else {
                // Alert if no robot selected
                alert('Please select a robot to track first');
            }
        }
    });

    // Set up robot selection
    const robotSelect = cameraFeed.querySelector('.robot-select');
    robotSelect.addEventListener('change', () => {
        const trackingToggle = cameraFeed.querySelector('.tracking-toggle');
        if (trackingToggle.textContent === 'Tracking') {
            socket.emit('viewer_set_tracking', {
                sim_id: simId,
                tracking_robot: robotSelect.value
            });
        }
    });

    // Setup mouse control for camera rotation
    setupMouseControls(canvas, simId);
    
    // Make camera feed clickable to select it
    cameraFeed.addEventListener('click', () => {
        setActiveSimulation(simId);
    });

    cameraGrid.appendChild(template);
    simulationElements[simId] = cameraFeed;
    
    // Initialize robot cameras list for this simulation
    robotCameras[simId] = [];

    // If this is the first simulation, make it active
    if (!activeSimulation) {
        setActiveSimulation(simId);
    }
}

// Create a robot camera element for the detailed view
function createRobotCameraElement(robotName, cameraName) {
    // Create a unique identifier using both robot name and camera name
    const uniqueId = `${robotName}-${cameraName}`;

    // Check if we already have this camera using the unique identifier
    const existingCamera = robotFeedsContainer.querySelector(`[data-unique-id="${uniqueId}"]`);
    if (existingCamera) {
        return existingCamera;
    }

    // Extract camera number for better display
    const cameraNumber = cameraName.split('_').pop();

    const template = robotCameraTemplate.content.cloneNode(true);
    const cameraFeed = template.querySelector('.robot-camera-feed');

    cameraFeed.dataset.uniqueId = uniqueId;
    cameraFeed.dataset.cameraName = cameraName;
    cameraFeed.dataset.robotName = robotName;

    const cameraTitle = cameraFeed.querySelector('.robot-camera-title');
    cameraTitle.textContent = `${robotName} Camera ${cameraNumber}`;

    const canvas = cameraFeed.querySelector('.robot-camera-image');
    canvas.width = 320;  // Smaller width for robot cameras
    canvas.height = 180; // Smaller height for robot cameras

    // Add click handler to make this the main camera view
    cameraFeed.addEventListener('click', () => {
        // Update main camera to show this view
        const mainCanvas = mainDetailedCanvas;
        const mainCtx = mainCanvas.getContext('2d');
        mainCtx.drawImage(canvas, 0, 0, mainCanvas.width, mainCanvas.height);
        mainCameraTitle.textContent = `${robotName} Camera ${cameraNumber}`;
    });

    robotFeedsContainer.appendChild(cameraFeed);
    return cameraFeed;
}

// Setup expand button functionality
function setupExpandButton(cameraFeed, simId) {
    const expandButton = cameraFeed.querySelector('.expand-button');

    expandButton.addEventListener('click', (e) => {
        e.stopPropagation(); // Prevent triggering the camera feed click event
        openDetailedView(simId);
    });
}

// Setup bird view button functionality
function setupBirdViewButton(cameraFeed, simId) {
    const birdViewButton = cameraFeed.querySelector('.bird-view-btn');

    birdViewButton.addEventListener('click', (e) => {
        e.stopPropagation(); // Prevent triggering the camera feed click event
        
        const robotSelect = cameraFeed.querySelector('.robot-select');
        const selectedRobot = robotSelect.value;
        
        if (selectedRobot) {
            setBirdEyeView(simId, selectedRobot);
        } else {
            alert('Please select a robot to view first');
        }
    });
}

// Open detailed view for a simulation
function openDetailedView(simId) {
    if (!simulationElements[simId]) {
        console.error(`Cannot open detailed view - simulation ${simId} not found`);
        return;
    }

    detailedViewSimId = simId;

    // Update main camera title
    const simName = simulationElements[simId].querySelector('.sim-name').textContent;
    mainCameraTitle.textContent = simName;

    // Set up the main camera canvas
    const originalCanvas = simulationElements[simId].querySelector('.camera-image');
    mainDetailedCanvas.width = originalCanvas.width;
    mainDetailedCanvas.height = originalCanvas.height;

    // Copy the current image to the detailed view canvas
    const ctx = mainDetailedCanvas.getContext('2d');
    ctx.drawImage(originalCanvas, 0, 0);

    // Update position display
    const infoOverlay = simulationElements[simId].querySelector('.info-overlay');
    mainDetailedInfo.textContent = infoOverlay.textContent;

    // Update time display
    const timeDisplay = simulationElements[simId].querySelector('.camera-time');
    mainDetailedTime.textContent = timeDisplay.textContent;

    // Update tracking toggle button
    const trackingToggle = simulationElements[simId].querySelector('.tracking-toggle');
    detailedTrackingToggle.textContent = trackingToggle.textContent;

    // Update recording button
    detailedRecordBtn.classList.toggle('recording', recorderState[simId].isRecording);
    if (recorderState[simId].isRecording) {
        detailedRecordBtn.innerHTML = '<i class="bi bi-stop-circle"></i>';
        detailedRecordBtn.title = 'Stop Recording';
    } else {
        detailedRecordBtn.innerHTML = '<i class="bi bi-record-circle"></i>';
        detailedRecordBtn.title = 'Start Recording';
    }

    // Set up recording button
    detailedRecordBtn.addEventListener('click', () => {
        if (recorderState[simId].isRecording) {
            stopRecording(simId);
            detailedRecordBtn.classList.remove('recording');
            detailedRecordBtn.innerHTML = '<i class="bi bi-record-circle"></i>';
            detailedRecordBtn.title = 'Start Recording';

            // Also update the original button state
            const recordBtn = simulationElements[simId].querySelector('.record-button');
            recordBtn.classList.remove('recording');
            recordBtn.innerHTML = '<i class="bi bi-record-circle"></i>';
            recordBtn.title = 'Start Recording';
        } else {
            startRecording(simId, mainDetailedCanvas);
            detailedRecordBtn.classList.add('recording');
            detailedRecordBtn.innerHTML = '<i class="bi bi-stop-circle"></i>';
            detailedRecordBtn.title = 'Stop Recording';

            // Also update the original button state
            const recordBtn = simulationElements[simId].querySelector('.record-button');
            recordBtn.classList.add('recording');
            recordBtn.innerHTML = '<i class="bi bi-stop-circle"></i>';
            recordBtn.title = 'Stop Recording';
        }
    });

    // Set up tracking toggle
    detailedTrackingToggle.addEventListener('click', () => {
        const isTracking = detailedTrackingToggle.textContent === 'Tracking';
        if (isTracking) {
            detailedTrackingToggle.textContent = 'Free Cam';
            socket.emit('viewer_set_tracking', {
                sim_id: simId,
                tracking_robot: null
            });

            // Also update the original view's toggle
            const originalToggle = simulationElements[simId].querySelector('.tracking-toggle');
            originalToggle.textContent = 'Free Cam';
        } else {
            const selectedRobot = detailedRobotSelect.value;
            if (selectedRobot) {
                detailedTrackingToggle.textContent = 'Tracking';
                socket.emit('viewer_set_tracking', {
                    sim_id: simId,
                    tracking_robot: selectedRobot
                });

                // Also update the original view's toggle
                const originalToggle = simulationElements[simId].querySelector('.tracking-toggle');
                originalToggle.textContent = 'Tracking';
            } else {
                // Alert if no robot selected
                alert('Please select a robot to track first');
            }
        }
    });

    // Set up bird view button for detailed view
    const detailedBirdViewBtn = document.querySelector('.detailed-bird-view-btn');
    detailedBirdViewBtn.addEventListener('click', () => {
        const selectedRobot = detailedRobotSelect.value;
        if (selectedRobot) {
            setBirdEyeView(simId, selectedRobot);
        } else {
            alert('Please select a robot to view first');
        }
    });

    // Update robot selection dropdown
    detailedRobotSelect.innerHTML = '<option value="">Select Robot to Track</option>';
    const originalRobotSelect = simulationElements[simId].querySelector('.robot-select');
    Array.from(originalRobotSelect.options).forEach(option => {
        if (option.value) {
            const newOption = document.createElement('option');
            newOption.value = option.value;
            newOption.textContent = option.textContent;
            if (option.selected) {
                newOption.selected = true;
            }
            detailedRobotSelect.appendChild(newOption);
        }
    });

    // Set up robot selection change handler
    detailedRobotSelect.addEventListener('change', () => {
        if (detailedTrackingToggle.textContent === 'Tracking') {
            socket.emit('viewer_set_tracking', {
                sim_id: simId,
                tracking_robot: detailedRobotSelect.value
            });
        }
    });

    // Clear robot cameras container
    robotFeedsContainer.innerHTML = '';

    // Add robot cameras if available
    if (robotCameras[simId]) {
        robotCameras[simId].forEach(robotCamera => {
            createRobotCameraElement(robotCamera.robot, robotCamera.camera_name);
        });
    }

    // Set up detailed view camera controls
    setupDetailedViewCameraControls();

    // Show the detailed view
    detailedViewContainer.classList.add('active');
    setActiveSimulation(simId);
}

// Close detailed view
function closeDetailedView() {
    detailedViewContainer.classList.remove('active');
    detailedViewSimId = null;

    // Remove event listeners
    detailedRecordBtn.removeEventListener('click', null);
    detailedTrackingToggle.removeEventListener('click', null);
    detailedRobotSelect.removeEventListener('change', null);
}

// Set up detailed view camera controls
function setupDetailedViewCameraControls() {
    // Setup mouse control for camera rotation
    mainDetailedCanvas.addEventListener('mousedown', (e) => {
        if (!detailedViewSimId) return;

        // Ensure this simulation is active
        setActiveSimulation(detailedViewSimId);

        const simId = detailedViewSimId;
        if (!mouseState[simId]) {
            mouseState[simId] = {
                isDragging: false,
                lastX: 0,
                lastY: 0
            };
        }

        mouseState[simId].isDragging = true;
        mouseState[simId].lastX = e.clientX;
        mouseState[simId].lastY = e.clientY;
    });

    // Setup control buttons
    const controlButtons = document.querySelectorAll('.control-panel-detailed .control-button');
    controlButtons.forEach(button => {
        const action = button.dataset.action;
        const actionMap = {
            'forward': [1, 0, 0],
            'backward': [-1, 0, 0],
            'left': [0, -1, 0],
            'right': [0, 1, 0],
            'roll-left': { roll: -0.05 },
            'roll-right': { roll: 0.05 }
        };

        // Handle button press
        button.addEventListener('mousedown', () => {
            if (!detailedViewSimId) return;

            setActiveSimulation(detailedViewSimId);
            const simId = detailedViewSimId;

            if (action in actionMap) {
                button.classList.add('active');
                if (Array.isArray(actionMap[action])) {
                    // Movement direction
                    socket.emit('viewer_camera_control', {
                        sim_id: simId,
                        action: 'movement',
                        movement: actionMap[action]
                    });
                } else if (actionMap[action].roll !== undefined) {
                    // Roll action
                    socket.emit('viewer_camera_control', {
                        sim_id: simId,
                        action: 'roll',
                        roll: actionMap[action].roll
                    });
                }
            }
        });

        // Handle button release
        button.addEventListener('mouseup', () => {
            if (!detailedViewSimId) return;

            const simId = detailedViewSimId;

            if (action in actionMap) {
                button.classList.remove('active');
                if (Array.isArray(actionMap[action])) {
                    // Stop movement
                    socket.emit('viewer_camera_control', {
                        sim_id: simId,
                        action: 'movement',
                        movement: [0, 0, 0]
                    });
                } else if (actionMap[action].roll !== undefined) {
                    // Stop roll
                    socket.emit('viewer_camera_control', {
                        sim_id: simId,
                        action: 'roll',
                        roll: 0
                    });
                }
            }
        });

        // Handle button pointer leave
        button.addEventListener('pointerleave', () => {
            button.classList.remove('active');
        });
    });

    // Setup wheel event for zoom
    mainDetailedCanvas.addEventListener('wheel', (e) => {
        if (!detailedViewSimId) return;

        // Check if shift key is pressed
        if (e.shiftKey) {
            e.preventDefault(); // Prevent default scrolling

            setActiveSimulation(detailedViewSimId);
            const simId = detailedViewSimId;

            // Calculate movement based on scroll direction
            const movementValue = e.deltaY > 0 ? -5 : 5;

            // Apply movement in forward/backward direction
            socket.emit('viewer_camera_control', {
                sim_id: simId,
                action: 'movement',
                movement: [movementValue, 0, 0]
            });

            // After a short delay, reset movement to zero
            setTimeout(() => {
                socket.emit('viewer_camera_control', {
                    sim_id: simId,
                    action: 'movement',
                    movement: [0, 0, 0]
                });
            }, 100);
        }
    });
}

// Setup recording functionality
function setupRecordingButton(cameraFeed, simId, canvas) {
    const recordButton = cameraFeed.querySelector('.record-button');

    recordButton.addEventListener('click', (e) => {
        e.stopPropagation(); // Prevent triggering the camera feed click event

        if (recorderState[simId].isRecording) {
            stopRecording(simId);
            recordButton.classList.remove('recording');
            recordButton.innerHTML = '<i class="bi bi-record-circle"></i>';
            recordButton.title = 'Start Recording';

            // Update detailed view recording button if open
            if (detailedViewSimId === simId) {
                detailedRecordBtn.classList.remove('recording');
                detailedRecordBtn.innerHTML = '<i class="bi bi-record-circle"></i>';
                detailedRecordBtn.title = 'Start Recording';
            }
        } else {
            startRecording(simId, canvas);
            recordButton.classList.add('recording');
            recordButton.innerHTML = '<i class="bi bi-stop-circle"></i>';
            recordButton.title = 'Stop Recording';

            // Update detailed view recording button if open
            if (detailedViewSimId === simId) {
                detailedRecordBtn.classList.add('recording');
                detailedRecordBtn.innerHTML = '<i class="bi bi-stop-circle"></i>';
                detailedRecordBtn.title = 'Stop Recording';
            }
        }
    });
}

// Setup control buttons for a simulation
function setupControlButtons(cameraFeed, simId) {
    const controlButtons = cameraFeed.querySelectorAll('.control-button');

    // Map of actions to camera movement
    const actionMap = {
        'forward': [1, 0, 0],
        'backward': [-1, 0, 0],
        'left': [0, -1, 0],
        'right': [0, 1, 0],
        'roll-left': { roll: -0.05 },
        'roll-right': { roll: 0.05 }
    };

    controlButtons.forEach(button => {
        const action = button.dataset.action;

        // Handle button press (mousedown)
        button.addEventListener('mousedown', () => {
            if (action in actionMap) {
                button.classList.add('active');
                if (Array.isArray(actionMap[action])) {
                    // Movement direction
                    socket.emit('viewer_camera_control', {
                        sim_id: simId,
                        action: 'movement',
                        movement: actionMap[action]
                    });
                } else if (actionMap[action].roll !== undefined) {
                    // Roll action
                    socket.emit('viewer_camera_control', {
                        sim_id: simId,
                        action: 'roll',
                        roll: actionMap[action].roll
                    });
                }
            }
        });

        // Handle button release (mouseup)
        button.addEventListener('mouseup', () => {
            if (action in actionMap) {
                button.classList.remove('active');
                if (Array.isArray(actionMap[action])) {
                    // Stop movement
                    socket.emit('viewer_camera_control', {
                        sim_id: simId,
                        action: 'movement',
                        movement: [0, 0, 0]
                    });
                } else if (actionMap[action].roll !== undefined) {
                    // Stop roll
                    socket.emit('viewer_camera_control', {
                        sim_id: simId,
                        action: 'roll',
                        roll: 0
                    });
                }
            }
        });

        // Handle button pointer leave (in case mouse up happens outside the button)
        button.addEventListener('pointerleave', () => {
            button.classList.remove('active');
        });
    });
}

// Setup mouse controls for camera rotation
function setupMouseControls(canvas, simId) {
    mouseState[simId] = {
        isDragging: false,
        lastX: 0,
        lastY: 0
    };

    canvas.addEventListener('mousedown', (e) => {
        // Set this simulation as active when clicking on its canvas
        setActiveSimulation(simId);

        mouseState[simId].isDragging = true;
        mouseState[simId].lastX = e.clientX;
        mouseState[simId].lastY = e.clientY;
    });

    window.addEventListener('mouseup', () => {
        if (mouseState[simId]) {
            mouseState[simId].isDragging = false;
        }
    });

    window.addEventListener('mousemove', (e) => {
        if (mouseState[simId] && mouseState[simId].isDragging) {
            const dx = e.clientX - mouseState[simId].lastX;
            const dy = e.clientY - mouseState[simId].lastY;

            if (dx !== 0 || dy !== 0) {
                socket.emit('viewer_camera_control', {
                    sim_id: simId,
                    action: 'rotate',
                    pitch: -dy * MOUSE_SENSITIVITY,
                    yaw: dx * MOUSE_SENSITIVITY
                });
            }

            mouseState[simId].lastX = e.clientX;
            mouseState[simId].lastY = e.clientY;
        }
    });
    canvas.addEventListener('wheel', (e) => {
        // Check if shift key is pressed
        if (e.shiftKey) {
            e.preventDefault(); // Prevent default scrolling

            // Calculate movement based on scroll direction
            // Positive deltaY is scroll down, negative is scroll up
            const movementValue = e.deltaY > 0 ? -5 : 5;

            // Apply movement in forward/backward direction (same as W/S keys)
            socket.emit('viewer_camera_control', {
                sim_id: simId,
                action: 'movement',
                movement: [movementValue, 0, 0]
            });

            // After a short delay, reset movement to zero
            setTimeout(() => {
                socket.emit('viewer_camera_control', {
                    sim_id: simId,
                    action: 'movement',
                    movement: [0, 0, 0]
                });
            }, 100);
        }
    });
}

// Setup keyboard controls
function setupKeyboardControls() {
    // Map of keys to control actions
    const keyToActionMap = {
        'w': { action: 'movement', value: [1, 0, 0] },
        's': { action: 'movement', value: [-1, 0, 0] },
        'a': { action: 'movement', value: [0, 1, 0] },
        'd': { action: 'movement', value: [0, -1, 0] },
        'q': { action: 'roll', value: -0.05 },
        'e': { action: 'roll', value: 0.05 }
    };

    // Key press handler
    document.addEventListener('keydown', (e) => {
        // Only process if an active simulation is selected
        if (!activeSimulation) return;

        // Get the key
        const key = e.key.toLowerCase();

        // Check if this is a key we care about
        if (key in keyToActionMap) {
            // Prevent default browser behavior (e.g., scrolling)
            e.preventDefault();

            // If key is already pressed, don't send duplicate events
            if (keyState[key]) return;

            // Update key state
            keyState[key] = true;

            // Get the action and value
            const { action, value } = keyToActionMap[key];

            // Find the corresponding button and add active class
            const buttons = [];
            if (detailedViewSimId === activeSimulation) {
                // If detailed view is active, find buttons there
                buttons.push(document.querySelector(`.control-panel-detailed .control-button[data-key="${key}"]`));
            }
            buttons.push(simulationElements[activeSimulation].querySelector(`.control-button[data-key="${key}"]`));

            buttons.forEach(button => {
                if (button) button.classList.add('active');
            });

            // Send the control action
            if (action === 'movement') {
                socket.emit('viewer_camera_control', {
                    sim_id: activeSimulation,
                    action: 'movement',
                    movement: value
                });
            } else if (action === 'roll') {
                socket.emit('viewer_camera_control', {
                    sim_id: activeSimulation,
                    action: 'roll',
                    roll: value
                });
            }
        }
    });

    // Key release handler
    document.addEventListener('keyup', (e) => {
        // Only process if we have an active simulation
        if (!activeSimulation) return;

        // Get the key
        const key = e.key.toLowerCase();

        // Check if this is a key we care about
        if (key in keyToActionMap) {
            // Update key state
            keyState[key] = false;

            // Get the action
            const { action } = keyToActionMap[key];

            // Find the corresponding button and remove active class
            const buttons = [];
            if (detailedViewSimId === activeSimulation) {
                // If detailed view is active, find buttons there
                buttons.push(document.querySelector(`.control-panel-detailed .control-button[data-key="${key}"]`));
            }
            buttons.push(simulationElements[activeSimulation].querySelector(`.control-button[data-key="${key}"]`));

            buttons.forEach(button => {
                if (button) button.classList.remove('active');
            });

            // Send stop command if all movement keys are released
            if (action === 'movement' && !keyState.w && !keyState.a && !keyState.s && !keyState.d) {
                socket.emit('viewer_camera_control', {
                    sim_id: activeSimulation,
                    action: 'movement',
                    movement: [0, 0, 0]
                });
            } else if (action === 'roll' && !keyState.q && !keyState.e) {
                // Send stop roll command if both Q and E are released
                socket.emit('viewer_camera_control', {
                    sim_id: activeSimulation,
                    action: 'roll',
                    roll: 0
                });
            }
        }
    });

    // Handle window blur - stop all movements when window loses focus
    window.addEventListener('blur', () => {
        if (activeSimulation) {
            // Reset all key states
            Object.keys(keyState).forEach(key => {
                keyState[key] = false;
            });

            // Remove active class from all buttons
            const buttons = simulationElements[activeSimulation].querySelectorAll('.control-button');
            buttons.forEach(button => button.classList.remove('active'));

            if (detailedViewSimId === activeSimulation) {
                const detailedButtons = document.querySelectorAll('.control-panel-detailed .control-button');
                detailedButtons.forEach(button => button.classList.remove('active'));
            }

            // Send stop commands
            socket.emit('viewer_camera_control', {
                sim_id: activeSimulation,
                action: 'movement',
                movement: [0, 0, 0]
            });

            socket.emit('viewer_camera_control', {
                sim_id: activeSimulation,
                action: 'roll',
                roll: 0
            });
        }
    });
}

// Start recording a simulation
function startRecording(simId, canvas) {
    console.log(`Starting recording for simulation ${simId}`);
    
    // Clear any previous recording data
    recorderState[simId].chunks = [];
    recorderState[simId].frames = [];
    recorderState[simId].startTime = Date.now();
    
    try {
        // Create a media stream from the canvas with higher frame rate
        const stream = canvas.captureStream(60); // 60 FPS for smoother video
        
        // Determine the best codec and options for high quality
        let options;
        
        // Try to use high-quality codecs with high bitrate
        if (MediaRecorder.isTypeSupported('video/webm;codecs=vp9')) {
            // VP9 with high bitrate for best quality/compression ratio
            options = { 
                mimeType: 'video/webm;codecs=vp9',
                videoBitsPerSecond: 8000000 // 8 Mbps for high quality
            };
        } else if (MediaRecorder.isTypeSupported('video/webm;codecs=h264')) {
            // H264 fallback with high bitrate
            options = { 
                mimeType: 'video/webm;codecs=h264',
                videoBitsPerSecond: 8000000 // 8 Mbps for high quality
            };
        } else if (MediaRecorder.isTypeSupported('video/webm')) {
            // Generic WebM fallback
            options = { 
                mimeType: 'video/webm',
                videoBitsPerSecond: 6000000 // 6 Mbps
            };
        } else {
            // Last resort fallback
            options = { videoBitsPerSecond: 6000000 };
        }
        
        console.log(`Recording with options: ${JSON.stringify(options)}`);
        recorderState[simId].recorder = new MediaRecorder(stream, options);
        
        // Set up event handlers
        recorderState[simId].recorder.ondataavailable = (event) => {
            if (event.data.size > 0) {
                recorderState[simId].chunks.push(event.data);
            }
        };
        
        recorderState[simId].recorder.onstop = () => {
            processRecording(simId);
        };
        
        // Start recording with smaller time slices for better control
        recorderState[simId].recorder.start(50); // Collect data every 50ms for more frequent chunks
        recorderState[simId].isRecording = true;
        
        // Update UI to show recording state
        const cameraFeed = simulationElements[simId];
        cameraFeed.classList.add('recording-active');
        
        console.log(`Recording started for simulation ${simId}`);
    } catch (error) {
        console.error(`Error starting recording: ${error.message}`);
        alert(`Failed to start recording: ${error.message}`);
    }
}

// Stop recording a simulation
function stopRecording(simId) {
    console.log(`Stopping recording for simulation ${simId}`);
    
    if (recorderState[simId].recorder && recorderState[simId].isRecording) {
        recorderState[simId].recorder.stop();
        recorderState[simId].isRecording = false;
        
        // Update UI to remove recording state
        const cameraFeed = simulationElements[simId];
        cameraFeed.classList.remove('recording-active');
        
        console.log(`Recording stopped for simulation ${simId}`);
    }
}

// Process and save the recorded video
function processRecording(simId) {
    console.log(`Processing recording for simulation ${simId}`);
    
    const chunks = recorderState[simId].chunks;
    if (chunks.length === 0) {
        console.warn('No data recorded');
        return;
    }
    
    // Create a blob from the recorded chunks
    const blob = new Blob(chunks, { type: 'video/webm' });
    
    // Display recording information
    const videoSize = (blob.size / (1024 * 1024)).toFixed(2);
    console.log(`Recording completed: ${videoSize} MB, ${chunks.length} chunks`);
    
    // Show a small notification to the user
    const notification = document.createElement('div');
    notification.className = 'recording-notification';
    notification.innerHTML = `
        <div style="position: absolute; top: 50%; left: 50%; transform: translate(-50%, -50%); 
                    background: rgba(0,0,0,0.8); padding: 15px; border-radius: 8px; z-index: 1000;">
            <p>Recording complete! (${videoSize} MB)</p>
            <p>Your download should begin automatically.</p>
        </div>
    `;

    // Append notification to appropriate container
    if (detailedViewSimId === simId) {
        document.querySelector('.main-camera-view').appendChild(notification);
    } else {
        simulationElements[simId].appendChild(notification);
    }

    // Create a URL for the blob
    const url = URL.createObjectURL(blob);
    
    // Create a download link
    const a = document.createElement('a');
    a.style.display = 'none';
    a.href = url;
    
    // Format the filename with date and simulation name
    const simName = simulationElements[simId].querySelector('.sim-name').textContent.replace(/[^a-z0-9]/gi, '-').toLowerCase();
    const timestamp = new Date().toISOString().replace(/:/g, '-').split('.')[0];
    a.download = `simulation-${simName}-${timestamp}.webm`;
    
    // Add to document, click it, and remove it
    document.body.appendChild(a);
    a.click();
    
    // Clean up
    setTimeout(() => {
        document.body.removeChild(a);
        URL.revokeObjectURL(url);
        // Remove notification after 3 seconds
        setTimeout(() => {
            if (notification && notification.parentNode) {
                notification.parentNode.removeChild(notification);
            }
        }, 3000);
    }, 100);
    
    console.log(`Recording processed and download initiated for simulation ${simId}`);
}

// Capture a frame with its timestamp
function captureFrameWithTimestamp(simId, simulationTime) {
    if (recorderState[simId].isRecording) {
        // Store the simulation time with a high-resolution timestamp
        recorderState[simId].frames.push({
            time: Date.now() - recorderState[simId].startTime,
            simulationTime: simulationTime
        });
    }
}

// Function to set the active simulation
function setActiveSimulation(simId) {
    // Remove active class from all simulations
    Object.values(simulationElements).forEach(element => {
        element.classList.remove('active');
    });
    
    // Add active class to the selected simulation
    if (simulationElements[simId]) {
        simulationElements[simId].classList.add('active');
        activeSimulation = simId;
        console.log(`Active simulation set to: ${simId}`);
    }
}

// Function to set bird's eye view for a robot
function setBirdEyeView(simId, robotName) {
    console.log(`Setting bird's eye view for robot ${robotName} in simulation ${simId}`);
    
    socket.emit('viewer_set_bird_view', {
        sim_id: simId,
        robot_name: robotName
    });
    
    // Update UI to show we're in free cam mode (not tracking)
    const cameraFeed = simulationElements[simId];
    if (cameraFeed) {
        const trackingToggle = cameraFeed.querySelector('.tracking-toggle');
        trackingToggle.textContent = 'Free Cam';
    }
    
    // Update detailed view if open
    if (detailedViewSimId === simId) {
        detailedTrackingToggle.textContent = 'Free Cam';
    }
}

// Function to remove a simulation from the UI
function removeSimulation(simId) {
    const simElement = simulationElements[simId];
    if (simElement) {
        // If recording is in progress, stop it
        if (recorderState[simId] && recorderState[simId].isRecording) {
            stopRecording(simId);
        }
        
        // Find the parent column element
        const columnElement = simElement.closest('.col-md-4');
        if (columnElement) {
            // Remove the column from the DOM
            columnElement.remove();
            console.log(`Removed simulation ${simId} from UI`);
        }

        // Clean up related objects
        delete simulationElements[simId];
        delete recorderState[simId];
        delete robotCameras[simId];
        if (mouseState[simId]) {
            delete mouseState[simId];
        }

        // If this was the active simulation, select another one
        if (activeSimulation === simId) {
            activeSimulation = null;

            // Find another simulation to make active
            const simIds = Object.keys(simulationElements);
            if (simIds.length > 0) {
                setActiveSimulation(simIds[0]);
            }
        }
    }
}

// Update the robot dropdown for a simulation
function updateRobotList(simId, robots) {
    console.log(`Updating robot list for ${simId}:`, robots);
    
    // Make sure element exists
    if (!simulationElements[simId]) {
        console.warn(`Simulation element for ${simId} not found, can't update robot list yet`);
        return; // Will be created later and updated when simulation_update arrives
    }
    
    const cameraFeed = simulationElements[simId];
    const robotSelect = cameraFeed.querySelector('.robot-select');
    
    // Save the currently selected value
    const currentValue = robotSelect.value;
    
    // Clear existing options except the first one
    while (robotSelect.options.length > 1) {
        robotSelect.remove(1);
    }
    
    // Add new options
    robots.forEach(robot => {
        const option = document.createElement('option');
        option.value = robot;
        option.textContent = robot;
        robotSelect.appendChild(option);
    });
    
    // Restore the selected value if it still exists
    if (robots.includes(currentValue)) {
        robotSelect.value = currentValue;
    }
    
    // Update detailed view robot select if open
    if (detailedViewSimId === simId) {
        // Save the currently selected value
        const detailedCurrentValue = detailedRobotSelect.value;

        // Clear existing options except the first one
        while (detailedRobotSelect.options.length > 1) {
            detailedRobotSelect.remove(1);
        }

        // Add new options
        robots.forEach(robot => {
            const option = document.createElement('option');
            option.value = robot;
            option.textContent = robot;
            detailedRobotSelect.appendChild(option);
        });

        // Restore the selected value if it still exists
        if (robots.includes(detailedCurrentValue)) {
            detailedRobotSelect.value = detailedCurrentValue;
        }
    }

    console.log(`Updated robot dropdown, now has ${robotSelect.options.length} options`);
}

// Update simulation information
function updateSimulation(data) {
    console.log(`Updating simulation ${data.sim_id}`);
    
    if (!simulationElements[data.sim_id]) {
        console.log(`Creating new simulation element for ${data.sim_id}`);
        createSimulationElement(data.sim_id, data.sim_name);
    }

    const cameraFeed = simulationElements[data.sim_id];
    if (cameraFeed) {
        // Update the position display
        const infoOverlay = cameraFeed.querySelector('.info-overlay');
        if (data.camera_state && data.camera_state.position) {
            const pos = data.camera_state.position;
            infoOverlay.textContent = `Position: [${pos[0].toFixed(2)}, ${pos[1].toFixed(2)}, ${pos[2].toFixed(2)}]`;

            // Update detailed view position if open
            if (detailedViewSimId === data.sim_id) {
                mainDetailedInfo.textContent = infoOverlay.textContent;
            }
        }

        // Update tracking toggle button state
        const trackingToggle = cameraFeed.querySelector('.tracking-toggle');
        if (data.camera_state && data.camera_state.tracking_robot) {
            trackingToggle.textContent = 'Tracking';

            // Update robot select dropdown
            const robotSelect = cameraFeed.querySelector('.robot-select');
            if (robotSelect.value !== data.camera_state.tracking_robot) {
                robotSelect.value = data.camera_state.tracking_robot;
            }

            // Update detailed view if open
            if (detailedViewSimId === data.sim_id) {
                detailedTrackingToggle.textContent = 'Tracking';
                if (detailedRobotSelect.value !== data.camera_state.tracking_robot) {
                    detailedRobotSelect.value = data.camera_state.tracking_robot;
                }
            }
        } else {
            trackingToggle.textContent = 'Free Cam';

            // Update detailed view if open
            if (detailedViewSimId === data.sim_id) {
                detailedTrackingToggle.textContent = 'Free Cam';
            }
        }
    }
}

// Update camera feed with new image
function updateMainCameraFeed(data) {
    if (!simulationElements[data.sim_id]) {
        createSimulationElement(data.sim_id, data.sim_name || `Simulation ${data.sim_id}`);
    }

    const cameraFeed = simulationElements[data.sim_id];
    if (cameraFeed) {
        const canvas = cameraFeed.querySelector('.camera-image');
        const ctx = canvas.getContext('2d');

        // Update canvas dimensions if needed
        if (data.width && data.height && 
            (canvas.width !== data.width || canvas.height !== data.height)) {
            canvas.width = data.width;
            canvas.height = data.height;
        }

        if (data.image_data) {
            // Draw the image data to the canvas
            const img = new Image();
            img.onload = () => {
                ctx.drawImage(img, 0, 0, canvas.width, canvas.height);
                
                // If recording, capture this frame with its timestamp
                if (recorderState[data.sim_id] && recorderState[data.sim_id].isRecording && 
                    data.simulation_time !== undefined) {
                    captureFrameWithTimestamp(data.sim_id, data.simulation_time);
                }

                // Update detailed view if open
                if (detailedViewSimId === data.sim_id) {
                    const detailedCtx = mainDetailedCanvas.getContext('2d');
                    if (mainDetailedCanvas.width !== canvas.width || mainDetailedCanvas.height !== canvas.height) {
                        mainDetailedCanvas.width = canvas.width;
                        mainDetailedCanvas.height = canvas.height;
                    }
                    detailedCtx.drawImage(img, 0, 0, mainDetailedCanvas.width, mainDetailedCanvas.height);
                }
            };
            img.src = `data:image/jpeg;base64,${data.image_data}`;
        }
        
        // Update the simulation time if provided
        if (data.simulation_time !== undefined) {
            const timeDisplay = cameraFeed.querySelector('.camera-time');
            if (timeDisplay) {
                timeDisplay.textContent = formatSimulationTime(data.simulation_time);

                // Update detailed view time if open
                if (detailedViewSimId === data.sim_id) {
                    mainDetailedTime.textContent = timeDisplay.textContent;
                }
            }
        }
    }
}

// Update robot camera feed in detailed view
function updateRobotCameraFeed(data) {
    if (!detailedViewSimId || detailedViewSimId !== data.sim_id) {
        return; // Not in detailed view or not the current simulation
    }

    // Create unique identifier using both robot name and camera name
    const uniqueId = `${data.robot_name}-${data.camera_name}`;

    // Find the robot camera element using the unique identifier or create it if it doesn't exist
    let cameraElement = robotFeedsContainer.querySelector(`[data-unique-id="${uniqueId}"]`);
    if (!cameraElement) {
        // Create a new camera element
        cameraElement = createRobotCameraElement(data.robot_name, data.camera_name);
    }

    // Update the canvas
    const canvas = cameraElement.querySelector('.robot-camera-image');
    const ctx = canvas.getContext('2d');

    // Update canvas dimensions if needed
    if (data.width && data.height &&
        (canvas.width !== data.width || canvas.height !== data.height)) {
        // Use smaller dimensions for robot cameras in detailed view
        const aspectRatio = data.width / data.height;
        canvas.width = 320;
        canvas.height = Math.round(320 / aspectRatio);
    }

    if (data.image_data) {
        // Draw the image data to the canvas
        const img = new Image();
        img.onload = () => {
            // Clear the canvas first
            ctx.clearRect(0, 0, canvas.width, canvas.height);

            // Calculate dimensions to maintain aspect ratio
            const imgAspectRatio = img.width / img.height;
            const canvasAspectRatio = canvas.width / canvas.height;

            let drawWidth, drawHeight, offsetX, offsetY;

            if (imgAspectRatio > canvasAspectRatio) {
                // Image is wider than canvas - fit to width
                drawWidth = canvas.width;
                drawHeight = canvas.width / imgAspectRatio;
                offsetX = 0;
                offsetY = (canvas.height - drawHeight) / 2;
            } else {
                // Image is taller than canvas - fit to height
                drawHeight = canvas.height;
                drawWidth = canvas.height * imgAspectRatio;
                offsetX = (canvas.width - drawWidth) / 2;
                offsetY = 0;
            }

            // Draw the image centered on the canvas
            ctx.drawImage(img, offsetX, offsetY, drawWidth, drawHeight);
        };
        img.src = `data:image/jpeg;base64,${data.image_data}`;
    }
}

// For time formatting
function formatSimulationTime(timeInSeconds) {
    // Calculate minutes, seconds, and milliseconds
    const minutes = Math.floor(timeInSeconds / 60);
    const seconds = Math.floor(timeInSeconds % 60);
    const milliseconds = Math.floor((timeInSeconds % 1) * 1000);
    
    // Format with padding
    const minStr = minutes.toString().padStart(2, '0');
    const secStr = seconds.toString().padStart(2, '0');
    const msStr = milliseconds.toString().padStart(3, '0');
    
    return `${minStr}:${secStr}:${msStr}`;
}

// Initialize keyboard controls when the page loads
document.addEventListener('DOMContentLoaded', () => {
    setupKeyboardControls();
    
    // Set up close button for detailed view
    closeDetailedViewBtn.addEventListener('click', closeDetailedView);

    // Check browser support for MediaRecorder
    if (typeof MediaRecorder === 'undefined') {
        console.warn('MediaRecorder is not supported in this browser. Recording functionality will be disabled.');
        // Could display a warning to the user here
    }
});