Files
sign/apps/remix/app/components/tables/documents-table-action-dropdown.tsx

237 lines
7.9 KiB
TypeScript
Raw Normal View History

2023-10-10 08:25:58 +05:30
import { useState } from 'react';
2025-01-02 15:33:37 +11:00
import { msg } from '@lingui/core/macro';
2024-08-27 20:34:39 +09:00
import { useLingui } from '@lingui/react';
2025-01-02 15:33:37 +11:00
import { Trans } from '@lingui/react/macro';
import type { Document, Recipient, Team, User } from '@prisma/client';
import { DocumentStatus, RecipientRole } from '@prisma/client';
2023-08-29 14:30:57 +10:00
import {
CheckCircle,
2023-08-29 14:30:57 +10:00
Copy,
Download,
Edit,
EyeIcon,
2023-09-21 00:51:02 +00:00
Loader,
2023-08-29 14:30:57 +10:00
MoreHorizontal,
MoveRight,
2023-08-29 14:30:57 +10:00
Pencil,
Share,
Trash2,
} from 'lucide-react';
2025-01-02 15:33:37 +11:00
import { Link } from 'react-router';
2023-08-29 14:30:57 +10:00
2024-01-02 04:38:35 +00:00
import { downloadPDF } from '@documenso/lib/client-only/download-pdf';
2025-01-02 15:33:37 +11:00
import { useSession } from '@documenso/lib/client-only/providers/session';
import { isDocumentCompleted } from '@documenso/lib/utils/document';
import { formatDocumentsPath } from '@documenso/lib/utils/teams';
2023-09-21 00:51:02 +00:00
import { trpc as trpcClient } from '@documenso/trpc/client';
import { DocumentShareButton } from '@documenso/ui/components/document/document-share-button';
2023-08-29 14:30:57 +10:00
import {
DropdownMenu,
DropdownMenuContent,
DropdownMenuItem,
DropdownMenuLabel,
DropdownMenuTrigger,
} from '@documenso/ui/primitives/dropdown-menu';
2024-01-02 04:52:15 +00:00
import { useToast } from '@documenso/ui/primitives/use-toast';
2023-09-21 00:51:02 +00:00
2025-01-02 15:33:37 +11:00
import { DocumentDeleteDialog } from '~/components/dialogs/document-delete-dialog';
import { DocumentDuplicateDialog } from '~/components/dialogs/document-duplicate-dialog';
import { DocumentMoveDialog } from '~/components/dialogs/document-move-dialog';
import { DocumentResendDialog } from '~/components/dialogs/document-resend-dialog';
import { DocumentRecipientLinkCopyDialog } from '~/components/general/document/document-recipient-link-copy-dialog';
import { useOptionalCurrentTeam } from '~/providers/team';
2023-10-10 08:25:58 +05:30
2025-01-02 15:33:37 +11:00
export type DocumentsTableActionDropdownProps = {
2023-08-29 14:30:57 +10:00
row: Document & {
2025-01-13 13:41:53 +11:00
user: Pick<User, 'id' | 'name' | 'email'>;
recipients: Recipient[];
team: Pick<Team, 'id' | 'url'> | null;
2023-08-29 14:30:57 +10:00
};
};
2025-01-02 15:33:37 +11:00
export const DocumentsTableActionDropdown = ({ row }: DocumentsTableActionDropdownProps) => {
const { user } = useSession();
const team = useOptionalCurrentTeam();
2024-01-02 04:52:15 +00:00
const { toast } = useToast();
2024-08-27 20:34:39 +09:00
const { _ } = useLingui();
2023-10-10 08:25:58 +05:30
const [isDeleteDialogOpen, setDeleteDialogOpen] = useState(false);
2023-11-08 09:25:44 +00:00
const [isDuplicateDialogOpen, setDuplicateDialogOpen] = useState(false);
const [isMoveDialogOpen, setMoveDialogOpen] = useState(false);
2023-10-10 08:25:58 +05:30
2025-01-02 15:33:37 +11:00
const recipient = row.recipients.find((recipient) => recipient.email === user.email);
2023-08-29 14:30:57 +10:00
2025-01-02 15:33:37 +11:00
const isOwner = row.user.id === user.id;
2023-08-29 14:30:57 +10:00
// const isRecipient = !!recipient;
const isDraft = row.status === DocumentStatus.DRAFT;
2024-11-06 21:34:06 +09:00
const isPending = row.status === DocumentStatus.PENDING;
const isComplete = isDocumentCompleted(row.status);
2023-08-29 14:30:57 +10:00
// const isSigned = recipient?.signingStatus === SigningStatus.SIGNED;
const isCurrentTeamDocument = team && row.team?.url === team.url;
const canManageDocument = Boolean(isOwner || isCurrentTeamDocument);
const documentsPath = formatDocumentsPath(team?.url);
2023-08-29 14:30:57 +10:00
const onDownloadClick = async () => {
2024-01-02 04:38:35 +00:00
try {
const document = !recipient
? await trpcClient.document.getDocumentById.query({
documentId: row.id,
})
: await trpcClient.document.getDocumentByToken.query({
token: recipient.token,
});
2024-01-02 04:38:35 +00:00
const documentData = document?.documentData;
if (!documentData) {
return;
}
await downloadPDF({ documentData, fileName: row.title });
} catch (err) {
toast({
2024-08-27 20:34:39 +09:00
title: _(msg`Something went wrong`),
description: _(msg`An error occurred while downloading your document.`),
2024-01-02 04:38:35 +00:00
variant: 'destructive',
});
}
2023-08-29 14:30:57 +10:00
};
2025-01-13 13:41:53 +11:00
const nonSignedRecipients = row.recipients.filter((item) => item.signingStatus !== 'SIGNED');
2023-10-06 22:54:24 +00:00
2023-08-29 14:30:57 +10:00
return (
<DropdownMenu>
<DropdownMenuTrigger data-testid="document-table-action-btn">
2023-09-24 14:45:50 +10:00
<MoreHorizontal className="text-muted-foreground h-5 w-5" />
2023-08-29 14:30:57 +10:00
</DropdownMenuTrigger>
<DropdownMenuContent className="w-52" align="start" forceMount>
2024-08-27 20:34:39 +09:00
<DropdownMenuLabel>
<Trans>Action</Trans>
</DropdownMenuLabel>
2023-08-29 14:30:57 +10:00
{!isDraft && recipient && recipient?.role !== RecipientRole.CC && (
<DropdownMenuItem disabled={!recipient || isComplete} asChild>
2025-01-02 15:33:37 +11:00
<Link to={`/sign/${recipient?.token}`}>
{recipient?.role === RecipientRole.VIEWER && (
<>
<EyeIcon className="mr-2 h-4 w-4" />
2024-08-27 20:34:39 +09:00
<Trans>View</Trans>
</>
)}
{recipient?.role === RecipientRole.SIGNER && (
<>
<Pencil className="mr-2 h-4 w-4" />
2024-08-27 20:34:39 +09:00
<Trans>Sign</Trans>
</>
)}
{recipient?.role === RecipientRole.APPROVER && (
<>
<CheckCircle className="mr-2 h-4 w-4" />
2024-08-27 20:34:39 +09:00
<Trans>Approve</Trans>
</>
)}
</Link>
</DropdownMenuItem>
)}
2023-08-29 14:30:57 +10:00
<DropdownMenuItem disabled={!canManageDocument || isComplete} asChild>
2025-01-02 15:33:37 +11:00
<Link to={`${documentsPath}/${row.id}/edit`}>
2023-08-29 14:30:57 +10:00
<Edit className="mr-2 h-4 w-4" />
2024-08-27 20:34:39 +09:00
<Trans>Edit</Trans>
2023-08-29 14:30:57 +10:00
</Link>
</DropdownMenuItem>
<DropdownMenuItem disabled={!isComplete} onClick={onDownloadClick}>
<Download className="mr-2 h-4 w-4" />
2024-08-27 20:34:39 +09:00
<Trans>Download</Trans>
2023-08-29 14:30:57 +10:00
</DropdownMenuItem>
2023-12-21 21:37:33 +11:00
<DropdownMenuItem onClick={() => setDuplicateDialogOpen(true)}>
2023-08-29 14:30:57 +10:00
<Copy className="mr-2 h-4 w-4" />
2024-08-27 20:34:39 +09:00
<Trans>Duplicate</Trans>
2023-08-29 14:30:57 +10:00
</DropdownMenuItem>
{/* We don't want to allow teams moving documents across at the moment. */}
{!team && !row.teamId && (
<DropdownMenuItem onClick={() => setMoveDialogOpen(true)}>
<MoveRight className="mr-2 h-4 w-4" />
2024-08-27 20:34:39 +09:00
<Trans>Move to Team</Trans>
</DropdownMenuItem>
)}
{/* No point displaying this if there's no functionality. */}
{/* <DropdownMenuItem disabled>
2023-08-29 14:30:57 +10:00
<XCircle className="mr-2 h-4 w-4" />
Void
</DropdownMenuItem> */}
2023-08-29 14:30:57 +10:00
2025-02-17 22:46:36 +11:00
<DropdownMenuItem onClick={() => setDeleteDialogOpen(true)}>
2023-08-29 14:30:57 +10:00
<Trash2 className="mr-2 h-4 w-4" />
2024-08-27 20:34:39 +09:00
{canManageDocument ? _(msg`Delete`) : _(msg`Hide`)}
2023-08-29 14:30:57 +10:00
</DropdownMenuItem>
2024-08-27 20:34:39 +09:00
<DropdownMenuLabel>
<Trans>Share</Trans>
</DropdownMenuLabel>
2023-08-29 14:30:57 +10:00
2024-11-06 21:34:06 +09:00
{canManageDocument && (
<DocumentRecipientLinkCopyDialog
2025-01-13 13:41:53 +11:00
recipients={row.recipients}
2024-11-06 21:34:06 +09:00
trigger={
<DropdownMenuItem disabled={!isPending} asChild onSelect={(e) => e.preventDefault()}>
<div>
<Copy className="mr-2 h-4 w-4" />
<Trans>Signing Links</Trans>
</div>
</DropdownMenuItem>
}
/>
)}
2025-01-02 15:33:37 +11:00
<DocumentResendDialog document={row} recipients={nonSignedRecipients} />
2023-08-29 14:30:57 +10:00
<DocumentShareButton
documentId={row.id}
token={isOwner ? undefined : recipient?.token}
trigger={({ loading, disabled }) => (
<DropdownMenuItem disabled={disabled || isDraft} onSelect={(e) => e.preventDefault()}>
<div className="flex items-center">
{loading ? <Loader className="mr-2 h-4 w-4" /> : <Share className="mr-2 h-4 w-4" />}
2024-08-27 20:34:39 +09:00
<Trans>Share Signing Card</Trans>
</div>
</DropdownMenuItem>
2023-09-21 00:51:02 +00:00
)}
/>
2023-08-29 14:30:57 +10:00
</DropdownMenuContent>
2023-10-10 08:25:58 +05:30
2025-01-02 15:33:37 +11:00
<DocumentDeleteDialog
id={row.id}
status={row.status}
documentTitle={row.title}
open={isDeleteDialogOpen}
onOpenChange={setDeleteDialogOpen}
teamId={team?.id}
canManageDocument={canManageDocument}
/>
2025-01-02 15:33:37 +11:00
<DocumentMoveDialog
documentId={row.id}
open={isMoveDialogOpen}
onOpenChange={setMoveDialogOpen}
/>
2025-01-02 15:33:37 +11:00
<DocumentDuplicateDialog
id={row.id}
open={isDuplicateDialogOpen}
onOpenChange={setDuplicateDialogOpen}
/>
2023-08-29 14:30:57 +10:00
</DropdownMenu>
);
};