Spaces:
Running
Running
import { useEffect, useState } from "react"; | |
import { useVLMContext } from "../context/useVLMContext"; | |
import GlassContainer from "./GlassContainer"; | |
import { GLASS_EFFECTS } from "../constants"; | |
interface LoadingScreenProps { | |
onComplete: () => void; | |
} | |
export default function LoadingScreen({ onComplete }: LoadingScreenProps) { | |
const [progress, setProgress] = useState(0); | |
const [currentStep, setCurrentStep] = useState("Initializing..."); | |
const [isError, setIsError] = useState(false); | |
const [hasStartedLoading, setHasStartedLoading] = useState(false); | |
const { loadModel, isLoaded, isLoading } = useVLMContext(); | |
useEffect(() => { | |
// Prevent multiple loading attempts | |
if (hasStartedLoading || isLoading || isLoaded) return; | |
const loadModelAndProgress = async () => { | |
setHasStartedLoading(true); | |
try { | |
setCurrentStep("Checking WebGPU support..."); | |
setProgress(5); | |
// Check for WebGPU support first | |
if (!navigator.gpu) { | |
setCurrentStep("WebGPU not available in this browser"); | |
setIsError(true); | |
return; | |
} | |
// Load the actual AI model | |
await loadModel((message) => { | |
setCurrentStep(message); | |
if (message.includes("Loading processor")) { | |
setProgress(10); | |
} else if (message.includes("Processor loaded")) { | |
setProgress(20); | |
} else if (message.includes("Model loaded")) { | |
setProgress(80); | |
} | |
}); | |
setCurrentStep("Ready to start!"); | |
setProgress(100); | |
// Small delay before completing | |
await new Promise((resolve) => setTimeout(resolve, 300)); | |
onComplete(); | |
} catch (error) { | |
console.error("Error loading model:", error); | |
setCurrentStep(`Error loading model: ${error instanceof Error ? error.message : String(error)}`); | |
setIsError(true); | |
} | |
}; | |
loadModelAndProgress(); | |
}, [hasStartedLoading, isLoading, isLoaded, loadModel, onComplete]); | |
// Handle case where model is already loaded | |
useEffect(() => { | |
if (isLoaded && !hasStartedLoading) { | |
setProgress(100); | |
setCurrentStep("Model already loaded!"); | |
setTimeout(onComplete, 300); | |
} | |
}, [isLoaded, hasStartedLoading, onComplete]); | |
return ( | |
<div className="absolute inset-0 text-white flex items-center justify-center p-8" style={{ opacity: 1 }}> | |
<GlassContainer | |
className="max-w-md w-full rounded-3xl shadow-2xl" | |
bgColor={isError ? GLASS_EFFECTS.COLORS.ERROR_BG : GLASS_EFFECTS.COLORS.DEFAULT_BG} | |
> | |
<div className="p-8 text-center space-y-8"> | |
<div className="space-y-4"> | |
<div className="w-16 h-16 mx-auto"> | |
{isError ? ( | |
<div className="w-16 h-16 rounded-full bg-red-500/20 flex items-center justify-center"> | |
<svg className="w-8 h-8 text-red-400" fill="currentColor" viewBox="0 0 20 20"> | |
<path | |
fillRule="evenodd" | |
d="M10 18a8 8 0 100-16 8 8 0 000 16zM8.707 7.293a1 1 0 00-1.414 1.414L8.586 10l-1.293 1.293a1 1 0 101.414 1.414L10 11.414l1.293 1.293a1 1 0 001.414-1.414L11.414 10l1.293-1.293a1 1 0 00-1.414-1.414L10 8.586 8.707 7.293z" | |
clipRule="evenodd" | |
/> | |
</svg> | |
</div> | |
) : ( | |
<div className="animate-spin rounded-full h-16 w-16 border-4 border-blue-500 border-t-transparent"></div> | |
)} | |
</div> | |
<h2 className="text-2xl font-bold text-gray-100">{isError ? "Loading Failed" : "Loading AI Model"}</h2> | |
<p className={`${isError ? "text-red-400" : "text-gray-400"}`}>{currentStep}</p> | |
</div> | |
{!isError && ( | |
<div className="space-y-2"> | |
<div className="w-full bg-gray-800/50 rounded-full h-3 overflow-hidden backdrop-blur-sm border border-gray-700/30"> | |
<div | |
className="h-full bg-gradient-to-r from-blue-500 to-blue-600 rounded-full transition-all duration-300 ease-out" | |
style={{ width: `${progress}%` }} | |
/> | |
</div> | |
<p className="text-sm text-gray-500">{Math.round(progress)}% complete</p> | |
</div> | |
)} | |
{isError && ( | |
<div className="mt-4"> | |
<button | |
onClick={() => window.location.reload()} | |
className="px-6 py-2 bg-red-600 hover:bg-red-700 rounded-lg text-white font-medium transition-colors" | |
> | |
Reload Page | |
</button> | |
</div> | |
)} | |
</div> | |
</GlassContainer> | |
</div> | |
); | |
} | |