chore: implement feedback part 1

new form component added for document attachments with Zod validation and TRPC integration.
This commit is contained in:
Catalin Pit
2025-07-04 16:29:57 +03:00
parent 0b03bd3fce
commit 52b474d12b
5 changed files with 241 additions and 136 deletions

View File

@ -0,0 +1,184 @@
import { useEffect } from 'react';
import { zodResolver } from '@hookform/resolvers/zod';
import { Trash } from 'lucide-react';
import { useFieldArray, useForm } from 'react-hook-form';
import type { TDocument } from '@documenso/lib/types/document';
import { nanoid } from '@documenso/lib/universal/id';
import { AttachmentType } from '@documenso/prisma/generated/types';
import { trpc } from '@documenso/trpc/react';
import type { TSetDocumentAttachmentsSchema } from '@documenso/trpc/server/attachment-router/schema';
import { ZSetDocumentAttachmentsSchema } from '@documenso/trpc/server/attachment-router/schema';
import { Button } from '@documenso/ui/primitives/button';
import {
Dialog,
DialogContent,
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 { useToast } from '@documenso/ui/primitives/use-toast';
export type AttachmentFormProps = {
document: TDocument;
};
export const AttachmentForm = ({ document }: AttachmentFormProps) => {
const { toast } = useToast();
const { data: attachmentsData, refetch: refetchAttachments } =
trpc.attachment.getAttachments.useQuery({
documentId: document.id,
});
const { mutateAsync: setDocumentAttachments } =
trpc.attachment.setDocumentAttachments.useMutation();
const defaultAttachments = [
{
id: nanoid(12),
label: '',
url: '',
type: AttachmentType.LINK,
},
];
const form = useForm<TSetDocumentAttachmentsSchema>({
resolver: zodResolver(ZSetDocumentAttachmentsSchema),
defaultValues: {
documentId: document.id,
attachments: attachmentsData ?? defaultAttachments,
},
});
const {
fields: attachments,
append: appendAttachment,
remove: removeAttachment,
} = useFieldArray({
control: form.control,
name: 'attachments',
});
const onAddAttachment = () => {
appendAttachment({
id: nanoid(12),
label: '',
url: '',
type: AttachmentType.LINK,
});
};
const onRemoveAttachment = (index: number) => {
removeAttachment(index);
};
useEffect(() => {
if (attachmentsData) {
form.setValue('attachments', attachmentsData);
}
}, [attachmentsData]);
const onSubmit = async (data: TSetDocumentAttachmentsSchema) => {
try {
await setDocumentAttachments({
documentId: document.id,
attachments: data.attachments,
});
toast({
title: 'Attachment(s) updated',
description: 'The attachment(s) have been updated successfully',
});
await refetchAttachments();
} catch (error) {
console.error(error);
toast({
title: 'Something went wrong',
description: 'We encountered an unknown error while attempting to create the attachments.',
variant: 'destructive',
duration: 5000,
});
}
};
return (
<Dialog>
<DialogTrigger asChild>
<Button variant="outline" size="sm">
Add Attachment
</Button>
</DialogTrigger>
<DialogContent position="center">
<DialogHeader>
<DialogTitle>Attachments</DialogTitle>
</DialogHeader>
<Form {...form}>
<form onSubmit={form.handleSubmit(onSubmit)}>
<fieldset disabled={form.formState.isSubmitting} className="space-y-4">
{attachments.map((attachment, index) => (
<div key={attachment.id} className="flex items-end gap-2">
<FormField
control={form.control}
name={`attachments.${index}.label`}
render={({ field }) => (
<FormItem className="flex-1">
<FormLabel required>Label</FormLabel>
<FormControl>
<Input {...field} placeholder="Attachment label" />
</FormControl>
<FormMessage />
</FormItem>
)}
/>
<FormField
control={form.control}
name={`attachments.${index}.url`}
render={({ field }) => (
<FormItem className="flex-1">
<FormLabel required>URL</FormLabel>
<FormControl>
<Input {...field} placeholder="https://..." />
</FormControl>
<FormMessage />
</FormItem>
)}
/>
<Button
type="button"
variant="ghost"
size="sm"
onClick={() => onRemoveAttachment(index)}
>
<Trash className="h-4 w-4" />
</Button>
</div>
))}
</fieldset>
<DialogFooter className="mt-4">
<Button type="button" variant="outline" onClick={onAddAttachment}>
Add
</Button>
<Button type="submit">Save</Button>
</DialogFooter>
</form>
</Form>
</DialogContent>
</Dialog>
);
};

View File

@ -11,6 +11,7 @@ import { DocumentVisibility } from '@documenso/lib/types/document-visibility';
import { isDocumentCompleted } from '@documenso/lib/utils/document'; import { isDocumentCompleted } from '@documenso/lib/utils/document';
import { formatDocumentsPath } from '@documenso/lib/utils/teams'; import { formatDocumentsPath } from '@documenso/lib/utils/teams';
import { AttachmentForm } from '~/components/forms/attachment-form';
import { DocumentEditForm } from '~/components/general/document/document-edit-form'; import { DocumentEditForm } from '~/components/general/document/document-edit-form';
import { DocumentStatus } from '~/components/general/document/document-status'; import { DocumentStatus } from '~/components/general/document/document-status';
import { LegacyFieldWarningPopover } from '~/components/general/legacy-field-warning-popover'; import { LegacyFieldWarningPopover } from '~/components/general/legacy-field-warning-popover';
@ -133,11 +134,18 @@ export default function DocumentEditPage() {
</div> </div>
</div> </div>
{document.useLegacyFieldInsertion && ( <div>
<div> {document.useLegacyFieldInsertion ? (
<LegacyFieldWarningPopover type="document" documentId={document.id} /> <div className="flex flex-col items-end gap-2 sm:flex-row sm:items-start">
</div> <LegacyFieldWarningPopover type="document" documentId={document.id} />
)} <AttachmentForm document={document} />
</div>
) : (
<div>
<AttachmentForm document={document} />
</div>
)}
</div>
</div> </div>
<DocumentEditForm <DocumentEditForm

View File

@ -0,0 +1,39 @@
import { z } from 'zod';
import { AttachmentType } from '@documenso/prisma/generated/types';
export const ZGetDocumentAttachmentsSchema = z.object({
documentId: z.number(),
});
export const ZGetDocumentAttachmentsResponseSchema = z.array(
z.object({
id: z.string(),
label: z.string(),
url: z.string(),
type: z.nativeEnum(AttachmentType),
}),
);
export const ZSetDocumentAttachmentsSchema = z.object({
documentId: z.number(),
attachments: z.array(
z.object({
id: z.string(),
label: z.string().min(1, 'Label is required'),
url: z.string().url('Invalid URL'),
type: z.nativeEnum(AttachmentType),
}),
),
});
export type TSetDocumentAttachmentsSchema = z.infer<typeof ZSetDocumentAttachmentsSchema>;
export const ZSetDocumentAttachmentsResponseSchema = z.array(
z.object({
id: z.string(),
label: z.string(),
url: z.string(),
type: z.nativeEnum(AttachmentType),
}),
);

View File

@ -1,5 +1,6 @@
import { adminRouter } from './admin-router/router'; import { adminRouter } from './admin-router/router';
import { apiTokenRouter } from './api-token-router/router'; import { apiTokenRouter } from './api-token-router/router';
import { attachmentRouter } from './attachment-router/router';
import { authRouter } from './auth-router/router'; import { authRouter } from './auth-router/router';
import { billingRouter } from './billing/router'; import { billingRouter } from './billing/router';
import { documentRouter } from './document-router/router'; import { documentRouter } from './document-router/router';
@ -31,6 +32,7 @@ export const appRouter = router({
template: templateRouter, template: templateRouter,
webhook: webhookRouter, webhook: webhookRouter,
embeddingPresign: embeddingPresignRouter, embeddingPresign: embeddingPresignRouter,
attachment: attachmentRouter,
}); });
export type AppRouter = typeof appRouter; export type AppRouter = typeof appRouter;

View File

@ -1,4 +1,4 @@
import { useEffect, useId } from 'react'; import { useEffect } from 'react';
import { zodResolver } from '@hookform/resolvers/zod'; import { zodResolver } from '@hookform/resolvers/zod';
import { Trans, useLingui } from '@lingui/react/macro'; import { Trans, useLingui } from '@lingui/react/macro';
@ -10,8 +10,8 @@ import {
SendStatus, SendStatus,
TeamMemberRole, TeamMemberRole,
} from '@prisma/client'; } from '@prisma/client';
import { InfoIcon, Plus, Trash } from 'lucide-react'; import { InfoIcon } from 'lucide-react';
import { useFieldArray, useForm } from 'react-hook-form'; import { useForm } from 'react-hook-form';
import { match } from 'ts-pattern'; import { match } from 'ts-pattern';
import { useCurrentOrganisation } from '@documenso/lib/client-only/providers/organisation'; import { useCurrentOrganisation } from '@documenso/lib/client-only/providers/organisation';
@ -20,7 +20,6 @@ import { DOCUMENT_SIGNATURE_TYPES } from '@documenso/lib/constants/document';
import { SUPPORTED_LANGUAGES } from '@documenso/lib/constants/i18n'; import { SUPPORTED_LANGUAGES } from '@documenso/lib/constants/i18n';
import { DEFAULT_DOCUMENT_TIME_ZONE, TIME_ZONES } from '@documenso/lib/constants/time-zones'; import { DEFAULT_DOCUMENT_TIME_ZONE, TIME_ZONES } from '@documenso/lib/constants/time-zones';
import type { TDocument } from '@documenso/lib/types/document'; import type { TDocument } from '@documenso/lib/types/document';
import { nanoid } from '@documenso/lib/universal/id';
import { extractDocumentAuthMethods } from '@documenso/lib/utils/document-auth'; import { extractDocumentAuthMethods } from '@documenso/lib/utils/document-auth';
import { extractTeamSignatureSettings } from '@documenso/lib/utils/teams'; import { extractTeamSignatureSettings } from '@documenso/lib/utils/teams';
import { import {
@ -45,7 +44,6 @@ import {
AccordionItem, AccordionItem,
AccordionTrigger, AccordionTrigger,
} from '@documenso/ui/primitives/accordion'; } from '@documenso/ui/primitives/accordion';
import { Button } from '@documenso/ui/primitives/button';
import { import {
Form, Form,
FormControl, FormControl,
@ -93,7 +91,6 @@ export const AddSettingsFormPartial = ({
onSubmit, onSubmit,
}: AddSettingsFormProps) => { }: AddSettingsFormProps) => {
const { t } = useLingui(); const { t } = useLingui();
const initialId = useId();
const organisation = useCurrentOrganisation(); const organisation = useCurrentOrganisation();
@ -101,16 +98,6 @@ export const AddSettingsFormPartial = ({
documentAuth: document.authOptions, documentAuth: document.authOptions,
}); });
const defaultAttachments = [
{
id: '',
formId: initialId,
label: '',
url: '',
type: 'LINK',
},
];
const form = useForm<TAddSettingsFormSchema>({ const form = useForm<TAddSettingsFormSchema>({
resolver: zodResolver(ZAddSettingsFormSchema), resolver: zodResolver(ZAddSettingsFormSchema),
defaultValues: { defaultValues: {
@ -131,55 +118,15 @@ export const AddSettingsFormPartial = ({
language: document.documentMeta?.language ?? 'en', language: document.documentMeta?.language ?? 'en',
signatureTypes: extractTeamSignatureSettings(document.documentMeta), signatureTypes: extractTeamSignatureSettings(document.documentMeta),
}, },
attachments:
document.attachments?.map((attachment) => ({
...attachment,
id: String(attachment.id),
formId: String(attachment.id),
})) ?? defaultAttachments,
}, },
}); });
const {
fields: attachments,
append: appendAttachment,
remove: removeAttachment,
} = useFieldArray({
control: form.control,
name: 'attachments',
});
const onRemoveAttachment = (index: number) => {
const attachment = attachments[index];
const formStateIndex =
form.getValues('attachments')?.findIndex((a) => a.formId === attachment.formId) ?? -1;
if (formStateIndex !== -1) {
removeAttachment(formStateIndex);
const updatedAttachments =
form.getValues('attachments')?.filter((a) => a.formId !== attachment.formId) ?? [];
form.setValue('attachments', updatedAttachments);
}
};
const { stepIndex, currentStep, totalSteps, previousStep } = useStep(); const { stepIndex, currentStep, totalSteps, previousStep } = useStep();
const documentHasBeenSent = recipients.some( const documentHasBeenSent = recipients.some(
(recipient) => recipient.sendStatus === SendStatus.SENT, (recipient) => recipient.sendStatus === SendStatus.SENT,
); );
const onAddAttachment = () => {
appendAttachment({
id: nanoid(12),
formId: nanoid(12),
label: '',
url: '',
});
};
const canUpdateVisibility = match(currentTeamMemberRole) const canUpdateVisibility = match(currentTeamMemberRole)
.with(TeamMemberRole.ADMIN, () => true) .with(TeamMemberRole.ADMIN, () => true)
.with( .with(
@ -526,81 +473,6 @@ export const AddSettingsFormPartial = ({
</AccordionContent> </AccordionContent>
</AccordionItem> </AccordionItem>
</Accordion> </Accordion>
<Accordion type="multiple" className="mt-6">
<AccordionItem value="attachments" className="border-none">
<AccordionTrigger className="text-foreground mb-2 rounded border px-3 py-2 text-left hover:bg-neutral-200/30 hover:no-underline">
<Trans>Attachments</Trans>
</AccordionTrigger>
<AccordionContent className="text-muted-foreground -mx-1 px-1 pt-2 text-sm leading-relaxed">
<div className="flex flex-col space-y-6">
{attachments.map((attachment, index) => (
<div key={index} className="flex items-start gap-x-3">
<div className="flex-1">
<FormField
control={form.control}
name={`attachments.${index}.label`}
render={({ field }) => (
<FormItem>
<FormLabel className="flex flex-row items-center">
<Trans>Label</Trans>
</FormLabel>
<FormControl>
<Input className="bg-background" {...field} />
</FormControl>
<FormMessage />
</FormItem>
)}
/>
</div>
<div className="flex-1">
<FormField
control={form.control}
name={`attachments.${index}.url`}
render={({ field }) => (
<FormItem>
<FormLabel className="flex flex-row items-center">
<Trans>Location link</Trans>
</FormLabel>
<FormControl>
<Input className="bg-background" {...field} />
</FormControl>
<FormMessage />
</FormItem>
)}
/>
</div>
<div className="flex-none pt-8">
<button
onClick={() => onRemoveAttachment(index)}
className="hover:bg-muted rounded-md"
>
<Trash className="h-4 w-4" />
</button>
</div>
</div>
))}
<Button
type="button"
variant="outline"
className="w-full"
onClick={onAddAttachment}
>
<Plus className="mr-2 h-4 w-4" />
<Trans>Add Attachment</Trans>
</Button>
</div>
</AccordionContent>
</AccordionItem>
</Accordion>
</fieldset> </fieldset>
</Form> </Form>
</DocumentFlowFormContainerContent> </DocumentFlowFormContainerContent>