UNPKG

matrix-react-sdk

Version:
339 lines (333 loc) 59.6 kB
"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = exports.EMOJI_HEIGHT = exports.EMOJIS_PER_ROW = exports.CATEGORY_HEADER_HEIGHT = void 0; var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _react = _interopRequireDefault(require("react")); var _emojibaseBindings = require("@matrix-org/emojibase-bindings"); var _languageHandler = require("../../../languageHandler"); var recent = _interopRequireWildcard(require("../../../emojipicker/recent")); var _AutoHideScrollbar = _interopRequireDefault(require("../../structures/AutoHideScrollbar")); var _Header = _interopRequireDefault(require("./Header")); var _Search = _interopRequireDefault(require("./Search")); var _Preview = _interopRequireDefault(require("./Preview")); var _QuickReactions = _interopRequireDefault(require("./QuickReactions")); var _Category = _interopRequireDefault(require("./Category")); var _arrays = require("../../../utils/arrays"); var _RovingTabIndex = require("../../../accessibility/RovingTabIndex"); var _Keyboard = require("../../../Keyboard"); var _numbers = require("../../../utils/numbers"); function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); } function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; } function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; } function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2.default)(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } /* Copyright 2024 New Vector Ltd. Copyright 2020 The Matrix.org Foundation C.I.C. Copyright 2019 Tulir Asokan <tulir@maunium.net> SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only Please see LICENSE files in the repository root for full details. */ const CATEGORY_HEADER_HEIGHT = exports.CATEGORY_HEADER_HEIGHT = 20; const EMOJI_HEIGHT = exports.EMOJI_HEIGHT = 35; const EMOJIS_PER_ROW = exports.EMOJIS_PER_ROW = 8; const ZERO_WIDTH_JOINER = "\u200D"; class EmojiPicker extends _react.default.Component { constructor(props) { super(props); (0, _defineProperty2.default)(this, "recentlyUsed", void 0); (0, _defineProperty2.default)(this, "memoizedDataByCategory", void 0); (0, _defineProperty2.default)(this, "categories", void 0); (0, _defineProperty2.default)(this, "scrollRef", /*#__PURE__*/_react.default.createRef()); (0, _defineProperty2.default)(this, "onScroll", () => { const body = this.scrollRef.current?.containerRef.current; if (!body) return; this.setState({ scrollTop: body.scrollTop, viewportHeight: body.clientHeight }); this.updateVisibility(); }); (0, _defineProperty2.default)(this, "onKeyDown", (ev, state, dispatch) => { if (state.activeRef?.current && [_Keyboard.Key.ARROW_DOWN, _Keyboard.Key.ARROW_RIGHT, _Keyboard.Key.ARROW_LEFT, _Keyboard.Key.ARROW_UP].includes(ev.key)) { this.keyboardNavigation(ev, state, dispatch); } }); (0, _defineProperty2.default)(this, "updateVisibility", () => { const body = this.scrollRef.current?.containerRef.current; if (!body) return; const rect = body.getBoundingClientRect(); for (const cat of this.categories) { const elem = body.querySelector(`[data-category-id="${cat.id}"]`); if (!elem) { cat.visible = false; cat.ref.current?.classList.remove("mx_EmojiPicker_anchor_visible"); continue; } const elemRect = elem.getBoundingClientRect(); const y = elemRect.y - rect.y; const yEnd = elemRect.y + elemRect.height - rect.y; cat.visible = y < rect.height && yEnd > 0; // We update this here instead of through React to avoid re-render on scroll. if (!cat.ref.current) continue; if (cat.visible) { cat.ref.current.classList.add("mx_EmojiPicker_anchor_visible"); cat.ref.current.setAttribute("aria-selected", "true"); cat.ref.current.setAttribute("tabindex", "0"); } else { cat.ref.current.classList.remove("mx_EmojiPicker_anchor_visible"); cat.ref.current.setAttribute("aria-selected", "false"); cat.ref.current.setAttribute("tabindex", "-1"); } } }); (0, _defineProperty2.default)(this, "scrollToCategory", category => { this.scrollRef.current?.containerRef.current?.querySelector(`[data-category-id="${category}"]`)?.scrollIntoView(); }); (0, _defineProperty2.default)(this, "onChangeFilter", filter => { const lcFilter = filter.toLowerCase().trim(); // filter is case insensitive for (const cat of this.categories) { let emojis; // If the new filter string includes the old filter string, we don't have to re-filter the whole dataset. if (lcFilter.includes(this.state.filter)) { emojis = this.memoizedDataByCategory[cat.id]; } else { emojis = cat.id === "recent" ? this.recentlyUsed : _emojibaseBindings.DATA_BY_CATEGORY[cat.id]; } if (lcFilter !== "") { emojis = emojis.filter(emoji => this.emojiMatchesFilter(emoji, lcFilter)); // Copy the array to not clobber the original unfiltered sorting emojis = [...emojis].sort((a, b) => { const indexA = a.shortcodes[0].indexOf(lcFilter); const indexB = b.shortcodes[0].indexOf(lcFilter); // Prioritize emojis containing the filter in its shortcode if (indexA == -1 || indexB == -1) { return indexB - indexA; } // If both emojis start with the filter // put the shorter emoji first if (indexA == 0 && indexB == 0) { return a.shortcodes[0].length - b.shortcodes[0].length; } // Prioritize emojis starting with the filter return indexA - indexB; }); } this.memoizedDataByCategory[cat.id] = emojis; cat.enabled = emojis.length > 0; // The setState below doesn't re-render the header and we already have the refs for updateVisibility, so... if (cat.ref.current) { cat.ref.current.disabled = !cat.enabled; } } this.setState({ filter }); // Header underlines need to be updated, but updating requires knowing // where the categories are, so we wait for a tick. window.setTimeout(this.updateVisibility, 0); }); (0, _defineProperty2.default)(this, "emojiMatchesFilter", (emoji, filter) => { // If the query is an emoji containing a variation then strip it to provide more useful matches if (filter.includes(ZERO_WIDTH_JOINER)) { filter = filter.split(ZERO_WIDTH_JOINER, 2)[0]; } return emoji.label.toLowerCase().includes(filter) || (Array.isArray(emoji.emoticon) ? emoji.emoticon.some(x => x.includes(filter)) : emoji.emoticon?.includes(filter)) || emoji.shortcodes.some(x => x.toLowerCase().includes(filter)) || emoji.unicode.split(ZERO_WIDTH_JOINER).includes(filter); }); (0, _defineProperty2.default)(this, "onEnterFilter", () => { const btn = this.scrollRef.current?.containerRef.current?.querySelector('.mx_EmojiPicker_item_wrapper[tabindex="0"]'); btn?.click(); this.props.onFinished(); }); (0, _defineProperty2.default)(this, "onHoverEmoji", emoji => { this.setState({ previewEmoji: emoji }); }); (0, _defineProperty2.default)(this, "onHoverEmojiEnd", () => { this.setState({ previewEmoji: undefined }); }); (0, _defineProperty2.default)(this, "onClickEmoji", (ev, emoji) => { if (this.props.onChoose(emoji.unicode) !== false) { recent.add(emoji.unicode); } if (ev.key === _Keyboard.Key.ENTER) { this.props.onFinished(); } }); this.state = { filter: "", scrollTop: 0, viewportHeight: 280 }; // Convert recent emoji characters to emoji data, removing unknowns and duplicates this.recentlyUsed = Array.from(new Set((0, _arrays.filterBoolean)(recent.get().map(_emojibaseBindings.getEmojiFromUnicode)))); this.memoizedDataByCategory = _objectSpread({ recent: this.recentlyUsed }, _emojibaseBindings.DATA_BY_CATEGORY); this.categories = [{ id: "recent", name: (0, _languageHandler._t)("emoji|category_frequently_used"), enabled: this.recentlyUsed.length > 0, visible: this.recentlyUsed.length > 0, ref: /*#__PURE__*/_react.default.createRef() }, { id: "people", name: (0, _languageHandler._t)("emoji|category_smileys_people"), enabled: true, visible: true, ref: /*#__PURE__*/_react.default.createRef() }, { id: "nature", name: (0, _languageHandler._t)("emoji|category_animals_nature"), enabled: true, visible: false, ref: /*#__PURE__*/_react.default.createRef() }, { id: "foods", name: (0, _languageHandler._t)("emoji|category_food_drink"), enabled: true, visible: false, ref: /*#__PURE__*/_react.default.createRef() }, { id: "activity", name: (0, _languageHandler._t)("emoji|category_activities"), enabled: true, visible: false, ref: /*#__PURE__*/_react.default.createRef() }, { id: "places", name: (0, _languageHandler._t)("emoji|category_travel_places"), enabled: true, visible: false, ref: /*#__PURE__*/_react.default.createRef() }, { id: "objects", name: (0, _languageHandler._t)("emoji|category_objects"), enabled: true, visible: false, ref: /*#__PURE__*/_react.default.createRef() }, { id: "symbols", name: (0, _languageHandler._t)("emoji|category_symbols"), enabled: true, visible: false, ref: /*#__PURE__*/_react.default.createRef() }, { id: "flags", name: (0, _languageHandler._t)("emoji|category_flags"), enabled: true, visible: false, ref: /*#__PURE__*/_react.default.createRef() }]; } keyboardNavigation(ev, state, dispatch) { const node = state.activeRef?.current; const parent = node?.parentElement; if (!parent || !state.activeRef) return; const rowIndex = Array.from(parent.children).indexOf(node); const refIndex = state.refs.indexOf(state.activeRef); let focusRef; let newParent; switch (ev.key) { case _Keyboard.Key.ARROW_LEFT: focusRef = state.refs[refIndex - 1]; newParent = focusRef?.current?.parentElement ?? undefined; break; case _Keyboard.Key.ARROW_RIGHT: focusRef = state.refs[refIndex + 1]; newParent = focusRef?.current?.parentElement ?? undefined; break; case _Keyboard.Key.ARROW_UP: case _Keyboard.Key.ARROW_DOWN: { // For up/down we find the prev/next parent by inspecting the refs either side of our row const ref = ev.key === _Keyboard.Key.ARROW_UP ? state.refs[refIndex - rowIndex - 1] : state.refs[refIndex - rowIndex + EMOJIS_PER_ROW]; newParent = ref?.current?.parentElement ?? undefined; const newTarget = newParent?.children[(0, _numbers.clamp)(rowIndex, 0, newParent.children.length - 1)]; focusRef = state.refs.find(r => r.current === newTarget); break; } } if (focusRef) { dispatch({ type: _RovingTabIndex.Type.SetFocus, payload: { ref: focusRef } }); if (parent !== newParent) { focusRef.current?.scrollIntoView({ behavior: "auto", block: "center", inline: "center" }); } } ev.preventDefault(); ev.stopPropagation(); } static categoryHeightForEmojiCount(count) { if (count === 0) { return 0; } return CATEGORY_HEADER_HEIGHT + Math.ceil(count / EMOJIS_PER_ROW) * EMOJI_HEIGHT; } render() { return /*#__PURE__*/_react.default.createElement(_RovingTabIndex.RovingTabIndexProvider, { onKeyDown: this.onKeyDown }, ({ onKeyDownHandler }) => { let heightBefore = 0; return /*#__PURE__*/_react.default.createElement("section", { className: "mx_EmojiPicker", "data-testid": "mx_EmojiPicker", onKeyDown: onKeyDownHandler, "aria-label": (0, _languageHandler._t)("a11y|emoji_picker") }, /*#__PURE__*/_react.default.createElement(_Header.default, { categories: this.categories, onAnchorClick: this.scrollToCategory }), /*#__PURE__*/_react.default.createElement(_Search.default, { query: this.state.filter, onChange: this.onChangeFilter, onEnter: this.onEnterFilter, onKeyDown: onKeyDownHandler }), /*#__PURE__*/_react.default.createElement(_AutoHideScrollbar.default, { id: "mx_EmojiPicker_body", className: "mx_EmojiPicker_body", ref: this.scrollRef, onScroll: this.onScroll }, this.categories.map(category => { const emojis = this.memoizedDataByCategory[category.id]; const categoryElement = /*#__PURE__*/_react.default.createElement(_Category.default, { key: category.id, id: category.id, name: category.name, heightBefore: heightBefore, viewportHeight: this.state.viewportHeight, scrollTop: this.state.scrollTop, emojis: emojis, onClick: this.onClickEmoji, onMouseEnter: this.onHoverEmoji, onMouseLeave: this.onHoverEmojiEnd, isEmojiDisabled: this.props.isEmojiDisabled, selectedEmojis: this.props.selectedEmojis }); const height = EmojiPicker.categoryHeightForEmojiCount(emojis.length); heightBefore += height; return categoryElement; })), this.state.previewEmoji ? /*#__PURE__*/_react.default.createElement(_Preview.default, { emoji: this.state.previewEmoji }) : /*#__PURE__*/_react.default.createElement(_QuickReactions.default, { onClick: this.onClickEmoji, selectedEmojis: this.props.selectedEmojis })); }); } } var _default = exports.default = EmojiPicker; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_emojibaseBindings","_languageHandler","recent","_interopRequireWildcard","_AutoHideScrollbar","_Header","_Search","_Preview","_QuickReactions","_Category","_arrays","_RovingTabIndex","_Keyboard","_numbers","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","ownKeys","keys","getOwnPropertySymbols","o","filter","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty2","getOwnPropertyDescriptors","defineProperties","CATEGORY_HEADER_HEIGHT","exports","EMOJI_HEIGHT","EMOJIS_PER_ROW","ZERO_WIDTH_JOINER","EmojiPicker","React","Component","constructor","props","createRef","body","scrollRef","current","containerRef","setState","scrollTop","viewportHeight","clientHeight","updateVisibility","ev","state","dispatch","activeRef","Key","ARROW_DOWN","ARROW_RIGHT","ARROW_LEFT","ARROW_UP","includes","key","keyboardNavigation","rect","getBoundingClientRect","cat","categories","elem","querySelector","id","visible","ref","classList","remove","elemRect","y","yEnd","height","add","setAttribute","category","scrollIntoView","lcFilter","toLowerCase","trim","emojis","memoizedDataByCategory","recentlyUsed","DATA_BY_CATEGORY","emoji","emojiMatchesFilter","sort","b","indexA","shortcodes","indexOf","indexB","enabled","disabled","window","setTimeout","split","label","Array","isArray","emoticon","some","x","unicode","btn","click","onFinished","previewEmoji","undefined","onChoose","ENTER","from","Set","filterBoolean","map","getEmojiFromUnicode","name","_t","node","parent","parentElement","rowIndex","children","refIndex","refs","focusRef","newParent","newTarget","clamp","find","type","Type","SetFocus","payload","behavior","block","inline","preventDefault","stopPropagation","categoryHeightForEmojiCount","count","Math","ceil","render","createElement","RovingTabIndexProvider","onKeyDown","onKeyDownHandler","heightBefore","className","onAnchorClick","scrollToCategory","query","onChange","onChangeFilter","onEnter","onEnterFilter","onScroll","categoryElement","onClick","onClickEmoji","onMouseEnter","onHoverEmoji","onMouseLeave","onHoverEmojiEnd","isEmojiDisabled","selectedEmojis","_default"],"sources":["../../../../src/components/views/emojipicker/EmojiPicker.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2020 The Matrix.org Foundation C.I.C.\nCopyright 2019 Tulir Asokan <tulir@maunium.net>\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport React, { Dispatch } from \"react\";\nimport { DATA_BY_CATEGORY, getEmojiFromUnicode, Emoji as IEmoji } from \"@matrix-org/emojibase-bindings\";\n\nimport { _t } from \"../../../languageHandler\";\nimport * as recent from \"../../../emojipicker/recent\";\nimport AutoHideScrollbar from \"../../structures/AutoHideScrollbar\";\nimport Header from \"./Header\";\nimport Search from \"./Search\";\nimport Preview from \"./Preview\";\nimport QuickReactions from \"./QuickReactions\";\nimport Category, { CategoryKey, ICategory } from \"./Category\";\nimport { filterBoolean } from \"../../../utils/arrays\";\nimport {\n    IAction as RovingAction,\n    IState as RovingState,\n    RovingTabIndexProvider,\n    Type,\n} from \"../../../accessibility/RovingTabIndex\";\nimport { Key } from \"../../../Keyboard\";\nimport { clamp } from \"../../../utils/numbers\";\nimport { ButtonEvent } from \"../elements/AccessibleButton\";\nimport { Ref } from \"../../../accessibility/roving/types\";\n\nexport const CATEGORY_HEADER_HEIGHT = 20;\nexport const EMOJI_HEIGHT = 35;\nexport const EMOJIS_PER_ROW = 8;\n\nconst ZERO_WIDTH_JOINER = \"\\u200D\";\n\ninterface IProps {\n    selectedEmojis?: Set<string>;\n    onChoose(unicode: string): boolean;\n    onFinished(): void;\n    isEmojiDisabled?: (unicode: string) => boolean;\n}\n\ninterface IState {\n    filter: string;\n    previewEmoji?: IEmoji;\n    scrollTop: number;\n    // initial estimation of height, dialog is hardcoded to 450px height.\n    // should be enough to never have blank rows of emojis as\n    // 3 rows of overflow are also rendered. The actual value is updated on scroll.\n    viewportHeight: number;\n}\n\nclass EmojiPicker extends React.Component<IProps, IState> {\n    private readonly recentlyUsed: IEmoji[];\n    private readonly memoizedDataByCategory: Record<CategoryKey, IEmoji[]>;\n    private readonly categories: ICategory[];\n\n    private scrollRef = React.createRef<AutoHideScrollbar<\"div\">>();\n\n    public constructor(props: IProps) {\n        super(props);\n\n        this.state = {\n            filter: \"\",\n            scrollTop: 0,\n            viewportHeight: 280,\n        };\n\n        // Convert recent emoji characters to emoji data, removing unknowns and duplicates\n        this.recentlyUsed = Array.from(new Set(filterBoolean(recent.get().map(getEmojiFromUnicode))));\n        this.memoizedDataByCategory = {\n            recent: this.recentlyUsed,\n            ...DATA_BY_CATEGORY,\n        };\n\n        this.categories = [\n            {\n                id: \"recent\",\n                name: _t(\"emoji|category_frequently_used\"),\n                enabled: this.recentlyUsed.length > 0,\n                visible: this.recentlyUsed.length > 0,\n                ref: React.createRef(),\n            },\n            {\n                id: \"people\",\n                name: _t(\"emoji|category_smileys_people\"),\n                enabled: true,\n                visible: true,\n                ref: React.createRef(),\n            },\n            {\n                id: \"nature\",\n                name: _t(\"emoji|category_animals_nature\"),\n                enabled: true,\n                visible: false,\n                ref: React.createRef(),\n            },\n            {\n                id: \"foods\",\n                name: _t(\"emoji|category_food_drink\"),\n                enabled: true,\n                visible: false,\n                ref: React.createRef(),\n            },\n            {\n                id: \"activity\",\n                name: _t(\"emoji|category_activities\"),\n                enabled: true,\n                visible: false,\n                ref: React.createRef(),\n            },\n            {\n                id: \"places\",\n                name: _t(\"emoji|category_travel_places\"),\n                enabled: true,\n                visible: false,\n                ref: React.createRef(),\n            },\n            {\n                id: \"objects\",\n                name: _t(\"emoji|category_objects\"),\n                enabled: true,\n                visible: false,\n                ref: React.createRef(),\n            },\n            {\n                id: \"symbols\",\n                name: _t(\"emoji|category_symbols\"),\n                enabled: true,\n                visible: false,\n                ref: React.createRef(),\n            },\n            {\n                id: \"flags\",\n                name: _t(\"emoji|category_flags\"),\n                enabled: true,\n                visible: false,\n                ref: React.createRef(),\n            },\n        ];\n    }\n\n    private onScroll = (): void => {\n        const body = this.scrollRef.current?.containerRef.current;\n        if (!body) return;\n        this.setState({\n            scrollTop: body.scrollTop,\n            viewportHeight: body.clientHeight,\n        });\n        this.updateVisibility();\n    };\n\n    private keyboardNavigation(ev: React.KeyboardEvent, state: RovingState, dispatch: Dispatch<RovingAction>): void {\n        const node = state.activeRef?.current;\n        const parent = node?.parentElement;\n        if (!parent || !state.activeRef) return;\n        const rowIndex = Array.from(parent.children).indexOf(node);\n        const refIndex = state.refs.indexOf(state.activeRef);\n\n        let focusRef: Ref | undefined;\n        let newParent: HTMLElement | undefined;\n        switch (ev.key) {\n            case Key.ARROW_LEFT:\n                focusRef = state.refs[refIndex - 1];\n                newParent = focusRef?.current?.parentElement ?? undefined;\n                break;\n\n            case Key.ARROW_RIGHT:\n                focusRef = state.refs[refIndex + 1];\n                newParent = focusRef?.current?.parentElement ?? undefined;\n                break;\n\n            case Key.ARROW_UP:\n            case Key.ARROW_DOWN: {\n                // For up/down we find the prev/next parent by inspecting the refs either side of our row\n                const ref =\n                    ev.key === Key.ARROW_UP\n                        ? state.refs[refIndex - rowIndex - 1]\n                        : state.refs[refIndex - rowIndex + EMOJIS_PER_ROW];\n                newParent = ref?.current?.parentElement ?? undefined;\n                const newTarget = newParent?.children[clamp(rowIndex, 0, newParent.children.length - 1)];\n                focusRef = state.refs.find((r) => r.current === newTarget);\n                break;\n            }\n        }\n\n        if (focusRef) {\n            dispatch({\n                type: Type.SetFocus,\n                payload: { ref: focusRef },\n            });\n\n            if (parent !== newParent) {\n                focusRef.current?.scrollIntoView({\n                    behavior: \"auto\",\n                    block: \"center\",\n                    inline: \"center\",\n                });\n            }\n        }\n\n        ev.preventDefault();\n        ev.stopPropagation();\n    }\n\n    private onKeyDown = (ev: React.KeyboardEvent, state: RovingState, dispatch: Dispatch<RovingAction>): void => {\n        if (\n            state.activeRef?.current &&\n            [Key.ARROW_DOWN, Key.ARROW_RIGHT, Key.ARROW_LEFT, Key.ARROW_UP].includes(ev.key)\n        ) {\n            this.keyboardNavigation(ev, state, dispatch);\n        }\n    };\n\n    private updateVisibility = (): void => {\n        const body = this.scrollRef.current?.containerRef.current;\n        if (!body) return;\n        const rect = body.getBoundingClientRect();\n        for (const cat of this.categories) {\n            const elem = body.querySelector(`[data-category-id=\"${cat.id}\"]`);\n            if (!elem) {\n                cat.visible = false;\n                cat.ref.current?.classList.remove(\"mx_EmojiPicker_anchor_visible\");\n                continue;\n            }\n            const elemRect = elem.getBoundingClientRect();\n            const y = elemRect.y - rect.y;\n            const yEnd = elemRect.y + elemRect.height - rect.y;\n            cat.visible = y < rect.height && yEnd > 0;\n            // We update this here instead of through React to avoid re-render on scroll.\n            if (!cat.ref.current) continue;\n            if (cat.visible) {\n                cat.ref.current.classList.add(\"mx_EmojiPicker_anchor_visible\");\n                cat.ref.current.setAttribute(\"aria-selected\", \"true\");\n                cat.ref.current.setAttribute(\"tabindex\", \"0\");\n            } else {\n                cat.ref.current.classList.remove(\"mx_EmojiPicker_anchor_visible\");\n                cat.ref.current.setAttribute(\"aria-selected\", \"false\");\n                cat.ref.current.setAttribute(\"tabindex\", \"-1\");\n            }\n        }\n    };\n\n    private scrollToCategory = (category: string): void => {\n        this.scrollRef.current?.containerRef.current\n            ?.querySelector(`[data-category-id=\"${category}\"]`)\n            ?.scrollIntoView();\n    };\n\n    private onChangeFilter = (filter: string): void => {\n        const lcFilter = filter.toLowerCase().trim(); // filter is case insensitive\n        for (const cat of this.categories) {\n            let emojis: IEmoji[];\n            // If the new filter string includes the old filter string, we don't have to re-filter the whole dataset.\n            if (lcFilter.includes(this.state.filter)) {\n                emojis = this.memoizedDataByCategory[cat.id];\n            } else {\n                emojis = cat.id === \"recent\" ? this.recentlyUsed : DATA_BY_CATEGORY[cat.id];\n            }\n\n            if (lcFilter !== \"\") {\n                emojis = emojis.filter((emoji) => this.emojiMatchesFilter(emoji, lcFilter));\n                // Copy the array to not clobber the original unfiltered sorting\n                emojis = [...emojis].sort((a, b) => {\n                    const indexA = a.shortcodes[0].indexOf(lcFilter);\n                    const indexB = b.shortcodes[0].indexOf(lcFilter);\n\n                    // Prioritize emojis containing the filter in its shortcode\n                    if (indexA == -1 || indexB == -1) {\n                        return indexB - indexA;\n                    }\n\n                    // If both emojis start with the filter\n                    // put the shorter emoji first\n                    if (indexA == 0 && indexB == 0) {\n                        return a.shortcodes[0].length - b.shortcodes[0].length;\n                    }\n\n                    // Prioritize emojis starting with the filter\n                    return indexA - indexB;\n                });\n            }\n\n            this.memoizedDataByCategory[cat.id] = emojis;\n            cat.enabled = emojis.length > 0;\n            // The setState below doesn't re-render the header and we already have the refs for updateVisibility, so...\n            if (cat.ref.current) {\n                cat.ref.current.disabled = !cat.enabled;\n            }\n        }\n        this.setState({ filter });\n        // Header underlines need to be updated, but updating requires knowing\n        // where the categories are, so we wait for a tick.\n        window.setTimeout(this.updateVisibility, 0);\n    };\n\n    private emojiMatchesFilter = (emoji: IEmoji, filter: string): boolean => {\n        // If the query is an emoji containing a variation then strip it to provide more useful matches\n        if (filter.includes(ZERO_WIDTH_JOINER)) {\n            filter = filter.split(ZERO_WIDTH_JOINER, 2)[0];\n        }\n        return (\n            emoji.label.toLowerCase().includes(filter) ||\n            (Array.isArray(emoji.emoticon)\n                ? emoji.emoticon.some((x) => x.includes(filter))\n                : emoji.emoticon?.includes(filter)) ||\n            emoji.shortcodes.some((x) => x.toLowerCase().includes(filter)) ||\n            emoji.unicode.split(ZERO_WIDTH_JOINER).includes(filter)\n        );\n    };\n\n    private onEnterFilter = (): void => {\n        const btn = this.scrollRef.current?.containerRef.current?.querySelector<HTMLButtonElement>(\n            '.mx_EmojiPicker_item_wrapper[tabindex=\"0\"]',\n        );\n        btn?.click();\n        this.props.onFinished();\n    };\n\n    private onHoverEmoji = (emoji: IEmoji): void => {\n        this.setState({\n            previewEmoji: emoji,\n        });\n    };\n\n    private onHoverEmojiEnd = (): void => {\n        this.setState({\n            previewEmoji: undefined,\n        });\n    };\n\n    private onClickEmoji = (ev: ButtonEvent, emoji: IEmoji): void => {\n        if (this.props.onChoose(emoji.unicode) !== false) {\n            recent.add(emoji.unicode);\n        }\n        if ((ev as React.KeyboardEvent).key === Key.ENTER) {\n            this.props.onFinished();\n        }\n    };\n\n    private static categoryHeightForEmojiCount(count: number): number {\n        if (count === 0) {\n            return 0;\n        }\n        return CATEGORY_HEADER_HEIGHT + Math.ceil(count / EMOJIS_PER_ROW) * EMOJI_HEIGHT;\n    }\n\n    public render(): React.ReactNode {\n        return (\n            <RovingTabIndexProvider onKeyDown={this.onKeyDown}>\n                {({ onKeyDownHandler }) => {\n                    let heightBefore = 0;\n                    return (\n                        <section\n                            className=\"mx_EmojiPicker\"\n                            data-testid=\"mx_EmojiPicker\"\n                            onKeyDown={onKeyDownHandler}\n                            aria-label={_t(\"a11y|emoji_picker\")}\n                        >\n                            <Header categories={this.categories} onAnchorClick={this.scrollToCategory} />\n                            <Search\n                                query={this.state.filter}\n                                onChange={this.onChangeFilter}\n                                onEnter={this.onEnterFilter}\n                                onKeyDown={onKeyDownHandler}\n                            />\n                            <AutoHideScrollbar\n                                id=\"mx_EmojiPicker_body\"\n                                className=\"mx_EmojiPicker_body\"\n                                ref={this.scrollRef}\n                                onScroll={this.onScroll}\n                            >\n                                {this.categories.map((category) => {\n                                    const emojis = this.memoizedDataByCategory[category.id];\n                                    const categoryElement = (\n                                        <Category\n                                            key={category.id}\n                                            id={category.id}\n                                            name={category.name}\n                                            heightBefore={heightBefore}\n                                            viewportHeight={this.state.viewportHeight}\n                                            scrollTop={this.state.scrollTop}\n                                            emojis={emojis}\n                                            onClick={this.onClickEmoji}\n                                            onMouseEnter={this.onHoverEmoji}\n                                            onMouseLeave={this.onHoverEmojiEnd}\n                                            isEmojiDisabled={this.props.isEmojiDisabled}\n                                            selectedEmojis={this.props.selectedEmojis}\n                                        />\n                                    );\n                                    const height = EmojiPicker.categoryHeightForEmojiCount(emojis.length);\n                                    heightBefore += height;\n                                    return categoryElement;\n                                })}\n                            </AutoHideScrollbar>\n                            {this.state.previewEmoji ? (\n                                <Preview emoji={this.state.previewEmoji} />\n                            ) : (\n                                <QuickReactions\n                                    onClick={this.onClickEmoji}\n                                    selectedEmojis={this.props.selectedEmojis}\n                                />\n                            )}\n                        </section>\n                    );\n                }}\n            </RovingTabIndexProvider>\n        );\n    }\n}\n\nexport default EmojiPicker;\n"],"mappings":"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