@ngrx/data 13.0.0-beta.0 → 13.0.0-rc.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/{esm2015/index.js → esm2020/index.mjs} +0 -0
- package/{esm2015/ngrx-data.js → esm2020/ngrx-data.mjs} +0 -0
- package/{esm2015/public_api.js → esm2020/public_api.mjs} +0 -0
- package/esm2020/src/actions/entity-action-factory.mjs +50 -0
- package/{esm2015/src/actions/entity-action-guard.js → esm2020/src/actions/entity-action-guard.mjs} +0 -0
- package/{esm2015/src/actions/entity-action-operators.js → esm2020/src/actions/entity-action-operators.mjs} +0 -0
- package/{esm2015/src/actions/entity-action.js → esm2020/src/actions/entity-action.mjs} +0 -0
- package/esm2020/src/actions/entity-cache-action.mjs +116 -0
- package/esm2020/src/actions/entity-cache-change-set.mjs +50 -0
- package/{esm2015/src/actions/entity-op.js → esm2020/src/actions/entity-op.mjs} +0 -0
- package/{esm2015/src/actions/merge-strategy.js → esm2020/src/actions/merge-strategy.mjs} +0 -0
- package/{esm2015/src/actions/update-response-data.js → esm2020/src/actions/update-response-data.mjs} +0 -0
- package/{esm2015/src/dataservices/data-service-error.js → esm2020/src/dataservices/data-service-error.mjs} +0 -0
- package/{esm2015/src/dataservices/default-data-service-config.js → esm2020/src/dataservices/default-data-service-config.mjs} +0 -0
- package/{esm2015/src/dataservices/default-data.service.js → esm2020/src/dataservices/default-data.service.mjs} +4 -4
- package/esm2020/src/dataservices/entity-cache-data.service.mjs +146 -0
- package/esm2020/src/dataservices/entity-data.service.mjs +63 -0
- package/esm2020/src/dataservices/http-url-generator.mjs +86 -0
- package/{esm2015/src/dataservices/interfaces.js → esm2020/src/dataservices/interfaces.mjs} +0 -0
- package/{esm2015/src/dataservices/persistence-result-handler.service.js → esm2020/src/dataservices/persistence-result-handler.service.mjs} +4 -4
- package/esm2020/src/dispatchers/entity-cache-dispatcher.mjs +163 -0
- package/{esm2015/src/dispatchers/entity-commands.js → esm2020/src/dispatchers/entity-commands.mjs} +0 -0
- package/esm2020/src/dispatchers/entity-dispatcher-base.mjs +405 -0
- package/{esm2015/src/dispatchers/entity-dispatcher-default-options.js → esm2020/src/dispatchers/entity-dispatcher-default-options.mjs} +4 -4
- package/esm2020/src/dispatchers/entity-dispatcher-factory.mjs +65 -0
- package/{esm2015/src/dispatchers/entity-dispatcher.js → esm2020/src/dispatchers/entity-dispatcher.mjs} +0 -0
- package/esm2020/src/effects/entity-cache-effects.mjs +115 -0
- package/{esm2015/src/effects/entity-effects-scheduler.js → esm2020/src/effects/entity-effects-scheduler.mjs} +0 -0
- package/esm2020/src/effects/entity-effects.mjs +148 -0
- package/{esm2015/src/entity-data-without-effects.module.js → esm2020/src/entity-data-without-effects.module.mjs} +5 -5
- package/{esm2015/src/entity-data.module.js → esm2020/src/entity-data.module.mjs} +5 -5
- package/esm2020/src/entity-metadata/entity-definition.mjs +33 -0
- package/esm2020/src/entity-metadata/entity-definition.service.mjs +93 -0
- package/{esm2015/src/entity-metadata/entity-filters.js → esm2020/src/entity-metadata/entity-filters.mjs} +0 -0
- package/{esm2015/src/entity-metadata/entity-metadata.js → esm2020/src/entity-metadata/entity-metadata.mjs} +0 -0
- package/{esm2015/src/entity-services/entity-collection-service-base.js → esm2020/src/entity-services/entity-collection-service-base.mjs} +0 -0
- package/{esm2015/src/entity-services/entity-collection-service-elements-factory.js → esm2020/src/entity-services/entity-collection-service-elements-factory.mjs} +4 -4
- package/{esm2015/src/entity-services/entity-collection-service-factory.js → esm2020/src/entity-services/entity-collection-service-factory.mjs} +4 -4
- package/{esm2015/src/entity-services/entity-collection-service.js → esm2020/src/entity-services/entity-collection-service.mjs} +0 -0
- package/{esm2015/src/entity-services/entity-services-base.js → esm2020/src/entity-services/entity-services-base.mjs} +4 -4
- package/{esm2015/src/entity-services/entity-services-elements.js → esm2020/src/entity-services/entity-services-elements.mjs} +4 -4
- package/{esm2015/src/entity-services/entity-services.js → esm2020/src/entity-services/entity-services.mjs} +0 -0
- package/{esm2015/src/index.js → esm2020/src/index.mjs} +0 -0
- package/{esm2015/src/reducers/constants.js → esm2020/src/reducers/constants.mjs} +0 -0
- package/esm2020/src/reducers/entity-cache-reducer.mjs +271 -0
- package/{esm2015/src/reducers/entity-cache.js → esm2020/src/reducers/entity-cache.mjs} +0 -0
- package/esm2020/src/reducers/entity-change-tracker-base.mjs +587 -0
- package/{esm2015/src/reducers/entity-change-tracker.js → esm2020/src/reducers/entity-change-tracker.mjs} +0 -0
- package/{esm2015/src/reducers/entity-collection-creator.js → esm2020/src/reducers/entity-collection-creator.mjs} +4 -4
- package/esm2020/src/reducers/entity-collection-reducer-methods.mjs +806 -0
- package/{esm2015/src/reducers/entity-collection-reducer-registry.js → esm2020/src/reducers/entity-collection-reducer-registry.mjs} +4 -4
- package/{esm2015/src/reducers/entity-collection-reducer.js → esm2020/src/reducers/entity-collection-reducer.mjs} +4 -4
- package/{esm2015/src/reducers/entity-collection.js → esm2020/src/reducers/entity-collection.mjs} +0 -0
- package/{esm2015/src/selectors/entity-cache-selector.js → esm2020/src/selectors/entity-cache-selector.mjs} +0 -0
- package/{esm2015/src/selectors/entity-selectors$.js → esm2020/src/selectors/entity-selectors$.mjs} +4 -4
- package/esm2020/src/selectors/entity-selectors.mjs +96 -0
- package/{esm2015/src/utils/correlation-id-generator.js → esm2020/src/utils/correlation-id-generator.mjs} +4 -4
- package/{esm2015/src/utils/default-logger.js → esm2020/src/utils/default-logger.mjs} +4 -4
- package/{esm2015/src/utils/default-pluralizer.js → esm2020/src/utils/default-pluralizer.mjs} +5 -5
- package/{esm2015/src/utils/guid-fns.js → esm2020/src/utils/guid-fns.mjs} +0 -0
- package/{esm2015/src/utils/interfaces.js → esm2020/src/utils/interfaces.mjs} +0 -0
- package/{esm2015/src/utils/utilities.js → esm2020/src/utils/utilities.mjs} +0 -0
- package/fesm2015/{ngrx-data.js → ngrx-data.mjs} +194 -169
- package/fesm2015/ngrx-data.mjs.map +1 -0
- package/fesm2020/ngrx-data.mjs +4936 -0
- package/fesm2020/ngrx-data.mjs.map +1 -0
- package/package.json +26 -13
- package/schematics/ng-add/index.js +29 -25
- package/schematics/ng-add/index.js.map +1 -1
- package/schematics-core/utility/angular-utils.js +6 -5
- package/schematics-core/utility/angular-utils.js.map +1 -1
- package/schematics-core/utility/ast-utils.js +12 -8
- package/schematics-core/utility/ast-utils.js.map +1 -1
- package/schematics-core/utility/find-component.js +12 -12
- package/schematics-core/utility/find-component.js.map +1 -1
- package/schematics-core/utility/find-module.js +12 -12
- package/schematics-core/utility/find-module.js.map +1 -1
- package/schematics-core/utility/json-utilts.js.map +1 -1
- package/schematics-core/utility/ngrx-utils.js +16 -12
- package/schematics-core/utility/ngrx-utils.js.map +1 -1
- package/schematics-core/utility/parse-name.js +3 -3
- package/schematics-core/utility/parse-name.js.map +1 -1
- package/schematics-core/utility/project.js +1 -1
- package/schematics-core/utility/project.js.map +1 -1
- package/schematics-core/utility/visitors.js +2 -2
- package/schematics-core/utility/visitors.js.map +1 -1
- package/bundles/ngrx-data.umd.js +0 -5491
- package/bundles/ngrx-data.umd.js.map +0 -1
- package/esm2015/src/actions/entity-action-factory.js +0 -46
- package/esm2015/src/actions/entity-cache-action.js +0 -116
- package/esm2015/src/actions/entity-cache-change-set.js +0 -50
- package/esm2015/src/dataservices/entity-cache-data.service.js +0 -140
- package/esm2015/src/dataservices/entity-data.service.js +0 -63
- package/esm2015/src/dataservices/http-url-generator.js +0 -83
- package/esm2015/src/dispatchers/entity-cache-dispatcher.js +0 -163
- package/esm2015/src/dispatchers/entity-dispatcher-base.js +0 -401
- package/esm2015/src/dispatchers/entity-dispatcher-factory.js +0 -62
- package/esm2015/src/effects/entity-cache-effects.js +0 -115
- package/esm2015/src/effects/entity-effects.js +0 -148
- package/esm2015/src/entity-metadata/entity-definition.js +0 -26
- package/esm2015/src/entity-metadata/entity-definition.service.js +0 -93
- package/esm2015/src/reducers/entity-cache-reducer.js +0 -271
- package/esm2015/src/reducers/entity-change-tracker-base.js +0 -575
- package/esm2015/src/reducers/entity-collection-reducer-methods.js +0 -784
- package/esm2015/src/selectors/entity-selectors.js +0 -89
- package/fesm2015/ngrx-data.js.map +0 -1
|
@@ -0,0 +1,587 @@
|
|
|
1
|
+
import { ChangeType } from './entity-collection';
|
|
2
|
+
import { defaultSelectId } from '../utils/utilities';
|
|
3
|
+
import { MergeStrategy } from '../actions/merge-strategy';
|
|
4
|
+
/**
|
|
5
|
+
* The default implementation of EntityChangeTracker with
|
|
6
|
+
* methods for tracking, committing, and reverting/undoing unsaved entity changes.
|
|
7
|
+
* Used by EntityCollectionReducerMethods which should call tracker methods BEFORE modifying the collection.
|
|
8
|
+
* See EntityChangeTracker docs.
|
|
9
|
+
*/
|
|
10
|
+
export class EntityChangeTrackerBase {
|
|
11
|
+
constructor(adapter, selectId) {
|
|
12
|
+
this.adapter = adapter;
|
|
13
|
+
this.selectId = selectId;
|
|
14
|
+
/** Extract the primary key (id); default to `id` */
|
|
15
|
+
this.selectId = selectId || defaultSelectId;
|
|
16
|
+
}
|
|
17
|
+
// #region commit methods
|
|
18
|
+
/**
|
|
19
|
+
* Commit all changes as when the collection has been completely reloaded from the server.
|
|
20
|
+
* Harmless when there are no entity changes to commit.
|
|
21
|
+
* @param collection The entity collection
|
|
22
|
+
*/
|
|
23
|
+
commitAll(collection) {
|
|
24
|
+
return Object.keys(collection.changeState).length === 0
|
|
25
|
+
? collection
|
|
26
|
+
: { ...collection, changeState: {} };
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Commit changes for the given entities as when they have been refreshed from the server.
|
|
30
|
+
* Harmless when there are no entity changes to commit.
|
|
31
|
+
* @param entityOrIdList The entities to clear tracking or their ids.
|
|
32
|
+
* @param collection The entity collection
|
|
33
|
+
*/
|
|
34
|
+
commitMany(entityOrIdList, collection) {
|
|
35
|
+
if (entityOrIdList == null || entityOrIdList.length === 0) {
|
|
36
|
+
return collection; // nothing to commit
|
|
37
|
+
}
|
|
38
|
+
let didMutate = false;
|
|
39
|
+
const changeState = entityOrIdList.reduce((chgState, entityOrId) => {
|
|
40
|
+
const id = typeof entityOrId === 'object'
|
|
41
|
+
? this.selectId(entityOrId)
|
|
42
|
+
: entityOrId;
|
|
43
|
+
if (chgState[id]) {
|
|
44
|
+
if (!didMutate) {
|
|
45
|
+
chgState = { ...chgState };
|
|
46
|
+
didMutate = true;
|
|
47
|
+
}
|
|
48
|
+
delete chgState[id];
|
|
49
|
+
}
|
|
50
|
+
return chgState;
|
|
51
|
+
}, collection.changeState);
|
|
52
|
+
return didMutate ? { ...collection, changeState } : collection;
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* Commit changes for the given entity as when it have been refreshed from the server.
|
|
56
|
+
* Harmless when no entity changes to commit.
|
|
57
|
+
* @param entityOrId The entity to clear tracking or its id.
|
|
58
|
+
* @param collection The entity collection
|
|
59
|
+
*/
|
|
60
|
+
commitOne(entityOrId, collection) {
|
|
61
|
+
return entityOrId == null
|
|
62
|
+
? collection
|
|
63
|
+
: this.commitMany([entityOrId], collection);
|
|
64
|
+
}
|
|
65
|
+
// #endregion commit methods
|
|
66
|
+
// #region merge query
|
|
67
|
+
/**
|
|
68
|
+
* Merge query results into the collection, adjusting the ChangeState per the mergeStrategy.
|
|
69
|
+
* @param entities Entities returned from querying the server.
|
|
70
|
+
* @param collection The entity collection
|
|
71
|
+
* @param [mergeStrategy] How to merge a queried entity when the corresponding entity in the collection has an unsaved change.
|
|
72
|
+
* Defaults to MergeStrategy.PreserveChanges.
|
|
73
|
+
* @returns The merged EntityCollection.
|
|
74
|
+
*/
|
|
75
|
+
mergeQueryResults(entities, collection, mergeStrategy) {
|
|
76
|
+
return this.mergeServerUpserts(entities, collection, MergeStrategy.PreserveChanges, mergeStrategy);
|
|
77
|
+
}
|
|
78
|
+
// #endregion merge query results
|
|
79
|
+
// #region merge save results
|
|
80
|
+
/**
|
|
81
|
+
* Merge result of saving new entities into the collection, adjusting the ChangeState per the mergeStrategy.
|
|
82
|
+
* The default is MergeStrategy.OverwriteChanges.
|
|
83
|
+
* @param entities Entities returned from saving new entities to the server.
|
|
84
|
+
* @param collection The entity collection
|
|
85
|
+
* @param [mergeStrategy] How to merge a saved entity when the corresponding entity in the collection has an unsaved change.
|
|
86
|
+
* Defaults to MergeStrategy.OverwriteChanges.
|
|
87
|
+
* @returns The merged EntityCollection.
|
|
88
|
+
*/
|
|
89
|
+
mergeSaveAdds(entities, collection, mergeStrategy) {
|
|
90
|
+
return this.mergeServerUpserts(entities, collection, MergeStrategy.OverwriteChanges, mergeStrategy);
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* Merge successful result of deleting entities on the server that have the given primary keys
|
|
94
|
+
* Clears the entity changeState for those keys unless the MergeStrategy is ignoreChanges.
|
|
95
|
+
* @param entities keys primary keys of the entities to remove/delete.
|
|
96
|
+
* @param collection The entity collection
|
|
97
|
+
* @param [mergeStrategy] How to adjust change tracking when the corresponding entity in the collection has an unsaved change.
|
|
98
|
+
* Defaults to MergeStrategy.OverwriteChanges.
|
|
99
|
+
* @returns The merged EntityCollection.
|
|
100
|
+
*/
|
|
101
|
+
mergeSaveDeletes(keys, collection, mergeStrategy) {
|
|
102
|
+
mergeStrategy =
|
|
103
|
+
mergeStrategy == null ? MergeStrategy.OverwriteChanges : mergeStrategy;
|
|
104
|
+
// same logic for all non-ignore merge strategies: always clear (commit) the changes
|
|
105
|
+
const deleteIds = keys; // make TypeScript happy
|
|
106
|
+
collection =
|
|
107
|
+
mergeStrategy === MergeStrategy.IgnoreChanges
|
|
108
|
+
? collection
|
|
109
|
+
: this.commitMany(deleteIds, collection);
|
|
110
|
+
return this.adapter.removeMany(deleteIds, collection);
|
|
111
|
+
}
|
|
112
|
+
/**
|
|
113
|
+
* Merge result of saving updated entities into the collection, adjusting the ChangeState per the mergeStrategy.
|
|
114
|
+
* The default is MergeStrategy.OverwriteChanges.
|
|
115
|
+
* @param updateResponseData Entity response data returned from saving updated entities to the server.
|
|
116
|
+
* @param collection The entity collection
|
|
117
|
+
* @param [mergeStrategy] How to merge a saved entity when the corresponding entity in the collection has an unsaved change.
|
|
118
|
+
* Defaults to MergeStrategy.OverwriteChanges.
|
|
119
|
+
* @param [skipUnchanged] True means skip update if server didn't change it. False by default.
|
|
120
|
+
* If the update was optimistic and the server didn't make more changes of its own
|
|
121
|
+
* then the updates are already in the collection and shouldn't make them again.
|
|
122
|
+
* @returns The merged EntityCollection.
|
|
123
|
+
*/
|
|
124
|
+
mergeSaveUpdates(updateResponseData, collection, mergeStrategy, skipUnchanged = false) {
|
|
125
|
+
if (updateResponseData == null || updateResponseData.length === 0) {
|
|
126
|
+
return collection; // nothing to merge.
|
|
127
|
+
}
|
|
128
|
+
let didMutate = false;
|
|
129
|
+
let changeState = collection.changeState;
|
|
130
|
+
mergeStrategy =
|
|
131
|
+
mergeStrategy == null ? MergeStrategy.OverwriteChanges : mergeStrategy;
|
|
132
|
+
let updates;
|
|
133
|
+
switch (mergeStrategy) {
|
|
134
|
+
case MergeStrategy.IgnoreChanges:
|
|
135
|
+
updates = filterChanged(updateResponseData);
|
|
136
|
+
return this.adapter.updateMany(updates, collection);
|
|
137
|
+
case MergeStrategy.OverwriteChanges:
|
|
138
|
+
changeState = updateResponseData.reduce((chgState, update) => {
|
|
139
|
+
const oldId = update.id;
|
|
140
|
+
const change = chgState[oldId];
|
|
141
|
+
if (change) {
|
|
142
|
+
if (!didMutate) {
|
|
143
|
+
chgState = { ...chgState };
|
|
144
|
+
didMutate = true;
|
|
145
|
+
}
|
|
146
|
+
delete chgState[oldId];
|
|
147
|
+
}
|
|
148
|
+
return chgState;
|
|
149
|
+
}, collection.changeState);
|
|
150
|
+
collection = didMutate ? { ...collection, changeState } : collection;
|
|
151
|
+
updates = filterChanged(updateResponseData);
|
|
152
|
+
return this.adapter.updateMany(updates, collection);
|
|
153
|
+
case MergeStrategy.PreserveChanges: {
|
|
154
|
+
const updateableEntities = [];
|
|
155
|
+
changeState = updateResponseData.reduce((chgState, update) => {
|
|
156
|
+
const oldId = update.id;
|
|
157
|
+
const change = chgState[oldId];
|
|
158
|
+
if (change) {
|
|
159
|
+
// Tracking a change so update original value but not the current value
|
|
160
|
+
if (!didMutate) {
|
|
161
|
+
chgState = { ...chgState };
|
|
162
|
+
didMutate = true;
|
|
163
|
+
}
|
|
164
|
+
const newId = this.selectId(update.changes);
|
|
165
|
+
const oldChangeState = change;
|
|
166
|
+
// If the server changed the id, register the new "originalValue" under the new id
|
|
167
|
+
// and remove the change tracked under the old id.
|
|
168
|
+
if (newId !== oldId) {
|
|
169
|
+
delete chgState[oldId];
|
|
170
|
+
}
|
|
171
|
+
const newOrigValue = {
|
|
172
|
+
...oldChangeState.originalValue,
|
|
173
|
+
...update.changes,
|
|
174
|
+
};
|
|
175
|
+
chgState[newId] = {
|
|
176
|
+
...oldChangeState,
|
|
177
|
+
originalValue: newOrigValue,
|
|
178
|
+
};
|
|
179
|
+
}
|
|
180
|
+
else {
|
|
181
|
+
updateableEntities.push(update);
|
|
182
|
+
}
|
|
183
|
+
return chgState;
|
|
184
|
+
}, collection.changeState);
|
|
185
|
+
collection = didMutate ? { ...collection, changeState } : collection;
|
|
186
|
+
updates = filterChanged(updateableEntities);
|
|
187
|
+
return this.adapter.updateMany(updates, collection);
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
/**
|
|
191
|
+
* Conditionally keep only those updates that have additional server changes.
|
|
192
|
+
* (e.g., for optimistic saves because they updates are already in the current collection)
|
|
193
|
+
* Strip off the `changed` property.
|
|
194
|
+
* @responseData Entity response data from server.
|
|
195
|
+
* May be an UpdateResponseData<T>, a subclass of Update<T> with a 'changed' flag.
|
|
196
|
+
* @returns Update<T> (without the changed flag)
|
|
197
|
+
*/
|
|
198
|
+
function filterChanged(responseData) {
|
|
199
|
+
if (skipUnchanged === true) {
|
|
200
|
+
// keep only those updates that the server changed (knowable if is UpdateResponseData<T>)
|
|
201
|
+
responseData = responseData.filter((r) => r.changed === true);
|
|
202
|
+
}
|
|
203
|
+
// Strip unchanged property from responseData, leaving just the pure Update<T>
|
|
204
|
+
// TODO: Remove? probably not necessary as the Update isn't stored and adapter will ignore `changed`.
|
|
205
|
+
return responseData.map((r) => ({ id: r.id, changes: r.changes }));
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
/**
|
|
209
|
+
* Merge result of saving upserted entities into the collection, adjusting the ChangeState per the mergeStrategy.
|
|
210
|
+
* The default is MergeStrategy.OverwriteChanges.
|
|
211
|
+
* @param entities Entities returned from saving upserts to the server.
|
|
212
|
+
* @param collection The entity collection
|
|
213
|
+
* @param [mergeStrategy] How to merge a saved entity when the corresponding entity in the collection has an unsaved change.
|
|
214
|
+
* Defaults to MergeStrategy.OverwriteChanges.
|
|
215
|
+
* @returns The merged EntityCollection.
|
|
216
|
+
*/
|
|
217
|
+
mergeSaveUpserts(entities, collection, mergeStrategy) {
|
|
218
|
+
return this.mergeServerUpserts(entities, collection, MergeStrategy.OverwriteChanges, mergeStrategy);
|
|
219
|
+
}
|
|
220
|
+
// #endregion merge save results
|
|
221
|
+
// #region query & save helpers
|
|
222
|
+
/**
|
|
223
|
+
*
|
|
224
|
+
* @param entities Entities to merge
|
|
225
|
+
* @param collection Collection into which entities are merged
|
|
226
|
+
* @param defaultMergeStrategy How to merge when action's MergeStrategy is unspecified
|
|
227
|
+
* @param [mergeStrategy] The action's MergeStrategy
|
|
228
|
+
*/
|
|
229
|
+
mergeServerUpserts(entities, collection, defaultMergeStrategy, mergeStrategy) {
|
|
230
|
+
if (entities == null || entities.length === 0) {
|
|
231
|
+
return collection; // nothing to merge.
|
|
232
|
+
}
|
|
233
|
+
let didMutate = false;
|
|
234
|
+
let changeState = collection.changeState;
|
|
235
|
+
mergeStrategy =
|
|
236
|
+
mergeStrategy == null ? defaultMergeStrategy : mergeStrategy;
|
|
237
|
+
switch (mergeStrategy) {
|
|
238
|
+
case MergeStrategy.IgnoreChanges:
|
|
239
|
+
return this.adapter.upsertMany(entities, collection);
|
|
240
|
+
case MergeStrategy.OverwriteChanges:
|
|
241
|
+
collection = this.adapter.upsertMany(entities, collection);
|
|
242
|
+
changeState = entities.reduce((chgState, entity) => {
|
|
243
|
+
const id = this.selectId(entity);
|
|
244
|
+
const change = chgState[id];
|
|
245
|
+
if (change) {
|
|
246
|
+
if (!didMutate) {
|
|
247
|
+
chgState = { ...chgState };
|
|
248
|
+
didMutate = true;
|
|
249
|
+
}
|
|
250
|
+
delete chgState[id];
|
|
251
|
+
}
|
|
252
|
+
return chgState;
|
|
253
|
+
}, collection.changeState);
|
|
254
|
+
return didMutate ? { ...collection, changeState } : collection;
|
|
255
|
+
case MergeStrategy.PreserveChanges: {
|
|
256
|
+
const upsertEntities = [];
|
|
257
|
+
changeState = entities.reduce((chgState, entity) => {
|
|
258
|
+
const id = this.selectId(entity);
|
|
259
|
+
const change = chgState[id];
|
|
260
|
+
if (change) {
|
|
261
|
+
if (!didMutate) {
|
|
262
|
+
chgState = {
|
|
263
|
+
...chgState,
|
|
264
|
+
[id]: {
|
|
265
|
+
...chgState[id],
|
|
266
|
+
originalValue: entity,
|
|
267
|
+
},
|
|
268
|
+
};
|
|
269
|
+
didMutate = true;
|
|
270
|
+
}
|
|
271
|
+
}
|
|
272
|
+
else {
|
|
273
|
+
upsertEntities.push(entity);
|
|
274
|
+
}
|
|
275
|
+
return chgState;
|
|
276
|
+
}, collection.changeState);
|
|
277
|
+
collection = this.adapter.upsertMany(upsertEntities, collection);
|
|
278
|
+
return didMutate ? { ...collection, changeState } : collection;
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
// #endregion query & save helpers
|
|
283
|
+
// #region track methods
|
|
284
|
+
/**
|
|
285
|
+
* Track multiple entities before adding them to the collection.
|
|
286
|
+
* Does NOT add to the collection (the reducer's job).
|
|
287
|
+
* @param entities The entities to add. They must all have their ids.
|
|
288
|
+
* @param collection The entity collection
|
|
289
|
+
* @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.
|
|
290
|
+
*/
|
|
291
|
+
trackAddMany(entities, collection, mergeStrategy) {
|
|
292
|
+
if (mergeStrategy === MergeStrategy.IgnoreChanges ||
|
|
293
|
+
entities == null ||
|
|
294
|
+
entities.length === 0) {
|
|
295
|
+
return collection; // nothing to track
|
|
296
|
+
}
|
|
297
|
+
let didMutate = false;
|
|
298
|
+
const changeState = entities.reduce((chgState, entity) => {
|
|
299
|
+
const id = this.selectId(entity);
|
|
300
|
+
if (id == null || id === '') {
|
|
301
|
+
throw new Error(`${collection.entityName} entity add requires a key to be tracked`);
|
|
302
|
+
}
|
|
303
|
+
const trackedChange = chgState[id];
|
|
304
|
+
if (!trackedChange) {
|
|
305
|
+
if (!didMutate) {
|
|
306
|
+
didMutate = true;
|
|
307
|
+
chgState = { ...chgState };
|
|
308
|
+
}
|
|
309
|
+
chgState[id] = { changeType: ChangeType.Added };
|
|
310
|
+
}
|
|
311
|
+
return chgState;
|
|
312
|
+
}, collection.changeState);
|
|
313
|
+
return didMutate ? { ...collection, changeState } : collection;
|
|
314
|
+
}
|
|
315
|
+
/**
|
|
316
|
+
* Track an entity before adding it to the collection.
|
|
317
|
+
* Does NOT add to the collection (the reducer's job).
|
|
318
|
+
* @param entity The entity to add. It must have an id.
|
|
319
|
+
* @param collection The entity collection
|
|
320
|
+
* @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.
|
|
321
|
+
* If not specified, implementation supplies a default strategy.
|
|
322
|
+
*/
|
|
323
|
+
trackAddOne(entity, collection, mergeStrategy) {
|
|
324
|
+
return entity == null
|
|
325
|
+
? collection
|
|
326
|
+
: this.trackAddMany([entity], collection, mergeStrategy);
|
|
327
|
+
}
|
|
328
|
+
/**
|
|
329
|
+
* Track multiple entities before removing them with the intention of deleting them on the server.
|
|
330
|
+
* Does NOT remove from the collection (the reducer's job).
|
|
331
|
+
* @param keys The primary keys of the entities to delete.
|
|
332
|
+
* @param collection The entity collection
|
|
333
|
+
* @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.
|
|
334
|
+
*/
|
|
335
|
+
trackDeleteMany(keys, collection, mergeStrategy) {
|
|
336
|
+
if (mergeStrategy === MergeStrategy.IgnoreChanges ||
|
|
337
|
+
keys == null ||
|
|
338
|
+
keys.length === 0) {
|
|
339
|
+
return collection; // nothing to track
|
|
340
|
+
}
|
|
341
|
+
let didMutate = false;
|
|
342
|
+
const entityMap = collection.entities;
|
|
343
|
+
const changeState = keys.reduce((chgState, id) => {
|
|
344
|
+
const originalValue = entityMap[id];
|
|
345
|
+
if (originalValue) {
|
|
346
|
+
const trackedChange = chgState[id];
|
|
347
|
+
if (trackedChange) {
|
|
348
|
+
if (trackedChange.changeType === ChangeType.Added) {
|
|
349
|
+
// Special case: stop tracking an added entity that you delete
|
|
350
|
+
// The caller must also detect this, remove it immediately from the collection
|
|
351
|
+
// and skip attempt to delete on the server.
|
|
352
|
+
cloneChgStateOnce();
|
|
353
|
+
delete chgState[id];
|
|
354
|
+
}
|
|
355
|
+
else if (trackedChange.changeType === ChangeType.Updated) {
|
|
356
|
+
// Special case: switch change type from Updated to Deleted.
|
|
357
|
+
cloneChgStateOnce();
|
|
358
|
+
chgState[id] = { ...chgState[id], changeType: ChangeType.Deleted };
|
|
359
|
+
}
|
|
360
|
+
}
|
|
361
|
+
else {
|
|
362
|
+
// Start tracking this entity
|
|
363
|
+
cloneChgStateOnce();
|
|
364
|
+
chgState[id] = { changeType: ChangeType.Deleted, originalValue };
|
|
365
|
+
}
|
|
366
|
+
}
|
|
367
|
+
return chgState;
|
|
368
|
+
function cloneChgStateOnce() {
|
|
369
|
+
if (!didMutate) {
|
|
370
|
+
didMutate = true;
|
|
371
|
+
chgState = { ...chgState };
|
|
372
|
+
}
|
|
373
|
+
}
|
|
374
|
+
}, collection.changeState);
|
|
375
|
+
return didMutate ? { ...collection, changeState } : collection;
|
|
376
|
+
}
|
|
377
|
+
/**
|
|
378
|
+
* Track an entity before it is removed with the intention of deleting it on the server.
|
|
379
|
+
* Does NOT remove from the collection (the reducer's job).
|
|
380
|
+
* @param key The primary key of the entity to delete.
|
|
381
|
+
* @param collection The entity collection
|
|
382
|
+
* @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.
|
|
383
|
+
*/
|
|
384
|
+
trackDeleteOne(key, collection, mergeStrategy) {
|
|
385
|
+
return key == null
|
|
386
|
+
? collection
|
|
387
|
+
: this.trackDeleteMany([key], collection, mergeStrategy);
|
|
388
|
+
}
|
|
389
|
+
/**
|
|
390
|
+
* Track multiple entities before updating them in the collection.
|
|
391
|
+
* Does NOT update the collection (the reducer's job).
|
|
392
|
+
* @param updates The entities to update.
|
|
393
|
+
* @param collection The entity collection
|
|
394
|
+
* @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.
|
|
395
|
+
*/
|
|
396
|
+
trackUpdateMany(updates, collection, mergeStrategy) {
|
|
397
|
+
if (mergeStrategy === MergeStrategy.IgnoreChanges ||
|
|
398
|
+
updates == null ||
|
|
399
|
+
updates.length === 0) {
|
|
400
|
+
return collection; // nothing to track
|
|
401
|
+
}
|
|
402
|
+
let didMutate = false;
|
|
403
|
+
const entityMap = collection.entities;
|
|
404
|
+
const changeState = updates.reduce((chgState, update) => {
|
|
405
|
+
const { id, changes: entity } = update;
|
|
406
|
+
if (id == null || id === '') {
|
|
407
|
+
throw new Error(`${collection.entityName} entity update requires a key to be tracked`);
|
|
408
|
+
}
|
|
409
|
+
const originalValue = entityMap[id];
|
|
410
|
+
// Only track if it is in the collection. Silently ignore if it is not.
|
|
411
|
+
// @ngrx/entity adapter would also silently ignore.
|
|
412
|
+
// Todo: should missing update entity really be reported as an error?
|
|
413
|
+
if (originalValue) {
|
|
414
|
+
const trackedChange = chgState[id];
|
|
415
|
+
if (!trackedChange) {
|
|
416
|
+
if (!didMutate) {
|
|
417
|
+
didMutate = true;
|
|
418
|
+
chgState = { ...chgState };
|
|
419
|
+
}
|
|
420
|
+
chgState[id] = { changeType: ChangeType.Updated, originalValue };
|
|
421
|
+
}
|
|
422
|
+
}
|
|
423
|
+
return chgState;
|
|
424
|
+
}, collection.changeState);
|
|
425
|
+
return didMutate ? { ...collection, changeState } : collection;
|
|
426
|
+
}
|
|
427
|
+
/**
|
|
428
|
+
* Track an entity before updating it in the collection.
|
|
429
|
+
* Does NOT update the collection (the reducer's job).
|
|
430
|
+
* @param update The entity to update.
|
|
431
|
+
* @param collection The entity collection
|
|
432
|
+
* @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.
|
|
433
|
+
*/
|
|
434
|
+
trackUpdateOne(update, collection, mergeStrategy) {
|
|
435
|
+
return update == null
|
|
436
|
+
? collection
|
|
437
|
+
: this.trackUpdateMany([update], collection, mergeStrategy);
|
|
438
|
+
}
|
|
439
|
+
/**
|
|
440
|
+
* Track multiple entities before upserting (adding and updating) them to the collection.
|
|
441
|
+
* Does NOT update the collection (the reducer's job).
|
|
442
|
+
* @param entities The entities to add or update. They must be complete entities with ids.
|
|
443
|
+
* @param collection The entity collection
|
|
444
|
+
* @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.
|
|
445
|
+
*/
|
|
446
|
+
trackUpsertMany(entities, collection, mergeStrategy) {
|
|
447
|
+
if (mergeStrategy === MergeStrategy.IgnoreChanges ||
|
|
448
|
+
entities == null ||
|
|
449
|
+
entities.length === 0) {
|
|
450
|
+
return collection; // nothing to track
|
|
451
|
+
}
|
|
452
|
+
let didMutate = false;
|
|
453
|
+
const entityMap = collection.entities;
|
|
454
|
+
const changeState = entities.reduce((chgState, entity) => {
|
|
455
|
+
const id = this.selectId(entity);
|
|
456
|
+
if (id == null || id === '') {
|
|
457
|
+
throw new Error(`${collection.entityName} entity upsert requires a key to be tracked`);
|
|
458
|
+
}
|
|
459
|
+
const trackedChange = chgState[id];
|
|
460
|
+
if (!trackedChange) {
|
|
461
|
+
if (!didMutate) {
|
|
462
|
+
didMutate = true;
|
|
463
|
+
chgState = { ...chgState };
|
|
464
|
+
}
|
|
465
|
+
const originalValue = entityMap[id];
|
|
466
|
+
chgState[id] =
|
|
467
|
+
originalValue == null
|
|
468
|
+
? { changeType: ChangeType.Added }
|
|
469
|
+
: { changeType: ChangeType.Updated, originalValue };
|
|
470
|
+
}
|
|
471
|
+
return chgState;
|
|
472
|
+
}, collection.changeState);
|
|
473
|
+
return didMutate ? { ...collection, changeState } : collection;
|
|
474
|
+
}
|
|
475
|
+
/**
|
|
476
|
+
* Track an entity before upsert (adding and updating) it to the collection.
|
|
477
|
+
* Does NOT update the collection (the reducer's job).
|
|
478
|
+
* @param entities The entity to add or update. It must be a complete entity with its id.
|
|
479
|
+
* @param collection The entity collection
|
|
480
|
+
* @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.
|
|
481
|
+
*/
|
|
482
|
+
trackUpsertOne(entity, collection, mergeStrategy) {
|
|
483
|
+
return entity == null
|
|
484
|
+
? collection
|
|
485
|
+
: this.trackUpsertMany([entity], collection, mergeStrategy);
|
|
486
|
+
}
|
|
487
|
+
// #endregion track methods
|
|
488
|
+
// #region undo methods
|
|
489
|
+
/**
|
|
490
|
+
* Revert the unsaved changes for all collection.
|
|
491
|
+
* Harmless when there are no entity changes to undo.
|
|
492
|
+
* @param collection The entity collection
|
|
493
|
+
*/
|
|
494
|
+
undoAll(collection) {
|
|
495
|
+
const ids = Object.keys(collection.changeState);
|
|
496
|
+
const { remove, upsert } = ids.reduce((acc, id) => {
|
|
497
|
+
const changeState = acc.chgState[id];
|
|
498
|
+
switch (changeState.changeType) {
|
|
499
|
+
case ChangeType.Added:
|
|
500
|
+
acc.remove.push(id);
|
|
501
|
+
break;
|
|
502
|
+
case ChangeType.Deleted:
|
|
503
|
+
const removed = changeState.originalValue;
|
|
504
|
+
if (removed) {
|
|
505
|
+
acc.upsert.push(removed);
|
|
506
|
+
}
|
|
507
|
+
break;
|
|
508
|
+
case ChangeType.Updated:
|
|
509
|
+
acc.upsert.push(changeState.originalValue);
|
|
510
|
+
break;
|
|
511
|
+
}
|
|
512
|
+
return acc;
|
|
513
|
+
},
|
|
514
|
+
// entitiesToUndo
|
|
515
|
+
{
|
|
516
|
+
remove: [],
|
|
517
|
+
upsert: [],
|
|
518
|
+
chgState: collection.changeState,
|
|
519
|
+
});
|
|
520
|
+
collection = this.adapter.removeMany(remove, collection);
|
|
521
|
+
collection = this.adapter.upsertMany(upsert, collection);
|
|
522
|
+
return { ...collection, changeState: {} };
|
|
523
|
+
}
|
|
524
|
+
/**
|
|
525
|
+
* Revert the unsaved changes for the given entities.
|
|
526
|
+
* Harmless when there are no entity changes to undo.
|
|
527
|
+
* @param entityOrIdList The entities to revert or their ids.
|
|
528
|
+
* @param collection The entity collection
|
|
529
|
+
*/
|
|
530
|
+
undoMany(entityOrIdList, collection) {
|
|
531
|
+
if (entityOrIdList == null || entityOrIdList.length === 0) {
|
|
532
|
+
return collection; // nothing to undo
|
|
533
|
+
}
|
|
534
|
+
let didMutate = false;
|
|
535
|
+
const { changeState, remove, upsert } = entityOrIdList.reduce((acc, entityOrId) => {
|
|
536
|
+
let chgState = acc.changeState;
|
|
537
|
+
const id = typeof entityOrId === 'object'
|
|
538
|
+
? this.selectId(entityOrId)
|
|
539
|
+
: entityOrId;
|
|
540
|
+
const change = chgState[id];
|
|
541
|
+
if (change) {
|
|
542
|
+
if (!didMutate) {
|
|
543
|
+
chgState = { ...chgState };
|
|
544
|
+
didMutate = true;
|
|
545
|
+
}
|
|
546
|
+
delete chgState[id]; // clear tracking of this entity
|
|
547
|
+
acc.changeState = chgState;
|
|
548
|
+
switch (change.changeType) {
|
|
549
|
+
case ChangeType.Added:
|
|
550
|
+
acc.remove.push(id);
|
|
551
|
+
break;
|
|
552
|
+
case ChangeType.Deleted:
|
|
553
|
+
const removed = change.originalValue;
|
|
554
|
+
if (removed) {
|
|
555
|
+
acc.upsert.push(removed);
|
|
556
|
+
}
|
|
557
|
+
break;
|
|
558
|
+
case ChangeType.Updated:
|
|
559
|
+
acc.upsert.push(change.originalValue);
|
|
560
|
+
break;
|
|
561
|
+
}
|
|
562
|
+
}
|
|
563
|
+
return acc;
|
|
564
|
+
},
|
|
565
|
+
// entitiesToUndo
|
|
566
|
+
{
|
|
567
|
+
remove: [],
|
|
568
|
+
upsert: [],
|
|
569
|
+
changeState: collection.changeState,
|
|
570
|
+
});
|
|
571
|
+
collection = this.adapter.removeMany(remove, collection);
|
|
572
|
+
collection = this.adapter.upsertMany(upsert, collection);
|
|
573
|
+
return didMutate ? { ...collection, changeState } : collection;
|
|
574
|
+
}
|
|
575
|
+
/**
|
|
576
|
+
* Revert the unsaved changes for the given entity.
|
|
577
|
+
* Harmless when there are no entity changes to undo.
|
|
578
|
+
* @param entityOrId The entity to revert or its id.
|
|
579
|
+
* @param collection The entity collection
|
|
580
|
+
*/
|
|
581
|
+
undoOne(entityOrId, collection) {
|
|
582
|
+
return entityOrId == null
|
|
583
|
+
? collection
|
|
584
|
+
: this.undoMany([entityOrId], collection);
|
|
585
|
+
}
|
|
586
|
+
}
|
|
587
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"entity-change-tracker-base.js","sourceRoot":"","sources":["../../../../../../modules/data/src/reducers/entity-change-tracker-base.ts"],"names":[],"mappings":"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","sourcesContent":["import { EntityAdapter, IdSelector, Update } from '@ngrx/entity';\n\nimport { ChangeType, EntityCollection } from './entity-collection';\nimport { defaultSelectId } from '../utils/utilities';\nimport { EntityChangeTracker } from './entity-change-tracker';\nimport { MergeStrategy } from '../actions/merge-strategy';\nimport { UpdateResponseData } from '../actions/update-response-data';\n\n/**\n * The default implementation of EntityChangeTracker with\n * methods for tracking, committing, and reverting/undoing unsaved entity changes.\n * Used by EntityCollectionReducerMethods which should call tracker methods BEFORE modifying the collection.\n * See EntityChangeTracker docs.\n */\nexport class EntityChangeTrackerBase<T> implements EntityChangeTracker<T> {\n  constructor(\n    private adapter: EntityAdapter<T>,\n    private selectId: IdSelector<T>\n  ) {\n    /** Extract the primary key (id); default to `id` */\n    this.selectId = selectId || defaultSelectId;\n  }\n\n  // #region commit methods\n  /**\n   * Commit all changes as when the collection has been completely reloaded from the server.\n   * Harmless when there are no entity changes to commit.\n   * @param collection The entity collection\n   */\n  commitAll(collection: EntityCollection<T>): EntityCollection<T> {\n    return Object.keys(collection.changeState).length === 0\n      ? collection\n      : { ...collection, changeState: {} };\n  }\n\n  /**\n   * Commit changes for the given entities as when they have been refreshed from the server.\n   * Harmless when there are no entity changes to commit.\n   * @param entityOrIdList The entities to clear tracking or their ids.\n   * @param collection The entity collection\n   */\n  commitMany(\n    entityOrIdList: (number | string | T)[],\n    collection: EntityCollection<T>\n  ): EntityCollection<T> {\n    if (entityOrIdList == null || entityOrIdList.length === 0) {\n      return collection; // nothing to commit\n    }\n    let didMutate = false;\n    const changeState = entityOrIdList.reduce((chgState, entityOrId) => {\n      const id =\n        typeof entityOrId === 'object'\n          ? this.selectId(entityOrId)\n          : (entityOrId as string | number);\n      if (chgState[id]) {\n        if (!didMutate) {\n          chgState = { ...chgState };\n          didMutate = true;\n        }\n        delete chgState[id];\n      }\n      return chgState;\n    }, collection.changeState);\n\n    return didMutate ? { ...collection, changeState } : collection;\n  }\n\n  /**\n   * Commit changes for the given entity as when it have been refreshed from the server.\n   * Harmless when no entity changes to commit.\n   * @param entityOrId The entity to clear tracking or its id.\n   * @param collection The entity collection\n   */\n  commitOne(\n    entityOrId: number | string | T,\n    collection: EntityCollection<T>\n  ): EntityCollection<T> {\n    return entityOrId == null\n      ? collection\n      : this.commitMany([entityOrId], collection);\n  }\n\n  // #endregion commit methods\n\n  // #region merge query\n  /**\n   * Merge query results into the collection, adjusting the ChangeState per the mergeStrategy.\n   * @param entities Entities returned from querying the server.\n   * @param collection The entity collection\n   * @param [mergeStrategy] How to merge a queried entity when the corresponding entity in the collection has an unsaved change.\n   * Defaults to MergeStrategy.PreserveChanges.\n   * @returns The merged EntityCollection.\n   */\n  mergeQueryResults(\n    entities: T[],\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    return this.mergeServerUpserts(\n      entities,\n      collection,\n      MergeStrategy.PreserveChanges,\n      mergeStrategy\n    );\n  }\n  // #endregion merge query results\n\n  // #region merge save results\n  /**\n   * Merge result of saving new entities into the collection, adjusting the ChangeState per the mergeStrategy.\n   * The default is MergeStrategy.OverwriteChanges.\n   * @param entities Entities returned from saving new entities to the server.\n   * @param collection The entity collection\n   * @param [mergeStrategy] How to merge a saved entity when the corresponding entity in the collection has an unsaved change.\n   * Defaults to MergeStrategy.OverwriteChanges.\n   * @returns The merged EntityCollection.\n   */\n  mergeSaveAdds(\n    entities: T[],\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    return this.mergeServerUpserts(\n      entities,\n      collection,\n      MergeStrategy.OverwriteChanges,\n      mergeStrategy\n    );\n  }\n\n  /**\n   * Merge successful result of deleting entities on the server that have the given primary keys\n   * Clears the entity changeState for those keys unless the MergeStrategy is ignoreChanges.\n   * @param entities keys primary keys of the entities to remove/delete.\n   * @param collection The entity collection\n   * @param [mergeStrategy] How to adjust change tracking when the corresponding entity in the collection has an unsaved change.\n   * Defaults to MergeStrategy.OverwriteChanges.\n   * @returns The merged EntityCollection.\n   */\n  mergeSaveDeletes(\n    keys: (number | string)[],\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    mergeStrategy =\n      mergeStrategy == null ? MergeStrategy.OverwriteChanges : mergeStrategy;\n    // same logic for all non-ignore merge strategies: always clear (commit) the changes\n    const deleteIds = keys as string[]; // make TypeScript happy\n    collection =\n      mergeStrategy === MergeStrategy.IgnoreChanges\n        ? collection\n        : this.commitMany(deleteIds, collection);\n    return this.adapter.removeMany(deleteIds, collection);\n  }\n\n  /**\n   * Merge result of saving updated entities into the collection, adjusting the ChangeState per the mergeStrategy.\n   * The default is MergeStrategy.OverwriteChanges.\n   * @param updateResponseData Entity response data returned from saving updated entities to the server.\n   * @param collection The entity collection\n   * @param [mergeStrategy] How to merge a saved entity when the corresponding entity in the collection has an unsaved change.\n   * Defaults to MergeStrategy.OverwriteChanges.\n   * @param [skipUnchanged] True means skip update if server didn't change it. False by default.\n   * If the update was optimistic and the server didn't make more changes of its own\n   * then the updates are already in the collection and shouldn't make them again.\n   * @returns The merged EntityCollection.\n   */\n  mergeSaveUpdates(\n    updateResponseData: UpdateResponseData<T>[],\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy,\n    skipUnchanged = false\n  ): EntityCollection<T> {\n    if (updateResponseData == null || updateResponseData.length === 0) {\n      return collection; // nothing to merge.\n    }\n\n    let didMutate = false;\n    let changeState = collection.changeState;\n    mergeStrategy =\n      mergeStrategy == null ? MergeStrategy.OverwriteChanges : mergeStrategy;\n    let updates: Update<T>[];\n\n    switch (mergeStrategy) {\n      case MergeStrategy.IgnoreChanges:\n        updates = filterChanged(updateResponseData);\n        return this.adapter.updateMany(updates, collection);\n\n      case MergeStrategy.OverwriteChanges:\n        changeState = updateResponseData.reduce((chgState, update) => {\n          const oldId = update.id;\n          const change = chgState[oldId];\n          if (change) {\n            if (!didMutate) {\n              chgState = { ...chgState };\n              didMutate = true;\n            }\n            delete chgState[oldId];\n          }\n          return chgState;\n        }, collection.changeState);\n\n        collection = didMutate ? { ...collection, changeState } : collection;\n\n        updates = filterChanged(updateResponseData);\n        return this.adapter.updateMany(updates, collection);\n\n      case MergeStrategy.PreserveChanges: {\n        const updateableEntities = [] as UpdateResponseData<T>[];\n        changeState = updateResponseData.reduce((chgState, update) => {\n          const oldId = update.id;\n          const change = chgState[oldId];\n          if (change) {\n            // Tracking a change so update original value but not the current value\n            if (!didMutate) {\n              chgState = { ...chgState };\n              didMutate = true;\n            }\n            const newId = this.selectId(update.changes as T);\n            const oldChangeState = change;\n            // If the server changed the id, register the new \"originalValue\" under the new id\n            // and remove the change tracked under the old id.\n            if (newId !== oldId) {\n              delete chgState[oldId];\n            }\n            const newOrigValue = {\n              ...(oldChangeState!.originalValue as any),\n              ...(update.changes as any),\n            };\n            (chgState as any)[newId] = {\n              ...oldChangeState,\n              originalValue: newOrigValue,\n            };\n          } else {\n            updateableEntities.push(update);\n          }\n          return chgState;\n        }, collection.changeState);\n        collection = didMutate ? { ...collection, changeState } : collection;\n\n        updates = filterChanged(updateableEntities);\n        return this.adapter.updateMany(updates, collection);\n      }\n    }\n\n    /**\n     * Conditionally keep only those updates that have additional server changes.\n     * (e.g., for optimistic saves because they updates are already in the current collection)\n     * Strip off the `changed` property.\n     * @responseData Entity response data from server.\n     * May be an UpdateResponseData<T>, a subclass of Update<T> with a 'changed' flag.\n     * @returns Update<T> (without the changed flag)\n     */\n    function filterChanged(responseData: UpdateResponseData<T>[]): Update<T>[] {\n      if (skipUnchanged === true) {\n        // keep only those updates that the server changed (knowable if is UpdateResponseData<T>)\n        responseData = responseData.filter((r) => r.changed === true);\n      }\n      // Strip unchanged property from responseData, leaving just the pure Update<T>\n      // TODO: Remove? probably not necessary as the Update isn't stored and adapter will ignore `changed`.\n      return responseData.map((r) => ({ id: r.id as any, changes: r.changes }));\n    }\n  }\n\n  /**\n   * Merge result of saving upserted entities into the collection, adjusting the ChangeState per the mergeStrategy.\n   * The default is MergeStrategy.OverwriteChanges.\n   * @param entities Entities returned from saving upserts to the server.\n   * @param collection The entity collection\n   * @param [mergeStrategy] How to merge a saved entity when the corresponding entity in the collection has an unsaved change.\n   * Defaults to MergeStrategy.OverwriteChanges.\n   * @returns The merged EntityCollection.\n   */\n  mergeSaveUpserts(\n    entities: T[],\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    return this.mergeServerUpserts(\n      entities,\n      collection,\n      MergeStrategy.OverwriteChanges,\n      mergeStrategy\n    );\n  }\n  // #endregion merge save results\n\n  // #region query & save helpers\n  /**\n   *\n   * @param entities Entities to merge\n   * @param collection Collection into which entities are merged\n   * @param defaultMergeStrategy How to merge when action's MergeStrategy is unspecified\n   * @param [mergeStrategy] The action's MergeStrategy\n   */\n  private mergeServerUpserts(\n    entities: T[],\n    collection: EntityCollection<T>,\n    defaultMergeStrategy: MergeStrategy,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    if (entities == null || entities.length === 0) {\n      return collection; // nothing to merge.\n    }\n\n    let didMutate = false;\n    let changeState = collection.changeState;\n    mergeStrategy =\n      mergeStrategy == null ? defaultMergeStrategy : mergeStrategy;\n\n    switch (mergeStrategy) {\n      case MergeStrategy.IgnoreChanges:\n        return this.adapter.upsertMany(entities, collection);\n\n      case MergeStrategy.OverwriteChanges:\n        collection = this.adapter.upsertMany(entities, collection);\n\n        changeState = entities.reduce((chgState, entity) => {\n          const id = this.selectId(entity);\n          const change = chgState[id];\n          if (change) {\n            if (!didMutate) {\n              chgState = { ...chgState };\n              didMutate = true;\n            }\n            delete chgState[id];\n          }\n          return chgState;\n        }, collection.changeState);\n\n        return didMutate ? { ...collection, changeState } : collection;\n\n      case MergeStrategy.PreserveChanges: {\n        const upsertEntities = [] as T[];\n        changeState = entities.reduce((chgState, entity) => {\n          const id = this.selectId(entity);\n          const change = chgState[id];\n          if (change) {\n            if (!didMutate) {\n              chgState = {\n                ...chgState,\n                [id]: {\n                  ...chgState[id]!,\n                  originalValue: entity,\n                },\n              };\n              didMutate = true;\n            }\n          } else {\n            upsertEntities.push(entity);\n          }\n          return chgState;\n        }, collection.changeState);\n\n        collection = this.adapter.upsertMany(upsertEntities, collection);\n        return didMutate ? { ...collection, changeState } : collection;\n      }\n    }\n  }\n  // #endregion query & save helpers\n\n  // #region track methods\n  /**\n   * Track multiple entities before adding them to the collection.\n   * Does NOT add to the collection (the reducer's job).\n   * @param entities The entities to add. They must all have their ids.\n   * @param collection The entity collection\n   * @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.\n   */\n  trackAddMany(\n    entities: T[],\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    if (\n      mergeStrategy === MergeStrategy.IgnoreChanges ||\n      entities == null ||\n      entities.length === 0\n    ) {\n      return collection; // nothing to track\n    }\n    let didMutate = false;\n    const changeState = entities.reduce((chgState, entity) => {\n      const id = this.selectId(entity);\n      if (id == null || id === '') {\n        throw new Error(\n          `${collection.entityName} entity add requires a key to be tracked`\n        );\n      }\n      const trackedChange = chgState[id];\n\n      if (!trackedChange) {\n        if (!didMutate) {\n          didMutate = true;\n          chgState = { ...chgState };\n        }\n        chgState[id] = { changeType: ChangeType.Added };\n      }\n      return chgState;\n    }, collection.changeState);\n    return didMutate ? { ...collection, changeState } : collection;\n  }\n\n  /**\n   * Track an entity before adding it to the collection.\n   * Does NOT add to the collection (the reducer's job).\n   * @param entity The entity to add. It must have an id.\n   * @param collection The entity collection\n   * @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.\n   * If not specified, implementation supplies a default strategy.\n   */\n  trackAddOne(\n    entity: T,\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    return entity == null\n      ? collection\n      : this.trackAddMany([entity], collection, mergeStrategy);\n  }\n\n  /**\n   * Track multiple entities before removing them with the intention of deleting them on the server.\n   * Does NOT remove from the collection (the reducer's job).\n   * @param keys The primary keys of the entities to delete.\n   * @param collection The entity collection\n   * @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.\n   */\n  trackDeleteMany(\n    keys: (number | string)[],\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    if (\n      mergeStrategy === MergeStrategy.IgnoreChanges ||\n      keys == null ||\n      keys.length === 0\n    ) {\n      return collection; // nothing to track\n    }\n    let didMutate = false;\n    const entityMap = collection.entities;\n    const changeState = keys.reduce((chgState, id) => {\n      const originalValue = entityMap[id];\n      if (originalValue) {\n        const trackedChange = chgState[id];\n        if (trackedChange) {\n          if (trackedChange.changeType === ChangeType.Added) {\n            // Special case: stop tracking an added entity that you delete\n            // The caller must also detect this, remove it immediately from the collection\n            // and skip attempt to delete on the server.\n            cloneChgStateOnce();\n            delete chgState[id];\n          } else if (trackedChange.changeType === ChangeType.Updated) {\n            // Special case: switch change type from Updated to Deleted.\n            cloneChgStateOnce();\n            chgState[id] = { ...chgState[id], changeType: ChangeType.Deleted };\n          }\n        } else {\n          // Start tracking this entity\n          cloneChgStateOnce();\n          chgState[id] = { changeType: ChangeType.Deleted, originalValue };\n        }\n      }\n      return chgState;\n\n      function cloneChgStateOnce() {\n        if (!didMutate) {\n          didMutate = true;\n          chgState = { ...chgState };\n        }\n      }\n    }, collection.changeState);\n\n    return didMutate ? { ...collection, changeState } : collection;\n  }\n\n  /**\n   * Track an entity before it is removed with the intention of deleting it on the server.\n   * Does NOT remove from the collection (the reducer's job).\n   * @param key The primary key of the entity to delete.\n   * @param collection The entity collection\n   * @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.\n   */\n  trackDeleteOne(\n    key: number | string,\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    return key == null\n      ? collection\n      : this.trackDeleteMany([key], collection, mergeStrategy);\n  }\n\n  /**\n   * Track multiple entities before updating them in the collection.\n   * Does NOT update the collection (the reducer's job).\n   * @param updates The entities to update.\n   * @param collection The entity collection\n   * @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.\n   */\n  trackUpdateMany(\n    updates: Update<T>[],\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    if (\n      mergeStrategy === MergeStrategy.IgnoreChanges ||\n      updates == null ||\n      updates.length === 0\n    ) {\n      return collection; // nothing to track\n    }\n    let didMutate = false;\n    const entityMap = collection.entities;\n    const changeState = updates.reduce((chgState, update) => {\n      const { id, changes: entity } = update;\n      if (id == null || id === '') {\n        throw new Error(\n          `${collection.entityName} entity update requires a key to be tracked`\n        );\n      }\n      const originalValue = entityMap[id];\n      // Only track if it is in the collection. Silently ignore if it is not.\n      // @ngrx/entity adapter would also silently ignore.\n      // Todo: should missing update entity really be reported as an error?\n      if (originalValue) {\n        const trackedChange = chgState[id];\n        if (!trackedChange) {\n          if (!didMutate) {\n            didMutate = true;\n            chgState = { ...chgState };\n          }\n          chgState[id] = { changeType: ChangeType.Updated, originalValue };\n        }\n      }\n      return chgState;\n    }, collection.changeState);\n    return didMutate ? { ...collection, changeState } : collection;\n  }\n\n  /**\n   * Track an entity before updating it in the collection.\n   * Does NOT update the collection (the reducer's job).\n   * @param update The entity to update.\n   * @param collection The entity collection\n   * @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.\n   */\n  trackUpdateOne(\n    update: Update<T>,\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    return update == null\n      ? collection\n      : this.trackUpdateMany([update], collection, mergeStrategy);\n  }\n\n  /**\n   * Track multiple entities before upserting (adding and updating) them to the collection.\n   * Does NOT update the collection (the reducer's job).\n   * @param entities The entities to add or update. They must be complete entities with ids.\n   * @param collection The entity collection\n   * @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.\n   */\n  trackUpsertMany(\n    entities: T[],\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    if (\n      mergeStrategy === MergeStrategy.IgnoreChanges ||\n      entities == null ||\n      entities.length === 0\n    ) {\n      return collection; // nothing to track\n    }\n    let didMutate = false;\n    const entityMap = collection.entities;\n    const changeState = entities.reduce((chgState, entity) => {\n      const id = this.selectId(entity);\n      if (id == null || id === '') {\n        throw new Error(\n          `${collection.entityName} entity upsert requires a key to be tracked`\n        );\n      }\n      const trackedChange = chgState[id];\n\n      if (!trackedChange) {\n        if (!didMutate) {\n          didMutate = true;\n          chgState = { ...chgState };\n        }\n\n        const originalValue = entityMap[id];\n        chgState[id] =\n          originalValue == null\n            ? { changeType: ChangeType.Added }\n            : { changeType: ChangeType.Updated, originalValue };\n      }\n      return chgState;\n    }, collection.changeState);\n    return didMutate ? { ...collection, changeState } : collection;\n  }\n\n  /**\n   * Track an entity before upsert (adding and updating) it to the collection.\n   * Does NOT update the collection (the reducer's job).\n   * @param entities The entity to add or update. It must be a complete entity with its id.\n   * @param collection The entity collection\n   * @param [mergeStrategy] Track by default. Don't track if is MergeStrategy.IgnoreChanges.\n   */\n  trackUpsertOne(\n    entity: T,\n    collection: EntityCollection<T>,\n    mergeStrategy?: MergeStrategy\n  ): EntityCollection<T> {\n    return entity == null\n      ? collection\n      : this.trackUpsertMany([entity], collection, mergeStrategy);\n  }\n  // #endregion track methods\n\n  // #region undo methods\n  /**\n   * Revert the unsaved changes for all collection.\n   * Harmless when there are no entity changes to undo.\n   * @param collection The entity collection\n   */\n  undoAll(collection: EntityCollection<T>): EntityCollection<T> {\n    const ids = Object.keys(collection.changeState);\n\n    const { remove, upsert } = ids.reduce(\n      (acc, id) => {\n        const changeState = acc.chgState[id]!;\n        switch (changeState.changeType) {\n          case ChangeType.Added:\n            acc.remove.push(id);\n            break;\n          case ChangeType.Deleted:\n            const removed = changeState!.originalValue;\n            if (removed) {\n              acc.upsert.push(removed);\n            }\n            break;\n          case ChangeType.Updated:\n            acc.upsert.push(changeState!.originalValue!);\n            break;\n        }\n        return acc;\n      },\n      // entitiesToUndo\n      {\n        remove: [] as (number | string)[],\n        upsert: [] as T[],\n        chgState: collection.changeState,\n      }\n    );\n\n    collection = this.adapter.removeMany(remove as string[], collection);\n    collection = this.adapter.upsertMany(upsert, collection);\n\n    return { ...collection, changeState: {} };\n  }\n\n  /**\n   * Revert the unsaved changes for the given entities.\n   * Harmless when there are no entity changes to undo.\n   * @param entityOrIdList The entities to revert or their ids.\n   * @param collection The entity collection\n   */\n  undoMany(\n    entityOrIdList: (number | string | T)[],\n    collection: EntityCollection<T>\n  ): EntityCollection<T> {\n    if (entityOrIdList == null || entityOrIdList.length === 0) {\n      return collection; // nothing to undo\n    }\n    let didMutate = false;\n\n    const { changeState, remove, upsert } = entityOrIdList.reduce(\n      (acc, entityOrId) => {\n        let chgState = acc.changeState;\n        const id =\n          typeof entityOrId === 'object'\n            ? this.selectId(entityOrId)\n            : (entityOrId as string | number);\n        const change = chgState[id]!;\n        if (change) {\n          if (!didMutate) {\n            chgState = { ...chgState };\n            didMutate = true;\n          }\n          delete chgState[id]; // clear tracking of this entity\n          acc.changeState = chgState;\n          switch (change.changeType) {\n            case ChangeType.Added:\n              acc.remove.push(id);\n              break;\n            case ChangeType.Deleted:\n              const removed = change!.originalValue;\n              if (removed) {\n                acc.upsert.push(removed);\n              }\n              break;\n            case ChangeType.Updated:\n              acc.upsert.push(change!.originalValue!);\n              break;\n          }\n        }\n        return acc;\n      },\n      // entitiesToUndo\n      {\n        remove: [] as (number | string)[],\n        upsert: [] as T[],\n        changeState: collection.changeState,\n      }\n    );\n\n    collection = this.adapter.removeMany(remove as string[], collection);\n    collection = this.adapter.upsertMany(upsert, collection);\n    return didMutate ? { ...collection, changeState } : collection;\n  }\n\n  /**\n   * Revert the unsaved changes for the given entity.\n   * Harmless when there are no entity changes to undo.\n   * @param entityOrId The entity to revert or its id.\n   * @param collection The entity collection\n   */\n  undoOne(\n    entityOrId: number | string | T,\n    collection: EntityCollection<T>\n  ): EntityCollection<T> {\n    return entityOrId == null\n      ? collection\n      : this.undoMany([entityOrId], collection);\n  }\n  // #endregion undo methods\n}\n"]}
|