2022-07-27 23:28:21 +00:00
|
|
|
import { zodResolver } from "@hookform/resolvers/zod";
|
2022-09-29 14:40:37 +00:00
|
|
|
import classNames from "classnames";
|
2022-12-07 21:47:02 +00:00
|
|
|
import { signIn } from "next-auth/react";
|
2022-07-27 23:28:21 +00:00
|
|
|
import { useRouter } from "next/router";
|
|
|
|
import { Controller, FormProvider, useForm } from "react-hook-form";
|
|
|
|
import * as z from "zod";
|
|
|
|
|
|
|
|
import { isPasswordValid } from "@calcom/lib/auth";
|
2022-12-07 21:47:02 +00:00
|
|
|
import { WEBSITE_URL } from "@calcom/lib/constants";
|
2022-07-27 23:28:21 +00:00
|
|
|
import { useLocale } from "@calcom/lib/hooks/useLocale";
|
2022-12-07 21:47:02 +00:00
|
|
|
import { EmailField, Label, PasswordField, TextField } from "@calcom/ui";
|
2022-07-27 23:28:21 +00:00
|
|
|
|
|
|
|
const SetupFormStep1 = (props: { setIsLoading: (val: boolean) => void }) => {
|
|
|
|
const router = useRouter();
|
|
|
|
const { t } = useLocale();
|
|
|
|
|
|
|
|
const formSchema = z.object({
|
2022-07-29 13:23:54 +00:00
|
|
|
username: z
|
|
|
|
.string()
|
|
|
|
.refine((val) => val.trim().length >= 1, { message: t("at_least_characters", { count: 1 }) }),
|
2022-07-27 23:28:21 +00:00
|
|
|
email_address: z.string().email({ message: t("enter_valid_email") }),
|
|
|
|
full_name: z.string().min(3, t("at_least_characters", { count: 3 })),
|
2022-07-29 13:23:54 +00:00
|
|
|
password: z.string().superRefine((data, ctx) => {
|
2022-11-21 08:03:45 +00:00
|
|
|
const isStrict = true;
|
|
|
|
const result = isPasswordValid(data, true, isStrict);
|
2022-07-29 13:23:54 +00:00
|
|
|
Object.keys(result).map((key: string) => {
|
|
|
|
if (!result[key as keyof typeof result]) {
|
|
|
|
ctx.addIssue({
|
|
|
|
code: z.ZodIssueCode.custom,
|
|
|
|
path: [key],
|
2022-08-03 16:59:40 +00:00
|
|
|
message: key,
|
2022-07-29 13:23:54 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
2022-07-27 23:28:21 +00:00
|
|
|
}),
|
|
|
|
});
|
|
|
|
|
|
|
|
const formMethods = useForm<{
|
|
|
|
username: string;
|
|
|
|
email_address: string;
|
|
|
|
full_name: string;
|
|
|
|
password: string;
|
|
|
|
}>({
|
|
|
|
resolver: zodResolver(formSchema),
|
|
|
|
});
|
|
|
|
|
|
|
|
const onError = () => {
|
|
|
|
props.setIsLoading(false);
|
|
|
|
};
|
|
|
|
|
|
|
|
const onSubmit = formMethods.handleSubmit(async (data: z.infer<typeof formSchema>) => {
|
|
|
|
props.setIsLoading(true);
|
|
|
|
const response = await fetch("/api/auth/setup", {
|
|
|
|
method: "POST",
|
|
|
|
body: JSON.stringify({
|
2022-07-29 13:23:54 +00:00
|
|
|
username: data.username.trim(),
|
2022-07-27 23:28:21 +00:00
|
|
|
full_name: data.full_name,
|
|
|
|
email_address: data.email_address.toLowerCase(),
|
|
|
|
password: data.password,
|
|
|
|
}),
|
|
|
|
headers: {
|
|
|
|
"Content-Type": "application/json",
|
|
|
|
},
|
|
|
|
});
|
|
|
|
if (response.status === 200) {
|
2022-12-07 21:47:02 +00:00
|
|
|
await signIn("credentials", {
|
|
|
|
redirect: false,
|
|
|
|
callbackUrl: "/",
|
|
|
|
email: data.email_address.toLowerCase(),
|
|
|
|
password: data.password,
|
|
|
|
});
|
|
|
|
router.replace(`/auth/setup?step=2&category=calendar`);
|
2022-07-27 23:28:21 +00:00
|
|
|
} else {
|
|
|
|
router.replace("/auth/setup");
|
|
|
|
}
|
|
|
|
}, onError);
|
|
|
|
|
2022-12-07 21:47:02 +00:00
|
|
|
const longWebsiteUrl = WEBSITE_URL.length > 30;
|
2022-09-29 14:40:37 +00:00
|
|
|
|
2022-07-27 23:28:21 +00:00
|
|
|
return (
|
|
|
|
<FormProvider {...formMethods}>
|
2022-11-16 13:41:40 +00:00
|
|
|
<form id="wizard-step-1" name="wizard-step-1" className="space-y-4" onSubmit={onSubmit}>
|
2022-07-27 23:28:21 +00:00
|
|
|
<div>
|
|
|
|
<Controller
|
|
|
|
name="username"
|
|
|
|
control={formMethods.control}
|
|
|
|
render={({ field: { onBlur, onChange, value } }) => (
|
2022-09-29 14:40:37 +00:00
|
|
|
<>
|
|
|
|
<Label htmlFor="username" className={classNames(longWebsiteUrl && "mb-0")}>
|
|
|
|
<span className="block">{t("username")}</span>
|
|
|
|
{longWebsiteUrl && (
|
|
|
|
<small className="items-centerpx-3 mt-2 inline-flex rounded-t-md border border-b-0 border-gray-300 bg-gray-100 py-1 px-3 text-gray-500">
|
|
|
|
{process.env.NEXT_PUBLIC_WEBSITE_URL}
|
|
|
|
</small>
|
|
|
|
)}
|
|
|
|
</Label>
|
|
|
|
<TextField
|
|
|
|
addOnLeading={
|
|
|
|
!longWebsiteUrl && (
|
2023-01-04 07:38:45 +00:00
|
|
|
<span className="inline-flex items-center rounded-none px-3 text-sm text-gray-500">
|
2022-09-29 14:40:37 +00:00
|
|
|
{process.env.NEXT_PUBLIC_WEBSITE_URL}/
|
|
|
|
</span>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
id="username"
|
|
|
|
labelSrOnly={true}
|
|
|
|
value={value || ""}
|
|
|
|
className={classNames("my-0", longWebsiteUrl && "rounded-t-none")}
|
|
|
|
onBlur={onBlur}
|
|
|
|
name="username"
|
|
|
|
onChange={async (e) => {
|
|
|
|
onChange(e.target.value);
|
|
|
|
formMethods.setValue("username", e.target.value);
|
|
|
|
await formMethods.trigger("username");
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</>
|
2022-07-27 23:28:21 +00:00
|
|
|
)}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
<Controller
|
|
|
|
name="full_name"
|
|
|
|
control={formMethods.control}
|
|
|
|
render={({ field: { onBlur, onChange, value } }) => (
|
|
|
|
<TextField
|
|
|
|
value={value || ""}
|
|
|
|
onBlur={onBlur}
|
|
|
|
onChange={async (e) => {
|
|
|
|
onChange(e.target.value);
|
|
|
|
formMethods.setValue("full_name", e.target.value);
|
|
|
|
await formMethods.trigger("full_name");
|
|
|
|
}}
|
|
|
|
color={formMethods.formState.errors.full_name ? "warn" : ""}
|
|
|
|
type="text"
|
|
|
|
name="full_name"
|
|
|
|
autoCapitalize="none"
|
|
|
|
autoComplete="name"
|
|
|
|
autoCorrect="off"
|
|
|
|
className="my-0"
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
<Controller
|
|
|
|
name="email_address"
|
|
|
|
control={formMethods.control}
|
|
|
|
render={({ field: { onBlur, onChange, value } }) => (
|
|
|
|
<EmailField
|
|
|
|
value={value || ""}
|
|
|
|
onBlur={onBlur}
|
|
|
|
onChange={async (e) => {
|
|
|
|
onChange(e.target.value);
|
|
|
|
formMethods.setValue("email_address", e.target.value);
|
|
|
|
await formMethods.trigger("email_address");
|
|
|
|
}}
|
|
|
|
className="my-0"
|
|
|
|
name="email_address"
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
<Controller
|
|
|
|
name="password"
|
|
|
|
control={formMethods.control}
|
|
|
|
render={({ field: { onBlur, onChange, value } }) => (
|
|
|
|
<PasswordField
|
|
|
|
value={value || ""}
|
|
|
|
onBlur={onBlur}
|
|
|
|
onChange={async (e) => {
|
|
|
|
onChange(e.target.value);
|
|
|
|
formMethods.setValue("password", e.target.value);
|
|
|
|
await formMethods.trigger("password");
|
|
|
|
}}
|
2022-11-21 08:03:45 +00:00
|
|
|
hintErrors={["caplow", "admin_min", "num"]}
|
2022-07-27 23:28:21 +00:00
|
|
|
name="password"
|
|
|
|
className="my-0"
|
|
|
|
autoComplete="off"
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</form>
|
|
|
|
</FormProvider>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2022-12-07 21:47:02 +00:00
|
|
|
export default SetupFormStep1;
|