add some feature

This commit is contained in:
oiov
2025-03-26 15:49:22 +08:00
parent e108ca121a
commit 4401aaabf2
6 changed files with 258 additions and 10 deletions
+2
View File
@@ -1,4 +1,5 @@
import HeroLanding, { LandingImages } from "@/components/sections/hero-landing";
import { PricingSection } from "@/components/sections/pricing";
// import PreviewLanding from "@/components/sections/preview-landing";
@@ -8,6 +9,7 @@ export default function IndexPage() {
<HeroLanding />
{/* <PreviewLanding /> */}
<LandingImages />
<PricingSection />
</>
);
}
-1
View File
@@ -8,7 +8,6 @@ import { cn, nFormatter } from "@/lib/utils";
import { buttonVariants } from "@/components/ui/button";
import { Icons } from "@/components/shared/icons";
import { Doc } from "../../.contentlayer/generated/types";
import GitHubStarsWithSuspense from "../shared/github-star-wrapper";
import UrlShortener from "./url-shortener";
+243
View File
@@ -0,0 +1,243 @@
"use client";
import type React from "react";
import type { CSSProperties, ReactNode } from "react";
import { motion } from "framer-motion";
import { Check, X } from "lucide-react";
import { cn } from "@/lib/utils";
import { Icons } from "../shared/icons";
export const PricingSection = () => {
return (
<section 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">
Pricing
</h2>
<p className="text-center text-base text-zinc-600 dark:text-zinc-400 md:text-lg">
Use it for free for yourself, upgrade when your team needs advanced
control.
</p>
</div>
<div className="grid grid-cols-1 gap-6 md:grid-cols-2">
<PriceCard
tier="Free"
price="$0/mo"
bestFor="For hobbyists and individuals looking to manage their links"
CTA={<GhostButton className="w-full">Get started free</GhostButton>}
benefits={[
{
text: "100K tracked clicks/mo",
checked: true,
icon: <Icons.mousePointerClick className="size-4" />,
},
{
text: "10k new links/mo",
checked: true,
icon: <Icons.link className="size-4" />,
},
{
text: "180-day analytics retention",
checked: true,
icon: <Icons.calendar className="size-4" />,
},
{
text: "One domain",
checked: true,
icon: <Icons.globe className="size-4" />,
},
{
text: "Advanced analytics",
checked: true,
icon: <Icons.lineChart className="size-4" />,
},
{
text: "Basic support",
checked: true,
icon: <Icons.help className="size-4" />,
},
{
text: "API Access",
checked: true,
icon: <Icons.unplug className="size-4" />,
},
]}
/>
{/* <PriceCard
tier="Pro"
price="$79/mo"
bestFor="Best for 5-50 users"
CTA={
<GhostButton className="w-full bg-zinc-800 text-zinc-50 hover:bg-zinc-700 hover:text-zinc-50 dark:bg-zinc-50 dark:text-zinc-950 dark:hover:bg-zinc-200 dark:hover:text-zinc-900">
14-day free trial
</GhostButton>
}
benefits={[
{ text: "Five workspaces", checked: true },
{ text: "Email support", checked: true },
{ text: "7 day data retention", checked: true },
{ text: "Custom roles", checked: true },
{ text: "Priority support", checked: false },
{ text: "SSO", checked: false },
]}
/> */}
<PriceCard
tier="Enterprise"
price="Contact us"
bestFor="For large organizations with custom needs"
CTA={<GhostButton className="w-full">Contact us</GhostButton>}
benefits={[
{
text: "Unlimited tracked clicks/mo",
checked: true,
icon: <Icons.mousePointerClick className="size-4" />,
},
{
text: "Unlimited new links",
checked: true,
icon: <Icons.link className="size-4" />,
},
{
text: "1-year analytics retention",
checked: true,
icon: <Icons.calendar className="size-4" />,
},
{
text: "Three domain",
checked: true,
icon: <Icons.globe className="size-4" />,
},
{
text: "Advanced analytics",
checked: true,
icon: <Icons.lineChart className="size-4" />,
},
{
text: "Basic support",
checked: true,
icon: <Icons.help className="size-4" />,
},
{
text: "API Access",
checked: true,
icon: <Icons.unplug className="size-4" />,
},
]}
/>
</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>
);
};
const GhostButton = ({ children, className, ...rest }: GhostButtonProps) => {
return (
<button
className={cn(
"rounded-md px-4 py-2 text-lg text-zinc-700 transition-all hover:scale-[1.02] hover:bg-zinc-200 hover:text-zinc-900 active:scale-[0.98] dark:text-zinc-100 dark:hover:bg-zinc-800 dark:hover:text-zinc-50",
className,
)}
{...rest}
>
{children}
</button>
);
};
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;
};
type GhostButtonProps = {
children: ReactNode;
className?: string;
} & React.ButtonHTMLAttributes<HTMLButtonElement>;
+8 -8
View File
@@ -42,7 +42,7 @@ export default function UrlShotenerExp() {
</div>
</div>
<div className="rounded-2xl border border-gray-100 bg-white p-5 shadow-sm">
<div className="rounded-2xl border border-gray-100 bg-white p-5 shadow-sm dark:border-gray-700 dark:bg-gray-900/70">
<div className="flex items-center justify-between gap-2">
<div className="flex items-center gap-4">
<div className="flex h-12 w-12 items-center justify-center overflow-hidden rounded-full bg-black">
@@ -65,14 +65,14 @@ export default function UrlShotenerExp() {
</div>
<div>
<div className="flex items-center gap-2">
<p className="text-base font-bold text-gray-700">
<p className="text-base font-bold text-gray-700 dark:text-gray-50">
wr.do/s/try
</p>
<div className="flex gap-1">
<button className="rounded-full border p-1.5 transition-colors hover:bg-gray-100">
<button className="rounded-full border p-1.5 transition-colors hover:bg-gray-100 dark:bg-gray-600/50">
<Icons.copy className="size-3 text-gray-500" />
</button>
<button className="rounded-full border p-1.5 transition-colors hover:bg-gray-100">
<button className="rounded-full border p-1.5 transition-colors hover:bg-gray-100 dark:bg-gray-600/50">
<Icons.qrcode className="size-3 text-gray-500" />
</button>
</div>
@@ -102,18 +102,18 @@ export default function UrlShotenerExp() {
></polyline>
</g>
</svg>
wr.do/login
wr.do/dashboard
</div>
</div>
</div>
<div className="flex items-center gap-2">
<div className="flex items-center gap-1 rounded-lg border bg-gray-50 px-3 py-1">
<div className="flex items-center gap-1 rounded-lg border bg-gray-50 px-3 py-1 dark:bg-gray-600/50">
<Icons.mousePointerClick className="size-4" />
<p className="text-sm font-medium text-gray-700">
<p className="text-sm font-medium text-gray-700 dark:text-gray-50">
12.6K <span className="hidden sm:inline">clicks</span>
</p>
</div>
<button className="rounded-lg p-2 transition-colors hover:bg-gray-100">
<button className="hidden rounded-lg p-2 transition-colors hover:bg-gray-100 sm:block">
<svg
width="20"
height="20"
+4
View File
@@ -6,6 +6,7 @@ import {
BadgeHelp,
BookOpen,
Bug,
Calendar,
Camera,
Check,
ChevronLeft,
@@ -43,6 +44,7 @@ import {
Settings,
SunMedium,
Trash2,
Unplug,
User,
Users,
X,
@@ -64,6 +66,7 @@ export const Icons = {
close: X,
copy: Copy,
camera: Camera,
calendar: Calendar,
fileText: FileText,
dashboard: LayoutPanelLeft,
download: Download,
@@ -148,6 +151,7 @@ export const Icons = {
mail: Mail,
bug: Bug,
CirclePlay: CirclePlay,
unplug: Unplug,
lineChart: ({ ...props }: LucideProps) => (
<svg
xmlns="http://www.w3.org/2000/svg"
+1 -1
View File
File diff suppressed because one or more lines are too long