mirror of
https://github.com/documenso/documenso.git
synced 2025-11-10 04:22:32 +10:00
This PR is handles the changes required to support envelopes. The new envelope editor/signing page will be hidden during release. The core changes here is to migrate the documents and templates model to a centralized envelopes model. Even though Documents and Templates are removed, from the user perspective they will still exist as we remap envelopes to documents and templates.
199 lines
6.0 KiB
TypeScript
199 lines
6.0 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 { 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 { putPdfFile } from '@documenso/lib/universal/upload/put-file';
|
|
import { formatDocumentsPath, formatTemplatesPath } from '@documenso/lib/utils/teams';
|
|
import { trpc } from '@documenso/trpc/react';
|
|
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 } = 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 result = await Promise.all(
|
|
files.map(async (file) => {
|
|
try {
|
|
const response = await putPdfFile(file);
|
|
|
|
return {
|
|
title: file.name,
|
|
documentDataId: response.id,
|
|
};
|
|
} catch (err) {
|
|
console.error(err);
|
|
throw new Error('Failed to upload document');
|
|
}
|
|
}),
|
|
);
|
|
|
|
const envelopeItemsToCreate = result.filter(
|
|
(item): item is { title: string; documentDataId: string } => item !== undefined,
|
|
);
|
|
|
|
const { id } = await createEnvelope({
|
|
folderId,
|
|
type,
|
|
title: files[0].name,
|
|
items: envelopeItemsToCreate,
|
|
meta: {
|
|
timezone: userTimezone,
|
|
},
|
|
}).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.`,
|
|
)
|
|
.otherwise(() => t`An error occurred while uploading your document.`);
|
|
|
|
toast({
|
|
title: t`Error`,
|
|
description: errorMessage,
|
|
variant: 'destructive',
|
|
duration: 7500,
|
|
});
|
|
} finally {
|
|
setIsLoading(false);
|
|
}
|
|
};
|
|
|
|
const onFileDropRejected = () => {
|
|
toast({
|
|
title:
|
|
type === EnvelopeType.DOCUMENT
|
|
? t`Your document failed to upload.`
|
|
: t`Your template failed to upload.`,
|
|
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"
|
|
/>
|
|
</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>
|
|
);
|
|
};
|