PteroTheme/resources/scripts/components/elements/dialog/Dialog.tsx

133 lines
5.5 KiB
TypeScript
Raw Normal View History

2022-11-25 20:25:03 +00:00
import { useRef, useState } from 'react';
import * as React from 'react';
2022-06-12 20:07:52 +01:00
import { Dialog as HDialog } from '@headlessui/react';
import { Button } from '@/components/elements/button/index';
import { XIcon } from '@heroicons/react/solid';
import { AnimatePresence, motion } from 'framer-motion';
import { DialogContext, IconPosition, RenderDialogProps, styles } from './';
2022-07-02 23:27:22 +01:00
const variants = {
open: {
scale: 1,
opacity: 1,
transition: {
type: 'spring',
damping: 15,
stiffness: 300,
duration: 0.15,
},
},
closed: {
scale: 0.75,
opacity: 0,
transition: {
type: 'easeIn',
duration: 0.15,
},
},
2022-07-02 23:27:22 +01:00
bounce: {
scale: 0.95,
opacity: 1,
2022-07-02 23:27:22 +01:00
transition: { type: 'linear', duration: 0.075 },
},
};
export default ({
2022-07-02 23:27:22 +01:00
open,
title,
description,
onClose,
hideCloseIcon,
preventExternalClose,
children,
}: RenderDialogProps) => {
const container = useRef<HTMLDivElement>(null);
2022-07-02 23:27:22 +01:00
const [icon, setIcon] = useState<React.ReactNode>();
const [footer, setFooter] = useState<React.ReactNode>();
const [iconPosition, setIconPosition] = useState<IconPosition>('title');
const [down, setDown] = useState(false);
const onContainerClick = (down: boolean, e: React.MouseEvent<HTMLDivElement>): void => {
if (e.target instanceof HTMLElement && container.current?.isSameNode(e.target)) {
setDown(down);
}
};
2022-07-02 23:27:22 +01:00
const onDialogClose = (): void => {
if (!preventExternalClose) {
return onClose();
}
};
return (
2022-06-12 20:07:52 +01:00
<AnimatePresence>
{open && (
2022-07-02 23:27:22 +01:00
<DialogContext.Provider value={{ setIcon, setFooter, setIconPosition }}>
<HDialog
static
as={motion.div}
initial={{ opacity: 0 }}
animate={{ opacity: 1 }}
exit={{ opacity: 0 }}
transition={{ duration: 0.15 }}
open={open}
2022-07-02 23:27:22 +01:00
onClose={onDialogClose}
>
<div className={'fixed inset-0 bg-gray-900/50 z-40'} />
<div className={'fixed inset-0 overflow-y-auto z-50'}>
<div
ref={container}
className={styles.container}
onMouseDown={onContainerClick.bind(this, true)}
onMouseUp={onContainerClick.bind(this, false)}
>
<HDialog.Panel
as={motion.div}
initial={'closed'}
animate={down ? 'bounce' : 'open'}
exit={'closed'}
2022-07-02 23:27:22 +01:00
variants={variants}
className={styles.panel}
>
<div className={'flex p-6 pb-0 overflow-y-auto'}>
2022-07-02 23:27:22 +01:00
{iconPosition === 'container' && icon}
<div className={'flex-1 max-h-[70vh] min-w-0'}>
<div className={'flex items-center'}>
2022-07-02 23:27:22 +01:00
{iconPosition !== 'container' && icon}
<div>
{title && (
<HDialog.Title className={styles.title}>{title}</HDialog.Title>
)}
{description && (
<HDialog.Description>{description}</HDialog.Description>
)}
</div>
</div>
{children}
<div className={'invisible h-6'} />
</div>
2022-06-12 20:07:52 +01:00
</div>
2022-07-02 23:27:22 +01:00
{footer}
{/* Keep this below the other buttons so that it isn't the default focus if they're present. */}
{!hideCloseIcon && (
<div className={'absolute right-0 top-0 m-4'}>
<Button.Text
size={Button.Sizes.Small}
shape={Button.Shapes.IconSquare}
onClick={onClose}
className={'group'}
>
<XIcon className={styles.close_icon} />
</Button.Text>
</div>
)}
</HDialog.Panel>
</div>
</div>
</HDialog>
</DialogContext.Provider>
2022-06-12 20:07:52 +01:00
)}
</AnimatePresence>
);
};