matrix-react-sdk
Version:
SDK for matrix.org using React
282 lines (245 loc) • 35.4 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = exports.EMOJIS_PER_ROW = exports.EMOJI_HEIGHT = exports.CATEGORY_HEADER_HEIGHT = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireDefault(require("react"));
var _languageHandler = require("../../../languageHandler");
var recent = _interopRequireWildcard(require("../../../emojipicker/recent"));
var _emoji = require("../../../emoji");
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 _replaceableComponent = require("../../../utils/replaceableComponent");
var _dec, _class, _temp;
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { (0, _defineProperty2.default)(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
const CATEGORY_HEADER_HEIGHT = 22;
exports.CATEGORY_HEADER_HEIGHT = CATEGORY_HEADER_HEIGHT;
const EMOJI_HEIGHT = 37;
exports.EMOJI_HEIGHT = EMOJI_HEIGHT;
const EMOJIS_PER_ROW = 8;
exports.EMOJIS_PER_ROW = EMOJIS_PER_ROW;
let EmojiPicker = (_dec = (0, _replaceableComponent.replaceableComponent)("views.emojipicker.EmojiPicker"), _dec(_class = (_temp = class EmojiPicker extends _react.default.Component
/*:: <IProps, IState>*/
{
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, "bodyRef", /*#__PURE__*/_react.default.createRef());
(0, _defineProperty2.default)(this, "onScroll", () => {
const body = this.bodyRef.current;
this.setState({
scrollTop: body.scrollTop,
viewportHeight: body.clientHeight
});
this.updateVisibility();
});
(0, _defineProperty2.default)(this, "updateVisibility", () => {
const body = this.bodyRef.current;
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.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
/*: string*/
) => {
this.bodyRef.current.querySelector(`[data-category-id="${category}"]`).scrollIntoView();
});
(0, _defineProperty2.default)(this, "onChangeFilter", (filter
/*: string*/
) => {
filter = filter.toLowerCase(); // filter is case insensitive stored lower-case
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 (filter.includes(this.state.filter)) {
emojis = this.memoizedDataByCategory[cat.id];
} else {
emojis = cat.id === "recent" ? this.recentlyUsed : _emoji.DATA_BY_CATEGORY[cat.id];
}
emojis = emojis.filter(emoji => emoji.filterString.includes(filter));
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...
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.
setTimeout(this.updateVisibility, 0);
});
(0, _defineProperty2.default)(this, "onEnterFilter", () => {
const btn = this.bodyRef.current.querySelector(".mx_EmojiPicker_item");
if (btn) {
btn.click();
}
});
(0, _defineProperty2.default)(this, "onHoverEmoji", (emoji
/*: IEmoji*/
) => {
this.setState({
previewEmoji: emoji
});
});
(0, _defineProperty2.default)(this, "onHoverEmojiEnd", (emoji
/*: IEmoji*/
) => {
this.setState({
previewEmoji: null
});
});
(0, _defineProperty2.default)(this, "onClickEmoji", (emoji
/*: IEmoji*/
) => {
if (this.props.onChoose(emoji.unicode) !== false) {
recent.add(emoji.unicode);
}
});
this.state = {
filter: "",
previewEmoji: null,
scrollTop: 0,
viewportHeight: 280
}; // Convert recent emoji characters to emoji data, removing unknowns and duplicates
this.recentlyUsed = Array.from(new Set(recent.get().map(_emoji.getEmojiFromUnicode).filter(Boolean)));
this.memoizedDataByCategory = _objectSpread({
recent: this.recentlyUsed
}, _emoji.DATA_BY_CATEGORY);
this.categories = [{
id: "recent",
name: (0, _languageHandler._t)("Frequently Used"),
enabled: this.recentlyUsed.length > 0,
visible: this.recentlyUsed.length > 0,
ref: /*#__PURE__*/_react.default.createRef()
}, {
id: "people",
name: (0, _languageHandler._t)("Smileys & People"),
enabled: true,
visible: true,
ref: /*#__PURE__*/_react.default.createRef()
}, {
id: "nature",
name: (0, _languageHandler._t)("Animals & Nature"),
enabled: true,
visible: false,
ref: /*#__PURE__*/_react.default.createRef()
}, {
id: "foods",
name: (0, _languageHandler._t)("Food & Drink"),
enabled: true,
visible: false,
ref: /*#__PURE__*/_react.default.createRef()
}, {
id: "activity",
name: (0, _languageHandler._t)("Activities"),
enabled: true,
visible: false,
ref: /*#__PURE__*/_react.default.createRef()
}, {
id: "places",
name: (0, _languageHandler._t)("Travel & Places"),
enabled: true,
visible: false,
ref: /*#__PURE__*/_react.default.createRef()
}, {
id: "objects",
name: (0, _languageHandler._t)("Objects"),
enabled: true,
visible: false,
ref: /*#__PURE__*/_react.default.createRef()
}, {
id: "symbols",
name: (0, _languageHandler._t)("Symbols"),
enabled: true,
visible: false,
ref: /*#__PURE__*/_react.default.createRef()
}, {
id: "flags",
name: (0, _languageHandler._t)("Flags"),
enabled: true,
visible: false,
ref: /*#__PURE__*/_react.default.createRef()
}];
}
static categoryHeightForEmojiCount(count
/*: number*/
) {
if (count === 0) {
return 0;
}
return CATEGORY_HEADER_HEIGHT + Math.ceil(count / EMOJIS_PER_ROW) * EMOJI_HEIGHT;
}
render() {
let heightBefore = 0;
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_EmojiPicker"
}, /*#__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
}), /*#__PURE__*/_react.default.createElement(_AutoHideScrollbar.default, {
className: "mx_EmojiPicker_body",
wrappedRef: ref => {
// @ts-ignore - AutoHideScrollbar should accept a RefObject or fall back to its own instead
this.bodyRef.current = ref;
},
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,
selectedEmojis: this.props.selectedEmojis
});
const height = EmojiPicker.categoryHeightForEmojiCount(emojis.length);
heightBefore += height;
return categoryElement;
})), this.state.previewEmoji || !this.props.showQuickReactions ? /*#__PURE__*/_react.default.createElement(_Preview.default, {
emoji: this.state.previewEmoji
}) : /*#__PURE__*/_react.default.createElement(_QuickReactions.default, {
onClick: this.onClickEmoji,
selectedEmojis: this.props.selectedEmojis
}));
}
}, _temp)) || _class);
var _default = EmojiPicker;
exports.default = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/emojipicker/EmojiPicker.tsx"],"names":["CATEGORY_HEADER_HEIGHT","EMOJI_HEIGHT","EMOJIS_PER_ROW","EmojiPicker","React","Component","constructor","props","createRef","body","bodyRef","current","setState","scrollTop","viewportHeight","clientHeight","updateVisibility","rect","getBoundingClientRect","cat","categories","elem","querySelector","id","visible","ref","classList","remove","elemRect","y","yEnd","height","add","setAttribute","category","scrollIntoView","filter","toLowerCase","emojis","includes","state","memoizedDataByCategory","recentlyUsed","DATA_BY_CATEGORY","emoji","filterString","enabled","length","disabled","setTimeout","btn","click","previewEmoji","onChoose","unicode","recent","Array","from","Set","get","map","getEmojiFromUnicode","Boolean","name","categoryHeightForEmojiCount","count","Math","ceil","render","heightBefore","scrollToCategory","onChangeFilter","onEnterFilter","onScroll","categoryElement","onClickEmoji","onHoverEmoji","onHoverEmojiEnd","selectedEmojis","showQuickReactions"],"mappings":"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","sourcesContent":["/*\nCopyright 2019 Tulir Asokan <tulir@maunium.net>\nCopyright 2020 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\nimport React from 'react';\n\nimport { _t } from '../../../languageHandler';\nimport * as recent from '../../../emojipicker/recent';\nimport {DATA_BY_CATEGORY, getEmojiFromUnicode, IEmoji} from \"../../../emoji\";\nimport AutoHideScrollbar from \"../../structures/AutoHideScrollbar\";\nimport Header from \"./Header\";\nimport Search from \"./Search\";\nimport Preview from \"./Preview\";\nimport QuickReactions from \"./QuickReactions\";\nimport Category, {ICategory, CategoryKey} from \"./Category\";\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\n\nexport const CATEGORY_HEADER_HEIGHT = 22;\nexport const EMOJI_HEIGHT = 37;\nexport const EMOJIS_PER_ROW = 8;\n\ninterface IProps {\n    selectedEmojis: Set<string>;\n    showQuickReactions?: boolean;\n    onChoose(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\n@replaceableComponent(\"views.emojipicker.EmojiPicker\")\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 bodyRef = React.createRef<HTMLDivElement>();\n\n    constructor(props) {\n        super(props);\n\n        this.state = {\n            filter: \"\",\n            previewEmoji: null,\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(recent.get().map(getEmojiFromUnicode).filter(Boolean)));\n        this.memoizedDataByCategory = {\n            recent: this.recentlyUsed,\n            ...DATA_BY_CATEGORY,\n        };\n\n        this.categories = [{\n            id: \"recent\",\n            name: _t(\"Frequently Used\"),\n            enabled: this.recentlyUsed.length > 0,\n            visible: this.recentlyUsed.length > 0,\n            ref: React.createRef(),\n        }, {\n            id: \"people\",\n            name: _t(\"Smileys & People\"),\n            enabled: true,\n            visible: true,\n            ref: React.createRef(),\n        }, {\n            id: \"nature\",\n            name: _t(\"Animals & Nature\"),\n            enabled: true,\n            visible: false,\n            ref: React.createRef(),\n        }, {\n            id: \"foods\",\n            name: _t(\"Food & Drink\"),\n            enabled: true,\n            visible: false,\n            ref: React.createRef(),\n        }, {\n            id: \"activity\",\n            name: _t(\"Activities\"),\n            enabled: true,\n            visible: false,\n            ref: React.createRef(),\n        }, {\n            id: \"places\",\n            name: _t(\"Travel & Places\"),\n            enabled: true,\n            visible: false,\n            ref: React.createRef(),\n        }, {\n            id: \"objects\",\n            name: _t(\"Objects\"),\n            enabled: true,\n            visible: false,\n            ref: React.createRef(),\n        }, {\n            id: \"symbols\",\n            name: _t(\"Symbols\"),\n            enabled: true,\n            visible: false,\n            ref: React.createRef(),\n        }, {\n            id: \"flags\",\n            name: _t(\"Flags\"),\n            enabled: true,\n            visible: false,\n            ref: React.createRef(),\n        }];\n    }\n\n    private onScroll = () => {\n        const body = this.bodyRef.current;\n        this.setState({\n            scrollTop: body.scrollTop,\n            viewportHeight: body.clientHeight,\n        });\n        this.updateVisibility();\n    };\n\n    private updateVisibility = () => {\n        const body = this.bodyRef.current;\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.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) => {\n        this.bodyRef.current.querySelector(`[data-category-id=\"${category}\"]`).scrollIntoView();\n    };\n\n    private onChangeFilter = (filter: string) => {\n        filter = filter.toLowerCase(); // filter is case insensitive stored lower-case\n        for (const cat of this.categories) {\n            let emojis;\n            // If the new filter string includes the old filter string, we don't have to re-filter the whole dataset.\n            if (filter.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            emojis = emojis.filter(emoji => emoji.filterString.includes(filter));\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            cat.ref.current.disabled = !cat.enabled;\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        setTimeout(this.updateVisibility, 0);\n    };\n\n    private onEnterFilter = () => {\n        const btn = this.bodyRef.current.querySelector<HTMLButtonElement>(\".mx_EmojiPicker_item\");\n        if (btn) {\n            btn.click();\n        }\n    };\n\n    private onHoverEmoji = (emoji: IEmoji) => {\n        this.setState({\n            previewEmoji: emoji,\n        });\n    };\n\n    private onHoverEmojiEnd = (emoji: IEmoji) => {\n        this.setState({\n            previewEmoji: null,\n        });\n    };\n\n    private onClickEmoji = (emoji: IEmoji) => {\n        if (this.props.onChoose(emoji.unicode) !== false) {\n            recent.add(emoji.unicode);\n        }\n    };\n\n    private static categoryHeightForEmojiCount(count: 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    render() {\n        let heightBefore = 0;\n        return (\n            <div className=\"mx_EmojiPicker\">\n                <Header categories={this.categories} onAnchorClick={this.scrollToCategory} />\n                <Search query={this.state.filter} onChange={this.onChangeFilter} onEnter={this.onEnterFilter} />\n                <AutoHideScrollbar\n                    className=\"mx_EmojiPicker_body\"\n                    wrappedRef={ref => {\n                        // @ts-ignore - AutoHideScrollbar should accept a RefObject or fall back to its own instead\n                        this.bodyRef.current = ref\n                    }}\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                                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 || !this.props.showQuickReactions\n                    ? <Preview emoji={this.state.previewEmoji} />\n                    : <QuickReactions onClick={this.onClickEmoji} selectedEmojis={this.props.selectedEmojis} /> }\n            </div>\n        );\n    }\n}\n\nexport default EmojiPicker;\n"]}