signal-desktop/ts/components/emoji/EmojiButton.tsx

164 lines
4.8 KiB
TypeScript
Raw Normal View History

2022-03-04 21:14:52 +00:00
// Copyright 2019-2022 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';
import classNames from 'classnames';
import { get, noop } from 'lodash';
2019-05-24 23:58:27 +00:00
import { Manager, Popper, Reference } from 'react-popper';
import { createPortal } from 'react-dom';
2021-07-19 19:26:06 +00:00
import { Emoji } from './Emoji';
import type { Props as EmojiPickerProps } from './EmojiPicker';
import { EmojiPicker } from './EmojiPicker';
import type { LocalizerType } from '../../types/Util';
import * as KeyboardLayout from '../../services/keyboardLayout';
2019-05-24 23:58:27 +00:00
export type OwnProps = {
2022-03-04 21:14:52 +00:00
readonly className?: string;
2021-07-19 19:26:06 +00:00
readonly closeOnPick?: boolean;
readonly emoji?: string;
2019-05-24 23:58:27 +00:00
readonly i18n: LocalizerType;
readonly onClose?: () => unknown;
2019-05-24 23:58:27 +00:00
};
export type Props = OwnProps &
Pick<
EmojiPickerProps,
2019-11-07 21:36:16 +00:00
'doSend' | 'onPickEmoji' | 'onSetSkinTone' | 'recentEmojis' | 'skinTone'
2019-05-24 23:58:27 +00:00
>;
export const EmojiButton = React.memo(
({
2022-03-04 21:14:52 +00:00
className,
2021-07-19 19:26:06 +00:00
closeOnPick,
emoji,
i18n,
doSend,
onClose,
onPickEmoji,
skinTone,
onSetSkinTone,
recentEmojis,
}: Props) => {
2019-05-24 23:58:27 +00:00
const [open, setOpen] = React.useState(false);
const [popperRoot, setPopperRoot] = React.useState<HTMLElement | null>(
null
);
2020-01-08 17:44:54 +00:00
const handleClickButton = React.useCallback(() => {
if (popperRoot) {
2019-05-24 23:58:27 +00:00
setOpen(false);
2020-01-08 17:44:54 +00:00
} else {
setOpen(true);
}
}, [popperRoot, setOpen]);
const handleClose = React.useCallback(() => {
setOpen(false);
if (onClose) {
onClose();
}
}, [setOpen, onClose]);
2019-05-24 23:58:27 +00:00
// Create popper root and handle outside clicks
2020-01-08 17:44:54 +00:00
React.useEffect(() => {
if (open) {
const root = document.createElement('div');
setPopperRoot(root);
document.body.appendChild(root);
const handleOutsideClick = (event: MouseEvent) => {
if (!root.contains(event.target as Node)) {
handleClose();
event.stopPropagation();
event.preventDefault();
2020-01-08 17:44:54 +00:00
}
};
document.addEventListener('click', handleOutsideClick);
2019-05-24 23:58:27 +00:00
2020-01-08 17:44:54 +00:00
return () => {
document.body.removeChild(root);
document.removeEventListener('click', handleOutsideClick);
setPopperRoot(null);
};
}
2019-05-24 23:58:27 +00:00
2020-01-08 17:44:54 +00:00
return noop;
}, [open, setOpen, setPopperRoot, handleClose]);
2019-05-24 23:58:27 +00:00
2019-11-07 21:36:16 +00:00
// Install keyboard shortcut to open emoji picker
2020-01-08 17:44:54 +00:00
React.useEffect(() => {
const handleKeydown = (event: KeyboardEvent) => {
const { ctrlKey, metaKey, shiftKey } = event;
2020-01-08 17:44:54 +00:00
const commandKey = get(window, 'platform') === 'darwin' && metaKey;
const controlKey = get(window, 'platform') !== 'darwin' && ctrlKey;
const commandOrCtrl = commandKey || controlKey;
const key = KeyboardLayout.lookup(event);
2019-11-07 21:36:16 +00:00
2020-01-08 17:44:54 +00:00
// We don't want to open up if the conversation has any panels open
const panels = document.querySelectorAll('.conversation .panel');
if (panels && panels.length > 1) {
return;
}
2020-01-08 17:44:54 +00:00
if (commandOrCtrl && shiftKey && (key === 'j' || key === 'J')) {
event.stopPropagation();
event.preventDefault();
2019-11-07 21:36:16 +00:00
2020-01-08 17:44:54 +00:00
setOpen(!open);
}
};
document.addEventListener('keydown', handleKeydown);
2019-11-07 21:36:16 +00:00
2020-01-08 17:44:54 +00:00
return () => {
document.removeEventListener('keydown', handleKeydown);
};
}, [open, setOpen]);
2019-11-07 21:36:16 +00:00
2019-05-24 23:58:27 +00:00
return (
<Manager>
<Reference>
{({ ref }) => (
<button
type="button"
2019-05-24 23:58:27 +00:00
ref={ref}
onClick={handleClickButton}
2022-03-04 21:14:52 +00:00
className={classNames(className, {
2019-05-24 23:58:27 +00:00
'module-emoji-button__button': true,
'module-emoji-button__button--active': open,
2021-07-19 19:26:06 +00:00
'module-emoji-button__button--has-emoji': Boolean(emoji),
2019-05-24 23:58:27 +00:00
})}
aria-label={i18n('EmojiButton__label')}
2021-07-19 19:26:06 +00:00
>
{emoji && <Emoji emoji={emoji} size={24} />}
</button>
2019-05-24 23:58:27 +00:00
)}
</Reference>
{open && popperRoot
? createPortal(
<Popper placement="top-start" strategy="fixed">
2019-05-24 23:58:27 +00:00
{({ ref, style }) => (
<EmojiPicker
ref={ref}
i18n={i18n}
style={style}
2021-07-19 19:26:06 +00:00
onPickEmoji={ev => {
onPickEmoji(ev);
if (closeOnPick) {
handleClose();
}
}}
doSend={doSend}
2019-05-24 23:58:27 +00:00
onClose={handleClose}
skinTone={skinTone}
onSetSkinTone={onSetSkinTone}
recentEmojis={recentEmojis}
/>
)}
</Popper>,
popperRoot
)
: null}
</Manager>
);
}
);