Spaces:
Running
Running
import { useState, useEffect, useRef, useCallback, useMemo } from "react"; | |
import LoadingScreen from "./components/LoadingScreen"; | |
import CaptioningView from "./components/CaptioningView"; | |
import WelcomeScreen from "./components/WelcomeScreen"; | |
import WebcamPermissionDialog from "./components/WebcamPermissionDialog"; | |
import type { AppState } from "./types"; | |
export default function App() { | |
const [appState, setAppState] = useState<AppState>("requesting-permission"); | |
const [webcamStream, setWebcamStream] = useState<MediaStream | null>(null); | |
const [isVideoReady, setIsVideoReady] = useState(false); | |
const videoRef = useRef<HTMLVideoElement | null>(null); | |
const handlePermissionGranted = useCallback((stream: MediaStream) => { | |
setWebcamStream(stream); | |
setAppState("welcome"); | |
}, []); | |
const handleStart = useCallback(() => { | |
setAppState("loading"); | |
}, []); | |
const handleLoadingComplete = useCallback(() => { | |
setAppState("captioning"); | |
}, []); | |
const playVideo = useCallback(async (video: HTMLVideoElement) => { | |
try { | |
await video.play(); | |
} catch (error) { | |
console.error("Failed to play video:", error); | |
} | |
}, []); | |
const setupVideo = useCallback( | |
(video: HTMLVideoElement, stream: MediaStream) => { | |
video.srcObject = stream; | |
const handleCanPlay = () => { | |
setIsVideoReady(true); | |
playVideo(video); | |
}; | |
video.addEventListener("canplay", handleCanPlay, { once: true }); | |
return () => { | |
video.removeEventListener("canplay", handleCanPlay); | |
}; | |
}, | |
[playVideo], | |
); | |
useEffect(() => { | |
if (webcamStream && videoRef.current) { | |
const video = videoRef.current; | |
video.srcObject = null; | |
video.load(); | |
const cleanup = setupVideo(video, webcamStream); | |
return cleanup; | |
} | |
}, [webcamStream, setupVideo]); | |
const videoBlurState = useMemo(() => { | |
switch (appState) { | |
case "requesting-permission": | |
return "blur(20px) brightness(0.2) saturate(0.5)"; | |
case "welcome": | |
return "blur(12px) brightness(0.3) saturate(0.7)"; | |
case "loading": | |
return "blur(8px) brightness(0.4) saturate(0.8)"; | |
case "captioning": | |
return "none"; | |
default: | |
return "blur(20px) brightness(0.2) saturate(0.5)"; | |
} | |
}, [appState]); | |
return ( | |
<div className="App relative h-screen overflow-hidden"> | |
<div className="absolute inset-0 bg-gray-900" /> | |
{webcamStream && ( | |
<video | |
ref={videoRef} | |
autoPlay | |
muted | |
playsInline | |
className="absolute inset-0 w-full h-full object-cover transition-all duration-1000 ease-out" | |
style={{ | |
filter: videoBlurState, | |
opacity: isVideoReady ? 1 : 0, | |
}} | |
/> | |
)} | |
{appState !== "captioning" && <div className="absolute inset-0 bg-gray-900/80 backdrop-blur-sm" />} | |
{appState === "requesting-permission" && <WebcamPermissionDialog onPermissionGranted={handlePermissionGranted} />} | |
{appState === "welcome" && <WelcomeScreen onStart={handleStart} />} | |
{appState === "loading" && <LoadingScreen onComplete={handleLoadingComplete} />} | |
{appState === "captioning" && <CaptioningView videoRef={videoRef} />} | |
</div> | |
); | |
} | |