| import { useEffect, useState } from "react"; |
| import { useNavigate, Link } from "react-router-dom"; |
| import { supabase } from "@/integrations/supabase/client"; |
| import { useAuth } from "@/hooks/useAuth"; |
| import { Button } from "@/components/ui/button"; |
| import { Input } from "@/components/ui/input"; |
| import { Label } from "@/components/ui/label"; |
| import { Card } from "@/components/ui/card"; |
| import { useToast } from "@/hooks/use-toast"; |
| import { Sparkles, Loader2 } from "lucide-react"; |
|
|
| export default function Auth() { |
| const { user, loading } = useAuth(); |
| const navigate = useNavigate(); |
| const { toast } = useToast(); |
| const [mode, setMode] = useState<"signin" | "signup">("signin"); |
| const [email, setEmail] = useState(""); |
| const [password, setPassword] = useState(""); |
| const [displayName, setDisplayName] = useState(""); |
| const [submitting, setSubmitting] = useState(false); |
|
|
| useEffect(() => { |
| if (!loading && user) navigate("/app", { replace: true }); |
| }, [user, loading, navigate]); |
|
|
| const handleSubmit = async (e: React.FormEvent) => { |
| e.preventDefault(); |
| setSubmitting(true); |
| try { |
| if (mode === "signup") { |
| const { error } = await supabase.auth.signUp({ |
| email, |
| password, |
| options: { |
| emailRedirectTo: `${window.location.origin}/app`, |
| data: { display_name: displayName || email.split("@")[0] }, |
| }, |
| }); |
| if (error) throw error; |
| toast({ title: "Account created", description: "You're signed in." }); |
| } else { |
| const { error } = await supabase.auth.signInWithPassword({ email, password }); |
| if (error) throw error; |
| } |
| } catch (err: any) { |
| toast({ |
| title: "Authentication failed", |
| description: err.message ?? "Something went wrong.", |
| variant: "destructive", |
| }); |
| } finally { |
| setSubmitting(false); |
| } |
| }; |
|
|
| return ( |
| <main className="min-h-screen flex items-center justify-center px-4 bg-background"> |
| <div className="w-full max-w-md"> |
| <div className="flex items-center gap-2 justify-center mb-8"> |
| <div className="h-9 w-9 rounded-lg bg-gradient-primary flex items-center justify-center shadow-glow"> |
| <Sparkles className="h-5 w-5 text-primary-foreground" /> |
| </div> |
| <h1 className="text-2xl font-semibold tracking-tight">Source.AI</h1> |
| </div> |
| |
| <Card className="p-6 bg-card border-border"> |
| <h2 className="text-xl font-semibold mb-1"> |
| {mode === "signin" ? "Welcome back" : "Create your account"} |
| </h2> |
| <p className="text-sm text-muted-foreground mb-6"> |
| {mode === "signin" ? "Sign in to continue to your workspace." : "Start turning content into study assets."} |
| </p> |
| |
| <form onSubmit={handleSubmit} className="space-y-4"> |
| {mode === "signup" && ( |
| <div className="space-y-1.5"> |
| <Label htmlFor="name">Display name</Label> |
| <Input id="name" value={displayName} onChange={(e) => setDisplayName(e.target.value)} placeholder="Your name" /> |
| </div> |
| )} |
| <div className="space-y-1.5"> |
| <Label htmlFor="email">Email</Label> |
| <Input id="email" type="email" value={email} onChange={(e) => setEmail(e.target.value)} required placeholder="you@example.com" /> |
| </div> |
| <div className="space-y-1.5"> |
| <Label htmlFor="password">Password</Label> |
| <Input id="password" type="password" value={password} onChange={(e) => setPassword(e.target.value)} required minLength={6} placeholder="••••••••" /> |
| </div> |
| |
| <Button type="submit" className="w-full" disabled={submitting}> |
| {submitting && <Loader2 className="h-4 w-4 mr-2 animate-spin" />} |
| {mode === "signin" ? "Sign in" : "Create account"} |
| </Button> |
| </form> |
| |
| <div className="mt-4 text-center text-sm text-muted-foreground"> |
| {mode === "signin" ? ( |
| <> |
| No account?{" "} |
| <button className="text-primary hover:underline" onClick={() => setMode("signup")}>Sign up</button> |
| </> |
| ) : ( |
| <> |
| Already have one?{" "} |
| <button className="text-primary hover:underline" onClick={() => setMode("signin")}>Sign in</button> |
| </> |
| )} |
| </div> |
| </Card> |
|
|
| <p className="text-center text-xs text-muted-foreground mt-6"> |
| <Link to="/" className="hover:underline">← Back to home</Link> |
| </p> |
| </div> |
| </main> |
| ); |
| } |
|
|