@abp/utils 5.0.10 → 5.1.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/abp.resourcemapping.js +5 -5
- package/dist/abp-utils.d.ts +4 -4
- package/dist/bundles/abp-utils.umd.js +671 -671
- package/dist/bundles/abp-utils.umd.js.map +1 -1
- package/dist/bundles/abp-utils.umd.min.js.map +1 -1
- package/dist/esm2015/abp-utils.js +4 -4
- package/dist/esm2015/lib/linked-list.js +289 -289
- package/dist/esm2015/public-api.js +5 -5
- package/dist/fesm2015/abp-utils.js +290 -290
- package/dist/fesm2015/abp-utils.js.map +1 -1
- package/dist/lib/linked-list.d.ts +80 -80
- package/dist/package.json +1 -1
- package/dist/public-api.d.ts +1 -1
- package/ngcc.config.js +6 -6
- package/package.json +1 -1
|
@@ -1,289 +1,289 @@
|
|
|
1
|
-
/* tslint:disable:no-non-null-assertion */
|
|
2
|
-
import compare from 'just-compare';
|
|
3
|
-
export class ListNode {
|
|
4
|
-
constructor(value) {
|
|
5
|
-
this.value = value;
|
|
6
|
-
}
|
|
7
|
-
}
|
|
8
|
-
export class LinkedList {
|
|
9
|
-
constructor() {
|
|
10
|
-
this.size = 0;
|
|
11
|
-
}
|
|
12
|
-
get head() {
|
|
13
|
-
return this.first;
|
|
14
|
-
}
|
|
15
|
-
get tail() {
|
|
16
|
-
return this.last;
|
|
17
|
-
}
|
|
18
|
-
get length() {
|
|
19
|
-
return this.size;
|
|
20
|
-
}
|
|
21
|
-
attach(value, previousNode, nextNode) {
|
|
22
|
-
if (!previousNode)
|
|
23
|
-
return this.addHead(value);
|
|
24
|
-
if (!nextNode)
|
|
25
|
-
return this.addTail(value);
|
|
26
|
-
const node = new ListNode(value);
|
|
27
|
-
node.previous = previousNode;
|
|
28
|
-
previousNode.next = node;
|
|
29
|
-
node.next = nextNode;
|
|
30
|
-
nextNode.previous = node;
|
|
31
|
-
this.size++;
|
|
32
|
-
return node;
|
|
33
|
-
}
|
|
34
|
-
attachMany(values, previousNode, nextNode) {
|
|
35
|
-
if (!values.length)
|
|
36
|
-
return [];
|
|
37
|
-
if (!previousNode)
|
|
38
|
-
return this.addManyHead(values);
|
|
39
|
-
if (!nextNode)
|
|
40
|
-
return this.addManyTail(values);
|
|
41
|
-
const list = new LinkedList();
|
|
42
|
-
list.addManyTail(values);
|
|
43
|
-
list.first.previous = previousNode;
|
|
44
|
-
previousNode.next = list.first;
|
|
45
|
-
list.last.next = nextNode;
|
|
46
|
-
nextNode.previous = list.last;
|
|
47
|
-
this.size += values.length;
|
|
48
|
-
return list.toNodeArray();
|
|
49
|
-
}
|
|
50
|
-
detach(node) {
|
|
51
|
-
if (!node.previous)
|
|
52
|
-
return this.dropHead();
|
|
53
|
-
if (!node.next)
|
|
54
|
-
return this.dropTail();
|
|
55
|
-
node.previous.next = node.next;
|
|
56
|
-
node.next.previous = node.previous;
|
|
57
|
-
this.size--;
|
|
58
|
-
return node;
|
|
59
|
-
}
|
|
60
|
-
add(value) {
|
|
61
|
-
return {
|
|
62
|
-
after: (...params) => this.addAfter.call(this, value, ...params),
|
|
63
|
-
before: (...params) => this.addBefore.call(this, value, ...params),
|
|
64
|
-
byIndex: (position) => this.addByIndex(value, position),
|
|
65
|
-
head: () => this.addHead(value),
|
|
66
|
-
tail: () => this.addTail(value),
|
|
67
|
-
};
|
|
68
|
-
}
|
|
69
|
-
addMany(values) {
|
|
70
|
-
return {
|
|
71
|
-
after: (...params) => this.addManyAfter.call(this, values, ...params),
|
|
72
|
-
before: (...params) => this.addManyBefore.call(this, values, ...params),
|
|
73
|
-
byIndex: (position) => this.addManyByIndex(values, position),
|
|
74
|
-
head: () => this.addManyHead(values),
|
|
75
|
-
tail: () => this.addManyTail(values),
|
|
76
|
-
};
|
|
77
|
-
}
|
|
78
|
-
addAfter(value, previousValue, compareFn = compare) {
|
|
79
|
-
const previous = this.find(node => compareFn(node.value, previousValue));
|
|
80
|
-
return previous ? this.attach(value, previous, previous.next) : this.addTail(value);
|
|
81
|
-
}
|
|
82
|
-
addBefore(value, nextValue, compareFn = compare) {
|
|
83
|
-
const next = this.find(node => compareFn(node.value, nextValue));
|
|
84
|
-
return next ? this.attach(value, next.previous, next) : this.addHead(value);
|
|
85
|
-
}
|
|
86
|
-
addByIndex(value, position) {
|
|
87
|
-
if (position < 0)
|
|
88
|
-
position += this.size;
|
|
89
|
-
else if (position >= this.size)
|
|
90
|
-
return this.addTail(value);
|
|
91
|
-
if (position <= 0)
|
|
92
|
-
return this.addHead(value);
|
|
93
|
-
const next = this.get(position);
|
|
94
|
-
return this.attach(value, next.previous, next);
|
|
95
|
-
}
|
|
96
|
-
addHead(value) {
|
|
97
|
-
const node = new ListNode(value);
|
|
98
|
-
node.next = this.first;
|
|
99
|
-
if (this.first)
|
|
100
|
-
this.first.previous = node;
|
|
101
|
-
else
|
|
102
|
-
this.last = node;
|
|
103
|
-
this.first = node;
|
|
104
|
-
this.size++;
|
|
105
|
-
return node;
|
|
106
|
-
}
|
|
107
|
-
addTail(value) {
|
|
108
|
-
const node = new ListNode(value);
|
|
109
|
-
if (this.first) {
|
|
110
|
-
node.previous = this.last;
|
|
111
|
-
this.last.next = node;
|
|
112
|
-
this.last = node;
|
|
113
|
-
}
|
|
114
|
-
else {
|
|
115
|
-
this.first = node;
|
|
116
|
-
this.last = node;
|
|
117
|
-
}
|
|
118
|
-
this.size++;
|
|
119
|
-
return node;
|
|
120
|
-
}
|
|
121
|
-
addManyAfter(values, previousValue, compareFn = compare) {
|
|
122
|
-
const previous = this.find(node => compareFn(node.value, previousValue));
|
|
123
|
-
return previous ? this.attachMany(values, previous, previous.next) : this.addManyTail(values);
|
|
124
|
-
}
|
|
125
|
-
addManyBefore(values, nextValue, compareFn = compare) {
|
|
126
|
-
const next = this.find(node => compareFn(node.value, nextValue));
|
|
127
|
-
return next ? this.attachMany(values, next.previous, next) : this.addManyHead(values);
|
|
128
|
-
}
|
|
129
|
-
addManyByIndex(values, position) {
|
|
130
|
-
if (position < 0)
|
|
131
|
-
position += this.size;
|
|
132
|
-
if (position <= 0)
|
|
133
|
-
return this.addManyHead(values);
|
|
134
|
-
if (position >= this.size)
|
|
135
|
-
return this.addManyTail(values);
|
|
136
|
-
const next = this.get(position);
|
|
137
|
-
return this.attachMany(values, next.previous, next);
|
|
138
|
-
}
|
|
139
|
-
addManyHead(values) {
|
|
140
|
-
return values.reduceRight((nodes, value) => {
|
|
141
|
-
nodes.unshift(this.addHead(value));
|
|
142
|
-
return nodes;
|
|
143
|
-
}, []);
|
|
144
|
-
}
|
|
145
|
-
addManyTail(values) {
|
|
146
|
-
return values.map(value => this.addTail(value));
|
|
147
|
-
}
|
|
148
|
-
drop() {
|
|
149
|
-
return {
|
|
150
|
-
byIndex: (position) => this.dropByIndex(position),
|
|
151
|
-
byValue: (...params) => this.dropByValue.apply(this, params),
|
|
152
|
-
byValueAll: (...params) => this.dropByValueAll.apply(this, params),
|
|
153
|
-
head: () => this.dropHead(),
|
|
154
|
-
tail: () => this.dropTail(),
|
|
155
|
-
};
|
|
156
|
-
}
|
|
157
|
-
dropMany(count) {
|
|
158
|
-
return {
|
|
159
|
-
byIndex: (position) => this.dropManyByIndex(count, position),
|
|
160
|
-
head: () => this.dropManyHead(count),
|
|
161
|
-
tail: () => this.dropManyTail(count),
|
|
162
|
-
};
|
|
163
|
-
}
|
|
164
|
-
dropByIndex(position) {
|
|
165
|
-
if (position < 0)
|
|
166
|
-
position += this.size;
|
|
167
|
-
const current = this.get(position);
|
|
168
|
-
return current ? this.detach(current) : undefined;
|
|
169
|
-
}
|
|
170
|
-
dropByValue(value, compareFn = compare) {
|
|
171
|
-
const position = this.findIndex(node => compareFn(node.value, value));
|
|
172
|
-
return position < 0 ? undefined : this.dropByIndex(position);
|
|
173
|
-
}
|
|
174
|
-
dropByValueAll(value, compareFn = compare) {
|
|
175
|
-
const dropped = [];
|
|
176
|
-
for (let current = this.first, position = 0; current; position++, current = current.next) {
|
|
177
|
-
if (compareFn(current.value, value)) {
|
|
178
|
-
dropped.push(this.dropByIndex(position - dropped.length));
|
|
179
|
-
}
|
|
180
|
-
}
|
|
181
|
-
return dropped;
|
|
182
|
-
}
|
|
183
|
-
dropHead() {
|
|
184
|
-
const head = this.first;
|
|
185
|
-
if (head) {
|
|
186
|
-
this.first = head.next;
|
|
187
|
-
if (this.first)
|
|
188
|
-
this.first.previous = undefined;
|
|
189
|
-
else
|
|
190
|
-
this.last = undefined;
|
|
191
|
-
this.size--;
|
|
192
|
-
return head;
|
|
193
|
-
}
|
|
194
|
-
return undefined;
|
|
195
|
-
}
|
|
196
|
-
dropTail() {
|
|
197
|
-
const tail = this.last;
|
|
198
|
-
if (tail) {
|
|
199
|
-
this.last = tail.previous;
|
|
200
|
-
if (this.last)
|
|
201
|
-
this.last.next = undefined;
|
|
202
|
-
else
|
|
203
|
-
this.first = undefined;
|
|
204
|
-
this.size--;
|
|
205
|
-
return tail;
|
|
206
|
-
}
|
|
207
|
-
return undefined;
|
|
208
|
-
}
|
|
209
|
-
dropManyByIndex(count, position) {
|
|
210
|
-
if (count <= 0)
|
|
211
|
-
return [];
|
|
212
|
-
if (position < 0)
|
|
213
|
-
position = Math.max(position + this.size, 0);
|
|
214
|
-
else if (position >= this.size)
|
|
215
|
-
return [];
|
|
216
|
-
count = Math.min(count, this.size - position);
|
|
217
|
-
const dropped = [];
|
|
218
|
-
while (count--) {
|
|
219
|
-
const current = this.get(position);
|
|
220
|
-
dropped.push(this.detach(current));
|
|
221
|
-
}
|
|
222
|
-
return dropped;
|
|
223
|
-
}
|
|
224
|
-
dropManyHead(count) {
|
|
225
|
-
if (count <= 0)
|
|
226
|
-
return [];
|
|
227
|
-
count = Math.min(count, this.size);
|
|
228
|
-
const dropped = [];
|
|
229
|
-
while (count--)
|
|
230
|
-
dropped.unshift(this.dropHead());
|
|
231
|
-
return dropped;
|
|
232
|
-
}
|
|
233
|
-
dropManyTail(count) {
|
|
234
|
-
if (count <= 0)
|
|
235
|
-
return [];
|
|
236
|
-
count = Math.min(count, this.size);
|
|
237
|
-
const dropped = [];
|
|
238
|
-
while (count--)
|
|
239
|
-
dropped.push(this.dropTail());
|
|
240
|
-
return dropped;
|
|
241
|
-
}
|
|
242
|
-
find(predicate) {
|
|
243
|
-
for (let current = this.first, position = 0; current; position++, current = current.next) {
|
|
244
|
-
if (predicate(current, position, this))
|
|
245
|
-
return current;
|
|
246
|
-
}
|
|
247
|
-
return undefined;
|
|
248
|
-
}
|
|
249
|
-
findIndex(predicate) {
|
|
250
|
-
for (let current = this.first, position = 0; current; position++, current = current.next) {
|
|
251
|
-
if (predicate(current, position, this))
|
|
252
|
-
return position;
|
|
253
|
-
}
|
|
254
|
-
return -1;
|
|
255
|
-
}
|
|
256
|
-
forEach(iteratorFn) {
|
|
257
|
-
for (let node = this.first, position = 0; node; position++, node = node.next) {
|
|
258
|
-
iteratorFn(node, position, this);
|
|
259
|
-
}
|
|
260
|
-
}
|
|
261
|
-
get(position) {
|
|
262
|
-
return this.find((_, index) => position === index);
|
|
263
|
-
}
|
|
264
|
-
indexOf(value, compareFn = compare) {
|
|
265
|
-
return this.findIndex(node => compareFn(node.value, value));
|
|
266
|
-
}
|
|
267
|
-
toArray() {
|
|
268
|
-
const array = new Array(this.size);
|
|
269
|
-
this.forEach((node, index) => (array[index] = node.value));
|
|
270
|
-
return array;
|
|
271
|
-
}
|
|
272
|
-
toNodeArray() {
|
|
273
|
-
const array = new Array(this.size);
|
|
274
|
-
this.forEach((node, index) => (array[index] = node));
|
|
275
|
-
return array;
|
|
276
|
-
}
|
|
277
|
-
toString(mapperFn = JSON.stringify) {
|
|
278
|
-
return this.toArray()
|
|
279
|
-
.map(value => mapperFn(value))
|
|
280
|
-
.join(' <-> ');
|
|
281
|
-
}
|
|
282
|
-
// Cannot use Generator type because of ng-packagr
|
|
283
|
-
*[Symbol.iterator]() {
|
|
284
|
-
for (let node = this.first, position = 0; node; position++, node = node.next) {
|
|
285
|
-
yield node.value;
|
|
286
|
-
}
|
|
287
|
-
}
|
|
288
|
-
}
|
|
289
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"linked-list.js","sourceRoot":"","sources":["../../../projects/utils/src/lib/linked-list.ts"],"names":[],"mappings":"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","sourcesContent":["/* tslint:disable:no-non-null-assertion */\n\nimport compare from 'just-compare';\n\nexport class ListNode<T = any> {\n  next: ListNode | undefined;\n  previous: ListNode | undefined;\n  constructor(public readonly value: T) {}\n}\n\nexport class LinkedList<T = any> {\n  private first: ListNode<T> | undefined;\n  private last: ListNode<T> | undefined;\n  private size = 0;\n\n  get head(): ListNode<T> | undefined {\n    return this.first;\n  }\n  get tail(): ListNode<T> | undefined {\n    return this.last;\n  }\n  get length(): number {\n    return this.size;\n  }\n\n  private attach(\n    value: T,\n    previousNode: ListNode<T> | undefined,\n    nextNode: ListNode<T> | undefined,\n  ): ListNode<T> {\n    if (!previousNode) return this.addHead(value);\n\n    if (!nextNode) return this.addTail(value);\n\n    const node = new ListNode(value);\n    node.previous = previousNode;\n    previousNode.next = node;\n    node.next = nextNode;\n    nextNode.previous = node;\n\n    this.size++;\n\n    return node;\n  }\n\n  private attachMany(\n    values: T[],\n    previousNode: ListNode<T> | undefined,\n    nextNode: ListNode<T> | undefined,\n  ): ListNode<T>[] {\n    if (!values.length) return [];\n\n    if (!previousNode) return this.addManyHead(values);\n\n    if (!nextNode) return this.addManyTail(values);\n\n    const list = new LinkedList<T>();\n    list.addManyTail(values);\n    list.first!.previous = previousNode;\n    previousNode.next = list.first;\n    list.last!.next = nextNode;\n    nextNode.previous = list.last;\n\n    this.size += values.length;\n\n    return list.toNodeArray();\n  }\n\n  private detach(node: ListNode<T>) {\n    if (!node.previous) return this.dropHead();\n\n    if (!node.next) return this.dropTail();\n\n    node.previous.next = node.next;\n    node.next.previous = node.previous;\n\n    this.size--;\n\n    return node;\n  }\n\n  add(value: T) {\n    return {\n      after: (...params: [T] | [any, ListComparisonFn<T>]) =>\n        this.addAfter.call(this, value, ...params),\n      before: (...params: [T] | [any, ListComparisonFn<T>]) =>\n        this.addBefore.call(this, value, ...params),\n      byIndex: (position: number) => this.addByIndex(value, position),\n      head: () => this.addHead(value),\n      tail: () => this.addTail(value),\n    };\n  }\n\n  addMany(values: T[]) {\n    return {\n      after: (...params: [T] | [any, ListComparisonFn<T>]) =>\n        this.addManyAfter.call(this, values, ...params),\n      before: (...params: [T] | [any, ListComparisonFn<T>]) =>\n        this.addManyBefore.call(this, values, ...params),\n      byIndex: (position: number) => this.addManyByIndex(values, position),\n      head: () => this.addManyHead(values),\n      tail: () => this.addManyTail(values),\n    };\n  }\n\n  addAfter(value: T, previousValue: T): ListNode<T>;\n  addAfter(value: T, previousValue: any, compareFn: ListComparisonFn<T>): ListNode<T>;\n  addAfter(value: T, previousValue: any, compareFn: ListComparisonFn<T> = compare): ListNode<T> {\n    const previous = this.find(node => compareFn(node.value, previousValue));\n\n    return previous ? this.attach(value, previous, previous.next) : this.addTail(value);\n  }\n\n  addBefore(value: T, nextValue: T): ListNode<T>;\n  addBefore(value: T, nextValue: any, compareFn: ListComparisonFn<T>): ListNode<T>;\n  addBefore(value: T, nextValue: any, compareFn: ListComparisonFn<T> = compare): ListNode<T> {\n    const next = this.find(node => compareFn(node.value, nextValue));\n\n    return next ? this.attach(value, next.previous, next) : this.addHead(value);\n  }\n\n  addByIndex(value: T, position: number): ListNode<T> {\n    if (position < 0) position += this.size;\n    else if (position >= this.size) return this.addTail(value);\n\n    if (position <= 0) return this.addHead(value);\n\n    const next = this.get(position)!;\n\n    return this.attach(value, next.previous, next);\n  }\n\n  addHead(value: T): ListNode<T> {\n    const node = new ListNode(value);\n\n    node.next = this.first;\n\n    if (this.first) this.first.previous = node;\n    else this.last = node;\n\n    this.first = node;\n    this.size++;\n\n    return node;\n  }\n\n  addTail(value: T): ListNode<T> {\n    const node = new ListNode(value);\n\n    if (this.first) {\n      node.previous = this.last;\n      this.last!.next = node;\n      this.last = node;\n    } else {\n      this.first = node;\n      this.last = node;\n    }\n\n    this.size++;\n\n    return node;\n  }\n\n  addManyAfter(values: T[], previousValue: T): ListNode<T>[];\n  addManyAfter(values: T[], previousValue: any, compareFn: ListComparisonFn<T>): ListNode<T>[];\n  addManyAfter(\n    values: T[],\n    previousValue: any,\n    compareFn: ListComparisonFn<T> = compare,\n  ): ListNode<T>[] {\n    const previous = this.find(node => compareFn(node.value, previousValue));\n\n    return previous ? this.attachMany(values, previous, previous.next) : this.addManyTail(values);\n  }\n\n  addManyBefore(values: T[], nextValue: T): ListNode<T>[];\n  addManyBefore(values: T[], nextValue: any, compareFn: ListComparisonFn<T>): ListNode<T>[];\n  addManyBefore(\n    values: T[],\n    nextValue: any,\n    compareFn: ListComparisonFn<T> = compare,\n  ): ListNode<T>[] {\n    const next = this.find(node => compareFn(node.value, nextValue));\n\n    return next ? this.attachMany(values, next.previous, next) : this.addManyHead(values);\n  }\n\n  addManyByIndex(values: T[], position: number): ListNode<T>[] {\n    if (position < 0) position += this.size;\n\n    if (position <= 0) return this.addManyHead(values);\n\n    if (position >= this.size) return this.addManyTail(values);\n\n    const next = this.get(position)!;\n\n    return this.attachMany(values, next.previous, next);\n  }\n\n  addManyHead(values: T[]): ListNode<T>[] {\n    return values.reduceRight<ListNode<T>[]>((nodes, value) => {\n      nodes.unshift(this.addHead(value));\n      return nodes;\n    }, []);\n  }\n\n  addManyTail(values: T[]): ListNode<T>[] {\n    return values.map(value => this.addTail(value));\n  }\n\n  drop() {\n    return {\n      byIndex: (position: number) => this.dropByIndex(position),\n      byValue: (...params: [T] | [any, ListComparisonFn<T>]) =>\n        this.dropByValue.apply(this, params),\n      byValueAll: (...params: [T] | [any, ListComparisonFn<T>]) =>\n        this.dropByValueAll.apply(this, params),\n      head: () => this.dropHead(),\n      tail: () => this.dropTail(),\n    };\n  }\n\n  dropMany(count: number) {\n    return {\n      byIndex: (position: number) => this.dropManyByIndex(count, position),\n      head: () => this.dropManyHead(count),\n      tail: () => this.dropManyTail(count),\n    };\n  }\n\n  dropByIndex(position: number): ListNode<T> | undefined {\n    if (position < 0) position += this.size;\n\n    const current = this.get(position);\n\n    return current ? this.detach(current) : undefined;\n  }\n\n  dropByValue(value: T): ListNode<T> | undefined;\n  dropByValue(value: any, compareFn: ListComparisonFn<T>): ListNode<T> | undefined;\n  dropByValue(value: any, compareFn: ListComparisonFn<T> = compare): ListNode<T> | undefined {\n    const position = this.findIndex(node => compareFn(node.value, value));\n\n    return position < 0 ? undefined : this.dropByIndex(position);\n  }\n\n  dropByValueAll(value: T): ListNode<T>[];\n  dropByValueAll(value: any, compareFn: ListComparisonFn<T>): ListNode<T>[];\n  dropByValueAll(value: any, compareFn: ListComparisonFn<T> = compare): ListNode<T>[] {\n    const dropped: ListNode<T>[] = [];\n\n    for (let current = this.first, position = 0; current; position++, current = current.next) {\n      if (compareFn(current.value, value)) {\n        dropped.push(this.dropByIndex(position - dropped.length)!);\n      }\n    }\n\n    return dropped;\n  }\n\n  dropHead(): ListNode<T> | undefined {\n    const head = this.first;\n\n    if (head) {\n      this.first = head.next;\n\n      if (this.first) this.first.previous = undefined;\n      else this.last = undefined;\n\n      this.size--;\n\n      return head;\n    }\n\n    return undefined;\n  }\n\n  dropTail(): ListNode<T> | undefined {\n    const tail = this.last;\n\n    if (tail) {\n      this.last = tail.previous;\n\n      if (this.last) this.last.next = undefined;\n      else this.first = undefined;\n\n      this.size--;\n\n      return tail;\n    }\n\n    return undefined;\n  }\n\n  dropManyByIndex(count: number, position: number): ListNode<T>[] {\n    if (count <= 0) return [];\n\n    if (position < 0) position = Math.max(position + this.size, 0);\n    else if (position >= this.size) return [];\n\n    count = Math.min(count, this.size - position);\n\n    const dropped: ListNode<T>[] = [];\n\n    while (count--) {\n      const current = this.get(position);\n      dropped.push(this.detach(current!)!);\n    }\n\n    return dropped;\n  }\n\n  dropManyHead(count: Exclude<number, 0>): ListNode<T>[] {\n    if (count <= 0) return [];\n\n    count = Math.min(count, this.size);\n\n    const dropped: ListNode<T>[] = [];\n\n    while (count--) dropped.unshift(this.dropHead()!);\n\n    return dropped;\n  }\n\n  dropManyTail(count: Exclude<number, 0>): ListNode<T>[] {\n    if (count <= 0) return [];\n\n    count = Math.min(count, this.size);\n\n    const dropped: ListNode<T>[] = [];\n\n    while (count--) dropped.push(this.dropTail()!);\n\n    return dropped;\n  }\n\n  find(predicate: ListIteratorFn<T>): ListNode<T> | undefined {\n    for (let current = this.first, position = 0; current; position++, current = current.next) {\n      if (predicate(current, position, this)) return current;\n    }\n\n    return undefined;\n  }\n\n  findIndex(predicate: ListIteratorFn<T>): number {\n    for (let current = this.first, position = 0; current; position++, current = current.next) {\n      if (predicate(current, position, this)) return position;\n    }\n\n    return -1;\n  }\n\n  forEach<R = boolean>(iteratorFn: ListIteratorFn<T, R>) {\n    for (let node = this.first, position = 0; node; position++, node = node.next) {\n      iteratorFn(node, position, this);\n    }\n  }\n\n  get(position: number): ListNode<T> | undefined {\n    return this.find((_, index) => position === index);\n  }\n\n  indexOf(value: T): number;\n  indexOf(value: any, compareFn: ListComparisonFn<T>): number;\n  indexOf(value: any, compareFn: ListComparisonFn<T> = compare): number {\n    return this.findIndex(node => compareFn(node.value, value));\n  }\n\n  toArray(): T[] {\n    const array = new Array(this.size);\n\n    this.forEach((node, index) => (array[index!] = node.value));\n\n    return array;\n  }\n\n  toNodeArray(): ListNode<T>[] {\n    const array = new Array(this.size);\n\n    this.forEach((node, index) => (array[index!] = node));\n\n    return array;\n  }\n\n  toString(mapperFn: ListMapperFn<T> = JSON.stringify): string {\n    return this.toArray()\n      .map(value => mapperFn(value))\n      .join(' <-> ');\n  }\n\n  // Cannot use Generator type because of ng-packagr\n  *[Symbol.iterator](): any {\n    for (let node = this.first, position = 0; node; position++, node = node.next) {\n      yield node.value;\n    }\n  }\n}\n\nexport type ListMapperFn<T = any> = (value: T) => any;\n\nexport type ListComparisonFn<T = any> = (value1: T, value2: any) => boolean;\n\nexport type ListIteratorFn<T = any, R = boolean> = (\n  node: ListNode<T>,\n  index?: number,\n  list?: LinkedList,\n) => R;\n"]}
|
|
1
|
+
/* tslint:disable:no-non-null-assertion */
|
|
2
|
+
import compare from 'just-compare';
|
|
3
|
+
export class ListNode {
|
|
4
|
+
constructor(value) {
|
|
5
|
+
this.value = value;
|
|
6
|
+
}
|
|
7
|
+
}
|
|
8
|
+
export class LinkedList {
|
|
9
|
+
constructor() {
|
|
10
|
+
this.size = 0;
|
|
11
|
+
}
|
|
12
|
+
get head() {
|
|
13
|
+
return this.first;
|
|
14
|
+
}
|
|
15
|
+
get tail() {
|
|
16
|
+
return this.last;
|
|
17
|
+
}
|
|
18
|
+
get length() {
|
|
19
|
+
return this.size;
|
|
20
|
+
}
|
|
21
|
+
attach(value, previousNode, nextNode) {
|
|
22
|
+
if (!previousNode)
|
|
23
|
+
return this.addHead(value);
|
|
24
|
+
if (!nextNode)
|
|
25
|
+
return this.addTail(value);
|
|
26
|
+
const node = new ListNode(value);
|
|
27
|
+
node.previous = previousNode;
|
|
28
|
+
previousNode.next = node;
|
|
29
|
+
node.next = nextNode;
|
|
30
|
+
nextNode.previous = node;
|
|
31
|
+
this.size++;
|
|
32
|
+
return node;
|
|
33
|
+
}
|
|
34
|
+
attachMany(values, previousNode, nextNode) {
|
|
35
|
+
if (!values.length)
|
|
36
|
+
return [];
|
|
37
|
+
if (!previousNode)
|
|
38
|
+
return this.addManyHead(values);
|
|
39
|
+
if (!nextNode)
|
|
40
|
+
return this.addManyTail(values);
|
|
41
|
+
const list = new LinkedList();
|
|
42
|
+
list.addManyTail(values);
|
|
43
|
+
list.first.previous = previousNode;
|
|
44
|
+
previousNode.next = list.first;
|
|
45
|
+
list.last.next = nextNode;
|
|
46
|
+
nextNode.previous = list.last;
|
|
47
|
+
this.size += values.length;
|
|
48
|
+
return list.toNodeArray();
|
|
49
|
+
}
|
|
50
|
+
detach(node) {
|
|
51
|
+
if (!node.previous)
|
|
52
|
+
return this.dropHead();
|
|
53
|
+
if (!node.next)
|
|
54
|
+
return this.dropTail();
|
|
55
|
+
node.previous.next = node.next;
|
|
56
|
+
node.next.previous = node.previous;
|
|
57
|
+
this.size--;
|
|
58
|
+
return node;
|
|
59
|
+
}
|
|
60
|
+
add(value) {
|
|
61
|
+
return {
|
|
62
|
+
after: (...params) => this.addAfter.call(this, value, ...params),
|
|
63
|
+
before: (...params) => this.addBefore.call(this, value, ...params),
|
|
64
|
+
byIndex: (position) => this.addByIndex(value, position),
|
|
65
|
+
head: () => this.addHead(value),
|
|
66
|
+
tail: () => this.addTail(value),
|
|
67
|
+
};
|
|
68
|
+
}
|
|
69
|
+
addMany(values) {
|
|
70
|
+
return {
|
|
71
|
+
after: (...params) => this.addManyAfter.call(this, values, ...params),
|
|
72
|
+
before: (...params) => this.addManyBefore.call(this, values, ...params),
|
|
73
|
+
byIndex: (position) => this.addManyByIndex(values, position),
|
|
74
|
+
head: () => this.addManyHead(values),
|
|
75
|
+
tail: () => this.addManyTail(values),
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
addAfter(value, previousValue, compareFn = compare) {
|
|
79
|
+
const previous = this.find(node => compareFn(node.value, previousValue));
|
|
80
|
+
return previous ? this.attach(value, previous, previous.next) : this.addTail(value);
|
|
81
|
+
}
|
|
82
|
+
addBefore(value, nextValue, compareFn = compare) {
|
|
83
|
+
const next = this.find(node => compareFn(node.value, nextValue));
|
|
84
|
+
return next ? this.attach(value, next.previous, next) : this.addHead(value);
|
|
85
|
+
}
|
|
86
|
+
addByIndex(value, position) {
|
|
87
|
+
if (position < 0)
|
|
88
|
+
position += this.size;
|
|
89
|
+
else if (position >= this.size)
|
|
90
|
+
return this.addTail(value);
|
|
91
|
+
if (position <= 0)
|
|
92
|
+
return this.addHead(value);
|
|
93
|
+
const next = this.get(position);
|
|
94
|
+
return this.attach(value, next.previous, next);
|
|
95
|
+
}
|
|
96
|
+
addHead(value) {
|
|
97
|
+
const node = new ListNode(value);
|
|
98
|
+
node.next = this.first;
|
|
99
|
+
if (this.first)
|
|
100
|
+
this.first.previous = node;
|
|
101
|
+
else
|
|
102
|
+
this.last = node;
|
|
103
|
+
this.first = node;
|
|
104
|
+
this.size++;
|
|
105
|
+
return node;
|
|
106
|
+
}
|
|
107
|
+
addTail(value) {
|
|
108
|
+
const node = new ListNode(value);
|
|
109
|
+
if (this.first) {
|
|
110
|
+
node.previous = this.last;
|
|
111
|
+
this.last.next = node;
|
|
112
|
+
this.last = node;
|
|
113
|
+
}
|
|
114
|
+
else {
|
|
115
|
+
this.first = node;
|
|
116
|
+
this.last = node;
|
|
117
|
+
}
|
|
118
|
+
this.size++;
|
|
119
|
+
return node;
|
|
120
|
+
}
|
|
121
|
+
addManyAfter(values, previousValue, compareFn = compare) {
|
|
122
|
+
const previous = this.find(node => compareFn(node.value, previousValue));
|
|
123
|
+
return previous ? this.attachMany(values, previous, previous.next) : this.addManyTail(values);
|
|
124
|
+
}
|
|
125
|
+
addManyBefore(values, nextValue, compareFn = compare) {
|
|
126
|
+
const next = this.find(node => compareFn(node.value, nextValue));
|
|
127
|
+
return next ? this.attachMany(values, next.previous, next) : this.addManyHead(values);
|
|
128
|
+
}
|
|
129
|
+
addManyByIndex(values, position) {
|
|
130
|
+
if (position < 0)
|
|
131
|
+
position += this.size;
|
|
132
|
+
if (position <= 0)
|
|
133
|
+
return this.addManyHead(values);
|
|
134
|
+
if (position >= this.size)
|
|
135
|
+
return this.addManyTail(values);
|
|
136
|
+
const next = this.get(position);
|
|
137
|
+
return this.attachMany(values, next.previous, next);
|
|
138
|
+
}
|
|
139
|
+
addManyHead(values) {
|
|
140
|
+
return values.reduceRight((nodes, value) => {
|
|
141
|
+
nodes.unshift(this.addHead(value));
|
|
142
|
+
return nodes;
|
|
143
|
+
}, []);
|
|
144
|
+
}
|
|
145
|
+
addManyTail(values) {
|
|
146
|
+
return values.map(value => this.addTail(value));
|
|
147
|
+
}
|
|
148
|
+
drop() {
|
|
149
|
+
return {
|
|
150
|
+
byIndex: (position) => this.dropByIndex(position),
|
|
151
|
+
byValue: (...params) => this.dropByValue.apply(this, params),
|
|
152
|
+
byValueAll: (...params) => this.dropByValueAll.apply(this, params),
|
|
153
|
+
head: () => this.dropHead(),
|
|
154
|
+
tail: () => this.dropTail(),
|
|
155
|
+
};
|
|
156
|
+
}
|
|
157
|
+
dropMany(count) {
|
|
158
|
+
return {
|
|
159
|
+
byIndex: (position) => this.dropManyByIndex(count, position),
|
|
160
|
+
head: () => this.dropManyHead(count),
|
|
161
|
+
tail: () => this.dropManyTail(count),
|
|
162
|
+
};
|
|
163
|
+
}
|
|
164
|
+
dropByIndex(position) {
|
|
165
|
+
if (position < 0)
|
|
166
|
+
position += this.size;
|
|
167
|
+
const current = this.get(position);
|
|
168
|
+
return current ? this.detach(current) : undefined;
|
|
169
|
+
}
|
|
170
|
+
dropByValue(value, compareFn = compare) {
|
|
171
|
+
const position = this.findIndex(node => compareFn(node.value, value));
|
|
172
|
+
return position < 0 ? undefined : this.dropByIndex(position);
|
|
173
|
+
}
|
|
174
|
+
dropByValueAll(value, compareFn = compare) {
|
|
175
|
+
const dropped = [];
|
|
176
|
+
for (let current = this.first, position = 0; current; position++, current = current.next) {
|
|
177
|
+
if (compareFn(current.value, value)) {
|
|
178
|
+
dropped.push(this.dropByIndex(position - dropped.length));
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
return dropped;
|
|
182
|
+
}
|
|
183
|
+
dropHead() {
|
|
184
|
+
const head = this.first;
|
|
185
|
+
if (head) {
|
|
186
|
+
this.first = head.next;
|
|
187
|
+
if (this.first)
|
|
188
|
+
this.first.previous = undefined;
|
|
189
|
+
else
|
|
190
|
+
this.last = undefined;
|
|
191
|
+
this.size--;
|
|
192
|
+
return head;
|
|
193
|
+
}
|
|
194
|
+
return undefined;
|
|
195
|
+
}
|
|
196
|
+
dropTail() {
|
|
197
|
+
const tail = this.last;
|
|
198
|
+
if (tail) {
|
|
199
|
+
this.last = tail.previous;
|
|
200
|
+
if (this.last)
|
|
201
|
+
this.last.next = undefined;
|
|
202
|
+
else
|
|
203
|
+
this.first = undefined;
|
|
204
|
+
this.size--;
|
|
205
|
+
return tail;
|
|
206
|
+
}
|
|
207
|
+
return undefined;
|
|
208
|
+
}
|
|
209
|
+
dropManyByIndex(count, position) {
|
|
210
|
+
if (count <= 0)
|
|
211
|
+
return [];
|
|
212
|
+
if (position < 0)
|
|
213
|
+
position = Math.max(position + this.size, 0);
|
|
214
|
+
else if (position >= this.size)
|
|
215
|
+
return [];
|
|
216
|
+
count = Math.min(count, this.size - position);
|
|
217
|
+
const dropped = [];
|
|
218
|
+
while (count--) {
|
|
219
|
+
const current = this.get(position);
|
|
220
|
+
dropped.push(this.detach(current));
|
|
221
|
+
}
|
|
222
|
+
return dropped;
|
|
223
|
+
}
|
|
224
|
+
dropManyHead(count) {
|
|
225
|
+
if (count <= 0)
|
|
226
|
+
return [];
|
|
227
|
+
count = Math.min(count, this.size);
|
|
228
|
+
const dropped = [];
|
|
229
|
+
while (count--)
|
|
230
|
+
dropped.unshift(this.dropHead());
|
|
231
|
+
return dropped;
|
|
232
|
+
}
|
|
233
|
+
dropManyTail(count) {
|
|
234
|
+
if (count <= 0)
|
|
235
|
+
return [];
|
|
236
|
+
count = Math.min(count, this.size);
|
|
237
|
+
const dropped = [];
|
|
238
|
+
while (count--)
|
|
239
|
+
dropped.push(this.dropTail());
|
|
240
|
+
return dropped;
|
|
241
|
+
}
|
|
242
|
+
find(predicate) {
|
|
243
|
+
for (let current = this.first, position = 0; current; position++, current = current.next) {
|
|
244
|
+
if (predicate(current, position, this))
|
|
245
|
+
return current;
|
|
246
|
+
}
|
|
247
|
+
return undefined;
|
|
248
|
+
}
|
|
249
|
+
findIndex(predicate) {
|
|
250
|
+
for (let current = this.first, position = 0; current; position++, current = current.next) {
|
|
251
|
+
if (predicate(current, position, this))
|
|
252
|
+
return position;
|
|
253
|
+
}
|
|
254
|
+
return -1;
|
|
255
|
+
}
|
|
256
|
+
forEach(iteratorFn) {
|
|
257
|
+
for (let node = this.first, position = 0; node; position++, node = node.next) {
|
|
258
|
+
iteratorFn(node, position, this);
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
get(position) {
|
|
262
|
+
return this.find((_, index) => position === index);
|
|
263
|
+
}
|
|
264
|
+
indexOf(value, compareFn = compare) {
|
|
265
|
+
return this.findIndex(node => compareFn(node.value, value));
|
|
266
|
+
}
|
|
267
|
+
toArray() {
|
|
268
|
+
const array = new Array(this.size);
|
|
269
|
+
this.forEach((node, index) => (array[index] = node.value));
|
|
270
|
+
return array;
|
|
271
|
+
}
|
|
272
|
+
toNodeArray() {
|
|
273
|
+
const array = new Array(this.size);
|
|
274
|
+
this.forEach((node, index) => (array[index] = node));
|
|
275
|
+
return array;
|
|
276
|
+
}
|
|
277
|
+
toString(mapperFn = JSON.stringify) {
|
|
278
|
+
return this.toArray()
|
|
279
|
+
.map(value => mapperFn(value))
|
|
280
|
+
.join(' <-> ');
|
|
281
|
+
}
|
|
282
|
+
// Cannot use Generator type because of ng-packagr
|
|
283
|
+
*[Symbol.iterator]() {
|
|
284
|
+
for (let node = this.first, position = 0; node; position++, node = node.next) {
|
|
285
|
+
yield node.value;
|
|
286
|
+
}
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"linked-list.js","sourceRoot":"","sources":["../../../projects/utils/src/lib/linked-list.ts"],"names":[],"mappings":"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","sourcesContent":["/* tslint:disable:no-non-null-assertion */\r\n\r\nimport compare from 'just-compare';\r\n\r\nexport class ListNode<T = any> {\r\n  next: ListNode | undefined;\r\n  previous: ListNode | undefined;\r\n  constructor(public readonly value: T) {}\r\n}\r\n\r\nexport class LinkedList<T = any> {\r\n  private first: ListNode<T> | undefined;\r\n  private last: ListNode<T> | undefined;\r\n  private size = 0;\r\n\r\n  get head(): ListNode<T> | undefined {\r\n    return this.first;\r\n  }\r\n  get tail(): ListNode<T> | undefined {\r\n    return this.last;\r\n  }\r\n  get length(): number {\r\n    return this.size;\r\n  }\r\n\r\n  private attach(\r\n    value: T,\r\n    previousNode: ListNode<T> | undefined,\r\n    nextNode: ListNode<T> | undefined,\r\n  ): ListNode<T> {\r\n    if (!previousNode) return this.addHead(value);\r\n\r\n    if (!nextNode) return this.addTail(value);\r\n\r\n    const node = new ListNode(value);\r\n    node.previous = previousNode;\r\n    previousNode.next = node;\r\n    node.next = nextNode;\r\n    nextNode.previous = node;\r\n\r\n    this.size++;\r\n\r\n    return node;\r\n  }\r\n\r\n  private attachMany(\r\n    values: T[],\r\n    previousNode: ListNode<T> | undefined,\r\n    nextNode: ListNode<T> | undefined,\r\n  ): ListNode<T>[] {\r\n    if (!values.length) return [];\r\n\r\n    if (!previousNode) return this.addManyHead(values);\r\n\r\n    if (!nextNode) return this.addManyTail(values);\r\n\r\n    const list = new LinkedList<T>();\r\n    list.addManyTail(values);\r\n    list.first!.previous = previousNode;\r\n    previousNode.next = list.first;\r\n    list.last!.next = nextNode;\r\n    nextNode.previous = list.last;\r\n\r\n    this.size += values.length;\r\n\r\n    return list.toNodeArray();\r\n  }\r\n\r\n  private detach(node: ListNode<T>) {\r\n    if (!node.previous) return this.dropHead();\r\n\r\n    if (!node.next) return this.dropTail();\r\n\r\n    node.previous.next = node.next;\r\n    node.next.previous = node.previous;\r\n\r\n    this.size--;\r\n\r\n    return node;\r\n  }\r\n\r\n  add(value: T) {\r\n    return {\r\n      after: (...params: [T] | [any, ListComparisonFn<T>]) =>\r\n        this.addAfter.call(this, value, ...params),\r\n      before: (...params: [T] | [any, ListComparisonFn<T>]) =>\r\n        this.addBefore.call(this, value, ...params),\r\n      byIndex: (position: number) => this.addByIndex(value, position),\r\n      head: () => this.addHead(value),\r\n      tail: () => this.addTail(value),\r\n    };\r\n  }\r\n\r\n  addMany(values: T[]) {\r\n    return {\r\n      after: (...params: [T] | [any, ListComparisonFn<T>]) =>\r\n        this.addManyAfter.call(this, values, ...params),\r\n      before: (...params: [T] | [any, ListComparisonFn<T>]) =>\r\n        this.addManyBefore.call(this, values, ...params),\r\n      byIndex: (position: number) => this.addManyByIndex(values, position),\r\n      head: () => this.addManyHead(values),\r\n      tail: () => this.addManyTail(values),\r\n    };\r\n  }\r\n\r\n  addAfter(value: T, previousValue: T): ListNode<T>;\r\n  addAfter(value: T, previousValue: any, compareFn: ListComparisonFn<T>): ListNode<T>;\r\n  addAfter(value: T, previousValue: any, compareFn: ListComparisonFn<T> = compare): ListNode<T> {\r\n    const previous = this.find(node => compareFn(node.value, previousValue));\r\n\r\n    return previous ? this.attach(value, previous, previous.next) : this.addTail(value);\r\n  }\r\n\r\n  addBefore(value: T, nextValue: T): ListNode<T>;\r\n  addBefore(value: T, nextValue: any, compareFn: ListComparisonFn<T>): ListNode<T>;\r\n  addBefore(value: T, nextValue: any, compareFn: ListComparisonFn<T> = compare): ListNode<T> {\r\n    const next = this.find(node => compareFn(node.value, nextValue));\r\n\r\n    return next ? this.attach(value, next.previous, next) : this.addHead(value);\r\n  }\r\n\r\n  addByIndex(value: T, position: number): ListNode<T> {\r\n    if (position < 0) position += this.size;\r\n    else if (position >= this.size) return this.addTail(value);\r\n\r\n    if (position <= 0) return this.addHead(value);\r\n\r\n    const next = this.get(position)!;\r\n\r\n    return this.attach(value, next.previous, next);\r\n  }\r\n\r\n  addHead(value: T): ListNode<T> {\r\n    const node = new ListNode(value);\r\n\r\n    node.next = this.first;\r\n\r\n    if (this.first) this.first.previous = node;\r\n    else this.last = node;\r\n\r\n    this.first = node;\r\n    this.size++;\r\n\r\n    return node;\r\n  }\r\n\r\n  addTail(value: T): ListNode<T> {\r\n    const node = new ListNode(value);\r\n\r\n    if (this.first) {\r\n      node.previous = this.last;\r\n      this.last!.next = node;\r\n      this.last = node;\r\n    } else {\r\n      this.first = node;\r\n      this.last = node;\r\n    }\r\n\r\n    this.size++;\r\n\r\n    return node;\r\n  }\r\n\r\n  addManyAfter(values: T[], previousValue: T): ListNode<T>[];\r\n  addManyAfter(values: T[], previousValue: any, compareFn: ListComparisonFn<T>): ListNode<T>[];\r\n  addManyAfter(\r\n    values: T[],\r\n    previousValue: any,\r\n    compareFn: ListComparisonFn<T> = compare,\r\n  ): ListNode<T>[] {\r\n    const previous = this.find(node => compareFn(node.value, previousValue));\r\n\r\n    return previous ? this.attachMany(values, previous, previous.next) : this.addManyTail(values);\r\n  }\r\n\r\n  addManyBefore(values: T[], nextValue: T): ListNode<T>[];\r\n  addManyBefore(values: T[], nextValue: any, compareFn: ListComparisonFn<T>): ListNode<T>[];\r\n  addManyBefore(\r\n    values: T[],\r\n    nextValue: any,\r\n    compareFn: ListComparisonFn<T> = compare,\r\n  ): ListNode<T>[] {\r\n    const next = this.find(node => compareFn(node.value, nextValue));\r\n\r\n    return next ? this.attachMany(values, next.previous, next) : this.addManyHead(values);\r\n  }\r\n\r\n  addManyByIndex(values: T[], position: number): ListNode<T>[] {\r\n    if (position < 0) position += this.size;\r\n\r\n    if (position <= 0) return this.addManyHead(values);\r\n\r\n    if (position >= this.size) return this.addManyTail(values);\r\n\r\n    const next = this.get(position)!;\r\n\r\n    return this.attachMany(values, next.previous, next);\r\n  }\r\n\r\n  addManyHead(values: T[]): ListNode<T>[] {\r\n    return values.reduceRight<ListNode<T>[]>((nodes, value) => {\r\n      nodes.unshift(this.addHead(value));\r\n      return nodes;\r\n    }, []);\r\n  }\r\n\r\n  addManyTail(values: T[]): ListNode<T>[] {\r\n    return values.map(value => this.addTail(value));\r\n  }\r\n\r\n  drop() {\r\n    return {\r\n      byIndex: (position: number) => this.dropByIndex(position),\r\n      byValue: (...params: [T] | [any, ListComparisonFn<T>]) =>\r\n        this.dropByValue.apply(this, params),\r\n      byValueAll: (...params: [T] | [any, ListComparisonFn<T>]) =>\r\n        this.dropByValueAll.apply(this, params),\r\n      head: () => this.dropHead(),\r\n      tail: () => this.dropTail(),\r\n    };\r\n  }\r\n\r\n  dropMany(count: number) {\r\n    return {\r\n      byIndex: (position: number) => this.dropManyByIndex(count, position),\r\n      head: () => this.dropManyHead(count),\r\n      tail: () => this.dropManyTail(count),\r\n    };\r\n  }\r\n\r\n  dropByIndex(position: number): ListNode<T> | undefined {\r\n    if (position < 0) position += this.size;\r\n\r\n    const current = this.get(position);\r\n\r\n    return current ? this.detach(current) : undefined;\r\n  }\r\n\r\n  dropByValue(value: T): ListNode<T> | undefined;\r\n  dropByValue(value: any, compareFn: ListComparisonFn<T>): ListNode<T> | undefined;\r\n  dropByValue(value: any, compareFn: ListComparisonFn<T> = compare): ListNode<T> | undefined {\r\n    const position = this.findIndex(node => compareFn(node.value, value));\r\n\r\n    return position < 0 ? undefined : this.dropByIndex(position);\r\n  }\r\n\r\n  dropByValueAll(value: T): ListNode<T>[];\r\n  dropByValueAll(value: any, compareFn: ListComparisonFn<T>): ListNode<T>[];\r\n  dropByValueAll(value: any, compareFn: ListComparisonFn<T> = compare): ListNode<T>[] {\r\n    const dropped: ListNode<T>[] = [];\r\n\r\n    for (let current = this.first, position = 0; current; position++, current = current.next) {\r\n      if (compareFn(current.value, value)) {\r\n        dropped.push(this.dropByIndex(position - dropped.length)!);\r\n      }\r\n    }\r\n\r\n    return dropped;\r\n  }\r\n\r\n  dropHead(): ListNode<T> | undefined {\r\n    const head = this.first;\r\n\r\n    if (head) {\r\n      this.first = head.next;\r\n\r\n      if (this.first) this.first.previous = undefined;\r\n      else this.last = undefined;\r\n\r\n      this.size--;\r\n\r\n      return head;\r\n    }\r\n\r\n    return undefined;\r\n  }\r\n\r\n  dropTail(): ListNode<T> | undefined {\r\n    const tail = this.last;\r\n\r\n    if (tail) {\r\n      this.last = tail.previous;\r\n\r\n      if (this.last) this.last.next = undefined;\r\n      else this.first = undefined;\r\n\r\n      this.size--;\r\n\r\n      return tail;\r\n    }\r\n\r\n    return undefined;\r\n  }\r\n\r\n  dropManyByIndex(count: number, position: number): ListNode<T>[] {\r\n    if (count <= 0) return [];\r\n\r\n    if (position < 0) position = Math.max(position + this.size, 0);\r\n    else if (position >= this.size) return [];\r\n\r\n    count = Math.min(count, this.size - position);\r\n\r\n    const dropped: ListNode<T>[] = [];\r\n\r\n    while (count--) {\r\n      const current = this.get(position);\r\n      dropped.push(this.detach(current!)!);\r\n    }\r\n\r\n    return dropped;\r\n  }\r\n\r\n  dropManyHead(count: Exclude<number, 0>): ListNode<T>[] {\r\n    if (count <= 0) return [];\r\n\r\n    count = Math.min(count, this.size);\r\n\r\n    const dropped: ListNode<T>[] = [];\r\n\r\n    while (count--) dropped.unshift(this.dropHead()!);\r\n\r\n    return dropped;\r\n  }\r\n\r\n  dropManyTail(count: Exclude<number, 0>): ListNode<T>[] {\r\n    if (count <= 0) return [];\r\n\r\n    count = Math.min(count, this.size);\r\n\r\n    const dropped: ListNode<T>[] = [];\r\n\r\n    while (count--) dropped.push(this.dropTail()!);\r\n\r\n    return dropped;\r\n  }\r\n\r\n  find(predicate: ListIteratorFn<T>): ListNode<T> | undefined {\r\n    for (let current = this.first, position = 0; current; position++, current = current.next) {\r\n      if (predicate(current, position, this)) return current;\r\n    }\r\n\r\n    return undefined;\r\n  }\r\n\r\n  findIndex(predicate: ListIteratorFn<T>): number {\r\n    for (let current = this.first, position = 0; current; position++, current = current.next) {\r\n      if (predicate(current, position, this)) return position;\r\n    }\r\n\r\n    return -1;\r\n  }\r\n\r\n  forEach<R = boolean>(iteratorFn: ListIteratorFn<T, R>) {\r\n    for (let node = this.first, position = 0; node; position++, node = node.next) {\r\n      iteratorFn(node, position, this);\r\n    }\r\n  }\r\n\r\n  get(position: number): ListNode<T> | undefined {\r\n    return this.find((_, index) => position === index);\r\n  }\r\n\r\n  indexOf(value: T): number;\r\n  indexOf(value: any, compareFn: ListComparisonFn<T>): number;\r\n  indexOf(value: any, compareFn: ListComparisonFn<T> = compare): number {\r\n    return this.findIndex(node => compareFn(node.value, value));\r\n  }\r\n\r\n  toArray(): T[] {\r\n    const array = new Array(this.size);\r\n\r\n    this.forEach((node, index) => (array[index!] = node.value));\r\n\r\n    return array;\r\n  }\r\n\r\n  toNodeArray(): ListNode<T>[] {\r\n    const array = new Array(this.size);\r\n\r\n    this.forEach((node, index) => (array[index!] = node));\r\n\r\n    return array;\r\n  }\r\n\r\n  toString(mapperFn: ListMapperFn<T> = JSON.stringify): string {\r\n    return this.toArray()\r\n      .map(value => mapperFn(value))\r\n      .join(' <-> ');\r\n  }\r\n\r\n  // Cannot use Generator type because of ng-packagr\r\n  *[Symbol.iterator](): any {\r\n    for (let node = this.first, position = 0; node; position++, node = node.next) {\r\n      yield node.value;\r\n    }\r\n  }\r\n}\r\n\r\nexport type ListMapperFn<T = any> = (value: T) => any;\r\n\r\nexport type ListComparisonFn<T = any> = (value1: T, value2: any) => boolean;\r\n\r\nexport type ListIteratorFn<T = any, R = boolean> = (\r\n  node: ListNode<T>,\r\n  index?: number,\r\n  list?: LinkedList,\r\n) => R;\r\n"]}
|