@plait/core 0.77.3 → 0.78.0
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/fesm2022/plait-core.mjs.map +1 -1
- package/package.json +1 -3
- package/esm2022/constants/cursor.mjs +0 -13
- package/esm2022/constants/index.mjs +0 -20
- package/esm2022/constants/keycodes.mjs +0 -127
- package/esm2022/constants/selection.mjs +0 -6
- package/esm2022/constants/zoom.mjs +0 -4
- package/esm2022/context.mjs +0 -23
- package/esm2022/core/element/context-change.mjs +0 -13
- package/esm2022/core/element/context.mjs +0 -2
- package/esm2022/core/element/element-flavour.mjs +0 -140
- package/esm2022/core/element/element-ref.mjs +0 -2
- package/esm2022/core/list-render.mjs +0 -217
- package/esm2022/differs/default_iterable_differ.mjs +0 -614
- package/esm2022/differs/iterable_differs.mjs +0 -9
- package/esm2022/interfaces/board.mjs +0 -108
- package/esm2022/interfaces/custom-types.mjs +0 -5
- package/esm2022/interfaces/direction.mjs +0 -8
- package/esm2022/interfaces/element.mjs +0 -43
- package/esm2022/interfaces/group.mjs +0 -6
- package/esm2022/interfaces/history.mjs +0 -5
- package/esm2022/interfaces/index.mjs +0 -19
- package/esm2022/interfaces/node.mjs +0 -57
- package/esm2022/interfaces/operation.mjs +0 -96
- package/esm2022/interfaces/path-ref.mjs +0 -15
- package/esm2022/interfaces/path.mjs +0 -183
- package/esm2022/interfaces/plugin.mjs +0 -6
- package/esm2022/interfaces/point.mjs +0 -27
- package/esm2022/interfaces/pointer.mjs +0 -6
- package/esm2022/interfaces/rectangle-client.mjs +0 -171
- package/esm2022/interfaces/selection.mjs +0 -13
- package/esm2022/interfaces/svg-arc-command.mjs +0 -2
- package/esm2022/interfaces/theme.mjs +0 -49
- package/esm2022/interfaces/viewport.mjs +0 -7
- package/esm2022/plait-core.mjs +0 -5
- package/esm2022/plugins/create-board.mjs +0 -122
- package/esm2022/plugins/index.mjs +0 -11
- package/esm2022/plugins/with-board.mjs +0 -20
- package/esm2022/plugins/with-hand.mjs +0 -113
- package/esm2022/plugins/with-history.mjs +0 -91
- package/esm2022/plugins/with-hotkey.mjs +0 -96
- package/esm2022/plugins/with-i18n.mjs +0 -13
- package/esm2022/plugins/with-moving.mjs +0 -282
- package/esm2022/plugins/with-options.mjs +0 -13
- package/esm2022/plugins/with-related-fragment.mjs +0 -23
- package/esm2022/plugins/with-selection.mjs +0 -230
- package/esm2022/public-api.mjs +0 -16
- package/esm2022/testing/core/create-board.mjs +0 -15
- package/esm2022/testing/core/fake-weak-map.mjs +0 -18
- package/esm2022/testing/core/index.mjs +0 -3
- package/esm2022/testing/fake-events/event-objects.mjs +0 -131
- package/esm2022/testing/fake-events/index.mjs +0 -2
- package/esm2022/testing/index.mjs +0 -3
- package/esm2022/testing/test-element.mjs +0 -9
- package/esm2022/transforms/board.mjs +0 -137
- package/esm2022/transforms/element.mjs +0 -22
- package/esm2022/transforms/general.mjs +0 -146
- package/esm2022/transforms/group.mjs +0 -64
- package/esm2022/transforms/index.mjs +0 -17
- package/esm2022/transforms/node.mjs +0 -37
- package/esm2022/transforms/selection.mjs +0 -26
- package/esm2022/transforms/theme.mjs +0 -8
- package/esm2022/transforms/viewport.mjs +0 -8
- package/esm2022/transforms/z-index.mjs +0 -20
- package/esm2022/utils/angle.mjs +0 -164
- package/esm2022/utils/board.mjs +0 -18
- package/esm2022/utils/clipboard/clipboard.mjs +0 -40
- package/esm2022/utils/clipboard/common.mjs +0 -82
- package/esm2022/utils/clipboard/data-transfer.mjs +0 -33
- package/esm2022/utils/clipboard/index.mjs +0 -3
- package/esm2022/utils/clipboard/navigator-clipboard.mjs +0 -71
- package/esm2022/utils/clipboard/types.mjs +0 -13
- package/esm2022/utils/common.mjs +0 -79
- package/esm2022/utils/debug.mjs +0 -91
- package/esm2022/utils/dnd.mjs +0 -8
- package/esm2022/utils/dom/common.mjs +0 -75
- package/esm2022/utils/dom/environment.mjs +0 -2
- package/esm2022/utils/dom/foreign.mjs +0 -26
- package/esm2022/utils/dom/index.mjs +0 -4
- package/esm2022/utils/drawing/arrow.mjs +0 -23
- package/esm2022/utils/drawing/circle.mjs +0 -4
- package/esm2022/utils/drawing/line.mjs +0 -47
- package/esm2022/utils/drawing/rectangle.mjs +0 -36
- package/esm2022/utils/element.mjs +0 -90
- package/esm2022/utils/environment.mjs +0 -14
- package/esm2022/utils/fragment.mjs +0 -27
- package/esm2022/utils/group.mjs +0 -239
- package/esm2022/utils/helper.mjs +0 -68
- package/esm2022/utils/history.mjs +0 -96
- package/esm2022/utils/hotkeys.mjs +0 -109
- package/esm2022/utils/id-creator.mjs +0 -11
- package/esm2022/utils/index.mjs +0 -35
- package/esm2022/utils/iterable.mjs +0 -32
- package/esm2022/utils/math.mjs +0 -480
- package/esm2022/utils/mobile.mjs +0 -6
- package/esm2022/utils/moving-element.mjs +0 -17
- package/esm2022/utils/pointer.mjs +0 -13
- package/esm2022/utils/position.mjs +0 -9
- package/esm2022/utils/selected-element.mjs +0 -145
- package/esm2022/utils/selection.mjs +0 -151
- package/esm2022/utils/snap/snap-moving.mjs +0 -199
- package/esm2022/utils/snap/snap.mjs +0 -211
- package/esm2022/utils/to-image.mjs +0 -204
- package/esm2022/utils/to-point.mjs +0 -74
- package/esm2022/utils/tree.mjs +0 -22
- package/esm2022/utils/viewport.mjs +0 -227
- package/esm2022/utils/weak-maps.mjs +0 -27
- package/esm2022/utils/z-index.mjs +0 -166
|
@@ -1,614 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license
|
|
3
|
-
* Copyright Google LLC All Rights Reserved.
|
|
4
|
-
*
|
|
5
|
-
* Use of this source code is governed by an MIT-style license that can be
|
|
6
|
-
* found in the LICENSE file at https://angular.io/license
|
|
7
|
-
*/
|
|
8
|
-
import { isListLikeIterable, iterateListLike } from '../utils/iterable';
|
|
9
|
-
const trackByIdentity = (index, item) => item;
|
|
10
|
-
export class DefaultIterableDiffer {
|
|
11
|
-
constructor(trackByFn) {
|
|
12
|
-
this.length = 0;
|
|
13
|
-
// Keeps track of the used records at any point in time (during & across `_check()` calls)
|
|
14
|
-
this._linkedRecords = null;
|
|
15
|
-
// Keeps track of the removed records at any point in time during `_check()` calls.
|
|
16
|
-
this._unlinkedRecords = null;
|
|
17
|
-
this._previousItHead = null;
|
|
18
|
-
this._itHead = null;
|
|
19
|
-
this._itTail = null;
|
|
20
|
-
this._additionsHead = null;
|
|
21
|
-
this._additionsTail = null;
|
|
22
|
-
this._movesHead = null;
|
|
23
|
-
this._movesTail = null;
|
|
24
|
-
this._removalsHead = null;
|
|
25
|
-
this._removalsTail = null;
|
|
26
|
-
// Keeps track of records where custom track by is the same, but item identity has changed
|
|
27
|
-
this._identityChangesHead = null;
|
|
28
|
-
this._identityChangesTail = null;
|
|
29
|
-
this._trackByFn = trackByFn || trackByIdentity;
|
|
30
|
-
}
|
|
31
|
-
forEachItem(fn) {
|
|
32
|
-
let record;
|
|
33
|
-
for (record = this._itHead; record !== null; record = record._next) {
|
|
34
|
-
fn(record);
|
|
35
|
-
}
|
|
36
|
-
}
|
|
37
|
-
forEachOperation(fn) {
|
|
38
|
-
let nextIt = this._itHead;
|
|
39
|
-
let nextRemove = this._removalsHead;
|
|
40
|
-
let addRemoveOffset = 0;
|
|
41
|
-
let moveOffsets = null;
|
|
42
|
-
while (nextIt || nextRemove) {
|
|
43
|
-
// Figure out which is the next record to process
|
|
44
|
-
// Order: remove, add, move
|
|
45
|
-
const record = !nextRemove ||
|
|
46
|
-
nextIt &&
|
|
47
|
-
nextIt.currentIndex <
|
|
48
|
-
getPreviousIndex(nextRemove, addRemoveOffset, moveOffsets) ?
|
|
49
|
-
nextIt :
|
|
50
|
-
nextRemove;
|
|
51
|
-
const adjPreviousIndex = getPreviousIndex(record, addRemoveOffset, moveOffsets);
|
|
52
|
-
const currentIndex = record.currentIndex;
|
|
53
|
-
// consume the item, and adjust the addRemoveOffset and update moveDistance if necessary
|
|
54
|
-
if (record === nextRemove) {
|
|
55
|
-
addRemoveOffset--;
|
|
56
|
-
nextRemove = nextRemove._nextRemoved;
|
|
57
|
-
}
|
|
58
|
-
else {
|
|
59
|
-
nextIt = nextIt._next;
|
|
60
|
-
if (record.previousIndex == null) {
|
|
61
|
-
addRemoveOffset++;
|
|
62
|
-
}
|
|
63
|
-
else {
|
|
64
|
-
// INVARIANT: currentIndex < previousIndex
|
|
65
|
-
if (!moveOffsets)
|
|
66
|
-
moveOffsets = [];
|
|
67
|
-
const localMovePreviousIndex = adjPreviousIndex - addRemoveOffset;
|
|
68
|
-
const localCurrentIndex = currentIndex - addRemoveOffset;
|
|
69
|
-
if (localMovePreviousIndex != localCurrentIndex) {
|
|
70
|
-
for (let i = 0; i < localMovePreviousIndex; i++) {
|
|
71
|
-
const offset = i < moveOffsets.length ? moveOffsets[i] : (moveOffsets[i] = 0);
|
|
72
|
-
const index = offset + i;
|
|
73
|
-
if (localCurrentIndex <= index && index < localMovePreviousIndex) {
|
|
74
|
-
moveOffsets[i] = offset + 1;
|
|
75
|
-
}
|
|
76
|
-
}
|
|
77
|
-
const previousIndex = record.previousIndex;
|
|
78
|
-
moveOffsets[previousIndex] = localCurrentIndex - localMovePreviousIndex;
|
|
79
|
-
}
|
|
80
|
-
}
|
|
81
|
-
}
|
|
82
|
-
if (adjPreviousIndex !== currentIndex) {
|
|
83
|
-
fn(record, adjPreviousIndex, currentIndex);
|
|
84
|
-
}
|
|
85
|
-
}
|
|
86
|
-
}
|
|
87
|
-
forEachPreviousItem(fn) {
|
|
88
|
-
let record;
|
|
89
|
-
for (record = this._previousItHead; record !== null; record = record._nextPrevious) {
|
|
90
|
-
fn(record);
|
|
91
|
-
}
|
|
92
|
-
}
|
|
93
|
-
forEachAddedItem(fn) {
|
|
94
|
-
let record;
|
|
95
|
-
for (record = this._additionsHead; record !== null; record = record._nextAdded) {
|
|
96
|
-
fn(record);
|
|
97
|
-
}
|
|
98
|
-
}
|
|
99
|
-
forEachMovedItem(fn) {
|
|
100
|
-
let record;
|
|
101
|
-
for (record = this._movesHead; record !== null; record = record._nextMoved) {
|
|
102
|
-
fn(record);
|
|
103
|
-
}
|
|
104
|
-
}
|
|
105
|
-
forEachRemovedItem(fn) {
|
|
106
|
-
let record;
|
|
107
|
-
for (record = this._removalsHead; record !== null; record = record._nextRemoved) {
|
|
108
|
-
fn(record);
|
|
109
|
-
}
|
|
110
|
-
}
|
|
111
|
-
forEachIdentityChange(fn) {
|
|
112
|
-
let record;
|
|
113
|
-
for (record = this._identityChangesHead; record !== null; record = record._nextIdentityChange) {
|
|
114
|
-
fn(record);
|
|
115
|
-
}
|
|
116
|
-
}
|
|
117
|
-
diff(collection) {
|
|
118
|
-
if (collection == null)
|
|
119
|
-
collection = [];
|
|
120
|
-
if (!isListLikeIterable(collection)) {
|
|
121
|
-
throw new Error('Exception: Error trying to diff. Only arrays and iterables are allowed');
|
|
122
|
-
}
|
|
123
|
-
if (this.check(collection)) {
|
|
124
|
-
return this;
|
|
125
|
-
}
|
|
126
|
-
else {
|
|
127
|
-
return null;
|
|
128
|
-
}
|
|
129
|
-
}
|
|
130
|
-
onDestroy() { }
|
|
131
|
-
check(collection) {
|
|
132
|
-
this._reset();
|
|
133
|
-
let record = this._itHead;
|
|
134
|
-
let mayBeDirty = false;
|
|
135
|
-
let index;
|
|
136
|
-
let item;
|
|
137
|
-
let itemTrackBy;
|
|
138
|
-
if (Array.isArray(collection)) {
|
|
139
|
-
this.length = collection.length;
|
|
140
|
-
for (let index = 0; index < this.length; index++) {
|
|
141
|
-
item = collection[index];
|
|
142
|
-
itemTrackBy = this._trackByFn(index, item);
|
|
143
|
-
if (record === null || !Object.is(record.trackById, itemTrackBy)) {
|
|
144
|
-
record = this._mismatch(record, item, itemTrackBy, index);
|
|
145
|
-
mayBeDirty = true;
|
|
146
|
-
}
|
|
147
|
-
else {
|
|
148
|
-
if (mayBeDirty) {
|
|
149
|
-
record = this._verifyReinsertion(record, item, itemTrackBy, index);
|
|
150
|
-
}
|
|
151
|
-
if (!Object.is(record.item, item))
|
|
152
|
-
this._addIdentityChange(record, item);
|
|
153
|
-
}
|
|
154
|
-
record = record._next;
|
|
155
|
-
}
|
|
156
|
-
}
|
|
157
|
-
else {
|
|
158
|
-
index = 0;
|
|
159
|
-
iterateListLike(collection, (item) => {
|
|
160
|
-
itemTrackBy = this._trackByFn(index, item);
|
|
161
|
-
if (record === null || !Object.is(record.trackById, itemTrackBy)) {
|
|
162
|
-
record = this._mismatch(record, item, itemTrackBy, index);
|
|
163
|
-
mayBeDirty = true;
|
|
164
|
-
}
|
|
165
|
-
else {
|
|
166
|
-
if (mayBeDirty) {
|
|
167
|
-
record = this._verifyReinsertion(record, item, itemTrackBy, index);
|
|
168
|
-
}
|
|
169
|
-
if (!Object.is(record.item, item))
|
|
170
|
-
this._addIdentityChange(record, item);
|
|
171
|
-
}
|
|
172
|
-
record = record._next;
|
|
173
|
-
index++;
|
|
174
|
-
});
|
|
175
|
-
this.length = index;
|
|
176
|
-
}
|
|
177
|
-
this._truncate(record);
|
|
178
|
-
this.collection = collection;
|
|
179
|
-
return this.isDirty;
|
|
180
|
-
}
|
|
181
|
-
/* CollectionChanges is considered dirty if it has any additions, moves, removals, or identity
|
|
182
|
-
* changes.
|
|
183
|
-
*/
|
|
184
|
-
get isDirty() {
|
|
185
|
-
return this._additionsHead !== null || this._movesHead !== null ||
|
|
186
|
-
this._removalsHead !== null || this._identityChangesHead !== null;
|
|
187
|
-
}
|
|
188
|
-
/**
|
|
189
|
-
* Reset the state of the change objects to show no changes. This means set previousKey to
|
|
190
|
-
* currentKey, and clear all of the queues (additions, moves, removals).
|
|
191
|
-
* Set the previousIndexes of moved and added items to their currentIndexes
|
|
192
|
-
* Reset the list of additions, moves and removals
|
|
193
|
-
*
|
|
194
|
-
* @internal
|
|
195
|
-
*/
|
|
196
|
-
_reset() {
|
|
197
|
-
if (this.isDirty) {
|
|
198
|
-
let record;
|
|
199
|
-
for (record = this._previousItHead = this._itHead; record !== null; record = record._next) {
|
|
200
|
-
record._nextPrevious = record._next;
|
|
201
|
-
}
|
|
202
|
-
for (record = this._additionsHead; record !== null; record = record._nextAdded) {
|
|
203
|
-
record.previousIndex = record.currentIndex;
|
|
204
|
-
}
|
|
205
|
-
this._additionsHead = this._additionsTail = null;
|
|
206
|
-
for (record = this._movesHead; record !== null; record = record._nextMoved) {
|
|
207
|
-
record.previousIndex = record.currentIndex;
|
|
208
|
-
}
|
|
209
|
-
this._movesHead = this._movesTail = null;
|
|
210
|
-
this._removalsHead = this._removalsTail = null;
|
|
211
|
-
this._identityChangesHead = this._identityChangesTail = null;
|
|
212
|
-
}
|
|
213
|
-
}
|
|
214
|
-
/**
|
|
215
|
-
* This is the core function which handles differences between collections.
|
|
216
|
-
*
|
|
217
|
-
* - `record` is the record which we saw at this position last time. If null then it is a new
|
|
218
|
-
* item.
|
|
219
|
-
* - `item` is the current item in the collection
|
|
220
|
-
* - `index` is the position of the item in the collection
|
|
221
|
-
*
|
|
222
|
-
* @internal
|
|
223
|
-
*/
|
|
224
|
-
_mismatch(record, item, itemTrackBy, index) {
|
|
225
|
-
// The previous record after which we will append the current one.
|
|
226
|
-
let previousRecord;
|
|
227
|
-
if (record === null) {
|
|
228
|
-
previousRecord = this._itTail;
|
|
229
|
-
}
|
|
230
|
-
else {
|
|
231
|
-
previousRecord = record._prev;
|
|
232
|
-
// Remove the record from the collection since we know it does not match the item.
|
|
233
|
-
this._remove(record);
|
|
234
|
-
}
|
|
235
|
-
// See if we have evicted the item, which used to be at some anterior position of _itHead list.
|
|
236
|
-
record = this._unlinkedRecords === null ? null : this._unlinkedRecords.get(itemTrackBy, null);
|
|
237
|
-
if (record !== null) {
|
|
238
|
-
// It is an item which we have evicted earlier: reinsert it back into the list.
|
|
239
|
-
// But first we need to check if identity changed, so we can update in view if necessary.
|
|
240
|
-
if (!Object.is(record.item, item))
|
|
241
|
-
this._addIdentityChange(record, item);
|
|
242
|
-
this._reinsertAfter(record, previousRecord, index);
|
|
243
|
-
}
|
|
244
|
-
else {
|
|
245
|
-
// Attempt to see if the item is at some posterior position of _itHead list.
|
|
246
|
-
record = this._linkedRecords === null ? null : this._linkedRecords.get(itemTrackBy, index);
|
|
247
|
-
if (record !== null) {
|
|
248
|
-
// We have the item in _itHead at/after `index` position. We need to move it forward in the
|
|
249
|
-
// collection.
|
|
250
|
-
// But first we need to check if identity changed, so we can update in view if necessary.
|
|
251
|
-
if (!Object.is(record.item, item))
|
|
252
|
-
this._addIdentityChange(record, item);
|
|
253
|
-
this._moveAfter(record, previousRecord, index);
|
|
254
|
-
}
|
|
255
|
-
else {
|
|
256
|
-
// It is a new item: add it.
|
|
257
|
-
record =
|
|
258
|
-
this._addAfter(new IterableChangeRecord_(item, itemTrackBy), previousRecord, index);
|
|
259
|
-
}
|
|
260
|
-
}
|
|
261
|
-
return record;
|
|
262
|
-
}
|
|
263
|
-
/**
|
|
264
|
-
* This check is only needed if an array contains duplicates. (Short circuit of nothing dirty)
|
|
265
|
-
*
|
|
266
|
-
* Use case: `[a, a]` => `[b, a, a]`
|
|
267
|
-
*
|
|
268
|
-
* If we did not have this check then the insertion of `b` would:
|
|
269
|
-
* 1) evict first `a`
|
|
270
|
-
* 2) insert `b` at `0` index.
|
|
271
|
-
* 3) leave `a` at index `1` as is. <-- this is wrong!
|
|
272
|
-
* 3) reinsert `a` at index 2. <-- this is wrong!
|
|
273
|
-
*
|
|
274
|
-
* The correct behavior is:
|
|
275
|
-
* 1) evict first `a`
|
|
276
|
-
* 2) insert `b` at `0` index.
|
|
277
|
-
* 3) reinsert `a` at index 1.
|
|
278
|
-
* 3) move `a` at from `1` to `2`.
|
|
279
|
-
*
|
|
280
|
-
*
|
|
281
|
-
* Double check that we have not evicted a duplicate item. We need to check if the item type may
|
|
282
|
-
* have already been removed:
|
|
283
|
-
* The insertion of b will evict the first 'a'. If we don't reinsert it now it will be reinserted
|
|
284
|
-
* at the end. Which will show up as the two 'a's switching position. This is incorrect, since a
|
|
285
|
-
* better way to think of it is as insert of 'b' rather then switch 'a' with 'b' and then add 'a'
|
|
286
|
-
* at the end.
|
|
287
|
-
*
|
|
288
|
-
* @internal
|
|
289
|
-
*/
|
|
290
|
-
_verifyReinsertion(record, item, itemTrackBy, index) {
|
|
291
|
-
let reinsertRecord = this._unlinkedRecords === null ? null : this._unlinkedRecords.get(itemTrackBy, null);
|
|
292
|
-
if (reinsertRecord !== null) {
|
|
293
|
-
record = this._reinsertAfter(reinsertRecord, record._prev, index);
|
|
294
|
-
}
|
|
295
|
-
else if (record.currentIndex != index) {
|
|
296
|
-
record.currentIndex = index;
|
|
297
|
-
this._addToMoves(record, index);
|
|
298
|
-
}
|
|
299
|
-
return record;
|
|
300
|
-
}
|
|
301
|
-
/**
|
|
302
|
-
* Get rid of any excess {@link IterableChangeRecord_}s from the previous collection
|
|
303
|
-
*
|
|
304
|
-
* - `record` The first excess {@link IterableChangeRecord_}.
|
|
305
|
-
*
|
|
306
|
-
* @internal
|
|
307
|
-
*/
|
|
308
|
-
_truncate(record) {
|
|
309
|
-
// Anything after that needs to be removed;
|
|
310
|
-
while (record !== null) {
|
|
311
|
-
const nextRecord = record._next;
|
|
312
|
-
this._addToRemovals(this._unlink(record));
|
|
313
|
-
record = nextRecord;
|
|
314
|
-
}
|
|
315
|
-
if (this._unlinkedRecords !== null) {
|
|
316
|
-
this._unlinkedRecords.clear();
|
|
317
|
-
}
|
|
318
|
-
if (this._additionsTail !== null) {
|
|
319
|
-
this._additionsTail._nextAdded = null;
|
|
320
|
-
}
|
|
321
|
-
if (this._movesTail !== null) {
|
|
322
|
-
this._movesTail._nextMoved = null;
|
|
323
|
-
}
|
|
324
|
-
if (this._itTail !== null) {
|
|
325
|
-
this._itTail._next = null;
|
|
326
|
-
}
|
|
327
|
-
if (this._removalsTail !== null) {
|
|
328
|
-
this._removalsTail._nextRemoved = null;
|
|
329
|
-
}
|
|
330
|
-
if (this._identityChangesTail !== null) {
|
|
331
|
-
this._identityChangesTail._nextIdentityChange = null;
|
|
332
|
-
}
|
|
333
|
-
}
|
|
334
|
-
/** @internal */
|
|
335
|
-
_reinsertAfter(record, prevRecord, index) {
|
|
336
|
-
if (this._unlinkedRecords !== null) {
|
|
337
|
-
this._unlinkedRecords.remove(record);
|
|
338
|
-
}
|
|
339
|
-
const prev = record._prevRemoved;
|
|
340
|
-
const next = record._nextRemoved;
|
|
341
|
-
if (prev === null) {
|
|
342
|
-
this._removalsHead = next;
|
|
343
|
-
}
|
|
344
|
-
else {
|
|
345
|
-
prev._nextRemoved = next;
|
|
346
|
-
}
|
|
347
|
-
if (next === null) {
|
|
348
|
-
this._removalsTail = prev;
|
|
349
|
-
}
|
|
350
|
-
else {
|
|
351
|
-
next._prevRemoved = prev;
|
|
352
|
-
}
|
|
353
|
-
this._insertAfter(record, prevRecord, index);
|
|
354
|
-
this._addToMoves(record, index);
|
|
355
|
-
return record;
|
|
356
|
-
}
|
|
357
|
-
/** @internal */
|
|
358
|
-
_moveAfter(record, prevRecord, index) {
|
|
359
|
-
this._unlink(record);
|
|
360
|
-
this._insertAfter(record, prevRecord, index);
|
|
361
|
-
this._addToMoves(record, index);
|
|
362
|
-
return record;
|
|
363
|
-
}
|
|
364
|
-
/** @internal */
|
|
365
|
-
_addAfter(record, prevRecord, index) {
|
|
366
|
-
this._insertAfter(record, prevRecord, index);
|
|
367
|
-
if (this._additionsTail === null) {
|
|
368
|
-
// assert(this._additionsHead === null);
|
|
369
|
-
this._additionsTail = this._additionsHead = record;
|
|
370
|
-
}
|
|
371
|
-
else {
|
|
372
|
-
// assert(_additionsTail._nextAdded === null);
|
|
373
|
-
// assert(record._nextAdded === null);
|
|
374
|
-
this._additionsTail = this._additionsTail._nextAdded = record;
|
|
375
|
-
}
|
|
376
|
-
return record;
|
|
377
|
-
}
|
|
378
|
-
/** @internal */
|
|
379
|
-
_insertAfter(record, prevRecord, index) {
|
|
380
|
-
const next = prevRecord === null ? this._itHead : prevRecord._next;
|
|
381
|
-
record._next = next;
|
|
382
|
-
record._prev = prevRecord;
|
|
383
|
-
if (next === null) {
|
|
384
|
-
this._itTail = record;
|
|
385
|
-
}
|
|
386
|
-
else {
|
|
387
|
-
next._prev = record;
|
|
388
|
-
}
|
|
389
|
-
if (prevRecord === null) {
|
|
390
|
-
this._itHead = record;
|
|
391
|
-
}
|
|
392
|
-
else {
|
|
393
|
-
prevRecord._next = record;
|
|
394
|
-
}
|
|
395
|
-
if (this._linkedRecords === null) {
|
|
396
|
-
this._linkedRecords = new _DuplicateMap();
|
|
397
|
-
}
|
|
398
|
-
this._linkedRecords.put(record);
|
|
399
|
-
record.currentIndex = index;
|
|
400
|
-
return record;
|
|
401
|
-
}
|
|
402
|
-
/** @internal */
|
|
403
|
-
_remove(record) {
|
|
404
|
-
return this._addToRemovals(this._unlink(record));
|
|
405
|
-
}
|
|
406
|
-
/** @internal */
|
|
407
|
-
_unlink(record) {
|
|
408
|
-
if (this._linkedRecords !== null) {
|
|
409
|
-
this._linkedRecords.remove(record);
|
|
410
|
-
}
|
|
411
|
-
const prev = record._prev;
|
|
412
|
-
const next = record._next;
|
|
413
|
-
if (prev === null) {
|
|
414
|
-
this._itHead = next;
|
|
415
|
-
}
|
|
416
|
-
else {
|
|
417
|
-
prev._next = next;
|
|
418
|
-
}
|
|
419
|
-
if (next === null) {
|
|
420
|
-
this._itTail = prev;
|
|
421
|
-
}
|
|
422
|
-
else {
|
|
423
|
-
next._prev = prev;
|
|
424
|
-
}
|
|
425
|
-
return record;
|
|
426
|
-
}
|
|
427
|
-
/** @internal */
|
|
428
|
-
_addToMoves(record, toIndex) {
|
|
429
|
-
if (record.previousIndex === toIndex) {
|
|
430
|
-
return record;
|
|
431
|
-
}
|
|
432
|
-
if (this._movesTail === null) {
|
|
433
|
-
this._movesTail = this._movesHead = record;
|
|
434
|
-
}
|
|
435
|
-
else {
|
|
436
|
-
this._movesTail = this._movesTail._nextMoved = record;
|
|
437
|
-
}
|
|
438
|
-
return record;
|
|
439
|
-
}
|
|
440
|
-
_addToRemovals(record) {
|
|
441
|
-
if (this._unlinkedRecords === null) {
|
|
442
|
-
this._unlinkedRecords = new _DuplicateMap();
|
|
443
|
-
}
|
|
444
|
-
this._unlinkedRecords.put(record);
|
|
445
|
-
record.currentIndex = null;
|
|
446
|
-
record._nextRemoved = null;
|
|
447
|
-
if (this._removalsTail === null) {
|
|
448
|
-
this._removalsTail = this._removalsHead = record;
|
|
449
|
-
record._prevRemoved = null;
|
|
450
|
-
}
|
|
451
|
-
else {
|
|
452
|
-
record._prevRemoved = this._removalsTail;
|
|
453
|
-
this._removalsTail = this._removalsTail._nextRemoved = record;
|
|
454
|
-
}
|
|
455
|
-
return record;
|
|
456
|
-
}
|
|
457
|
-
/** @internal */
|
|
458
|
-
_addIdentityChange(record, item) {
|
|
459
|
-
record.item = item;
|
|
460
|
-
if (this._identityChangesTail === null) {
|
|
461
|
-
this._identityChangesTail = this._identityChangesHead = record;
|
|
462
|
-
}
|
|
463
|
-
else {
|
|
464
|
-
this._identityChangesTail = this._identityChangesTail._nextIdentityChange = record;
|
|
465
|
-
}
|
|
466
|
-
return record;
|
|
467
|
-
}
|
|
468
|
-
}
|
|
469
|
-
export class IterableChangeRecord_ {
|
|
470
|
-
constructor(item, trackById) {
|
|
471
|
-
this.item = item;
|
|
472
|
-
this.trackById = trackById;
|
|
473
|
-
this.currentIndex = null;
|
|
474
|
-
this.previousIndex = null;
|
|
475
|
-
/** @internal */
|
|
476
|
-
this._nextPrevious = null;
|
|
477
|
-
/** @internal */
|
|
478
|
-
this._prev = null;
|
|
479
|
-
/** @internal */
|
|
480
|
-
this._next = null;
|
|
481
|
-
/** @internal */
|
|
482
|
-
this._prevDup = null;
|
|
483
|
-
/** @internal */
|
|
484
|
-
this._nextDup = null;
|
|
485
|
-
/** @internal */
|
|
486
|
-
this._prevRemoved = null;
|
|
487
|
-
/** @internal */
|
|
488
|
-
this._nextRemoved = null;
|
|
489
|
-
/** @internal */
|
|
490
|
-
this._nextAdded = null;
|
|
491
|
-
/** @internal */
|
|
492
|
-
this._nextMoved = null;
|
|
493
|
-
/** @internal */
|
|
494
|
-
this._nextIdentityChange = null;
|
|
495
|
-
}
|
|
496
|
-
}
|
|
497
|
-
// A linked list of IterableChangeRecords with the same IterableChangeRecord_.item
|
|
498
|
-
class _DuplicateItemRecordList {
|
|
499
|
-
constructor() {
|
|
500
|
-
/** @internal */
|
|
501
|
-
this._head = null;
|
|
502
|
-
/** @internal */
|
|
503
|
-
this._tail = null;
|
|
504
|
-
}
|
|
505
|
-
/**
|
|
506
|
-
* Append the record to the list of duplicates.
|
|
507
|
-
*
|
|
508
|
-
* Note: by design all records in the list of duplicates hold the same value in record.item.
|
|
509
|
-
*/
|
|
510
|
-
add(record) {
|
|
511
|
-
if (this._head === null) {
|
|
512
|
-
this._head = this._tail = record;
|
|
513
|
-
record._nextDup = null;
|
|
514
|
-
record._prevDup = null;
|
|
515
|
-
}
|
|
516
|
-
else {
|
|
517
|
-
this._tail._nextDup = record;
|
|
518
|
-
record._prevDup = this._tail;
|
|
519
|
-
record._nextDup = null;
|
|
520
|
-
this._tail = record;
|
|
521
|
-
}
|
|
522
|
-
}
|
|
523
|
-
// Returns a IterableChangeRecord_ having IterableChangeRecord_.trackById == trackById and
|
|
524
|
-
// IterableChangeRecord_.currentIndex >= atOrAfterIndex
|
|
525
|
-
get(trackById, atOrAfterIndex) {
|
|
526
|
-
let record;
|
|
527
|
-
for (record = this._head; record !== null; record = record._nextDup) {
|
|
528
|
-
if ((atOrAfterIndex === null || atOrAfterIndex <= record.currentIndex) &&
|
|
529
|
-
Object.is(record.trackById, trackById)) {
|
|
530
|
-
return record;
|
|
531
|
-
}
|
|
532
|
-
}
|
|
533
|
-
return null;
|
|
534
|
-
}
|
|
535
|
-
/**
|
|
536
|
-
* Remove one {@link IterableChangeRecord_} from the list of duplicates.
|
|
537
|
-
*
|
|
538
|
-
* Returns whether the list of duplicates is empty.
|
|
539
|
-
*/
|
|
540
|
-
remove(record) {
|
|
541
|
-
const prev = record._prevDup;
|
|
542
|
-
const next = record._nextDup;
|
|
543
|
-
if (prev === null) {
|
|
544
|
-
this._head = next;
|
|
545
|
-
}
|
|
546
|
-
else {
|
|
547
|
-
prev._nextDup = next;
|
|
548
|
-
}
|
|
549
|
-
if (next === null) {
|
|
550
|
-
this._tail = prev;
|
|
551
|
-
}
|
|
552
|
-
else {
|
|
553
|
-
next._prevDup = prev;
|
|
554
|
-
}
|
|
555
|
-
return this._head === null;
|
|
556
|
-
}
|
|
557
|
-
}
|
|
558
|
-
class _DuplicateMap {
|
|
559
|
-
constructor() {
|
|
560
|
-
this.map = new Map();
|
|
561
|
-
}
|
|
562
|
-
put(record) {
|
|
563
|
-
const key = record.trackById;
|
|
564
|
-
let duplicates = this.map.get(key);
|
|
565
|
-
if (!duplicates) {
|
|
566
|
-
duplicates = new _DuplicateItemRecordList();
|
|
567
|
-
this.map.set(key, duplicates);
|
|
568
|
-
}
|
|
569
|
-
duplicates.add(record);
|
|
570
|
-
}
|
|
571
|
-
/**
|
|
572
|
-
* Retrieve the `value` using key. Because the IterableChangeRecord_ value may be one which we
|
|
573
|
-
* have already iterated over, we use the `atOrAfterIndex` to pretend it is not there.
|
|
574
|
-
*
|
|
575
|
-
* Use case: `[a, b, c, a, a]` if we are at index `3` which is the second `a` then asking if we
|
|
576
|
-
* have any more `a`s needs to return the second `a`.
|
|
577
|
-
*/
|
|
578
|
-
get(trackById, atOrAfterIndex) {
|
|
579
|
-
const key = trackById;
|
|
580
|
-
const recordList = this.map.get(key);
|
|
581
|
-
return recordList ? recordList.get(trackById, atOrAfterIndex) : null;
|
|
582
|
-
}
|
|
583
|
-
/**
|
|
584
|
-
* Removes a {@link IterableChangeRecord_} from the list of duplicates.
|
|
585
|
-
*
|
|
586
|
-
* The list of duplicates also is removed from the map if it gets empty.
|
|
587
|
-
*/
|
|
588
|
-
remove(record) {
|
|
589
|
-
const key = record.trackById;
|
|
590
|
-
const recordList = this.map.get(key);
|
|
591
|
-
// Remove the list of duplicates when it gets empty
|
|
592
|
-
if (recordList.remove(record)) {
|
|
593
|
-
this.map.delete(key);
|
|
594
|
-
}
|
|
595
|
-
return record;
|
|
596
|
-
}
|
|
597
|
-
get isEmpty() {
|
|
598
|
-
return this.map.size === 0;
|
|
599
|
-
}
|
|
600
|
-
clear() {
|
|
601
|
-
this.map.clear();
|
|
602
|
-
}
|
|
603
|
-
}
|
|
604
|
-
function getPreviousIndex(item, addRemoveOffset, moveOffsets) {
|
|
605
|
-
const previousIndex = item.previousIndex;
|
|
606
|
-
if (previousIndex === null)
|
|
607
|
-
return previousIndex;
|
|
608
|
-
let moveOffset = 0;
|
|
609
|
-
if (moveOffsets && previousIndex < moveOffsets.length) {
|
|
610
|
-
moveOffset = moveOffsets[previousIndex];
|
|
611
|
-
}
|
|
612
|
-
return previousIndex + addRemoveOffset + moveOffset;
|
|
613
|
-
}
|
|
614
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"default_iterable_differ.js","sourceRoot":"","sources":["../../../../packages/core/src/differs/default_iterable_differ.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {isListLikeIterable, iterateListLike} from '../utils/iterable';\n\nimport {IterableChangeRecord, IterableChanges, IterableDiffer, CustomIterable, TrackByFunction} from './iterable_differs';\n\n\nconst trackByIdentity = (index: number, item: any) => item;\n\nexport type Writable<T> = {\n  -readonly[K in keyof T]: T[K];\n};\n\nexport class DefaultIterableDiffer<V> implements IterableDiffer<V>, IterableChanges<V> {\n  public readonly length: number = 0;\n  public readonly collection!: V[]|Iterable<V>|null;\n  // Keeps track of the used records at any point in time (during & across `_check()` calls)\n  private _linkedRecords: _DuplicateMap<V>|null = null;\n  // Keeps track of the removed records at any point in time during `_check()` calls.\n  private _unlinkedRecords: _DuplicateMap<V>|null = null;\n  private _previousItHead: IterableChangeRecord_<V>|null = null;\n  private _itHead: IterableChangeRecord_<V>|null = null;\n  private _itTail: IterableChangeRecord_<V>|null = null;\n  private _additionsHead: IterableChangeRecord_<V>|null = null;\n  private _additionsTail: IterableChangeRecord_<V>|null = null;\n  private _movesHead: IterableChangeRecord_<V>|null = null;\n  private _movesTail: IterableChangeRecord_<V>|null = null;\n  private _removalsHead: IterableChangeRecord_<V>|null = null;\n  private _removalsTail: IterableChangeRecord_<V>|null = null;\n  // Keeps track of records where custom track by is the same, but item identity has changed\n  private _identityChangesHead: IterableChangeRecord_<V>|null = null;\n  private _identityChangesTail: IterableChangeRecord_<V>|null = null;\n  private _trackByFn: TrackByFunction<V>;\n\n  constructor(trackByFn?: TrackByFunction<V>) {\n    this._trackByFn = trackByFn || trackByIdentity;\n  }\n\n  forEachItem(fn: (record: IterableChangeRecord_<V>) => void) {\n    let record: IterableChangeRecord_<V>|null;\n    for (record = this._itHead; record !== null; record = record._next) {\n      fn(record);\n    }\n  }\n\n  forEachOperation(\n      fn: (item: IterableChangeRecord<V>, previousIndex: number|null, currentIndex: number|null) =>\n          void) {\n    let nextIt = this._itHead;\n    let nextRemove = this._removalsHead;\n    let addRemoveOffset = 0;\n    let moveOffsets: number[]|null = null;\n    while (nextIt || nextRemove) {\n      // Figure out which is the next record to process\n      // Order: remove, add, move\n      const record: IterableChangeRecord<V> = !nextRemove ||\n              nextIt &&\n                  nextIt.currentIndex! <\n                      getPreviousIndex(nextRemove, addRemoveOffset, moveOffsets) ?\n          nextIt! :\n          nextRemove;\n      const adjPreviousIndex = getPreviousIndex(record, addRemoveOffset, moveOffsets);\n      const currentIndex = record.currentIndex;\n\n      // consume the item, and adjust the addRemoveOffset and update moveDistance if necessary\n      if (record === nextRemove) {\n        addRemoveOffset--;\n        nextRemove = nextRemove._nextRemoved;\n      } else {\n        nextIt = nextIt!._next;\n        if (record.previousIndex == null) {\n          addRemoveOffset++;\n        } else {\n          // INVARIANT:  currentIndex < previousIndex\n          if (!moveOffsets) moveOffsets = [];\n          const localMovePreviousIndex = adjPreviousIndex - addRemoveOffset;\n          const localCurrentIndex = currentIndex! - addRemoveOffset;\n          if (localMovePreviousIndex != localCurrentIndex) {\n            for (let i = 0; i < localMovePreviousIndex; i++) {\n              const offset = i < moveOffsets.length ? moveOffsets[i] : (moveOffsets[i] = 0);\n              const index = offset + i;\n              if (localCurrentIndex <= index && index < localMovePreviousIndex) {\n                moveOffsets[i] = offset + 1;\n              }\n            }\n            const previousIndex = record.previousIndex;\n            moveOffsets[previousIndex] = localCurrentIndex - localMovePreviousIndex;\n          }\n        }\n      }\n\n      if (adjPreviousIndex !== currentIndex) {\n        fn(record, adjPreviousIndex, currentIndex);\n      }\n    }\n  }\n\n  forEachPreviousItem(fn: (record: IterableChangeRecord_<V>) => void) {\n    let record: IterableChangeRecord_<V>|null;\n    for (record = this._previousItHead; record !== null; record = record._nextPrevious) {\n      fn(record);\n    }\n  }\n\n  forEachAddedItem(fn: (record: IterableChangeRecord_<V>) => void) {\n    let record: IterableChangeRecord_<V>|null;\n    for (record = this._additionsHead; record !== null; record = record._nextAdded) {\n      fn(record);\n    }\n  }\n\n  forEachMovedItem(fn: (record: IterableChangeRecord_<V>) => void) {\n    let record: IterableChangeRecord_<V>|null;\n    for (record = this._movesHead; record !== null; record = record._nextMoved) {\n      fn(record);\n    }\n  }\n\n  forEachRemovedItem(fn: (record: IterableChangeRecord_<V>) => void) {\n    let record: IterableChangeRecord_<V>|null;\n    for (record = this._removalsHead; record !== null; record = record._nextRemoved) {\n      fn(record);\n    }\n  }\n\n  forEachIdentityChange(fn: (record: IterableChangeRecord_<V>) => void) {\n    let record: IterableChangeRecord_<V>|null;\n    for (record = this._identityChangesHead; record !== null; record = record._nextIdentityChange) {\n      fn(record);\n    }\n  }\n\n  diff(collection: CustomIterable<V>|null|undefined): DefaultIterableDiffer<V>|null {\n    if (collection == null) collection = [];\n    if (!isListLikeIterable(collection)) {\n      throw new Error('Exception: Error trying to diff. Only arrays and iterables are allowed');\n    }\n\n    if (this.check(collection)) {\n      return this;\n    } else {\n      return null;\n    }\n  }\n\n  onDestroy() {}\n\n  check(collection: CustomIterable<V>): boolean {\n    this._reset();\n\n    let record: IterableChangeRecord_<V>|null = this._itHead;\n    let mayBeDirty: boolean = false;\n    let index: number;\n    let item: V;\n    let itemTrackBy: any;\n    if (Array.isArray(collection)) {\n      (this as Writable<this>).length = collection.length;\n\n      for (let index = 0; index < this.length; index++) {\n        item = collection[index];\n        itemTrackBy = this._trackByFn(index, item);\n        if (record === null || !Object.is(record.trackById, itemTrackBy)) {\n          record = this._mismatch(record, item, itemTrackBy, index);\n          mayBeDirty = true;\n        } else {\n          if (mayBeDirty) {\n            record = this._verifyReinsertion(record, item, itemTrackBy, index);\n          }\n          if (!Object.is(record.item, item)) this._addIdentityChange(record, item);\n        }\n\n        record = record._next;\n      }\n    } else {\n      index = 0;\n      iterateListLike(collection, (item: V) => {\n        itemTrackBy = this._trackByFn(index, item);\n        if (record === null || !Object.is(record.trackById, itemTrackBy)) {\n          record = this._mismatch(record, item, itemTrackBy, index);\n          mayBeDirty = true;\n        } else {\n          if (mayBeDirty) {\n            record = this._verifyReinsertion(record, item, itemTrackBy, index);\n          }\n          if (!Object.is(record.item, item)) this._addIdentityChange(record, item);\n        }\n        record = record._next;\n        index++;\n      });\n      (this as Writable<this>).length = index;\n    }\n\n    this._truncate(record);\n    (this as Writable<this>).collection = collection;\n    return this.isDirty;\n  }\n\n  /* CollectionChanges is considered dirty if it has any additions, moves, removals, or identity\n   * changes.\n   */\n  get isDirty(): boolean {\n    return this._additionsHead !== null || this._movesHead !== null ||\n        this._removalsHead !== null || this._identityChangesHead !== null;\n  }\n\n  /**\n   * Reset the state of the change objects to show no changes. This means set previousKey to\n   * currentKey, and clear all of the queues (additions, moves, removals).\n   * Set the previousIndexes of moved and added items to their currentIndexes\n   * Reset the list of additions, moves and removals\n   *\n   * @internal\n   */\n  _reset() {\n    if (this.isDirty) {\n      let record: IterableChangeRecord_<V>|null;\n\n      for (record = this._previousItHead = this._itHead; record !== null; record = record._next) {\n        record._nextPrevious = record._next;\n      }\n\n      for (record = this._additionsHead; record !== null; record = record._nextAdded) {\n        record.previousIndex = record.currentIndex;\n      }\n      this._additionsHead = this._additionsTail = null;\n\n      for (record = this._movesHead; record !== null; record = record._nextMoved) {\n        record.previousIndex = record.currentIndex;\n      }\n      this._movesHead = this._movesTail = null;\n      this._removalsHead = this._removalsTail = null;\n      this._identityChangesHead = this._identityChangesTail = null;\n    }\n  }\n\n  /**\n   * This is the core function which handles differences between collections.\n   *\n   * - `record` is the record which we saw at this position last time. If null then it is a new\n   *   item.\n   * - `item` is the current item in the collection\n   * - `index` is the position of the item in the collection\n   *\n   * @internal\n   */\n  _mismatch(record: IterableChangeRecord_<V>|null, item: V, itemTrackBy: any, index: number):\n      IterableChangeRecord_<V> {\n    // The previous record after which we will append the current one.\n    let previousRecord: IterableChangeRecord_<V>|null;\n\n    if (record === null) {\n      previousRecord = this._itTail;\n    } else {\n      previousRecord = record._prev;\n      // Remove the record from the collection since we know it does not match the item.\n      this._remove(record);\n    }\n\n    // See if we have evicted the item, which used to be at some anterior position of _itHead list.\n    record = this._unlinkedRecords === null ? null : this._unlinkedRecords.get(itemTrackBy, null);\n    if (record !== null) {\n      // It is an item which we have evicted earlier: reinsert it back into the list.\n      // But first we need to check if identity changed, so we can update in view if necessary.\n      if (!Object.is(record.item, item)) this._addIdentityChange(record, item);\n\n      this._reinsertAfter(record, previousRecord, index);\n    } else {\n      // Attempt to see if the item is at some posterior position of _itHead list.\n      record = this._linkedRecords === null ? null : this._linkedRecords.get(itemTrackBy, index);\n      if (record !== null) {\n        // We have the item in _itHead at/after `index` position. We need to move it forward in the\n        // collection.\n        // But first we need to check if identity changed, so we can update in view if necessary.\n        if (!Object.is(record.item, item)) this._addIdentityChange(record, item);\n\n        this._moveAfter(record, previousRecord, index);\n      } else {\n        // It is a new item: add it.\n        record =\n            this._addAfter(new IterableChangeRecord_<V>(item, itemTrackBy), previousRecord, index);\n      }\n    }\n    return record;\n  }\n\n  /**\n   * This check is only needed if an array contains duplicates. (Short circuit of nothing dirty)\n   *\n   * Use case: `[a, a]` => `[b, a, a]`\n   *\n   * If we did not have this check then the insertion of `b` would:\n   *   1) evict first `a`\n   *   2) insert `b` at `0` index.\n   *   3) leave `a` at index `1` as is. <-- this is wrong!\n   *   3) reinsert `a` at index 2. <-- this is wrong!\n   *\n   * The correct behavior is:\n   *   1) evict first `a`\n   *   2) insert `b` at `0` index.\n   *   3) reinsert `a` at index 1.\n   *   3) move `a` at from `1` to `2`.\n   *\n   *\n   * Double check that we have not evicted a duplicate item. We need to check if the item type may\n   * have already been removed:\n   * The insertion of b will evict the first 'a'. If we don't reinsert it now it will be reinserted\n   * at the end. Which will show up as the two 'a's switching position. This is incorrect, since a\n   * better way to think of it is as insert of 'b' rather then switch 'a' with 'b' and then add 'a'\n   * at the end.\n   *\n   * @internal\n   */\n  _verifyReinsertion(record: IterableChangeRecord_<V>, item: V, itemTrackBy: any, index: number):\n      IterableChangeRecord_<V> {\n    let reinsertRecord: IterableChangeRecord_<V>|null =\n        this._unlinkedRecords === null ? null : this._unlinkedRecords.get(itemTrackBy, null);\n    if (reinsertRecord !== null) {\n      record = this._reinsertAfter(reinsertRecord, record._prev!, index);\n    } else if (record.currentIndex != index) {\n      record.currentIndex = index;\n      this._addToMoves(record, index);\n    }\n    return record;\n  }\n\n  /**\n   * Get rid of any excess {@link IterableChangeRecord_}s from the previous collection\n   *\n   * - `record` The first excess {@link IterableChangeRecord_}.\n   *\n   * @internal\n   */\n  _truncate(record: IterableChangeRecord_<V>|null) {\n    // Anything after that needs to be removed;\n    while (record !== null) {\n      const nextRecord: IterableChangeRecord_<V>|null = record._next;\n      this._addToRemovals(this._unlink(record));\n      record = nextRecord;\n    }\n    if (this._unlinkedRecords !== null) {\n      this._unlinkedRecords.clear();\n    }\n\n    if (this._additionsTail !== null) {\n      this._additionsTail._nextAdded = null;\n    }\n    if (this._movesTail !== null) {\n      this._movesTail._nextMoved = null;\n    }\n    if (this._itTail !== null) {\n      this._itTail._next = null;\n    }\n    if (this._removalsTail !== null) {\n      this._removalsTail._nextRemoved = null;\n    }\n    if (this._identityChangesTail !== null) {\n      this._identityChangesTail._nextIdentityChange = null;\n    }\n  }\n\n  /** @internal */\n  _reinsertAfter(\n      record: IterableChangeRecord_<V>, prevRecord: IterableChangeRecord_<V>|null,\n      index: number): IterableChangeRecord_<V> {\n    if (this._unlinkedRecords !== null) {\n      this._unlinkedRecords.remove(record);\n    }\n    const prev = record._prevRemoved;\n    const next = record._nextRemoved;\n\n    if (prev === null) {\n      this._removalsHead = next;\n    } else {\n      prev._nextRemoved = next;\n    }\n    if (next === null) {\n      this._removalsTail = prev;\n    } else {\n      next._prevRemoved = prev;\n    }\n\n    this._insertAfter(record, prevRecord, index);\n    this._addToMoves(record, index);\n    return record;\n  }\n\n  /** @internal */\n  _moveAfter(\n      record: IterableChangeRecord_<V>, prevRecord: IterableChangeRecord_<V>|null,\n      index: number): IterableChangeRecord_<V> {\n    this._unlink(record);\n    this._insertAfter(record, prevRecord, index);\n    this._addToMoves(record, index);\n    return record;\n  }\n\n  /** @internal */\n  _addAfter(\n      record: IterableChangeRecord_<V>, prevRecord: IterableChangeRecord_<V>|null,\n      index: number): IterableChangeRecord_<V> {\n    this._insertAfter(record, prevRecord, index);\n\n    if (this._additionsTail === null) {\n      // assert(this._additionsHead === null);\n      this._additionsTail = this._additionsHead = record;\n    } else {\n      // assert(_additionsTail._nextAdded === null);\n      // assert(record._nextAdded === null);\n      this._additionsTail = this._additionsTail._nextAdded = record;\n    }\n    return record;\n  }\n\n  /** @internal */\n  _insertAfter(\n      record: IterableChangeRecord_<V>, prevRecord: IterableChangeRecord_<V>|null,\n      index: number): IterableChangeRecord_<V> {\n\n    const next: IterableChangeRecord_<V>|null =\n        prevRecord === null ? this._itHead : prevRecord._next;\n    record._next = next;\n    record._prev = prevRecord;\n    if (next === null) {\n      this._itTail = record;\n    } else {\n      next._prev = record;\n    }\n    if (prevRecord === null) {\n      this._itHead = record;\n    } else {\n      prevRecord._next = record;\n    }\n\n    if (this._linkedRecords === null) {\n      this._linkedRecords = new _DuplicateMap<V>();\n    }\n    this._linkedRecords.put(record);\n\n    record.currentIndex = index;\n    return record;\n  }\n\n  /** @internal */\n  _remove(record: IterableChangeRecord_<V>): IterableChangeRecord_<V> {\n    return this._addToRemovals(this._unlink(record));\n  }\n\n  /** @internal */\n  _unlink(record: IterableChangeRecord_<V>): IterableChangeRecord_<V> {\n    if (this._linkedRecords !== null) {\n      this._linkedRecords.remove(record);\n    }\n\n    const prev = record._prev;\n    const next = record._next;\n\n    if (prev === null) {\n      this._itHead = next;\n    } else {\n      prev._next = next;\n    }\n    if (next === null) {\n      this._itTail = prev;\n    } else {\n      next._prev = prev;\n    }\n\n    return record;\n  }\n\n  /** @internal */\n  _addToMoves(record: IterableChangeRecord_<V>, toIndex: number): IterableChangeRecord_<V> {\n    if (record.previousIndex === toIndex) {\n      return record;\n    }\n\n    if (this._movesTail === null) {\n      this._movesTail = this._movesHead = record;\n    } else {\n      this._movesTail = this._movesTail._nextMoved = record;\n    }\n\n    return record;\n  }\n\n  private _addToRemovals(record: IterableChangeRecord_<V>): IterableChangeRecord_<V> {\n    if (this._unlinkedRecords === null) {\n      this._unlinkedRecords = new _DuplicateMap<V>();\n    }\n    this._unlinkedRecords.put(record);\n    record.currentIndex = null;\n    record._nextRemoved = null;\n\n    if (this._removalsTail === null) {\n      this._removalsTail = this._removalsHead = record;\n      record._prevRemoved = null;\n    } else {\n      record._prevRemoved = this._removalsTail;\n      this._removalsTail = this._removalsTail._nextRemoved = record;\n    }\n    return record;\n  }\n\n  /** @internal */\n  _addIdentityChange(record: IterableChangeRecord_<V>, item: V) {\n    record.item = item;\n    if (this._identityChangesTail === null) {\n      this._identityChangesTail = this._identityChangesHead = record;\n    } else {\n      this._identityChangesTail = this._identityChangesTail._nextIdentityChange = record;\n    }\n    return record;\n  }\n}\n\nexport class IterableChangeRecord_<V> implements IterableChangeRecord<V> {\n  currentIndex: number|null = null;\n  previousIndex: number|null = null;\n\n  /** @internal */\n  _nextPrevious: IterableChangeRecord_<V>|null = null;\n  /** @internal */\n  _prev: IterableChangeRecord_<V>|null = null;\n  /** @internal */\n  _next: IterableChangeRecord_<V>|null = null;\n  /** @internal */\n  _prevDup: IterableChangeRecord_<V>|null = null;\n  /** @internal */\n  _nextDup: IterableChangeRecord_<V>|null = null;\n  /** @internal */\n  _prevRemoved: IterableChangeRecord_<V>|null = null;\n  /** @internal */\n  _nextRemoved: IterableChangeRecord_<V>|null = null;\n  /** @internal */\n  _nextAdded: IterableChangeRecord_<V>|null = null;\n  /** @internal */\n  _nextMoved: IterableChangeRecord_<V>|null = null;\n  /** @internal */\n  _nextIdentityChange: IterableChangeRecord_<V>|null = null;\n\n\n  constructor(public item: V, public trackById: any) {}\n}\n\n// A linked list of IterableChangeRecords with the same IterableChangeRecord_.item\nclass _DuplicateItemRecordList<V> {\n  /** @internal */\n  _head: IterableChangeRecord_<V>|null = null;\n  /** @internal */\n  _tail: IterableChangeRecord_<V>|null = null;\n\n  /**\n   * Append the record to the list of duplicates.\n   *\n   * Note: by design all records in the list of duplicates hold the same value in record.item.\n   */\n  add(record: IterableChangeRecord_<V>): void {\n    if (this._head === null) {\n      this._head = this._tail = record;\n      record._nextDup = null;\n      record._prevDup = null;\n    } else {\n      this._tail!._nextDup = record;\n      record._prevDup = this._tail;\n      record._nextDup = null;\n      this._tail = record;\n    }\n  }\n\n  // Returns a IterableChangeRecord_ having IterableChangeRecord_.trackById == trackById and\n  // IterableChangeRecord_.currentIndex >= atOrAfterIndex\n  get(trackById: any, atOrAfterIndex: number|null): IterableChangeRecord_<V>|null {\n    let record: IterableChangeRecord_<V>|null;\n    for (record = this._head; record !== null; record = record._nextDup) {\n      if ((atOrAfterIndex === null || atOrAfterIndex <= record.currentIndex!) &&\n          Object.is(record.trackById, trackById)) {\n        return record;\n      }\n    }\n    return null;\n  }\n\n  /**\n   * Remove one {@link IterableChangeRecord_} from the list of duplicates.\n   *\n   * Returns whether the list of duplicates is empty.\n   */\n  remove(record: IterableChangeRecord_<V>): boolean {\n\n    const prev: IterableChangeRecord_<V>|null = record._prevDup;\n    const next: IterableChangeRecord_<V>|null = record._nextDup;\n    if (prev === null) {\n      this._head = next;\n    } else {\n      prev._nextDup = next;\n    }\n    if (next === null) {\n      this._tail = prev;\n    } else {\n      next._prevDup = prev;\n    }\n    return this._head === null;\n  }\n}\n\nclass _DuplicateMap<V> {\n  map = new Map<any, _DuplicateItemRecordList<V>>();\n\n  put(record: IterableChangeRecord_<V>) {\n    const key = record.trackById;\n\n    let duplicates = this.map.get(key);\n    if (!duplicates) {\n      duplicates = new _DuplicateItemRecordList<V>();\n      this.map.set(key, duplicates);\n    }\n    duplicates.add(record);\n  }\n\n  /**\n   * Retrieve the `value` using key. Because the IterableChangeRecord_ value may be one which we\n   * have already iterated over, we use the `atOrAfterIndex` to pretend it is not there.\n   *\n   * Use case: `[a, b, c, a, a]` if we are at index `3` which is the second `a` then asking if we\n   * have any more `a`s needs to return the second `a`.\n   */\n  get(trackById: any, atOrAfterIndex: number|null): IterableChangeRecord_<V>|null {\n    const key = trackById;\n    const recordList = this.map.get(key);\n    return recordList ? recordList.get(trackById, atOrAfterIndex) : null;\n  }\n\n  /**\n   * Removes a {@link IterableChangeRecord_} from the list of duplicates.\n   *\n   * The list of duplicates also is removed from the map if it gets empty.\n   */\n  remove(record: IterableChangeRecord_<V>): IterableChangeRecord_<V> {\n    const key = record.trackById;\n    const recordList: _DuplicateItemRecordList<V> = this.map.get(key)!;\n    // Remove the list of duplicates when it gets empty\n    if (recordList.remove(record)) {\n      this.map.delete(key);\n    }\n    return record;\n  }\n\n  get isEmpty(): boolean {\n    return this.map.size === 0;\n  }\n\n  clear() {\n    this.map.clear();\n  }\n}\n\nfunction getPreviousIndex(item: any, addRemoveOffset: number, moveOffsets: number[]|null): number {\n  const previousIndex = item.previousIndex;\n  if (previousIndex === null) return previousIndex;\n  let moveOffset = 0;\n  if (moveOffsets && previousIndex < moveOffsets.length) {\n    moveOffset = moveOffsets[previousIndex];\n  }\n  return previousIndex + addRemoveOffset + moveOffset;\n}\n"]}
|