mirror of
https://github.com/documenso/documenso.git
synced 2025-11-09 20:12:31 +10:00
206 lines
6.3 KiB
TypeScript
206 lines
6.3 KiB
TypeScript
import { useMemo, useState } from 'react';
|
|
|
|
import { msg } from '@lingui/core/macro';
|
|
import { useLingui } from '@lingui/react/macro';
|
|
import { Trans } from '@lingui/react/macro';
|
|
import { EnvelopeType } from '@prisma/client';
|
|
import { ErrorCode as DropzoneErrorCode, type FileRejection } from 'react-dropzone';
|
|
import { useNavigate } from 'react-router';
|
|
import { match } from 'ts-pattern';
|
|
|
|
import { useLimits } from '@documenso/ee/server-only/limits/provider/client';
|
|
import { useCurrentOrganisation } from '@documenso/lib/client-only/providers/organisation';
|
|
import { useSession } from '@documenso/lib/client-only/providers/session';
|
|
import { APP_DOCUMENT_UPLOAD_SIZE_LIMIT } from '@documenso/lib/constants/app';
|
|
import { TIME_ZONES } from '@documenso/lib/constants/time-zones';
|
|
import { AppError, AppErrorCode } from '@documenso/lib/errors/app-error';
|
|
import { formatDocumentsPath, formatTemplatesPath } from '@documenso/lib/utils/teams';
|
|
import { trpc } from '@documenso/trpc/react';
|
|
import type { TCreateEnvelopePayload } from '@documenso/trpc/server/envelope-router/create-envelope.types';
|
|
import { cn } from '@documenso/ui/lib/utils';
|
|
import { DocumentDropzone } from '@documenso/ui/primitives/document-upload';
|
|
import {
|
|
Tooltip,
|
|
TooltipContent,
|
|
TooltipProvider,
|
|
TooltipTrigger,
|
|
} from '@documenso/ui/primitives/tooltip';
|
|
import { useToast } from '@documenso/ui/primitives/use-toast';
|
|
|
|
import { useCurrentTeam } from '~/providers/team';
|
|
|
|
export type EnvelopeUploadButtonProps = {
|
|
className?: string;
|
|
type: EnvelopeType;
|
|
folderId?: string;
|
|
};
|
|
|
|
/**
|
|
* Upload an envelope
|
|
*/
|
|
export const EnvelopeUploadButton = ({ className, type, folderId }: EnvelopeUploadButtonProps) => {
|
|
const { t } = useLingui();
|
|
const { toast } = useToast();
|
|
const { user } = useSession();
|
|
|
|
const team = useCurrentTeam();
|
|
|
|
const navigate = useNavigate();
|
|
const organisation = useCurrentOrganisation();
|
|
|
|
const userTimezone = TIME_ZONES.find(
|
|
(timezone) => timezone === Intl.DateTimeFormat().resolvedOptions().timeZone,
|
|
);
|
|
|
|
const { quota, remaining, refreshLimits, maximumEnvelopeItemCount } = useLimits();
|
|
|
|
const [isLoading, setIsLoading] = useState(false);
|
|
|
|
const { mutateAsync: createEnvelope } = trpc.envelope.create.useMutation();
|
|
|
|
const disabledMessage = useMemo(() => {
|
|
if (organisation.subscription && remaining.documents === 0) {
|
|
return msg`Document upload disabled due to unpaid invoices`;
|
|
}
|
|
|
|
if (remaining.documents === 0) {
|
|
return msg`You have reached your document limit.`;
|
|
}
|
|
|
|
if (!user.emailVerified) {
|
|
return msg`Verify your email to upload documents.`;
|
|
}
|
|
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
}, [remaining.documents, user.emailVerified, team]);
|
|
|
|
const onFileDrop = async (files: File[]) => {
|
|
try {
|
|
setIsLoading(true);
|
|
|
|
const payload = {
|
|
folderId,
|
|
type,
|
|
title: files[0].name,
|
|
meta: {
|
|
timezone: userTimezone,
|
|
},
|
|
} satisfies TCreateEnvelopePayload;
|
|
|
|
const formData = new FormData();
|
|
|
|
formData.append('payload', JSON.stringify(payload));
|
|
|
|
for (const file of files) {
|
|
formData.append('files', file);
|
|
}
|
|
|
|
const { id } = await createEnvelope(formData).catch((error) => {
|
|
console.error(error);
|
|
|
|
throw error;
|
|
});
|
|
|
|
void refreshLimits();
|
|
|
|
const pathPrefix =
|
|
type === EnvelopeType.DOCUMENT
|
|
? formatDocumentsPath(team.url)
|
|
: formatTemplatesPath(team.url);
|
|
|
|
await navigate(`${pathPrefix}/${id}/edit`);
|
|
|
|
toast({
|
|
title: type === EnvelopeType.DOCUMENT ? t`Document uploaded` : t`Template uploaded`,
|
|
description:
|
|
type === EnvelopeType.DOCUMENT
|
|
? t`Your document has been uploaded successfully.`
|
|
: t`Your template has been uploaded successfully.`,
|
|
duration: 5000,
|
|
});
|
|
} catch (err) {
|
|
const error = AppError.parseError(err);
|
|
|
|
console.error(err);
|
|
|
|
const errorMessage = match(error.code)
|
|
.with('INVALID_DOCUMENT_FILE', () => t`You cannot upload encrypted PDFs`)
|
|
.with(
|
|
AppErrorCode.LIMIT_EXCEEDED,
|
|
() => t`You have reached your document limit for this month. Please upgrade your plan.`,
|
|
)
|
|
.with(
|
|
'ENVELOPE_ITEM_LIMIT_EXCEEDED',
|
|
() => t`You have reached the limit of the number of files per envelope`,
|
|
)
|
|
.otherwise(() => t`An error occurred while uploading your document.`);
|
|
|
|
toast({
|
|
title: t`Error`,
|
|
description: errorMessage,
|
|
variant: 'destructive',
|
|
duration: 7500,
|
|
});
|
|
} finally {
|
|
setIsLoading(false);
|
|
}
|
|
};
|
|
|
|
const onFileDropRejected = (fileRejections: FileRejection[]) => {
|
|
const maxItemsReached = fileRejections.some((fileRejection) =>
|
|
fileRejection.errors.some((error) => error.code === DropzoneErrorCode.TooManyFiles),
|
|
);
|
|
|
|
if (maxItemsReached) {
|
|
toast({
|
|
title: t`You cannot upload more than ${maximumEnvelopeItemCount} items per envelope.`,
|
|
duration: 5000,
|
|
variant: 'destructive',
|
|
});
|
|
|
|
return;
|
|
}
|
|
|
|
toast({
|
|
title: t`Upload failed`,
|
|
description: t`File cannot be larger than ${APP_DOCUMENT_UPLOAD_SIZE_LIMIT}MB`,
|
|
duration: 5000,
|
|
variant: 'destructive',
|
|
});
|
|
};
|
|
|
|
return (
|
|
<div className={cn('relative', className)}>
|
|
<TooltipProvider>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<div>
|
|
<DocumentDropzone
|
|
loading={isLoading}
|
|
disabled={remaining.documents === 0 || !user.emailVerified}
|
|
disabledMessage={disabledMessage}
|
|
onDrop={onFileDrop}
|
|
onDropRejected={onFileDropRejected}
|
|
type="envelope"
|
|
maxFiles={maximumEnvelopeItemCount}
|
|
/>
|
|
</div>
|
|
</TooltipTrigger>
|
|
|
|
{type === EnvelopeType.DOCUMENT &&
|
|
remaining.documents > 0 &&
|
|
Number.isFinite(remaining.documents) && (
|
|
<TooltipContent>
|
|
<p className="text-sm">
|
|
<Trans>
|
|
{remaining.documents} of {quota.documents} documents remaining this month.
|
|
</Trans>
|
|
</p>
|
|
</TooltipContent>
|
|
)}
|
|
</Tooltip>
|
|
</TooltipProvider>
|
|
</div>
|
|
);
|
|
};
|