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

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

  <style>
    button:focus-visible,
    input[type="checkbox"]:focus-visible { /* Kept for any other potential inputs */
      outline: 2px solid #9CA3AF;
      outline-offset: 2px;
    }

    .toggle-label {
      font-size: 0.875rem; /* text-sm */
      color: #D1D5DB; /* gray-300 */
    }

    .model-scroll-area {
        max-height: 510px;
        overflow-y: auto;
        padding: 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;
    }

    .border-gray-500 {
        border-color: transparent !important;
    }
    .border-gray-800 {
        border-color: transparent !important;
    }
  </style>
</head>

<body class="bg-black text-gray-200 p-4 md:p-8 font-sans">
  <div class="flex items-center gap-x-3 mb-1">
    <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 videogenerators</h1>
  </div>
  <h2 class="text-2xl  mb-4 text-left text-gray-100">Kubric</h2>

  <div class="flex justify-center items-center gap-x-6 gap-y-3 mb-6 p-4 rounded-lg shadow-md flex-wrap">
    <div class="flex items-center">
        <label for="globalToggleBg" class="toggle-label mr-3 font-medium text-gray-100">Background:</label>
        <button type="button" id="globalToggleBg" 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">Toggle Background</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 class="flex items-center">
        <label for="globalToggleRgb" class="toggle-label mr-3 font-medium text-gray-100">RGB:</label>
        <button type="button" id="globalToggleRgb" 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">Toggle RGB</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="flex flex-col lg:flex-row justify-center items-start gap-6 md:gap-8 mt-8">

    <div
      class="bg-black rounded-xl shadow-xl p-2 w-full lg:max-w-[320px] flex-shrink-0 border border-gray-500 mx-auto lg:mx-0">
      <h2 id="initialConditionTitle" class="text-xl font-semibold mb-4 text-center text-blue-400">Initial Condition</h2>
      <div class="flex flex-col items-center gap-2">
        <img id="initialConditionImage" src="https://placehold.co/256x256/000000/D1D5DB?text=Loading..."
          alt="Initial Condition Scene 1" class="w-[256px] h-[256px] object-cover rounded-md shadow-lg" />
        <img id="initialConditionImageSecondary" src="https://placehold.co/256x256/000000/D1D5DB?text=Secondary"
          alt="Secondary Initial Condition" class="w-[256px] h-[256px] object-cover rounded-md shadow-lg" />
      </div>
    </div>

    <div class="flex flex-col md:flex-row justify-center items-start gap-6 md:gap-8">
      <div class="bg-black rounded-xl shadow-xl p-2 w-fit border border-gray-800 mx-auto md:mx-0">
        <h2 class="text-xl font-semibold mb-2 text-center text-gray-100">Ours</h2>
        <div id="Ours-scroll-area" class="model-scroll-area mt-2">
          </div>
      </div>

      <div class="bg-black rounded-xl shadow-xl p-2 w-fit border border-gray-800 mx-auto md:mx-0">
        <h2 class="text-xl font-semibold mb-2 text-center text-gray-100">WAN (1.3B, fine-tuned)</h2>
        <div id="WAN-scroll-area" class="model-scroll-area mt-2">
          </div>
      </div>

      <div class="bg-black rounded-xl shadow-xl p-2 w-fit border border-gray-800 mx-auto md:mx-0">
        <h2 class="text-xl font-semibold mb-2 text-center text-gray-100">SVD (fine-tuned)</h2>
        <div id="SVD-scroll-area" class="model-scroll-area mt-2">
          </div>
      </div>
    </div>
  </div>

  <div class="flex justify-center items-center gap-4 mt-10 pt-4">
    <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 / 5</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>

  <script>
    document.addEventListener('DOMContentLoaded', () => {
      const TOTAL_SCENES = 5;
      const ITEMS_PER_GROUP = 4;
      const TOTAL_SAMPLES_PER_MODEL = 8;
      const BENCHMARK = "kubric";
      const MODELS = [
        {
          id: 'Ours',
          name: 'Ours',
          scrollAreaId: 'Ours-scroll-area',
          samples: { "kubric": "samples/kubric/ours(L)" }
        },
        {
          id: 'WAN',
          name: 'WAN',
          scrollAreaId: 'WAN-scroll-area',
          samples: { "kubric": "samples/kubric/wan" }
        },
        {
          id: 'SVD',
          name: 'SVD',
          scrollAreaId: 'SVD-scroll-area',
          samples: { "kubric": "samples/kubric/svd" }
        }
      ];
      const initialConditionPlaceholderColor = '000000/D1D5DB';

      const LS_KEY_SHOW_BG = 'genModelCompareShowBg';
      const LS_KEY_SHOW_RGB = 'genModelCompareShowRgb';
      const LS_KEY_SCENE_INDEX = 'genModelCompareSceneIndex';

      let globalShowBg, globalShowRgb, currentSceneIndex;

      const savedShowBg = localStorage.getItem(LS_KEY_SHOW_BG);
      if (savedShowBg !== null) {
          globalShowBg = savedShowBg === 'true';
      } else {
          globalShowBg = true;
      }

      const savedShowRgb = localStorage.getItem(LS_KEY_SHOW_RGB);
      if (savedShowRgb !== null) {
          globalShowRgb = savedShowRgb === 'true';
      } else {
          globalShowRgb = false;
      }

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

      const prevSceneBtn = document.getElementById('prevSceneBtn');
      const nextSceneBtn = document.getElementById('nextSceneBtn');
      const sceneIndicator = document.getElementById('sceneIndicator');
      const initialConditionImageEl = document.getElementById('initialConditionImage');
      const initialConditionImageSecondaryEl = document.getElementById('initialConditionImageSecondary');
      const initialConditionTitleEl = document.getElementById('initialConditionTitle');

      const globalToggleBgBtn = document.getElementById('globalToggleBg');
      const globalToggleBgCircle = globalToggleBgBtn.querySelector('span[aria-hidden="true"]');
      const globalToggleRgbBtn = document.getElementById('globalToggleRgb');
      const globalToggleRgbCircle = globalToggleRgbBtn.querySelector('span[aria-hidden="true"]');


      function createErrorPlaceholder(modelName, sceneNum, rolloutIdx, messagePrefix = "Error loading video") {
          const placeholder = document.createElement('img');
          placeholder.className = 'w-full h-auto aspect-square object-cover rounded shadow-sm media-element';
          placeholder.src = `https://placehold.co/128x128/000000/FCA5A5?text=Error`;
          placeholder.alt = `${messagePrefix} ${modelName} Scene ${sceneNum} Sample ${rolloutIdx}`;
          return placeholder;
      }

      function getMediaProperties(modelId, sceneNum, rolloutIdx) {
        const bgndFileSuffix = globalShowBg ? 'with_bg' : 'without_bg';
        const bgAltSuffix = globalShowBg ? 'BG' : 'NoBG';
        const fileExtension = 'mp4';
        const typePrefixAlt = 'Video';

        let rgbFileSuffix;
        const isOursModel = modelId === 'Ours';

        if (isOursModel) {
            rgbFileSuffix = 'black';
        } else {
            if (globalShowRgb) {
                rgbFileSuffix = 'rgb';
            } else {
                rgbFileSuffix = 'black';
            }
        }

        const finalAltSuffix = `${typePrefixAlt}${bgAltSuffix}`;

        const formattedScene = `scene=${String(sceneNum).padStart(6, '0')}`;
        const rollout = String(rolloutIdx).padStart(6, '0');
        const modelConfig = MODELS.find(m => m.id === modelId);
        const basePath = modelConfig.samples[BENCHMARK];

        const src = `${basePath}/${formattedScene}/${rollout}_tab20_${bgndFileSuffix}_${rgbFileSuffix}.${fileExtension}`;
        const ariaLabel = `${modelConfig.name} Scene ${sceneNum} Sample ${rolloutIdx} ${finalAltSuffix}`;

        return { src, ariaLabel };
      }

      function updateModelPaths(model, sceneNum) {
        const scrollArea = document.getElementById(model.scrollAreaId);
        if (!scrollArea) {
            return;
        }
        const itemContainers = scrollArea.querySelectorAll('.item-container');

        itemContainers.forEach(container => {
            const videoEl = container.querySelector('.media-element');
            if (videoEl && videoEl.tagName === 'VIDEO') {
                const rolloutIdx = parseInt(videoEl.dataset.rolloutIdx, 10);

                if (isNaN(rolloutIdx)) {
                    return;
                }

                const props = getMediaProperties(model.id, sceneNum, rolloutIdx);
                const currentAssignedSrc = videoEl.dataset.currentSrc;

                if (currentAssignedSrc !== props.src) {
                    const sourceEl = videoEl.querySelector('source');
                    if (sourceEl) {
                        sourceEl.src = props.src;
                        videoEl.dataset.currentSrc = props.src;
                        videoEl.load();
                    }
                }

                if (videoEl.getAttribute('aria-label') !== props.ariaLabel) {
                    videoEl.setAttribute('aria-label', props.ariaLabel);
                }
            }
        });
      }

    function populateModelScrollArea(model, sceneNum) {
      const scrollArea = document.getElementById(model.scrollAreaId);
      if (!scrollArea) {
        return;
      }
      scrollArea.innerHTML = '';

      const numGroups = TOTAL_SAMPLES_PER_MODEL / ITEMS_PER_GROUP;

      for (let groupIndex = 0; groupIndex < numGroups; groupIndex++) {
        const startSampleOffset = groupIndex * ITEMS_PER_GROUP;
        const gridDiv = document.createElement('div');

        const isLastGroup = groupIndex === numGroups - 1;
        gridDiv.className = `grid grid-cols-2 gap-2 ${!isLastGroup ? 'mb-2' : ''}`;

        for (let i = 0; i < ITEMS_PER_GROUP; i++) {
          const rolloutIdx = startSampleOffset + i + 1;
          const props = getMediaProperties(model.id, sceneNum, rolloutIdx);

          const itemContainer = document.createElement('div');
          itemContainer.className = 'relative flex flex-col items-center item-container';

          // MODIFIED: Removed sample number display
          // const indexElement = document.createElement('div');
          // indexElement.textContent = `${rolloutIdx}`;
          // indexElement.className = 'text-xs text-gray-400 font-medium mb-0.5';

          const videoElement = document.createElement('video');
          videoElement.className = 'w-full h-auto 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.rolloutIdx = rolloutIdx;
          videoElement.dataset.currentSrc = props.src;

          const source = document.createElement('source');
          source.src = props.src;
          source.type = 'video/mp4';
          source.onerror = function(e) {
            if (itemContainer.parentElement) {
              const errorPlaceholder = createErrorPlaceholder(model.name, sceneNum, rolloutIdx);

              const errorItemContainer = document.createElement('div');
              errorItemContainer.className = 'relative flex flex-col items-center item-container';

              // MODIFIED: Removed sample number display from error
              // const errorIndexElement = document.createElement('div');
              // errorIndexElement.textContent = `${rolloutIdx}`;
              // errorIndexElement.className = 'text-xs text-gray-400 font-medium mb-0.5';

              // errorItemContainer.appendChild(errorIndexElement);
              errorItemContainer.appendChild(errorPlaceholder);

              itemContainer.parentElement.replaceChild(errorItemContainer, itemContainer);
            }
          };
          videoElement.appendChild(source);

          // MODIFIED: Removed appending sample number
          // itemContainer.appendChild(indexElement);
          itemContainer.appendChild(videoElement);

          gridDiv.appendChild(itemContainer);
        }
        scrollArea.appendChild(gridDiv);
      }
    }

      function loadSceneContent(sceneIdx) {
        const sceneNum = sceneIdx + 1;
        if (initialConditionImageEl && initialConditionImageSecondaryEl) {
            const formattedScene = `scene=${String(sceneNum).padStart(6, '0')}`;
            const basePathForInitial = MODELS[0].samples[BENCHMARK];

            initialConditionImageEl.src = `${basePathForInitial}/${formattedScene}/initial_frame.jpg`;
            initialConditionImageEl.alt = `Initial Condition Scene ${sceneNum} - Frame`;

            initialConditionImageSecondaryEl.src = `${basePathForInitial}/${formattedScene}/initial_frame_with_queries_rgb.jpg`;
            initialConditionImageSecondaryEl.alt = `Initial Condition Scene ${sceneNum} - Frame with Queries`;

            initialConditionImageEl.onerror = function () {
              this.onerror = null;
              this.src = `https://placehold.co/256x256/${initialConditionPlaceholderColor}?text=Error`;
              this.alt = `Error loading Initial Condition Scene ${sceneNum} - Frame`;
            };
            initialConditionImageSecondaryEl.onerror = function () {
              this.onerror = null;
              this.src = `https://placehold.co/256x256/${initialConditionPlaceholderColor}?text=Error`;
              this.alt = `Error loading Initial Condition Scene ${sceneNum} - Frame with Queries`;
            };
        }

        MODELS.forEach(model => {
            populateModelScrollArea(model, sceneNum);
        });
      }

      function updateSceneDisplay() {
        const displaySceneNum = currentSceneIndex + 1;
        sceneIndicator.textContent = `Scene ${displaySceneNum} / ${TOTAL_SCENES}`;
        if (initialConditionTitleEl) {
            // MODIFIED: Removed scene number from Initial Condition title
            initialConditionTitleEl.textContent = `Initial Condition`;
        }
      }

      function updateButtonStates() {
        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();
        }
      }

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

      function updateToggleVisuals(buttonEl, circleEl, isOnState, onColorClass = 'bg-blue-600', offColorClass = 'bg-gray-700') {
        buttonEl.setAttribute('aria-checked', String(isOnState));
        if (isOnState) {
            buttonEl.classList.remove(offColorClass);
            buttonEl.classList.add(onColorClass);
            circleEl.classList.remove('translate-x-0');
            circleEl.classList.add('translate-x-5');
        } else {
            buttonEl.classList.remove(onColorClass);
            buttonEl.classList.add(offColorClass);
            circleEl.classList.remove('translate-x-5');
            circleEl.classList.add('translate-x-0');
        }
      }

      updateToggleVisuals(globalToggleBgBtn, globalToggleBgCircle, globalShowBg, 'bg-blue-600');
      updateToggleVisuals(globalToggleRgbBtn, globalToggleRgbCircle, globalShowRgb, 'bg-blue-600');

      globalToggleBgBtn.addEventListener('click', function () {
          globalShowBg = !globalShowBg;
          localStorage.setItem(LS_KEY_SHOW_BG, String(globalShowBg));
          updateToggleVisuals(globalToggleBgBtn, globalToggleBgCircle, globalShowBg, 'bg-blue-600');
          MODELS.forEach(model => {
            updateModelPaths(model, currentSceneIndex + 1);
          });
      });

      globalToggleRgbBtn.addEventListener('click', function () {
          globalShowRgb = !globalShowRgb;
          localStorage.setItem(LS_KEY_SHOW_RGB, String(globalShowRgb));
          updateToggleVisuals(globalToggleRgbBtn, globalToggleRgbCircle, globalShowRgb, 'bg-blue-600');
          MODELS.forEach(model => {
            if (model.id !== 'Ours') {
                updateModelPaths(model, currentSceneIndex + 1);
            }
          });
      });

      loadSceneContent(currentSceneIndex);
      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>