UNPKG

matrix-react-sdk

Version:
135 lines (130 loc) 19.7 kB
"use strict"; 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 _lodash = require("lodash"); var _utils = require("matrix-js-sdk/src/utils"); 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 2018 Michael Telatynski <7t3chguy@gmail.com> Copyright 2018 New Vector Ltd Copyright 2017 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. */ /** * Simple search matcher that matches any results with the query string anywhere * in the search string. Returns matches in the order the query string appears * in the search key, earliest first, then in the order the search key appears * in the provided array of keys, then in the order the items appeared in the * source array. * * @param {Object[]} objects Initial list of objects. Equivalent to calling * setObjects() after construction * @param {Object} options Options object * @param {string[]} options.keys List of keys to use as indexes on the objects * @param {function[]} options.funcs List of functions that when called with the * object as an arg will return a string to use as an index */ class QueryMatcher { constructor(objects, options = { keys: [] }) { (0, _defineProperty2.default)(this, "_options", void 0); (0, _defineProperty2.default)(this, "_items", new Map()); this._options = options; this.setObjects(objects); // By default, we remove any non-alphanumeric characters ([^A-Za-z0-9_]) from the // query and the value being queried before matching if (this._options.shouldMatchWordsOnly === undefined) { this._options.shouldMatchWordsOnly = true; } } setObjects(objects) { this._items = new Map(); for (const object of objects) { // Need to use unsafe coerce here because the objects can have any // type for their values. We assume that those values who's keys have // been specified will be string. Also, we cannot infer all the // types of the keys of the objects at compile. const keyValues = (0, _lodash.at)(object, this._options.keys); if (this._options.funcs) { for (const f of this._options.funcs) { const v = f(object); if (Array.isArray(v)) { keyValues.push(...v); } else { keyValues.push(v); } } } for (const [index, keyValue] of Object.entries(keyValues)) { if (!keyValue) continue; // skip falsy keyValues const key = this.processQuery(keyValue); if (!this._items.has(key)) { this._items.set(key, []); } this._items.get(key).push({ keyWeight: Number(index), object }); } } } match(query, limit = -1) { query = this.processQuery(query); if (this._options.shouldMatchWordsOnly) { query = query.replace(/[^\w]/g, ""); } if (query.length === 0) { return []; } const matches = []; // Iterate through the map & check each key. // ES6 Map iteration order is defined to be insertion order, so results // here will come out in the order they were put in. for (const [key, candidates] of this._items.entries()) { let resultKey = key; if (this._options.shouldMatchWordsOnly) { resultKey = resultKey.replace(/[^\w]/g, ""); } const index = resultKey.indexOf(query); if (index !== -1) { matches.push(...candidates.map(candidate => _objectSpread({ index }, candidate))); } } // Sort matches by where the query appeared in the search key, then by // where the matched key appeared in the provided array of keys. matches.sort((a, b) => { if (a.index < b.index) { return -1; } else if (a.index === b.index) { if (a.keyWeight < b.keyWeight) { return -1; } else if (a.keyWeight === b.keyWeight) { return 0; } } return 1; }); // Now map the keys to the result objects. Also remove any duplicates. const dedupped = (0, _lodash.uniq)(matches.map(match => match.object)); const maxLength = limit === -1 ? dedupped.length : limit; return dedupped.slice(0, maxLength); } processQuery(query) { if (this._options.fuzzy !== false) { // lower case both the input and the output for consistency return (0, _utils.removeHiddenChars)(query.toLowerCase()).toLowerCase(); } return query.toLowerCase(); } } exports.default = QueryMatcher; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_lodash","require","_utils","ownKeys","e","r","t","Object","keys","getOwnPropertySymbols","o","filter","getOwnPropertyDescriptor","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty2","default","getOwnPropertyDescriptors","defineProperties","defineProperty","QueryMatcher","constructor","objects","options","Map","_options","setObjects","shouldMatchWordsOnly","undefined","_items","object","keyValues","at","funcs","f","v","Array","isArray","index","keyValue","entries","key","processQuery","has","set","get","keyWeight","Number","match","query","limit","replace","matches","candidates","resultKey","indexOf","map","candidate","sort","a","b","dedupped","uniq","maxLength","slice","fuzzy","removeHiddenChars","toLowerCase","exports"],"sources":["../../src/autocomplete/QueryMatcher.ts"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2018 Michael Telatynski <7t3chguy@gmail.com>\nCopyright 2018 New Vector Ltd\nCopyright 2017 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 { at, uniq } from \"lodash\";\nimport { removeHiddenChars } from \"matrix-js-sdk/src/utils\";\n\nimport { TimelineRenderingType } from \"../contexts/RoomContext\";\nimport { Leaves } from \"../@types/common\";\n\ninterface IOptions<T extends {}> {\n    keys: Array<Leaves<T>>;\n    funcs?: Array<(o: T) => string | string[]>;\n    shouldMatchWordsOnly?: boolean;\n    // whether to apply unhomoglyph and strip diacritics to fuzz up the search. Defaults to true\n    fuzzy?: boolean;\n    context?: TimelineRenderingType;\n}\n\n/**\n * Simple search matcher that matches any results with the query string anywhere\n * in the search string. Returns matches in the order the query string appears\n * in the search key, earliest first, then in the order the search key appears\n * in the provided array of keys, then in the order the items appeared in the\n * source array.\n *\n * @param {Object[]} objects Initial list of objects. Equivalent to calling\n *     setObjects() after construction\n * @param {Object} options Options object\n * @param {string[]} options.keys List of keys to use as indexes on the objects\n * @param {function[]} options.funcs List of functions that when called with the\n *     object as an arg will return a string to use as an index\n */\nexport default class QueryMatcher<T extends {}> {\n    private _options: IOptions<T>;\n    private _items = new Map<string, { object: T; keyWeight: number }[]>();\n\n    public constructor(objects: T[], options: IOptions<T> = { keys: [] }) {\n        this._options = options;\n\n        this.setObjects(objects);\n\n        // By default, we remove any non-alphanumeric characters ([^A-Za-z0-9_]) from the\n        // query and the value being queried before matching\n        if (this._options.shouldMatchWordsOnly === undefined) {\n            this._options.shouldMatchWordsOnly = true;\n        }\n    }\n\n    public setObjects(objects: T[]): void {\n        this._items = new Map();\n\n        for (const object of objects) {\n            // Need to use unsafe coerce here because the objects can have any\n            // type for their values. We assume that those values who's keys have\n            // been specified will be string. Also, we cannot infer all the\n            // types of the keys of the objects at compile.\n            const keyValues = at<string>(<any>object, this._options.keys);\n\n            if (this._options.funcs) {\n                for (const f of this._options.funcs) {\n                    const v = f(object);\n                    if (Array.isArray(v)) {\n                        keyValues.push(...v);\n                    } else {\n                        keyValues.push(v);\n                    }\n                }\n            }\n\n            for (const [index, keyValue] of Object.entries(keyValues)) {\n                if (!keyValue) continue; // skip falsy keyValues\n                const key = this.processQuery(keyValue);\n                if (!this._items.has(key)) {\n                    this._items.set(key, []);\n                }\n                this._items.get(key)!.push({\n                    keyWeight: Number(index),\n                    object,\n                });\n            }\n        }\n    }\n\n    public match(query: string, limit = -1): T[] {\n        query = this.processQuery(query);\n        if (this._options.shouldMatchWordsOnly) {\n            query = query.replace(/[^\\w]/g, \"\");\n        }\n        if (query.length === 0) {\n            return [];\n        }\n        const matches: {\n            index: number;\n            object: T;\n            keyWeight: number;\n        }[] = [];\n        // Iterate through the map & check each key.\n        // ES6 Map iteration order is defined to be insertion order, so results\n        // here will come out in the order they were put in.\n        for (const [key, candidates] of this._items.entries()) {\n            let resultKey = key;\n            if (this._options.shouldMatchWordsOnly) {\n                resultKey = resultKey.replace(/[^\\w]/g, \"\");\n            }\n            const index = resultKey.indexOf(query);\n            if (index !== -1) {\n                matches.push(...candidates.map((candidate) => ({ index, ...candidate })));\n            }\n        }\n\n        // Sort matches by where the query appeared in the search key, then by\n        // where the matched key appeared in the provided array of keys.\n        matches.sort((a, b) => {\n            if (a.index < b.index) {\n                return -1;\n            } else if (a.index === b.index) {\n                if (a.keyWeight < b.keyWeight) {\n                    return -1;\n                } else if (a.keyWeight === b.keyWeight) {\n                    return 0;\n                }\n            }\n\n            return 1;\n        });\n\n        // Now map the keys to the result objects. Also remove any duplicates.\n        const dedupped = uniq(matches.map((match) => match.object));\n        const maxLength = limit === -1 ? dedupped.length : limit;\n\n        return dedupped.slice(0, maxLength);\n    }\n\n    private processQuery(query: string): string {\n        if (this._options.fuzzy !== false) {\n            // lower case both the input and the output for consistency\n            return removeHiddenChars(query.toLowerCase()).toLowerCase();\n        }\n        return query.toLowerCase();\n    }\n}\n"],"mappings":"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","ignoreList":[]}