152 lines
5.2 KiB
TypeScript
152 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;
|
||
|