signal-desktop/ts/components/Modal.tsx

117 lines
2.8 KiB
TypeScript
Raw Normal View History

// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import React, { useState, ReactElement, ReactNode } from 'react';
import classNames from 'classnames';
import { noop } from 'lodash';
import { LocalizerType } from '../types/Util';
import { ModalHost } from './ModalHost';
import { Theme } from '../util/theme';
2021-05-11 00:50:43 +00:00
import { getClassNamesFor } from '../util/getClassNamesFor';
import { useHasWrapped } from '../util/hooks';
type PropsType = {
children: ReactNode;
hasXButton?: boolean;
i18n: LocalizerType;
2021-04-21 16:31:12 +00:00
moduleClassName?: string;
2021-05-28 16:15:17 +00:00
noMouseClose?: boolean;
onClose?: () => void;
title?: ReactNode;
theme?: Theme;
};
2021-05-11 00:50:43 +00:00
const BASE_CLASS_NAME = 'module-Modal';
export function Modal({
children,
hasXButton,
i18n,
2021-04-21 16:31:12 +00:00
moduleClassName,
2021-05-28 16:15:17 +00:00
noMouseClose,
onClose = noop,
title,
theme,
}: Readonly<PropsType>): ReactElement {
const [scrolled, setScrolled] = useState(false);
const hasHeader = Boolean(hasXButton || title);
2021-05-11 00:50:43 +00:00
const getClassName = getClassNamesFor(BASE_CLASS_NAME, moduleClassName);
return (
2021-05-28 16:15:17 +00:00
<ModalHost noMouseClose={noMouseClose} onClose={onClose} theme={theme}>
<div
className={classNames(
2021-05-11 00:50:43 +00:00
getClassName(''),
getClassName(hasHeader ? '--has-header' : '--no-header')
)}
>
{hasHeader && (
2021-05-11 00:50:43 +00:00
<div className={getClassName('__header')}>
{hasXButton && (
<button
aria-label={i18n('close')}
type="button"
2021-05-11 00:50:43 +00:00
className={getClassName('__close-button')}
tabIndex={0}
onClick={() => {
onClose();
}}
/>
)}
{title && (
<h1
className={classNames(
2021-05-11 00:50:43 +00:00
getClassName('__title'),
hasXButton ? getClassName('__title--with-x-button') : null
)}
>
{title}
</h1>
)}
</div>
)}
<div
2021-05-11 00:50:43 +00:00
className={classNames(
getClassName('__body'),
scrolled ? getClassName('__body--scrolled') : null
)}
onScroll={event => {
setScrolled((event.target as HTMLDivElement).scrollTop > 2);
}}
>
{children}
</div>
</div>
</ModalHost>
);
}
Modal.ButtonFooter = function ButtonFooter({
children,
2021-05-11 00:50:43 +00:00
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>
);
};