// Copyright 2021-2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { KeyboardEvent, MouseEvent, ReactNode } from 'react'; import React, { memo, useEffect } from 'react'; import classNames from 'classnames'; import { createPortal } from 'react-dom'; import { useRestoreFocus } from '../hooks/useRestoreFocus'; import { clearTimeoutIfNecessary } from '../util/clearTimeoutIfNecessary'; export type PropsType = { autoDismissDisabled?: boolean; children: ReactNode; align?: 'left' | 'center'; className?: string; disableCloseOnClick?: boolean; onClose: () => unknown; timeout?: number; toastAction?: { label: string; onClick: () => unknown; }; style?: React.CSSProperties; }; export const Toast = memo( ({ autoDismissDisabled = false, children, align = 'center', className, disableCloseOnClick = false, onClose, style, timeout = 8000, toastAction, }: PropsType): JSX.Element | null => { const [root, setRoot] = React.useState(null); const [focusRef] = useRestoreFocus(); useEffect(() => { const div = document.createElement('div'); document.body.appendChild(div); setRoot(div); return () => { document.body.removeChild(div); setRoot(null); }; }, []); useEffect(() => { if (!root || autoDismissDisabled) { return; } const timeoutId = setTimeout(onClose, timeout); return () => { clearTimeoutIfNecessary(timeoutId); }; }, [autoDismissDisabled, onClose, root, timeout]); return root ? createPortal(
{ if (!disableCloseOnClick) { onClose(); } }} onKeyDown={(ev: KeyboardEvent) => { if (ev.key === 'Enter' || ev.key === ' ') { if (!disableCloseOnClick) { onClose(); } } }} role="button" tabIndex={0} style={style} >
{children}
{toastAction && (
) => { ev.stopPropagation(); ev.preventDefault(); toastAction.onClick(); onClose(); }} onKeyDown={(ev: KeyboardEvent) => { if (ev.key === 'Enter' || ev.key === ' ') { ev.stopPropagation(); ev.preventDefault(); toastAction.onClick(); onClose(); } }} ref={focusRef} role="button" tabIndex={0} > {toastAction.label}
)}
, root ) : null; } );