// export type File = IdBase & { readonly _: unique symbol }; export type IdentId = IdBase & { readonly _: unique symbol }; export type AstId = IdBase & { readonly _: unique symbol }; export type DefId = IdBase & { readonly _: unique symbol }; // export type IdBase = { rawId: number }; export type IdRaw = IdType["rawId"]; export const idRaw = (id: IdType): IdRaw => id.rawId; export const idFromRaw = ( rawId: IdRaw, ): IdType => ({ rawId } as IdType); export class Ids { private next = 0; public nextThenStep(): IdType { const rawId = this.next; this.next += 1; return idFromRaw(rawId); } } export class IdMap implements Map { private map = new Map, V>(); set(id: Id, val: V) { this.map.set(idRaw(id), val); return this; } get(id: Id): V | undefined { return this.map.get(idRaw(id)); } has(id: Id): boolean { return this.map.has(idRaw(id)); } keys(): MapIterator { return this.map.keys() .map((rawId) => idFromRaw(rawId)); } clear(): void { this.map.clear(); } delete(id: Id): boolean { return this.map.delete(idRaw(id)); } forEach( callbackfn: (value: V, key: Id, map: Map) => void, thisArg?: unknown, ): void { this.map.forEach( (value, key, _map) => callbackfn(value, idFromRaw(key), this), thisArg, ); } get size(): number { return this.map.size; } entries(): MapIterator<[Id, V]> { return this.map.entries() .map(([rawId, v]) => [idFromRaw(rawId), v]); } values(): MapIterator { return this.map.values(); } [Symbol.iterator](): MapIterator<[Id, V]> { return this.map[Symbol.iterator]() .map(([rawId, v]) => [idFromRaw(rawId), v]); } get [Symbol.toStringTag](): string { return this.map[Symbol.toStringTag]; } }