mirror of
https://github.com/documenso/documenso.git
synced 2025-11-12 07:43:16 +10:00
Currently users who cancel their plan are stuck without the ability to resubscribe. This allows them to choose a plan to subscribe This assumes that a Subscription in the "INACTIVE" state means that the plan has been paid but canceled. No tests have been done to determine the relation between "PAST_DUE" and "INACTIVE" states within our context.
151 lines
5.3 KiB
TypeScript
151 lines
5.3 KiB
TypeScript
import { useLingui } from '@lingui/react';
|
|
import { Trans } from '@lingui/react/macro';
|
|
import { SubscriptionStatus } from '@prisma/client';
|
|
import { Loader } from 'lucide-react';
|
|
import type Stripe from 'stripe';
|
|
import { match } from 'ts-pattern';
|
|
|
|
import { useCurrentOrganisation } from '@documenso/lib/client-only/providers/organisation';
|
|
import { canExecuteOrganisationAction } from '@documenso/lib/utils/organisations';
|
|
import { trpc } from '@documenso/trpc/react';
|
|
|
|
import { BillingPlans } from '~/components/general/billing-plans';
|
|
import { OrganisationBillingPortalButton } from '~/components/general/organisations/organisation-billing-portal-button';
|
|
import { OrganisationBillingInvoicesTable } from '~/components/tables/organisation-billing-invoices-table';
|
|
import { appMetaTags } from '~/utils/meta';
|
|
|
|
export function meta() {
|
|
return appMetaTags('Billing');
|
|
}
|
|
|
|
export default function TeamsSettingBillingPage() {
|
|
const { _, i18n } = useLingui();
|
|
|
|
const organisation = useCurrentOrganisation();
|
|
|
|
const { data: subscriptionQuery, isLoading: isLoadingSubscription } =
|
|
trpc.billing.subscription.get.useQuery({
|
|
organisationId: organisation.id,
|
|
});
|
|
|
|
if (isLoadingSubscription || !subscriptionQuery) {
|
|
return (
|
|
<div className="flex items-center justify-center rounded-lg py-32">
|
|
<Loader className="text-muted-foreground h-6 w-6 animate-spin" />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
const { subscription, plans } = subscriptionQuery;
|
|
|
|
const canManageBilling = canExecuteOrganisationAction(
|
|
'MANAGE_BILLING',
|
|
organisation.currentOrganisationRole,
|
|
);
|
|
|
|
const { organisationSubscription, stripeSubscription } = subscription || {};
|
|
|
|
const currentProductName =
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
(stripeSubscription?.items.data[0].price.product as Stripe.Product | undefined)?.name;
|
|
|
|
return (
|
|
<div>
|
|
<div className="flex flex-row items-end justify-between">
|
|
<div>
|
|
<h3 className="text-2xl font-semibold">
|
|
<Trans>Billing</Trans>
|
|
</h3>
|
|
|
|
<div className="text-muted-foreground mt-2 text-sm">
|
|
{!organisationSubscription && (
|
|
<p>
|
|
<Trans>
|
|
You are currently on the <span className="font-semibold">Free Plan</span>.
|
|
</Trans>
|
|
</p>
|
|
)}
|
|
|
|
{organisationSubscription &&
|
|
match(organisationSubscription.status)
|
|
.with('ACTIVE', () => (
|
|
<p>
|
|
{currentProductName ? (
|
|
<span>
|
|
You are currently subscribed to{' '}
|
|
<span className="font-semibold">{currentProductName}</span>
|
|
</span>
|
|
) : (
|
|
<span>You currently have an active plan</span>
|
|
)}
|
|
|
|
{organisationSubscription.periodEnd && (
|
|
<span>
|
|
{' '}
|
|
which is set to{' '}
|
|
{organisationSubscription.cancelAtPeriodEnd ? (
|
|
<span>
|
|
end on{' '}
|
|
<span className="font-semibold">
|
|
{i18n.date(organisationSubscription.periodEnd)}.
|
|
</span>
|
|
</span>
|
|
) : (
|
|
<span>
|
|
automatically renew on{' '}
|
|
<span className="font-semibold">
|
|
{i18n.date(organisationSubscription.periodEnd)}.
|
|
</span>
|
|
</span>
|
|
)}
|
|
</span>
|
|
)}
|
|
</p>
|
|
))
|
|
.with('INACTIVE', () => (
|
|
<p>
|
|
{currentProductName ? (
|
|
<Trans>
|
|
You currently have an inactive{' '}
|
|
<span className="font-semibold">{currentProductName}</span> subscription
|
|
</Trans>
|
|
) : (
|
|
<Trans>Your current plan is inactive.</Trans>
|
|
)}
|
|
</p>
|
|
))
|
|
.with('PAST_DUE', () => (
|
|
<p>
|
|
{currentProductName ? (
|
|
<Trans>
|
|
Your current {currentProductName} plan is past due. Please update your
|
|
payment information.
|
|
</Trans>
|
|
) : (
|
|
<Trans>Your current plan is past due.</Trans>
|
|
)}
|
|
</p>
|
|
))
|
|
.otherwise(() => null)}
|
|
</div>
|
|
</div>
|
|
|
|
<OrganisationBillingPortalButton />
|
|
</div>
|
|
|
|
<hr className="my-4" />
|
|
|
|
{(!subscription ||
|
|
subscription.organisationSubscription.status === SubscriptionStatus.INACTIVE) &&
|
|
canManageBilling && <BillingPlans plans={plans} />}
|
|
|
|
<section className="mt-6">
|
|
<OrganisationBillingInvoicesTable
|
|
organisationId={organisation.id}
|
|
subscriptionExists={Boolean(subscription)}
|
|
/>
|
|
</section>
|
|
</div>
|
|
);
|
|
}
|