2022-08-31 02:35:06 -07:00
|
|
|
import classNames from 'clsx';
|
2022-11-25 09:04:11 -08:00
|
|
|
import React from 'react';
|
2022-03-21 11:09:01 -07:00
|
|
|
import { defineMessages, useIntl } from 'react-intl';
|
|
|
|
|
|
|
|
import Button from '../button/button';
|
2022-08-16 11:21:04 -07:00
|
|
|
import { ButtonThemes } from '../button/useButtonStyles';
|
2022-11-29 13:49:35 -08:00
|
|
|
import HStack from '../hstack/hstack';
|
2022-03-21 11:09:01 -07:00
|
|
|
import IconButton from '../icon-button/icon-button';
|
|
|
|
|
|
|
|
const messages = defineMessages({
|
|
|
|
close: { id: 'lightbox.close', defaultMessage: 'Close' },
|
|
|
|
confirm: { id: 'confirmations.delete.confirm', defaultMessage: 'Delete' },
|
|
|
|
});
|
|
|
|
|
2022-05-17 08:45:38 -07:00
|
|
|
const widths = {
|
|
|
|
xs: 'max-w-xs',
|
|
|
|
sm: 'max-w-sm',
|
|
|
|
md: 'max-w-base',
|
|
|
|
lg: 'max-w-lg',
|
|
|
|
xl: 'max-w-xl',
|
|
|
|
'2xl': 'max-w-2xl',
|
|
|
|
'3xl': 'max-w-3xl',
|
|
|
|
'4xl': 'max-w-4xl',
|
|
|
|
};
|
|
|
|
|
2022-03-21 11:09:01 -07:00
|
|
|
interface IModal {
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Callback when the modal is cancelled. */
|
2022-03-21 11:09:01 -07:00
|
|
|
cancelAction?: () => void,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Cancel button text. */
|
2022-06-12 07:14:46 -07:00
|
|
|
cancelText?: React.ReactNode,
|
2022-05-15 00:06:40 -07:00
|
|
|
/** URL to an SVG icon for the close button. */
|
|
|
|
closeIcon?: string,
|
|
|
|
/** Position of the close button. */
|
|
|
|
closePosition?: 'left' | 'right',
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Callback when the modal is confirmed. */
|
2022-05-18 11:08:08 -07:00
|
|
|
confirmationAction?: (event?: React.MouseEvent<HTMLButtonElement>) => void,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Whether the confirmation button is disabled. */
|
2022-03-21 11:09:01 -07:00
|
|
|
confirmationDisabled?: boolean,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Confirmation button text. */
|
2022-05-11 13:26:37 -07:00
|
|
|
confirmationText?: React.ReactNode,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Confirmation button theme. */
|
2022-08-16 11:21:04 -07:00
|
|
|
confirmationTheme?: ButtonThemes,
|
2022-12-15 14:51:30 -08:00
|
|
|
/** Whether to use full width style for confirmation button. */
|
|
|
|
confirmationFullWidth?: boolean,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Callback when the modal is closed. */
|
2022-04-07 08:01:38 -07:00
|
|
|
onClose?: () => void,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Callback when the secondary action is chosen. */
|
2022-05-18 11:08:08 -07:00
|
|
|
secondaryAction?: (event?: React.MouseEvent<HTMLButtonElement>) => void,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Secondary button text. */
|
2022-05-11 13:26:37 -07:00
|
|
|
secondaryText?: React.ReactNode,
|
2022-05-18 11:08:08 -07:00
|
|
|
secondaryDisabled?: boolean,
|
2022-04-27 07:11:21 -07:00
|
|
|
/** Don't focus the "confirm" button on mount. */
|
|
|
|
skipFocus?: boolean,
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Title text for the modal. */
|
2022-06-21 15:29:17 -07:00
|
|
|
title?: React.ReactNode,
|
2022-10-04 12:08:22 -07:00
|
|
|
width?: keyof typeof widths,
|
2023-01-10 15:03:15 -08:00
|
|
|
children?: React.ReactNode,
|
2022-03-21 11:09:01 -07:00
|
|
|
}
|
|
|
|
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Displays a modal dialog box. */
|
2022-03-21 11:09:01 -07:00
|
|
|
const Modal: React.FC<IModal> = ({
|
|
|
|
cancelAction,
|
|
|
|
cancelText,
|
|
|
|
children,
|
2022-07-09 09:20:02 -07:00
|
|
|
closeIcon = require('@tabler/icons/x.svg'),
|
2022-05-15 00:06:40 -07:00
|
|
|
closePosition = 'right',
|
2022-03-21 11:09:01 -07:00
|
|
|
confirmationAction,
|
|
|
|
confirmationDisabled,
|
|
|
|
confirmationText,
|
|
|
|
confirmationTheme,
|
2022-12-15 14:51:30 -08:00
|
|
|
confirmationFullWidth,
|
2022-03-21 11:09:01 -07:00
|
|
|
onClose,
|
|
|
|
secondaryAction,
|
2022-05-18 11:08:08 -07:00
|
|
|
secondaryDisabled = false,
|
2022-03-21 11:09:01 -07:00
|
|
|
secondaryText,
|
2022-04-27 07:11:21 -07:00
|
|
|
skipFocus = false,
|
2022-03-21 11:09:01 -07:00
|
|
|
title,
|
2022-05-17 08:45:38 -07:00
|
|
|
width = 'xl',
|
2022-03-21 11:09:01 -07:00
|
|
|
}) => {
|
|
|
|
const intl = useIntl();
|
|
|
|
const buttonRef = React.useRef<HTMLButtonElement>(null);
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
2022-04-27 07:11:21 -07:00
|
|
|
if (buttonRef?.current && !skipFocus) {
|
2022-03-21 11:09:01 -07:00
|
|
|
buttonRef.current.focus();
|
|
|
|
}
|
2022-04-27 07:11:21 -07:00
|
|
|
}, [skipFocus, buttonRef]);
|
2022-03-21 11:09:01 -07:00
|
|
|
|
|
|
|
return (
|
2022-11-25 09:04:11 -08:00
|
|
|
<div data-testid='modal' className={classNames('block w-full p-6 mx-auto text-start align-middle transition-all transform bg-white dark:bg-primary-900 text-gray-900 dark:text-gray-100 shadow-xl rounded-2xl pointer-events-auto', widths[width])}>
|
2022-03-21 11:09:01 -07:00
|
|
|
<div className='sm:flex sm:items-start w-full justify-between'>
|
|
|
|
<div className='w-full'>
|
2022-06-21 15:29:17 -07:00
|
|
|
{title && (
|
|
|
|
<div
|
|
|
|
className={classNames('w-full flex items-center gap-2', {
|
|
|
|
'flex-row-reverse': closePosition === 'left',
|
|
|
|
})}
|
|
|
|
>
|
2022-07-22 10:30:16 -07:00
|
|
|
<h3 className='flex-grow text-lg leading-6 font-bold text-gray-900 dark:text-white'>
|
2022-06-21 15:29:17 -07:00
|
|
|
{title}
|
|
|
|
</h3>
|
2022-03-21 11:09:01 -07:00
|
|
|
|
2022-06-21 15:29:17 -07:00
|
|
|
{onClose && (
|
|
|
|
<IconButton
|
|
|
|
src={closeIcon}
|
|
|
|
title={intl.formatMessage(messages.close)}
|
|
|
|
onClick={onClose}
|
2022-12-04 17:40:09 -08:00
|
|
|
className='text-gray-500 hover:text-gray-700 dark:text-gray-300 dark:hover:text-gray-200 rtl:rotate-180'
|
2022-06-21 15:29:17 -07:00
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
)}
|
2022-03-21 11:09:01 -07:00
|
|
|
|
2022-06-21 15:29:17 -07:00
|
|
|
{title ? (
|
|
|
|
<div className='w-full mt-2'>
|
|
|
|
{children}
|
|
|
|
</div>
|
|
|
|
) : children}
|
2022-03-21 11:09:01 -07:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
{confirmationAction && (
|
2022-11-29 13:49:35 -08:00
|
|
|
<HStack className='mt-5' justifyContent='between' data-testid='modal-actions'>
|
2022-12-15 14:51:30 -08:00
|
|
|
<div className={classNames({ 'flex-grow': !confirmationFullWidth })}>
|
2022-03-21 11:09:01 -07:00
|
|
|
{cancelAction && (
|
|
|
|
<Button
|
2022-07-22 10:30:16 -07:00
|
|
|
theme='tertiary'
|
2022-03-21 11:09:01 -07:00
|
|
|
onClick={cancelAction}
|
|
|
|
>
|
2022-04-27 07:11:21 -07:00
|
|
|
{cancelText || 'Cancel'}
|
2022-03-21 11:09:01 -07:00
|
|
|
</Button>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
|
2022-12-15 14:51:30 -08:00
|
|
|
<HStack space={2} className={classNames({ 'flex-grow': confirmationFullWidth })}>
|
2022-03-21 11:09:01 -07:00
|
|
|
{secondaryAction && (
|
|
|
|
<Button
|
|
|
|
theme='secondary'
|
|
|
|
onClick={secondaryAction}
|
2022-05-18 11:08:08 -07:00
|
|
|
disabled={secondaryDisabled}
|
2022-03-21 11:09:01 -07:00
|
|
|
>
|
|
|
|
{secondaryText}
|
|
|
|
</Button>
|
|
|
|
)}
|
|
|
|
|
|
|
|
<Button
|
|
|
|
theme={confirmationTheme || 'primary'}
|
|
|
|
onClick={confirmationAction}
|
|
|
|
disabled={confirmationDisabled}
|
|
|
|
ref={buttonRef}
|
2022-12-15 14:51:30 -08:00
|
|
|
block={confirmationFullWidth}
|
2022-03-21 11:09:01 -07:00
|
|
|
>
|
|
|
|
{confirmationText}
|
|
|
|
</Button>
|
2022-11-29 13:49:35 -08:00
|
|
|
</HStack>
|
|
|
|
</HStack>
|
2022-03-21 11:09:01 -07:00
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default Modal;
|