mirror of
https://github.com/Unleash/unleash.git
synced 2025-09-05 17:53:12 +02:00
feat: lazy load youtube
This commit is contained in:
parent
8f6d440d86
commit
74d127e205
@ -1,29 +1,98 @@
|
|||||||
// biome-ignore lint/correctness/noUnusedImports: Needs this for React to work
|
// biome-ignore lint/correctness/noUnusedImports: Needs this for React to work
|
||||||
import React from 'react';
|
import React, { useState, useCallback } from 'react';
|
||||||
import Admonition from '@theme/Admonition';
|
import Admonition from '@theme/Admonition';
|
||||||
|
import styles from './VideoContent.module.css';
|
||||||
|
|
||||||
|
// Extract YouTube video ID from various URL formats
|
||||||
|
const extractVideoId = (url) => {
|
||||||
|
const regex =
|
||||||
|
/(?:youtube\.com\/(?:[^\/]+\/.+\/|(?:v|e(?:mbed)?)\/|.*[?&]v=)|youtu\.be\/)([^"&?\/\s]{11})/;
|
||||||
|
const match = url.match(regex);
|
||||||
|
return match ? match[1] : null;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Lazy video component that shows thumbnail until clicked
|
||||||
|
const LazyVideo = ({ url, title = 'YouTube video player' }) => {
|
||||||
|
const [isLoaded, setIsLoaded] = useState(false);
|
||||||
|
const videoId = extractVideoId(url);
|
||||||
|
|
||||||
|
const handleLoad = useCallback(() => {
|
||||||
|
setIsLoaded(true);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
if (!videoId) {
|
||||||
|
return (
|
||||||
|
<Admonition type='warning'>Invalid YouTube URL: {url}</Admonition>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isLoaded) {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={styles.videoThumbnail}
|
||||||
|
onClick={handleLoad}
|
||||||
|
onKeyDown={(e) => e.key === 'Enter' && handleLoad()}
|
||||||
|
role='button'
|
||||||
|
tabIndex={0}
|
||||||
|
style={{
|
||||||
|
backgroundImage: `url(https://img.youtube.com/vi/${videoId}/maxresdefault.jpg)`,
|
||||||
|
}}
|
||||||
|
aria-label={`Load ${title}`}
|
||||||
|
>
|
||||||
|
{/* Play button overlay */}
|
||||||
|
<div className={styles.playButton}>
|
||||||
|
<svg
|
||||||
|
width='32'
|
||||||
|
height='32'
|
||||||
|
viewBox='0 0 24 24'
|
||||||
|
fill='white'
|
||||||
|
style={{ marginLeft: '2px' }}
|
||||||
|
>
|
||||||
|
<path d='M8 5v14l11-7z' />
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Loading hint */}
|
||||||
|
<div className={styles.loadingHint}>Click to load video</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
const Component = ({ videoUrls }) => {
|
|
||||||
return (
|
return (
|
||||||
<article className='unleash-video-container'>
|
|
||||||
{videoUrls ? (
|
|
||||||
videoUrls.map((url) => (
|
|
||||||
<iframe
|
<iframe
|
||||||
key={url}
|
className={styles.loadedVideo}
|
||||||
width='100%'
|
width='100%'
|
||||||
height='auto'
|
height='315'
|
||||||
src={url}
|
src={`${url}${url.includes('?') ? '&' : '?'}autoplay=1`}
|
||||||
title='YouTube video player'
|
title={title}
|
||||||
|
allow='accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share'
|
||||||
allowFullScreen
|
allowFullScreen
|
||||||
|
loading='lazy'
|
||||||
/>
|
/>
|
||||||
))
|
);
|
||||||
) : (
|
};
|
||||||
|
|
||||||
|
const VideoContent = ({ videoUrls }) => {
|
||||||
|
if (!videoUrls || videoUrls.length === 0) {
|
||||||
|
return (
|
||||||
<Admonition type='danger'>
|
<Admonition type='danger'>
|
||||||
You need to provide YouTube video URLs for this component to
|
You need to provide YouTube video URLs for this component to
|
||||||
work properly.
|
work properly.
|
||||||
</Admonition>
|
</Admonition>
|
||||||
)}
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<article className={`unleash-video-container ${styles.videoContainer}`}>
|
||||||
|
{videoUrls.map((url, index) => (
|
||||||
|
<LazyVideo
|
||||||
|
key={url}
|
||||||
|
url={url}
|
||||||
|
title={`YouTube video player ${index + 1}`}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
</article>
|
</article>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default Component;
|
export default VideoContent;
|
||||||
|
108
website/src/components/VideoContent.module.css
Normal file
108
website/src/components/VideoContent.module.css
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
/* Video thumbnail hover effects */
|
||||||
|
.videoThumbnail {
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
padding-bottom: 56.25%; /* 16:9 aspect ratio */
|
||||||
|
background-size: cover;
|
||||||
|
background-position: center;
|
||||||
|
cursor: pointer;
|
||||||
|
border-radius: 8px;
|
||||||
|
overflow: hidden;
|
||||||
|
border: 1px solid var(--ifm-color-emphasis-200);
|
||||||
|
transition: transform 0.2s ease, box-shadow 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.videoThumbnail:hover {
|
||||||
|
transform: translateY(-2px);
|
||||||
|
box-shadow: 0 4px 20px rgba(0, 0, 0, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.videoThumbnail:hover .playButton {
|
||||||
|
transform: translate(-50%, -50%) scale(1.1);
|
||||||
|
background-color: var(--ifm-color-primary);
|
||||||
|
}
|
||||||
|
|
||||||
|
.playButton {
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
left: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
width: 80px;
|
||||||
|
height: 80px;
|
||||||
|
background-color: rgba(0, 0, 0, 0.8);
|
||||||
|
border-radius: 50%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
border: 3px solid rgba(255, 255, 255, 0.9);
|
||||||
|
}
|
||||||
|
|
||||||
|
.loadingHint {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 10px;
|
||||||
|
left: 10px;
|
||||||
|
background-color: rgba(0, 0, 0, 0.7);
|
||||||
|
color: white;
|
||||||
|
padding: 4px 8px;
|
||||||
|
border-radius: 4px;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 500;
|
||||||
|
backdrop-filter: blur(4px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.videoContainer {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 1.5rem;
|
||||||
|
margin: 1rem 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loadedVideo {
|
||||||
|
border: none;
|
||||||
|
border-radius: 8px;
|
||||||
|
box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Dark mode adjustments */
|
||||||
|
[data-theme='dark'] .videoThumbnail {
|
||||||
|
border-color: var(--ifm-color-emphasis-300);
|
||||||
|
}
|
||||||
|
|
||||||
|
[data-theme='dark'] .videoThumbnail:hover {
|
||||||
|
box-shadow: 0 4px 20px rgba(255, 255, 255, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Responsive design */
|
||||||
|
@media (max-width: 768px) {
|
||||||
|
.playButton {
|
||||||
|
width: 60px;
|
||||||
|
height: 60px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.videoContainer {
|
||||||
|
gap: 1rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Loading state */
|
||||||
|
.videoThumbnail:focus-visible {
|
||||||
|
outline: 2px solid var(--ifm-color-primary);
|
||||||
|
outline-offset: 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Animation for smooth loading */
|
||||||
|
.loadedVideo {
|
||||||
|
animation: fadeInScale 0.3s ease-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes fadeInScale {
|
||||||
|
from {
|
||||||
|
opacity: 0;
|
||||||
|
transform: scale(0.95);
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
opacity: 1;
|
||||||
|
transform: scale(1);
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user