2023-02-08 18:26:37 -08:00
|
|
|
import { Portal } from '@reach/portal';
|
2022-05-01 12:31:40 -07:00
|
|
|
import React, { useState, useEffect, useRef } from 'react';
|
2022-04-10 18:31:24 -07:00
|
|
|
|
2022-11-16 05:32:32 -08:00
|
|
|
import { simpleEmojiReact } from 'soapbox/actions/emoji-reacts';
|
2022-04-10 18:31:24 -07:00
|
|
|
import { openModal } from 'soapbox/actions/modals';
|
2022-11-25 10:28:43 -08:00
|
|
|
import { EmojiSelector } from 'soapbox/components/ui';
|
2023-01-09 14:13:12 -08:00
|
|
|
import { useAppDispatch, useAppSelector, useOwnAccount, useSoapboxConfig } from 'soapbox/hooks';
|
2022-11-15 12:48:54 -08:00
|
|
|
import { isUserTouching } from 'soapbox/is-mobile';
|
2022-11-15 12:46:23 -08:00
|
|
|
import { getReactForStatus } from 'soapbox/utils/emoji-reacts';
|
2022-04-10 18:31:24 -07:00
|
|
|
|
2023-02-08 09:58:01 -08:00
|
|
|
interface IStatusReactionWrapper {
|
2022-04-10 18:31:24 -07:00
|
|
|
statusId: string,
|
|
|
|
children: JSX.Element,
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Provides emoji reaction functionality to the underlying button component */
|
2023-02-08 09:58:01 -08:00
|
|
|
const StatusReactionWrapper: React.FC<IStatusReactionWrapper> = ({ statusId, children }): JSX.Element | null => {
|
2023-01-09 14:13:12 -08:00
|
|
|
const dispatch = useAppDispatch();
|
2022-04-10 18:31:24 -07:00
|
|
|
const ownAccount = useOwnAccount();
|
|
|
|
const status = useAppSelector(state => state.statuses.get(statusId));
|
|
|
|
const soapboxConfig = useSoapboxConfig();
|
|
|
|
|
2022-05-01 12:31:40 -07:00
|
|
|
const timeout = useRef<NodeJS.Timeout>();
|
2022-04-10 18:31:24 -07:00
|
|
|
const [visible, setVisible] = useState(false);
|
|
|
|
|
2022-04-20 14:35:09 -07:00
|
|
|
const [referenceElement, setReferenceElement] = useState<HTMLDivElement | null>(null);
|
2022-04-10 18:31:24 -07:00
|
|
|
|
2022-05-01 12:31:40 -07:00
|
|
|
useEffect(() => {
|
|
|
|
return () => {
|
|
|
|
if (timeout.current) {
|
|
|
|
clearTimeout(timeout.current);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}, []);
|
|
|
|
|
2022-04-10 18:31:24 -07:00
|
|
|
if (!status) return null;
|
|
|
|
|
|
|
|
const handleMouseEnter = () => {
|
2022-05-01 12:31:40 -07:00
|
|
|
if (timeout.current) {
|
|
|
|
clearTimeout(timeout.current);
|
|
|
|
}
|
|
|
|
|
2022-04-13 15:37:38 -07:00
|
|
|
if (!isUserTouching()) {
|
|
|
|
setVisible(true);
|
|
|
|
}
|
2022-04-10 18:31:24 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
const handleMouseLeave = () => {
|
2022-05-01 12:31:40 -07:00
|
|
|
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);
|
|
|
|
}
|
2022-04-10 18:31:24 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
const handleReact = (emoji: string): void => {
|
|
|
|
if (ownAccount) {
|
|
|
|
dispatch(simpleEmojiReact(status, emoji));
|
|
|
|
} else {
|
2022-10-24 15:01:53 -07:00
|
|
|
handleUnauthorized();
|
2022-04-10 18:31:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
setVisible(false);
|
|
|
|
};
|
|
|
|
|
2022-04-10 18:41:00 -07:00
|
|
|
const handleClick: React.EventHandler<React.MouseEvent> = e => {
|
|
|
|
const meEmojiReact = getReactForStatus(status, soapboxConfig.allowedEmoji) || '👍';
|
|
|
|
|
|
|
|
if (isUserTouching()) {
|
2022-10-24 15:01:53 -07:00
|
|
|
if (ownAccount) {
|
|
|
|
if (visible) {
|
|
|
|
handleReact(meEmojiReact);
|
|
|
|
} else {
|
|
|
|
setVisible(true);
|
|
|
|
}
|
2022-04-10 18:41:00 -07:00
|
|
|
} else {
|
2022-10-24 15:01:53 -07:00
|
|
|
handleUnauthorized();
|
2022-04-10 18:41:00 -07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
handleReact(meEmojiReact);
|
|
|
|
}
|
|
|
|
|
2022-04-13 15:37:38 -07:00
|
|
|
e.preventDefault();
|
2022-04-10 18:41:00 -07:00
|
|
|
e.stopPropagation();
|
|
|
|
};
|
|
|
|
|
2022-10-24 15:01:53 -07:00
|
|
|
const handleUnauthorized = () => {
|
|
|
|
dispatch(openModal('UNAUTHORIZED', {
|
|
|
|
action: 'FAVOURITE',
|
|
|
|
ap_id: status.url,
|
|
|
|
}));
|
|
|
|
};
|
|
|
|
|
2022-04-10 18:31:24 -07:00
|
|
|
return (
|
2022-05-01 12:10:35 -07:00
|
|
|
<div className='relative' onMouseEnter={handleMouseEnter} onMouseLeave={handleMouseLeave}>
|
2022-04-10 18:31:24 -07:00
|
|
|
{React.cloneElement(children, {
|
2022-04-10 18:41:00 -07:00
|
|
|
onClick: handleClick,
|
2022-04-20 14:35:09 -07:00
|
|
|
ref: setReferenceElement,
|
2022-04-10 18:31:24 -07:00
|
|
|
})}
|
|
|
|
|
2023-02-08 18:26:37 -08:00
|
|
|
<Portal>
|
|
|
|
<EmojiSelector
|
|
|
|
placement='top-start'
|
|
|
|
referenceElement={referenceElement}
|
|
|
|
onReact={handleReact}
|
|
|
|
visible={visible}
|
|
|
|
/>
|
|
|
|
</Portal>
|
2022-04-10 18:31:24 -07:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2023-02-08 09:58:01 -08:00
|
|
|
export default StatusReactionWrapper;
|