Files
stats/apps/public/src/app/(home)/_sections/pricing.tsx
Carl-Gerhard Lindesvärd 1b898660ad fix: improve landing page
2026-03-10 22:30:31 +01:00

130 lines
4.6 KiB
TypeScript

'use client';
import NumberFlow from '@number-flow/react';
import { PRICING } from '@openpanel/payments/prices';
import { CheckIcon, ServerIcon, StarIcon } from 'lucide-react';
import Link from 'next/link';
import { useState } from 'react';
import { GetStartedButton } from '@/components/get-started-button';
import { Section, SectionHeader } from '@/components/section';
import { Button } from '@/components/ui/button';
import { cn, formatEventsCount } from '@/lib/utils';
const features = [
'Unlimited websites or apps',
'Unlimited users',
'Unlimited dashboards',
'Unlimited charts',
'Unlimited tracked profiles',
'Yes, we have no limits or hidden costs',
];
export function Pricing() {
const [selectedIndex, setSelectedIndex] = useState(2);
const selected = PRICING[selectedIndex];
return (
<Section className="container">
<div className="col md:row gap-16">
<div className="col w-full min-w-sm gap-4 rounded-3xl border bg-linear-to-b from-card to-background p-6 md:w-1/3">
<p className="text-muted-foreground text-sm">
Choose how many events you'll track this month
</p>
<div className="row flex-wrap gap-2">
{PRICING.map((tier, index) => (
<Button
className={cn('relative h-8 rounded-full border px-4')}
key={tier.price}
onClick={() => setSelectedIndex(index)}
size="sm"
variant={selectedIndex === index ? 'default' : 'outline'}
>
{tier.popular && <StarIcon className="size-4" />}
{formatEventsCount(tier.events)}
</Button>
))}
<Button
className={cn('relative h-8 rounded-full border px-4')}
onClick={() => setSelectedIndex(-1)}
size="sm"
variant={selectedIndex === -1 ? 'default' : 'outline'}
>
Custom
</Button>
</div>
<div className="col mt-8 w-full items-baseline md:mt-auto">
{selected ? (
<>
<span className="mb-2 rounded-full bg-primary/10 px-2.5 py-0.5 font-medium text-primary text-xs">
30-day free trial
</span>
<div className="row items-end gap-3">
<NumberFlow
className="font-bold text-5xl"
format={{
style: 'currency',
currency: 'USD',
minimumFractionDigits: 0,
maximumFractionDigits: 1,
}}
locales={'en-US'}
value={selected.price}
/>
</div>
<div className="row w-full justify-between">
<span className="-mt-2 text-muted-foreground/80 text-sm">
Per month
</span>
<span className="-mt-2 text-muted-foreground/80 text-sm">
+ VAT if applicable
</span>
</div>
</>
) : (
<div className="text-lg">
Contact us at{' '}
<a className="underline" href="mailto:hello@openpanel.dev">
hello@openpanel.dev
</a>{' '}
to get a custom quote.
</div>
)}
</div>
</div>
<div className="col flex-1 shrink-0 justify-center gap-8">
<div className="col gap-4">
<SectionHeader
description="Pay only for what you use. Choose your event volume - everything else is unlimited. No surprises, no hidden fees."
title="Simple, transparent pricing"
/>
</div>
<ul className="col gap-2">
{features.map((feature) => (
<li className="row items-start gap-2 text-sm" key={feature}>
<CheckIcon className="mt-0.5 size-4 shrink-0" />
<span>{feature}</span>
</li>
))}
</ul>
<GetStartedButton className="w-fit" />
<p className="row items-center gap-2 text-muted-foreground text-sm">
<ServerIcon className="size-4 shrink-0" />
Prefer to self-host?{' '}
<Link
className="text-primary hover:underline"
href="/docs/self-hosting/self-hosting"
>
Deploy for free
</Link>{' '}
with unlimited events.
</p>
</div>
</div>
</Section>
);
}