|
1 |
| -import { Folder, TItem, ItemType } from './Tree' |
2 |
| -import { Mapping } from './Mappings' |
| 1 | +import { Folder, TItem, ItemType, TItemLocation, ItemLocation } from './Tree' |
| 2 | +import Mappings, { MappingSnapshot } from './Mappings' |
3 | 3 | import Ordering from './interfaces/Ordering'
|
4 | 4 | import batchingToposort from 'batching-toposort'
|
5 | 5 | import Logger from './Logger'
|
@@ -40,7 +40,7 @@ export interface ReorderAction {
|
40 | 40 | type: 'REORDER',
|
41 | 41 | payload: TItem,
|
42 | 42 | oldItem?: TItem,
|
43 |
| - order?: Ordering, |
| 43 | + order: Ordering, |
44 | 44 | oldOrder?: Ordering,
|
45 | 45 | }
|
46 | 46 |
|
@@ -112,16 +112,6 @@ export default class Diff {
|
112 | 112 | }
|
113 | 113 | }
|
114 | 114 |
|
115 |
| - add(diff: Diff, types:TActionType[] = []):void { |
116 |
| - if (types.length === 0) { |
117 |
| - diff.getActions().forEach(action => this.commit({...action})) |
118 |
| - return |
119 |
| - } |
120 |
| - types.forEach(type => |
121 |
| - diff.getActions(type).forEach(action => this.commit({...action})) |
122 |
| - ) |
123 |
| - } |
124 |
| - |
125 | 115 | getActions(type?: TActionType):Action[] {
|
126 | 116 | if (type) {
|
127 | 117 | return this.actions[type].slice()
|
@@ -170,62 +160,76 @@ export default class Diff {
|
170 | 160 | /**
|
171 | 161 | * on ServerToLocal: don't map removals
|
172 | 162 | * on LocalToServer:
|
173 |
| - * @param mappings |
174 |
| - * @param isLocalToServer |
| 163 | + * @param mappingsSnapshot |
| 164 | + * @param targetLocation |
175 | 165 | * @param filter
|
176 | 166 | */
|
177 |
| - map(mappings:Mapping, isLocalToServer: boolean, filter: (Action)=>boolean = () => true):void { |
| 167 | + map(mappingsSnapshot:MappingSnapshot, targetLocation: TItemLocation, filter: (Action)=>boolean = () => true): Diff { |
| 168 | + const newDiff = new Diff |
| 169 | + |
178 | 170 | // Map payloads
|
179 |
| - this.getActions().forEach(action => { |
180 |
| - if (action.type === ActionType.REMOVE && !isLocalToServer) { |
181 |
| - return |
182 |
| - } |
| 171 | + this.getActions() |
| 172 | + .map(a => a as Action) |
| 173 | + .forEach(action => { |
| 174 | + let newAction |
| 175 | + |
| 176 | + if (!filter(action)) { |
| 177 | + newDiff.commit(action) |
| 178 | + return |
| 179 | + } |
183 | 180 |
|
184 |
| - if (!filter(action)) { |
185 |
| - return |
186 |
| - } |
| 181 | + /* if (action.type === ActionType.REMOVE && targetLocation !== ItemLocation.SERVER) { |
| 182 | + newDiff.commit({...action}) |
| 183 | + return |
| 184 | + } */ |
187 | 185 |
|
188 |
| - Logger.log('Mapping action ' + action.type + ' ' + (isLocalToServer ? 'LocalToServer' : 'ServerToLocal'), {...action}) |
| 186 | + Logger.log('Mapping action ' + action.type + ' to ' + targetLocation, {...action}) |
189 | 187 |
|
190 |
| - if (action.type === ActionType.REORDER) { |
191 |
| - action.oldOrder = action.order |
192 |
| - action.order = action.order.slice().map(item => { |
193 |
| - return {...item, id: mappings[item.type ][item.id]} |
194 |
| - }) |
195 |
| - } |
196 |
| - |
197 |
| - // needed because we set oldItem in the first section, so we wouldn't know anymore if it was set before |
198 |
| - const oldItem = action.oldItem |
199 |
| - |
200 |
| - // We have two sections here, because we want to be able to take IDs from oldItem even for moves |
201 |
| - // but not parentIds (which do change during moves, obviously) |
202 |
| - |
203 |
| - if (oldItem && !isLocalToServer) { |
204 |
| - const oldId = action.oldItem.id |
205 |
| - const newId = action.payload.id |
206 |
| - action.oldItem = action.oldItem.clone() |
207 |
| - action.payload = action.payload.clone() |
208 |
| - action.payload.id = oldId |
209 |
| - action.oldItem.id = newId |
210 |
| - } else { |
211 |
| - const newPayload = action.payload.clone() |
212 |
| - newPayload.id = mappings[newPayload.type][newPayload.id] |
213 |
| - action.oldItem = action.payload.clone() |
214 |
| - action.payload = newPayload |
215 |
| - } |
216 |
| - |
217 |
| - if (oldItem && !isLocalToServer && action.type !== ActionType.MOVE) { |
218 |
| - const oldParent = action.oldItem.parentId |
219 |
| - const newParent = action.payload.parentId |
220 |
| - action.payload.parentId = oldParent |
221 |
| - action.oldItem.parentId = newParent |
222 |
| - } else { |
223 |
| - if (typeof action.payload.parentId !== 'undefined' && typeof mappings.folder[action.payload.parentId] === 'undefined') { |
224 |
| - throw new Error('Cannot map parentId:' + action.payload.parentId) |
| 188 | + // needed because we set oldItem in the first section, so we wouldn't know anymore if it was set before |
| 189 | + const oldItem = action.oldItem |
| 190 | + |
| 191 | + // We have two sections here, because we want to be able to take IDs from oldItem even for moves |
| 192 | + // but not parentIds (which do change during moves, obviously) |
| 193 | + |
| 194 | + if (oldItem && targetLocation !== ItemLocation.SERVER) { |
| 195 | + const oldId = action.oldItem.id |
| 196 | + const newId = action.payload.id |
| 197 | + newAction = { |
| 198 | + ...action, |
| 199 | + payload: action.payload.clone(false, targetLocation), |
| 200 | + oldItem: action.oldItem.clone(false) |
| 201 | + } |
| 202 | + newAction.payload.id = oldId |
| 203 | + newAction.oldItem.id = newId |
| 204 | + } else { |
| 205 | + newAction = { |
| 206 | + ...action, |
| 207 | + payload: action.payload.clone(false, targetLocation), |
| 208 | + oldItem: action.payload.clone(false) |
| 209 | + } |
| 210 | + newAction.payload.id = Mappings.mapId(mappingsSnapshot, action.payload, targetLocation) |
| 211 | + } |
| 212 | + |
| 213 | + if (oldItem && targetLocation !== ItemLocation.SERVER && action.type !== ActionType.MOVE) { |
| 214 | + newAction.payload.parentId = action.oldItem.parentId |
| 215 | + newAction.oldItem.parentId = action.payload.parentId |
| 216 | + } else { |
| 217 | + newAction.oldItem.parentId = action.payload.parentId |
| 218 | + newAction.payload.parentId = Mappings.mapParentId(mappingsSnapshot, action.payload, targetLocation) |
| 219 | + if (typeof newAction.payload.parentId === 'undefined' && typeof action.payload.parentId !== 'undefined') { |
| 220 | + throw new Error('Failed to map parentId: ' + action.payload.parentId) |
| 221 | + } |
| 222 | + } |
| 223 | + |
| 224 | + if (action.type === ActionType.REORDER) { |
| 225 | + newAction.oldOrder = action.order |
| 226 | + newAction.order = action.order.slice().map(item => { |
| 227 | + return {...item, id: mappingsSnapshot[(targetLocation === ItemLocation.LOCAL ? ItemLocation.SERVER : ItemLocation.LOCAL) + 'To' + targetLocation][item.type][item.id]} |
| 228 | + }) |
225 | 229 | }
|
226 |
| - action.oldItem.parentId = action.payload.parentId |
227 |
| - action.payload.parentId = mappings.folder[action.payload.parentId] |
228 |
| - } |
229 |
| - }) |
| 230 | + |
| 231 | + newDiff.commit(newAction) |
| 232 | + }) |
| 233 | + return newDiff |
230 | 234 | }
|
231 | 235 | }
|
0 commit comments