mirror of https://github.com/portainer/portainer
57 lines
1.4 KiB
TypeScript
57 lines
1.4 KiB
TypeScript
import { DialogContent, DialogOverlay } from '@reach/dialog';
|
|
import clsx from 'clsx';
|
|
import { createContext, PropsWithChildren, useContext } from 'react';
|
|
|
|
import { CloseButton } from './CloseButton';
|
|
import styles from './Modal.module.css';
|
|
|
|
const Context = createContext<boolean | null>(null);
|
|
Context.displayName = 'ModalContext';
|
|
|
|
export function useModalContext() {
|
|
const context = useContext(Context);
|
|
if (!context) {
|
|
throw new Error('should be nested under Modal');
|
|
}
|
|
|
|
return context;
|
|
}
|
|
|
|
interface Props {
|
|
onDismiss?(): void;
|
|
'aria-label'?: string;
|
|
'aria-labelledby'?: string;
|
|
}
|
|
|
|
export function Modal({
|
|
children,
|
|
onDismiss,
|
|
'aria-label': ariaLabel,
|
|
'aria-labelledby': ariaLabelledBy,
|
|
}: PropsWithChildren<Props>) {
|
|
return (
|
|
<Context.Provider value>
|
|
<DialogOverlay
|
|
isOpen
|
|
className={clsx(
|
|
styles.overlay,
|
|
'flex items-center justify-center z-50'
|
|
)}
|
|
onDismiss={onDismiss}
|
|
role="dialog"
|
|
>
|
|
<DialogContent
|
|
aria-label={ariaLabel}
|
|
aria-labelledby={ariaLabelledBy}
|
|
className={clsx(styles.modalDialog, 'p-0 bg-transparent')}
|
|
>
|
|
<div className={clsx(styles.modalContent, 'relative')}>
|
|
{children}
|
|
{onDismiss && <CloseButton onClose={onDismiss} />}
|
|
</div>
|
|
</DialogContent>
|
|
</DialogOverlay>
|
|
</Context.Provider>
|
|
);
|
|
}
|