Xenova's picture
Xenova HF Staff
Upload 20 files
be54278 verified
import { useVoxtral } from "./VoxtralContext";
import { THEME } from "../constants";
import {
AppGridBackground,
ErrorMessageBox,
useMountedTransition,
} from "./SharedUI";
export const LoadingPage = () => {
const { loadingProgress, loadingMessage, error } = useVoxtral();
const mounted = useMountedTransition();
const progressClamped = Math.min(100, Math.max(0, loadingProgress));
const isError = !!error;
return (
<AppGridBackground className="min-h-screen flex items-center justify-center p-8">
<div
className={`max-w-md w-full backdrop-blur-sm rounded-sm border shadow-xl transition-all duration-700 transform ${mounted ? "opacity-100 translate-y-0" : "opacity-0 translate-y-4"}`}
style={{
backgroundColor: `${THEME.beigeLight}F2`,
borderColor: THEME.beigeDark,
}}
>
<div
className={`h-1 w-full transition-colors duration-300 ${isError ? "bg-[var(--mistral-red)]" : "bg-[var(--mistral-orange)]"}`}
/>
<div className="p-8 space-y-8">
<div className="flex justify-center">
{isError ? (
<div
className="w-20 h-20 rounded-full flex items-center justify-center border"
style={{
backgroundColor: `${THEME.errorRed}1A`,
borderColor: `${THEME.errorRed}33`,
}}
>
<svg
className="w-10 h-10"
style={{ color: THEME.errorRed }}
fill="none"
viewBox="0 0 24 24"
stroke="currentColor"
strokeWidth={2}
>
<path
strokeLinecap="round"
strokeLinejoin="round"
d="M12 9v3.75m9-.75a9 9 0 1 1-18 0 9 9 0 0 1 18 0Zm-9 3.75h.008v.008H12v-.008Z"
/>
</svg>
</div>
) : (
<div className="relative">
<div
className="w-20 h-20 border-4 rounded-full animate-spin"
style={{
borderColor: THEME.beigeDark,
borderTopColor: THEME.mistralOrange,
}}
/>
<div className="absolute inset-0 flex items-center justify-center">
<div
className="w-2 h-2 rounded-full animate-pulse"
style={{ backgroundColor: THEME.mistralOrange }}
/>
</div>
</div>
)}
</div>
<div className="text-center space-y-2">
<h2
className="text-2xl font-bold tracking-tight"
style={{ color: THEME.textBlack }}
>
{isError ? "Initialization Failed" : "Loading Model"}
</h2>
<p className="text-sm text-gray-500 font-mono uppercase tracking-widest">
{isError ? "Voxtral-Mini-4B-Realtime" : loadingMessage}
</p>
</div>
{!isError && (
<div className="space-y-4">
<div className="flex justify-between text-xs font-mono font-bold text-gray-500">
<span>PROGRESS</span>
<span>{Math.round(progressClamped)}%</span>
</div>
<div
className="w-full rounded-full h-4 overflow-hidden border"
style={{
backgroundColor: `${THEME.beigeDark}80`,
borderColor: THEME.beigeDark,
}}
>
<div
className="h-full progress-stripe transition-all duration-500 ease-out"
style={{
width: `${progressClamped}%`,
backgroundColor: THEME.mistralOrange,
}}
/>
</div>
<div
className="bg-white border p-3 rounded-sm"
style={{ borderColor: THEME.beigeDark }}
>
<div className="flex items-center space-x-2">
<div className="w-2 h-2 bg-green-500 rounded-full animate-pulse" />
<p className="font-mono text-xs text-gray-600 truncate">
{`> ${loadingMessage}`}
</p>
</div>
</div>
</div>
)}
{isError && (
<div className="space-y-4">
<ErrorMessageBox
className="border p-4 rounded text-left"
message={error}
/>
<button
onClick={() => window.location.reload()}
className="w-full py-3 text-white font-bold transition-colors shadow-lg hover:bg-black cursor-pointer"
style={{ backgroundColor: THEME.textBlack }}
>
RELOAD APPLICATION
</button>
</div>
)}
</div>
</div>
</AppGridBackground>
);
};