import { zodResolver } from "@hookform/resolvers/zod"; import { useForm } from "react-hook-form"; import { Link, redirect, useNavigation, useSubmit } from "react-router"; import { createAuthSession, getOptionalAuth, login } from "../lib/auth.server"; import { loginSchema, type LoginSchema } from "../lib/schemas"; import type { Route } from "./+types/login"; export function meta({}: Route.MetaArgs) { return [{ title: "Login | The Biergarten App" }]; } export async function loader({ request }: Route.LoaderArgs) { const auth = await getOptionalAuth(request); if (auth) throw redirect("/dashboard"); return null; } export async function action({ request }: Route.ActionArgs) { const formData = await request.formData(); const result = loginSchema.safeParse({ username: formData.get("username"), password: formData.get("password"), }); if (!result.success) { return { error: result.error.issues[0].message }; } try { const payload = await login(result.data.username, result.data.password); return createAuthSession(payload, "/dashboard"); } catch (err) { return { error: err instanceof Error ? err.message : "Login failed." }; } } export default function Login({ actionData }: Route.ComponentProps) { const navigation = useNavigation(); const submit = useSubmit(); const isSubmitting = navigation.state === "submitting"; const { register, handleSubmit, formState: { errors }, } = useForm({ resolver: zodResolver(loginSchema) }); const onSubmit = handleSubmit((data) => { submit(data, { method: "post" }); }); return (

Login

Sign in to your Biergarten account

{actionData?.error && (
{actionData.error}
)}
Username {errors.username && (

{errors.username.message}

)}
Password {errors.password && (

{errors.password.message}

)}
New here?
Create an account ← Back to home
); }