mirror of
https://github.com/netbirdio/netbird.git
synced 2026-05-20 15:49:55 +00:00
131 lines
5.0 KiB
TypeScript
131 lines
5.0 KiB
TypeScript
import { forwardRef, ComponentPropsWithoutRef, ElementRef, HTMLAttributes } from "react";
|
|
import * as DialogPrimitive from "@radix-ui/react-dialog";
|
|
import { VisuallyHidden } from "@radix-ui/react-visually-hidden";
|
|
import { X } from "lucide-react";
|
|
import { cn } from "@/lib/cn";
|
|
|
|
export const Root = DialogPrimitive.Root;
|
|
export const Trigger = DialogPrimitive.Trigger;
|
|
export const Close = DialogPrimitive.Close;
|
|
export const Portal = DialogPrimitive.Portal;
|
|
|
|
export const Overlay = forwardRef<
|
|
ElementRef<typeof DialogPrimitive.Overlay>,
|
|
ComponentPropsWithoutRef<typeof DialogPrimitive.Overlay>
|
|
>(function DialogOverlay({ className, ...props }, ref) {
|
|
return (
|
|
<DialogPrimitive.Overlay
|
|
ref={ref}
|
|
className={cn(
|
|
"fixed inset-0 z-50 grid items-center justify-items-center overflow-y-auto px-10 py-16",
|
|
"bg-black/40 backdrop-blur-sm",
|
|
"data-[state=open]:animate-in data-[state=closed]:animate-out",
|
|
"data-[state=open]:fade-in-0 data-[state=closed]:fade-out-0",
|
|
"duration-150 ease-out",
|
|
className,
|
|
)}
|
|
{...props}
|
|
/>
|
|
);
|
|
});
|
|
|
|
type ContentProps = ComponentPropsWithoutRef<typeof DialogPrimitive.Content> & {
|
|
showClose?: boolean;
|
|
maxWidthClass?: string;
|
|
};
|
|
|
|
export const Content = forwardRef<ElementRef<typeof DialogPrimitive.Content>, ContentProps>(
|
|
function DialogContent(
|
|
{ className, children, showClose = true, maxWidthClass = "max-w-md", ...props },
|
|
ref,
|
|
) {
|
|
return (
|
|
<DialogPrimitive.Portal>
|
|
<Overlay>
|
|
<DialogPrimitive.Content
|
|
ref={ref}
|
|
className={cn(
|
|
"mx-auto relative z-[52] w-full outline-none ring-0",
|
|
"focus:outline-none focus-visible:outline-none focus:ring-0 focus-visible:ring-0",
|
|
"border border-nb-gray-900 bg-nb-gray py-7 shadow-2xl rounded-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-[state=closed]:slide-out-to-left-1 data-[state=open]:slide-in-from-left-1",
|
|
"duration-150 ease-out",
|
|
maxWidthClass,
|
|
className,
|
|
)}
|
|
onClick={(e) => e.stopPropagation()}
|
|
{...props}
|
|
>
|
|
<VisuallyHidden asChild>
|
|
<DialogPrimitive.Title>Dialog</DialogPrimitive.Title>
|
|
</VisuallyHidden>
|
|
{children}
|
|
{showClose && (
|
|
<DialogPrimitive.Close
|
|
className={cn(
|
|
"absolute right-3 top-3 z-10 rounded-md p-2 transition-colors",
|
|
"text-nb-gray-300 hover:text-nb-gray-100 hover:bg-nb-gray-900",
|
|
"focus:outline-none disabled:pointer-events-none",
|
|
)}
|
|
aria-label="Close"
|
|
>
|
|
<X className="h-4 w-4" />
|
|
</DialogPrimitive.Close>
|
|
)}
|
|
</DialogPrimitive.Content>
|
|
</Overlay>
|
|
</DialogPrimitive.Portal>
|
|
);
|
|
},
|
|
);
|
|
|
|
export const Title = forwardRef<
|
|
ElementRef<typeof DialogPrimitive.Title>,
|
|
ComponentPropsWithoutRef<typeof DialogPrimitive.Title>
|
|
>(function DialogTitle({ className, ...props }, ref) {
|
|
return (
|
|
<DialogPrimitive.Title
|
|
ref={ref}
|
|
className={cn(
|
|
"text-md font-semibold leading-none tracking-tight text-nb-gray-50",
|
|
className,
|
|
)}
|
|
{...props}
|
|
/>
|
|
);
|
|
});
|
|
|
|
export const Description = forwardRef<
|
|
ElementRef<typeof DialogPrimitive.Description>,
|
|
ComponentPropsWithoutRef<typeof DialogPrimitive.Description>
|
|
>(function DialogDescription({ className, ...props }, ref) {
|
|
return (
|
|
<DialogPrimitive.Description
|
|
ref={ref}
|
|
className={cn("text-sm text-nb-gray-400 mt-2 leading-snug", className)}
|
|
{...props}
|
|
/>
|
|
);
|
|
});
|
|
|
|
type FooterProps = HTMLAttributes<HTMLDivElement> & {
|
|
separator?: boolean;
|
|
};
|
|
|
|
export const Footer = ({ className, separator = true, ...props }: FooterProps) => (
|
|
<div className={cn(separator && "border-t border-nb-gray-900 mt-6")}>
|
|
<div
|
|
className={cn(
|
|
"flex flex-col-reverse gap-3 sm:flex-row sm:justify-end",
|
|
"[&>*]:w-full sm:[&>*]:w-auto",
|
|
"px-8 pt-6",
|
|
className,
|
|
)}
|
|
{...props}
|
|
/>
|
|
</div>
|
|
);
|