mirror of
https://github.com/mountain-loop/yaak.git
synced 2026-01-15 16:23:25 +01:00
93 lines
2.8 KiB
TypeScript
93 lines
2.8 KiB
TypeScript
import classNames from 'classnames';
|
|
import { motion } from 'framer-motion';
|
|
import type { ReactNode } from 'react';
|
|
import { useMemo } from 'react';
|
|
import { useKeyPressEvent } from 'react-use';
|
|
import { Overlay } from '../Overlay';
|
|
import { Heading } from './Heading';
|
|
import { IconButton } from './IconButton';
|
|
|
|
export interface DialogProps {
|
|
children: ReactNode;
|
|
open: boolean;
|
|
onClose: () => void;
|
|
title?: ReactNode;
|
|
description?: ReactNode;
|
|
className?: string;
|
|
size?: 'sm' | 'md' | 'full' | 'dynamic';
|
|
hideX?: boolean;
|
|
}
|
|
|
|
export function Dialog({
|
|
children,
|
|
className,
|
|
size = 'full',
|
|
open,
|
|
onClose,
|
|
title,
|
|
description,
|
|
hideX,
|
|
}: DialogProps) {
|
|
const titleId = useMemo(() => Math.random().toString(36).slice(2), []);
|
|
const descriptionId = useMemo(
|
|
() => (description ? Math.random().toString(36).slice(2) : undefined),
|
|
[description],
|
|
);
|
|
|
|
useKeyPressEvent('Escape', (e) => {
|
|
e.preventDefault();
|
|
onClose();
|
|
});
|
|
|
|
return (
|
|
<Overlay open={open} onClose={onClose} portalName="dialog">
|
|
<div className="absolute inset-0 flex items-center justify-center pointer-events-none">
|
|
<div
|
|
role="dialog"
|
|
aria-labelledby={titleId}
|
|
aria-describedby={descriptionId}
|
|
className="pointer-events-auto"
|
|
>
|
|
<motion.div
|
|
initial={{ top: 5, scale: 0.97 }}
|
|
animate={{ top: 0, scale: 1 }}
|
|
className={classNames(
|
|
className,
|
|
'gap-2 grid grid-rows-[auto_minmax(0,1fr)]',
|
|
'relative bg-gray-50 pointer-events-auto',
|
|
'p-5 rounded-lg overflow-auto',
|
|
'dark:border border-highlight shadow shadow-black/10',
|
|
'max-w-[90vw] max-h-[calc(100vh-8em)]',
|
|
size === 'sm' && 'w-[25rem] max-h-[80vh]',
|
|
size === 'md' && 'w-[45rem] max-h-[80vh]',
|
|
size === 'full' && 'w-[100vw] h-[100vh]',
|
|
size === 'dynamic' && 'min-w-[30vw] max-w-[80vw]',
|
|
)}
|
|
>
|
|
{title ? (
|
|
<Heading className="text-xl font-semibold w-full" id={titleId}>
|
|
{title}
|
|
</Heading>
|
|
) : (
|
|
<span />
|
|
)}
|
|
{description && <p id={descriptionId}>{description}</p>}
|
|
<div className="h-full w-full">{children}</div>
|
|
{/*Put close at the end so that it's the last thing to be tabbed to*/}
|
|
{!hideX && (
|
|
<IconButton
|
|
onClick={onClose}
|
|
title="Close dialog"
|
|
aria-label="Close"
|
|
icon="x"
|
|
size="sm"
|
|
className="ml-auto absolute right-1 top-1"
|
|
/>
|
|
)}
|
|
</motion.div>
|
|
</div>
|
|
</div>
|
|
</Overlay>
|
|
);
|
|
}
|