70 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			70 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
|  | /** | ||
|  |  * SetArray acts like a `Set` (allowing only one occurrence of a string `key`), but provides the | ||
|  |  * index of the `key` in the backing array. | ||
|  |  * | ||
|  |  * This is designed to allow synchronizing a second array with the contents of the backing array, | ||
|  |  * like how in a sourcemap `sourcesContent[i]` is the source content associated with `source[i]`, | ||
|  |  * and there are never duplicates. | ||
|  |  */ | ||
|  | class SetArray { | ||
|  |     constructor() { | ||
|  |         this._indexes = { __proto__: null }; | ||
|  |         this.array = []; | ||
|  |     } | ||
|  | } | ||
|  | /** | ||
|  |  * Typescript doesn't allow friend access to private fields, so this just casts the set into a type | ||
|  |  * with public access modifiers. | ||
|  |  */ | ||
|  | function cast(set) { | ||
|  |     return set; | ||
|  | } | ||
|  | /** | ||
|  |  * Gets the index associated with `key` in the backing array, if it is already present. | ||
|  |  */ | ||
|  | function get(setarr, key) { | ||
|  |     return cast(setarr)._indexes[key]; | ||
|  | } | ||
|  | /** | ||
|  |  * Puts `key` into the backing array, if it is not already present. Returns | ||
|  |  * the index of the `key` in the backing array. | ||
|  |  */ | ||
|  | function put(setarr, key) { | ||
|  |     // The key may or may not be present. If it is present, it's a number.
 | ||
|  |     const index = get(setarr, key); | ||
|  |     if (index !== undefined) | ||
|  |         return index; | ||
|  |     const { array, _indexes: indexes } = cast(setarr); | ||
|  |     const length = array.push(key); | ||
|  |     return (indexes[key] = length - 1); | ||
|  | } | ||
|  | /** | ||
|  |  * Pops the last added item out of the SetArray. | ||
|  |  */ | ||
|  | function pop(setarr) { | ||
|  |     const { array, _indexes: indexes } = cast(setarr); | ||
|  |     if (array.length === 0) | ||
|  |         return; | ||
|  |     const last = array.pop(); | ||
|  |     indexes[last] = undefined; | ||
|  | } | ||
|  | /** | ||
|  |  * Removes the key, if it exists in the set. | ||
|  |  */ | ||
|  | function remove(setarr, key) { | ||
|  |     const index = get(setarr, key); | ||
|  |     if (index === undefined) | ||
|  |         return; | ||
|  |     const { array, _indexes: indexes } = cast(setarr); | ||
|  |     for (let i = index + 1; i < array.length; i++) { | ||
|  |         const k = array[i]; | ||
|  |         array[i - 1] = k; | ||
|  |         indexes[k]--; | ||
|  |     } | ||
|  |     indexes[key] = undefined; | ||
|  |     array.pop(); | ||
|  | } | ||
|  | 
 | ||
|  | export { SetArray, get, pop, put, remove }; | ||
|  | //# sourceMappingURL=set-array.mjs.map
 |