<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <title>DexNDM</title>
  <link rel="icon" href="./figs/logo.png" type="image/png">
  <link rel="stylesheet" href="style.css">
</head>
<body>
  <div class="toc">
    <h3>Content</h3>
    <hr>
    <ul>
      <li><a href="#abstract">Abstract</a></li>
      <li><a href="#skill-highlight">In-Hand Rotation Skills (Highlight)</a></li>
      <!-- <li><a href="#more-primitive-skills">Skill Galleries</a></li> -->
      <li><a href="#long-complex-tasks">Application: Teleoperating Complex Dexterous Manipulation Tasks</a></li>
    </ul>
  </div>

  <div class="main-content">
    <div class="hero-text">DexNDM</div>
    <div class="sub-hero-text">CLOSING THE REALITY GAP FOR <br>
      Dexterous IN-HAND ROTATION VIA Joint-wise NEURAL DYNAMICS Model 
    </div>

    <!-- Add Authors -->
    <div class="authors">
      Anonymous Author(s)
    </div>
    <!-- End Authors -->

    <div class="tagline" id="skill-highlight">In-Hand Rotation Skills (Highlight)</div>

    <div class="video-gallery-section" id="highlightGallerySection">
      <div class="video-gallery-container">

        <div class="video-grid" id="videoGridHighlight">
          <div class="row" style="display: flex; flex-wrap: wrap; justify-content: center; gap: 20px; margin-bottom: 20px;">

            <div class="col" style="flex: 1; min-width: 300px; max-width: 100%;">
              <h3 class="title is-5" style="text-align: center; margin-bottom: 10px;">16cm x 3cm x 3cm (2x)</h3>
              <video autoplay muted loop playsinline controls 
                     class="lazy"
                     width="100%"
                     style="border-radius: 10px; display: block;">
                     <source data-src="./static/videos_lowres3/palmdown_16cm_360(1)(1).mp4" type="video/mp4">
              </video>
            </div>
            
            <div class="col" style="flex: 1; min-width: 300px; max-width: 100%;">
              <h3 class="title is-5" style="text-align: center; margin-bottom: 10px;">20cm x 3cm x 3cm (2x)</h3>
              <video autoplay muted loop playsinline controls 
                     class="lazy"
                     width="100%"
                     style="border-radius: 10px; display: block;">
                     <source data-src="./static/videos_lowres3/thumbup_20_3_3_morethan360(1)(1).mp4" type="video/mp4">
              </video>
            </div>

          </div>
        </div>


        <div class="video-grid" id="videoGridHighlight">
          <div class="row" style="display: flex; flex-wrap: wrap; justify-content: center; gap: 20px; margin-bottom: 20px;">

            <div class="col" style="flex: 1; min-width: 300px; max-width: 100%;">
              <h3 class="title is-5" style="text-align: center; margin-bottom: 10px;">14cm x 3cm x 3cm (2x)</h3>
              <video autoplay muted loop playsinline controls 
                     class="lazy"
                     width="100%"
                     style="border-radius: 10px; display: block;">
                     <source data-src="./static/videos_lowres3/palmdown_14_3_3_more360(1)(1).mp4" type="video/mp4">
              </video>
            </div>
            
            <div class="col" style="flex: 1; min-width: 300px; max-width: 100%;">
              <h3 class="title is-5" style="text-align: center; margin-bottom: 10px;">20cm x 4cm x 4cm (2x)</h3>
              <video autoplay muted loop playsinline controls 
                     class="lazy"
                     width="100%"
                     style="border-radius: 10px; display: block;">
                     <source data-src="./static/videos_lowres3/thumbup_20_4_4_near450(1)(1).mp4" type="video/mp4">
              </video>
            </div>

          </div>
        </div>

        <div class="video-grid" id="videoGridHighlight">
          <div class="row" style="display: flex; flex-wrap: wrap; justify-content: center; gap: 20px; margin-bottom: 20px;">


            <div class="col" style="flex: 1; min-width: 300px; max-width: 100%;">
              <h3 class="title is-5" style="text-align: center; margin-bottom: 10px;">16cm x 3cm x 3cm (2x)</h3>
              <video autoplay muted loop playsinline controls 
                     class="lazy"
                     width="100%"
                     style="border-radius: 10px; display: block;">
                     <source data-src="./static/videos_lowres3/thumbup_16_3_3_360more(1)(1).mp4" type="video/mp4">
              </video>
            </div>
            

            <!-- almost_360_rot(1)(1), thumbup_legoleg_360(1)(1), thumbup_cuboidleg_z_rot -->
            <div class="col" style="flex: 1; min-width: 300px; max-width: 100%;">
              <h3 class="title is-5" style="text-align: center; margin-bottom: 10px;">16cm x 3 cm x 23cm (2x)</h3>
              <video autoplay muted loop playsinline controls 
                     class="lazy"
                     width="100%"
                     style="border-radius: 10px; display: block;">
                     <source data-src="./static/videos_lowres3/rot_book_1(1)(1).mp4" type="video/mp4">
              </video>
            </div>


          </div>
        </div>



        <div class="video-grid" id="videoGridHighlight">
          <div class="row" style="display: flex; flex-wrap: wrap; justify-content: center; gap: 20px; margin-bottom: 20px;">

            <div class="col" style="flex: 1; min-width: 300px; max-width: 100%;">
              <h3 class="title is-5" style="text-align: center; margin-bottom: 10px;">Complex Geometry (2x)</h3>
              <video autoplay muted loop playsinline controls 
                     class="lazy"
                     width="100%"
                     style="border-radius: 10px; display: block;">
                     <source data-src="./static/videos_lowres3/complex_geometry_video_2(1).mp4" type="video/mp4">
              </video>
            </div>
            
            <div class="col" style="flex: 1; min-width: 300px; max-width: 100%;">
              <h3 class="title is-5" style="text-align: center; margin-bottom: 10px; ">Small Objects (2x)</h3>
              <video autoplay muted loop playsinline controls 
                     class="lazy"
                     width="100%"
                     style="border-radius:10px; display: block;">
                     <source data-src="./static/videos_lowres3/small_obj_rot_collection(1)(1).mp4" type="video/mp4">
                    </video>
            </div>


          </div>
        </div>

        <div class="video-grid" id="videoGridHighlight">
          <div class="row" style="display: flex; flex-wrap: wrap; justify-content: center; gap: 20px; margin-bottom: 20px;">

            <div class="col" style="flex: 1; min-width: 300px; max-width: 100%;">
              <h3 class="title is-5" style="text-align: center; margin-bottom: 10px;">Diverse Wrist Orientations (2x)</h3>
              <video autoplay muted loop playsinline controls 
                     class="lazy"
                     width="100%"
                     style="border-radius: 10px; display: block;">
                     <source data-src="./static/videos_lowres3/anywristorient(1).mp4" type="video/mp4">
              </video>
            </div>
            
            <div class="col" style="flex: 1; min-width: 300px; max-width: 100%;">
              <h3 class="title is-5" style="text-align: center; margin-bottom: 10px; ">Rich Rotation Axes (2x)</h3>
              <video autoplay muted loop playsinline controls 
                     class="lazy"
                     width="100%"
                     style="border-radius:10px; display: block;">
                     <source data-src="./static/videos_lowres3/rotaxis.mp4" type="video/mp4">
                    </video>
            </div>


          </div>
        </div>

      </div>

    </div>





    <div class="tagline" id="long-complex-tasks">Application: Teleoperating Complex Dexterous Manipulation Tasks</div>

    <div class="video-gallery-section" id="assemblyTaskGallerySection">
      <div class="video-gallery-container">

        <div class="video-gallery" id="videoGalleryAssemblyTask">
          <video class="gallery-video" src="./static/videos_lowres3/screwdriver_longer_4x_v3(1).mp4" autoplay muted playsinline loop style="border-radius: 10px; display: block; min-width: 400px; max-width: 400px;"></video>
          <!-- <video class="gallery-video" src="./static/videos_lowres3/pen_write_v2(1).mp4" autoplay muted playsinline loop style="border-radius: 10px; display: block; min-width: 400px; max-width: 400px;"></video> -->
          <video class="gallery-video" src="./static/videos_lowres3/hammer_6(1).mp4" autoplay muted playsinline loop style="border-radius: 10px; display: block; min-width: 400px; max-width: 400px;"></video>
          <!-- <video class="gallery-video" src="./static/videos_lowres3/brush_v5_2(1).mp4" autoplay muted playsinline loop style="border-radius: 10px; display: block; min-width: 400px; max-width: 400px;"></video> -->
          <!-- <video class="gallery-video" src="./static/videos_lowres3/hammer_v2_7(1).mp4" autoplay muted playsinline loop style="border-radius: 10px; display: block; min-width: 400px; max-width: 400px;"></video> -->
          <video class="gallery-video" src="./static/videos_lowres3/cut_banana_3(1).mp4" autoplay muted playsinline loop style="border-radius: 10px; display: block; min-width: 400px; max-width: 400px;"></video>
          <!-- <video class="gallery-video" src="./static/videos_lowres3/spoon_3(1).mp4" autoplay muted playsinline loop style="border-radius: 10px; display: block; min-width: 400px; max-width: 400px;"></video> -->
          <!-- <video class="gallery-video" src="./static/videos_lowres3/brush_small_v2(1).mp4" autoplay muted playsinline loop style="border-radius: 10px; display: block; min-width: 400px; max-width: 400px;"></video> -->
        </div>
      </div>
      <!-- Container for the caption AND buttons -->
      <div class="gallery-caption-container">
          <!-- Move button controls INSIDE caption container -->
          <div class="gallery-nav-controls">
              <button class="gallery-nav left" id="scrollLeftBtnAssemblyTask">&lt;</button>
              <button class="gallery-nav right" id="scrollRightBtnAssemblyTask">&gt;</button>
              <p class="figure-caption gallery-caption">
                <b>Teleoperation Tasks.</b> Teleoperating the robot to use tools: screwdriver (4x), thin pen (4x), hammer-90 (2x), cut banana using knife (2x).
            </p>
          </div>
          
      </div>
    </div>

  </div> <!-- End of main-content div -->


  <div class="footer">
     Website template modified from <a href="https://www.videomimic.net/">VideoMimic</a> and <a href="https://humanoid-clone.github.io/">CLONE</a>.
  </div>

  <!-- Teaser Video Autoplay with Delay and Loop -->
  <script>
  document.addEventListener('DOMContentLoaded', function() {
    const video = document.getElementById('teaser-video');
    // const initialDelay = 1000; // No longer needed, autoplay attribute handles initial play
    const loopDelay = 3000;    // 3 seconds delay before looping

    // let initialPlayTimeout; // No longer needed
    let loopTimeout;

    if (video) {
      // Ensure video is muted (already in HTML, but good practice)
      video.muted = true;
      // Controls are already in HTML, ensuring user can play if autoplay fails

      // REMOVE JavaScript-based initial play:
      /*
      initialPlayTimeout = setTimeout(function() {
        video.play().then(function() {
          // Autoplay started
        }).catch(function(error) {
          console.log('Initial autoplay prevented for teaser video. User interaction might be needed.', error);
          video.controls = true; // Ensure controls are visible
        });
      }, initialDelay);
      */

      // Loop with delay - this part can stay
      video.addEventListener('ended', function() {
        clearTimeout(loopTimeout); 
        loopTimeout = setTimeout(function() {
          video.currentTime = 0; 
          video.play().catch(function(error) {
            console.log('Delayed loop play prevented for teaser video:', error);
          });
        }, loopDelay);
      });

      // --- Clear Timeouts on Manual Pause ---
      video.addEventListener('pause', function() {
        if (!video.ended && video.currentTime > 0) {
           // clearTimeout(initialPlayTimeout); // No longer needed
           clearTimeout(loopTimeout);
           console.log('Teaser video: Manual pause detected, clearing loop timeout.');
        }
      });

      // --- Clear Timeouts on Manual Play (if paused before initial delay finishes) ---
       video.addEventListener('play', function() {
           // if (initialPlayTimeout) { // No longer needed
           //     clearTimeout(initialPlayTimeout);
           // }
       });

    } else {
      console.error('Video element with ID "teaser-video" not found.');
    }
  });
  </script>


  <script>
    document.addEventListener("DOMContentLoaded", function() {
      var lazyVideos = [].slice.call(document.querySelectorAll("video.lazy"));
      if ("IntersectionObserver" in window) {
        var lazyVideoObserver = new IntersectionObserver(function(entries, observer) {
          entries.forEach(function(video) {
            if (video.isIntersecting) {
              for (var source in video.target.children) {
                var videoSource = video.target.children[source];
                if (typeof videoSource.tagName === "string" && videoSource.tagName === "SOURCE") {
                  videoSource.src = videoSource.dataset.src;
                }
              }
              video.target.load();
              video.target.classList.remove("lazy");
              lazyVideoObserver.unobserve(video.target);
            }
          });
        });
        lazyVideos.forEach(function(lazyVideo) {
          lazyVideoObserver.observe(lazyVideo);
        });
      }
    });
  </script>

  <script>
    document.addEventListener('DOMContentLoaded', function() {
      const galleries = [
        {
          sectionId: 'gallery-section-anchor', 
          galleryInnerId: 'videoGallerySitting',
          scrollLeftBtnId: 'scrollLeftBtnSitting',
          scrollRightBtnId: 'scrollRightBtnSitting'
        },
        {
          sectionId: 'traversingGallerySection', 
          galleryInnerId: 'videoGalleryTraversing',
          scrollLeftBtnId: 'scrollLeftBtnTraversing',
          scrollRightBtnId: 'scrollRightBtnTraversing'
        },
        {
          sectionId: 'stairsGallerySection', 
          galleryInnerId: 'videoGalleryStairs',
          scrollLeftBtnId: 'scrollLeftBtnStairs',
          scrollRightBtnId: 'scrollRightBtnStairs'
        },
        {
          sectionId: 'reconstructionGallerySection', 
          galleryInnerId: 'videoGalleryReconstruction',
          scrollLeftBtnId: 'scrollLeftBtnReconstruction',
          scrollRightBtnId: 'scrollRightBtnReconstruction'
        },
        {
          sectionId: 'nyrotGallerySection', 
          galleryInnerId: 'videoGalleryNyrot',
          scrollLeftBtnId: 'scrollLeftBtnNyrot',
          scrollRightBtnId: 'scrollRightBtnNyrot'
        },
        {
          sectionId: 'zrotGallerySection',
          galleryInnerId: 'videoGalleryZrot',
          scrollLeftBtnId: 'scrollLeftBtnZrot',
          scrollRightBtnId: 'scrollRightBtnZrot'
        },
        {
          sectionId: 'zrotSmallGallerySection', 
          galleryInnerId: 'videoGalleryZrotSmall',
          scrollLeftBtnId: 'scrollLeftBtnZrotSmall',
          scrollRightBtnId: 'scrollRightBtnZrotSmall'
        },
        {
          sectionId: 'xyrotGallerySection', 
          galleryInnerId: 'videoGalleryXYrot',
          scrollLeftBtnId: 'scrollLeftBtnXYrot',
          scrollRightBtnId: 'scrollRightBtnXYrot'
        },
        {
          sectionId: 'anyAxisGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryAnyAxis',
          scrollLeftBtnId: 'scrollLeftBtnAnyAxis',
          scrollRightBtnId: 'scrollRightBtnAnyAxis'
        },
        {
          sectionId: 'gravityInvGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryGravityInv',
          scrollLeftBtnId: 'scrollLeftBtnGravityInv',
          scrollRightBtnId: 'scrollRightBtnGravityInv'
        },
        {
          sectionId: 'assemblyTaskGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryAssemblyTask',
          scrollLeftBtnId: 'scrollLeftBtnAssemblyTask',
          scrollRightBtnId: 'scrollRightBtnAssemblyTask'
        },
        {
          sectionId: 'toolUseGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryToolUse',
          scrollLeftBtnId: 'scrollLeftBtnToolUse',
          scrollRightBtnId: 'scrollRightBtnToolUse'
        },
        {
          sectionId: 'orientationSensativePlacementGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryOrientationSensativePlacement',
          scrollLeftBtnId: 'scrollLeftBtnOrientationSensativePlacement',
          scrollRightBtnId: 'scrollRightBtnOrientationSensativePlacement'
        },
        {
          sectionId: 'nzrotGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryNZrot',
          scrollLeftBtnId: 'scrollLeftBtnNZrot',
          scrollRightBtnId: 'scrollRightBtnNZrot'
        },
        {
          sectionId: 'nyrotGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryNyrot',
          scrollLeftBtnId: 'scrollLeftBtnNyrot',
          scrollRightBtnId: 'scrollRightBtnNyrot'
        },
        {
          sectionId: 'nxrotGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryNxrot',
          scrollLeftBtnId: 'scrollLeftBtnNxrot',
          scrollRightBtnId: 'scrollRightBtnNxrot'
        },
        {
          sectionId: 'yrotGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryYrot',
          scrollLeftBtnId: 'scrollLeftBtnYrot',
          scrollRightBtnId: 'scrollRightBtnYrot'
        },
        {
          sectionId: 'xrotGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryXrot',
          scrollLeftBtnId: 'scrollLeftBtnXrot',
          scrollRightBtnId: 'scrollRightBtnXrot'
        },
        {
          sectionId: 'gravityInvAnyAxisGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryGravityInvAnyAxis',
          scrollLeftBtnId: 'scrollLeftBtnGravityInvAnyAxis',
          scrollRightBtnId: 'scrollRightBtnGravityInvAnyAxis'
        },
        {
          sectionId: 'cmpAnyRotateGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryCmpAnyRotate',
          scrollLeftBtnId: 'scrollLeftBtnCmpAnyRotate',
          scrollRightBtnId: 'scrollRightBtnCmpAnyRotate'
        },
        {
          sectionId: 'cmpVisualDexterityGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryCmpVisualDexterity',
          scrollLeftBtnId: 'scrollLeftBtnCmpVisualDexterity',
          scrollRightBtnId: 'scrollRightBtnCmpVisualDexterity'
        },
        {
          sectionId: 'cmpUANASAPGallerySection', // ID of the .video-gallery-section for reconstruction
          galleryInnerId: 'videoGalleryCmpUANASAP',
          scrollLeftBtnId: 'scrollLeftBtnCmpUANASAP',
          scrollRightBtnId: 'scrollRightBtnCmpUANASAP'
        }
      ];

      galleries.forEach(galleryConfig => {
        const gallerySection = document.getElementById(galleryConfig.sectionId);
        if (!gallerySection) {
          console.error(`Gallery section with ID ${galleryConfig.sectionId} not found.`);
          return;
        }

        const galleryContainer = gallerySection.querySelector('.video-gallery-container');
        const galleryInner = document.getElementById(galleryConfig.galleryInnerId);
        const scrollLeftBtn = document.getElementById(galleryConfig.scrollLeftBtnId);
        const scrollRightBtn = document.getElementById(galleryConfig.scrollRightBtnId);

        if (galleryContainer && galleryInner && scrollLeftBtn && scrollRightBtn) {
          // Calculate the scroll amount based on the width of the first video + gap
          const scrollAmount = (galleryInner.firstElementChild?.offsetWidth || 300) + 15; // 15 is the gap

          scrollLeftBtn.addEventListener('click', () => {
            // Scroll the CONTAINER element
            galleryContainer.scrollBy({ left: -scrollAmount, behavior: 'smooth' });
          });

          scrollRightBtn.addEventListener('click', () => {
            // Scroll the CONTAINER element
            galleryContainer.scrollBy({ left: scrollAmount, behavior: 'smooth' });
          });

          /* --- REMOVE OR COMMENT OUT HOVER LOGIC ---
          // Optional: Add hover-to-play functionality for gallery videos
          // Target videos within galleryInner
          const galleryVideos = galleryInner.querySelectorAll('.gallery-video');
          galleryVideos.forEach(video => {
              video.addEventListener('mouseenter', () => {
                  video.play().catch(e => console.log("Autoplay prevented:", e));
              });
              video.addEventListener('mouseleave', () => {
                  video.pause();
                  // video.currentTime = 0; // Optional: reset video on mouse leave
              });
          });
          */ // --- END OF REMOVED HOVER LOGIC ---

        } else {
          console.error(`Gallery elements not found for navigation setup in section ${galleryConfig.sectionId}.`);
          // Log which elements might be missing
          if (!galleryContainer) console.error('Missing element: .video-gallery-container in section ' + galleryConfig.sectionId);
          if (!galleryInner) console.error(`Missing element with ID ${galleryConfig.galleryInnerId}`);
          if (!scrollLeftBtn) console.error(`Missing element with ID ${galleryConfig.scrollLeftBtnId}`);
          if (!scrollRightBtn) console.error(`Missing element with ID ${galleryConfig.scrollRightBtnId}`);
        }
      });
    });
  </script>

  <!-- JavaScript for Real-to-Sim Video Synchronization -->
  <script>
    document.addEventListener('DOMContentLoaded', function() {
      const videosToSync = [
        document.querySelector('.r2s-video-input'),
        document.querySelector('.r2s-video-smpl'),
        document.querySelector('.r2s-video-g1'),
        document.querySelector('.r2s-video-ego-rgb'),
        document.querySelector('.r2s-video-ego-depth'),
        document.querySelector('.r2s-video-sim')
      ].filter(Boolean); // Filter out nulls if any class name is wrong or video missing

      function synchronizeAndPlayR2SVideos() {
        if (videosToSync.length === 0) {
          console.warn('No videos found for Real-to-Sim synchronization.');
          return;
        }

        const readyPromises = videosToSync.map(video => {
          return new Promise((resolve, reject) => {
            // If video is already ready (e.g., cached), resolve immediately
            if (video.readyState >= 4) { // HAVE_ENOUGH_DATA (canplaythrough)
              resolve();
            } else {
              video.addEventListener('canplaythrough', resolve, { once: true });
              video.addEventListener('error', reject, { once: true }); // Handle potential loading errors
            }
          });
        });

        Promise.all(readyPromises)
          .then(() => {
            console.log('All Real-to-Sim videos are ready to play. Starting playback.');
            videosToSync.forEach(video => {
              video.currentTime = 0; // Ensure starting from the beginning
              video.play().catch(error => {
                console.warn(`Autoplay was prevented for video ${video.src}. User interaction might be needed.`, error);
                // Ensure controls are visible if autoplay fails for any video
                video.controls = true;
              });
            });
          })
          .catch(error => {
            console.error('Error waiting for Real-to-Sim videos to be ready:', error);
            // Optionally, provide a fallback or user message here
            videosToSync.forEach(video => video.controls = true); // Show controls on all if any failed to load
          });
      }

      synchronizeAndPlayR2SVideos();

      // The `loop` attribute on the HTML video tags will handle continuous looping.
      // The videos will naturally re-synchronize at their LCM due to the loop attribute
      // if they have different durations and all successfully start.
    });
  </script>

  <!-- JavaScript to prevent default click on specific image links -->
  <script>
    document.addEventListener('DOMContentLoaded', function() {
      const imageLinkIds = ['figure-1-img', 'figure-2-img', 'figure-3-img'];
      imageLinkIds.forEach(id => {
        const linkElement = document.getElementById(id);
        if (linkElement) {
          linkElement.addEventListener('click', function(event) {
            event.preventDefault();
          });
        }
      });
    });
  </script>

</body>
</html>
