UNPKG

@theia/core

Version:

Theia is a cloud & desktop IDE framework implemented in TypeScript.

122 lines 4.37 kB
"use strict"; // ***************************************************************************** // Copyright (C) 2023 STMicroelectronics and others. // // This program and the accompanying materials are made available under the // terms of the Eclipse Public License v. 2.0 which is available at // http://www.eclipse.org/legal/epl-2.0. // // This Source Code may also be made available under the following Secondary // Licenses when the conditions for such availability set forth in the Eclipse // Public License v. 2.0 are satisfied: GNU General Public License, version 2 // with the GNU Classpath Exception which is available at // https://www.gnu.org/software/classpath/license.html. // // SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0 // ***************************************************************************** Object.defineProperty(exports, "__esModule", { value: true }); exports.MultiKeyMap = void 0; /** * A convenience class for managing a "map of maps" of arbitrary depth */ class MultiKeyMap { constructor(keyLength) { this.keyLength = keyLength; this.rootMap = new Map(); } static create(keyLength, data) { const result = new MultiKeyMap(keyLength); for (const entry of data) { result.set(entry[0], entry[1]); } return result; } set(key, value) { if (this.keyLength !== key.length) { throw new Error(`inappropriate key length: ${key.length}, should be ${this.keyLength}`); } let map = this.rootMap; for (let i = 0; i < this.keyLength - 1; i++) { let existing = map.get(key[i]); if (!existing) { existing = new Map(); map.set(key[i], existing); } map = existing; } const oldValue = map.get(key[this.keyLength - 1]); map.set(key[this.keyLength - 1], value); return oldValue; } get(key) { if (this.keyLength !== key.length) { throw new Error(`inappropriate key length: ${key.length}, should be ${this.keyLength}`); } let map = this.rootMap; for (let i = 0; i < this.keyLength - 1; i++) { map = map.get(key[i]); if (!map) { return undefined; } } return map.get(key[this.keyLength - 1]); } /** * Checks whether the given key is present in the map * @param key the key to test. It can have a length < the key length * @returns whether the key exists */ has(key) { if (this.keyLength < key.length) { throw new Error(`inappropriate key length: ${key.length}, should <= ${this.keyLength}`); } let map = this.rootMap; for (let i = 0; i < key.length - 1; i++) { map = map.get(key[i]); if (!map) { return false; } } return map.has(key[key.length - 1]); } /** * Deletes the value with the given key from the map * @param key the key to remove. It can have a length < the key length * @returns whether the key was present in the map */ delete(key) { if (this.keyLength < key.length) { throw new Error(`inappropriate key length: ${key.length}, should <= ${this.keyLength}`); } let map = this.rootMap; for (let i = 0; i < this.keyLength - 1; i++) { map = map.get(key[i]); if (!map) { return false; } } return map.delete(key[key.length - 1]); } /** * Iterates over all entries in the map. The ordering semantics are like iterating over a map of maps. * @param handler Handler for each entry */ forEach(handler) { this.doForeach(handler, this.rootMap, []); } // eslint-disable-next-line @typescript-eslint/no-explicit-any doForeach(handler, currentMap, keys) { if (keys.length === this.keyLength - 1) { currentMap.forEach((v, k) => { handler(v, [...keys, k]); }); } else { currentMap.forEach((v, k) => { this.doForeach(handler, v, [...keys, k]); }); } } } exports.MultiKeyMap = MultiKeyMap; //# sourceMappingURL=collections.js.map