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

583 lines
19 KiB
TypeScript
Raw Normal View History

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';
import classNames from 'classnames';
import type {
2019-05-24 23:58:27 +00:00
GridCellRenderer,
SectionRenderedParams,
} from 'react-virtualized';
import { AutoSizer, Grid } from 'react-virtualized';
2019-05-24 23:58:27 +00:00
import {
chunk,
2022-02-07 23:00:04 +00:00
clamp,
2019-05-24 23:58:27 +00:00
debounce,
findLast,
flatMap,
initial,
last,
zipObject,
} from 'lodash';
import FocusTrap from 'focus-trap-react';
2019-05-24 23:58:27 +00:00
import { Emoji } from './Emoji';
2024-03-21 16:35:54 +00:00
import { dataByCategory } from './lib';
import type { LocalizerType } from '../../types/Util';
import { isSingleGrapheme } from '../../util/grapheme';
2023-04-03 20:16:27 +00:00
import { missingCaseError } from '../../util/missingCaseError';
2024-03-21 16:35:54 +00:00
import { useEmojiSearch } from '../../hooks/useEmojiSearch';
2019-05-24 23:58:27 +00:00
export type EmojiPickDataType = {
skinTone?: number;
shortName: string;
};
2019-05-24 23:58:27 +00:00
export type OwnProps = {
readonly i18n: LocalizerType;
readonly recentEmojis?: ReadonlyArray<string>;
readonly skinTone?: number;
readonly onClickSettings?: () => unknown;
readonly onClose?: () => unknown;
readonly onPickEmoji: (o: EmojiPickDataType) => unknown;
readonly onSetSkinTone?: (tone: number) => unknown;
readonly wasInvokedFromKeyboard: boolean;
2019-05-24 23:58:27 +00:00
};
export type Props = OwnProps & Pick<React.HTMLProps<HTMLDivElement>, 'style'>;
function isEventFromMouse(
event:
| React.MouseEvent<HTMLButtonElement>
| React.KeyboardEvent<HTMLButtonElement>
): boolean {
return (
('clientX' in event && event.clientX !== 0) ||
('clientY' in event && event.clientY !== 0)
);
}
2019-11-07 21:36:16 +00:00
function focusOnRender(el: HTMLElement | null) {
2019-05-24 23:58:27 +00:00
if (el) {
el.focus();
}
}
const COL_COUNT = 8;
const categories = [
'recents',
'emoji',
'animal',
'food',
'activity',
'travel',
'object',
'symbol',
'flag',
] as const;
2024-07-24 00:31:40 +00:00
type Category = (typeof categories)[number];
2019-05-24 23:58:27 +00:00
export const EmojiPicker = React.memo(
React.forwardRef<HTMLDivElement, Props>(
(
{
i18n,
onPickEmoji,
skinTone = 0,
onSetSkinTone,
2019-12-17 20:25:57 +00:00
recentEmojis = [],
2019-05-24 23:58:27 +00:00
style,
onClickSettings,
2019-05-24 23:58:27 +00:00
onClose,
wasInvokedFromKeyboard,
2019-05-24 23:58:27 +00:00
}: Props,
ref
) => {
const isRTL = i18n.getLocaleDirection() === 'rtl';
const [isUsingKeyboard, setIsUsingKeyboard] = React.useState(
wasInvokedFromKeyboard
);
const [firstRecent] = React.useState(recentEmojis);
const [selectedCategory, setSelectedCategory] = React.useState<Category>(
2019-05-24 23:58:27 +00:00
categories[0]
);
const [searchMode, setSearchMode] = React.useState(false);
const [searchText, setSearchText] = React.useState('');
const [scrollToRow, setScrollToRow] = React.useState(0);
2020-10-02 20:05:09 +00:00
const [selectedTone, setSelectedTone] = React.useState(skinTone);
2019-05-24 23:58:27 +00:00
2024-03-21 16:35:54 +00:00
const search = useEmojiSearch(i18n.getLocale());
const handleToggleSearch = React.useCallback(
(
e:
| React.MouseEvent<HTMLButtonElement>
| React.KeyboardEvent<HTMLButtonElement>
) => {
if (isEventFromMouse(e)) {
setIsUsingKeyboard(false);
}
e.stopPropagation();
e.preventDefault();
setSearchText('');
setSelectedCategory(categories[0]);
setSearchMode(m => !m);
},
[setSearchText, setSearchMode]
);
2019-05-24 23:58:27 +00:00
const debounceSearchChange = React.useMemo(
() =>
debounce((query: string) => {
2019-05-24 23:58:27 +00:00
setScrollToRow(0);
2022-02-07 23:00:04 +00:00
setSearchText(query);
2019-05-24 23:58:27 +00:00
}, 200),
[setSearchText, setScrollToRow]
);
const handleSearchChange = React.useCallback(
(e: React.ChangeEvent<HTMLInputElement>) => {
debounceSearchChange(e.currentTarget.value);
},
[debounceSearchChange]
);
const handlePickTone = React.useCallback(
(
e:
| React.MouseEvent<HTMLButtonElement>
| React.KeyboardEvent<HTMLButtonElement>
) => {
if (isEventFromMouse(e)) {
setIsUsingKeyboard(false);
}
e.preventDefault();
e.stopPropagation();
2019-05-24 23:58:27 +00:00
const { tone = '0' } = e.currentTarget.dataset;
const parsedTone = parseInt(tone, 10);
setSelectedTone(parsedTone);
2020-05-11 23:14:02 +00:00
if (onSetSkinTone) {
onSetSkinTone(parsedTone);
}
2019-05-24 23:58:27 +00:00
},
2020-05-11 23:14:02 +00:00
[onSetSkinTone]
2019-05-24 23:58:27 +00:00
);
const handlePickEmoji = React.useCallback(
(
e:
| React.MouseEvent<HTMLButtonElement>
| React.KeyboardEvent<HTMLButtonElement>
) => {
const { shortName } = e.currentTarget.dataset;
if ('key' in e) {
if (e.key === 'Enter') {
if (shortName && isUsingKeyboard) {
onPickEmoji({ skinTone: selectedTone, shortName });
e.stopPropagation();
e.preventDefault();
} else if (onClose) {
onClose();
e.stopPropagation();
e.preventDefault();
}
}
} else if (shortName) {
if (isEventFromMouse(e)) {
setIsUsingKeyboard(false);
}
e.stopPropagation();
e.preventDefault();
onPickEmoji({ skinTone: selectedTone, shortName });
2019-05-24 23:58:27 +00:00
}
},
[
onClose,
onPickEmoji,
isUsingKeyboard,
selectedTone,
setIsUsingKeyboard,
]
2019-05-24 23:58:27 +00:00
);
// Handle key presses, particularly Escape
2020-01-08 17:44:54 +00:00
React.useEffect(() => {
const handler = (event: KeyboardEvent) => {
if (event.key === 'Tab') {
// We do NOT prevent default here to allow Tab to be used normally
setIsUsingKeyboard(true);
return;
}
if (event.key === 'Escape') {
if (searchMode) {
event.preventDefault();
event.stopPropagation();
setScrollToRow(0);
setSearchText('');
setSearchMode(false);
} else if (onClose) {
event.preventDefault();
event.stopPropagation();
onClose();
}
} else if (!searchMode && !event.ctrlKey && !event.metaKey) {
if (
[
'ArrowUp',
'ArrowDown',
'ArrowLeft',
'ArrowRight',
'Enter',
'Shift',
' ', // Space
].includes(event.key)
) {
// Do nothing, these can be used to navigate around the picker.
} else if (isSingleGrapheme(event.key)) {
// A single grapheme means the user is typing text. Switch to search mode.
setSelectedCategory(categories[0]);
setSearchMode(true);
// Continue propagation, typing the first letter for search.
} else {
// For anything else, assume it's a special key that isn't one of the ones
// above (such as Delete or ContextMenu).
onClose?.();
event.preventDefault();
event.stopPropagation();
}
2020-01-08 17:44:54 +00:00
}
};
2019-05-24 23:58:27 +00:00
2020-01-08 17:44:54 +00:00
document.addEventListener('keydown', handler);
return () => {
document.removeEventListener('keydown', handler);
};
}, [onClose, setIsUsingKeyboard, searchMode, setSearchMode]);
2019-05-24 23:58:27 +00:00
2020-01-23 21:17:06 +00:00
const [, ...renderableCategories] = categories;
2020-01-08 17:44:54 +00:00
const emojiGrid = React.useMemo(() => {
if (searchText) {
2024-03-21 16:35:54 +00:00
return chunk(search(searchText), COL_COUNT);
2020-01-08 17:44:54 +00:00
}
2019-05-24 23:58:27 +00:00
2020-01-23 21:17:06 +00:00
const chunks = flatMap(renderableCategories, cat =>
2020-01-08 17:44:54 +00:00
chunk(
dataByCategory[cat].map(e => e.short_name),
COL_COUNT
)
);
2019-05-24 23:58:27 +00:00
2020-01-08 17:44:54 +00:00
return [...chunk(firstRecent, COL_COUNT), ...chunks];
2024-03-21 16:35:54 +00:00
}, [firstRecent, renderableCategories, searchText, search]);
2019-05-24 23:58:27 +00:00
2022-02-07 23:00:04 +00:00
const rowCount = emojiGrid.length;
2020-01-08 17:44:54 +00:00
const catRowEnds = React.useMemo(() => {
const rowEnds: Array<number> = [
Math.ceil(firstRecent.length / COL_COUNT) - 1,
];
2019-05-24 23:58:27 +00:00
2020-01-23 21:17:06 +00:00
renderableCategories.forEach(cat => {
2020-01-08 17:44:54 +00:00
rowEnds.push(
Math.ceil(dataByCategory[cat].length / COL_COUNT) +
(last(rowEnds) as number)
);
});
2019-05-24 23:58:27 +00:00
2020-01-08 17:44:54 +00:00
return rowEnds;
}, [firstRecent.length, renderableCategories]);
2020-01-08 17:44:54 +00:00
const catToRowOffsets = React.useMemo(() => {
const offsets = initial(catRowEnds).map(i => i + 1);
return zipObject(categories, [0, ...offsets]);
}, [catRowEnds]);
2019-05-24 23:58:27 +00:00
const catOffsetEntries = React.useMemo(
() => Object.entries(catToRowOffsets),
[catToRowOffsets]
);
const handleSelectCategory = React.useCallback(
(
e:
| React.MouseEvent<HTMLButtonElement>
| React.KeyboardEvent<HTMLButtonElement>
) => {
e.stopPropagation();
e.preventDefault();
const { category } = e.currentTarget.dataset;
2019-05-24 23:58:27 +00:00
if (category) {
setSelectedCategory(category as Category);
2019-05-24 23:58:27 +00:00
setScrollToRow(catToRowOffsets[category]);
}
},
[catToRowOffsets, setSelectedCategory, setScrollToRow]
);
const cellRenderer = React.useCallback<GridCellRenderer>(
({ key, style: cellStyle, rowIndex, columnIndex }) => {
const shortName = emojiGrid[rowIndex][columnIndex];
return shortName ? (
<div
key={key}
className="module-emoji-picker__body__emoji-cell"
style={cellStyle}
>
<button
type="button"
2019-05-24 23:58:27 +00:00
className="module-emoji-picker__button"
onClick={handlePickEmoji}
onKeyDown={handlePickEmoji}
2019-05-24 23:58:27 +00:00
data-short-name={shortName}
title={shortName}
>
<Emoji shortName={shortName} skinTone={selectedTone} />
</button>
</div>
) : null;
},
[emojiGrid, handlePickEmoji, selectedTone]
2019-05-24 23:58:27 +00:00
);
const getRowHeight = React.useCallback(
({ index }: { index: number }) => {
if (searchText) {
return 34;
}
if (catRowEnds.includes(index) && index !== last(catRowEnds)) {
return 44;
}
return 34;
},
[catRowEnds, searchText]
);
const onSectionRendered = React.useMemo(
() =>
debounce(({ rowStartIndex }: SectionRenderedParams) => {
const [cat] =
findLast(catOffsetEntries, ([, row]) => rowStartIndex >= row) ||
categories;
setSelectedCategory(cat as Category);
2019-05-24 23:58:27 +00:00
}, 10),
[catOffsetEntries]
2019-05-24 23:58:27 +00:00
);
function getCategoryButtonLabel(category: Category): string {
switch (category) {
case 'recents':
2023-03-30 00:03:25 +00:00
return i18n('icu:EmojiPicker__button--recents');
case 'emoji':
2023-03-30 00:03:25 +00:00
return i18n('icu:EmojiPicker__button--emoji');
case 'animal':
2023-03-30 00:03:25 +00:00
return i18n('icu:EmojiPicker__button--animal');
case 'food':
2023-03-30 00:03:25 +00:00
return i18n('icu:EmojiPicker__button--food');
case 'activity':
2023-03-30 00:03:25 +00:00
return i18n('icu:EmojiPicker__button--activity');
case 'travel':
2023-03-30 00:03:25 +00:00
return i18n('icu:EmojiPicker__button--travel');
case 'object':
2023-03-30 00:03:25 +00:00
return i18n('icu:EmojiPicker__button--object');
case 'symbol':
2023-03-30 00:03:25 +00:00
return i18n('icu:EmojiPicker__button--symbol');
case 'flag':
2023-03-30 00:03:25 +00:00
return i18n('icu:EmojiPicker__button--flag');
default:
throw missingCaseError(category);
}
}
2019-05-24 23:58:27 +00:00
return (
<FocusTrap
focusTrapOptions={{
allowOutsideClick: true,
returnFocusOnDeactivate: false,
}}
>
<div className="module-emoji-picker" ref={ref} style={style}>
<header className="module-emoji-picker__header">
<button
type="button"
onClick={handleToggleSearch}
onKeyDown={event => {
if (event.key === 'Enter' || event.key === 'Select') {
handleToggleSearch(event);
}
}}
title={
searchMode
? i18n('icu:EmojiPicker--search-close')
: i18n('icu:EmojiPicker--search-placeholder')
}
className={classNames(
'module-emoji-picker__button',
'module-emoji-picker__button--icon',
searchMode
? 'module-emoji-picker__button--icon--close'
: 'module-emoji-picker__button--icon--search'
)}
2023-03-30 00:03:25 +00:00
aria-label={i18n('icu:EmojiPicker--search-placeholder')}
/>
{searchMode ? (
<div className="module-emoji-picker__header__search-field">
<input
ref={focusOnRender}
className="module-emoji-picker__header__search-field__input"
2023-03-30 00:03:25 +00:00
placeholder={i18n('icu:EmojiPicker--search-placeholder')}
onChange={handleSearchChange}
2023-04-20 17:03:43 +00:00
dir="auto"
/>
</div>
) : (
categories.map(cat =>
cat === 'recents' && firstRecent.length === 0 ? null : (
<button
2023-05-04 23:41:45 +00:00
aria-pressed={selectedCategory === cat}
type="button"
key={cat}
data-category={cat}
title={cat}
onClick={handleSelectCategory}
onKeyDown={event => {
if (event.key === 'Enter' || event.key === 'Space') {
handleSelectCategory(event);
}
}}
className={classNames(
'module-emoji-picker__button',
'module-emoji-picker__button--icon',
`module-emoji-picker__button--icon--${cat}`,
selectedCategory === cat
? 'module-emoji-picker__button--selected'
: null
)}
aria-label={getCategoryButtonLabel(cat)}
/>
)
)
2019-05-24 23:58:27 +00:00
)}
</header>
2022-02-07 23:00:04 +00:00
{rowCount > 0 ? (
<div>
<AutoSizer>
{({ width, height }) => (
<Grid
key={searchText}
className="module-emoji-picker__body"
width={width}
height={height}
columnCount={COL_COUNT}
columnWidth={38}
// react-virtualized Grid default style has direction: 'ltr'
style={{ direction: isRTL ? 'rtl' : 'ltr' }}
rowHeight={getRowHeight}
2022-02-07 23:00:04 +00:00
rowCount={rowCount}
cellRenderer={cellRenderer}
2022-02-07 23:00:04 +00:00
// In some cases, `scrollToRow` can be too high for a short period
// during state changes. This ensures that the value is never too
// large.
scrollToRow={clamp(scrollToRow, 0, rowCount - 1)}
scrollToAlignment="start"
onSectionRendered={onSectionRendered}
/>
)}
</AutoSizer>
2019-05-24 23:58:27 +00:00
</div>
) : (
<div
className={classNames(
'module-emoji-picker__body',
'module-emoji-picker__body--empty'
)}
>
2023-03-30 00:03:25 +00:00
{i18n('icu:EmojiPicker--empty')}
<Emoji
shortName="slightly_frowning_face"
size={16}
2023-04-20 17:03:43 +00:00
style={{ marginInlineStart: '4px' }}
/>
</div>
)}
<footer className="module-emoji-picker__footer">
{Boolean(onClickSettings) && (
<button
2023-03-30 00:03:25 +00:00
aria-label={i18n('icu:CustomizingPreferredReactions__title')}
className="module-emoji-picker__button module-emoji-picker__button--footer module-emoji-picker__button--settings"
onClick={event => {
if (onClickSettings) {
event.preventDefault();
event.stopPropagation();
onClickSettings();
}
}}
onKeyDown={event => {
if (
onClickSettings &&
(event.key === 'Enter' || event.key === 'Space')
) {
event.preventDefault();
event.stopPropagation();
onClickSettings();
}
}}
2023-03-30 00:03:25 +00:00
title={i18n('icu:CustomizingPreferredReactions__title')}
type="button"
/>
)}
{onSetSkinTone ? (
<div className="module-emoji-picker__footer__skin-tones">
{[0, 1, 2, 3, 4, 5].map(tone => (
<button
2023-05-04 23:41:45 +00:00
aria-pressed={selectedTone === tone}
type="button"
key={tone}
data-tone={tone}
onClick={handlePickTone}
onKeyDown={event => {
if (event.key === 'Enter' || event.key === 'Space') {
handlePickTone(event);
}
}}
2023-03-30 00:03:25 +00:00
title={i18n('icu:EmojiPicker--skin-tone', {
2023-03-27 23:37:39 +00:00
tone: `${tone}`,
})}
className={classNames(
'module-emoji-picker__button',
'module-emoji-picker__button--footer',
selectedTone === tone
? 'module-emoji-picker__button--selected'
: null
)}
>
<Emoji shortName="hand" skinTone={tone} size={20} />
</button>
))}
</div>
) : null}
{Boolean(onClickSettings) && (
<div className="module-emoji-picker__footer__settings-spacer" />
2019-05-24 23:58:27 +00:00
)}
</footer>
</div>
</FocusTrap>
2019-05-24 23:58:27 +00:00
);
}
)
);