| 
									
										
										
										
											2020-07-28 20:06:04 +08:00
										 |  |  | /* | 
					
						
							|  |  |  | 	MIT License http://www.opensource.org/licenses/mit-license.php
 | 
					
						
							|  |  |  | 	Author Tobias Koppers @sokra | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | "use strict"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @template {any[]} T | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class TupleSet { | 
					
						
							| 
									
										
										
										
											2020-07-29 04:05:13 +08:00
										 |  |  | 	constructor(init) { | 
					
						
							| 
									
										
										
										
											2020-07-28 20:06:04 +08:00
										 |  |  | 		this._map = new Map(); | 
					
						
							|  |  |  | 		this.size = 0; | 
					
						
							| 
									
										
										
										
											2020-07-29 04:05:13 +08:00
										 |  |  | 		if (init) { | 
					
						
							|  |  |  | 			for (const tuple of init) { | 
					
						
							|  |  |  | 				this.add(...tuple); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2020-07-28 20:06:04 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param  {T} args tuple | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	add(...args) { | 
					
						
							|  |  |  | 		let map = this._map; | 
					
						
							|  |  |  | 		for (let i = 0; i < args.length - 2; i++) { | 
					
						
							|  |  |  | 			const arg = args[i]; | 
					
						
							|  |  |  | 			const innerMap = map.get(arg); | 
					
						
							|  |  |  | 			if (innerMap === undefined) { | 
					
						
							|  |  |  | 				map.set(arg, (map = new Map())); | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				map = innerMap; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const beforeLast = args[args.length - 2]; | 
					
						
							|  |  |  | 		let set = map.get(beforeLast); | 
					
						
							|  |  |  | 		if (set === undefined) { | 
					
						
							|  |  |  | 			map.set(beforeLast, (set = new Set())); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const last = args[args.length - 1]; | 
					
						
							|  |  |  | 		this.size -= set.size; | 
					
						
							|  |  |  | 		set.add(last); | 
					
						
							|  |  |  | 		this.size += set.size; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param  {T} args tuple | 
					
						
							|  |  |  | 	 * @returns {boolean} true, if the tuple is in the Set | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	has(...args) { | 
					
						
							|  |  |  | 		let map = this._map; | 
					
						
							|  |  |  | 		for (let i = 0; i < args.length - 2; i++) { | 
					
						
							|  |  |  | 			const arg = args[i]; | 
					
						
							|  |  |  | 			map = map.get(arg); | 
					
						
							|  |  |  | 			if (map === undefined) { | 
					
						
							|  |  |  | 				return false; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const beforeLast = args[args.length - 2]; | 
					
						
							|  |  |  | 		let set = map.get(beforeLast); | 
					
						
							|  |  |  | 		if (set === undefined) { | 
					
						
							|  |  |  | 			return false; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const last = args[args.length - 1]; | 
					
						
							|  |  |  | 		return set.has(last); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-29 04:05:13 +08:00
										 |  |  | 	/** | 
					
						
							|  |  |  | 	 * @param {T} args tuple | 
					
						
							|  |  |  | 	 * @returns {void} | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	delete(...args) { | 
					
						
							|  |  |  | 		let map = this._map; | 
					
						
							|  |  |  | 		for (let i = 0; i < args.length - 2; i++) { | 
					
						
							|  |  |  | 			const arg = args[i]; | 
					
						
							|  |  |  | 			map = map.get(arg); | 
					
						
							|  |  |  | 			if (map === undefined) { | 
					
						
							|  |  |  | 				return; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const beforeLast = args[args.length - 2]; | 
					
						
							|  |  |  | 		let set = map.get(beforeLast); | 
					
						
							|  |  |  | 		if (set === undefined) { | 
					
						
							|  |  |  | 			return; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const last = args[args.length - 1]; | 
					
						
							|  |  |  | 		this.size -= set.size; | 
					
						
							|  |  |  | 		set.delete(last); | 
					
						
							|  |  |  | 		this.size += set.size; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/** | 
					
						
							|  |  |  | 	 * @returns {Iterator<T>} iterator | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2020-07-28 20:06:04 +08:00
										 |  |  | 	[Symbol.iterator]() { | 
					
						
							|  |  |  | 		const iteratorStack = []; | 
					
						
							|  |  |  | 		const tuple = []; | 
					
						
							|  |  |  | 		let currentSetIterator = undefined; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		const next = it => { | 
					
						
							|  |  |  | 			const result = it.next(); | 
					
						
							|  |  |  | 			if (result.done) { | 
					
						
							|  |  |  | 				if (iteratorStack.length === 0) return false; | 
					
						
							|  |  |  | 				tuple.pop(); | 
					
						
							|  |  |  | 				return next(iteratorStack.pop()); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			const [key, value] = result.value; | 
					
						
							|  |  |  | 			iteratorStack.push(it); | 
					
						
							|  |  |  | 			tuple.push(key); | 
					
						
							|  |  |  | 			if (value instanceof Set) { | 
					
						
							|  |  |  | 				currentSetIterator = value[Symbol.iterator](); | 
					
						
							|  |  |  | 				return true; | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				return next(value[Symbol.iterator]()); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		next(this._map[Symbol.iterator]()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return { | 
					
						
							|  |  |  | 			next() { | 
					
						
							|  |  |  | 				while (currentSetIterator) { | 
					
						
							|  |  |  | 					const result = currentSetIterator.next(); | 
					
						
							|  |  |  | 					if (result.done) { | 
					
						
							|  |  |  | 						tuple.pop(); | 
					
						
							|  |  |  | 						if (!next(iteratorStack.pop())) { | 
					
						
							|  |  |  | 							currentSetIterator = undefined; | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} else { | 
					
						
							|  |  |  | 						return { | 
					
						
							|  |  |  | 							done: false, | 
					
						
							| 
									
										
										
										
											2020-07-29 04:05:13 +08:00
										 |  |  | 							value: /** @type {T} */ (tuple.concat(result.value)) | 
					
						
							| 
									
										
										
										
											2020-07-28 20:06:04 +08:00
										 |  |  | 						}; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				return { done: true, value: undefined }; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports = TupleSet; |