import { useMemo, useState } from 'react'; import { zodResolver } from '@hookform/resolvers/zod'; import { Trans } from '@lingui/react/macro'; import type { Field } from '@prisma/client'; import { RecipientRole } from '@prisma/client'; import { useForm } from 'react-hook-form'; import { match } from 'ts-pattern'; import { z } from 'zod'; import { fieldsContainUnsignedRequiredField } from '@documenso/lib/utils/advanced-fields-helpers'; import { Button } from '@documenso/ui/primitives/button'; import { Dialog, DialogContent, DialogFooter, 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 { DocumentSigningDisclosure } from '~/components/general/document-signing/document-signing-disclosure'; export type DocumentSigningCompleteDialogProps = { isSubmitting: boolean; documentTitle: string; fields: Field[]; fieldsValidated: () => void | Promise; onSignatureComplete: (nextSigner?: { name: string; email: string }) => void | Promise; role: RecipientRole; disabled?: boolean; allowDictateNextSigner?: boolean; defaultNextSigner?: { name: string; email: string; }; }; const ZNextSignerFormSchema = z.object({ name: z.string().min(1, 'Name is required'), email: z.string().email('Invalid email address'), }); type TNextSignerFormSchema = z.infer; export const DocumentSigningCompleteDialog = ({ isSubmitting, documentTitle, fields, fieldsValidated, onSignatureComplete, role, disabled = false, allowDictateNextSigner = false, defaultNextSigner, }: DocumentSigningCompleteDialogProps) => { const [showDialog, setShowDialog] = useState(false); const [isEditingNextSigner, setIsEditingNextSigner] = useState(false); const form = useForm({ resolver: allowDictateNextSigner ? zodResolver(ZNextSignerFormSchema) : undefined, defaultValues: { name: defaultNextSigner?.name ?? '', email: defaultNextSigner?.email ?? '', }, }); const isComplete = useMemo(() => !fieldsContainUnsignedRequiredField(fields), [fields]); const handleOpenChange = (open: boolean) => { if (form.formState.isSubmitting || !isComplete) { return; } if (open) { form.reset({ name: defaultNextSigner?.name ?? '', email: defaultNextSigner?.email ?? '', }); } setIsEditingNextSigner(false); setShowDialog(open); }; const onFormSubmit = async (data: TNextSignerFormSchema) => { console.log('data', data); console.log('form.formState.errors', form.formState.errors); try { if (allowDictateNextSigner && data.name && data.email) { await onSignatureComplete({ name: data.name, email: data.email }); } else { await onSignatureComplete(); } } catch (error) { console.error('Error completing signature:', error); } }; const isNextSignerValid = !allowDictateNextSigner || (form.watch('name') && form.watch('email')); return (
{match(role) .with(RecipientRole.VIEWER, () => Complete Viewing) .with(RecipientRole.SIGNER, () => Complete Signing) .with(RecipientRole.APPROVER, () => Complete Approval) .with(RecipientRole.CC, () => Complete Viewing) .with(RecipientRole.ASSISTANT, () => Complete Assisting) .exhaustive()}
{match(role) .with(RecipientRole.VIEWER, () => ( You are about to complete viewing " {documentTitle} ".
Are you sure?
)) .with(RecipientRole.SIGNER, () => ( You are about to complete signing " {documentTitle} ".
Are you sure?
)) .with(RecipientRole.APPROVER, () => ( You are about to complete approving{' '} "{documentTitle}" .
Are you sure?
)) .otherwise(() => ( You are about to complete viewing " {documentTitle} ".
Are you sure?
))}
{allowDictateNextSigner && (
{!isEditingNextSigner && (

The next recipient to sign this document will be{' '} {form.watch('name')} ( {form.watch('email')}).

)} {isEditingNextSigner && (
( Name )} /> ( Email )} />
)}
)}
); };