shadcn-expansions
UI
Typography
Layout
Utils

heading-with-anchor

Add anchor for every heading.

heading
anchor
typography
link
navigation
documentation
View Docs

Source Code

Files
heading-with-anchor.tsx
1"use client";
2import { Link } from "@/components/primitives/link-with-transition";
3import { cn } from "@/lib/utils";
4import { Slot } from "@radix-ui/react-slot";
5import { type VariantProps, cva } from "class-variance-authority";
6import { LinkIcon } from "lucide-react";
7import React from "react";
8
9type AnchorProps = {
10	anchor?: string;
11	anchorVisibility?: "hover" | "always" | "never";
12	disableCopyToClipboard?: boolean;
13};
14
15const Anchor = ({
16	anchor,
17	disableCopyToClipboard = false,
18	anchorVisibility = "always",
19}: AnchorProps) => {
20	function copyToClipboard() {
21		if (disableCopyToClipboard) return;
22		const currentUrl = window.location.href.replace(/#.*$/, "");
23		const urlWithId = `${currentUrl}#${anchor}`;
24
25		void navigator?.clipboard?.writeText(urlWithId);
26	}
27
28	return (
29		<div
30			className={cn(
31				"ms-2 pt-1",
32				anchorVisibility === "always" && "visible",
33				anchorVisibility === "never" && "hidden",
34				anchorVisibility === "hover" && "invisible group-hover:visible",
35			)}
36		>
37			{/* modify `Link` to `a` if you are not using Next.js */}
38			<Link href={`#${anchor}`} onClick={copyToClipboard}>
39				<LinkIcon className="text-gray-600 hover:text-gray-400" />
40			</Link>
41		</div>
42	);
43};
44
45const headingVariants = cva("font-bold text-primary", {
46	variants: {
47		variant: {
48			h1: "leading-14 text-4xl lg:text-5xl",
49			h2: "leading-14 text-3xl lg:text-4xl",
50			h3: "leading-10 text-2xl lg:text-3xl",
51			h4: "leading-8 text-xl lg:text-2xl",
52			h5: "leading-8 text-lg lg:text-xl",
53			h6: "leading-7 text-sm lg:text-base",
54			p: "leading-5 text-lg lg:text-xl font-normal",
55		},
56	},
57	defaultVariants: {
58		variant: "h6",
59	},
60});
61
62type BaseHeadingProps = {
63	children?: React.ReactNode;
64	variant?: string;
65	className?: string;
66	asChild?: boolean;
67	anchor?: string;
68	anchorAlignment?: "close" | "spaced";
69	anchorVisibility?: "hover" | "always" | "never";
70	disableCopyToClipboard?: boolean;
71} & React.HTMLAttributes<HTMLHeadingElement> &
72	VariantProps<typeof headingVariants>;
73
74const BaseHeading = ({
75	children,
76	className,
77	variant = "h6",
78	asChild = false,
79	anchor,
80	anchorAlignment = "spaced",
81	anchorVisibility = "always",
82	disableCopyToClipboard = false,
83	...props
84}: BaseHeadingProps) => {
85	const Comp = asChild ? Slot : variant;
86	return (
87		<>
88			<Comp
89				id={anchor}
90				{...props}
91				className={cn(
92					anchor && "flex scroll-m-20 items-center gap-1", // modify `scroll-m-20` according to your header height.
93					anchorAlignment === "spaced" && "justify-between",
94					anchorVisibility === "hover" && "group",
95					headingVariants({ variant, className }),
96				)}
97			>
98				{children}
99				{anchor && (
100					<Anchor
101						anchor={anchor}
102						anchorVisibility={anchorVisibility}
103						disableCopyToClipboard={disableCopyToClipboard}
104					/>
105				)}
106			</Comp>
107		</>
108	);
109};
110
111type TypographyProps = Omit<BaseHeadingProps, "variant" | "asChild">;
112
113const H1 = (props: TypographyProps) => {
114	return <BaseHeading {...props} variant="h1" />;
115};
116
117const H2 = (props: TypographyProps) => {
118	return <BaseHeading {...props} variant="h2" />;
119};
120
121const H3 = (props: TypographyProps) => {
122	return <BaseHeading {...props} variant="h3" />;
123};
124
125const H4 = (props: TypographyProps) => {
126	return <BaseHeading {...props} variant="h4" />;
127};
128
129const H5 = (props: TypographyProps) => {
130	return <BaseHeading {...props} variant="h5" />;
131};
132
133const H6 = (props: TypographyProps) => {
134	return <BaseHeading {...props} variant="h6" />;
135};
136
137const P = (props: TypographyProps) => {
138	return <BaseHeading {...props} variant="p" />;
139};
140
141export { H1, H2, H3, H4, H5, H6, P };
142