2026-02-17 12:47:39 +05:30
|
|
|
"use client";
|
2026-02-16 00:11:34 +05:30
|
|
|
|
2026-02-17 12:47:39 +05:30
|
|
|
import * as ToolbarPrimitive from "@radix-ui/react-toolbar";
|
|
|
|
|
import * as TooltipPrimitive from "@radix-ui/react-tooltip";
|
2026-02-20 22:44:56 -08:00
|
|
|
import { cva, type VariantProps } from "class-variance-authority";
|
2026-02-17 12:47:39 +05:30
|
|
|
import { ChevronDown } from "lucide-react";
|
2026-02-20 22:44:56 -08:00
|
|
|
import * as React from "react";
|
2026-02-16 00:11:34 +05:30
|
|
|
|
|
|
|
|
import {
|
2026-02-17 12:47:39 +05:30
|
|
|
DropdownMenuLabel,
|
|
|
|
|
DropdownMenuRadioGroup,
|
|
|
|
|
DropdownMenuSeparator,
|
|
|
|
|
} from "@/components/ui/dropdown-menu";
|
|
|
|
|
import { Separator } from "@/components/ui/separator";
|
|
|
|
|
import { Tooltip, TooltipTrigger } from "@/components/ui/tooltip";
|
|
|
|
|
import { cn } from "@/lib/utils";
|
2026-02-16 00:11:34 +05:30
|
|
|
|
|
|
|
|
export function Toolbar({
|
2026-02-17 12:47:39 +05:30
|
|
|
className,
|
|
|
|
|
...props
|
2026-02-16 00:11:34 +05:30
|
|
|
}: React.ComponentProps<typeof ToolbarPrimitive.Root>) {
|
2026-02-17 12:47:39 +05:30
|
|
|
return (
|
|
|
|
|
<ToolbarPrimitive.Root
|
|
|
|
|
className={cn("relative flex select-none items-center", className)}
|
|
|
|
|
{...props}
|
|
|
|
|
/>
|
|
|
|
|
);
|
2026-02-16 00:11:34 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function ToolbarToggleGroup({
|
2026-02-17 12:47:39 +05:30
|
|
|
className,
|
|
|
|
|
...props
|
2026-02-16 00:11:34 +05:30
|
|
|
}: React.ComponentProps<typeof ToolbarPrimitive.ToolbarToggleGroup>) {
|
2026-02-17 12:47:39 +05:30
|
|
|
return (
|
|
|
|
|
<ToolbarPrimitive.ToolbarToggleGroup
|
|
|
|
|
className={cn("flex items-center", className)}
|
|
|
|
|
{...props}
|
|
|
|
|
/>
|
|
|
|
|
);
|
2026-02-16 00:11:34 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function ToolbarLink({
|
2026-02-17 12:47:39 +05:30
|
|
|
className,
|
|
|
|
|
...props
|
2026-02-16 00:11:34 +05:30
|
|
|
}: React.ComponentProps<typeof ToolbarPrimitive.Link>) {
|
2026-02-17 12:47:39 +05:30
|
|
|
return (
|
|
|
|
|
<ToolbarPrimitive.Link
|
|
|
|
|
className={cn("font-medium underline underline-offset-4", className)}
|
|
|
|
|
{...props}
|
|
|
|
|
/>
|
|
|
|
|
);
|
2026-02-16 00:11:34 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function ToolbarSeparator({
|
2026-02-17 12:47:39 +05:30
|
|
|
className,
|
|
|
|
|
...props
|
2026-02-16 00:11:34 +05:30
|
|
|
}: React.ComponentProps<typeof ToolbarPrimitive.Separator>) {
|
2026-02-17 12:47:39 +05:30
|
|
|
return (
|
|
|
|
|
<ToolbarPrimitive.Separator
|
|
|
|
|
className={cn("mx-2 my-1 w-px shrink-0 bg-border", className)}
|
|
|
|
|
{...props}
|
|
|
|
|
/>
|
|
|
|
|
);
|
2026-02-16 00:11:34 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// From toggleVariants
|
|
|
|
|
const toolbarButtonVariants = cva(
|
2026-02-17 12:47:39 +05:30
|
|
|
"inline-flex cursor-pointer items-center justify-center gap-2 whitespace-nowrap rounded-md font-medium text-sm outline-none transition-[color,box-shadow] hover:bg-muted hover:text-muted-foreground focus-visible:border-ring focus-visible:ring-[3px] focus-visible:ring-ring/50 disabled:pointer-events-none disabled:opacity-50 aria-checked:bg-accent aria-checked:text-accent-foreground aria-invalid:border-destructive aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 [&_svg:not([class*='size-'])]:size-4 [&_svg]:pointer-events-none [&_svg]:shrink-0",
|
|
|
|
|
{
|
|
|
|
|
defaultVariants: {
|
|
|
|
|
size: "default",
|
|
|
|
|
variant: "default",
|
|
|
|
|
},
|
|
|
|
|
variants: {
|
|
|
|
|
size: {
|
|
|
|
|
default: "h-9 min-w-9 px-2",
|
|
|
|
|
lg: "h-10 min-w-10 px-2.5",
|
|
|
|
|
sm: "h-8 min-w-8 px-1.5",
|
|
|
|
|
},
|
|
|
|
|
variant: {
|
|
|
|
|
default: "bg-transparent",
|
|
|
|
|
outline:
|
|
|
|
|
"border border-input bg-transparent shadow-xs hover:bg-accent hover:text-accent-foreground",
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
2026-02-16 00:11:34 +05:30
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const dropdownArrowVariants = cva(
|
2026-02-17 12:47:39 +05:30
|
|
|
cn(
|
|
|
|
|
"inline-flex items-center justify-center rounded-r-md font-medium text-foreground text-sm transition-colors disabled:pointer-events-none disabled:opacity-50"
|
|
|
|
|
),
|
|
|
|
|
{
|
|
|
|
|
defaultVariants: {
|
|
|
|
|
size: "sm",
|
|
|
|
|
variant: "default",
|
|
|
|
|
},
|
|
|
|
|
variants: {
|
|
|
|
|
size: {
|
|
|
|
|
default: "h-9 w-6",
|
|
|
|
|
lg: "h-10 w-8",
|
|
|
|
|
sm: "h-8 w-4",
|
|
|
|
|
},
|
|
|
|
|
variant: {
|
|
|
|
|
default:
|
|
|
|
|
"bg-transparent hover:bg-muted hover:text-muted-foreground aria-checked:bg-accent aria-checked:text-accent-foreground",
|
|
|
|
|
outline:
|
|
|
|
|
"border border-input border-l-0 bg-transparent hover:bg-accent hover:text-accent-foreground",
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
2026-02-16 00:11:34 +05:30
|
|
|
);
|
|
|
|
|
|
|
|
|
|
type ToolbarButtonProps = {
|
2026-02-17 12:47:39 +05:30
|
|
|
isDropdown?: boolean;
|
|
|
|
|
pressed?: boolean;
|
|
|
|
|
} & Omit<React.ComponentPropsWithoutRef<typeof ToolbarToggleItem>, "asChild" | "value"> &
|
|
|
|
|
VariantProps<typeof toolbarButtonVariants>;
|
2026-02-16 00:11:34 +05:30
|
|
|
|
|
|
|
|
export const ToolbarButton = withTooltip(function ToolbarButton({
|
2026-02-17 12:47:39 +05:30
|
|
|
children,
|
|
|
|
|
className,
|
|
|
|
|
isDropdown,
|
|
|
|
|
pressed,
|
|
|
|
|
size = "sm",
|
|
|
|
|
variant,
|
|
|
|
|
...props
|
2026-02-16 00:11:34 +05:30
|
|
|
}: ToolbarButtonProps) {
|
2026-02-17 12:47:39 +05:30
|
|
|
return typeof pressed === "boolean" ? (
|
|
|
|
|
<ToolbarToggleGroup disabled={props.disabled} value="single" type="single">
|
|
|
|
|
<ToolbarToggleItem
|
|
|
|
|
className={cn(
|
|
|
|
|
toolbarButtonVariants({
|
|
|
|
|
size,
|
|
|
|
|
variant,
|
|
|
|
|
}),
|
|
|
|
|
isDropdown && "justify-between gap-1 pr-1",
|
|
|
|
|
className
|
|
|
|
|
)}
|
|
|
|
|
value={pressed ? "single" : ""}
|
|
|
|
|
{...props}
|
|
|
|
|
>
|
|
|
|
|
{isDropdown ? (
|
|
|
|
|
<>
|
|
|
|
|
<div className="flex flex-1 items-center gap-2 whitespace-nowrap">{children}</div>
|
|
|
|
|
<div>
|
|
|
|
|
<ChevronDown className="size-3.5 text-muted-foreground" data-icon />
|
|
|
|
|
</div>
|
|
|
|
|
</>
|
|
|
|
|
) : (
|
|
|
|
|
children
|
|
|
|
|
)}
|
|
|
|
|
</ToolbarToggleItem>
|
|
|
|
|
</ToolbarToggleGroup>
|
|
|
|
|
) : (
|
|
|
|
|
<ToolbarPrimitive.Button
|
|
|
|
|
className={cn(
|
|
|
|
|
toolbarButtonVariants({
|
|
|
|
|
size,
|
|
|
|
|
variant,
|
|
|
|
|
}),
|
|
|
|
|
isDropdown && "pr-1",
|
|
|
|
|
className
|
|
|
|
|
)}
|
|
|
|
|
{...props}
|
|
|
|
|
>
|
|
|
|
|
{children}
|
|
|
|
|
</ToolbarPrimitive.Button>
|
|
|
|
|
);
|
2026-02-16 00:11:34 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
|
|
export function ToolbarSplitButton({
|
2026-02-17 12:47:39 +05:30
|
|
|
className,
|
|
|
|
|
...props
|
2026-02-16 00:11:34 +05:30
|
|
|
}: React.ComponentPropsWithoutRef<typeof ToolbarButton>) {
|
2026-02-17 12:47:39 +05:30
|
|
|
return (
|
|
|
|
|
<ToolbarButton
|
|
|
|
|
className={cn("group flex gap-0 px-0 hover:bg-transparent", className)}
|
|
|
|
|
{...props}
|
|
|
|
|
/>
|
|
|
|
|
);
|
2026-02-16 00:11:34 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type ToolbarSplitButtonPrimaryProps = Omit<
|
2026-02-17 12:47:39 +05:30
|
|
|
React.ComponentPropsWithoutRef<typeof ToolbarToggleItem>,
|
|
|
|
|
"value"
|
2026-02-16 00:11:34 +05:30
|
|
|
> &
|
2026-02-17 12:47:39 +05:30
|
|
|
VariantProps<typeof toolbarButtonVariants>;
|
2026-02-16 00:11:34 +05:30
|
|
|
|
|
|
|
|
export function ToolbarSplitButtonPrimary({
|
2026-02-17 12:47:39 +05:30
|
|
|
children,
|
|
|
|
|
className,
|
|
|
|
|
size = "sm",
|
|
|
|
|
variant,
|
|
|
|
|
...props
|
2026-02-16 00:11:34 +05:30
|
|
|
}: ToolbarSplitButtonPrimaryProps) {
|
2026-02-17 12:47:39 +05:30
|
|
|
return (
|
|
|
|
|
<span
|
|
|
|
|
className={cn(
|
|
|
|
|
toolbarButtonVariants({
|
|
|
|
|
size,
|
|
|
|
|
variant,
|
|
|
|
|
}),
|
|
|
|
|
"rounded-r-none",
|
|
|
|
|
"group-data-[pressed=true]:bg-accent group-data-[pressed=true]:text-accent-foreground",
|
|
|
|
|
className
|
|
|
|
|
)}
|
|
|
|
|
{...props}
|
|
|
|
|
>
|
|
|
|
|
{children}
|
|
|
|
|
</span>
|
|
|
|
|
);
|
2026-02-16 00:11:34 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function ToolbarSplitButtonSecondary({
|
2026-02-17 12:47:39 +05:30
|
|
|
className,
|
|
|
|
|
size,
|
|
|
|
|
variant,
|
|
|
|
|
...props
|
|
|
|
|
}: React.ComponentPropsWithoutRef<"span"> & VariantProps<typeof dropdownArrowVariants>) {
|
|
|
|
|
return (
|
|
|
|
|
<span
|
|
|
|
|
className={cn(
|
|
|
|
|
dropdownArrowVariants({
|
|
|
|
|
size,
|
|
|
|
|
variant,
|
|
|
|
|
}),
|
|
|
|
|
"group-data-[pressed=true]:bg-accent group-data-[pressed=true]:text-accent-foreground",
|
|
|
|
|
className
|
|
|
|
|
)}
|
|
|
|
|
onClick={(e) => e.stopPropagation()}
|
|
|
|
|
role="button"
|
|
|
|
|
{...props}
|
|
|
|
|
>
|
|
|
|
|
<ChevronDown className="size-3.5 text-muted-foreground" data-icon />
|
|
|
|
|
</span>
|
|
|
|
|
);
|
2026-02-16 00:11:34 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export function ToolbarToggleItem({
|
2026-02-17 12:47:39 +05:30
|
|
|
className,
|
|
|
|
|
size = "sm",
|
|
|
|
|
variant,
|
|
|
|
|
...props
|
2026-02-16 00:11:34 +05:30
|
|
|
}: React.ComponentProps<typeof ToolbarPrimitive.ToggleItem> &
|
2026-02-17 12:47:39 +05:30
|
|
|
VariantProps<typeof toolbarButtonVariants>) {
|
|
|
|
|
return (
|
|
|
|
|
<ToolbarPrimitive.ToggleItem
|
|
|
|
|
className={cn(toolbarButtonVariants({ size, variant }), className)}
|
|
|
|
|
{...props}
|
|
|
|
|
/>
|
|
|
|
|
);
|
2026-02-16 00:11:34 +05:30
|
|
|
}
|
|
|
|
|
|
2026-02-17 12:47:39 +05:30
|
|
|
export function ToolbarGroup({ children, className }: React.ComponentProps<"div">) {
|
|
|
|
|
return (
|
|
|
|
|
<div className={cn("group/toolbar-group", "relative hidden has-[button]:flex", className)}>
|
|
|
|
|
<div className="flex items-center">{children}</div>
|
2026-02-16 00:11:34 +05:30
|
|
|
|
2026-02-17 12:47:39 +05:30
|
|
|
<div className="group-last/toolbar-group:hidden! mx-1.5 py-0.5">
|
|
|
|
|
<Separator orientation="vertical" />
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
);
|
2026-02-16 00:11:34 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type TooltipProps<T extends React.ElementType> = {
|
2026-02-17 12:47:39 +05:30
|
|
|
tooltip?: React.ReactNode;
|
|
|
|
|
tooltipContentProps?: Omit<React.ComponentPropsWithoutRef<typeof TooltipContent>, "children">;
|
|
|
|
|
tooltipProps?: Omit<React.ComponentPropsWithoutRef<typeof Tooltip>, "children">;
|
|
|
|
|
tooltipTriggerProps?: React.ComponentPropsWithoutRef<typeof TooltipTrigger>;
|
2026-02-16 00:11:34 +05:30
|
|
|
} & React.ComponentProps<T>;
|
|
|
|
|
|
|
|
|
|
function withTooltip<T extends React.ElementType>(Component: T) {
|
2026-02-17 12:47:39 +05:30
|
|
|
return function ExtendComponent({
|
|
|
|
|
tooltip,
|
|
|
|
|
tooltipContentProps,
|
|
|
|
|
tooltipProps,
|
|
|
|
|
tooltipTriggerProps,
|
|
|
|
|
...props
|
|
|
|
|
}: TooltipProps<T>) {
|
|
|
|
|
const [mounted, setMounted] = React.useState(false);
|
2026-02-16 00:11:34 +05:30
|
|
|
|
2026-02-17 12:47:39 +05:30
|
|
|
React.useEffect(() => {
|
|
|
|
|
setMounted(true);
|
|
|
|
|
}, []);
|
2026-02-16 00:11:34 +05:30
|
|
|
|
2026-02-17 12:47:39 +05:30
|
|
|
const component = <Component {...(props as React.ComponentProps<T>)} />;
|
2026-02-16 00:11:34 +05:30
|
|
|
|
2026-02-17 12:47:39 +05:30
|
|
|
if (tooltip && mounted) {
|
|
|
|
|
return (
|
|
|
|
|
<Tooltip {...tooltipProps}>
|
|
|
|
|
<TooltipTrigger asChild {...tooltipTriggerProps}>
|
|
|
|
|
{component}
|
|
|
|
|
</TooltipTrigger>
|
2026-02-16 00:11:34 +05:30
|
|
|
|
2026-02-17 12:47:39 +05:30
|
|
|
<TooltipContent {...tooltipContentProps}>{tooltip}</TooltipContent>
|
|
|
|
|
</Tooltip>
|
|
|
|
|
);
|
|
|
|
|
}
|
2026-02-16 00:11:34 +05:30
|
|
|
|
2026-02-17 12:47:39 +05:30
|
|
|
return component;
|
|
|
|
|
};
|
2026-02-16 00:11:34 +05:30
|
|
|
}
|
|
|
|
|
|
2026-02-17 01:30:38 +05:30
|
|
|
function TooltipContent({
|
2026-02-17 12:47:39 +05:30
|
|
|
children,
|
|
|
|
|
className,
|
|
|
|
|
sideOffset = 4,
|
|
|
|
|
...props
|
2026-02-17 01:30:38 +05:30
|
|
|
}: React.ComponentProps<typeof TooltipPrimitive.Content>) {
|
2026-02-17 12:47:39 +05:30
|
|
|
return (
|
|
|
|
|
<TooltipPrimitive.Portal>
|
|
|
|
|
<TooltipPrimitive.Content
|
|
|
|
|
className={cn(
|
|
|
|
|
"bg-black text-white font-medium shadow-xl px-3 py-1.5 dark:bg-zinc-800 dark:text-zinc-50 border-none animate-in fade-in-0 zoom-in-95 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=closed]:zoom-out-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 z-50 w-fit rounded-md text-xs text-balance pointer-events-none",
|
|
|
|
|
className
|
|
|
|
|
)}
|
|
|
|
|
data-slot="tooltip-content"
|
|
|
|
|
sideOffset={sideOffset}
|
|
|
|
|
{...props}
|
|
|
|
|
>
|
|
|
|
|
{children}
|
|
|
|
|
</TooltipPrimitive.Content>
|
|
|
|
|
</TooltipPrimitive.Portal>
|
|
|
|
|
);
|
2026-02-17 01:30:38 +05:30
|
|
|
}
|
|
|
|
|
|
2026-02-16 00:11:34 +05:30
|
|
|
export function ToolbarMenuGroup({
|
2026-02-17 12:47:39 +05:30
|
|
|
children,
|
|
|
|
|
className,
|
|
|
|
|
label,
|
|
|
|
|
...props
|
2026-02-16 00:11:34 +05:30
|
|
|
}: React.ComponentProps<typeof DropdownMenuRadioGroup> & { label?: string }) {
|
2026-02-17 12:47:39 +05:30
|
|
|
return (
|
|
|
|
|
<>
|
|
|
|
|
<DropdownMenuSeparator
|
|
|
|
|
className={cn(
|
|
|
|
|
"hidden",
|
|
|
|
|
"mb-0 mx-2 shrink-0 peer-has-[[role=menuitem]]/menu-group:block peer-has-[[role=menuitemradio]]/menu-group:block peer-has-[[role=option]]/menu-group:block",
|
|
|
|
|
"dark:bg-neutral-700"
|
|
|
|
|
)}
|
|
|
|
|
/>
|
2026-02-16 00:11:34 +05:30
|
|
|
|
2026-02-17 12:47:39 +05:30
|
|
|
<DropdownMenuRadioGroup
|
|
|
|
|
{...props}
|
|
|
|
|
className={cn(
|
|
|
|
|
"hidden",
|
|
|
|
|
"peer/menu-group group/menu-group my-1.5 has-[[role=menuitem]]:block has-[[role=menuitemradio]]:block has-[[role=option]]:block",
|
|
|
|
|
className
|
|
|
|
|
)}
|
|
|
|
|
>
|
|
|
|
|
{label && (
|
|
|
|
|
<DropdownMenuLabel className="select-none font-semibold text-muted-foreground text-xs">
|
|
|
|
|
{label}
|
|
|
|
|
</DropdownMenuLabel>
|
|
|
|
|
)}
|
|
|
|
|
{children}
|
|
|
|
|
</DropdownMenuRadioGroup>
|
|
|
|
|
</>
|
|
|
|
|
);
|
2026-02-16 00:11:34 +05:30
|
|
|
}
|