mirror of
https://github.com/shadcn-ui/ui.git
synced 2026-07-01 08:34:12 +00:00
* feat(@shadcn/react): add message-scroller package Add the @shadcn/react headless primitives package with MessageScroller scroll anchoring, streaming follow, history prepend, and jump-to-message behavior. Includes geometry helpers, use-render utility, and unit, browser, and perf tests. * feat(registry): add chat components Add MessageScroller, Message, Bubble, Attachment, and Marker registry sources for base and radix, style variants, preview-03 chat blocks, and registry index wiring. * feat(v4): integrate chat components into docs site Wire chat components into the v4 app with docs routes, example preview pages, message part renderers, markdown support, registry build updates, and supporting lib utilities. * feat(examples): add chat component demos Add base and radix example demos for MessageScroller, Message, Bubble, Attachment, Marker, scroll-fade, and shimmer. * docs: add chat component documentation Add component and utility docs for the chat component set, update docs navigation, and add the June 2026 chat components changelog entry. * chore: regenerate registry JSON output Rebuild public registry artifacts for all style variants with the new chat components. * chore(release): add @shadcn/react publish and CI pipeline Add Changesets prerelease workflow, browser test job, RELEASING docs, and monorepo wiring for publishing @shadcn/react independently from the shadcn CLI. * docs: fix display of component preview on mobile * fix * fix * docs: add message scroller docs * style: format * fix
485 lines
13 KiB
TypeScript
485 lines
13 KiB
TypeScript
import * as React from "react"
|
|
import Image from "next/image"
|
|
import Link from "next/link"
|
|
|
|
import { source } from "@/lib/source"
|
|
import { cn } from "@/lib/utils"
|
|
import { Callout } from "@/components/callout"
|
|
import { CodeBlockCommand } from "@/components/code-block-command"
|
|
import { CodeCollapsibleWrapper } from "@/components/code-collapsible-wrapper"
|
|
import { CodeTabs } from "@/components/code-tabs"
|
|
import { ComponentPreview } from "@/components/component-preview"
|
|
import { ComponentSource } from "@/components/component-source"
|
|
import { ComponentsList } from "@/components/components-list"
|
|
import { CopyButton } from "@/components/copy-button"
|
|
import { DirectoryList } from "@/components/directory-list"
|
|
import { getIconForLanguageExtension } from "@/components/icons"
|
|
import {
|
|
Accordion,
|
|
AccordionContent,
|
|
AccordionItem,
|
|
AccordionTrigger,
|
|
} from "@/registry/new-york-v4/ui/accordion"
|
|
import {
|
|
Alert,
|
|
AlertDescription,
|
|
AlertTitle,
|
|
} from "@/registry/new-york-v4/ui/alert"
|
|
import { AspectRatio } from "@/registry/new-york-v4/ui/aspect-ratio"
|
|
import { Button } from "@/registry/new-york-v4/ui/button"
|
|
import { Kbd } from "@/registry/new-york-v4/ui/kbd"
|
|
import {
|
|
Tabs,
|
|
TabsContent,
|
|
TabsList,
|
|
TabsTrigger,
|
|
} from "@/registry/new-york-v4/ui/tabs"
|
|
|
|
function getComponentsFolder() {
|
|
const componentsFolder = source.pageTree.children.find(
|
|
(page) => page.$id === "components"
|
|
)
|
|
|
|
if (componentsFolder?.type !== "folder") {
|
|
return null
|
|
}
|
|
|
|
return componentsFolder
|
|
}
|
|
|
|
// This is only used on /docs/components/ index page, so default to radix.
|
|
function ComponentsListWrapper({ variant }: { variant?: "all" | "new" }) {
|
|
const componentsFolder = getComponentsFolder()
|
|
|
|
if (!componentsFolder) {
|
|
return null
|
|
}
|
|
|
|
return (
|
|
<ComponentsList
|
|
componentsFolder={componentsFolder}
|
|
currentBase="radix"
|
|
variant={variant}
|
|
/>
|
|
)
|
|
}
|
|
|
|
function getNodeText(node: React.ReactNode): string {
|
|
if (typeof node === "string" || typeof node === "number") {
|
|
return String(node)
|
|
}
|
|
|
|
if (Array.isArray(node)) {
|
|
return node.map((child) => getNodeText(child)).join("")
|
|
}
|
|
|
|
if (React.isValidElement<{ children?: React.ReactNode }>(node)) {
|
|
return getNodeText(node.props.children)
|
|
}
|
|
|
|
return ""
|
|
}
|
|
|
|
function getHeadingId(children: React.ReactNode) {
|
|
const id = getNodeText(children)
|
|
.trim()
|
|
.replace(/\s+/g, "-")
|
|
.replace(/'/g, "")
|
|
.replace(/\?/g, "")
|
|
.toLowerCase()
|
|
|
|
return id || undefined
|
|
}
|
|
|
|
function HeadingAnchor({
|
|
id,
|
|
children,
|
|
}: {
|
|
id?: string
|
|
children: React.ReactNode
|
|
}) {
|
|
if (!id) {
|
|
return children
|
|
}
|
|
|
|
return (
|
|
<a className="group no-underline" href={`#${id}`}>
|
|
<span className="underline-offset-4 group-hover:underline">
|
|
{children}
|
|
</span>
|
|
<span
|
|
aria-hidden="true"
|
|
className="ml-2 text-muted-foreground opacity-0 group-hover:opacity-100"
|
|
>
|
|
#
|
|
</span>
|
|
</a>
|
|
)
|
|
}
|
|
|
|
export const mdxComponents = {
|
|
h1: ({ className, children, id, ...props }: React.ComponentProps<"h1">) => {
|
|
const headingId = id ?? getHeadingId(children)
|
|
|
|
return (
|
|
<h1
|
|
id={headingId}
|
|
className={cn(
|
|
"mt-2 scroll-m-28 font-heading text-3xl font-bold tracking-tight",
|
|
className
|
|
)}
|
|
{...props}
|
|
>
|
|
<HeadingAnchor id={headingId}>{children}</HeadingAnchor>
|
|
</h1>
|
|
)
|
|
},
|
|
h2: ({ className, children, id, ...props }: React.ComponentProps<"h2">) => {
|
|
const headingId = id ?? getHeadingId(children)
|
|
|
|
return (
|
|
<h2
|
|
id={headingId}
|
|
className={cn(
|
|
"[&+]*:[code]:text-xl mt-10 scroll-m-28 font-heading text-xl font-medium tracking-tight first:mt-0 lg:mt-12 [&+.steps]:mt-0! [&+.steps>h3]:mt-4! [&+h3]:mt-6! [&+p]:mt-4!",
|
|
className
|
|
)}
|
|
{...props}
|
|
>
|
|
<HeadingAnchor id={headingId}>{children}</HeadingAnchor>
|
|
</h2>
|
|
)
|
|
},
|
|
h3: ({ className, children, id, ...props }: React.ComponentProps<"h3">) => {
|
|
const headingId = id ?? getHeadingId(children)
|
|
|
|
return (
|
|
<h3
|
|
id={headingId}
|
|
className={cn(
|
|
"mt-12 scroll-m-28 font-heading text-lg font-medium tracking-tight [&+p]:mt-4! *:[code]:text-xl",
|
|
className
|
|
)}
|
|
{...props}
|
|
>
|
|
<HeadingAnchor id={headingId}>{children}</HeadingAnchor>
|
|
</h3>
|
|
)
|
|
},
|
|
h4: ({ className, children, id, ...props }: React.ComponentProps<"h4">) => {
|
|
const headingId = id ?? getHeadingId(children)
|
|
|
|
return (
|
|
<h4
|
|
id={headingId}
|
|
className={cn(
|
|
"mt-8 scroll-m-28 font-heading text-base font-medium tracking-tight",
|
|
className
|
|
)}
|
|
{...props}
|
|
>
|
|
<HeadingAnchor id={headingId}>{children}</HeadingAnchor>
|
|
</h4>
|
|
)
|
|
},
|
|
h5: ({ className, children, id, ...props }: React.ComponentProps<"h5">) => {
|
|
const headingId = id ?? getHeadingId(children)
|
|
|
|
return (
|
|
<h5
|
|
id={headingId}
|
|
className={cn(
|
|
"mt-8 scroll-m-28 text-base font-medium tracking-tight",
|
|
className
|
|
)}
|
|
{...props}
|
|
>
|
|
<HeadingAnchor id={headingId}>{children}</HeadingAnchor>
|
|
</h5>
|
|
)
|
|
},
|
|
h6: ({ className, children, id, ...props }: React.ComponentProps<"h6">) => {
|
|
const headingId = id ?? getHeadingId(children)
|
|
|
|
return (
|
|
<h6
|
|
id={headingId}
|
|
className={cn(
|
|
"mt-8 scroll-m-28 text-base font-medium tracking-tight",
|
|
className
|
|
)}
|
|
{...props}
|
|
>
|
|
<HeadingAnchor id={headingId}>{children}</HeadingAnchor>
|
|
</h6>
|
|
)
|
|
},
|
|
a: ({ className, ...props }: React.ComponentProps<"a">) => (
|
|
<a
|
|
className={cn("font-medium underline underline-offset-4", className)}
|
|
{...props}
|
|
/>
|
|
),
|
|
p: ({ className, ...props }: React.ComponentProps<"p">) => (
|
|
<p
|
|
className={cn("leading-relaxed [&:not(:first-child)]:mt-6", className)}
|
|
{...props}
|
|
/>
|
|
),
|
|
strong: ({ className, ...props }: React.HTMLAttributes<HTMLElement>) => (
|
|
<strong className={cn("font-medium", className)} {...props} />
|
|
),
|
|
ul: ({ className, ...props }: React.ComponentProps<"ul">) => (
|
|
<ul className={cn("my-6 ml-6 list-disc", className)} {...props} />
|
|
),
|
|
ol: ({ className, ...props }: React.ComponentProps<"ol">) => (
|
|
<ol className={cn("my-6 ml-6 list-decimal", className)} {...props} />
|
|
),
|
|
li: ({ className, ...props }: React.ComponentProps<"li">) => (
|
|
<li className={cn("mt-2", className)} {...props} />
|
|
),
|
|
blockquote: ({ className, ...props }: React.ComponentProps<"blockquote">) => (
|
|
<blockquote
|
|
className={cn("mt-6 border-l-2 pl-6 italic", className)}
|
|
{...props}
|
|
/>
|
|
),
|
|
img: ({ className, alt, ...props }: React.ComponentProps<"img">) => (
|
|
<img className={cn("rounded-md", className)} alt={alt} {...props} />
|
|
),
|
|
hr: ({ ...props }: React.ComponentProps<"hr">) => (
|
|
<hr className="my-4 md:my-8" {...props} />
|
|
),
|
|
table: ({ className, ...props }: React.ComponentProps<"table">) => (
|
|
<div className="my-6 no-scrollbar w-full overflow-y-auto rounded-xl border">
|
|
<table
|
|
className={cn(
|
|
"relative w-full overflow-hidden border-none text-sm [&_tbody_tr:last-child]:border-b-0",
|
|
className
|
|
)}
|
|
{...props}
|
|
/>
|
|
</div>
|
|
),
|
|
tr: ({ className, ...props }: React.ComponentProps<"tr">) => (
|
|
<tr className={cn("m-0 border-b", className)} {...props} />
|
|
),
|
|
th: ({ className, ...props }: React.ComponentProps<"th">) => (
|
|
<th
|
|
className={cn(
|
|
"px-4 py-2 text-left font-bold [&[align=center]]:text-center [&[align=right]]:text-right",
|
|
className
|
|
)}
|
|
{...props}
|
|
/>
|
|
),
|
|
td: ({ className, ...props }: React.ComponentProps<"td">) => (
|
|
<td
|
|
className={cn(
|
|
"px-4 py-2 text-left whitespace-nowrap [&[align=center]]:text-center [&[align=right]]:text-right",
|
|
className
|
|
)}
|
|
{...props}
|
|
/>
|
|
),
|
|
pre: ({ className, children, ...props }: React.ComponentProps<"pre">) => {
|
|
return (
|
|
<pre
|
|
className={cn(
|
|
"no-scrollbar min-w-0 overflow-x-auto overflow-y-auto overscroll-x-contain overscroll-y-auto px-4 py-3.5 outline-none has-[[data-highlighted-line]]:px-0 has-[[data-line-numbers]]:px-0 has-[[data-slot=tabs]]:p-0",
|
|
className
|
|
)}
|
|
{...props}
|
|
>
|
|
{children}
|
|
</pre>
|
|
)
|
|
},
|
|
figure: ({ className, ...props }: React.ComponentProps<"figure">) => {
|
|
return <figure className={cn(className)} {...props} />
|
|
},
|
|
figcaption: ({
|
|
className,
|
|
children,
|
|
...props
|
|
}: React.ComponentProps<"figcaption">) => {
|
|
const iconExtension =
|
|
"data-language" in props && typeof props["data-language"] === "string"
|
|
? getIconForLanguageExtension(props["data-language"])
|
|
: null
|
|
|
|
return (
|
|
<figcaption
|
|
className={cn(
|
|
"flex items-center gap-2 text-code-foreground [&_svg]:size-4 [&_svg]:text-code-foreground [&_svg]:opacity-70",
|
|
className
|
|
)}
|
|
{...props}
|
|
>
|
|
{iconExtension}
|
|
{children}
|
|
</figcaption>
|
|
)
|
|
},
|
|
code: ({
|
|
className,
|
|
__raw__,
|
|
__src__,
|
|
__npm__,
|
|
__yarn__,
|
|
__pnpm__,
|
|
__bun__,
|
|
...props
|
|
}: React.ComponentProps<"code"> & {
|
|
__raw__?: string
|
|
__src__?: string
|
|
__npm__?: string
|
|
__yarn__?: string
|
|
__pnpm__?: string
|
|
__bun__?: string
|
|
}) => {
|
|
// Inline Code.
|
|
if (typeof props.children === "string") {
|
|
return (
|
|
<code
|
|
className={cn(
|
|
"relative rounded-md bg-muted px-[0.3rem] py-[0.2rem] font-mono text-[0.8rem] break-words outline-none",
|
|
className
|
|
)}
|
|
{...props}
|
|
/>
|
|
)
|
|
}
|
|
|
|
// npm command.
|
|
const isNpmCommand = __npm__ && __yarn__ && __pnpm__ && __bun__
|
|
if (isNpmCommand) {
|
|
return (
|
|
<CodeBlockCommand
|
|
__npm__={__npm__}
|
|
__yarn__={__yarn__}
|
|
__pnpm__={__pnpm__}
|
|
__bun__={__bun__}
|
|
/>
|
|
)
|
|
}
|
|
|
|
// Default codeblock.
|
|
return (
|
|
<>
|
|
{__raw__ && <CopyButton value={__raw__} src={__src__} />}
|
|
<code {...props} />
|
|
</>
|
|
)
|
|
},
|
|
Step: ({ className, ...props }: React.ComponentProps<"h3">) => (
|
|
<h3
|
|
className={cn(
|
|
"mt-8 scroll-m-32 font-heading text-lg font-medium tracking-tight",
|
|
className
|
|
)}
|
|
{...props}
|
|
/>
|
|
),
|
|
Steps: ({ className, ...props }: React.ComponentProps<"div">) => (
|
|
<div
|
|
className={cn(
|
|
"steps mb-12 [counter-reset:step] md:ml-4 md:border-l md:pl-8 [&>h3]:step",
|
|
className
|
|
)}
|
|
{...props}
|
|
/>
|
|
),
|
|
Image: ({
|
|
src,
|
|
className,
|
|
width,
|
|
height,
|
|
alt,
|
|
...props
|
|
}: React.ComponentProps<"img">) => (
|
|
<Image
|
|
className={cn("mt-6 rounded-md border", className)}
|
|
src={(src as string) || ""}
|
|
width={Number(width)}
|
|
height={Number(height)}
|
|
alt={alt || ""}
|
|
{...props}
|
|
/>
|
|
),
|
|
Tabs: ({ className, ...props }: React.ComponentProps<typeof Tabs>) => {
|
|
return <Tabs className={cn("relative mt-6 w-full", className)} {...props} />
|
|
},
|
|
TabsList: ({
|
|
className,
|
|
...props
|
|
}: React.ComponentProps<typeof TabsList>) => (
|
|
<TabsList
|
|
className={cn(
|
|
"justify-start gap-4 rounded-none bg-transparent px-0",
|
|
className
|
|
)}
|
|
{...props}
|
|
/>
|
|
),
|
|
TabsTrigger: ({
|
|
className,
|
|
...props
|
|
}: React.ComponentProps<typeof TabsTrigger>) => (
|
|
<TabsTrigger
|
|
className={cn(
|
|
"rounded-none border-0 border-b-2 border-transparent bg-transparent px-0 pb-3 text-base text-muted-foreground hover:text-primary data-[state=active]:border-primary data-[state=active]:bg-transparent data-[state=active]:text-foreground data-[state=active]:shadow-none! dark:data-[state=active]:border-primary dark:data-[state=active]:bg-transparent",
|
|
className
|
|
)}
|
|
{...props}
|
|
/>
|
|
),
|
|
TabsContent: ({
|
|
className,
|
|
...props
|
|
}: React.ComponentProps<typeof TabsContent>) => (
|
|
<TabsContent
|
|
className={cn(
|
|
"relative [&_h3.font-heading]:text-base [&_h3.font-heading]:font-medium *:[figure]:first:mt-0 [&>.steps]:mt-6",
|
|
className
|
|
)}
|
|
{...props}
|
|
/>
|
|
),
|
|
Tab: ({ className, ...props }: React.ComponentProps<"div">) => (
|
|
<div className={cn(className)} {...props} />
|
|
),
|
|
Button,
|
|
Callout,
|
|
Accordion,
|
|
AccordionContent,
|
|
AccordionItem,
|
|
AccordionTrigger,
|
|
Alert,
|
|
AlertTitle,
|
|
AlertDescription,
|
|
AspectRatio,
|
|
CodeTabs,
|
|
ComponentPreview,
|
|
ComponentSource,
|
|
CodeCollapsibleWrapper,
|
|
ComponentsList: ComponentsListWrapper,
|
|
DirectoryList,
|
|
Link: ({ className, ...props }: React.ComponentProps<typeof Link>) => (
|
|
<Link
|
|
className={cn("font-medium underline underline-offset-4", className)}
|
|
{...props}
|
|
/>
|
|
),
|
|
LinkedCard: ({ className, ...props }: React.ComponentProps<typeof Link>) => (
|
|
<Link
|
|
className={cn(
|
|
"flex w-full flex-col items-center rounded-xl bg-surface p-6 text-surface-foreground transition-colors hover:bg-surface/80 sm:p-10",
|
|
className
|
|
)}
|
|
{...props}
|
|
/>
|
|
),
|
|
Kbd,
|
|
}
|