<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8" />
  <meta name="viewport" content="width=device-width, initial-scale=1" />
  <title>Comparison with ATM</title>
  <script src="https://cdn.tailwindcss.com"></script>

  <style>
    button:focus-visible {
      outline: 2px solid #9CA3AF;
      outline-offset: 2px;
    }

    .model-scroll-area {
        max-height: 610px; 
        overflow-y: auto;
        padding: 0 8px 8px 8px; 
        border-radius: 0.375rem; /* rounded-md */
        background-color: black;
    }

    .model-scroll-area::-webkit-scrollbar { width: 8px; }
    .model-scroll-area::-webkit-scrollbar-track { background: #1f2937; border-radius: 10px; }
    .model-scroll-area::-webkit-scrollbar-thumb { background: #6b7280; border-radius: 10px; }
    .model-scroll-area::-webkit-scrollbar-thumb:hover { background: #9ca3af; }

     .model-card {
        background-color: #000000;
        border-radius: 0.75rem; /* rounded-xl */
        box-shadow: 0 10px 15px -3px rgba(0,0,0,0.1), 0 4px 6px -2px rgba(0,0,0,0.05);
        padding: 0.5rem; /* p-2 */
     }
     .instruction-text-container {
        max-height: 150px;
        overflow-y: auto;
        background-color: #111827; /* gray-900 */
        padding: 0.75rem; /* p-3 */
        border-radius: 0.375rem; /* rounded-md */
        box-shadow: inset 0 2px 4px 0 rgba(0,0,0,0.06);
     }
     .instruction-text-container::-webkit-scrollbar { width: 6px; }
     .instruction-text-container::-webkit-scrollbar-track { background: #1f2937; border-radius: 6px; }
     .instruction-text-container::-webkit-scrollbar-thumb { background: #4b5563; border-radius: 6px; }
     .instruction-text-container::-webkit-scrollbar-thumb:hover { background: #6b7280; }
  </style>
</head>

<body class="bg-black text-gray-200 p-4 md:p-8 font-sans">

  <div class="container mx-auto px-4">
    <div class="flex items-baseline gap-x-3 mb-2">
      <a href="index.html"
         aria-label="Go to Homepage"
         class="text-blue-400 hover:text-blue-500 text-3xl font-semibold transition-colors duration-150 ease-in-out focus:outline-none focus-visible:ring-2 focus-visible:ring-blue-300 focus-visible:ring-offset-2 focus-visible:ring-offset-black rounded-md">
        &larr;
      </a>
      <h1 class="text-3xl font-bold text-left text-gray-100">Comparison with ATM</h1>
      <h2 class="text-2xl font-semibold text-gray-300 ml-1">LIBERO</h2>
    </div>
    <h2 id="sceneInstructionDisplay" class="text-xl mb-5 text-left text-gray-300 instruction-text-container">Loading instruction...</h2>

    <div class="my-4 flex justify-end items-center">
      <label for="rgbToggle" class="mr-3 text-sm font-medium text-gray-300">Show RGB for GT:</label>
      <button type="button" id="rgbToggle" role="switch" aria-checked="false"
        class="bg-gray-700 relative inline-flex flex-shrink-0 h-6 w-11 border-2 border-transparent rounded-full cursor-pointer transition-colors ease-in-out duration-200 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500 focus:ring-offset-black">
        <span class="sr-only">Show RGB for GT</span>
        <span aria-hidden="true"
          class="translate-x-0 pointer-events-none inline-block h-5 w-5 rounded-full bg-white shadow transform ring-0 transition ease-in-out duration-200"></span>
      </button>
    </div>
  </div>

  <div class="w-full px-4">
    <div class="flex flex-col lg:flex-row justify-center items-center lg:items-start gap-4 md:gap-5 mt-5">
      <div id="initialConditionContainer" class="bg-black rounded-xl shadow-xl pt-2 px-4 pb-4 w-full lg:max-w-sm flex-shrink-0 initial-condition-card mx-auto lg:mx-0">
        <h2 class="text-xl font-semibold mb-3 text-center text-blue-400">Initial Condition</h2>
        <div class="flex flex-col items-center gap-2">
          <div>
              <h3 class="text-sm font-semibold text-center text-gray-200 mb-2">Side View</h3>
              <img id="initialConditionAgentImage" src="https://placehold.co/256x256/000000/D1D5DB?text=Loading..."
                   alt="Initial Condition Side View" class="w-[256px] h-[256px] object-cover rounded-md shadow-lg" />
          </div>
          <div>
              <h3 class="text-sm font-semibold text-center text-gray-200 mb-2">Effector View</h3>
              <img id="initialConditionGripperImage" src="https://placehold.co/256x256/000000/D1D5DB?text=Loading..."
                   alt="Initial Condition Effector View" class="w-[256px] h-[256px] object-cover rounded-md shadow-lg" />
          </div>
        </div>
      </div>

      <div id="modelsComparisonContainer" class="flex flex-row flex-wrap lg:flex-nowrap justify-center items-stretch gap-3 md:gap-4">
          </div>
    </div>
  </div>

  <div class="container mx-auto px-4">
    <div class="flex justify-center items-center gap-4 mt-8 pt-4 mb-6">
      <button id="prevSceneBtn"
        class="bg-blue-700 hover:bg-blue-800 text-white font-semibold py-2 px-4 rounded-lg shadow-md transition-colors duration-150 ease-in-out disabled:opacity-50 disabled:bg-gray-700 disabled:cursor-not-allowed">
        &larr; Previous
      </button>
      <span id="sceneIndicator" class="text-lg font-medium text-gray-300">Scene 1 / N</span>
      <button id="nextSceneBtn"
        class="bg-blue-700 hover:bg-blue-800 text-white font-semibold py-2 px-4 rounded-lg shadow-md transition-colors duration-150 ease-in-out disabled:opacity-50 disabled:bg-gray-700 disabled:cursor-not-allowed">
        Next &rarr;
      </button>
    </div>
  </div>

  <script>
    document.addEventListener('DOMContentLoaded', () => {
      const BENCHMARK = "libero";
      // const SCENE_INSTRUCTIONS = {};
      // for (let i = 0; i <= 128; i++) {
      //   const key = i.toString().padStart(6, '0');
      //   SCENE_INSTRUCTIONS[key] = ""; // or "" if you prefer an empty string
      // }

      // // Add specific instructions
      // SCENE_INSTRUCTIONS['000007'] = "turn on the stove and put the moka pot on it";
      // SCENE_INSTRUCTIONS['000067'] = "turn on the stove and put the moka pot on it";
      // SCENE_INSTRUCTIONS['000086'] = "turn on the stove and put the moka pot on it";
      // SCENE_INSTRUCTIONS['000107'] = "put the black bowl in the bottom drawer of the cabinet and close it";
      // SCENE_INSTRUCTIONS['000114'] = "put the black bowl in the bottom drawer of the cabinet and close it";
      const SCENE_INSTRUCTIONS = {
        '000024': "turn on the stove and put the moka pot on it",
        '000062': "put the black bowl in the bottom drawer of the cabinet and close it",
        '000034': "turn on the stove and put the moka pot on it",
        '000081': "put the black bowl in the bottom drawer of the cabinet and close it",
        '000116': "put the yellow and white mug in the microwave and close it",
      };

      const SCENE_NUMBERS = Object.keys(SCENE_INSTRUCTIONS);
      const TOTAL_SCENES = SCENE_NUMBERS.length;


      const MODELS = [
        { id: 'gt', name: 'Ground Truth', folderName: 'gt', rollouts: 1 },
        { id: 'ours', name: 'Ours', folderName: 'ours', rollouts: 8 },
        { id: 'atm', name: 'ATM', folderName: 'atm', rollouts: 1 }
      ];
      const initialConditionPlaceholderColor = '000000/D1D5DB';
      const LS_KEY_SCENE_INDEX = 'liberoCompareSceneIndex_v2';
      let currentSceneIndex;

      const LS_KEY_RGB_MODE = 'liberoRgbModeEnabled_v1';
      const storedRgbMode = localStorage.getItem(LS_KEY_RGB_MODE);
      let isRgbModeEnabled;
      if (storedRgbMode === null) {
          isRgbModeEnabled = true; 
          localStorage.setItem(LS_KEY_RGB_MODE, String(isRgbModeEnabled)); 
      } else {
          isRgbModeEnabled = storedRgbMode === 'true';
      }
      
      const rgbToggleBtn = document.getElementById('rgbToggle');
      const rgbToggleCircle = rgbToggleBtn.querySelector('span[aria-hidden="true"]');

      function updateRgbToggleVisuals() {
        rgbToggleBtn.setAttribute('aria-checked', isRgbModeEnabled);
        if (isRgbModeEnabled) {
            rgbToggleBtn.classList.remove('bg-gray-700');
            rgbToggleBtn.classList.add('bg-blue-600');
            rgbToggleCircle.classList.remove('translate-x-0');
            rgbToggleCircle.classList.add('translate-x-5');
        } else {
            rgbToggleBtn.classList.remove('bg-blue-600');
            rgbToggleBtn.classList.add('bg-gray-700');
            rgbToggleCircle.classList.remove('translate-x-5');
            rgbToggleCircle.classList.add('translate-x-0');
        }
      }
      updateRgbToggleVisuals();

      rgbToggleBtn.addEventListener('click', () => {
        isRgbModeEnabled = !isRgbModeEnabled;
        localStorage.setItem(LS_KEY_RGB_MODE, String(isRgbModeEnabled));
        updateRgbToggleVisuals();
        if (TOTAL_SCENES > 0 && currentSceneIndex >= 0 && currentSceneIndex < TOTAL_SCENES) {
             loadSceneContent(currentSceneIndex); 
        }
      });

      const savedSceneIndexStr = localStorage.getItem(LS_KEY_SCENE_INDEX);
      if (savedSceneIndexStr !== null) {
          const savedSceneIndex = parseInt(savedSceneIndexStr, 10);
          currentSceneIndex = (!isNaN(savedSceneIndex) && savedSceneIndex >= 0 && savedSceneIndex < TOTAL_SCENES) ? savedSceneIndex : 0;
      } else {
          currentSceneIndex = 0;
      }
      if (currentSceneIndex >= TOTAL_SCENES && TOTAL_SCENES > 0) currentSceneIndex = TOTAL_SCENES - 1;
      else if (TOTAL_SCENES === 0) currentSceneIndex = 0;

      const prevSceneBtn = document.getElementById('prevSceneBtn');
      const nextSceneBtn = document.getElementById('nextSceneBtn');
      const sceneIndicator = document.getElementById('sceneIndicator');
      const initialConditionAgentImageEl = document.getElementById('initialConditionAgentImage');
      const initialConditionGripperImageEl = document.getElementById('initialConditionGripperImage');
      const instructionDisplayEl = document.getElementById('sceneInstructionDisplay');
      const modelsComparisonContainer = document.getElementById('modelsComparisonContainer');

      function createErrorPlaceholder(modelName, sceneFolder, rolloutIdx, messagePrefix = "Error") {
          const placeholder = document.createElement('img');
          placeholder.className = 'w-[256px] h-[256px] object-cover rounded shadow-sm media-element';
          placeholder.src = `https://placehold.co/256x256/${initialConditionPlaceholderColor.replace('/', '/FCA5A5/')}?text=${encodeURIComponent(messagePrefix)}`;
          let altText = `${messagePrefix}`;
          if (modelName) altText += ` ${modelName}`;
          if (sceneFolder) altText += ` Scene ${sceneFolder.split('=')[1] || sceneFolder}`;
          if (rolloutIdx !== undefined) altText += ` Sample ${rolloutIdx + 1}`;
          placeholder.alt = altText;
          return placeholder;
      }

      function getMediaProperties(modelId, sceneFolder, rolloutIdx, viewType) { 
        const bgndFileSuffix = 'without_bg'; 
        const fileExtension = 'mp4';
        const modelConfig = MODELS.find(m => m.id === modelId);

        if (!modelConfig) {
            console.error(`Configuration error for modelId: ${modelId}`);
            return { src: '', ariaLabel: 'Error: Model configuration missing' };
        }

        const rolloutFileNum = String(rolloutIdx).padStart(6, '0');
        let videoQualitySuffix = 'black'; 

        if (modelId === 'gt' && isRgbModeEnabled) {
            videoQualitySuffix = 'rgb';
        }
        
        const src = `samples/${BENCHMARK}/scene=${sceneFolder}/${modelConfig.folderName}/${viewType}/${rolloutFileNum}_${bgndFileSuffix}_${videoQualitySuffix}.${fileExtension}`;
        
        const displayViewType = viewType === 'agent' ? 'Side' : 'Effector';
        let ariaLabel = `${modelConfig.name} Scene ${sceneFolder} Rollout ${rolloutIdx + 1} ${displayViewType} view`;
        if (modelId === 'gt' && isRgbModeEnabled) {
            ariaLabel += ' (RGB)';
        }
        return { src, ariaLabel };
      }

      function createVideoElement(props, model, sceneFolder, rolloutIdx, viewType) { 
          const videoElement = document.createElement('video');
          videoElement.className = 'w-[256px] h-[256px] aspect-square object-cover rounded shadow-sm media-element';
          videoElement.autoplay = true;
          videoElement.loop = true;
          videoElement.muted = true;
          videoElement.playsInline = true;
          videoElement.setAttribute('aria-label', props.ariaLabel);
          videoElement.dataset.modelId = model.id;
          videoElement.dataset.rolloutIdx = rolloutIdx;
          videoElement.dataset.viewType = viewType; 
          videoElement.dataset.currentSrc = props.src;

          const source = document.createElement('source');
          source.src = props.src;
          source.type = 'video/mp4';
          source.onerror = function(e) {
            console.error(`Error loading video: ${props.src}`, e);
            if (videoElement.parentElement) {
              const errorDisplayViewType = viewType === 'agent' ? 'Side' : 'Effector';
              const errorPlaceholder = createErrorPlaceholder(model.name, sceneFolder, rolloutIdx, `Error ${errorDisplayViewType}`);
              videoElement.parentElement.replaceChild(errorPlaceholder, videoElement);
            }
          };
          videoElement.appendChild(source);
          return videoElement;
      }

      function populateModelVideos(model, sceneFolder) {
        const scrollArea = document.getElementById(model.id + '-scroll-area');
        if (!scrollArea) {
          console.error(`Scroll area not found for model: ${model.id}-scroll-area`);
          return;
        }
        scrollArea.innerHTML = '';
        
        for (let r = 0; r < model.rollouts; r++) {
          const rolloutIdx = r;
          const rolloutContainer = document.createElement('div');
          rolloutContainer.className = 'p-1 rounded-md'; 
          const videosDiv = document.createElement('div');
          videosDiv.className = 'flex flex-col gap-2 items-center'; 

          const agentViewTitle = document.createElement('h4');
          agentViewTitle.className = 'text-sm font-semibold text-center text-gray-200';
          agentViewTitle.textContent = 'Side View';
          videosDiv.appendChild(agentViewTitle);

          const agentProps = getMediaProperties(model.id, sceneFolder, rolloutIdx, 'agent');
          const agentVideoElement = createVideoElement(agentProps, model, sceneFolder, rolloutIdx, 'agent');
          videosDiv.appendChild(agentVideoElement);

          const gripperViewTitle = document.createElement('h4');
          gripperViewTitle.className = 'text-sm font-semibold text-center text-gray-200';
          gripperViewTitle.textContent = 'Effector View';
          videosDiv.appendChild(gripperViewTitle);

          const gripperProps = getMediaProperties(model.id, sceneFolder, rolloutIdx, 'gripper');
          const gripperVideoElement = createVideoElement(gripperProps, model, sceneFolder, rolloutIdx, 'gripper');
          videosDiv.appendChild(gripperVideoElement);

          rolloutContainer.appendChild(videosDiv);
          scrollArea.appendChild(rolloutContainer); 
        }
      }

      function loadSceneContent(sceneIdx) {
        if (TOTAL_SCENES === 0) {
            instructionDisplayEl.textContent = "No scenes configured.";
            initialConditionAgentImageEl.src = `https://placehold.co/256x256/${initialConditionPlaceholderColor}?text=No+Scenes`;
            initialConditionGripperImageEl.src = `https://placehold.co/256x256/${initialConditionPlaceholderColor}?text=No+Scenes`;
            MODELS.forEach(model => {
                const scrollArea = document.getElementById(model.id + '-scroll-area');
                if(scrollArea) scrollArea.innerHTML = '<p class="text-center text-gray-400">No scenes</p>';
            });
            return;
        }
        const currentSceneFolder = SCENE_NUMBERS[sceneIdx];

        const instructionText = SCENE_INSTRUCTIONS[currentSceneFolder];
        if (instructionText) {
            instructionDisplayEl.textContent = instructionText;
        } else {
            console.error("Instruction not found for scene:", currentSceneFolder);
            instructionDisplayEl.textContent = 'Instruction not available for this scene.';
        }

        if (initialConditionAgentImageEl && initialConditionGripperImageEl) {
            const basePathForInitial = `samples/${BENCHMARK}/scene=${currentSceneFolder}`;
            initialConditionAgentImageEl.src = `${basePathForInitial}/initial_frame_agent_queries.jpg`; 
            initialConditionAgentImageEl.alt = `Initial Condition Scene ${currentSceneFolder} - Side View`;
            initialConditionGripperImageEl.src = `${basePathForInitial}/initial_frame_gripper_queries.jpg`;
            initialConditionGripperImageEl.alt = `Initial Condition Scene ${currentSceneFolder} - Effector View`;
            
            initialConditionAgentImageEl.onerror = function () { 
                this.onerror = null; 
                this.src = `https://placehold.co/256x256/${initialConditionPlaceholderColor.replace('/', '/FCA5A5/')}?text=Error`; 
                this.alt = `Error loading Side View for Scene ${currentSceneFolder}`; 
            };
            initialConditionGripperImageEl.onerror = function () { 
                this.onerror = null; 
                this.src = `https://placehold.co/256x256/${initialConditionPlaceholderColor.replace('/', '/FCA5A5/')}?text=Error`; 
                this.alt = `Error loading Effector View for Scene ${currentSceneFolder}`; 
            };
        }

        MODELS.forEach(model => { populateModelVideos(model, currentSceneFolder); });
      }

      function updateSceneDisplay() {
        if (TOTAL_SCENES === 0) { sceneIndicator.textContent = "Scene 0 / 0"; return; }
        sceneIndicator.textContent = `Scene ${currentSceneIndex + 1} / ${TOTAL_SCENES}`;
      }

      function updateButtonStates() {
        if (TOTAL_SCENES === 0) { prevSceneBtn.disabled = true; nextSceneBtn.disabled = true; return; }
        prevSceneBtn.disabled = currentSceneIndex === 0;
        nextSceneBtn.disabled = currentSceneIndex === TOTAL_SCENES - 1;
      }

      function goToNextScene() {
        if (currentSceneIndex < TOTAL_SCENES - 1) {
          currentSceneIndex++;
          localStorage.setItem(LS_KEY_SCENE_INDEX, currentSceneIndex);
          loadSceneContent(currentSceneIndex);
          updateSceneDisplay();
          updateButtonStates();
        }
      }

      function goToPrevScene() {
        if (currentSceneIndex > 0) {
          currentSceneIndex--;
          localStorage.setItem(LS_KEY_SCENE_INDEX, currentSceneIndex);
          loadSceneContent(currentSceneIndex);
          updateSceneDisplay();
          updateButtonStates();
        }
      }

      MODELS.forEach(model => {
          const modelCard = document.createElement('div');
          modelCard.className = 'model-card flex flex-col w-full max-w-xs'; 

          const modelTitle = document.createElement('h2');
          modelTitle.className = 'text-xl font-semibold mb-2 text-center text-gray-100';
          modelTitle.textContent = model.name;
          modelCard.appendChild(modelTitle);

          const scrollArea = document.createElement('div');
          scrollArea.id = model.id + '-scroll-area';
          scrollArea.className = 'model-scroll-area flex-grow'; 
          scrollArea.classList.add('flex', 'flex-col', 'gap-2');
          
          modelCard.appendChild(scrollArea);
          modelsComparisonContainer.appendChild(modelCard);
      });

      prevSceneBtn.addEventListener('click', goToPrevScene);
      nextSceneBtn.addEventListener('click', goToNextScene);

      if (TOTAL_SCENES > 0) {
        loadSceneContent(currentSceneIndex);
      } else { 
        loadSceneContent(0); 
      }
      updateSceneDisplay();
      updateButtonStates();

      document.addEventListener('keydown', (event) => {
        if (event.key === 'ArrowLeft' && !['INPUT', 'TEXTAREA', 'BUTTON'].includes(document.activeElement.tagName)) { goToPrevScene(); }
        else if (event.key === 'ArrowRight' && !['INPUT', 'TEXTAREA', 'BUTTON'].includes(document.activeElement.tagName)) { goToNextScene(); }
      });
    });
  </script>
</body>
</html>