Navigation Menu
React navigationTop-of-page horizontal navigation with hover/click triggered megamenus. Use for marketing sites and product navs that need rich content — featured links, mini-cards, and submenu columns.
Also available for Vue ->Installation
$ pnpm dlx shadcn@latest add https://react.uipkge.dev/r/react/navigation-menu.json$ npx shadcn@latest add https://react.uipkge.dev/r/react/navigation-menu.json$ yarn dlx shadcn@latest add https://react.uipkge.dev/r/react/navigation-menu.json$ bunx shadcn@latest add https://react.uipkge.dev/r/react/navigation-menu.json
Or with the named registry:
npx shadcn@latest add @uipkge-react/navigation-menu
Examples
Dependencies
Files (3)
-
components/ui/navigation-menu/navigation-menu.tsx 7.6 kB
'use client' import * as React from 'react' import * as NavigationMenuPrimitive from '@radix-ui/react-navigation-menu' import { ChevronDown } from 'lucide-react' import { cn } from '@/lib/utils' import { navigationMenuTriggerStyle } from './navigation-menu.variants' const NavigationMenu = React.forwardRef< React.ElementRef<typeof NavigationMenuPrimitive.Root>, React.ComponentPropsWithoutRef<typeof NavigationMenuPrimitive.Root> & { viewport?: boolean } >(({ className, children, viewport = true, ...props }, ref) => ( <NavigationMenuPrimitive.Root ref={ref} data-uipkge="" data-slot="navigation-menu" data-viewport={viewport} className={cn('group/navigation-menu relative flex max-w-max flex-1 items-center justify-center', className)} {...props} > {children} {viewport && <NavigationMenuViewport />} </NavigationMenuPrimitive.Root> )) NavigationMenu.displayName = NavigationMenuPrimitive.Root.displayName const NavigationMenuList = React.forwardRef< React.ElementRef<typeof NavigationMenuPrimitive.List>, React.ComponentPropsWithoutRef<typeof NavigationMenuPrimitive.List> >(({ className, ...props }, ref) => ( <NavigationMenuPrimitive.List ref={ref} data-uipkge="" data-slot="navigation-menu-list" className={cn('group flex flex-1 list-none items-center justify-center gap-1', className)} {...props} /> )) NavigationMenuList.displayName = NavigationMenuPrimitive.List.displayName const NavigationMenuItem = React.forwardRef< React.ElementRef<typeof NavigationMenuPrimitive.Item>, React.ComponentPropsWithoutRef<typeof NavigationMenuPrimitive.Item> >(({ className, ...props }, ref) => ( <NavigationMenuPrimitive.Item ref={ref} data-uipkge="" data-slot="navigation-menu-item" className={cn('relative', className)} {...props} /> )) NavigationMenuItem.displayName = NavigationMenuPrimitive.Item.displayName const NavigationMenuTrigger = React.forwardRef< React.ElementRef<typeof NavigationMenuPrimitive.Trigger>, React.ComponentPropsWithoutRef<typeof NavigationMenuPrimitive.Trigger> >(({ className, children, ...props }, ref) => ( <NavigationMenuPrimitive.Trigger ref={ref} data-uipkge="" data-slot="navigation-menu-trigger" className={cn(navigationMenuTriggerStyle(), 'group', className)} {...props} > {children} <ChevronDown className="relative top-[1px] ml-1 size-3 transition duration-300 group-data-[state=open]:rotate-180" aria-hidden="true" /> </NavigationMenuPrimitive.Trigger> )) NavigationMenuTrigger.displayName = NavigationMenuPrimitive.Trigger.displayName const NavigationMenuContent = React.forwardRef< React.ElementRef<typeof NavigationMenuPrimitive.Content>, React.ComponentPropsWithoutRef<typeof NavigationMenuPrimitive.Content> >(({ className, ...props }, ref) => ( <NavigationMenuPrimitive.Content ref={ref} data-uipkge="" data-slot="navigation-menu-content" className={cn( 'data-[motion^=from-]:motion-safe:animate-in data-[motion^=to-]:motion-safe:animate-out data-[motion^=from-]:motion-safe:fade-in data-[motion^=to-]:motion-safe:fade-out data-[motion=from-end]:motion-safe:slide-in-from-right-52 data-[motion=from-start]:motion-safe:slide-in-from-left-52 data-[motion=to-end]:motion-safe:slide-out-to-right-52 data-[motion=to-start]:motion-safe:slide-out-to-left-52 top-0 left-0 w-full p-2 pr-2.5 md:absolute md:w-auto', 'group-data-[viewport=false]/navigation-menu:bg-popover group-data-[viewport=false]/navigation-menu:text-popover-foreground group-data-[viewport=false]/navigation-menu:data-[state=open]:motion-safe:animate-in group-data-[viewport=false]/navigation-menu:data-[state=closed]:motion-safe:animate-out group-data-[viewport=false]/navigation-menu:data-[state=closed]:motion-safe:zoom-out-95 group-data-[viewport=false]/navigation-menu:data-[state=open]:motion-safe:zoom-in-95 group-data-[viewport=false]/navigation-menu:data-[state=open]:motion-safe:fade-in-0 group-data-[viewport=false]/navigation-menu:data-[state=closed]:motion-safe:fade-out-0 group-data-[viewport=false]/navigation-menu:top-full group-data-[viewport=false]/navigation-menu:mt-1.5 group-data-[viewport=false]/navigation-menu:overflow-hidden group-data-[viewport=false]/navigation-menu:rounded-md group-data-[viewport=false]/navigation-menu:border group-data-[viewport=false]/navigation-menu:shadow group-data-[viewport=false]/navigation-menu:duration-200 **:data-[slot=navigation-menu-link]:focus:ring-0 **:data-[slot=navigation-menu-link]:focus:outline-none', className, )} {...props} /> )) NavigationMenuContent.displayName = NavigationMenuPrimitive.Content.displayName const NavigationMenuLink = React.forwardRef< React.ElementRef<typeof NavigationMenuPrimitive.Link>, React.ComponentPropsWithoutRef<typeof NavigationMenuPrimitive.Link> >(({ className, ...props }, ref) => ( <NavigationMenuPrimitive.Link ref={ref} data-uipkge="" data-slot="navigation-menu-link" className={cn( "data-active:focus:bg-accent data-active:hover:bg-accent data-active:bg-accent/50 data-active:text-accent-foreground hover:bg-accent hover:text-accent-foreground focus:bg-accent focus:text-accent-foreground ring-ring/10 dark:ring-ring/20 dark:outline-ring/40 outline-ring/50 [&_svg:not([class*='text-'])]:text-muted-foreground flex flex-col gap-1 rounded-sm p-2 text-sm transition-[color,box-shadow] focus-visible:ring-4 focus-visible:outline-1 [&_svg:not([class*='size-'])]:size-4", className, )} {...props} /> )) NavigationMenuLink.displayName = NavigationMenuPrimitive.Link.displayName const NavigationMenuIndicator = React.forwardRef< React.ElementRef<typeof NavigationMenuPrimitive.Indicator>, React.ComponentPropsWithoutRef<typeof NavigationMenuPrimitive.Indicator> >(({ className, ...props }, ref) => ( <NavigationMenuPrimitive.Indicator ref={ref} data-uipkge="" data-slot="navigation-menu-indicator" className={cn( 'data-[state=visible]:motion-safe:animate-in data-[state=hidden]:motion-safe:animate-out data-[state=hidden]:motion-safe:fade-out data-[state=visible]:motion-safe:fade-in top-full z-[1] flex h-1.5 items-end justify-center overflow-hidden', className, )} {...props} > <div className="bg-border relative top-[60%] h-2 w-2 rotate-45 rounded-tl-sm shadow-md" /> </NavigationMenuPrimitive.Indicator> )) NavigationMenuIndicator.displayName = NavigationMenuPrimitive.Indicator.displayName const NavigationMenuViewport = React.forwardRef< React.ElementRef<typeof NavigationMenuPrimitive.Viewport>, React.ComponentPropsWithoutRef<typeof NavigationMenuPrimitive.Viewport> >(({ className, ...props }, ref) => ( <div className="absolute top-full left-0 isolate z-50 flex justify-center"> <NavigationMenuPrimitive.Viewport ref={ref} data-uipkge="" data-slot="navigation-menu-viewport" className={cn( 'origin-top-center bg-popover text-popover-foreground data-[state=open]:motion-safe:animate-in data-[state=closed]:motion-safe:animate-out data-[state=closed]:motion-safe:zoom-out-95 data-[state=open]:motion-safe:zoom-in-90 relative left-[var(--radix-navigation-menu-viewport-left)] mt-1.5 h-[var(--radix-navigation-menu-viewport-height)] w-full overflow-hidden rounded-md border shadow md:w-[var(--radix-navigation-menu-viewport-width)]', className, )} {...props} /> </div> )) NavigationMenuViewport.displayName = NavigationMenuPrimitive.Viewport.displayName export { NavigationMenu, NavigationMenuList, NavigationMenuItem, NavigationMenuContent, NavigationMenuTrigger, NavigationMenuLink, NavigationMenuIndicator, NavigationMenuViewport, } -
components/ui/navigation-menu/navigation-menu.variants.ts 0.9 kB
import type { VariantProps } from 'class-variance-authority' import { cva } from 'class-variance-authority' /** * Variant definitions live in their own file (rather than the package * `index.ts`) so consuming Vue SFCs can import without creating a circular * dependency through the index. See card.variants.ts for the canonical * example + the SSR symptom that motivated the split. */ export const navigationMenuTriggerStyle = cva( 'group inline-flex h-9 w-max items-center justify-center rounded-md bg-background px-4 py-2 text-sm font-medium hover:bg-accent hover:text-accent-foreground focus:bg-accent focus:text-accent-foreground disabled:pointer-events-none disabled:opacity-50 data-[state=open]:hover:bg-accent data-[state=open]:text-accent-foreground data-[state=open]:focus:bg-accent data-[state=open]:bg-accent/50 focus-visible:ring-ring/50 outline-none transition-[color,box-shadow] focus-visible:ring-[3px] focus-visible:outline-1', ) -
components/ui/navigation-menu/index.ts 0.3 kB
export { NavigationMenu, NavigationMenuList, NavigationMenuItem, NavigationMenuContent, NavigationMenuTrigger, NavigationMenuLink, NavigationMenuIndicator, NavigationMenuViewport, } from './navigation-menu' export { navigationMenuTriggerStyle } from './navigation-menu.variants'
Raw manifest: https://react.uipkge.dev/r/react/navigation-menu.json