Update billing page (#1578)

This commit is contained in:
Luke Vella 2025-02-26 14:22:02 +00:00 committed by GitHub
parent 34f5555791
commit b3aafb5af6
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
12 changed files with 766 additions and 493 deletions

View file

@ -129,23 +129,6 @@
"goToInvite": "Go to Invite Page",
"planPro": "Pro",
"Billing": "Billing",
"subscriptionUpdatePayment": "Update Payment Details",
"subscriptionCancel": "Cancel Subscription",
"billingStatus": "Billing Status",
"billingStatusDescription": "Manage your subscription and billing details",
"freeForever": "free forever",
"billingStatusState": "Status",
"billingStatusActive": "Active",
"billingStatusPaused": "Paused",
"billingStatusDeleted": "Cancelled",
"endDate": "End date",
"dueDate": "Next payment due",
"billingStatusPlan": "Plan",
"billingPeriod": "Period",
"billingPeriodMonthly": "Monthly",
"billingPeriodYearly": "Yearly",
"monthlyBillingDescription": "per month",
"plan_extendedPollLife": "Keep polls indefinitely",
"becomeATranslator": "Help translate",
"noPolls": "No polls",
"noPollsDescription": "Get started by creating a new poll.",
@ -173,20 +156,10 @@
"scrollRight": "Scroll Right",
"shrink": "Shrink",
"expand": "Expand",
"activeSubscription": "Thank you for subscribing to Rallly Pro. You can manage your subscription and billing details from the billing portal.",
"billingPortal": "Billing Portal",
"supportDescription": "Need help with anything?",
"supportBilling": "Please reach out if you need any assistance.",
"contactSupport": "Contact Support",
"planFreeDescription": "For casual users",
"currentPlan": "Current Plan",
"limitedAccess": "Access to core features",
"pollsDeleted": "Polls are automatically deleted once they become inactive",
"planProDescription": "For power users and professionals",
"accessAllFeatures": "Access all features",
"earlyAccess": "Get early access to new features",
"earlyAdopterDescription": "As an early adopter, you'll lock in your subscription rate and won't be affected by future price increases.",
"upgradeNowSaveLater": "Upgrade now, save later",
"pricing": "Pricing",
"pollSettingsDescription": "Customize the behaviour of your poll",
"requireParticipantEmailLabel": "Make email address required for participants",
@ -236,7 +209,6 @@
"pastEventsEmptyStateDescription": "When you schedule events, they will appear here.",
"activePollCount": "{{activePollCount}} Live",
"createPoll": "Create poll",
"yearlyBillingDescription": "per year",
"addToCalendar": "Add to Calendar",
"microsoft365": "Microsoft 365",
"outlook": "Outlook",
@ -247,7 +219,6 @@
"timeZoneChangeDetectorMessage": "Your timezone has changed to <b>{currentTimeZone}</b>. Do you want to update your preferences?",
"yesUpdateTimezone": "Yes, update my timezone",
"noKeepCurrentTimezone": "No, keep the current timezone",
"annualBenefit": "{count} months free",
"removeAvatar": "Remove",
"uploadProfilePicture": "Upload",
"profilePictureDescription": "Up to 2MB, JPG or PNG",
@ -304,5 +275,33 @@
"createAccount": "Create Account",
"tooManyRequests": "Too many requests",
"tooManyRequestsDescription": "Please try again later.",
"loginMagicLinkError": "This link is invalid or expired. Please request a new link."
"loginMagicLinkError": "This link is invalid or expired. Please request a new link.",
"subscriptionPriceMonthly": "{{price}} per month",
"subscriptionPriceYearly": "{{price}} per year",
"billingSubscriptionTitle": "Subscription",
"billingSubscriptionDescription": "View and manage your current subscription plan",
"billingSubscriptionNotActive": "You are not currently subscribed to a plan.",
"billingSubscriptionUpgradeToProDescription": "Upgrade to Pro to get access to all features and benefits.",
"billingSubscriptionUpgradeToPro": "Upgrade to Pro",
"subscriptionStatusActive": "Active",
"subscriptionStatusTrialing": "Trialing",
"subscriptionStatusPastDue": "Past due",
"subscriptionStatusCanceled": "Canceled",
"subscriptionStatusUnpaid": "Unpaid",
"subscriptionStatusUnknown": "Unknown",
"paymentMethodUnknown": "Unknown",
"subscriptionCancelOn": "Cancels {date}",
"subscriptionStatusPaused": "Paused",
"subscriptionStatusIncomplete": "Incomplete",
"subscriptionStatusIncompleteExpired": "Incomplete expired",
"billingSubscriptionPlan": "Plan",
"billingSubscriptionPrice": "Price",
"billingSubscriptionNextPaymentDue": "Next Payment Due",
"billingPaymentMethod": "Payment Method",
"noPaymentMethodSet": "No Payment Method Set",
"addPaymentMethodDescription": "Please add a payment method to ensure uninterrupted service for your subscription.",
"addPaymentMethod": "Add Payment Method",
"needToMakeChanges": "Need to make changes?",
"billingPortalDescription": "Visit the billing portal to manage your subscription, update payment methods, or view billing history.",
"priceFree": "Free"
}

View file

@ -1,273 +0,0 @@
"use client";
import { Button } from "@rallly/ui/button";
import { Card } from "@rallly/ui/card";
import { Label } from "@rallly/ui/label";
import dayjs from "dayjs";
import { ArrowUpRight, CreditCardIcon, SendIcon } from "lucide-react";
import Link from "next/link";
import Script from "next/script";
import {
Settings,
SettingsContent,
SettingsSection,
} from "@/components/settings/settings";
import { Trans } from "@/components/trans";
import { useSubscription } from "@/contexts/plan";
import { trpc } from "@/trpc/client";
import type { PricingData } from "./billing-plans";
import { BillingPlans } from "./billing-plans";
declare global {
interface Window {
// eslint-disable-next-line @typescript-eslint/no-explicit-any
Paddle: any;
}
}
const BillingPortal = () => {
return (
<div>
<p className="text-sm">
<Trans
i18nKey="activeSubscription"
defaults="Thank you for subscribing to Rallly Pro. You can manage your subscription and billing details from the billing portal."
/>
</p>
<div className="mt-6">
<Button asChild>
<Link
href={`/api/stripe/portal?return_path=${encodeURIComponent(
window.location.pathname,
)}`}
>
<CreditCardIcon className="size-4" />
<span>
<Trans i18nKey="billingPortal" defaults="Billing Portal" />
</span>
<ArrowUpRight className="size-4" />
</Link>
</Button>
</div>
</div>
);
};
const proPlanIdMonthly = process.env.NEXT_PUBLIC_PRO_PLAN_ID_MONTHLY as string;
const SubscriptionStatus = ({ pricingData }: { pricingData: PricingData }) => {
const data = useSubscription();
if (!data) {
return null;
}
return (
<div className="space-y-6">
{!data.active ? (
<BillingPlans pricingData={pricingData} />
) : data.legacy ? (
<LegacyBilling />
) : (
<SettingsSection
title={<Trans i18nKey="billingStatus" defaults="Billing Status" />}
description={
<Trans
i18nKey="billingStatusDescription"
defaults="Manage your subscription and billing details."
/>
}
>
<BillingPortal />
</SettingsSection>
)}
</div>
);
};
const LegacyBilling = () => {
const { data: userPaymentData } = trpc.user.getBilling.useQuery();
if (!userPaymentData) {
return null;
}
const { status, endDate, planId } = userPaymentData;
if (status === "trialing" || status === "past_due") {
return <p>Invalid billing status</p>;
}
return (
<>
{process.env.NEXT_PUBLIC_PADDLE_VENDOR_ID ? (
<Script
src="https://cdn.paddle.com/paddle/paddle.js"
onLoad={() => {
if (process.env.NEXT_PUBLIC_PADDLE_SANDBOX === "true") {
window.Paddle.Environment.set("sandbox");
}
window.Paddle.Setup({
vendor: Number(process.env.NEXT_PUBLIC_PADDLE_VENDOR_ID),
});
}}
/>
) : null}
<Card>
<div className="grid gap-4 p-4 sm:grid-cols-2">
<div>
<Label>
<Trans i18nKey="billingStatusState" defaults="Status" />
</Label>
<div>
{(() => {
switch (status) {
case "active":
return (
<Trans i18nKey="billingStatusActive" defaults="Active" />
);
case "paused":
return (
<Trans i18nKey="billingStatusPaused" defaults="Paused" />
);
case "deleted":
return (
<Trans
i18nKey="billingStatusDeleted"
defaults="Cancelled"
/>
);
}
})()}
</div>
</div>
<div>
{status === "deleted" ? (
<Label>
<Trans i18nKey="endDate" defaults="End date" />
</Label>
) : (
<Label>
<Trans i18nKey="dueDate" defaults="Due date" />
</Label>
)}
<div>{dayjs(endDate).format("LL")}</div>
</div>
<div>
<Label>
<Trans i18nKey="billingStatusPlan" defaults="Plan" />
</Label>
<div>
<Trans i18nKey="planPro" />
</div>
</div>
<div>
<Label>
<Trans i18nKey="billingPeriod" defaults="Period" />
</Label>
<div>
{planId === proPlanIdMonthly ? (
<Trans i18nKey="billingPeriodMonthly" defaults="Monthly" />
) : (
<Trans i18nKey="billingPeriodYearly" defaults="Yearly" />
)}
</div>
</div>
</div>
{status === "active" || status === "paused" ? (
<div className="flex items-center gap-x-2 border-t bg-gray-50 p-3">
<Button
asChild
onClick={(e) => {
e.preventDefault();
window.Paddle.Checkout.open({
override: userPaymentData.updateUrl,
});
}}
>
<Link href={userPaymentData.updateUrl}>
<CreditCardIcon className="size-4" />
<Trans
i18nKey="subscriptionUpdatePayment"
defaults="Update Payment Details"
/>
</Link>
</Button>
<Button
asChild
variant="destructive"
onClick={(e) => {
e.preventDefault();
window.Paddle.Checkout.open({
override: userPaymentData.cancelUrl,
});
}}
>
<Link href={userPaymentData.cancelUrl}>
<Trans i18nKey="subscriptionCancel" defaults="Cancel" />
</Link>
</Button>
</div>
) : null}
</Card>
</>
);
};
<SettingsSection
title={<Trans i18nKey="support" defaults="Support" />}
description={
<Trans i18nKey="supportDescription" defaults="Need help with anything?" />
}
>
<div className="space-y-6">
<p className="text-sm">
<Trans
i18nKey="supportBilling"
defaults="Please reach out if you need any assistance."
/>
</p>
<Button asChild>
<Link href="mailto:support@rallly.co">
<SendIcon className="size-4" />
<Trans i18nKey="contactSupport" defaults="Contact Support" />
</Link>
</Button>
</div>
</SettingsSection>;
export function BillingPage({ pricingData }: { pricingData: PricingData }) {
return (
<Settings>
<SettingsContent>
<SubscriptionStatus pricingData={pricingData} />
<hr />
<SettingsSection
title={<Trans i18nKey="support" defaults="Support" />}
description={
<Trans
i18nKey="supportDescription"
defaults="Need help with anything?"
/>
}
>
<div className="space-y-6">
<p className="text-sm">
<Trans
i18nKey="supportBilling"
defaults="Please reach out if you need any assistance."
/>
</p>
<Button asChild>
<Link href="mailto:support@rallly.co">
<SendIcon className="size-4" />
<Trans i18nKey="contactSupport" defaults="Contact Support" />
</Link>
</Button>
</div>
</SettingsSection>
</SettingsContent>
</Settings>
);
}

View file

@ -1,176 +0,0 @@
import { Badge } from "@rallly/ui/badge";
import {
BillingPlan,
BillingPlanDescription,
BillingPlanHeader,
BillingPlanPeriod,
BillingPlanPerk,
BillingPlanPerks,
BillingPlanPrice,
BillingPlanTitle,
} from "@rallly/ui/billing-plan";
import { Button } from "@rallly/ui/button";
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@rallly/ui/tabs";
import { TrendingUpIcon } from "lucide-react";
import React from "react";
import { Trans } from "@/components/trans";
import { UpgradeButton } from "@/components/upgrade-button";
export type PricingData = {
monthly: {
amount: number;
currency: string;
};
yearly: {
amount: number;
currency: string;
};
};
export const BillingPlans = ({ pricingData }: { pricingData: PricingData }) => {
const [tab, setTab] = React.useState("yearly");
return (
<Tabs value={tab} onValueChange={setTab}>
<div className="space-y-4">
<div>
<TabsList>
<TabsTrigger value="monthly">
<Trans i18nKey="billingPeriodMonthly" />
</TabsTrigger>
<TabsTrigger value="yearly" className="inline-flex gap-x-2.5">
<Trans i18nKey="billingPeriodYearly" />
</TabsTrigger>
</TabsList>
</div>
<div className="grid gap-4 rounded-md md:grid-cols-2">
<BillingPlan>
<BillingPlanHeader>
<BillingPlanTitle>
<Trans i18nKey="planFree" />
</BillingPlanTitle>
<BillingPlanDescription>
<Trans
i18nKey="planFreeDescription"
defaults="For casual users"
/>
</BillingPlanDescription>
</BillingPlanHeader>
<div>
<BillingPlanPrice>$0</BillingPlanPrice>
<BillingPlanPeriod>
<Trans i18nKey="freeForever" />
</BillingPlanPeriod>
</div>
<hr />
<Button disabled className="w-full">
<Trans i18nKey="currentPlan" defaults="Current Plan" />
</Button>
<BillingPlanPerks>
<BillingPlanPerk>
<Trans
i18nKey="limitedAccess"
defaults="Access to core features"
/>
</BillingPlanPerk>
<BillingPlanPerk>
<Trans
i18nKey="pollsDeleted"
defaults="Polls are automatically deleted once they become inactive"
/>
</BillingPlanPerk>
</BillingPlanPerks>
</BillingPlan>
<div className="space-y-4 rounded-md border bg-white p-4 shadow-sm">
<div>
<BillingPlanTitle>
<Trans i18nKey="planPro" />
</BillingPlanTitle>
<p className="text-muted-foreground text-sm">
<Trans
i18nKey="planProDescription"
defaults="For power users and professionals"
/>
</p>
</div>
<div className="flex">
<TabsContent value="yearly">
<div className="flex items-center gap-x-2">
<BillingPlanPrice>
${pricingData.yearly.amount / 100}
</BillingPlanPrice>
<Badge variant="green">
<Trans
i18nKey="annualBenefit"
defaults="{count} months free"
values={{
count: 4,
}}
/>
</Badge>
</div>
<BillingPlanPeriod>
<Trans
i18nKey="yearlyBillingDescription"
defaults="per year"
/>
</BillingPlanPeriod>
</TabsContent>
<TabsContent value="monthly">
<BillingPlanPrice>
${pricingData.monthly.amount / 100}
</BillingPlanPrice>
<BillingPlanPeriod>
<Trans
i18nKey="monthlyBillingDescription"
defaults="per month"
/>
</BillingPlanPeriod>
</TabsContent>
</div>
<hr />
<UpgradeButton annual={tab === "yearly"} />
<BillingPlanPerks>
<BillingPlanPerk pro={true}>
<Trans
i18nKey="accessAllFeatures"
defaults="Access all features"
/>
</BillingPlanPerk>
<BillingPlanPerk pro={true}>
<Trans i18nKey="plan_extendedPollLife" />
</BillingPlanPerk>
<BillingPlanPerk pro={true}>
<Trans
i18nKey="earlyAccess"
defaults="Get early access to new features"
/>
</BillingPlanPerk>
</BillingPlanPerks>
</div>
</div>
<div className="rounded-lg border border-cyan-800/10 bg-cyan-50 px-4 py-3 text-cyan-800 shadow-sm">
<div className="mb-2">
<TrendingUpIcon className="text-indigo mr-2 mt-0.5 size-6 shrink-0" />
</div>
<div className="mb-2 flex items-center gap-x-2">
<h3 className="text-sm font-semibold">
<Trans
i18nKey="upgradeNowSaveLater"
defaults="Upgrade now, save later"
/>
</h3>
</div>
<p className="text-sm">
<Trans
i18nKey="earlyAdopterDescription"
defaults="As an early adopter, you'll lock in your subscription rate and won't be affected by future price increases."
/>
</p>
</div>
</div>
</Tabs>
);
};

View file

@ -0,0 +1,66 @@
"use client";
import { Badge } from "@rallly/ui/badge";
import { z } from "zod";
import { Trans } from "@/components/trans";
const brandLabels = {
visa: "Visa",
mastercard: "Mastercard",
amex: "American Express",
discover: "Discover",
unionpay: "UnionPay",
eftpos_au: "Eftpos (AU)",
jcb: "JCB",
link: "Link",
diners: "Diners Club",
unknown: "Unknown",
};
type CardBrands = keyof typeof brandLabels;
function CardDetails({ brand, last4 }: { brand: CardBrands; last4: string }) {
const brandLabel = brandLabels[brand];
return (
<div className="flex items-center gap-1">
<Badge>{brandLabel}</Badge>
<span>**** {last4}</span>
</div>
);
}
const cardDataSchema = z.object({
brand: z
.enum([
"visa",
"mastercard",
"amex",
"discover",
"unionpay",
"eftpos_au",
"jcb",
"diners",
"link",
"unknown",
])
.catch("unknown"),
last4: z.string(),
});
export function PaymentMethod({ type, data }: { type: string; data: unknown }) {
switch (type) {
case "card": {
const cardData = cardDataSchema.parse(data);
return <CardDetails brand={cardData.brand} last4={cardData.last4} />;
}
case "link":
return "Link";
default:
return (
<Badge>
<Trans i18nKey="paymentMethodUnknown" defaults="Unknown" />
</Badge>
);
}
}

View file

@ -0,0 +1,42 @@
"use client";
import { Trans } from "@/components/trans";
/**
* Returns the formatted price for the given amount, interval and currency.
* eg. $7 per month, £40 per year, etc
*/
export function SubscriptionPrice({
amount,
interval,
currency,
}: {
amount: number;
interval: string;
currency: string;
}) {
const formattedAmount = new Intl.NumberFormat("en", {
style: "currency",
currency: currency,
minimumFractionDigits: 0,
maximumFractionDigits: 0,
}).format(amount / 100); // Divide by 100 since amount is in cents
return (
<span>
{interval === "month" ? (
<Trans
i18nKey="subscriptionPriceMonthly"
defaults="{{price}} per month"
values={{ price: formattedAmount }}
/>
) : (
<Trans
i18nKey="subscriptionPriceYearly"
defaults="{{price}} per year"
values={{ price: formattedAmount }}
/>
)}
</span>
);
}

View file

@ -0,0 +1,71 @@
"use client";
import type { SubscriptionStatus as SubscriptionStatusType } from "@prisma/client";
import dayjs from "dayjs";
import { Trans } from "@/components/trans";
import { useTranslation } from "@/i18n/client";
interface SubscriptionStatusProps {
status: SubscriptionStatusType;
cancelAtPeriodEnd: boolean;
periodEnd: Date;
}
export const SubscriptionStatus = ({
status,
cancelAtPeriodEnd,
periodEnd,
}: SubscriptionStatusProps) => {
const { t } = useTranslation();
const statusConfig: Record<
string,
{
label: string;
}
> = {
active: {
label: t("subscriptionStatusActive", { defaultValue: "Active" }),
},
paused: {
label: t("subscriptionStatusPaused", { defaultValue: "Paused" }),
},
trialing: {
label: t("subscriptionStatusTrialing", { defaultValue: "Trialing" }),
},
past_due: {
label: t("subscriptionStatusPastDue", { defaultValue: "Past due" }),
},
canceled: {
label: t("subscriptionStatusCanceled", { defaultValue: "Canceled" }),
},
unpaid: {
label: t("subscriptionStatusUnpaid", { defaultValue: "Unpaid" }),
},
incomplete: {
label: t("subscriptionStatusIncomplete", { defaultValue: "Incomplete" }),
},
incomplete_expired: {
label: t("subscriptionStatusIncompleteExpired", {
defaultValue: "Incomplete expired",
}),
},
unknown: {
label: t("subscriptionStatusUnknown", { defaultValue: "Unknown" }),
},
};
const config = statusConfig[status] || statusConfig.unknown;
if (status === "active" && cancelAtPeriodEnd) {
return (
<Trans
i18nKey="subscriptionCancelOn"
defaults="Cancels {date}"
values={{ date: dayjs(periodEnd).format("MMM D") }}
/>
);
}
return <>{config.label}</>;
};

View file

@ -1,21 +1,307 @@
import { pricingData } from "@rallly/billing/pricing";
import { prisma } from "@rallly/database";
import { Badge } from "@rallly/ui/badge";
import { Button } from "@rallly/ui/button";
import { DialogTrigger } from "@rallly/ui/dialog";
import { Icon } from "@rallly/ui/icon";
import {
ArrowUpRightIcon,
CircleAlertIcon,
CreditCardIcon,
GemIcon,
PlusIcon,
SendIcon,
SparklesIcon,
} from "lucide-react";
import Link from "next/link";
import { notFound } from "next/navigation";
import { BillingPage } from "@/app/[locale]/(admin)/settings/billing/billing-page";
import type { Params } from "@/app/[locale]/types";
import { env } from "@/env";
import { getTranslation } from "@/i18n/server";
import {
DescriptionDetails,
DescriptionList,
DescriptionTerm,
} from "@/components/description-list";
import {
EmptyState,
EmptyStateDescription,
EmptyStateFooter,
EmptyStateIcon,
EmptyStateTitle,
} from "@/components/empty-state";
import { FormattedDate } from "@/components/formatted-date";
import { PayWallDialog } from "@/components/pay-wall-dialog";
import { Settings, SettingsSection } from "@/components/settings/settings";
import { Trans } from "@/components/trans";
import { requireUser } from "@/next-auth";
import { isSelfHosted } from "@/utils/constants";
import { PaymentMethod } from "./components/payment-method";
import { SubscriptionPrice } from "./components/subscription-price";
import { SubscriptionStatus } from "./components/subscription-status";
async function getData() {
const user = await requireUser();
const data = await prisma.user.findUnique({
where: { id: user.id },
select: {
customerId: true,
subscription: {
select: {
id: true,
active: true,
currency: true,
amount: true,
priceId: true,
periodStart: true,
periodEnd: true,
interval: true,
cancelAtPeriodEnd: true,
status: true,
},
},
paymentMethods: {
select: {
id: true,
type: true,
data: true,
},
},
},
});
if (!data) {
throw new Error("User not found");
}
return data;
}
export default async function Page() {
if (env.NEXT_PUBLIC_SELF_HOSTED === "true") {
if (isSelfHosted) {
notFound();
}
return <BillingPage pricingData={pricingData} />;
}
export async function generateMetadata({ params }: { params: Params }) {
const { t } = await getTranslation(params.locale);
return {
title: t("billing"),
};
const data = await getData();
const { subscription } = data;
return (
<Settings>
<SettingsSection
title={
<Trans i18nKey="billingSubscriptionTitle" defaults="Subscription" />
}
description={
<Trans
i18nKey="billingSubscriptionDescription"
defaults="View and manage your current subscription plan"
/>
}
>
{subscription ? (
<div className="space-y-4">
<DescriptionList>
<DescriptionTerm>
<Trans i18nKey="billingSubscriptionPlan" defaults="Plan" />
</DescriptionTerm>
<DescriptionDetails>
{subscription ? (
<div className="flex items-center gap-x-2">
<span className="font-bold">Pro</span>
<Badge
variant={
subscription.status === "active" ? "green" : "default"
}
>
<SubscriptionStatus
status={subscription.status}
cancelAtPeriodEnd={subscription.cancelAtPeriodEnd}
periodEnd={subscription.periodEnd}
/>
</Badge>
</div>
) : (
"Hobby"
)}
</DescriptionDetails>
<DescriptionTerm>
<Trans i18nKey="billingSubscriptionPrice" defaults="Price" />
</DescriptionTerm>
<DescriptionDetails>
{subscription ? (
<SubscriptionPrice
amount={subscription.amount}
currency={subscription.currency}
interval={subscription.interval}
/>
) : (
<Trans i18nKey="priceFree" defaults="Free" />
)}
</DescriptionDetails>
<DescriptionTerm>
<Trans
i18nKey="billingSubscriptionNextPaymentDue"
defaults="Next Payment Due"
/>
</DescriptionTerm>
<DescriptionDetails>
{subscription.cancelAtPeriodEnd ? (
"-"
) : (
<FormattedDate date={subscription.periodEnd} format="short" />
)}
</DescriptionDetails>
<DescriptionTerm>
<Trans
i18nKey="billingPaymentMethod"
defaults="Payment Method"
/>
</DescriptionTerm>
<DescriptionDetails>
{data.paymentMethods.length === 0 ? (
<div>
<div className="space-y-2">
<div className="text-destructive flex items-center gap-x-2">
<CircleAlertIcon className="size-4" />
<span className="font-medium">
<Trans
i18nKey="noPaymentMethodSet"
defaults="No Payment Method Set"
/>
</span>
</div>
<p className="text-muted-foreground text-sm">
<Trans
i18nKey="addPaymentMethodDescription"
defaults="Please add a payment method to ensure uninterrupted service for your subscription."
/>
</p>
</div>
<div className="mt-6">
<Button variant="default" asChild>
<Link
prefetch={false}
href="/api/stripe/portal/payment-methods"
>
<Icon>
<PlusIcon />
</Icon>
<Trans
i18nKey="addPaymentMethod"
defaults="Add Payment Method"
/>
</Link>
</Button>
</div>
</div>
) : (
<ul className="space-y-2">
{data.paymentMethods.map((method) => (
<li key={method.id}>
<PaymentMethod type={method.type} data={method.data} />
</li>
))}
</ul>
)}
</DescriptionDetails>
</DescriptionList>
{data.customerId ? (
<div className="flex flex-col items-start justify-between gap-6 gap-x-4 rounded-lg border bg-gray-50 p-4 sm:flex-row">
<div className="space-y-1">
<h3 className="text-sm font-medium">
<Trans
i18nKey="needToMakeChanges"
defaults="Need to make changes?"
/>
</h3>
<p className="text-muted-foreground text-sm">
<Trans
i18nKey="billingPortalDescription"
defaults="Visit the billing portal to manage your subscription, update payment methods, or view billing history."
/>
</p>
</div>
<Button asChild variant="default">
<Link
href="/api/stripe/portal"
prefetch={false}
className="whitespace-nowrap"
>
<Icon>
<CreditCardIcon />
</Icon>
<Trans i18nKey="billingPortal" defaults="Billing Portal" />
<Icon>
<ArrowUpRightIcon />
</Icon>
</Link>
</Button>
</div>
) : null}
</div>
) : (
<EmptyState className="py-8">
<EmptyStateIcon>
<GemIcon />
</EmptyStateIcon>
<EmptyStateTitle>
<Trans
i18nKey="billingSubscriptionNotActive"
defaults="You are not currently subscribed to a plan."
/>
</EmptyStateTitle>
<EmptyStateDescription>
<Trans
i18nKey="billingSubscriptionUpgradeToProDescription"
defaults="Upgrade to Pro to get access to all features and benefits."
/>
</EmptyStateDescription>
<EmptyStateFooter>
<PayWallDialog>
<DialogTrigger>
<Button>
<Icon>
<SparklesIcon />
</Icon>
<Trans
i18nKey="billingSubscriptionUpgradeToPro"
defaults="Upgrade to Pro"
/>
</Button>
</DialogTrigger>
</PayWallDialog>
</EmptyStateFooter>
</EmptyState>
)}
</SettingsSection>
<hr />
<SettingsSection
title={<Trans i18nKey="support" defaults="Support" />}
description={
<Trans
i18nKey="supportDescription"
defaults="Need help with anything?"
/>
}
>
<div className="space-y-6">
<p className="text-sm">
<Trans
i18nKey="supportBilling"
defaults="Please reach out if you need any assistance."
/>
</p>
<Button asChild>
<Link href="mailto:support@rallly.co">
<SendIcon className="size-4" />
<Trans i18nKey="contactSupport" defaults="Contact Support" />
</Link>
</Button>
</div>
</SettingsSection>
</Settings>
);
}

View file

@ -0,0 +1,46 @@
import { cn } from "@rallly/ui";
export function DescriptionList({
className,
...props
}: React.ComponentPropsWithoutRef<"dl">) {
return (
<dl
{...props}
className={cn(
className,
"grid grid-cols-1 text-sm/6 sm:grid-cols-[min(50%,320px)_auto]",
)}
/>
);
}
export function DescriptionTerm({
className,
...props
}: React.ComponentPropsWithoutRef<"dt">) {
return (
<dt
{...props}
className={cn(
className,
"col-start-1 border-t border-gray-800/5 pt-3 text-gray-500 first:border-none sm:border-t sm:border-gray-800/5 sm:py-3 dark:border-white/5 dark:text-gray-400 sm:dark:border-white/5",
)}
/>
);
}
export function DescriptionDetails({
className,
...props
}: React.ComponentPropsWithoutRef<"dd">) {
return (
<dd
{...props}
className={cn(
className,
"pb-3 pt-1 text-gray-800 sm:border-t sm:border-gray-800/5 sm:py-3 dark:text-white dark:sm:border-white/5 sm:[&:nth-child(2)]:border-none",
)}
/>
);
}

View file

@ -0,0 +1,19 @@
"use client";
import dayjs from "dayjs";
const formatMap = {
short: "D MMM YYYY",
};
type Format = keyof typeof formatMap;
export function FormattedDate({
date,
format,
}: {
date: Date;
format: Format;
}) {
return <>{dayjs(date).format(formatMap[format])}</>;
}

View file

@ -0,0 +1,183 @@
"use client";
import { cn } from "@rallly/ui";
import Link from "next/link";
import React, { createContext, useContext, useState } from "react";
const TableContext = createContext<{
bleed: boolean;
dense: boolean;
grid: boolean;
striped: boolean;
}>({
bleed: false,
dense: false,
grid: false,
striped: false,
});
export function Table({
bleed = false,
dense = false,
grid = false,
striped = false,
className,
children,
...props
}: {
bleed?: boolean;
dense?: boolean;
grid?: boolean;
striped?: boolean;
} & React.ComponentPropsWithoutRef<"div">) {
return (
<TableContext.Provider
value={
{ bleed, dense, grid, striped } as React.ContextType<
typeof TableContext
>
}
>
<div className="flow-root">
<div
{...props}
className={cn(
className,
"-mx-(--gutter) overflow-x-auto whitespace-nowrap",
)}
>
<div
className={cn(
"inline-block min-w-full align-middle",
!bleed && "sm:px-(--gutter)",
)}
>
<table className="min-w-full text-left text-sm/6 text-zinc-950 dark:text-white">
{children}
</table>
</div>
</div>
</div>
</TableContext.Provider>
);
}
export function TableHead({
className,
...props
}: React.ComponentPropsWithoutRef<"thead">) {
return (
<thead
{...props}
className={cn(className, "text-zinc-500 dark:text-zinc-400")}
/>
);
}
export function TableBody(props: React.ComponentPropsWithoutRef<"tbody">) {
return <tbody {...props} />;
}
const TableRowContext = createContext<{
href?: string;
target?: string;
title?: string;
}>({
href: undefined,
target: undefined,
title: undefined,
});
export function TableRow({
href,
target,
title,
className,
...props
}: {
href?: string;
target?: string;
title?: string;
} & React.ComponentPropsWithoutRef<"tr">) {
const { striped } = useContext(TableContext);
return (
<TableRowContext.Provider
value={
{ href, target, title } as React.ContextType<typeof TableRowContext>
}
>
<tr
{...props}
className={cn(
className,
href &&
"has-[[data-row-link][data-focus]]:outline-2 has-[[data-row-link][data-focus]]:-outline-offset-2 has-[[data-row-link][data-focus]]:outline-blue-500 dark:focus-within:bg-white/[2.5%]",
striped && "even:bg-zinc-950/[2.5%] dark:even:bg-white/[2.5%]",
href && striped && "hover:bg-zinc-950/5 dark:hover:bg-white/5",
href &&
!striped &&
"hover:bg-zinc-950/[2.5%] dark:hover:bg-white/[2.5%]",
)}
/>
</TableRowContext.Provider>
);
}
export function TableHeader({
className,
...props
}: React.ComponentPropsWithoutRef<"th">) {
const { bleed, grid } = useContext(TableContext);
return (
<th
{...props}
className={cn(
className,
"first:pl-(--gutter,--spacing(2)) last:pr-(--gutter,--spacing(2)) border-b border-b-zinc-950/10 px-4 py-2 font-medium dark:border-b-white/10",
grid &&
"border-l border-l-zinc-950/5 first:border-l-0 dark:border-l-white/5",
!bleed && "sm:first:pl-1 sm:last:pr-1",
)}
/>
);
}
export function TableCell({
className,
children,
...props
}: React.ComponentPropsWithoutRef<"td">) {
const { bleed, dense, grid, striped } = useContext(TableContext);
const { href, target, title } = useContext(TableRowContext);
const [cellRef, setCellRef] = useState<HTMLElement | null>(null);
return (
<td
ref={href ? setCellRef : undefined}
{...props}
className={cn(
className,
"first:pl-(--gutter,--spacing(2)) last:pr-(--gutter,--spacing(2)) relative px-4",
!striped && "border-b border-zinc-950/5 dark:border-white/5",
grid &&
"border-l border-l-zinc-950/5 first:border-l-0 dark:border-l-white/5",
dense ? "py-2.5" : "py-4",
!bleed && "sm:first:pl-1 sm:last:pr-1",
)}
>
{href && (
<Link
data-row-link
href={href}
target={target}
aria-label={title}
tabIndex={cellRef?.previousElementSibling === null ? 0 : -1}
className="focus:outline-hidden absolute inset-0"
/>
)}
{children}
</td>
);
}

View file

@ -1,3 +1,4 @@
"use client";
import { Trans as BaseTrans, useTranslation } from "react-i18next";
import type { TxKeyPath } from "../i18n/types";

View file

@ -1,5 +1,6 @@
import { prisma } from "@rallly/database";
import { posthog } from "@rallly/posthog/server";
import { redirect } from "next/navigation";
import NextAuth from "next-auth";
import type { Provider } from "next-auth/providers";
import z from "zod";
@ -185,4 +186,12 @@ const auth = async () => {
return getLegacySession();
};
export { auth, handlers, signIn, signOut };
const requireUser = async () => {
const session = await auth();
if (!session?.user) {
redirect("/login");
}
return session?.user;
};
export { auth, handlers, requireUser, signIn, signOut };