2023-01-03 19:55:46 +00:00
|
|
|
// Copyright 2019 Signal Messenger, LLC
|
2020-10-30 20:34:04 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2019-05-24 23:58:27 +00:00
|
|
|
import * as React from 'react';
|
2022-07-07 21:39:22 +00:00
|
|
|
import type { MutableRefObject } from 'react';
|
2019-05-24 23:58:27 +00:00
|
|
|
import classNames from 'classnames';
|
2019-12-17 18:52:36 +00:00
|
|
|
import { get, noop } from 'lodash';
|
2019-05-24 23:58:27 +00:00
|
|
|
import { Manager, Popper, Reference } from 'react-popper';
|
2021-07-19 19:26:06 +00:00
|
|
|
import { Emoji } from './Emoji';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { Props as EmojiPickerProps } from './EmojiPicker';
|
|
|
|
import { EmojiPicker } from './EmojiPicker';
|
|
|
|
import type { LocalizerType } from '../../types/Util';
|
2022-06-15 17:53:08 +00:00
|
|
|
import { useRefMerger } from '../../hooks/useRefMerger';
|
2022-09-15 01:58:35 +00:00
|
|
|
import { handleOutsideClick } from '../../util/handleOutsideClick';
|
2022-03-31 05:58:28 +00:00
|
|
|
import * as KeyboardLayout from '../../services/keyboardLayout';
|
2019-05-24 23:58:27 +00:00
|
|
|
|
2022-10-18 17:12:02 +00:00
|
|
|
export enum EmojiButtonVariant {
|
|
|
|
Normal,
|
|
|
|
ProfileEditor,
|
|
|
|
}
|
|
|
|
|
2022-07-07 21:39:22 +00:00
|
|
|
export type OwnProps = Readonly<{
|
|
|
|
className?: string;
|
|
|
|
closeOnPick?: boolean;
|
|
|
|
emoji?: string;
|
|
|
|
i18n: LocalizerType;
|
|
|
|
onClose?: () => unknown;
|
2023-09-14 17:04:48 +00:00
|
|
|
onOpen?: () => unknown;
|
2022-07-07 21:39:22 +00:00
|
|
|
emojiButtonApi?: MutableRefObject<EmojiButtonAPI | undefined>;
|
2022-10-18 17:12:02 +00:00
|
|
|
variant?: EmojiButtonVariant;
|
2022-07-07 21:39:22 +00:00
|
|
|
}>;
|
2019-05-24 23:58:27 +00:00
|
|
|
|
|
|
|
export type Props = OwnProps &
|
|
|
|
Pick<
|
|
|
|
EmojiPickerProps,
|
2023-11-21 19:15:39 +00:00
|
|
|
'onPickEmoji' | 'onSetSkinTone' | 'recentEmojis' | 'skinTone'
|
2019-05-24 23:58:27 +00:00
|
|
|
>;
|
|
|
|
|
2022-07-07 21:39:22 +00:00
|
|
|
export type EmojiButtonAPI = Readonly<{
|
|
|
|
close: () => void;
|
|
|
|
}>;
|
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
export const EmojiButton = React.memo(function EmojiButtonInner({
|
|
|
|
className,
|
|
|
|
closeOnPick,
|
|
|
|
emoji,
|
|
|
|
emojiButtonApi,
|
|
|
|
i18n,
|
|
|
|
onClose,
|
2023-09-14 17:04:48 +00:00
|
|
|
onOpen,
|
2022-11-18 00:45:19 +00:00
|
|
|
onPickEmoji,
|
|
|
|
skinTone,
|
|
|
|
onSetSkinTone,
|
|
|
|
recentEmojis,
|
|
|
|
variant = EmojiButtonVariant.Normal,
|
|
|
|
}: Props) {
|
2023-11-20 20:46:49 +00:00
|
|
|
const isRTL = i18n.getLocaleDirection() === 'rtl';
|
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
const [open, setOpen] = React.useState(false);
|
2024-03-19 13:23:31 +00:00
|
|
|
const [wasInvokedFromKeyboard, setWasInvokedFromKeyboard] =
|
|
|
|
React.useState(false);
|
2022-11-18 00:45:19 +00:00
|
|
|
const buttonRef = React.useRef<HTMLButtonElement | null>(null);
|
2023-01-19 00:09:18 +00:00
|
|
|
const popperRef = React.useRef<HTMLDivElement | null>(null);
|
2022-11-18 00:45:19 +00:00
|
|
|
const refMerger = useRefMerger();
|
|
|
|
|
2023-09-14 17:04:48 +00:00
|
|
|
React.useEffect(() => {
|
|
|
|
if (!open) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
onOpen?.();
|
|
|
|
}, [open, onOpen]);
|
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
const handleClickButton = React.useCallback(() => {
|
2024-03-19 13:23:31 +00:00
|
|
|
setWasInvokedFromKeyboard(false);
|
2023-01-19 00:09:18 +00:00
|
|
|
if (open) {
|
2020-01-08 17:44:54 +00:00
|
|
|
setOpen(false);
|
2022-11-18 00:45:19 +00:00
|
|
|
} else {
|
|
|
|
setOpen(true);
|
|
|
|
}
|
2024-03-19 13:23:31 +00:00
|
|
|
}, [open, setOpen, setWasInvokedFromKeyboard]);
|
2019-05-24 23:58:27 +00:00
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
const handleClose = React.useCallback(() => {
|
|
|
|
setOpen(false);
|
2024-03-19 13:23:31 +00:00
|
|
|
setWasInvokedFromKeyboard(false);
|
2022-11-18 00:45:19 +00:00
|
|
|
if (onClose) {
|
|
|
|
onClose();
|
|
|
|
}
|
2024-03-19 13:23:31 +00:00
|
|
|
}, [setOpen, setWasInvokedFromKeyboard, onClose]);
|
2022-11-18 00:45:19 +00:00
|
|
|
|
|
|
|
const api = React.useMemo(
|
|
|
|
() => ({
|
2024-03-19 13:23:31 +00:00
|
|
|
close: () => {
|
|
|
|
setOpen(false);
|
|
|
|
setWasInvokedFromKeyboard(false);
|
|
|
|
},
|
2022-11-18 00:45:19 +00:00
|
|
|
}),
|
2024-03-19 13:23:31 +00:00
|
|
|
[setOpen, setWasInvokedFromKeyboard]
|
2022-11-18 00:45:19 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
if (emojiButtonApi) {
|
|
|
|
// Using a React.MutableRefObject, so we need to reassign this prop.
|
|
|
|
// eslint-disable-next-line no-param-reassign
|
|
|
|
emojiButtonApi.current = api;
|
|
|
|
}
|
2022-07-07 21:39:22 +00:00
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
React.useEffect(() => {
|
|
|
|
if (!open) {
|
2020-01-08 17:44:54 +00:00
|
|
|
return noop;
|
2022-11-18 00:45:19 +00:00
|
|
|
}
|
2019-05-24 23:58:27 +00:00
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
return handleOutsideClick(
|
|
|
|
() => {
|
|
|
|
handleClose();
|
|
|
|
return true;
|
|
|
|
},
|
2023-01-19 00:09:18 +00:00
|
|
|
{
|
|
|
|
containerElements: [popperRef, buttonRef],
|
|
|
|
name: 'EmojiButton',
|
|
|
|
}
|
2022-11-18 00:45:19 +00:00
|
|
|
);
|
2023-01-19 00:09:18 +00:00
|
|
|
}, [open, handleClose]);
|
2022-11-18 00:45:19 +00:00
|
|
|
|
|
|
|
// Install keyboard shortcut to open emoji picker
|
|
|
|
React.useEffect(() => {
|
|
|
|
const handleKeydown = (event: KeyboardEvent) => {
|
|
|
|
const { ctrlKey, metaKey, shiftKey } = event;
|
|
|
|
const commandKey = get(window, 'platform') === 'darwin' && metaKey;
|
|
|
|
const controlKey = get(window, 'platform') !== 'darwin' && ctrlKey;
|
|
|
|
const commandOrCtrl = commandKey || controlKey;
|
|
|
|
const key = KeyboardLayout.lookup(event);
|
|
|
|
|
2024-01-02 19:56:21 +00:00
|
|
|
// We don't want to open up if the current conversation panel is hidden
|
|
|
|
const parentPanel = buttonRef.current?.closest('.ConversationPanel');
|
|
|
|
if (parentPanel?.classList.contains('ConversationPanel__hidden')) {
|
2022-11-18 00:45:19 +00:00
|
|
|
return;
|
2022-09-15 01:58:35 +00:00
|
|
|
}
|
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
if (commandOrCtrl && shiftKey && (key === 'j' || key === 'J')) {
|
|
|
|
event.stopPropagation();
|
|
|
|
event.preventDefault();
|
2019-11-07 21:36:16 +00:00
|
|
|
|
2024-03-19 13:23:31 +00:00
|
|
|
setWasInvokedFromKeyboard(true);
|
2022-11-18 00:45:19 +00:00
|
|
|
setOpen(!open);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
document.addEventListener('keydown', handleKeydown);
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
document.removeEventListener('keydown', handleKeydown);
|
|
|
|
};
|
|
|
|
}, [open, setOpen]);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Manager>
|
|
|
|
<Reference>
|
|
|
|
{({ ref }) => (
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
ref={refMerger(buttonRef, ref)}
|
|
|
|
onClick={handleClickButton}
|
|
|
|
className={classNames(className, {
|
|
|
|
'module-emoji-button__button': true,
|
|
|
|
'module-emoji-button__button--active': open,
|
|
|
|
'module-emoji-button__button--has-emoji': Boolean(emoji),
|
|
|
|
'module-emoji-button__button--profile-editor':
|
|
|
|
variant === EmojiButtonVariant.ProfileEditor,
|
|
|
|
})}
|
2023-03-30 00:03:25 +00:00
|
|
|
aria-label={i18n('icu:EmojiButton__label')}
|
2022-11-18 00:45:19 +00:00
|
|
|
>
|
|
|
|
{emoji && <Emoji emoji={emoji} size={24} />}
|
|
|
|
</button>
|
|
|
|
)}
|
|
|
|
</Reference>
|
2023-01-19 00:09:18 +00:00
|
|
|
{open ? (
|
|
|
|
<div ref={popperRef}>
|
2023-11-20 20:46:49 +00:00
|
|
|
<Popper placement={isRTL ? 'top-end' : 'top-start'} strategy="fixed">
|
2023-01-19 00:09:18 +00:00
|
|
|
{({ ref, style }) => (
|
|
|
|
<EmojiPicker
|
|
|
|
ref={ref}
|
|
|
|
i18n={i18n}
|
|
|
|
style={style}
|
|
|
|
onPickEmoji={ev => {
|
|
|
|
onPickEmoji(ev);
|
|
|
|
if (closeOnPick) {
|
|
|
|
handleClose();
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
onClose={handleClose}
|
|
|
|
skinTone={skinTone}
|
|
|
|
onSetSkinTone={onSetSkinTone}
|
2024-03-19 13:23:31 +00:00
|
|
|
wasInvokedFromKeyboard={wasInvokedFromKeyboard}
|
2023-01-19 00:09:18 +00:00
|
|
|
recentEmojis={recentEmojis}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</Popper>
|
|
|
|
</div>
|
|
|
|
) : null}
|
2022-11-18 00:45:19 +00:00
|
|
|
</Manager>
|
|
|
|
);
|
|
|
|
});
|