1
1
import { deprecate } from '@ember/debug' ;
2
2
3
- import { DEPRECATE_NON_UNIQUE_PAYLOADS , DISABLE_6X_DEPRECATIONS } from '@warp-drive/build-config/deprecations' ;
3
+ import {
4
+ DEPRECATE_NON_UNIQUE_PAYLOADS ,
5
+ DEPRECATE_RELATIONSHIP_REMOTE_UPDATE_CLEARING_LOCAL_STATE ,
6
+ DISABLE_6X_DEPRECATIONS ,
7
+ } from '@warp-drive/build-config/deprecations' ;
4
8
import { DEBUG } from '@warp-drive/build-config/env' ;
5
9
import { assert } from '@warp-drive/build-config/macros' ;
6
10
import type { StableRecordIdentifier } from '@warp-drive/core-types' ;
7
11
8
- import { isBelongsTo } from './-utils' ;
12
+ import { isBelongsTo , notifyChange } from './-utils' ;
9
13
import { assertPolymorphicType } from './debug/assert-polymorphic-type' ;
10
14
import type { CollectionEdge } from './edges/collection' ;
11
15
import type { ResourceEdge } from './edges/resource' ;
@@ -20,12 +24,14 @@ function _deprecatedCompare<T>(
20
24
prevState : T [ ] ,
21
25
prevSet : Set < T > ,
22
26
onAdd : ( v : T ) => void ,
23
- onDel : ( v : T ) => void
27
+ onDel : ( v : T ) => void ,
28
+ remoteClearsLocal : boolean
24
29
) : { duplicates : Map < T , number [ ] > ; diff : Diff < T > } {
25
30
const newLength = newState . length ;
26
31
const prevLength = prevState . length ;
27
32
const iterationLength = Math . max ( newLength , prevLength ) ;
28
33
let changed : boolean = newMembers . size !== prevSet . size ;
34
+ let remoteOrderChanged = false ;
29
35
const added = new Set < T > ( ) ;
30
36
const removed = new Set < T > ( ) ;
31
37
const duplicates = new Map < T , number [ ] > ( ) ;
@@ -70,11 +76,43 @@ function _deprecatedCompare<T>(
70
76
// detect reordering, adjusting index for duplicates
71
77
// j is always less than i and so if i < prevLength, j < prevLength
72
78
if ( member !== prevState [ j ] ) {
73
- changed = true ;
74
- } else if ( ! changed && j < priorLocalLength ) {
75
- const priorLocalMember = priorLocalState ! [ j ] ;
76
- if ( priorLocalMember !== member ) {
77
- changed = true ;
79
+ // the new remote order does not match the current remote order
80
+ // indicating a change in membership or reordering
81
+ remoteOrderChanged = true ;
82
+ // however: if the new remote order matches the current local order
83
+ // we can disregard the change notification generation so long as
84
+ // we are not configured to reset on remote update (which is deprecated)
85
+ if ( DEPRECATE_RELATIONSHIP_REMOTE_UPDATE_CLEARING_LOCAL_STATE ) {
86
+ if ( ! remoteClearsLocal && i < priorLocalLength ) {
87
+ const priorLocalMember = priorLocalState ! [ j ] ;
88
+ if ( priorLocalMember !== member ) {
89
+ changed = true ;
90
+ }
91
+ } else {
92
+ changed = true ;
93
+ }
94
+ } else {
95
+ if ( i < priorLocalLength ) {
96
+ const priorLocalMember = priorLocalState ! [ j ] ;
97
+ if ( priorLocalMember !== member ) {
98
+ changed = true ;
99
+ }
100
+ } else {
101
+ changed = true ;
102
+ }
103
+ }
104
+
105
+ // if remote order hasn't changed but local order differs
106
+ // and we are configured to reset on remote update (which is deprecated)
107
+ // then we still need to mark the relationship as changed
108
+ } else if ( DEPRECATE_RELATIONSHIP_REMOTE_UPDATE_CLEARING_LOCAL_STATE ) {
109
+ if ( remoteClearsLocal ) {
110
+ if ( ! changed && j < priorLocalLength ) {
111
+ const priorLocalMember = priorLocalState ! [ j ] ;
112
+ if ( priorLocalMember !== member ) {
113
+ changed = true ;
114
+ }
115
+ }
78
116
}
79
117
}
80
118
@@ -98,6 +136,7 @@ function _deprecatedCompare<T>(
98
136
finalState,
99
137
finalSet,
100
138
changed,
139
+ remoteOrderChanged,
101
140
} ;
102
141
103
142
return {
@@ -113,13 +152,15 @@ function _compare<T>(
113
152
prevState : T [ ] ,
114
153
prevSet : Set < T > ,
115
154
onAdd : ( v : T ) => void ,
116
- onDel : ( v : T ) => void
155
+ onDel : ( v : T ) => void ,
156
+ remoteClearsLocal : boolean
117
157
) : Diff < T > {
118
158
const finalLength = finalState . length ;
119
159
const prevLength = prevState . length ;
120
160
const iterationLength = Math . max ( finalLength , prevLength ) ;
121
161
const equalLength = finalLength === prevLength ;
122
162
let changed : boolean = finalSet . size !== prevSet . size ;
163
+ let remoteOrderChanged = false ;
123
164
const added = new Set < T > ( ) ;
124
165
const removed = new Set < T > ( ) ;
125
166
const priorLocalLength = priorLocalState ?. length ?? 0 ;
@@ -143,11 +184,43 @@ function _compare<T>(
143
184
144
185
// detect reordering
145
186
if ( equalLength && member !== prevMember ) {
146
- changed = true ;
147
- } else if ( equalLength && ! changed && i < priorLocalLength ) {
148
- const priorLocalMember = priorLocalState ! [ i ] ;
149
- if ( priorLocalMember !== prevMember ) {
150
- changed = true ;
187
+ // the new remote order does not match the current remote order
188
+ // indicating a change in membership or reordering
189
+ remoteOrderChanged = true ;
190
+ // however: if the new remote order matches the current local order
191
+ // we can disregard the change notification generation so long as
192
+ // we are not configured to reset on remote update (which is deprecated)
193
+ if ( DEPRECATE_RELATIONSHIP_REMOTE_UPDATE_CLEARING_LOCAL_STATE ) {
194
+ if ( ! remoteClearsLocal && i < priorLocalLength ) {
195
+ const priorLocalMember = priorLocalState ! [ i ] ;
196
+ if ( priorLocalMember !== member ) {
197
+ changed = true ;
198
+ }
199
+ } else {
200
+ changed = true ;
201
+ }
202
+ } else {
203
+ if ( i < priorLocalLength ) {
204
+ const priorLocalMember = priorLocalState ! [ i ] ;
205
+ if ( priorLocalMember !== member ) {
206
+ changed = true ;
207
+ }
208
+ } else {
209
+ changed = true ;
210
+ }
211
+ }
212
+
213
+ // if remote order hasn't changed but local order differs
214
+ // and we are configured to reset on remote update (which is deprecated)
215
+ // then we still need to mark the relationship as changed
216
+ } else if ( DEPRECATE_RELATIONSHIP_REMOTE_UPDATE_CLEARING_LOCAL_STATE ) {
217
+ if ( remoteClearsLocal ) {
218
+ if ( equalLength && ! changed && i < priorLocalLength ) {
219
+ const priorLocalMember = priorLocalState ! [ i ] ;
220
+ if ( priorLocalMember !== prevMember ) {
221
+ changed = true ;
222
+ }
223
+ }
151
224
}
152
225
}
153
226
@@ -165,6 +238,7 @@ function _compare<T>(
165
238
finalState,
166
239
finalSet,
167
240
changed,
241
+ remoteOrderChanged,
168
242
} ;
169
243
}
170
244
@@ -174,6 +248,7 @@ type Diff<T> = {
174
248
finalState : T [ ] ;
175
249
finalSet : Set < T > ;
176
250
changed : boolean ;
251
+ remoteOrderChanged : boolean ;
177
252
} ;
178
253
179
254
export function diffCollection (
@@ -194,7 +269,8 @@ export function diffCollection(
194
269
remoteState ,
195
270
remoteMembers ,
196
271
onAdd ,
197
- onDel
272
+ onDel ,
273
+ relationship . definition . resetOnRemoteUpdate
198
274
) ;
199
275
200
276
if ( DEBUG ) {
@@ -221,7 +297,16 @@ export function diffCollection(
221
297
) ;
222
298
}
223
299
224
- return _compare ( priorLocalState , finalState , finalSet , remoteState , remoteMembers , onAdd , onDel ) ;
300
+ return _compare (
301
+ priorLocalState ,
302
+ finalState ,
303
+ finalSet ,
304
+ remoteState ,
305
+ remoteMembers ,
306
+ onAdd ,
307
+ onDel ,
308
+ relationship . definition . resetOnRemoteUpdate
309
+ ) ;
225
310
}
226
311
227
312
export function computeLocalState ( storage : CollectionEdge ) : StableRecordIdentifier [ ] {
@@ -421,5 +506,12 @@ export function rollbackRelationship(
421
506
} ,
422
507
false
423
508
) ;
509
+
510
+ // when the change was a "reorder" only we wont have generated
511
+ // a notification yet.
512
+ // if we give rollback a unique operation we can use the ability of
513
+ // diff to report a separate `remoteOrderChanged` flag to trigger this
514
+ // if needed to avoid the duplicate.
515
+ notifyChange ( graph , relationship ) ;
424
516
}
425
517
}
0 commit comments