mirror of
https://github.com/blakeblackshear/frigate.git
synced 2026-06-25 22:01:51 +03:00
Some checks failed
CI / AMD64 Build (push) Has been cancelled
CI / ARM Build (push) Has been cancelled
CI / Jetson Jetpack 6 (push) Has been cancelled
CI / AMD64 Extra Build (push) Has been cancelled
CI / ARM Extra Build (push) Has been cancelled
CI / Synaptics Build (push) Has been cancelled
CI / Assemble and push default build (push) Has been cancelled
* remove redundant per-view toasters in settings * add variants to standardize dialog footer button layouts * remove text-md this class name compiles to nothing in tailwind. we used to add it to prevent iOS from zooming when focusing on an input, but that is now solved via the viewport meta in index.html * make wizard footers consistent with dialog footers * consistent destructive button style remove text-white from individual buttons and add it to the variant
171 lines
5.5 KiB
TypeScript
171 lines
5.5 KiB
TypeScript
"use client";
|
|
|
|
import * as React from "react";
|
|
|
|
import { baseUrl } from "../../api/baseUrl";
|
|
import { cn } from "@/lib/utils";
|
|
import { Input } from "@/components/ui/input";
|
|
import { Button } from "@/components/ui/button";
|
|
import ActivityIndicator from "@/components/indicators/activity-indicator";
|
|
import axios, { AxiosError } from "axios";
|
|
import { Toaster } from "@/components/ui/sonner";
|
|
import { toast } from "sonner";
|
|
import {
|
|
Form,
|
|
FormControl,
|
|
FormField,
|
|
FormItem,
|
|
FormLabel,
|
|
} from "@/components/ui/form";
|
|
import { useForm } from "react-hook-form";
|
|
import { zodResolver } from "@hookform/resolvers/zod";
|
|
import { z } from "zod";
|
|
import { AuthContext } from "@/context/auth-context";
|
|
import { useTranslation } from "react-i18next";
|
|
import useSWR from "swr";
|
|
import { LuExternalLink } from "react-icons/lu";
|
|
import { useDocDomain } from "@/hooks/use-doc-domain";
|
|
import { Card, CardContent } from "@/components/ui/card";
|
|
|
|
interface UserAuthFormProps extends React.HTMLAttributes<HTMLDivElement> {}
|
|
|
|
export function UserAuthForm({ className, ...props }: UserAuthFormProps) {
|
|
const { t } = useTranslation(["components/auth", "common"]);
|
|
const { getLocaleDocUrl } = useDocDomain();
|
|
const [isLoading, setIsLoading] = React.useState<boolean>(false);
|
|
const { login } = React.useContext(AuthContext);
|
|
|
|
// need to use local fetcher because useSWR default fetcher is not set up in this context
|
|
const fetcher = (path: string) => axios.get(path).then((res) => res.data);
|
|
const { data } = useSWR("/auth/first_time_login", fetcher);
|
|
const showFirstTimeLink = data?.admin_first_time_login === true;
|
|
|
|
const formSchema = z.object({
|
|
user: z.string().min(1, t("form.errors.usernameRequired")),
|
|
password: z.string().min(1, t("form.errors.passwordRequired")),
|
|
});
|
|
|
|
const form = useForm<z.infer<typeof formSchema>>({
|
|
resolver: zodResolver(formSchema),
|
|
mode: "onChange",
|
|
defaultValues: { user: "", password: "" },
|
|
});
|
|
|
|
const onSubmit = async (values: z.infer<typeof formSchema>) => {
|
|
setIsLoading(true);
|
|
try {
|
|
await axios.post(
|
|
"/login",
|
|
{
|
|
user: values.user,
|
|
password: values.password,
|
|
},
|
|
{
|
|
headers: { "X-CSRF-TOKEN": 1 },
|
|
},
|
|
);
|
|
const profileRes = await axios.get("/profile", { withCredentials: true });
|
|
login({
|
|
username: profileRes.data.username,
|
|
role: profileRes.data.role || "viewer",
|
|
});
|
|
window.location.href = baseUrl;
|
|
} catch (error) {
|
|
if (axios.isAxiosError(error)) {
|
|
const err = error as AxiosError;
|
|
if (err.response?.status === 429) {
|
|
toast.error(t("form.errors.rateLimit"), {
|
|
position: "top-center",
|
|
});
|
|
} else if (err.response?.status === 401) {
|
|
toast.error(t("form.errors.loginFailed"), {
|
|
position: "top-center",
|
|
});
|
|
} else {
|
|
toast.error(t("form.errors.unknownError"), {
|
|
position: "top-center",
|
|
});
|
|
}
|
|
} else {
|
|
toast.error(t("form.errors.webUnknownError"), {
|
|
position: "top-center",
|
|
});
|
|
}
|
|
|
|
setIsLoading(false);
|
|
}
|
|
};
|
|
|
|
return (
|
|
<div className={cn("grid gap-6", className)} {...props}>
|
|
<Form {...form}>
|
|
<form onSubmit={form.handleSubmit(onSubmit)} className="space-y-4">
|
|
<FormField
|
|
name="user"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>{t("form.user")}</FormLabel>
|
|
<FormControl>
|
|
<Input
|
|
className="w-full border border-input bg-background p-2 hover:bg-accent hover:text-accent-foreground dark:[color-scheme:dark]"
|
|
autoFocus
|
|
autoCapitalize="off"
|
|
autoCorrect="off"
|
|
spellCheck="false"
|
|
{...field}
|
|
/>
|
|
</FormControl>
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<FormField
|
|
name="password"
|
|
render={({ field }) => (
|
|
<FormItem>
|
|
<FormLabel>{t("form.password")}</FormLabel>
|
|
<FormControl>
|
|
<Input
|
|
className="w-full border border-input bg-background p-2 hover:bg-accent hover:text-accent-foreground dark:[color-scheme:dark]"
|
|
type="password"
|
|
{...field}
|
|
/>
|
|
</FormControl>
|
|
</FormItem>
|
|
)}
|
|
/>
|
|
<div className="flex flex-row gap-2 pt-5">
|
|
<Button
|
|
variant="select"
|
|
disabled={isLoading}
|
|
className="flex flex-1"
|
|
aria-label={t("form.login")}
|
|
>
|
|
{isLoading && <ActivityIndicator className="mr-2 h-4 w-4" />}
|
|
{t("form.login")}
|
|
</Button>
|
|
</div>
|
|
</form>
|
|
</Form>
|
|
{showFirstTimeLink && (
|
|
<Card className="mt-4 p-4 text-center text-sm">
|
|
<CardContent className="p-2">
|
|
<p className="mb-2 text-primary-variant">
|
|
{t("form.firstTimeLogin")}
|
|
</p>
|
|
<a
|
|
href={getLocaleDocUrl("configuration/authentication#onboarding")}
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
className="inline-flex items-center text-primary"
|
|
>
|
|
{t("readTheDocumentation", { ns: "common" })}
|
|
<LuExternalLink className="ml-2 size-3" />
|
|
</a>
|
|
</CardContent>
|
|
</Card>
|
|
)}
|
|
<Toaster />
|
|
</div>
|
|
);
|
|
}
|