Files
wr.do/components/sections/pricing.tsx
T
2025-06-11 15:06:36 +08:00

217 lines
6.3 KiB
TypeScript

"use client";
import type React from "react";
import type { CSSProperties, ReactNode } from "react";
import Link from "next/link";
import { motion } from "framer-motion";
import { X } from "lucide-react";
import { useTranslations } from "next-intl";
import useSWR from "swr";
import { PlanQuotaFormData } from "@/lib/dto/plan";
import { cn, fetcher, nFormatter } from "@/lib/utils";
import { Icons } from "../shared/icons";
import { Button } from "../ui/button";
const getBenefits = (plan: PlanQuotaFormData) => [
{
text: `${nFormatter(plan.slTrackedClicks)} tracked clicks/mo`,
checked: true,
icon: <Icons.mousePointerClick className="size-4" />,
},
{
text: `${nFormatter(plan.slNewLinks)} new links/mo`,
checked: true,
icon: <Icons.link className="size-4" />,
},
{
text: `${plan.slAnalyticsRetention}-day analytics retention`,
checked: true,
icon: <Icons.calendar className="size-4" />,
},
{
text: `Customize short link QR code`,
checked: plan.slCustomQrCodeLogo,
icon: <Icons.qrcode className="size-4" />,
},
{
text: `${nFormatter(plan.emEmailAddresses)} email addresses/mo`,
checked: true,
icon: <Icons.mail className="size-4" />,
},
{
text: `${nFormatter(plan.emSendEmails)} send emails/mo`,
checked: true,
icon: <Icons.send className="size-4" />,
},
{
text: `${plan.slDomains === 1 ? "One" : plan.slDomains} domain${plan.slDomains > 1 ? "s" : ""}`,
checked: true,
icon: <Icons.globe className="size-4" />,
},
{
text: "Advanced analytics",
checked: plan.slAdvancedAnalytics,
icon: <Icons.lineChart className="size-4" />,
},
{
text: `${plan.appSupport.charAt(0).toUpperCase() + plan.appSupport.slice(1)} support`,
checked: true,
icon: <Icons.help className="size-4" />,
},
{
text: "Open API Access",
checked: plan.appApiAccess,
icon: <Icons.unplug className="size-4" />,
},
];
export const PricingSection = () => {
const t = useTranslations("Landing");
const { data: plan } = useSWR<{
total: number;
list: PlanQuotaFormData[];
}>(`/api/plan?all=1`, fetcher);
return (
<section
id="pricing"
className="relative overflow-hidden bg-zinc-50 text-zinc-800 selection:bg-zinc-200 dark:bg-zinc-950 dark:text-zinc-200 dark:selection:bg-zinc-600"
>
<div className="absolute inset-0 bg-[radial-gradient(100%_100%_at_50%_0%,rgba(245,245,245,0.8),rgba(240,240,240,1))] dark:bg-[radial-gradient(100%_100%_at_50%_0%,rgba(13,13,17,1),rgba(9,9,11,1))]"></div>
<div className="relative z-10 mx-auto max-w-5xl px-4 py-20 md:px-8">
<div className="mb-12 space-y-3">
<h2 className="text-center text-xl font-semibold leading-tight sm:text-3xl sm:leading-tight md:text-4xl md:leading-tight">
{t("pricingTitle")}
</h2>
<p className="text-center text-base text-zinc-600 dark:text-zinc-400 md:text-lg">
{t("pricingDescription")}
</p>
</div>
<div className="grid grid-cols-1 gap-6 md:grid-cols-2">
{plan && (
<PriceCard
tier={t("freeTier")}
price={t("freePrice")}
bestFor={t("freeBestFor")}
CTA={
<Link href={"/dashboard"}>
<Button className="w-full" variant={"default"}>
{t("getStartedFree")}
</Button>
</Link>
}
benefits={getBenefits(plan.list[0])}
/>
)}
{plan && (
<PriceCard
tier={t("enterpriseTier")}
price={t("enterprisePrice")}
bestFor={t("enterpriseBestFor")}
CTA={
<Link href={"mailto:support@wr.do"}>
<Button className="w-full" variant="outline">
{t("contactUs")}
</Button>
</Link>
}
benefits={getBenefits(plan.list[plan.list.length - 1])}
/>
)}
</div>
</div>
</section>
);
};
const PriceCard = ({ tier, price, bestFor, CTA, benefits }: PriceCardProps) => {
return (
<Card>
<div className="flex flex-col items-center border-b border-zinc-200 pb-6 dark:border-zinc-700">
<span className="mb-6 inline-block text-zinc-800 dark:text-zinc-50">
{tier}
</span>
<span className="mb-3 inline-block text-4xl font-medium text-zinc-900 dark:text-zinc-100">
{price}
</span>
<span className="bg-gradient-to-br from-zinc-700 to-zinc-900 bg-clip-text text-center text-transparent dark:from-zinc-200 dark:to-zinc-500">
{bestFor}
</span>
</div>
<div className="space-y-3 py-9">
{benefits.map((b, i) => (
<Benefit {...b} key={i} />
))}
</div>
{CTA}
</Card>
);
};
const Benefit = ({ text, checked, icon }: BenefitType) => {
return (
<div className="flex items-center gap-3">
{checked ? (
<span className="grid size-5 place-content-center">{icon}</span>
) : (
<span className="grid size-5 place-content-center rounded-full bg-zinc-200 text-sm text-zinc-500 dark:bg-zinc-800 dark:text-zinc-400">
<X className="h-3 w-3" />
</span>
)}
<span className="text-sm text-neutral-600 dark:text-zinc-300">
{text}
</span>
</div>
);
};
const Card = ({ className, children, style = {} }: CardProps) => {
return (
<motion.div
initial={{
filter: "blur(2px)",
}}
whileInView={{
filter: "blur(0px)",
}}
transition={{
duration: 0.5,
ease: "easeInOut",
delay: 0.25,
}}
style={style}
className={cn(
"relative h-full w-full overflow-hidden rounded-2xl border border-zinc-200 bg-gradient-to-br from-zinc-50/50 to-zinc-100/80 p-6 dark:border-zinc-700 dark:from-zinc-950/50 dark:to-zinc-900/80",
className,
)}
>
{children}
</motion.div>
);
};
type PriceCardProps = {
tier: string;
price: string;
bestFor: string;
CTA: ReactNode;
benefits: BenefitType[];
};
type CardProps = {
className?: string;
children?: ReactNode;
style?: CSSProperties;
};
type BenefitType = {
text: string;
checked: boolean;
icon: ReactNode;
};