|
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>
|
|
);
|
|
}
|
|
|