import React, { useEffect, useState } from "react"; // hooks // types import { useRouter } from "next/router"; // icons import { CircleCheck, XCircle } from "lucide-react"; // ui import { Button, Input } from "@plane/ui"; // helpers import { API_BASE_URL } from "@/helpers/common.helper"; // services import { AuthService } from "@/services/authentication.service"; import useTimer from "hooks/use-timer"; import useToast from "hooks/use-toast"; import { IEmailCheckData } from "types/auth"; import { EAuthModes } from "./root"; type Props = { email: string; mode: EAuthModes; handleEmailClear: () => void; submitButtonText: string; }; type TUniqueCodeFormValues = { email: string; code: string; }; const defaultValues: TUniqueCodeFormValues = { email: "", code: "", }; // services const authService = new AuthService(); export const UniqueCodeForm: React.FC<Props> = (props) => { const { email, mode, handleEmailClear, submitButtonText } = props; // states const [uniqueCodeFormData, setUniqueCodeFormData] = useState<TUniqueCodeFormValues>({ ...defaultValues, email }); const [isRequestingNewCode, setIsRequestingNewCode] = useState(false); const [csrfToken, setCsrfToken] = useState<string | undefined>(undefined); // router const router = useRouter(); const { next_path } = router.query; // toast alert const { setToastAlert } = useToast(); // timer const { timer: resendTimerCode, setTimer: setResendCodeTimer } = useTimer(30); const handleFormChange = (key: keyof TUniqueCodeFormValues, value: string) => setUniqueCodeFormData((prev) => ({ ...prev, [key]: value })); const handleSendNewCode = async (email: string) => { const payload: IEmailCheckData = { email, }; await authService .generateUniqueCode(payload) .then(() => { setResendCodeTimer(30); setToastAlert({ type: "success", title: "Success!", message: "A new unique code has been sent to your email.", }); handleFormChange("code", ""); }) .catch((err) => setToastAlert({ type: "error", title: "Error!", message: err?.error ?? "Something went wrong. Please try again.", }) ); }; const handleRequestNewCode = async () => { setIsRequestingNewCode(true); await handleSendNewCode(uniqueCodeFormData.email) .then(() => setResendCodeTimer(30)) .finally(() => setIsRequestingNewCode(false)); }; useEffect(() => { if (csrfToken === undefined) authService.requestCSRFToken().then((data) => data?.csrf_token && setCsrfToken(data.csrf_token)); }, [csrfToken]); useEffect(() => { setIsRequestingNewCode(true); handleSendNewCode(email) .then(() => setResendCodeTimer(30)) .finally(() => setIsRequestingNewCode(false)); // eslint-disable-next-line react-hooks/exhaustive-deps }, []); const isRequestNewCodeDisabled = isRequestingNewCode || resendTimerCode > 0; return ( <form className="mx-auto mt-5 space-y-4 w-5/6 sm:w-96" method="POST" action={`${API_BASE_URL}/auth/spaces/${mode === EAuthModes.SIGN_IN ? "magic-sign-in" : "magic-sign-up"}/`} > <input type="hidden" name="csrfmiddlewaretoken" value={csrfToken} /> <input type="hidden" name="next_path" value={next_path} /> <div className="space-y-1"> <label className="text-sm font-medium text-onboarding-text-300" htmlFor="email"> Email </label> <div className="relative flex items-center rounded-md bg-onboarding-background-200"> <Input id="email" name="email" type="email" value={uniqueCodeFormData.email} onChange={(e) => handleFormChange("email", e.target.value)} // FIXME: // hasError={Boolean(errors.email)} placeholder="name@company.com" className="h-[46px] w-full border border-onboarding-border-100 pr-12 placeholder:text-onboarding-text-400" disabled /> {uniqueCodeFormData.email.length > 0 && ( <XCircle className="absolute right-3 h-5 w-5 stroke-custom-text-400 hover:cursor-pointer" onClick={handleEmailClear} /> )} </div> </div> <div className="space-y-1"> <label className="text-sm font-medium text-onboarding-text-300" htmlFor="code"> Unique code </label> <Input name="code" value={uniqueCodeFormData.code} onChange={(e) => handleFormChange("code", e.target.value)} // FIXME: // hasError={Boolean(errors.code)} placeholder="gets-sets-flys" className="h-[46px] w-full border border-onboarding-border-100 !bg-onboarding-background-200 pr-12 placeholder:text-onboarding-text-400" autoFocus /> <div className="flex w-full items-center justify-between px-1 text-xs"> <p className="flex items-center gap-1 font-medium text-green-700"> <CircleCheck height={12} width={12} /> Paste the code sent to your email </p> <button type="button" onClick={handleRequestNewCode} className={`${ isRequestNewCodeDisabled ? "text-onboarding-text-400" : "font-medium text-custom-primary-300 hover:text-custom-primary-200" }`} disabled={isRequestNewCodeDisabled} > {resendTimerCode > 0 ? `Resend in ${resendTimerCode}s` : isRequestingNewCode ? "Requesting new code" : "Resend"} </button> </div> </div> <Button type="submit" variant="primary" className="w-full" size="lg" // disabled={!isValid || hasEmailChanged} loading={isRequestingNewCode} > {isRequestingNewCode ? "Sending code" : submitButtonText} </Button> </form> ); };