mirror of
https://github.com/shadcn-ui/ui.git
synced 2026-06-27 06:34:12 +00:00
* feat: init * fix * fix * fix * feat * feat * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * feat: implement icons * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * feat: update init command * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * feat: dialog * feat * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * feat: add registry:base item type * feat: rename frame to canva * fix * feat * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fi * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * feat: add all colors * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * fix * feat: add outfit font * fix * fix * fix * fix * fix * chore: changeset * fix * fix * fix * fix * fix * fix * fix * fix
192 lines
5.8 KiB
TypeScript
192 lines
5.8 KiB
TypeScript
"use client"
|
|
|
|
import * as React from "react"
|
|
import Link, { type LinkProps } from "next/link"
|
|
import { useRouter } from "next/navigation"
|
|
|
|
import { PAGES_NEW } from "@/lib/docs"
|
|
import { showMcpDocs } from "@/lib/flags"
|
|
import { type source } from "@/lib/source"
|
|
import { cn } from "@/lib/utils"
|
|
import { Button } from "@/registry/new-york-v4/ui/button"
|
|
import {
|
|
Popover,
|
|
PopoverContent,
|
|
PopoverTrigger,
|
|
} from "@/registry/new-york-v4/ui/popover"
|
|
|
|
const TOP_LEVEL_SECTIONS = [
|
|
{ name: "Get Started", href: "/docs" },
|
|
{
|
|
name: "Components",
|
|
href: "/docs/components",
|
|
},
|
|
{
|
|
name: "Directory",
|
|
href: "/docs/directory",
|
|
},
|
|
{
|
|
name: "MCP Server",
|
|
href: "/docs/mcp",
|
|
},
|
|
{
|
|
name: "Forms",
|
|
href: "/docs/forms",
|
|
},
|
|
{
|
|
name: "Changelog",
|
|
href: "/docs/changelog",
|
|
},
|
|
]
|
|
|
|
export function MobileNav({
|
|
tree,
|
|
items,
|
|
className,
|
|
}: {
|
|
tree: typeof source.pageTree
|
|
items: { href: string; label: string }[]
|
|
className?: string
|
|
}) {
|
|
const [open, setOpen] = React.useState(false)
|
|
|
|
return (
|
|
<Popover open={open} onOpenChange={setOpen}>
|
|
<PopoverTrigger asChild>
|
|
<Button
|
|
variant="ghost"
|
|
className={cn(
|
|
"extend-touch-target h-8 touch-manipulation items-center justify-start gap-2.5 !p-0 hover:bg-transparent focus-visible:bg-transparent focus-visible:ring-0 active:bg-transparent dark:hover:bg-transparent",
|
|
className
|
|
)}
|
|
>
|
|
<div className="relative flex h-8 w-4 items-center justify-center">
|
|
<div className="relative size-4">
|
|
<span
|
|
className={cn(
|
|
"bg-foreground absolute left-0 block h-0.5 w-4 transition-all duration-100",
|
|
open ? "top-[0.4rem] -rotate-45" : "top-1"
|
|
)}
|
|
/>
|
|
<span
|
|
className={cn(
|
|
"bg-foreground absolute left-0 block h-0.5 w-4 transition-all duration-100",
|
|
open ? "top-[0.4rem] rotate-45" : "top-2.5"
|
|
)}
|
|
/>
|
|
</div>
|
|
<span className="sr-only">Toggle Menu</span>
|
|
</div>
|
|
<span className="flex h-8 items-center text-lg leading-none font-medium">
|
|
Menu
|
|
</span>
|
|
</Button>
|
|
</PopoverTrigger>
|
|
<PopoverContent
|
|
className="bg-background/90 no-scrollbar h-(--radix-popper-available-height) w-(--radix-popper-available-width) overflow-y-auto rounded-none border-none p-0 shadow-none backdrop-blur duration-100"
|
|
align="start"
|
|
side="bottom"
|
|
alignOffset={-16}
|
|
sideOffset={14}
|
|
>
|
|
<div className="flex flex-col gap-12 overflow-auto px-6 py-6">
|
|
<div className="flex flex-col gap-4">
|
|
<div className="text-muted-foreground text-sm font-medium">
|
|
Menu
|
|
</div>
|
|
<div className="flex flex-col gap-3">
|
|
<MobileLink href="/" onOpenChange={setOpen}>
|
|
Home
|
|
</MobileLink>
|
|
{items.map((item, index) => (
|
|
<MobileLink key={index} href={item.href} onOpenChange={setOpen}>
|
|
{item.label}
|
|
</MobileLink>
|
|
))}
|
|
</div>
|
|
</div>
|
|
<div className="flex flex-col gap-4">
|
|
<div className="text-muted-foreground text-sm font-medium">
|
|
Sections
|
|
</div>
|
|
<div className="flex flex-col gap-3">
|
|
{TOP_LEVEL_SECTIONS.map(({ name, href }) => {
|
|
if (!showMcpDocs && href.includes("/mcp")) {
|
|
return null
|
|
}
|
|
return (
|
|
<MobileLink key={name} href={href} onOpenChange={setOpen}>
|
|
{name}
|
|
</MobileLink>
|
|
)
|
|
})}
|
|
</div>
|
|
</div>
|
|
<div className="flex flex-col gap-8">
|
|
{tree?.children?.map((group, index) => {
|
|
if (group.type === "folder") {
|
|
return (
|
|
<div key={index} className="flex flex-col gap-4">
|
|
<div className="text-muted-foreground text-sm font-medium">
|
|
{group.name}
|
|
</div>
|
|
<div className="flex flex-col gap-3">
|
|
{group.children.map((item) => {
|
|
if (item.type === "page") {
|
|
if (!showMcpDocs && item.url.includes("/mcp")) {
|
|
return null
|
|
}
|
|
return (
|
|
<MobileLink
|
|
key={`${item.url}-${index}`}
|
|
href={item.url}
|
|
onOpenChange={setOpen}
|
|
className="flex items-center gap-2"
|
|
>
|
|
{item.name}{" "}
|
|
{PAGES_NEW.includes(item.url) && (
|
|
<span className="flex size-2 rounded-full bg-blue-500" />
|
|
)}
|
|
</MobileLink>
|
|
)
|
|
}
|
|
})}
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
})}
|
|
</div>
|
|
</div>
|
|
</PopoverContent>
|
|
</Popover>
|
|
)
|
|
}
|
|
|
|
function MobileLink({
|
|
href,
|
|
onOpenChange,
|
|
className,
|
|
children,
|
|
...props
|
|
}: LinkProps & {
|
|
onOpenChange?: (open: boolean) => void
|
|
children: React.ReactNode
|
|
className?: string
|
|
}) {
|
|
const router = useRouter()
|
|
return (
|
|
<Link
|
|
href={href}
|
|
onClick={() => {
|
|
router.push(href.toString())
|
|
onOpenChange?.(false)
|
|
}}
|
|
className={cn("text-2xl font-medium", className)}
|
|
{...props}
|
|
>
|
|
{children}
|
|
</Link>
|
|
)
|
|
}
|