Components
Loading preview...
Here is Feature component
npx shadcn@latest add https://21st.dev/r/brijr/feature-modern// app/components/Feature.tsx
"use client";
// React / Next
import React from "react";
import Link from "next/link";
// Third-party
import Balancer from "react-wrap-balancer";
import { Coins, ArrowRight } from "lucide-react";
/* =========================
craft-ds (v0.2.8 merged)
========================= */
import { type ClassValue, clsx } from "clsx";
import { twMerge } from "tailwind-merge";
export function cn(...inputs: ClassValue[]) {
return twMerge(clsx(inputs));
}
type LayoutProps = { children: React.ReactNode; className?: string };
type MainProps = { children: React.ReactNode; className?: string; id?: string };
type SectionProps = { children: React.ReactNode; className?: string; id?: string };
type ContainerProps = { children: React.ReactNode; className?: string; id?: string };
type ArticleProps = {
children?: React.ReactNode;
className?: string;
id?: string;
dangerouslySetInnerHTML?: { __html: string };
};
type BoxProps = {
children: React.ReactNode;
className?: string;
direction?:
| "row"
| "col"
| { sm?: "row" | "col"; md?: "row" | "col"; lg?: "row" | "col"; xl?: "row" | "col"; "2xl"?: "row" | "col" };
wrap?:
| boolean
| { sm?: boolean; md?: boolean; lg?: boolean; xl?: boolean; "2xl"?: boolean };
gap?: number | { sm?: number; md?: number; lg?: number; xl?: number; "2xl"?: number };
cols?: number | { sm?: number; md?: number; lg?: number; xl?: number; "2xl"?: number };
rows?: number | { sm?: number; md?: number; lg?: number; xl?: number; "2xl"?: number };
};
export const Layout = ({ children, className }: LayoutProps) => (
<html lang="en" suppressHydrationWarning className={cn("scroll-smooth antialiased focus:scroll-auto", className)}>
{children}
</html>
);
export const Main = ({ children, className, id }: MainProps) => (
<main
className={cn(
"max-w-none prose-p:m-0",
"prose prose-neutral prose:font-sans dark:prose-invert xl:prose-lg",
"prose-headings:font-normal",
"prose-strong:font-semibold",
"prose-a:underline prose-a:decoration-primary/50 prose-a:underline-offset-2 prose-a:text-foreground/75 prose-a:transition-all",
"hover:prose-a:decoration-primary hover:prose-a:text-foreground",
"prose-blockquote:not-italic",
"prose-pre:border prose-pre:bg-muted/25 prose-pre:text-foreground",
className
)}
id={id}
>
{children}
</main>
);
export const Section = ({ children, className, id }: SectionProps) => (
<section className={cn("py-8 md:py-12", className)} id={id}>
{children}
</section>
);
export const Container = ({ children, className, id }: ContainerProps) => (
<div className={cn("mx-auto max-w-5xl", "p-6 sm:p-8", className)} id={id}>
{children}
</div>
);
export const Article = ({ children, className, id, dangerouslySetInnerHTML }: ArticleProps) => (
<article
dangerouslySetInnerHTML={dangerouslySetInnerHTML}
className={cn(
"prose prose-neutral prose:font-sans dark:prose-invert xl:prose-lg",
"prose-headings:font-normal",
"prose-p:mb-0",
"prose-strong:font-semibold",
"prose-a:underline prose-a:decoration-primary/50 prose-a:underline-offset-2 prose-a:text-foreground/75 prose-a:transition-all",
"hover:prose-a:decoration-primary hover:prose-a:text-foreground",
"prose-blockquote:not-italic",
"prose-pre:border prose-pre:bg-muted/25",
className
)}
id={id}
>
{children}
</article>
);
export const Box = ({ children, className, direction = "row", wrap = false, gap = 0, cols, rows }: BoxProps) => {
const directionClasses = { row: "flex-row", col: "flex-col" } as const;
const wrapClasses = wrap ? "flex-wrap" : "flex-nowrap";
const gapClasses = { 0: "gap-0", 1: "gap-1", 2: "gap-2", 3: "gap-3", 4: "gap-4", 5: "gap-5", 6: "gap-6", 8: "gap-8", 10: "gap-10", 12: "gap-12" };
const colsClasses = { 1: "grid-cols-1", 2: "grid-cols-2", 3: "grid-cols-3", 4: "grid-cols-4", 5: "grid-cols-5", 6: "grid-cols-6", 7: "grid-cols-7", 8: "grid-cols-8", 9: "grid-cols-9", 10: "grid-cols-10", 11: "grid-cols-11", 12: "grid-cols-12" };
const getResp = (prop: any, map: Record<string | number, string>) =>
typeof prop === "object"
? Object.entries(prop)
.map(([bp, val]) => `${bp === "sm" ? "" : `${bp}:`}${map[val as keyof typeof map] || ""}`)
.join(" ")
: map[prop as keyof typeof map] || "";
const stack = cn(
cols || rows ? "grid" : "flex",
getResp(direction, directionClasses),
typeof wrap === "boolean" ? wrapClasses : getResp(wrap, { true: "flex-wrap", false: "flex-nowrap" } as any),
getResp(gap, gapClasses),
cols && getResp(cols, colsClasses),
rows && getResp(rows, colsClasses),
className
);
return <div className={stack}>{children}</div>;
};
/* =========================
Feature
========================= */
type FeatureText = {
icon: JSX.Element;
title: string;
description: string;
href?: string;
cta?: string;
};
const featureText: FeatureText[] = [
{
icon: <Coins className="h-6 w-6" />,
title: "Lorem Ipsum",
href: "/",
description:
"Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur.",
cta: "Learn More",
},
{
icon: <Coins className="h-6 w-6" />,
title: "Lorem Ipsum",
href: "/",
description:
"Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur.",
cta: "Learn More",
},
];
const Feature = () => {
return (
<Section className="border-b">
<Container className="not-prose">
<div className="flex flex-col gap-6">
<h3 className="text-4xl">
<Balancer>
Lorem ipsum dolor sit amet, consectetur adipiscing elit
</Balancer>
</h3>
<h4 className="text-2xl font-light opacity-70">
<Balancer>
Sed do eiusmod tempor incididunt ut labore et dolore magna aliqua
</Balancer>
</h4>
<div className="mt-6 grid gap-6 md:mt-12 md:grid-cols-2">
{featureText.map(({ icon, title, description, href, cta }, index) => (
<a
key={index}
href={href || "#"}
className="flex flex-col justify-between gap-6 rounded-lg border p-6 transition-all hover:-mt-2 hover:mb-2"
>
<div className="grid gap-4">
{icon}
<h4 className="text-xl text-primary">{title}</h4>
<p className="text-base opacity-75">{description}</p>
</div>
{cta && (
<div className="flex h-fit items-center text-sm font-semibold">
<span>{cta}</span> <ArrowRight className="ml-2 h-4 w-4" />
</div>
)}
</a>
))}
</div>
</div>
</Container>
</Section>
);
};
export default Feature;