2 MIT License http://www.opensource.org/licenses/mit-license.php
3 Author Tobias Koppers @sokra
8 const makeSerializable = require("./makeSerializable.js");
12 * @param {Set<T>} targetSet set where items should be added
13 * @param {Set<Iterable<T>>} toMerge iterables to be merged
16 const merge = (targetSet, toMerge) => {
17 for (const set of toMerge) {
18 for (const item of set) {
26 * @param {Set<Iterable<T>>} targetSet set where iterables should be added
27 * @param {Array<LazySet<T>>} toDeepMerge lazy sets to be flattened
30 const flatten = (targetSet, toDeepMerge) => {
31 for (const set of toDeepMerge) {
32 if (set._set.size > 0) targetSet.add(set._set);
34 for (const mergedSet of set._toMerge) {
35 targetSet.add(mergedSet);
37 flatten(targetSet, set._toDeepMerge);
43 * Like Set but with an addAll method to eventually add items from another iterable.
44 * Access methods make sure that all delayed operations are executed.
45 * Iteration methods deopts to normal Set performance until clear is called again (because of the chance of modifications during iteration).
50 * @param {Iterable<T>=} iterable init iterable
52 constructor(iterable) {
54 this._set = new Set(iterable);
55 /** @type {Set<Iterable<T>>} */
56 this._toMerge = new Set();
57 /** @type {Array<LazySet<T>>} */
58 this._toDeepMerge = [];
59 this._needMerge = false;
64 flatten(this._toMerge, this._toDeepMerge);
65 this._toDeepMerge.length = 0;
70 merge(this._set, this._toMerge);
71 this._toMerge.clear();
72 this._needMerge = false;
77 this._set.size === 0 &&
78 this._toMerge.size === 0 &&
79 this._toDeepMerge.length === 0
84 if (this._needMerge) this._merge();
85 return this._set.size;
89 * @param {T} item an item
90 * @returns {this} itself
98 * @param {Iterable<T> | LazySet<T>} iterable a immutable iterable or another immutable LazySet which will eventually be merged into the Set
99 * @returns {this} itself
103 const _set = this._set;
104 for (const item of iterable) {
108 if (iterable instanceof LazySet) {
109 if (iterable._isEmpty()) return this;
110 this._toDeepMerge.push(iterable);
111 this._needMerge = true;
112 if (this._toDeepMerge.length > 100000) {
116 this._toMerge.add(iterable);
117 this._needMerge = true;
119 if (this._toMerge.size > 100000) this._merge();
126 this._toMerge.clear();
127 this._toDeepMerge.length = 0;
128 this._needMerge = false;
133 * @param {T} value an item
134 * @returns {boolean} true, if the value was in the Set before
137 if (this._needMerge) this._merge();
138 return this._set.delete(value);
143 if (this._needMerge) this._merge();
144 return this._set.entries();
148 * @param {function(T, T, Set<T>): void} callbackFn function called for each entry
149 * @param {any} thisArg this argument for the callbackFn
152 forEach(callbackFn, thisArg) {
154 if (this._needMerge) this._merge();
155 this._set.forEach(callbackFn, thisArg);
159 * @param {T} item an item
160 * @returns {boolean} true, when the item is in the Set
163 if (this._needMerge) this._merge();
164 return this._set.has(item);
169 if (this._needMerge) this._merge();
170 return this._set.keys();
175 if (this._needMerge) this._merge();
176 return this._set.values();
179 [Symbol.iterator]() {
181 if (this._needMerge) this._merge();
182 return this._set[Symbol.iterator]();
185 /* istanbul ignore next */
186 get [Symbol.toStringTag]() {
190 serialize({ write }) {
191 if (this._needMerge) this._merge();
192 write(this._set.size);
193 for (const item of this._set) write(item);
196 static deserialize({ read }) {
197 const count = read();
199 for (let i = 0; i < count; i++) {
202 return new LazySet(items);
206 makeSerializable(LazySet, "webpack/lib/util/LazySet");
208 module.exports = LazySet;