dropdown-menu.tsx 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  1. "use client"
  2. import * as React from "react"
  3. import * as DropdownMenuPrimitive from "@radix-ui/react-dropdown-menu"
  4. import { CheckIcon, ChevronRightIcon, CircleIcon } from "lucide-react"
  5. import { cn } from "@/lib/utils"
  6. function DropdownMenu({
  7. ...props
  8. }: React.ComponentProps<typeof DropdownMenuPrimitive.Root>) {
  9. return <DropdownMenuPrimitive.Root data-slot="dropdown-menu" {...props} />
  10. }
  11. function DropdownMenuPortal({
  12. ...props
  13. }: React.ComponentProps<typeof DropdownMenuPrimitive.Portal>) {
  14. return (
  15. <DropdownMenuPrimitive.Portal data-slot="dropdown-menu-portal" {...props} />
  16. )
  17. }
  18. function DropdownMenuTrigger({
  19. ...props
  20. }: React.ComponentProps<typeof DropdownMenuPrimitive.Trigger>) {
  21. return (
  22. <DropdownMenuPrimitive.Trigger
  23. data-slot="dropdown-menu-trigger"
  24. {...props}
  25. />
  26. )
  27. }
  28. function DropdownMenuContent({
  29. className,
  30. sideOffset = 4,
  31. ...props
  32. }: React.ComponentProps<typeof DropdownMenuPrimitive.Content>) {
  33. return (
  34. <DropdownMenuPrimitive.Portal>
  35. <DropdownMenuPrimitive.Content
  36. data-slot="dropdown-menu-content"
  37. sideOffset={sideOffset}
  38. className={cn(
  39. "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 max-h-(--radix-dropdown-menu-content-available-height) min-w-[8rem] origin-(--radix-dropdown-menu-content-transform-origin) overflow-x-hidden overflow-y-auto rounded-md border p-1 shadow-md",
  40. className
  41. )}
  42. {...props}
  43. />
  44. </DropdownMenuPrimitive.Portal>
  45. )
  46. }
  47. function DropdownMenuGroup({
  48. ...props
  49. }: React.ComponentProps<typeof DropdownMenuPrimitive.Group>) {
  50. return (
  51. <DropdownMenuPrimitive.Group data-slot="dropdown-menu-group" {...props} />
  52. )
  53. }
  54. function DropdownMenuItem({
  55. className,
  56. inset,
  57. variant = "default",
  58. ...props
  59. }: React.ComponentProps<typeof DropdownMenuPrimitive.Item> & {
  60. inset?: boolean
  61. variant?: "default" | "destructive"
  62. }) {
  63. return (
  64. <DropdownMenuPrimitive.Item
  65. data-slot="dropdown-menu-item"
  66. data-inset={inset}
  67. data-variant={variant}
  68. className={cn(
  69. "focus:bg-accent focus:text-accent-foreground data-[variant=destructive]:text-destructive data-[variant=destructive]:focus:bg-destructive/10 dark:data-[variant=destructive]:focus:bg-destructive/20 data-[variant=destructive]:focus:text-destructive data-[variant=destructive]:*:[svg]:!text-destructive [&_svg:not([class*='text-'])]:text-muted-foreground relative flex cursor-default items-center gap-2 rounded-sm px-2 py-1.5 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 data-[inset]:pl-8 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
  70. className
  71. )}
  72. {...props}
  73. />
  74. )
  75. }
  76. function DropdownMenuCheckboxItem({
  77. className,
  78. children,
  79. checked,
  80. ...props
  81. }: React.ComponentProps<typeof DropdownMenuPrimitive.CheckboxItem>) {
  82. return (
  83. <DropdownMenuPrimitive.CheckboxItem
  84. data-slot="dropdown-menu-checkbox-item"
  85. className={cn(
  86. "focus:bg-accent focus:text-accent-foreground relative flex cursor-default items-center gap-2 rounded-sm py-1.5 pr-2 pl-8 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
  87. className
  88. )}
  89. checked={checked}
  90. {...props}
  91. >
  92. <span className="pointer-events-none absolute left-2 flex size-3.5 items-center justify-center">
  93. <DropdownMenuPrimitive.ItemIndicator>
  94. <CheckIcon className="size-4" />
  95. </DropdownMenuPrimitive.ItemIndicator>
  96. </span>
  97. {children}
  98. </DropdownMenuPrimitive.CheckboxItem>
  99. )
  100. }
  101. function DropdownMenuRadioGroup({
  102. ...props
  103. }: React.ComponentProps<typeof DropdownMenuPrimitive.RadioGroup>) {
  104. return (
  105. <DropdownMenuPrimitive.RadioGroup
  106. data-slot="dropdown-menu-radio-group"
  107. {...props}
  108. />
  109. )
  110. }
  111. function DropdownMenuRadioItem({
  112. className,
  113. children,
  114. ...props
  115. }: React.ComponentProps<typeof DropdownMenuPrimitive.RadioItem>) {
  116. return (
  117. <DropdownMenuPrimitive.RadioItem
  118. data-slot="dropdown-menu-radio-item"
  119. className={cn(
  120. "focus:bg-accent focus:text-accent-foreground relative flex cursor-default items-center gap-2 rounded-sm py-1.5 pr-2 pl-8 text-sm outline-hidden select-none data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
  121. className
  122. )}
  123. {...props}
  124. >
  125. <span className="pointer-events-none absolute left-2 flex size-3.5 items-center justify-center">
  126. <DropdownMenuPrimitive.ItemIndicator>
  127. <CircleIcon className="size-2 fill-current" />
  128. </DropdownMenuPrimitive.ItemIndicator>
  129. </span>
  130. {children}
  131. </DropdownMenuPrimitive.RadioItem>
  132. )
  133. }
  134. function DropdownMenuLabel({
  135. className,
  136. inset,
  137. ...props
  138. }: React.ComponentProps<typeof DropdownMenuPrimitive.Label> & {
  139. inset?: boolean
  140. }) {
  141. return (
  142. <DropdownMenuPrimitive.Label
  143. data-slot="dropdown-menu-label"
  144. data-inset={inset}
  145. className={cn(
  146. "px-2 py-1.5 text-sm font-medium data-[inset]:pl-8",
  147. className
  148. )}
  149. {...props}
  150. />
  151. )
  152. }
  153. function DropdownMenuSeparator({
  154. className,
  155. ...props
  156. }: React.ComponentProps<typeof DropdownMenuPrimitive.Separator>) {
  157. return (
  158. <DropdownMenuPrimitive.Separator
  159. data-slot="dropdown-menu-separator"
  160. className={cn("bg-border -mx-1 my-1 h-px", className)}
  161. {...props}
  162. />
  163. )
  164. }
  165. function DropdownMenuShortcut({
  166. className,
  167. ...props
  168. }: React.ComponentProps<"span">) {
  169. return (
  170. <span
  171. data-slot="dropdown-menu-shortcut"
  172. className={cn(
  173. "text-muted-foreground ml-auto text-xs tracking-widest",
  174. className
  175. )}
  176. {...props}
  177. />
  178. )
  179. }
  180. function DropdownMenuSub({
  181. ...props
  182. }: React.ComponentProps<typeof DropdownMenuPrimitive.Sub>) {
  183. return <DropdownMenuPrimitive.Sub data-slot="dropdown-menu-sub" {...props} />
  184. }
  185. function DropdownMenuSubTrigger({
  186. className,
  187. inset,
  188. children,
  189. ...props
  190. }: React.ComponentProps<typeof DropdownMenuPrimitive.SubTrigger> & {
  191. inset?: boolean
  192. }) {
  193. return (
  194. <DropdownMenuPrimitive.SubTrigger
  195. data-slot="dropdown-menu-sub-trigger"
  196. data-inset={inset}
  197. className={cn(
  198. "focus:bg-accent focus:text-accent-foreground data-[state=open]:bg-accent data-[state=open]:text-accent-foreground flex cursor-default items-center rounded-sm px-2 py-1.5 text-sm outline-hidden select-none data-[inset]:pl-8",
  199. className
  200. )}
  201. {...props}
  202. >
  203. {children}
  204. <ChevronRightIcon className="ml-auto size-4" />
  205. </DropdownMenuPrimitive.SubTrigger>
  206. )
  207. }
  208. function DropdownMenuSubContent({
  209. className,
  210. ...props
  211. }: React.ComponentProps<typeof DropdownMenuPrimitive.SubContent>) {
  212. return (
  213. <DropdownMenuPrimitive.SubContent
  214. data-slot="dropdown-menu-sub-content"
  215. className={cn(
  216. "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 min-w-[8rem] origin-(--radix-dropdown-menu-content-transform-origin) overflow-hidden rounded-md border p-1 shadow-lg",
  217. className
  218. )}
  219. {...props}
  220. />
  221. )
  222. }
  223. export {
  224. DropdownMenu,
  225. DropdownMenuPortal,
  226. DropdownMenuTrigger,
  227. DropdownMenuContent,
  228. DropdownMenuGroup,
  229. DropdownMenuLabel,
  230. DropdownMenuItem,
  231. DropdownMenuCheckboxItem,
  232. DropdownMenuRadioGroup,
  233. DropdownMenuRadioItem,
  234. DropdownMenuSeparator,
  235. DropdownMenuShortcut,
  236. DropdownMenuSub,
  237. DropdownMenuSubTrigger,
  238. DropdownMenuSubContent,
  239. }