"use client" import type React from "react" import { useState, useEffect } from "react" import { Button } from "./ui/button" import { Input } from "./ui/input" import { Label } from "./ui/label" import { Checkbox } from "./ui/checkbox" import { Lock, User, AlertCircle, Server, Shield } from "lucide-react" import { getApiUrl } from "../lib/api-config" import Image from "next/image" interface LoginProps { onLogin: () => void } export function Login({ onLogin }: LoginProps) { const [username, setUsername] = useState("") const [password, setPassword] = useState("") const [totpCode, setTotpCode] = useState("") const [requiresTotp, setRequiresTotp] = useState(false) const [rememberMe, setRememberMe] = useState(false) const [error, setError] = useState("") const [loading, setLoading] = useState(false) useEffect(() => { const savedUsername = localStorage.getItem("proxmenux-saved-username") const savedPassword = localStorage.getItem("proxmenux-saved-password") if (savedUsername && savedPassword) { setUsername(savedUsername) setPassword(savedPassword) setRememberMe(true) } }, []) const handleLogin = async (e: React.FormEvent) => { e.preventDefault() setError("") if (!username || !password) { setError("Please enter username and password") return } if (requiresTotp && !totpCode) { setError("Please enter your 2FA code") return } setLoading(true) try { const response = await fetch(getApiUrl("/api/auth/login"), { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ username, password, totp_token: totpCode || undefined, // Include 2FA code if provided }), }) const data = await response.json() if (data.requires_totp) { setRequiresTotp(true) setLoading(false) return } if (!response.ok) { throw new Error(data.message || "Login failed") } localStorage.setItem("proxmenux-auth-token", data.token) if (rememberMe) { localStorage.setItem("proxmenux-saved-username", username) localStorage.setItem("proxmenux-saved-password", password) } else { localStorage.removeItem("proxmenux-saved-username") localStorage.removeItem("proxmenux-saved-password") } onLogin() } catch (err) { setError(err instanceof Error ? err.message : "Login failed") } finally { setLoading(false) } } return (
ProxMenux Logo { const target = e.target as HTMLImageElement target.style.display = "none" const fallback = target.parentElement?.querySelector(".fallback-icon") if (fallback) { fallback.classList.remove("hidden") } }} />

ProxMenux Monitor

Sign in to access your dashboard

{error && (

{error}

)} {!requiresTotp ? ( <>
setUsername(e.target.value)} className="pl-10 text-base" disabled={loading} autoComplete="username" />
setPassword(e.target.value)} className="pl-10 text-base" disabled={loading} autoComplete="current-password" />
setRememberMe(checked as boolean)} disabled={loading} />
) : (

Two-Factor Authentication

Enter the 6-digit code from your authentication app

setTotpCode(e.target.value.replace(/\D/g, "").slice(0, 6))} className="text-center text-lg tracking-widest font-mono text-base" maxLength={6} disabled={loading} autoComplete="one-time-code" autoFocus />

You can also use a backup code (format: XXXX-XXXX)

)}

ProxMenux Monitor v1.0.1

) }