Presupuestador_web/client/src/app/settings/edit.tsx

280 lines
9.4 KiB
TypeScript
Raw Normal View History

2024-08-09 18:12:47 +00:00
import { ErrorOverlay, FormTextAreaField, LoadingOverlay } from "@/components";
2024-06-14 12:07:20 +00:00
import {
2024-06-17 16:54:30 +00:00
Alert,
AlertDescription,
AlertTitle,
2024-06-14 12:07:20 +00:00
Button,
Card,
CardContent,
CardDescription,
CardFooter,
CardHeader,
CardTitle,
2024-06-17 16:54:30 +00:00
Form,
2024-06-14 12:07:20 +00:00
} from "@/ui";
2024-06-17 16:54:30 +00:00
import { t } from "i18next";
import { AlertCircleIcon } from "lucide-react";
import { useMemo, useState } from "react";
2024-06-17 16:54:30 +00:00
import { SubmitHandler, useForm } from "react-hook-form";
import { Trans } from "react-i18next";
2024-06-14 12:07:20 +00:00
import { cn } from "@/lib/utils";
2024-08-09 18:12:47 +00:00
import { IUpdateProfile_Request_DTO } from "@shared/contexts";
2024-06-14 12:07:20 +00:00
import { Link } from "react-router-dom";
2024-08-09 18:12:47 +00:00
import { toast } from "react-toastify";
2024-06-17 16:54:30 +00:00
import { useSettings } from "./hooks";
2024-08-09 18:12:47 +00:00
type SettingsDataForm = IUpdateProfile_Request_DTO;
2024-06-14 12:07:20 +00:00
export const SettingsEditor = () => {
const [activeSection, setActiveSection] = useState("profile");
2024-08-09 18:12:47 +00:00
const { useOne, useUpdate } = useSettings();
2024-06-17 16:54:30 +00:00
2024-08-09 18:12:47 +00:00
const { data, status, error: queryError } = useOne();
2024-06-17 16:54:30 +00:00
2024-08-09 18:12:47 +00:00
const defaultValues = useMemo(
() => ({
2024-06-18 14:23:27 +00:00
contact_information: "",
default_payment_method: "",
default_notes: "",
default_legal_terms: "",
default_quote_validity: "",
2024-08-09 18:12:47 +00:00
}),
[]
);
const { mutate } = useUpdate();
const form = useForm<SettingsDataForm>({
mode: "onBlur",
values: data?.dealer,
defaultValues,
2024-06-17 16:54:30 +00:00
/*resolver: joiResolver(
Joi.object({
email: Joi.string()
.email({ tlds: { allow: false } })
.required(),
password: Joi.string().min(4).alphanum().required(),
}),
{
messages: SpanishJoiMessages,
}
),*/
});
const { formState, reset } = form;
2024-08-09 18:12:47 +00:00
const { isSubmitting } = formState;
2024-06-17 16:54:30 +00:00
const onSubmit: SubmitHandler<SettingsDataForm> = async (data) => {
2024-08-09 18:12:47 +00:00
console.log(data);
mutate(data, {
onError: (error) => {
console.debug(error);
toast.error(error.message);
//alert(error.message);
},
//onSettled: () => {},
onSuccess: () => {
reset({}, { keepValues: true, keepIsValid: true });
2024-08-09 18:12:47 +00:00
toast("Ajustes guardados");
//clear();
},
});
2024-06-17 16:54:30 +00:00
};
2024-08-09 18:12:47 +00:00
if (isSubmitting) {
return <LoadingOverlay title='Guardando ajustes' />;
}
if (status === "error") {
return <ErrorOverlay errorMessage={queryError.message} />;
}
if (status !== "success") {
return <LoadingOverlay />;
}
2024-06-14 12:07:20 +00:00
return (
2024-08-09 18:12:47 +00:00
<>
<div className='grid w-full max-w-6xl gap-2 mx-auto'>
<h1 className='text-2xl font-semibold md:text-3xl'>
<Trans i18nKey='settings.edit.title' />
</h1>
</div>
<Form {...form}>
<form onSubmit={form.handleSubmit(onSubmit)}>
<div className='mx-auto grid w-full max-w-6xl items-start gap-6 md:grid-cols-[180px_1fr] lg:grid-cols-[250px_1fr]'>
{form.formState.errors.root?.message && (
<Alert variant='destructive'>
<AlertCircleIcon className='w-4 h-4' />
<AlertTitle>
<Trans i18nKey='common.error' />
</AlertTitle>
<AlertDescription>{form.formState.errors.root?.message}</AlertDescription>
</Alert>
)}
<nav className='grid gap-4 text-sm text-muted-foreground'>
<Link
to='#'
onClick={() => setActiveSection("profile")}
className={activeSection === "profile" ? "font-semibold text-primary" : ""}
>
2024-08-09 18:12:47 +00:00
<Trans i18nKey='settings.edit.tabs.profile' />
</Link>
<Link
to='#'
onClick={() => setActiveSection("quotes")}
className={activeSection === "quotes" ? "font-semibold text-primary" : ""}
>
2024-08-09 18:12:47 +00:00
<Trans i18nKey='settings.edit.tabs.quotes' />
</Link>
<Link
to='#'
onClick={() => setActiveSection("legal")}
className={activeSection === "legal" ? "font-semibold text-primary" : ""}
>
2024-08-09 18:12:47 +00:00
<Trans i18nKey='settings.edit.tabs.legal' />
</Link>
</nav>
<div className={cn("grid gap-6", activeSection === "profile" ? "visible" : "hidden")}>
2024-08-09 18:12:47 +00:00
<Card>
<CardHeader>
<CardTitle>
<Trans i18nKey='settings.form_fields.contact_information.label' />
</CardTitle>
<CardDescription>
<Trans i18nKey='settings.form_fields.contact_information.desc' />
</CardDescription>
</CardHeader>
<CardContent>
<FormTextAreaField
autoSize
2024-08-09 18:12:47 +00:00
placeholder={t("settings.form_fields.contact_information.placeholder")}
{...form.register("contact_information", {
required: true,
})}
errors={form.formState.errors}
/>
</CardContent>
<CardFooter className='px-6 py-4 border-t'>
<Button>
<Trans i18nKey='common.save' />
</Button>
</CardFooter>
</Card>
</div>
<div className={cn("grid gap-6", activeSection === "quotes" ? "visible" : "hidden")}>
2024-08-09 18:12:47 +00:00
<Card>
<CardHeader>
<CardTitle>
<Trans i18nKey='settings.form_fields.default_payment_method.label' />
</CardTitle>
<CardDescription>
<Trans i18nKey='settings.form_fields.default_payment_method.desc' />
</CardDescription>
</CardHeader>
<CardContent>
<FormTextAreaField
autoSize
2024-08-09 18:12:47 +00:00
placeholder={t("settings.form_fields.default_payment_method.placeholder")}
{...form.register("default_payment_method", {
required: true,
})}
errors={form.formState.errors}
/>
</CardContent>
<CardFooter className='px-6 py-4 border-t'>
<Button>
<Trans i18nKey='common.save' />
</Button>
</CardFooter>
</Card>
<Card>
<CardHeader>
<CardTitle>
<Trans i18nKey='settings.form_fields.default_quote_validity.label' />
</CardTitle>
<CardDescription>
<Trans i18nKey='settings.form_fields.default_quote_validity.desc' />
</CardDescription>
</CardHeader>
<CardContent>
<FormTextAreaField
autoSize
2024-08-09 18:12:47 +00:00
placeholder={t("settings.form_fields.default_quote_validity.placeholder")}
{...form.register("default_quote_validity", {
required: true,
})}
errors={form.formState.errors}
/>
</CardContent>
<CardFooter className='px-6 py-4 border-t'>
<Button>
<Trans i18nKey='common.save' />
</Button>
</CardFooter>
</Card>
<Card>
<CardHeader>
<CardTitle>
<Trans i18nKey='settings.form_fields.default_notes.label' />
</CardTitle>
<CardDescription>
<Trans i18nKey='settings.form_fields.default_notes.desc' />
</CardDescription>
</CardHeader>
<CardContent>
<FormTextAreaField
autoSize
2024-08-09 18:12:47 +00:00
placeholder={t("settings.form_fields.default_notes.placeholder")}
{...form.register("default_notes", {
required: true,
})}
errors={form.formState.errors}
/>
</CardContent>
<CardFooter className='px-6 py-4 border-t'>
<Button>
<Trans i18nKey='common.save' />
</Button>
</CardFooter>
</Card>
</div>
<div className={cn("grid gap-6", activeSection === "legal" ? "visible" : "hidden")}>
2024-08-09 18:12:47 +00:00
<Card>
<CardHeader>
<CardTitle>
<Trans i18nKey='settings.form_fields.default_legal_terms.label' />
</CardTitle>
<CardDescription>
<Trans i18nKey='settings.form_fields.default_legal_terms.desc' />
</CardDescription>
</CardHeader>
<CardContent>
<FormTextAreaField
autoSize
2024-08-09 18:12:47 +00:00
placeholder={t("settings.form_fields.default_legal_terms.placeholder")}
{...form.register("default_legal_terms", {
required: true,
})}
errors={form.formState.errors}
/>
</CardContent>
<CardFooter className='px-6 py-4 border-t'>
<Button>
<Trans i18nKey='common.save' />
</Button>
</CardFooter>
</Card>
</div>
2024-06-17 16:54:30 +00:00
</div>
2024-08-09 18:12:47 +00:00
</form>
</Form>
</>
2024-06-14 12:07:20 +00:00
);
};