/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the MIT License. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ /** * An interface for a JavaScript object that * acts a dictionary. The keys are strings. */ export type IStringDictionary = Record; /** * An interface for a JavaScript object that * acts a dictionary. The keys are numbers. */ export type INumberDictionary = Record; const hasOwnProperty = Object.prototype.hasOwnProperty; /** * Returns an array which contains all values that reside * in the given dictionary. */ export function values(from: IStringDictionary | INumberDictionary): T[] { const result: T[] = []; for (let key in from) { if (hasOwnProperty.call(from, key)) { result.push((from as any)[key]); } } return result; } /** * Iterates over each entry in the provided dictionary. The iterator allows * to remove elements and will stop when the callback returns {{false}}. */ export function forEach(from: IStringDictionary | INumberDictionary, callback: (entry: { key: any; value: T; }, remove: () => void) => any): void { for (let key in from) { if (hasOwnProperty.call(from, key)) { const result = callback({ key: key, value: (from as any)[key] }, function () { delete (from as any)[key]; }); if (result === false) { return; } } } } /** * Groups the collection into a dictionary based on the provided * group function. */ export function groupBy(data: V[], groupFn: (element: V) => K): Record { const result: Record = Object.create(null); for (const element of data) { const key = groupFn(element); let target = result[key]; if (!target) { target = result[key] = []; } target.push(element); } return result; } export function fromMap(original: Map): IStringDictionary { const result: IStringDictionary = Object.create(null); if (original) { original.forEach((value, key) => { result[key] = value; }); } return result; } export function diffSets(before: Set, after: Set): { removed: T[], added: T[] } { const removed: T[] = []; const added: T[] = []; for (let element of before) { if (!after.has(element)) { removed.push(element); } } for (let element of after) { if (!before.has(element)) { added.push(element); } } return { removed, added }; } export function diffMaps(before: Map, after: Map): { removed: V[], added: V[] } { const removed: V[] = []; const added: V[] = []; for (let [index, value] of before) { if (!after.has(index)) { removed.push(value); } } for (let [index, value] of after) { if (!before.has(index)) { added.push(value); } } return { removed, added }; } export class SetMap { private map = new Map>(); add(key: K, value: V): void { let values = this.map.get(key); if (!values) { values = new Set(); this.map.set(key, values); } values.add(value); } delete(key: K, value: V): void { const values = this.map.get(key); if (!values) { return; } values.delete(value); if (values.size === 0) { this.map.delete(key); } } forEach(key: K, fn: (value: V) => void): void { const values = this.map.get(key); if (!values) { return; } values.forEach(fn); } }