-
Notifications
You must be signed in to change notification settings - Fork 3
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feature(webui): add context menu in breadcrumb links to copy relative…
…/absolute paths
- Loading branch information
Showing
4 changed files
with
232 additions
and
15 deletions.
There are no files selected for viewing
Binary file not shown.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,183 @@ | ||
// see: https://ui.shadcn.com/docs/components/context-menu | ||
|
||
import * as ContextMenuPrimitive from '@radix-ui/react-context-menu'; | ||
import { cx } from 'class-variance-authority'; | ||
// @ts-expect-error | ||
import CheckIcon from 'lucide-react/dist/esm/icons/check'; | ||
// @ts-expect-error | ||
import ChevronRightIcon from 'lucide-react/dist/esm/icons/chevron-right'; | ||
// @ts-expect-error | ||
import DotFilledIcon from 'lucide-react/dist/esm/icons/dot'; | ||
import { | ||
type ComponentPropsWithoutRef, | ||
type ElementRef, | ||
forwardRef, | ||
type HTMLAttributes, | ||
} from 'react'; | ||
|
||
export const ContextMenu = ContextMenuPrimitive.Root; | ||
|
||
export const ContextMenuTrigger = ContextMenuPrimitive.Trigger; | ||
|
||
export const ContextMenuGroup = ContextMenuPrimitive.Group; | ||
|
||
export const ContextMenuPortal = ContextMenuPrimitive.Portal; | ||
|
||
export const ContextMenuSub = ContextMenuPrimitive.Sub; | ||
|
||
export const ContextMenuRadioGroup = ContextMenuPrimitive.RadioGroup; | ||
|
||
export const ContextMenuSubTrigger = forwardRef< | ||
ElementRef<typeof ContextMenuPrimitive.SubTrigger>, | ||
ComponentPropsWithoutRef<typeof ContextMenuPrimitive.SubTrigger> & { | ||
inset?: boolean; | ||
} | ||
>(({ className, inset, children, ...props }, ref) => ( | ||
<ContextMenuPrimitive.SubTrigger | ||
ref={ref} | ||
className={cx( | ||
'flex cursor-default select-none items-center rounded-sm px-2 py-1.5 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[state=open]:bg-accent data-[state=open]:text-accent-foreground', | ||
inset && 'pl-8', | ||
className | ||
)} | ||
{...props} | ||
> | ||
{children} | ||
<ChevronRightIcon className="ml-auto h-4 w-4" /> | ||
</ContextMenuPrimitive.SubTrigger> | ||
)); | ||
ContextMenuSubTrigger.displayName = ContextMenuPrimitive.SubTrigger.displayName; | ||
|
||
export const ContextMenuSubContent = forwardRef< | ||
ElementRef<typeof ContextMenuPrimitive.SubContent>, | ||
ComponentPropsWithoutRef<typeof ContextMenuPrimitive.SubContent> | ||
>(({ className, ...props }, ref) => ( | ||
<ContextMenuPrimitive.SubContent | ||
ref={ref} | ||
className={cx( | ||
'z-50 min-w-[8rem] overflow-hidden rounded-md border bg-popover p-1 text-popover-foreground shadow-lg data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 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', | ||
className | ||
)} | ||
{...props} | ||
/> | ||
)); | ||
ContextMenuSubContent.displayName = ContextMenuPrimitive.SubContent.displayName; | ||
|
||
export const ContextMenuContent = forwardRef< | ||
ElementRef<typeof ContextMenuPrimitive.Content>, | ||
ComponentPropsWithoutRef<typeof ContextMenuPrimitive.Content> | ||
>(({ className, ...props }, ref) => ( | ||
<ContextMenuPrimitive.Portal> | ||
<ContextMenuPrimitive.Content | ||
ref={ref} | ||
className={cx( | ||
'p-1 z-50 min-w-[8rem] overflow-hidden border border-default rounded-md bg-screen text-default shadow-md', | ||
'data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 ', | ||
'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', | ||
className | ||
)} | ||
{...props} | ||
/> | ||
</ContextMenuPrimitive.Portal> | ||
)); | ||
ContextMenuContent.displayName = ContextMenuPrimitive.Content.displayName; | ||
|
||
export const ContextMenuItem = forwardRef< | ||
ElementRef<typeof ContextMenuPrimitive.Item>, | ||
ComponentPropsWithoutRef<typeof ContextMenuPrimitive.Item> & { | ||
inset?: boolean; | ||
} | ||
>(({ className, inset, ...props }, ref) => ( | ||
<ContextMenuPrimitive.Item | ||
ref={ref} | ||
className={cx( | ||
'px-2 py-1.5 relative flex items-center cursor-default select-none outline-none rounded-sm text-sm', | ||
'focus:bg-default focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50', | ||
inset && 'pl-8', | ||
className | ||
)} | ||
{...props} | ||
/> | ||
)); | ||
ContextMenuItem.displayName = ContextMenuPrimitive.Item.displayName; | ||
|
||
export const ContextMenuCheckboxItem = forwardRef< | ||
ElementRef<typeof ContextMenuPrimitive.CheckboxItem>, | ||
ComponentPropsWithoutRef<typeof ContextMenuPrimitive.CheckboxItem> | ||
>(({ className, children, checked, ...props }, ref) => ( | ||
<ContextMenuPrimitive.CheckboxItem | ||
ref={ref} | ||
className={cx( | ||
'relative flex cursor-default select-none items-center rounded-sm py-1.5 pl-8 pr-2 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50', | ||
className | ||
)} | ||
checked={checked} | ||
{...props} | ||
> | ||
<span className="absolute left-2 flex h-3.5 w-3.5 items-center justify-center"> | ||
<ContextMenuPrimitive.ItemIndicator> | ||
<CheckIcon className="h-4 w-4" /> | ||
</ContextMenuPrimitive.ItemIndicator> | ||
</span> | ||
{children} | ||
</ContextMenuPrimitive.CheckboxItem> | ||
)); | ||
ContextMenuCheckboxItem.displayName = ContextMenuPrimitive.CheckboxItem.displayName; | ||
|
||
export const ContextMenuRadioItem = forwardRef< | ||
ElementRef<typeof ContextMenuPrimitive.RadioItem>, | ||
ComponentPropsWithoutRef<typeof ContextMenuPrimitive.RadioItem> | ||
>(({ className, children, ...props }, ref) => ( | ||
<ContextMenuPrimitive.RadioItem | ||
ref={ref} | ||
className={cx( | ||
'relative flex cursor-default select-none items-center rounded-sm py-1.5 pl-8 pr-2 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50', | ||
className | ||
)} | ||
{...props} | ||
> | ||
<span className="absolute left-2 flex h-3.5 w-3.5 items-center justify-center"> | ||
<ContextMenuPrimitive.ItemIndicator> | ||
<DotFilledIcon className="h-4 w-4 fill-current" /> | ||
</ContextMenuPrimitive.ItemIndicator> | ||
</span> | ||
{children} | ||
</ContextMenuPrimitive.RadioItem> | ||
)); | ||
ContextMenuRadioItem.displayName = ContextMenuPrimitive.RadioItem.displayName; | ||
|
||
export const ContextMenuLabel = forwardRef< | ||
ElementRef<typeof ContextMenuPrimitive.Label>, | ||
ComponentPropsWithoutRef<typeof ContextMenuPrimitive.Label> & { | ||
inset?: boolean; | ||
} | ||
>(({ className, inset, ...props }, ref) => ( | ||
<ContextMenuPrimitive.Label | ||
ref={ref} | ||
className={cx('px-2 py-1.5 text-sm font-semibold text-foreground', inset && 'pl-8', className)} | ||
{...props} | ||
/> | ||
)); | ||
ContextMenuLabel.displayName = ContextMenuPrimitive.Label.displayName; | ||
|
||
export const ContextMenuSeparator = forwardRef< | ||
ElementRef<typeof ContextMenuPrimitive.Separator>, | ||
ComponentPropsWithoutRef<typeof ContextMenuPrimitive.Separator> | ||
>(({ className, ...props }, ref) => ( | ||
<ContextMenuPrimitive.Separator | ||
ref={ref} | ||
className={cx('-mx-1 my-1 h-px bg-hover', className)} | ||
{...props} | ||
/> | ||
)); | ||
ContextMenuSeparator.displayName = ContextMenuPrimitive.Separator.displayName; | ||
|
||
export const ContextMenuShortcut = ({ className, ...props }: HTMLAttributes<HTMLSpanElement>) => { | ||
return ( | ||
<span | ||
className={cx('ml-auto text-xs tracking-widest text-muted-foreground', className)} | ||
{...props} | ||
/> | ||
); | ||
}; | ||
ContextMenuShortcut.displayName = 'ContextMenuShortcut'; |