matrix-react-sdk
Version:
SDK for matrix.org using React
170 lines (162 loc) • 28.1 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireDefault(require("react"));
var _lodash = require("lodash");
var _emoticon = _interopRequireDefault(require("emojibase-regex/emoticon"));
var _emojibaseBindings = require("@matrix-org/emojibase-bindings");
var _languageHandler = require("../languageHandler");
var _AutocompleteProvider = _interopRequireDefault(require("./AutocompleteProvider"));
var _QueryMatcher = _interopRequireDefault(require("./QueryMatcher"));
var _Components = require("./Components");
var _SettingsStore = _interopRequireDefault(require("../settings/SettingsStore"));
var recent = _interopRequireWildcard(require("../emojipicker/recent"));
var _arrays = require("../utils/arrays");
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; }
/*
Copyright 2024 New Vector Ltd.
Copyright 2022 Ryan Browne <code@commonlawfeature.com>
Copyright 2019 The Matrix.org Foundation C.I.C.
Copyright 2017, 2018 New Vector Ltd
Copyright 2017 Vector Creations Ltd
Copyright 2016 Aviral Dasgupta
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
const LIMIT = 20;
// Match for ascii-style ";-)" emoticons or ":wink:" shortcodes provided by emojibase
// anchored to only match from the start of parts otherwise it'll show emoji suggestions whilst typing matrix IDs
const EMOJI_REGEX = new RegExp("(" + _emoticon.default.source + "|(?:^|\\s):[+-\\w]*:?)$", "g");
const SORTED_EMOJI = _emojibaseBindings.EMOJI.sort((a, b) => {
if (a.group === b.group) {
return a.order - b.order;
}
return a.group - b.group;
}).map((emoji, index) => ({
emoji,
// Include the index so that we can preserve the original order
_orderBy: index
}));
function score(query, space) {
if (Array.isArray(space)) {
return Math.min(...space.map(s => score(query, s)));
}
const index = space.indexOf(query);
if (index === -1) {
return Infinity;
} else {
return index;
}
}
function colonsTrimmed(str) {
// Trim off leading and potentially trailing `:` to correctly match the emoji data as they exist in emojibase.
// Notes: The regex is pinned to the start and end of the string so that we can use the lazy-capturing `*?` matcher.
// It needs to be lazy so that the trailing `:` is not captured in the replacement group, if it exists.
return str.replace(/^:(.*?):?$/, "$1");
}
class EmojiProvider extends _AutocompleteProvider.default {
constructor(room, renderingType) {
super({
commandRegex: EMOJI_REGEX,
renderingType
});
(0, _defineProperty2.default)(this, "matcher", void 0);
(0, _defineProperty2.default)(this, "nameMatcher", void 0);
(0, _defineProperty2.default)(this, "recentlyUsed", void 0);
this.matcher = new _QueryMatcher.default(SORTED_EMOJI, {
keys: [],
funcs: [o => o.emoji.shortcodes.map(s => `:${s}:`)],
// For matching against ascii equivalents
shouldMatchWordsOnly: false
});
this.nameMatcher = new _QueryMatcher.default(SORTED_EMOJI, {
keys: ["emoji.label"],
// For removing punctuation
shouldMatchWordsOnly: true
});
this.recentlyUsed = Array.from(new Set((0, _arrays.filterBoolean)(recent.get().map(_emojibaseBindings.getEmojiFromUnicode))));
}
async getCompletions(query, selection, force, limit = -1) {
if (!_SettingsStore.default.getValue("MessageComposerInput.suggestEmoji")) {
return []; // don't give any suggestions if the user doesn't want them
}
let completions = [];
const {
command,
range
} = this.getCurrentCommand(query, selection);
if (command && command[0].length > 2) {
const matchedString = command[0];
completions = this.matcher.match(matchedString, limit);
// Do second match with shouldMatchWordsOnly in order to match against 'name'
completions = completions.concat(this.nameMatcher.match(matchedString));
const sorters = [];
// make sure that emoticons come first
sorters.push(c => score(matchedString, c.emoji.emoticon || ""));
// then sort by score (Infinity if matchedString not in shortcode)
sorters.push(c => score(matchedString, c.emoji.shortcodes[0]));
// then sort by max score of all shortcodes, trim off the `:`
const trimmedMatch = colonsTrimmed(matchedString);
sorters.push(c => Math.min(...c.emoji.shortcodes.map(s => score(trimmedMatch, s))));
// If the matchedString is not empty, sort by length of shortcode. Example:
// matchedString = ":bookmark"
// completions = [":bookmark:", ":bookmark_tabs:", ...]
if (matchedString.length > 1) {
sorters.push(c => c.emoji.shortcodes[0].length);
}
// Finally, sort by original ordering
sorters.push(c => c._orderBy);
completions = (0, _lodash.sortBy)((0, _lodash.uniq)(completions), sorters);
completions = completions.slice(0, LIMIT);
// Do a second sort to place emoji matching with frequently used one on top
const recentlyUsedAutocomplete = [];
this.recentlyUsed.forEach(emoji => {
if (emoji.shortcodes[0].indexOf(trimmedMatch) === 0) {
recentlyUsedAutocomplete.push({
emoji: emoji,
_orderBy: 0
});
}
});
//if there is an exact shortcode match in the frequently used emojis, it goes before everything
for (let i = 0; i < recentlyUsedAutocomplete.length; i++) {
if (recentlyUsedAutocomplete[i].emoji.shortcodes[0] === trimmedMatch) {
const exactMatchEmoji = recentlyUsedAutocomplete[i];
for (let j = i; j > 0; j--) {
recentlyUsedAutocomplete[j] = recentlyUsedAutocomplete[j - 1];
}
recentlyUsedAutocomplete[0] = exactMatchEmoji;
break;
}
}
completions = recentlyUsedAutocomplete.concat(completions);
completions = (0, _lodash.uniqBy)(completions, "emoji");
return completions.map(c => ({
completion: c.emoji.unicode,
component: /*#__PURE__*/_react.default.createElement(_Components.PillCompletion, {
title: `:${c.emoji.shortcodes[0]}:`,
"aria-label": c.emoji.unicode
}, /*#__PURE__*/_react.default.createElement("span", null, c.emoji.unicode)),
range: range
}));
}
return [];
}
getName() {
return "😃 " + (0, _languageHandler._t)("common|emoji");
}
renderCompletions(completions) {
return /*#__PURE__*/_react.default.createElement("div", {
className: "mx_Autocomplete_Completion_container_pill",
role: "presentation",
"aria-label": (0, _languageHandler._t)("composer|autocomplete|emoji_a11y")
}, completions);
}
}
exports.default = EmojiProvider;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_lodash","_emoticon","_emojibaseBindings","_languageHandler","_AutocompleteProvider","_QueryMatcher","_Components","_SettingsStore","recent","_interopRequireWildcard","_arrays","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","LIMIT","EMOJI_REGEX","RegExp","EMOTICON_REGEX","source","SORTED_EMOJI","EMOJI","sort","b","group","order","map","emoji","index","_orderBy","score","query","space","Array","isArray","Math","min","s","indexOf","Infinity","colonsTrimmed","str","replace","EmojiProvider","AutocompleteProvider","constructor","room","renderingType","commandRegex","_defineProperty2","matcher","QueryMatcher","keys","funcs","o","shortcodes","shouldMatchWordsOnly","nameMatcher","recentlyUsed","from","Set","filterBoolean","getEmojiFromUnicode","getCompletions","selection","force","limit","SettingsStore","getValue","completions","command","range","getCurrentCommand","length","matchedString","match","concat","sorters","push","c","emoticon","trimmedMatch","sortBy","uniq","slice","recentlyUsedAutocomplete","forEach","exactMatchEmoji","j","uniqBy","completion","unicode","component","createElement","PillCompletion","title","getName","_t","renderCompletions","className","role","exports"],"sources":["../../src/autocomplete/EmojiProvider.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2022 Ryan Browne <code@commonlawfeature.com>\nCopyright 2019 The Matrix.org Foundation C.I.C.\nCopyright 2017, 2018 New Vector Ltd\nCopyright 2017 Vector Creations Ltd\nCopyright 2016 Aviral Dasgupta\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 from \"react\";\nimport { uniq, sortBy, uniqBy, ListIteratee } from \"lodash\";\nimport EMOTICON_REGEX from \"emojibase-regex/emoticon\";\nimport { Room } from \"matrix-js-sdk/src/matrix\";\nimport { EMOJI, Emoji, getEmojiFromUnicode } from \"@matrix-org/emojibase-bindings\";\n\nimport { _t } from \"../languageHandler\";\nimport AutocompleteProvider from \"./AutocompleteProvider\";\nimport QueryMatcher from \"./QueryMatcher\";\nimport { PillCompletion } from \"./Components\";\nimport { ICompletion, ISelectionRange } from \"./Autocompleter\";\nimport SettingsStore from \"../settings/SettingsStore\";\nimport { TimelineRenderingType } from \"../contexts/RoomContext\";\nimport * as recent from \"../emojipicker/recent\";\nimport { filterBoolean } from \"../utils/arrays\";\n\nconst LIMIT = 20;\n\n// Match for ascii-style \";-)\" emoticons or \":wink:\" shortcodes provided by emojibase\n// anchored to only match from the start of parts otherwise it'll show emoji suggestions whilst typing matrix IDs\nconst EMOJI_REGEX = new RegExp(\"(\" + EMOTICON_REGEX.source + \"|(?:^|\\\\s):[+-\\\\w]*:?)$\", \"g\");\n\ninterface ISortedEmoji {\n    emoji: Emoji;\n    _orderBy: number;\n}\n\nconst SORTED_EMOJI: ISortedEmoji[] = EMOJI.sort((a, b) => {\n    if (a.group === b.group) {\n        return a.order! - b.order!;\n    }\n    return a.group! - b.group!;\n}).map((emoji, index) => ({\n    emoji,\n    // Include the index so that we can preserve the original order\n    _orderBy: index,\n}));\n\nfunction score(query: string, space: string[] | string): number {\n    if (Array.isArray(space)) {\n        return Math.min(...space.map((s) => score(query, s)));\n    }\n\n    const index = space.indexOf(query);\n    if (index === -1) {\n        return Infinity;\n    } else {\n        return index;\n    }\n}\n\nfunction colonsTrimmed(str: string): string {\n    // Trim off leading and potentially trailing `:` to correctly match the emoji data as they exist in emojibase.\n    // Notes: The regex is pinned to the start and end of the string so that we can use the lazy-capturing `*?` matcher.\n    // It needs to be lazy so that the trailing `:` is not captured in the replacement group, if it exists.\n    return str.replace(/^:(.*?):?$/, \"$1\");\n}\n\nexport default class EmojiProvider extends AutocompleteProvider {\n    public matcher: QueryMatcher<ISortedEmoji>;\n    public nameMatcher: QueryMatcher<ISortedEmoji>;\n    private readonly recentlyUsed: Emoji[];\n\n    public constructor(room: Room, renderingType?: TimelineRenderingType) {\n        super({ commandRegex: EMOJI_REGEX, renderingType });\n        this.matcher = new QueryMatcher<ISortedEmoji>(SORTED_EMOJI, {\n            keys: [],\n            funcs: [(o) => o.emoji.shortcodes.map((s) => `:${s}:`)],\n            // For matching against ascii equivalents\n            shouldMatchWordsOnly: false,\n        });\n        this.nameMatcher = new QueryMatcher(SORTED_EMOJI, {\n            keys: [\"emoji.label\"],\n            // For removing punctuation\n            shouldMatchWordsOnly: true,\n        });\n\n        this.recentlyUsed = Array.from(new Set(filterBoolean(recent.get().map(getEmojiFromUnicode))));\n    }\n\n    public async getCompletions(\n        query: string,\n        selection: ISelectionRange,\n        force?: boolean,\n        limit = -1,\n    ): Promise<ICompletion[]> {\n        if (!SettingsStore.getValue(\"MessageComposerInput.suggestEmoji\")) {\n            return []; // don't give any suggestions if the user doesn't want them\n        }\n\n        let completions: ISortedEmoji[] = [];\n        const { command, range } = this.getCurrentCommand(query, selection);\n\n        if (command && command[0].length > 2) {\n            const matchedString = command[0];\n            completions = this.matcher.match(matchedString, limit);\n\n            // Do second match with shouldMatchWordsOnly in order to match against 'name'\n            completions = completions.concat(this.nameMatcher.match(matchedString));\n\n            const sorters: ListIteratee<ISortedEmoji>[] = [];\n            // make sure that emoticons come first\n            sorters.push((c) => score(matchedString, c.emoji.emoticon || \"\"));\n\n            // then sort by score (Infinity if matchedString not in shortcode)\n            sorters.push((c) => score(matchedString, c.emoji.shortcodes[0]));\n            // then sort by max score of all shortcodes, trim off the `:`\n            const trimmedMatch = colonsTrimmed(matchedString);\n            sorters.push((c) => Math.min(...c.emoji.shortcodes.map((s) => score(trimmedMatch, s))));\n            // If the matchedString is not empty, sort by length of shortcode. Example:\n            //  matchedString = \":bookmark\"\n            //  completions = [\":bookmark:\", \":bookmark_tabs:\", ...]\n            if (matchedString.length > 1) {\n                sorters.push((c) => c.emoji.shortcodes[0].length);\n            }\n            // Finally, sort by original ordering\n            sorters.push((c) => c._orderBy);\n            completions = sortBy<ISortedEmoji>(uniq(completions), sorters);\n\n            completions = completions.slice(0, LIMIT);\n\n            // Do a second sort to place emoji matching with frequently used one on top\n            const recentlyUsedAutocomplete: ISortedEmoji[] = [];\n            this.recentlyUsed.forEach((emoji) => {\n                if (emoji.shortcodes[0].indexOf(trimmedMatch) === 0) {\n                    recentlyUsedAutocomplete.push({ emoji: emoji, _orderBy: 0 });\n                }\n            });\n\n            //if there is an exact shortcode match in the frequently used emojis, it goes before everything\n            for (let i = 0; i < recentlyUsedAutocomplete.length; i++) {\n                if (recentlyUsedAutocomplete[i].emoji.shortcodes[0] === trimmedMatch) {\n                    const exactMatchEmoji = recentlyUsedAutocomplete[i];\n                    for (let j = i; j > 0; j--) {\n                        recentlyUsedAutocomplete[j] = recentlyUsedAutocomplete[j - 1];\n                    }\n                    recentlyUsedAutocomplete[0] = exactMatchEmoji;\n                    break;\n                }\n            }\n\n            completions = recentlyUsedAutocomplete.concat(completions);\n            completions = uniqBy(completions, \"emoji\");\n\n            return completions.map((c) => ({\n                completion: c.emoji.unicode,\n                component: (\n                    <PillCompletion title={`:${c.emoji.shortcodes[0]}:`} aria-label={c.emoji.unicode}>\n                        <span>{c.emoji.unicode}</span>\n                    </PillCompletion>\n                ),\n                range: range!,\n            }));\n        }\n        return [];\n    }\n\n    public getName(): string {\n        return \"😃 \" + _t(\"common|emoji\");\n    }\n\n    public renderCompletions(completions: React.ReactNode[]): React.ReactNode {\n        return (\n            <div\n                className=\"mx_Autocomplete_Completion_container_pill\"\n                role=\"presentation\"\n                aria-label={_t(\"composer|autocomplete|emoji_a11y\")}\n            >\n                {completions}\n            </div>\n        );\n    }\n}\n"],"mappings":"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","ignoreList":[]}