| | 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(() => { |
| | |
| | if (hasStartedLoading || isLoading || isLoaded) return; |
| |
|
| | const loadModelAndProgress = async () => { |
| | setHasStartedLoading(true); |
| |
|
| | try { |
| | setCurrentStep("Checking WebGPU support..."); |
| | setProgress(5); |
| |
|
| | |
| | if (!navigator.gpu) { |
| | setCurrentStep("WebGPU not available in this browser"); |
| | setIsError(true); |
| | return; |
| | } |
| |
|
| | |
| | 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); |
| |
|
| | |
| | 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]); |
| |
|
| | |
| | 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> |
| | ); |
| | } |
| |
|