sudo-archive/src/components/video/VideoPlayer.tsx

54 lines
1.5 KiB
TypeScript
Raw Normal View History

2023-01-10 18:53:55 +00:00
import { forwardRef, useContext, useEffect, useRef } from "react";
2023-01-08 14:37:16 +00:00
import { VideoPlayerContext, VideoPlayerContextProvider } from "./VideoContext";
2023-01-08 12:15:32 +00:00
2023-01-08 21:29:38 +00:00
export interface VideoPlayerProps {
autoPlay?: boolean;
2023-01-08 12:15:32 +00:00
children?: React.ReactNode;
}
const VideoPlayerInternals = forwardRef<
HTMLVideoElement,
{ autoPlay: boolean }
>((props, ref) => {
2023-01-08 12:15:32 +00:00
const video = useContext(VideoPlayerContext);
2023-01-10 18:53:55 +00:00
const didInitialize = useRef<true | null>(null);
2023-01-08 12:15:32 +00:00
2023-01-10 18:53:55 +00:00
useEffect(() => {
if (didInitialize.current) return;
if (!video.state.hasInitialized || !video.source) return;
video.state.initPlayer(video.source, video.sourceType);
didInitialize.current = true;
}, [didInitialize, video]);
// muted attribute is required for safari, as they cant change the volume itself
2023-01-08 12:15:32 +00:00
return (
<video
ref={ref}
autoPlay={props.autoPlay}
2023-01-10 18:53:55 +00:00
muted={video.state.volume === 0}
playsInline
className="h-full w-full"
2023-01-10 18:53:55 +00:00
/>
2023-01-08 12:15:32 +00:00
);
});
export function VideoPlayer(props: VideoPlayerProps) {
const playerRef = useRef<HTMLVideoElement | null>(null);
2023-01-08 15:23:42 +00:00
const playerWrapperRef = useRef<HTMLDivElement | null>(null);
2023-01-08 12:15:32 +00:00
return (
2023-01-08 15:23:42 +00:00
<VideoPlayerContextProvider player={playerRef} wrapper={playerWrapperRef}>
<div
2023-01-10 18:53:55 +00:00
className="relative aspect-video w-full select-none overflow-hidden bg-black"
2023-01-08 15:23:42 +00:00
ref={playerWrapperRef}
>
<VideoPlayerInternals
autoPlay={props.autoPlay ?? false}
ref={playerRef}
/>
2023-01-08 15:23:42 +00:00
<div className="absolute inset-0">{props.children}</div>
</div>
2023-01-08 12:15:32 +00:00
</VideoPlayerContextProvider>
);
}