153 lines
5.2 KiB
TypeScript
153 lines
5.2 KiB
TypeScript
|
|
// components/ui/select.tsx
|
|||
|
|
"use client";
|
|||
|
|
|
|||
|
|
import * as React from "react";
|
|||
|
|
import * as SelectPrimitive from "@radix-ui/react-select";
|
|||
|
|
import { Check, ChevronDown, ChevronUp } from "lucide-react";
|
|||
|
|
import { cn } from "@/lib/utils";
|
|||
|
|
|
|||
|
|
export const Select = SelectPrimitive.Root;
|
|||
|
|
export const SelectGroup = SelectPrimitive.Group;
|
|||
|
|
export const SelectValue = SelectPrimitive.Value;
|
|||
|
|
|
|||
|
|
export const SelectTrigger = React.forwardRef<
|
|||
|
|
React.ElementRef<typeof SelectPrimitive.Trigger>,
|
|||
|
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Trigger>
|
|||
|
|
>(({ className, children, ...props }, ref) => (
|
|||
|
|
<SelectPrimitive.Trigger
|
|||
|
|
ref={ref}
|
|||
|
|
className={cn(
|
|||
|
|
"flex h-10 w-full items-center justify-between rounded-md border border-neutral-700",
|
|||
|
|
"bg-neutral-900 px-3 text-sm text-neutral-100 outline-none",
|
|||
|
|
"ring-offset-neutral-900 transition-colors",
|
|||
|
|
"placeholder:text-neutral-400",
|
|||
|
|
"focus:border-neutral-500 focus:ring-1 focus:ring-neutral-500",
|
|||
|
|
"data-[state=open]:border-neutral-600",
|
|||
|
|
className
|
|||
|
|
)}
|
|||
|
|
{...props}
|
|||
|
|
>
|
|||
|
|
{children}
|
|||
|
|
<SelectPrimitive.Icon className="ml-2 opacity-80">
|
|||
|
|
<ChevronDown className="h-4 w-4" />
|
|||
|
|
</SelectPrimitive.Icon>
|
|||
|
|
</SelectPrimitive.Trigger>
|
|||
|
|
));
|
|||
|
|
SelectTrigger.displayName = SelectPrimitive.Trigger.displayName;
|
|||
|
|
|
|||
|
|
export const SelectContent = React.forwardRef<
|
|||
|
|
React.ElementRef<typeof SelectPrimitive.Content>,
|
|||
|
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Content>
|
|||
|
|
>(({ className, children, position = "popper", ...props }, ref) => (
|
|||
|
|
<SelectPrimitive.Portal>
|
|||
|
|
<SelectPrimitive.Content
|
|||
|
|
ref={ref}
|
|||
|
|
position={position}
|
|||
|
|
className={cn(
|
|||
|
|
// 🔒 Explicit background + border so it’s never transparent
|
|||
|
|
"z-50 overflow-hidden rounded-md border border-neutral-700",
|
|||
|
|
"bg-neutral-900 text-neutral-100 shadow-lg",
|
|||
|
|
"data-[state=open]:animate-in data-[state=open]:fade-in-0 data-[state=open]:zoom-in-95",
|
|||
|
|
"data-[side=bottom]:slide-in-from-top-2 data-[side=top]:slide-in-from-bottom-2",
|
|||
|
|
className
|
|||
|
|
)}
|
|||
|
|
{...props}
|
|||
|
|
>
|
|||
|
|
<SelectScrollUpButton />
|
|||
|
|
<SelectPrimitive.Viewport className="p-1 bg-neutral-900">
|
|||
|
|
{children}
|
|||
|
|
</SelectPrimitive.Viewport>
|
|||
|
|
<SelectScrollDownButton />
|
|||
|
|
</SelectPrimitive.Content>
|
|||
|
|
</SelectPrimitive.Portal>
|
|||
|
|
));
|
|||
|
|
SelectContent.displayName = SelectPrimitive.Content.displayName;
|
|||
|
|
|
|||
|
|
export const SelectLabel = React.forwardRef<
|
|||
|
|
React.ElementRef<typeof SelectPrimitive.Label>,
|
|||
|
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Label>
|
|||
|
|
>(({ className, ...props }, ref) => (
|
|||
|
|
<SelectPrimitive.Label
|
|||
|
|
ref={ref}
|
|||
|
|
className={cn("px-2 py-1.5 text-xs font-medium text-neutral-300", className)}
|
|||
|
|
{...props}
|
|||
|
|
/>
|
|||
|
|
));
|
|||
|
|
SelectLabel.displayName = SelectPrimitive.Label.displayName;
|
|||
|
|
|
|||
|
|
export const SelectItem = React.forwardRef<
|
|||
|
|
React.ElementRef<typeof SelectPrimitive.Item>,
|
|||
|
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Item>
|
|||
|
|
>(({ className, children, ...props }, ref) => (
|
|||
|
|
<SelectPrimitive.Item
|
|||
|
|
ref={ref}
|
|||
|
|
className={cn(
|
|||
|
|
"relative flex w-full cursor-default select-none items-center rounded-sm",
|
|||
|
|
"px-2 py-1.5 text-sm outline-none",
|
|||
|
|
"text-neutral-100",
|
|||
|
|
"focus:bg-neutral-800 focus:text-neutral-100",
|
|||
|
|
"data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
|
|||
|
|
className
|
|||
|
|
)}
|
|||
|
|
{...props}
|
|||
|
|
>
|
|||
|
|
<span className="mr-2 inline-flex h-4 w-4 items-center justify-center">
|
|||
|
|
<SelectPrimitive.ItemIndicator>
|
|||
|
|
<Check className="h-4 w-4" />
|
|||
|
|
</SelectPrimitive.ItemIndicator>
|
|||
|
|
</span>
|
|||
|
|
<SelectPrimitive.ItemText>{children}</SelectPrimitive.ItemText>
|
|||
|
|
</SelectPrimitive.Item>
|
|||
|
|
));
|
|||
|
|
SelectItem.displayName = SelectPrimitive.Item.displayName;
|
|||
|
|
|
|||
|
|
export const SelectSeparator = React.forwardRef<
|
|||
|
|
React.ElementRef<typeof SelectPrimitive.Separator>,
|
|||
|
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Separator>
|
|||
|
|
>(({ className, ...props }, ref) => (
|
|||
|
|
<SelectPrimitive.Separator
|
|||
|
|
ref={ref}
|
|||
|
|
className={cn("my-1 h-px bg-neutral-700", className)}
|
|||
|
|
{...props}
|
|||
|
|
/>
|
|||
|
|
));
|
|||
|
|
SelectSeparator.displayName = SelectPrimitive.Separator.displayName;
|
|||
|
|
|
|||
|
|
export const SelectScrollUpButton = React.forwardRef<
|
|||
|
|
React.ElementRef<typeof SelectPrimitive.ScrollUpButton>,
|
|||
|
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.ScrollUpButton>
|
|||
|
|
>(({ className, ...props }, ref) => (
|
|||
|
|
<SelectPrimitive.ScrollUpButton
|
|||
|
|
ref={ref}
|
|||
|
|
className={cn(
|
|||
|
|
"flex cursor-default items-center justify-center py-1",
|
|||
|
|
"bg-neutral-900 text-neutral-300",
|
|||
|
|
className
|
|||
|
|
)}
|
|||
|
|
{...props}
|
|||
|
|
>
|
|||
|
|
<ChevronUp className="h-4 w-4" />
|
|||
|
|
</SelectPrimitive.ScrollUpButton>
|
|||
|
|
));
|
|||
|
|
SelectScrollUpButton.displayName = SelectPrimitive.ScrollUpButton.displayName;
|
|||
|
|
|
|||
|
|
export const SelectScrollDownButton = React.forwardRef<
|
|||
|
|
React.ElementRef<typeof SelectPrimitive.ScrollDownButton>,
|
|||
|
|
React.ComponentPropsWithoutRef<typeof SelectPrimitive.ScrollDownButton>
|
|||
|
|
>(({ className, ...props }, ref) => (
|
|||
|
|
<SelectPrimitive.ScrollDownButton
|
|||
|
|
ref={ref}
|
|||
|
|
className={cn(
|
|||
|
|
"flex cursor-default items-center justify-center py-1",
|
|||
|
|
"bg-neutral-900 text-neutral-300",
|
|||
|
|
className
|
|||
|
|
)}
|
|||
|
|
{...props}
|
|||
|
|
>
|
|||
|
|
<ChevronDown className="h-4 w-4" />
|
|||
|
|
</SelectPrimitive.ScrollDownButton>
|
|||
|
|
));
|
|||
|
|
SelectScrollDownButton.displayName =
|
|||
|
|
SelectPrimitive.ScrollDownButton.displayName;
|
|||
|
|
|