import classNames from 'clsx'; import React, { useCallback, useEffect, useRef, useState } from 'react'; import 'wicg-inert'; import { FormattedMessage, defineMessages, useIntl } from 'react-intl'; import { useHistory } from 'react-router-dom'; import { cancelReplyCompose } from 'soapbox/actions/compose'; import { cancelEventCompose } from 'soapbox/actions/events'; import { openModal, closeModal } from 'soapbox/actions/modals'; import { useAppDispatch, usePrevious } from 'soapbox/hooks'; import { queryClient } from 'soapbox/queries/client'; import { IPolicy, PolicyKeys } from 'soapbox/queries/policies'; import type { UnregisterCallback } from 'history'; import type { ModalType } from 'soapbox/features/ui/components/modal-root'; import type { ReducerCompose } from 'soapbox/reducers/compose'; import type { ReducerRecord as ReducerComposeEvent } from 'soapbox/reducers/compose-event'; const messages = defineMessages({ confirm: { id: 'confirmations.delete.confirm', defaultMessage: 'Delete' }, cancelEditing: { id: 'confirmations.cancel_editing.confirm', defaultMessage: 'Cancel editing' }, }); export const checkComposeContent = (compose?: ReturnType) => { return !!compose && [ compose.text.length > 0, compose.spoiler_text.length > 0, compose.media_attachments.size > 0, compose.poll !== null, ].some(check => check === true); }; export const checkEventComposeContent = (compose?: ReturnType) => { return !!compose && [ compose.name.length > 0, compose.status.length > 0, compose.location !== null, compose.banner !== null, ].some(check => check === true); }; interface IModalRoot { onCancel?: () => void, onClose: (type?: ModalType) => void, type: ModalType, } const ModalRoot: React.FC = ({ children, onCancel, onClose, type }) => { const intl = useIntl(); const history = useHistory(); const dispatch = useAppDispatch(); const [revealed, setRevealed] = useState(!!children); const ref = useRef(null); const activeElement = useRef(revealed ? document.activeElement as HTMLDivElement | null : null); const modalHistoryKey = useRef(); const unlistenHistory = useRef(); const prevChildren = usePrevious(children); const prevType = usePrevious(type); const visible = !!children; const handleKeyUp = (e: KeyboardEvent) => { if (e.key === 'Escape' || e.key === 'Esc' || e.keyCode === 27) { handleOnClose(); } }; const handleOnClose = () => { dispatch((_, getState) => { const compose = getState().compose.get('compose-modal'); const hasComposeContent = checkComposeContent(compose); const hasEventComposeContent = checkEventComposeContent(getState().compose_event); if (hasComposeContent && type === 'COMPOSE') { const isEditing = compose!.id !== null; dispatch(openModal('CONFIRM', { icon: require('@tabler/icons/trash.svg'), heading: isEditing ? : , message: isEditing ? : , confirm: intl.formatMessage(messages.confirm), onConfirm: () => { dispatch(closeModal('COMPOSE')); dispatch(cancelReplyCompose()); }, onCancel: () => { dispatch(closeModal('CONFIRM')); }, })); } else if (hasEventComposeContent && type === 'COMPOSE_EVENT') { const isEditing = getState().compose_event.id !== null; dispatch(openModal('CONFIRM', { icon: require('@tabler/icons/trash.svg'), heading: isEditing ? : , message: isEditing ? : , confirm: intl.formatMessage(isEditing ? messages.cancelEditing : messages.confirm), onConfirm: () => { dispatch(closeModal('COMPOSE_EVENT')); dispatch(cancelEventCompose()); }, onCancel: () => { dispatch(closeModal('CONFIRM')); }, })); } else if ((hasComposeContent || hasEventComposeContent) && type === 'CONFIRM') { dispatch(closeModal('CONFIRM')); } else if (type === 'POLICY') { // If the user has not accepted the Policy, prevent them // from closing the Modal. const pendingPolicy = queryClient.getQueryData(PolicyKeys.policy) as IPolicy; if (pendingPolicy?.pending_policy_id) { return; } onClose(); } else { onClose(); } }); }; const handleKeyDown = useCallback((e) => { if (e.key === 'Tab') { const focusable = Array.from(ref.current!.querySelectorAll('button:not([disabled]), [href], input:not([disabled]), select:not([disabled]), textarea:not([disabled]), [tabindex]:not([tabindex="-1"])')).filter((x) => window.getComputedStyle(x).display !== 'none'); const index = focusable.indexOf(e.target); let element; if (e.shiftKey) { element = focusable[index - 1] || focusable[focusable.length - 1]; } else { element = focusable[index + 1] || focusable[0]; } if (element) { (element as HTMLDivElement).focus(); e.stopPropagation(); e.preventDefault(); } } }, []); const handleModalOpen = () => { modalHistoryKey.current = Date.now(); unlistenHistory.current = history.listen((_, action) => { if (action === 'POP') { handleOnClose(); if (onCancel) onCancel(); } }); }; const handleModalClose = (type: string) => { if (unlistenHistory.current) { unlistenHistory.current(); } if (!['FAVOURITES', 'MENTIONS', 'REACTIONS', 'REBLOGS', 'MEDIA'].includes(type)) { const { state } = history.location; if (state && (state as any).soapboxModalKey === modalHistoryKey.current) { history.goBack(); } } }; const ensureHistoryBuffer = () => { const { pathname, state } = history.location; if (!state || (state as any).soapboxModalKey !== modalHistoryKey.current) { history.push(pathname, { ...(state as any), soapboxModalKey: modalHistoryKey.current }); } }; const getSiblings = () => { return Array(...(ref.current!.parentElement!.childNodes as any as ChildNode[])).filter(node => node !== ref.current); }; useEffect(() => { if (!visible) return; window.addEventListener('keyup', handleKeyUp, false); window.addEventListener('keydown', handleKeyDown, false); return () => { window.removeEventListener('keyup', handleKeyUp); window.removeEventListener('keydown', handleKeyDown); }; }, [visible]); useEffect(() => { if (!!children && !prevChildren) { activeElement.current = document.activeElement as HTMLDivElement; getSiblings().forEach(sibling => (sibling as HTMLDivElement).setAttribute('inert', 'true')); handleModalOpen(); } else if (!prevChildren) { setRevealed(false); } if (!children && !!prevChildren) { activeElement.current?.focus(); activeElement.current = null; getSiblings().forEach(sibling => (sibling as HTMLDivElement).removeAttribute('inert')); handleModalClose(prevType!); } if (children) { requestAnimationFrame(() => { setRevealed(true); }); ensureHistoryBuffer(); } }); if (!visible) { return (
); } return (
); }; export default ModalRoot;