select.tsx 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. import * as SelectPrimitive from '@radix-ui/react-select';
  2. import { CheckIcon, ChevronDownIcon, ChevronUpIcon } from 'lucide-react';
  3. import * as React from 'react';
  4. import { cn } from '@/common/helpers/cn';
  5. function Select({ ...props }: React.ComponentProps<typeof SelectPrimitive.Root>) {
  6. return <SelectPrimitive.Root data-slot="select" {...props} />;
  7. }
  8. function SelectGroup({ ...props }: React.ComponentProps<typeof SelectPrimitive.Group>) {
  9. return <SelectPrimitive.Group data-slot="select-group" {...props} />;
  10. }
  11. function SelectValue({ ...props }: React.ComponentProps<typeof SelectPrimitive.Value>) {
  12. return <SelectPrimitive.Value data-slot="select-value" {...props} />;
  13. }
  14. function SelectTrigger({ className, children, ...props }: React.ComponentProps<typeof SelectPrimitive.Trigger>) {
  15. return (
  16. <SelectPrimitive.Trigger
  17. data-slot="select-trigger"
  18. className={cn(
  19. "flex h-9 w-full items-center justify-between rounded-md border border-input bg-transparent px-3 py-2 text-sm shadow-xs transition-[color,box-shadow] outline-none focus-visible:border-ring focus-visible:ring-[3px] focus-visible:ring-ring/50 disabled:cursor-not-allowed disabled:opacity-50 aria-invalid:border-destructive aria-invalid:ring-destructive/20 data-[placeholder]:text-muted-foreground *:data-[slot=select-value]:flex *:data-[slot=select-value]:items-center *:data-[slot=select-value]:gap-2 dark:aria-invalid:ring-destructive/40 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4 [&_svg:not([class*='text-'])]:text-muted-foreground [&>span]:line-clamp-1",
  20. className,
  21. )}
  22. {...props}
  23. >
  24. {children}
  25. <SelectPrimitive.Icon asChild>
  26. <ChevronDownIcon className="size-4 opacity-50" />
  27. </SelectPrimitive.Icon>
  28. </SelectPrimitive.Trigger>
  29. );
  30. }
  31. function SelectContent({ className, children, position = 'popper', ...props }: React.ComponentProps<typeof SelectPrimitive.Content>) {
  32. return (
  33. <SelectPrimitive.Portal>
  34. <SelectPrimitive.Content
  35. data-slot="select-content"
  36. className={cn(
  37. 'relative z-50 max-h-96 min-w-[8rem] overflow-hidden rounded-md border bg-popover 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',
  38. position === 'popper' &&
  39. 'data-[side=bottom]:translate-y-1 data-[side=left]:-translate-x-1 data-[side=right]:translate-x-1 data-[side=top]:-translate-y-1',
  40. className,
  41. )}
  42. position={position}
  43. {...props}
  44. >
  45. <SelectScrollUpButton />
  46. <SelectPrimitive.Viewport
  47. className={cn(
  48. 'p-1',
  49. position === 'popper' &&
  50. 'h-[var(--radix-select-trigger-height)] w-full min-w-[var(--radix-select-trigger-width)] scroll-my-1',
  51. )}
  52. >
  53. {children}
  54. </SelectPrimitive.Viewport>
  55. <SelectScrollDownButton />
  56. </SelectPrimitive.Content>
  57. </SelectPrimitive.Portal>
  58. );
  59. }
  60. function SelectLabel({ className, ...props }: React.ComponentProps<typeof SelectPrimitive.Label>) {
  61. return <SelectPrimitive.Label data-slot="select-label" className={cn('px-2 py-1.5 text-sm font-medium', className)} {...props} />;
  62. }
  63. function SelectItem({ className, children, ...props }: React.ComponentProps<typeof SelectPrimitive.Item>) {
  64. return (
  65. <SelectPrimitive.Item
  66. data-slot="select-item"
  67. className={cn(
  68. "relative flex w-full cursor-default items-center gap-2 rounded-sm py-1.5 pr-8 pl-2 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 [&_svg:not([class*='text-'])]:text-muted-foreground *:[span]:last:flex *:[span]:last:items-center *:[span]:last:gap-2",
  69. className,
  70. )}
  71. {...props}
  72. >
  73. <span className="absolute right-2 flex size-3.5 items-center justify-center">
  74. <SelectPrimitive.ItemIndicator>
  75. <CheckIcon className="size-4" />
  76. </SelectPrimitive.ItemIndicator>
  77. </span>
  78. <SelectPrimitive.ItemText>{children}</SelectPrimitive.ItemText>
  79. </SelectPrimitive.Item>
  80. );
  81. }
  82. function SelectSeparator({ className, ...props }: React.ComponentProps<typeof SelectPrimitive.Separator>) {
  83. return (
  84. <SelectPrimitive.Separator
  85. data-slot="select-separator"
  86. className={cn('pointer-events-none -mx-1 my-1 h-px bg-border', className)}
  87. {...props}
  88. />
  89. );
  90. }
  91. function SelectScrollUpButton({ className, ...props }: React.ComponentProps<typeof SelectPrimitive.ScrollUpButton>) {
  92. return (
  93. <SelectPrimitive.ScrollUpButton
  94. data-slot="select-scroll-up-button"
  95. className={cn('flex cursor-default items-center justify-center py-1', className)}
  96. {...props}
  97. >
  98. <ChevronUpIcon className="size-4" />
  99. </SelectPrimitive.ScrollUpButton>
  100. );
  101. }
  102. function SelectScrollDownButton({ className, ...props }: React.ComponentProps<typeof SelectPrimitive.ScrollDownButton>) {
  103. return (
  104. <SelectPrimitive.ScrollDownButton
  105. data-slot="select-scroll-down-button"
  106. className={cn('flex cursor-default items-center justify-center py-1', className)}
  107. {...props}
  108. >
  109. <ChevronDownIcon className="size-4" />
  110. </SelectPrimitive.ScrollDownButton>
  111. );
  112. }
  113. export {
  114. Select,
  115. SelectContent,
  116. SelectGroup,
  117. SelectItem,
  118. SelectLabel,
  119. SelectScrollDownButton,
  120. SelectScrollUpButton,
  121. SelectSeparator,
  122. SelectTrigger,
  123. SelectValue,
  124. };