225 lines
6.1 KiB
TypeScript
225 lines
6.1 KiB
TypeScript
// Copyright 2021-2022 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import type { ReactElement, ReactNode } from 'react';
|
|
import React, { useRef, useState } from 'react';
|
|
import type { ContentRect, MeasuredComponentProps } from 'react-measure';
|
|
import Measure from 'react-measure';
|
|
import classNames from 'classnames';
|
|
import { noop } from 'lodash';
|
|
import { animated } from '@react-spring/web';
|
|
|
|
import type { LocalizerType } from '../types/Util';
|
|
import { ModalHost } from './ModalHost';
|
|
import type { Theme } from '../util/theme';
|
|
import { getClassNamesFor } from '../util/getClassNamesFor';
|
|
import { useAnimated } from '../hooks/useAnimated';
|
|
import { useHasWrapped } from '../hooks/useHasWrapped';
|
|
import { useRefMerger } from '../hooks/useRefMerger';
|
|
|
|
type PropsType = {
|
|
children: ReactNode;
|
|
hasStickyButtons?: boolean;
|
|
hasXButton?: boolean;
|
|
i18n: LocalizerType;
|
|
moduleClassName?: string;
|
|
onBackButtonClick?: () => unknown;
|
|
onClose?: () => void;
|
|
title?: ReactNode;
|
|
useFocusTrap?: boolean;
|
|
};
|
|
|
|
type ModalPropsType = PropsType & {
|
|
noMouseClose?: boolean;
|
|
theme?: Theme;
|
|
};
|
|
|
|
const BASE_CLASS_NAME = 'module-Modal';
|
|
|
|
export function Modal({
|
|
children,
|
|
hasStickyButtons,
|
|
hasXButton,
|
|
i18n,
|
|
moduleClassName,
|
|
noMouseClose,
|
|
onBackButtonClick,
|
|
onClose = noop,
|
|
title,
|
|
theme,
|
|
useFocusTrap,
|
|
}: Readonly<ModalPropsType>): ReactElement {
|
|
const { close, modalStyles, overlayStyles } = useAnimated(onClose, {
|
|
getFrom: () => ({ opacity: 0, transform: 'translateY(48px)' }),
|
|
getTo: isOpen =>
|
|
isOpen
|
|
? { opacity: 1, transform: 'translateY(0px)' }
|
|
: { opacity: 0, transform: 'translateY(48px)' },
|
|
});
|
|
|
|
return (
|
|
<ModalHost
|
|
moduleClassName={moduleClassName}
|
|
noMouseClose={noMouseClose}
|
|
onClose={close}
|
|
overlayStyles={overlayStyles}
|
|
theme={theme}
|
|
useFocusTrap={useFocusTrap}
|
|
>
|
|
<animated.div style={modalStyles}>
|
|
<ModalWindow
|
|
hasStickyButtons={hasStickyButtons}
|
|
hasXButton={hasXButton}
|
|
i18n={i18n}
|
|
moduleClassName={moduleClassName}
|
|
onBackButtonClick={onBackButtonClick}
|
|
onClose={close}
|
|
title={title}
|
|
>
|
|
{children}
|
|
</ModalWindow>
|
|
</animated.div>
|
|
</ModalHost>
|
|
);
|
|
}
|
|
|
|
export function ModalWindow({
|
|
children,
|
|
hasStickyButtons,
|
|
hasXButton,
|
|
i18n,
|
|
moduleClassName,
|
|
onBackButtonClick,
|
|
onClose = noop,
|
|
title,
|
|
}: Readonly<PropsType>): JSX.Element {
|
|
const modalRef = useRef<HTMLDivElement | null>(null);
|
|
|
|
const refMerger = useRefMerger();
|
|
|
|
const bodyRef = useRef<HTMLDivElement | null>(null);
|
|
const [scrolled, setScrolled] = useState(false);
|
|
const [hasOverflow, setHasOverflow] = useState(false);
|
|
|
|
const hasHeader = Boolean(hasXButton || title || onBackButtonClick);
|
|
const getClassName = getClassNamesFor(BASE_CLASS_NAME, moduleClassName);
|
|
|
|
function handleResize({ scroll }: ContentRect) {
|
|
const modalNode = modalRef?.current;
|
|
if (!modalNode) {
|
|
return;
|
|
}
|
|
if (scroll) {
|
|
setHasOverflow(scroll.height > modalNode.clientHeight);
|
|
}
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{/* We don't want the click event to propagate to its container node. */}
|
|
{/* eslint-disable-next-line max-len */}
|
|
{/* eslint-disable-next-line jsx-a11y/no-static-element-interactions, jsx-a11y/click-events-have-key-events */}
|
|
<div
|
|
className={classNames(
|
|
getClassName(''),
|
|
getClassName(hasHeader ? '--has-header' : '--no-header'),
|
|
hasStickyButtons && getClassName('--sticky-buttons')
|
|
)}
|
|
ref={modalRef}
|
|
onClick={event => {
|
|
event.stopPropagation();
|
|
}}
|
|
>
|
|
{hasHeader && (
|
|
<div
|
|
className={classNames(
|
|
getClassName('__header'),
|
|
onBackButtonClick
|
|
? getClassName('__header--with-back-button')
|
|
: null
|
|
)}
|
|
>
|
|
{onBackButtonClick && (
|
|
<button
|
|
aria-label={i18n('back')}
|
|
className={getClassName('__back-button')}
|
|
onClick={onBackButtonClick}
|
|
tabIndex={0}
|
|
type="button"
|
|
/>
|
|
)}
|
|
{title && (
|
|
<h1
|
|
className={classNames(
|
|
getClassName('__title'),
|
|
hasXButton ? getClassName('__title--with-x-button') : null
|
|
)}
|
|
>
|
|
{title}
|
|
</h1>
|
|
)}
|
|
{hasXButton && !title && (
|
|
<div className={getClassName('__title')} />
|
|
)}
|
|
{hasXButton && (
|
|
<button
|
|
aria-label={i18n('close')}
|
|
className={getClassName('__close-button')}
|
|
onClick={onClose}
|
|
tabIndex={0}
|
|
type="button"
|
|
/>
|
|
)}
|
|
</div>
|
|
)}
|
|
<Measure scroll onResize={handleResize}>
|
|
{({ measureRef }: MeasuredComponentProps) => (
|
|
<div
|
|
className={classNames(
|
|
getClassName('__body'),
|
|
scrolled ? getClassName('__body--scrolled') : null,
|
|
hasOverflow || scrolled
|
|
? getClassName('__body--overflow')
|
|
: null
|
|
)}
|
|
onScroll={() => {
|
|
const scrollTop = bodyRef.current?.scrollTop || 0;
|
|
setScrolled(scrollTop > 2);
|
|
}}
|
|
ref={refMerger(measureRef, bodyRef)}
|
|
>
|
|
{children}
|
|
</div>
|
|
)}
|
|
</Measure>
|
|
</div>
|
|
</>
|
|
);
|
|
}
|
|
|
|
Modal.ButtonFooter = function ButtonFooter({
|
|
children,
|
|
moduleClassName,
|
|
}: Readonly<{
|
|
children: ReactNode;
|
|
moduleClassName?: string;
|
|
}>): ReactElement {
|
|
const [ref, hasWrapped] = useHasWrapped<HTMLDivElement>();
|
|
|
|
const className = getClassNamesFor(
|
|
BASE_CLASS_NAME,
|
|
moduleClassName
|
|
)('__button-footer');
|
|
|
|
return (
|
|
<div
|
|
className={classNames(
|
|
className,
|
|
hasWrapped ? `${className}--one-button-per-line` : undefined
|
|
)}
|
|
ref={ref}
|
|
>
|
|
{children}
|
|
</div>
|
|
);
|
|
};
|