dropdown-menu.tsx 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. import * as DropdownMenuPrimitive from '@radix-ui/react-dropdown-menu';
  2. import { CheckIcon, ChevronRightIcon, CircleIcon } from 'lucide-react';
  3. import * as React from 'react';
  4. import { cn } from '@/common/helpers/cn';
  5. function DropdownMenu({ ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Root>) {
  6. return <DropdownMenuPrimitive.Root data-slot="dropdown-menu" {...props} />;
  7. }
  8. function DropdownMenuPortal({ ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Portal>) {
  9. return <DropdownMenuPrimitive.Portal data-slot="dropdown-menu-portal" {...props} />;
  10. }
  11. function DropdownMenuTrigger({ ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Trigger>) {
  12. return <DropdownMenuPrimitive.Trigger data-slot="dropdown-menu-trigger" {...props} />;
  13. }
  14. function DropdownMenuContent({ className, sideOffset = 4, ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Content>) {
  15. return (
  16. <DropdownMenuPrimitive.Portal>
  17. <DropdownMenuPrimitive.Content
  18. data-slot="dropdown-menu-content"
  19. sideOffset={sideOffset}
  20. className={cn(
  21. 'z-50 min-w-[8rem] overflow-hidden rounded-md border bg-popover p-1 text-popover-foreground shadow-md 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 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=closed]:zoom-out-95 data-[state=open]:animate-in data-[state=open]:fade-in-0 data-[state=open]:zoom-in-95',
  22. className,
  23. )}
  24. {...props}
  25. />
  26. </DropdownMenuPrimitive.Portal>
  27. );
  28. }
  29. function DropdownMenuGroup({ ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Group>) {
  30. return <DropdownMenuPrimitive.Group data-slot="dropdown-menu-group" {...props} />;
  31. }
  32. function DropdownMenuItem({
  33. className,
  34. inset,
  35. variant = 'default',
  36. ...props
  37. }: React.ComponentProps<typeof DropdownMenuPrimitive.Item> & {
  38. inset?: boolean;
  39. variant?: 'default' | 'destructive';
  40. }) {
  41. return (
  42. <DropdownMenuPrimitive.Item
  43. data-slot="dropdown-menu-item"
  44. data-inset={inset}
  45. data-variant={variant}
  46. className={cn(
  47. "relative flex cursor-default items-center gap-2 rounded-sm px-2 py-1.5 text-sm outline-hidden select-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50 data-[inset]:pl-8 data-[variant=destructive]:text-destructive-foreground data-[variant=destructive]:focus:bg-destructive/10 data-[variant=destructive]:focus:text-destructive-foreground dark:data-[variant=destructive]:focus:bg-destructive/40 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4 [&_svg:not([class*='text-'])]:text-muted-foreground data-[variant=destructive]:*:[svg]:!text-destructive-foreground",
  48. className,
  49. )}
  50. {...props}
  51. />
  52. );
  53. }
  54. function DropdownMenuCheckboxItem({ className, children, checked, ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.CheckboxItem>) {
  55. return (
  56. <DropdownMenuPrimitive.CheckboxItem
  57. data-slot="dropdown-menu-checkbox-item"
  58. className={cn(
  59. "relative flex cursor-default items-center gap-2 rounded-sm py-1.5 pr-2 pl-8 text-sm outline-hidden select-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
  60. className,
  61. )}
  62. checked={checked}
  63. {...props}
  64. >
  65. <span className="pointer-events-none absolute left-2 flex size-3.5 items-center justify-center">
  66. <DropdownMenuPrimitive.ItemIndicator>
  67. <CheckIcon className="size-4" />
  68. </DropdownMenuPrimitive.ItemIndicator>
  69. </span>
  70. {children}
  71. </DropdownMenuPrimitive.CheckboxItem>
  72. );
  73. }
  74. function DropdownMenuRadioGroup({ ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.RadioGroup>) {
  75. return <DropdownMenuPrimitive.RadioGroup data-slot="dropdown-menu-radio-group" {...props} />;
  76. }
  77. function DropdownMenuRadioItem({ className, children, ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.RadioItem>) {
  78. return (
  79. <DropdownMenuPrimitive.RadioItem
  80. data-slot="dropdown-menu-radio-item"
  81. className={cn(
  82. "relative flex cursor-default items-center gap-2 rounded-sm py-1.5 pr-2 pl-8 text-sm outline-hidden select-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
  83. className,
  84. )}
  85. {...props}
  86. >
  87. <span className="pointer-events-none absolute left-2 flex size-3.5 items-center justify-center">
  88. <DropdownMenuPrimitive.ItemIndicator>
  89. <CircleIcon className="size-2 fill-current" />
  90. </DropdownMenuPrimitive.ItemIndicator>
  91. </span>
  92. {children}
  93. </DropdownMenuPrimitive.RadioItem>
  94. );
  95. }
  96. function DropdownMenuLabel({
  97. className,
  98. inset,
  99. ...props
  100. }: React.ComponentProps<typeof DropdownMenuPrimitive.Label> & {
  101. inset?: boolean;
  102. }) {
  103. return (
  104. <DropdownMenuPrimitive.Label
  105. data-slot="dropdown-menu-label"
  106. data-inset={inset}
  107. className={cn('px-2 py-1.5 text-sm font-medium data-[inset]:pl-8', className)}
  108. {...props}
  109. />
  110. );
  111. }
  112. function DropdownMenuSeparator({ className, ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Separator>) {
  113. return <DropdownMenuPrimitive.Separator data-slot="dropdown-menu-separator" className={cn('-mx-1 my-1 h-px bg-border', className)} {...props} />;
  114. }
  115. function DropdownMenuShortcut({ className, ...props }: React.ComponentProps<'span'>) {
  116. return <span data-slot="dropdown-menu-shortcut" className={cn('ml-auto text-xs tracking-widest text-muted-foreground', className)} {...props} />;
  117. }
  118. function DropdownMenuSub({ ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Sub>) {
  119. return <DropdownMenuPrimitive.Sub data-slot="dropdown-menu-sub" {...props} />;
  120. }
  121. function DropdownMenuSubTrigger({
  122. className,
  123. inset,
  124. children,
  125. ...props
  126. }: React.ComponentProps<typeof DropdownMenuPrimitive.SubTrigger> & {
  127. inset?: boolean;
  128. }) {
  129. return (
  130. <DropdownMenuPrimitive.SubTrigger
  131. data-slot="dropdown-menu-sub-trigger"
  132. data-inset={inset}
  133. className={cn(
  134. 'flex cursor-default items-center rounded-sm px-2 py-1.5 text-sm outline-hidden select-none focus:bg-accent focus:text-accent-foreground data-[inset]:pl-8 data-[state=open]:bg-accent data-[state=open]:text-accent-foreground',
  135. className,
  136. )}
  137. {...props}
  138. >
  139. {children}
  140. <ChevronRightIcon className="ml-auto size-4" />
  141. </DropdownMenuPrimitive.SubTrigger>
  142. );
  143. }
  144. function DropdownMenuSubContent({ className, ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.SubContent>) {
  145. return (
  146. <DropdownMenuPrimitive.SubContent
  147. data-slot="dropdown-menu-sub-content"
  148. className={cn(
  149. 'z-50 min-w-[8rem] overflow-hidden rounded-md border bg-popover p-1 text-popover-foreground shadow-lg 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 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=closed]:zoom-out-95 data-[state=open]:animate-in data-[state=open]:fade-in-0 data-[state=open]:zoom-in-95',
  150. className,
  151. )}
  152. {...props}
  153. />
  154. );
  155. }
  156. export {
  157. DropdownMenu,
  158. DropdownMenuCheckboxItem,
  159. DropdownMenuContent,
  160. DropdownMenuGroup,
  161. DropdownMenuItem,
  162. DropdownMenuLabel,
  163. DropdownMenuPortal,
  164. DropdownMenuRadioGroup,
  165. DropdownMenuRadioItem,
  166. DropdownMenuSeparator,
  167. DropdownMenuShortcut,
  168. DropdownMenuSub,
  169. DropdownMenuSubContent,
  170. DropdownMenuSubTrigger,
  171. DropdownMenuTrigger,
  172. };