command.tsx
1 import * as React from "react"; 2 import { type DialogProps } from "@radix-ui/react-dialog"; 3 import { MagnifyingGlassIcon } from "@radix-ui/react-icons"; 4 import { Command as CommandPrimitive } from "cmdk"; 5 6 import { cn } from "../../lib/utils"; 7 import { Dialog, DialogContent } from "./dialog"; 8 9 const Command = React.forwardRef< 10 React.ElementRef<typeof CommandPrimitive>, 11 React.ComponentPropsWithoutRef<typeof CommandPrimitive> 12 >(({ className, ...props }, ref) => ( 13 <CommandPrimitive 14 ref={ref} 15 className={cn( 16 "flex h-full w-full flex-col overflow-hidden rounded-md bg-white text-slate-950 dark:bg-slate-950 dark:text-slate-50", 17 className, 18 )} 19 {...props} 20 /> 21 )); 22 Command.displayName = CommandPrimitive.displayName; 23 24 interface CommandDialogProps extends DialogProps {} 25 26 const CommandDialog = ({ children, ...props }: CommandDialogProps) => { 27 return ( 28 <Dialog {...props}> 29 <DialogContent className="overflow-hidden p-0"> 30 <Command className="[&_[cmdk-group-heading]]:px-2 [&_[cmdk-group-heading]]:font-medium [&_[cmdk-group-heading]]:text-slate-500 [&_[cmdk-group]:not([hidden])_~[cmdk-group]]:pt-0 [&_[cmdk-group]]:px-2 [&_[cmdk-input-wrapper]_svg]:h-5 [&_[cmdk-input-wrapper]_svg]:w-5 [&_[cmdk-input]]:h-12 [&_[cmdk-item]]:px-2 [&_[cmdk-item]]:py-3 [&_[cmdk-item]_svg]:h-5 [&_[cmdk-item]_svg]:w-5 dark:[&_[cmdk-group-heading]]:text-slate-400"> 31 {children} 32 </Command> 33 </DialogContent> 34 </Dialog> 35 ); 36 }; 37 38 const CommandInput = React.forwardRef< 39 React.ElementRef<typeof CommandPrimitive.Input>, 40 React.ComponentPropsWithoutRef<typeof CommandPrimitive.Input> 41 >(({ className, ...props }, ref) => ( 42 <div className="flex items-center border-b px-3" cmdk-input-wrapper=""> 43 <MagnifyingGlassIcon className="mr-2 h-4 w-4 shrink-0 opacity-50" /> 44 <CommandPrimitive.Input 45 ref={ref} 46 className={cn( 47 "flex h-10 w-full rounded-md bg-transparent py-3 text-sm outline-none placeholder:text-slate-500 disabled:cursor-not-allowed disabled:opacity-50 dark:placeholder:text-slate-400", 48 className, 49 )} 50 {...props} 51 /> 52 </div> 53 )); 54 55 CommandInput.displayName = CommandPrimitive.Input.displayName; 56 57 const CommandList = React.forwardRef< 58 React.ElementRef<typeof CommandPrimitive.List>, 59 React.ComponentPropsWithoutRef<typeof CommandPrimitive.List> 60 >(({ className, ...props }, ref) => ( 61 <CommandPrimitive.List 62 ref={ref} 63 className={cn("max-h-[300px] overflow-y-auto overflow-x-hidden", className)} 64 {...props} 65 /> 66 )); 67 68 CommandList.displayName = CommandPrimitive.List.displayName; 69 70 const CommandEmpty = React.forwardRef< 71 React.ElementRef<typeof CommandPrimitive.Empty>, 72 React.ComponentPropsWithoutRef<typeof CommandPrimitive.Empty> 73 >((props, ref) => ( 74 <CommandPrimitive.Empty 75 ref={ref} 76 className="py-6 text-center text-sm" 77 {...props} 78 /> 79 )); 80 81 CommandEmpty.displayName = CommandPrimitive.Empty.displayName; 82 83 const CommandGroup = React.forwardRef< 84 React.ElementRef<typeof CommandPrimitive.Group>, 85 React.ComponentPropsWithoutRef<typeof CommandPrimitive.Group> 86 >(({ className, ...props }, ref) => ( 87 <CommandPrimitive.Group 88 ref={ref} 89 className={cn( 90 "overflow-hidden p-1 text-slate-950 [&_[cmdk-group-heading]]:px-2 [&_[cmdk-group-heading]]:py-1.5 [&_[cmdk-group-heading]]:text-xs [&_[cmdk-group-heading]]:font-medium [&_[cmdk-group-heading]]:text-slate-500 dark:text-slate-50 dark:[&_[cmdk-group-heading]]:text-slate-400", 91 className, 92 )} 93 {...props} 94 /> 95 )); 96 97 CommandGroup.displayName = CommandPrimitive.Group.displayName; 98 99 const CommandSeparator = React.forwardRef< 100 React.ElementRef<typeof CommandPrimitive.Separator>, 101 React.ComponentPropsWithoutRef<typeof CommandPrimitive.Separator> 102 >(({ className, ...props }, ref) => ( 103 <CommandPrimitive.Separator 104 ref={ref} 105 className={cn("-mx-1 h-px bg-slate-200 dark:bg-slate-800", className)} 106 {...props} 107 /> 108 )); 109 CommandSeparator.displayName = CommandPrimitive.Separator.displayName; 110 111 const CommandItem = React.forwardRef< 112 React.ElementRef<typeof CommandPrimitive.Item>, 113 React.ComponentPropsWithoutRef<typeof CommandPrimitive.Item> 114 >(({ className, ...props }, ref) => ( 115 <CommandPrimitive.Item 116 ref={ref} 117 className={cn( 118 "relative flex cursor-default select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none aria-selected:bg-slate-100 aria-selected:text-slate-900 data-[disabled]:pointer-events-none data-[disabled]:opacity-50 dark:aria-selected:bg-slate-800 dark:aria-selected:text-slate-50", 119 className, 120 )} 121 {...props} 122 /> 123 )); 124 125 CommandItem.displayName = CommandPrimitive.Item.displayName; 126 127 const CommandShortcut = ({ 128 className, 129 ...props 130 }: React.HTMLAttributes<HTMLSpanElement>) => { 131 return ( 132 <span 133 className={cn( 134 "ml-auto text-xs tracking-widest text-slate-500 dark:text-slate-400", 135 className, 136 )} 137 {...props} 138 /> 139 ); 140 }; 141 CommandShortcut.displayName = "CommandShortcut"; 142 143 export { 144 Command, 145 CommandDialog, 146 CommandInput, 147 CommandList, 148 CommandEmpty, 149 CommandGroup, 150 CommandItem, 151 CommandShortcut, 152 CommandSeparator, 153 };