import React, { useState, useEffect, useRef } from 'react'; import { simpleEmojiReact } from 'soapbox/actions/emoji-reacts'; import { openModal } from 'soapbox/actions/modals'; import { EmojiSelector, Portal } from 'soapbox/components/ui'; import { useAppDispatch, useAppSelector, useOwnAccount, useSoapboxConfig } from 'soapbox/hooks'; import { isUserTouching } from 'soapbox/is-mobile'; import { getReactForStatus } from 'soapbox/utils/emoji-reacts'; interface IStatusReactionWrapper { statusId: string, children: JSX.Element, } /** Provides emoji reaction functionality to the underlying button component */ const StatusReactionWrapper: React.FC = ({ statusId, children }): JSX.Element | null => { const dispatch = useAppDispatch(); const ownAccount = useOwnAccount(); const status = useAppSelector(state => state.statuses.get(statusId)); const soapboxConfig = useSoapboxConfig(); const timeout = useRef(); const [visible, setVisible] = useState(false); const [referenceElement, setReferenceElement] = useState(null); useEffect(() => { return () => { if (timeout.current) { clearTimeout(timeout.current); } }; }, []); if (!status) return null; const handleMouseEnter = () => { if (timeout.current) { clearTimeout(timeout.current); } if (!isUserTouching()) { setVisible(true); } }; const handleMouseLeave = () => { if (timeout.current) { clearTimeout(timeout.current); } // Unless the user is touching, delay closing the emoji selector briefly // so the user can move the mouse diagonally to make a selection. if (isUserTouching()) { setVisible(false); } else { timeout.current = setTimeout(() => { setVisible(false); }, 500); } }; const handleReact = (emoji: string): void => { if (ownAccount) { dispatch(simpleEmojiReact(status, emoji)); } else { handleUnauthorized(); } setVisible(false); }; const handleClick: React.EventHandler = e => { const meEmojiReact = getReactForStatus(status, soapboxConfig.allowedEmoji) || '👍'; if (isUserTouching()) { if (ownAccount) { if (visible) { handleReact(meEmojiReact); } else { setVisible(true); } } else { handleUnauthorized(); } } else { handleReact(meEmojiReact); } e.preventDefault(); e.stopPropagation(); }; const handleUnauthorized = () => { dispatch(openModal('UNAUTHORIZED', { action: 'FAVOURITE', ap_id: status.url, })); }; return (
{React.cloneElement(children, { onClick: handleClick, ref: setReferenceElement, })} {visible && ( )}
); }; export default StatusReactionWrapper;