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';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type {
|
2019-05-24 23:58:27 +00:00
|
|
|
GridCellRenderer,
|
|
|
|
SectionRenderedParams,
|
|
|
|
} from 'react-virtualized';
|
2021-10-26 19:15:33 +00:00
|
|
|
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';
|
2021-10-04 17:14:00 +00:00
|
|
|
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';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { LocalizerType } from '../../types/Util';
|
2022-08-22 23:31:35 +00:00
|
|
|
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
|
|
|
|
2020-10-21 16:53:32 +00:00
|
|
|
export type EmojiPickDataType = {
|
|
|
|
skinTone?: number;
|
|
|
|
shortName: string;
|
|
|
|
};
|
2019-06-27 20:35:21 +00:00
|
|
|
|
2019-05-24 23:58:27 +00:00
|
|
|
export type OwnProps = {
|
|
|
|
readonly i18n: LocalizerType;
|
2022-12-22 00:07:02 +00:00
|
|
|
readonly recentEmojis?: ReadonlyArray<string>;
|
2024-03-19 13:23:31 +00:00
|
|
|
readonly skinTone?: number;
|
2021-09-09 16:29:01 +00:00
|
|
|
readonly onClickSettings?: () => unknown;
|
2020-05-05 19:49:34 +00:00
|
|
|
readonly onClose?: () => unknown;
|
2024-03-19 13:23:31 +00:00
|
|
|
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'>;
|
|
|
|
|
2024-03-19 13:23:31 +00:00
|
|
|
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',
|
2023-03-29 17:15:54 +00:00
|
|
|
] 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,
|
2021-09-09 16:29:01 +00:00
|
|
|
onClickSettings,
|
2019-05-24 23:58:27 +00:00
|
|
|
onClose,
|
2024-03-19 13:23:31 +00:00
|
|
|
wasInvokedFromKeyboard,
|
2019-05-24 23:58:27 +00:00
|
|
|
}: Props,
|
|
|
|
ref
|
|
|
|
) => {
|
2023-11-20 20:46:49 +00:00
|
|
|
const isRTL = i18n.getLocaleDirection() === 'rtl';
|
|
|
|
|
2024-03-19 13:23:31 +00:00
|
|
|
const [isUsingKeyboard, setIsUsingKeyboard] = React.useState(
|
|
|
|
wasInvokedFromKeyboard
|
|
|
|
);
|
|
|
|
|
2020-09-01 00:09:28 +00:00
|
|
|
const [firstRecent] = React.useState(recentEmojis);
|
2023-03-29 17:15:54 +00:00
|
|
|
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());
|
|
|
|
|
2020-05-05 19:49:34 +00:00
|
|
|
const handleToggleSearch = React.useCallback(
|
2022-09-07 18:29:08 +00:00
|
|
|
(
|
|
|
|
e:
|
|
|
|
| React.MouseEvent<HTMLButtonElement>
|
|
|
|
| React.KeyboardEvent<HTMLButtonElement>
|
|
|
|
) => {
|
2024-03-19 13:23:31 +00:00
|
|
|
if (isEventFromMouse(e)) {
|
|
|
|
setIsUsingKeyboard(false);
|
|
|
|
}
|
2020-05-05 19:49:34 +00:00
|
|
|
e.stopPropagation();
|
2022-09-07 18:29:08 +00:00
|
|
|
e.preventDefault();
|
|
|
|
|
2020-05-05 19:49:34 +00:00
|
|
|
setSearchText('');
|
|
|
|
setSelectedCategory(categories[0]);
|
|
|
|
setSearchMode(m => !m);
|
|
|
|
},
|
|
|
|
[setSearchText, setSearchMode]
|
|
|
|
);
|
2019-05-24 23:58:27 +00:00
|
|
|
|
|
|
|
const debounceSearchChange = React.useMemo(
|
|
|
|
() =>
|
2020-02-07 19:37:04 +00:00
|
|
|
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(
|
2022-09-07 18:29:08 +00:00
|
|
|
(
|
|
|
|
e:
|
|
|
|
| React.MouseEvent<HTMLButtonElement>
|
|
|
|
| React.KeyboardEvent<HTMLButtonElement>
|
|
|
|
) => {
|
2024-03-19 13:23:31 +00:00
|
|
|
if (isEventFromMouse(e)) {
|
|
|
|
setIsUsingKeyboard(false);
|
|
|
|
}
|
2020-10-12 14:53:39 +00:00
|
|
|
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(
|
2019-05-31 21:58:53 +00:00
|
|
|
(
|
|
|
|
e:
|
|
|
|
| React.MouseEvent<HTMLButtonElement>
|
|
|
|
| React.KeyboardEvent<HTMLButtonElement>
|
|
|
|
) => {
|
2022-09-07 18:29:08 +00:00
|
|
|
const { shortName } = e.currentTarget.dataset;
|
2019-05-31 21:58:53 +00:00
|
|
|
if ('key' in e) {
|
2022-09-07 18:29:08 +00:00
|
|
|
if (e.key === 'Enter') {
|
2024-03-19 13:23:31 +00:00
|
|
|
if (shortName && isUsingKeyboard) {
|
2022-09-07 18:29:08 +00:00
|
|
|
onPickEmoji({ skinTone: selectedTone, shortName });
|
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
2024-03-19 13:23:31 +00:00
|
|
|
} else if (onClose) {
|
|
|
|
onClose();
|
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
2022-09-07 18:29:08 +00:00
|
|
|
}
|
2019-05-31 21:58:53 +00:00
|
|
|
}
|
2022-09-07 18:29:08 +00:00
|
|
|
} else if (shortName) {
|
2024-03-19 13:23:31 +00:00
|
|
|
if (isEventFromMouse(e)) {
|
|
|
|
setIsUsingKeyboard(false);
|
|
|
|
}
|
2022-09-07 18:29:08 +00:00
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
|
|
|
onPickEmoji({ skinTone: selectedTone, shortName });
|
2019-05-24 23:58:27 +00:00
|
|
|
}
|
|
|
|
},
|
2024-03-19 13:23:31 +00:00
|
|
|
[
|
|
|
|
onClose,
|
|
|
|
onPickEmoji,
|
|
|
|
isUsingKeyboard,
|
|
|
|
selectedTone,
|
|
|
|
setIsUsingKeyboard,
|
|
|
|
]
|
2019-05-24 23:58:27 +00:00
|
|
|
);
|
|
|
|
|
2022-08-22 23:31:35 +00:00
|
|
|
// Handle key presses, particularly Escape
|
2020-01-08 17:44:54 +00:00
|
|
|
React.useEffect(() => {
|
|
|
|
const handler = (event: KeyboardEvent) => {
|
2024-03-19 13:23:31 +00:00
|
|
|
if (event.key === 'Tab') {
|
|
|
|
// We do NOT prevent default here to allow Tab to be used normally
|
|
|
|
setIsUsingKeyboard(true);
|
|
|
|
return;
|
|
|
|
}
|
2022-08-22 23:31:35 +00:00
|
|
|
if (event.key === 'Escape') {
|
|
|
|
if (searchMode) {
|
2022-09-07 18:29:08 +00:00
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
2022-08-22 23:31:35 +00:00
|
|
|
setScrollToRow(0);
|
|
|
|
setSearchText('');
|
|
|
|
setSearchMode(false);
|
2022-09-07 18:29:08 +00:00
|
|
|
} else if (onClose) {
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
onClose();
|
2020-05-05 19:49:34 +00:00
|
|
|
}
|
2022-08-22 23:31:35 +00:00
|
|
|
} 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);
|
|
|
|
};
|
2024-03-19 13:23:31 +00:00
|
|
|
}, [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;
|
2020-09-01 00:09:28 +00:00
|
|
|
}, [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]);
|
2020-09-01 00:09:28 +00:00
|
|
|
}, [catRowEnds]);
|
2019-05-24 23:58:27 +00:00
|
|
|
|
|
|
|
const catOffsetEntries = React.useMemo(
|
|
|
|
() => Object.entries(catToRowOffsets),
|
|
|
|
[catToRowOffsets]
|
|
|
|
);
|
|
|
|
|
|
|
|
const handleSelectCategory = React.useCallback(
|
2022-09-07 18:29:08 +00:00
|
|
|
(
|
|
|
|
e:
|
|
|
|
| React.MouseEvent<HTMLButtonElement>
|
|
|
|
| React.KeyboardEvent<HTMLButtonElement>
|
|
|
|
) => {
|
2020-05-05 19:49:34 +00:00
|
|
|
e.stopPropagation();
|
2022-09-07 18:29:08 +00:00
|
|
|
e.preventDefault();
|
|
|
|
|
2020-05-05 19:49:34 +00:00
|
|
|
const { category } = e.currentTarget.dataset;
|
2019-05-24 23:58:27 +00:00
|
|
|
if (category) {
|
2023-03-29 17:15:54 +00:00
|
|
|
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
|
2020-09-01 00:09:28 +00:00
|
|
|
type="button"
|
2019-05-24 23:58:27 +00:00
|
|
|
className="module-emoji-picker__button"
|
|
|
|
onClick={handlePickEmoji}
|
2019-05-31 21:58:53 +00:00
|
|
|
onKeyDown={handlePickEmoji}
|
2019-05-24 23:58:27 +00:00
|
|
|
data-short-name={shortName}
|
|
|
|
title={shortName}
|
|
|
|
>
|
|
|
|
<Emoji shortName={shortName} skinTone={selectedTone} />
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
) : null;
|
|
|
|
},
|
2020-09-01 00:09:28 +00:00
|
|
|
[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;
|
|
|
|
|
2023-03-29 17:15:54 +00:00
|
|
|
setSelectedCategory(cat as Category);
|
2019-05-24 23:58:27 +00:00
|
|
|
}, 10),
|
2020-09-01 00:09:28 +00:00
|
|
|
[catOffsetEntries]
|
2019-05-24 23:58:27 +00:00
|
|
|
);
|
|
|
|
|
2023-03-29 17:15:54 +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');
|
2023-03-29 17:15:54 +00:00
|
|
|
case 'emoji':
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:EmojiPicker__button--emoji');
|
2023-03-29 17:15:54 +00:00
|
|
|
case 'animal':
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:EmojiPicker__button--animal');
|
2023-03-29 17:15:54 +00:00
|
|
|
case 'food':
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:EmojiPicker__button--food');
|
2023-03-29 17:15:54 +00:00
|
|
|
case 'activity':
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:EmojiPicker__button--activity');
|
2023-03-29 17:15:54 +00:00
|
|
|
case 'travel':
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:EmojiPicker__button--travel');
|
2023-03-29 17:15:54 +00:00
|
|
|
case 'object':
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:EmojiPicker__button--object');
|
2023-03-29 17:15:54 +00:00
|
|
|
case 'symbol':
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:EmojiPicker__button--symbol');
|
2023-03-29 17:15:54 +00:00
|
|
|
case 'flag':
|
2023-03-30 00:03:25 +00:00
|
|
|
return i18n('icu:EmojiPicker__button--flag');
|
2023-03-29 17:15:54 +00:00
|
|
|
default:
|
|
|
|
throw missingCaseError(category);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-24 23:58:27 +00:00
|
|
|
return (
|
2021-10-06 20:56:37 +00:00
|
|
|
<FocusTrap
|
|
|
|
focusTrapOptions={{
|
|
|
|
allowOutsideClick: true,
|
2024-12-10 00:03:33 +00:00
|
|
|
returnFocusOnDeactivate: false,
|
2021-10-06 20:56:37 +00:00
|
|
|
}}
|
|
|
|
>
|
2021-10-04 17:14:00 +00:00
|
|
|
<div className="module-emoji-picker" ref={ref} style={style}>
|
|
|
|
<header className="module-emoji-picker__header">
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
onClick={handleToggleSearch}
|
2022-09-07 18:29:08 +00:00
|
|
|
onKeyDown={event => {
|
|
|
|
if (event.key === 'Enter' || event.key === 'Select') {
|
|
|
|
handleToggleSearch(event);
|
|
|
|
}
|
|
|
|
}}
|
2023-04-25 18:01:21 +00:00
|
|
|
title={
|
|
|
|
searchMode
|
|
|
|
? i18n('icu:EmojiPicker--search-close')
|
|
|
|
: i18n('icu:EmojiPicker--search-placeholder')
|
|
|
|
}
|
2021-10-04 17:14:00 +00:00
|
|
|
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')}
|
2021-10-04 17:14:00 +00:00
|
|
|
/>
|
|
|
|
{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')}
|
2021-10-04 17:14:00 +00:00
|
|
|
onChange={handleSearchChange}
|
2023-04-20 17:03:43 +00:00
|
|
|
dir="auto"
|
2021-10-04 17:14:00 +00:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
) : (
|
|
|
|
categories.map(cat =>
|
|
|
|
cat === 'recents' && firstRecent.length === 0 ? null : (
|
|
|
|
<button
|
2023-05-04 23:41:45 +00:00
|
|
|
aria-pressed={selectedCategory === cat}
|
2021-10-04 17:14:00 +00:00
|
|
|
type="button"
|
|
|
|
key={cat}
|
|
|
|
data-category={cat}
|
|
|
|
title={cat}
|
|
|
|
onClick={handleSelectCategory}
|
2022-09-07 18:29:08 +00:00
|
|
|
onKeyDown={event => {
|
|
|
|
if (event.key === 'Enter' || event.key === 'Space') {
|
|
|
|
handleSelectCategory(event);
|
|
|
|
}
|
|
|
|
}}
|
2021-10-04 17:14:00 +00:00
|
|
|
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
|
|
|
|
)}
|
2023-03-29 17:15:54 +00:00
|
|
|
aria-label={getCategoryButtonLabel(cat)}
|
2021-10-04 17:14:00 +00:00
|
|
|
/>
|
|
|
|
)
|
|
|
|
)
|
2019-05-24 23:58:27 +00:00
|
|
|
)}
|
2021-10-04 17:14:00 +00:00
|
|
|
</header>
|
2022-02-07 23:00:04 +00:00
|
|
|
{rowCount > 0 ? (
|
2021-10-04 17:14:00 +00:00
|
|
|
<div>
|
|
|
|
<AutoSizer>
|
|
|
|
{({ width, height }) => (
|
|
|
|
<Grid
|
|
|
|
key={searchText}
|
|
|
|
className="module-emoji-picker__body"
|
|
|
|
width={width}
|
|
|
|
height={height}
|
|
|
|
columnCount={COL_COUNT}
|
|
|
|
columnWidth={38}
|
2023-11-20 20:46:49 +00:00
|
|
|
// react-virtualized Grid default style has direction: 'ltr'
|
|
|
|
style={{ direction: isRTL ? 'rtl' : 'ltr' }}
|
2021-10-04 17:14:00 +00:00
|
|
|
rowHeight={getRowHeight}
|
2022-02-07 23:00:04 +00:00
|
|
|
rowCount={rowCount}
|
2021-10-04 17:14:00 +00:00
|
|
|
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)}
|
2021-10-04 17:14:00 +00:00
|
|
|
scrollToAlignment="start"
|
|
|
|
onSectionRendered={onSectionRendered}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</AutoSizer>
|
2019-05-24 23:58:27 +00:00
|
|
|
</div>
|
|
|
|
) : (
|
2021-10-04 17:14:00 +00:00
|
|
|
<div
|
|
|
|
className={classNames(
|
|
|
|
'module-emoji-picker__body',
|
|
|
|
'module-emoji-picker__body--empty'
|
|
|
|
)}
|
|
|
|
>
|
2023-03-30 00:03:25 +00:00
|
|
|
{i18n('icu:EmojiPicker--empty')}
|
2021-10-04 17:14:00 +00:00
|
|
|
<Emoji
|
|
|
|
shortName="slightly_frowning_face"
|
|
|
|
size={16}
|
2023-04-20 17:03:43 +00:00
|
|
|
style={{ marginInlineStart: '4px' }}
|
2021-10-04 17:14:00 +00:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
<footer className="module-emoji-picker__footer">
|
|
|
|
{Boolean(onClickSettings) && (
|
|
|
|
<button
|
2023-03-30 00:03:25 +00:00
|
|
|
aria-label={i18n('icu:CustomizingPreferredReactions__title')}
|
2021-10-04 17:14:00 +00:00
|
|
|
className="module-emoji-picker__button module-emoji-picker__button--footer module-emoji-picker__button--settings"
|
2022-09-07 18:29:08 +00:00
|
|
|
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')}
|
2021-10-04 17:14:00 +00:00
|
|
|
type="button"
|
|
|
|
/>
|
|
|
|
)}
|
2022-06-14 01:48:07 +00:00
|
|
|
{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}
|
2022-06-14 01:48:07 +00:00
|
|
|
type="button"
|
|
|
|
key={tone}
|
|
|
|
data-tone={tone}
|
|
|
|
onClick={handlePickTone}
|
2022-09-07 18:29:08 +00:00
|
|
|
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}`,
|
|
|
|
})}
|
2022-06-14 01:48:07 +00:00
|
|
|
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}
|
2021-10-04 17:14:00 +00:00
|
|
|
{Boolean(onClickSettings) && (
|
|
|
|
<div className="module-emoji-picker__footer__settings-spacer" />
|
2019-05-24 23:58:27 +00:00
|
|
|
)}
|
2021-10-04 17:14:00 +00:00
|
|
|
</footer>
|
|
|
|
</div>
|
|
|
|
</FocusTrap>
|
2019-05-24 23:58:27 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
)
|
|
|
|
);
|