import { useMemo, useState } from 'react'; import { zodResolver } from '@hookform/resolvers/zod'; import { Trans, useLingui } from '@lingui/react/macro'; import type { Field, Recipient } from '@prisma/client'; import { RecipientRole } from '@prisma/client'; import { useForm } from 'react-hook-form'; import { match } from 'ts-pattern'; import { z } from 'zod'; import { AppError, AppErrorCode } from '@documenso/lib/errors/app-error'; import { type TRecipientAccessAuth, ZDocumentAccessAuthSchema, } from '@documenso/lib/types/document-auth'; import { fieldsContainUnsignedRequiredField } from '@documenso/lib/utils/advanced-fields-helpers'; import { Button } from '@documenso/ui/primitives/button'; import { Dialog, DialogContent, DialogDescription, DialogFooter, DialogHeader, DialogTitle, DialogTrigger, } from '@documenso/ui/primitives/dialog'; import { Form, FormControl, FormField, FormItem, FormLabel, FormMessage, } from '@documenso/ui/primitives/form/form'; import { Input } from '@documenso/ui/primitives/input'; import { AccessAuth2FAForm } from '~/components/general/document-signing/access-auth-2fa-form'; import { DocumentSigningDisclosure } from '~/components/general/document-signing/document-signing-disclosure'; import { useRequiredDocumentSigningAuthContext } from './document-signing-auth-provider'; export type DocumentSigningCompleteDialogProps = { isSubmitting: boolean; documentTitle: string; fields: Field[]; fieldsValidated: () => void | Promise; onSignatureComplete: ( nextSigner?: { name: string; email: string }, accessAuthOptions?: TRecipientAccessAuth, directRecipient?: { name: string; email: string }, ) => void | Promise; recipient: Pick; disabled?: boolean; allowDictateNextSigner?: boolean; defaultNextSigner?: { name: string; email: string; }; directTemplatePayload?: { name: string; email: string; }; buttonSize?: 'sm' | 'lg'; position?: 'start' | 'end' | 'center'; }; const ZNextSignerFormSchema = z.object({ name: z.string().min(1, 'Name is required'), email: z.string().email('Invalid email address'), accessAuthOptions: ZDocumentAccessAuthSchema.optional(), }); type TNextSignerFormSchema = z.infer; const ZDirectRecipientFormSchema = z.object({ name: z.string(), email: z.string().email('Invalid email address'), }); type TDirectRecipientFormSchema = z.infer; export const DocumentSigningCompleteDialog = ({ isSubmitting, documentTitle, fields, fieldsValidated, onSignatureComplete, recipient, disabled = false, allowDictateNextSigner = false, directTemplatePayload, defaultNextSigner, buttonSize = 'lg', position, }: DocumentSigningCompleteDialogProps) => { const { t } = useLingui(); const [showDialog, setShowDialog] = useState(false); const [showTwoFactorForm, setShowTwoFactorForm] = useState(false); const [twoFactorValidationError, setTwoFactorValidationError] = useState(null); const { derivedRecipientAccessAuth } = useRequiredDocumentSigningAuthContext(); const form = useForm({ resolver: allowDictateNextSigner ? zodResolver(ZNextSignerFormSchema) : undefined, defaultValues: { name: defaultNextSigner?.name ?? '', email: defaultNextSigner?.email ?? '', }, }); const directRecipientForm = useForm({ resolver: zodResolver(ZDirectRecipientFormSchema), defaultValues: { name: directTemplatePayload?.name ?? '', email: directTemplatePayload?.email ?? '', }, }); const isComplete = useMemo(() => !fieldsContainUnsignedRequiredField(fields), [fields]); const completionRequires2FA = useMemo( () => derivedRecipientAccessAuth.includes('TWO_FACTOR_AUTH'), [derivedRecipientAccessAuth], ); const handleOpenChange = (open: boolean) => { if (form.formState.isSubmitting || !isComplete) { return; } if (open) { form.reset({ name: defaultNextSigner?.name ?? '', email: defaultNextSigner?.email ?? '', }); } setShowDialog(open); }; const onFormSubmit = async (data: TNextSignerFormSchema) => { try { let directRecipient: { name: string; email: string } | undefined; if (directTemplatePayload && !directTemplatePayload.email) { const isFormValid = await directRecipientForm.trigger(); if (!isFormValid) { return; } directRecipient = directRecipientForm.getValues(); } // Check if 2FA is required if (completionRequires2FA && !data.accessAuthOptions) { setShowTwoFactorForm(true); return; } const nextSigner = allowDictateNextSigner && data.name && data.email ? { name: data.name, email: data.email } : undefined; await onSignatureComplete(nextSigner, data.accessAuthOptions, directRecipient); } catch (error) { const err = AppError.parseError(error); if (AppErrorCode.TWO_FACTOR_AUTH_FAILED === err.code) { // This was a 2FA validation failure - show the 2FA dialog again with error form.setValue('accessAuthOptions', undefined); setTwoFactorValidationError('Invalid verification code. Please try again.'); setShowTwoFactorForm(true); return; } } }; const onTwoFactorFormSubmit = (validatedAuthOptions: TRecipientAccessAuth) => { form.setValue('accessAuthOptions', validatedAuthOptions); setShowTwoFactorForm(false); setTwoFactorValidationError(null); // Now trigger the form submission with auth options void form.handleSubmit(onFormSubmit)(); }; return ( Are you sure?
{match(recipient.role) .with(RecipientRole.VIEWER, () => ( You are about to complete viewing the following document )) .with(RecipientRole.SIGNER, () => ( You are about to complete signing the following document )) .with(RecipientRole.APPROVER, () => ( You are about to complete approving the following document )) .with(RecipientRole.ASSISTANT, () => ( You are about to complete assisting the following document )) .with(RecipientRole.CC, () => null) .exhaustive()}

{documentTitle}

{!showTwoFactorForm && ( <>
{directTemplatePayload && !directTemplatePayload.email && (
( Your Name )} /> ( Your Email )} />
)}
{allowDictateNextSigner && defaultNextSigner && (
( Next Recipient Name )} /> ( Next Recipient Email )} />
)}
)} {showTwoFactorForm && ( )}
); };