@plait/core 0.0.55 → 0.0.56

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.
@@ -16,11 +16,20 @@ export const inverse = (op) => {
16
16
  if (Path.equals(newPath, path)) {
17
17
  return op;
18
18
  }
19
- if (Path.isSibling(path, newPath)) {
20
- const inversePath = Path.transform(path, op);
21
- return { ...op, path: inversePath, newPath: path };
22
- }
23
- return { ...op, path: newPath, newPath: path };
19
+ // when operation path is [0,0] -> [0,2], should exec Path.transform to get [0,1] -> [0,0]
20
+ // shoud not return [0,2] -> [0,0] #WIK-8981
21
+ // if (Path.isSibling(path, newPath)) {
22
+ // return { ...op, path: newPath, newPath: path };
23
+ // }
24
+ // If the move does not happen within a single parent it is possible
25
+ // for the move to impact the true path to the location where the node
26
+ // was removed from and where it was inserted. We have to adjust for this
27
+ // and find the original path. We can accomplish this (only in non-sibling)
28
+ // moves by looking at the impact of the move operation on the node
29
+ // after the original move path.
30
+ const inversePath = Path.transform(path, op);
31
+ const inverseNewPath = Path.transform(Path.next(path), op);
32
+ return { ...op, path: inversePath, newPath: inverseNewPath };
24
33
  }
25
34
  case 'set_node': {
26
35
  const { properties, newProperties } = op;
@@ -72,4 +81,4 @@ export const PlaitOperation = {
72
81
  isSetViewportOperation,
73
82
  inverse
74
83
  };
75
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib3BlcmF0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcGFja2FnZXMvcGxhaXQvc3JjL2ludGVyZmFjZXMvb3BlcmF0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSxJQUFJLEVBQUUsTUFBTSxRQUFRLENBQUM7QUFzRDlCLE1BQU0sQ0FBQyxNQUFNLHNCQUFzQixHQUFHLENBQUMsS0FBVSxFQUFpQyxFQUFFO0lBQ2hGLE9BQU8sS0FBSyxDQUFDLElBQUksS0FBSyxjQUFjLENBQUM7QUFDekMsQ0FBQyxDQUFDO0FBRUYsTUFBTSxDQUFDLE1BQU0sT0FBTyxHQUFHLENBQUMsRUFBa0IsRUFBa0IsRUFBRTtJQUMxRCxRQUFRLEVBQUUsQ0FBQyxJQUFJLEVBQUU7UUFDYixLQUFLLGFBQWEsQ0FBQyxDQUFDO1lBQ2hCLE9BQU8sRUFBRSxHQUFHLEVBQUUsRUFBRSxJQUFJLEVBQUUsYUFBYSxFQUFFLENBQUM7U0FDekM7UUFFRCxLQUFLLGFBQWEsQ0FBQyxDQUFDO1lBQ2hCLE9BQU8sRUFBRSxHQUFHLEVBQUUsRUFBRSxJQUFJLEVBQUUsYUFBYSxFQUFFLENBQUM7U0FDekM7UUFFRCxLQUFLLFdBQVcsQ0FBQyxDQUFDO1lBQ2QsTUFBTSxFQUFFLE9BQU8sRUFBRSxJQUFJLEVBQUUsR0FBRyxFQUFFLENBQUM7WUFFN0IsNERBQTREO1lBQzVELElBQUksSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLEVBQUUsSUFBSSxDQUFDLEVBQUU7Z0JBQzVCLE9BQU8sRUFBRSxDQUFDO2FBQ2I7WUFFRCxJQUFJLElBQUksQ0FBQyxTQUFTLENBQUMsSUFBSSxFQUFFLE9BQU8sQ0FBQyxFQUFFO2dCQUMvQixNQUFNLFdBQVcsR0FBRyxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksRUFBRSxFQUFFLENBQUUsQ0FBQztnQkFDOUMsT0FBTyxFQUFFLEdBQUcsRUFBRSxFQUFFLElBQUksRUFBRSxXQUFXLEVBQUUsT0FBTyxFQUFFLElBQUksRUFBRSxDQUFDO2FBQ3REO1lBRUQsT0FBTyxFQUFFLEdBQUcsRUFBRSxFQUFFLElBQUksRUFBRSxPQUFPLEVBQUUsT0FBTyxFQUFFLElBQUksRUFBRSxDQUFDO1NBQ2xEO1FBRUQsS0FBSyxVQUFVLENBQUMsQ0FBQztZQUNiLE1BQU0sRUFBRSxVQUFVLEVBQUUsYUFBYSxFQUFFLEdBQUcsRUFBRSxDQUFDO1lBQ3pDLE9BQU8sRUFBRSxHQUFHLEVBQUUsRUFBRSxVQUFVLEVBQUUsYUFBYSxFQUFFLGFBQWEsRUFBRSxVQUFVLEVBQUUsQ0FBQztTQUMxRTtRQUVELEtBQUssZUFBZSxDQUFDLENBQUM7WUFDbEIsTUFBTSxFQUFFLFVBQVUsRUFBRSxhQUFhLEVBQUUsR0FBRyxFQUFFLENBQUM7WUFFekMsSUFBSSxVQUFVLElBQUksSUFBSSxFQUFFO2dCQUNwQixPQUFPO29CQUNILEdBQUcsRUFBRTtvQkFDTCxVQUFVLEVBQUUsYUFBYTtvQkFDekIsYUFBYSxFQUFFLElBQUk7aUJBQ3RCLENBQUM7YUFDTDtpQkFBTSxJQUFJLGFBQWEsSUFBSSxJQUFJLEVBQUU7Z0JBQzlCLE9BQU87b0JBQ0gsR0FBRyxFQUFFO29CQUNMLFVBQVUsRUFBRSxJQUFJO29CQUNoQixhQUFhLEVBQUUsVUFBVTtpQkFDNUIsQ0FBQzthQUNMO2lCQUFNO2dCQUNILE9BQU8sRUFBRSxHQUFHLEVBQUUsRUFBRSxVQUFVLEVBQUUsYUFBYSxFQUFFLGFBQWEsRUFBRSxVQUFVLEVBQUUsQ0FBQzthQUMxRTtTQUNKO1FBRUQsS0FBSyxjQUFjLENBQUMsQ0FBQztZQUNqQixNQUFNLEVBQUUsVUFBVSxFQUFFLGFBQWEsRUFBRSxHQUFHLEVBQUUsQ0FBQztZQUN6QyxJQUFJLFVBQVUsSUFBSSxJQUFJLEVBQUU7Z0JBQ3BCLE9BQU87b0JBQ0gsR0FBRyxFQUFFO29CQUNMLFVBQVUsRUFBRSxhQUFhO29CQUN6QixhQUFhLEVBQUUsYUFBYTtpQkFDL0IsQ0FBQzthQUNMO2lCQUFNLElBQUksYUFBYSxJQUFJLElBQUksRUFBRTtnQkFDOUIsT0FBTztvQkFDSCxHQUFHLEVBQUU7b0JBQ0wsVUFBVSxFQUFFLFVBQVU7b0JBQ3RCLGFBQWEsRUFBRSxVQUFVO2lCQUM1QixDQUFDO2FBQ0w7aUJBQU07Z0JBQ0gsT0FBTyxFQUFFLEdBQUcsRUFBRSxFQUFFLFVBQVUsRUFBRSxhQUFhLEVBQUUsYUFBYSxFQUFFLFVBQVUsRUFBRSxDQUFDO2FBQzFFO1NBQ0o7S0FDSjtBQUNMLENBQUMsQ0FBQztBQUVGLE1BQU0sQ0FBQyxNQUFNLGNBQWMsR0FBNEI7SUFDbkQsc0JBQXNCO0lBQ3RCLE9BQU87Q0FDVixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUGxhaXROb2RlIH0gZnJvbSAnLi9ub2RlJztcbmltcG9ydCB7IFBhdGggfSBmcm9tICcuL3BhdGgnO1xuaW1wb3J0IHsgU2VsZWN0aW9uIH0gZnJvbSAnLi9zZWxlY3Rpb24nO1xuaW1wb3J0IHsgVmlld3BvcnQgfSBmcm9tICcuL3ZpZXdwb3J0JztcblxuZXhwb3J0IHR5cGUgSW5zZXJ0Tm9kZU9wZXJhdGlvbiA9IHtcbiAgICB0eXBlOiAnaW5zZXJ0X25vZGUnO1xuICAgIHBhdGg6IFBhdGg7XG4gICAgbm9kZTogUGxhaXROb2RlO1xufTtcblxuZXhwb3J0IHR5cGUgUmVtb3ZlTm9kZU9wZXJhdGlvbiA9IHtcbiAgICB0eXBlOiAncmVtb3ZlX25vZGUnO1xuICAgIHBhdGg6IFBhdGg7XG4gICAgbm9kZTogUGxhaXROb2RlO1xufTtcblxuZXhwb3J0IHR5cGUgTW92ZU5vZGVPcGVyYXRpb24gPSB7XG4gICAgdHlwZTogJ21vdmVfbm9kZSc7XG4gICAgcGF0aDogUGF0aDtcbiAgICBuZXdQYXRoOiBQYXRoO1xufTtcblxuZXhwb3J0IHR5cGUgU2V0Vmlld3BvcnRPcGVyYXRpb24gPSB7XG4gICAgdHlwZTogJ3NldF92aWV3cG9ydCc7XG4gICAgcHJvcGVydGllczogUGFydGlhbDxWaWV3cG9ydD47XG4gICAgbmV3UHJvcGVydGllczogUGFydGlhbDxWaWV3cG9ydD47XG59O1xuXG5leHBvcnQgdHlwZSBTZXRTZWxlY3Rpb25PcGVyYXRpb24gPSB7XG4gICAgdHlwZTogJ3NldF9zZWxlY3Rpb24nO1xuICAgIHByb3BlcnRpZXM6IFNlbGVjdGlvbiB8IG51bGw7XG4gICAgbmV3UHJvcGVydGllczogU2VsZWN0aW9uIHwgbnVsbDtcbn07XG5cbmV4cG9ydCB0eXBlIFNldE5vZGVPcGVyYXRpb24gPSB7XG4gICAgdHlwZTogJ3NldF9ub2RlJztcbiAgICBwYXRoOiBQYXRoO1xuICAgIHByb3BlcnRpZXM6IFBhcnRpYWw8UGxhaXROb2RlPjtcbiAgICBuZXdQcm9wZXJ0aWVzOiBQYXJ0aWFsPFBsYWl0Tm9kZT47XG59O1xuXG5leHBvcnQgdHlwZSBQbGFpdE9wZXJhdGlvbiA9XG4gICAgfCBJbnNlcnROb2RlT3BlcmF0aW9uXG4gICAgfCBTZXRWaWV3cG9ydE9wZXJhdGlvblxuICAgIHwgU2V0U2VsZWN0aW9uT3BlcmF0aW9uXG4gICAgfCBTZXROb2RlT3BlcmF0aW9uXG4gICAgfCBSZW1vdmVOb2RlT3BlcmF0aW9uXG4gICAgfCBNb3ZlTm9kZU9wZXJhdGlvbjtcblxuZXhwb3J0IGludGVyZmFjZSBQbGFpdE9wZXJhdGlvbkludGVyZmFjZSB7XG4gICAgaW52ZXJzZTogKG9wOiBQbGFpdE9wZXJhdGlvbikgPT4gUGxhaXRPcGVyYXRpb247XG4gICAgaXNTZXRWaWV3cG9ydE9wZXJhdGlvbjogKHZhbHVlOiBhbnkpID0+IGJvb2xlYW47XG59XG5cbmV4cG9ydCBjb25zdCBpc1NldFZpZXdwb3J0T3BlcmF0aW9uID0gKHZhbHVlOiBhbnkpOiB2YWx1ZSBpcyBTZXRWaWV3cG9ydE9wZXJhdGlvbiA9PiB7XG4gICAgcmV0dXJuIHZhbHVlLnR5cGUgPT09ICdzZXRfdmlld3BvcnQnO1xufTtcblxuZXhwb3J0IGNvbnN0IGludmVyc2UgPSAob3A6IFBsYWl0T3BlcmF0aW9uKTogUGxhaXRPcGVyYXRpb24gPT4ge1xuICAgIHN3aXRjaCAob3AudHlwZSkge1xuICAgICAgICBjYXNlICdpbnNlcnRfbm9kZSc6IHtcbiAgICAgICAgICAgIHJldHVybiB7IC4uLm9wLCB0eXBlOiAncmVtb3ZlX25vZGUnIH07XG4gICAgICAgIH1cblxuICAgICAgICBjYXNlICdyZW1vdmVfbm9kZSc6IHtcbiAgICAgICAgICAgIHJldHVybiB7IC4uLm9wLCB0eXBlOiAnaW5zZXJ0X25vZGUnIH07XG4gICAgICAgIH1cblxuICAgICAgICBjYXNlICdtb3ZlX25vZGUnOiB7XG4gICAgICAgICAgICBjb25zdCB7IG5ld1BhdGgsIHBhdGggfSA9IG9wO1xuXG4gICAgICAgICAgICAvLyBQRVJGOiBpbiB0aGlzIGNhc2UgdGhlIG1vdmUgb3BlcmF0aW9uIGlzIGEgbm8tb3AgYW55d2F5cy5cbiAgICAgICAgICAgIGlmIChQYXRoLmVxdWFscyhuZXdQYXRoLCBwYXRoKSkge1xuICAgICAgICAgICAgICAgIHJldHVybiBvcDtcbiAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgaWYgKFBhdGguaXNTaWJsaW5nKHBhdGgsIG5ld1BhdGgpKSB7XG4gICAgICAgICAgICAgICAgY29uc3QgaW52ZXJzZVBhdGggPSBQYXRoLnRyYW5zZm9ybShwYXRoLCBvcCkhO1xuICAgICAgICAgICAgICAgIHJldHVybiB7IC4uLm9wLCBwYXRoOiBpbnZlcnNlUGF0aCwgbmV3UGF0aDogcGF0aCB9O1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICByZXR1cm4geyAuLi5vcCwgcGF0aDogbmV3UGF0aCwgbmV3UGF0aDogcGF0aCB9O1xuICAgICAgICB9XG5cbiAgICAgICAgY2FzZSAnc2V0X25vZGUnOiB7XG4gICAgICAgICAgICBjb25zdCB7IHByb3BlcnRpZXMsIG5ld1Byb3BlcnRpZXMgfSA9IG9wO1xuICAgICAgICAgICAgcmV0dXJuIHsgLi4ub3AsIHByb3BlcnRpZXM6IG5ld1Byb3BlcnRpZXMsIG5ld1Byb3BlcnRpZXM6IHByb3BlcnRpZXMgfTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNhc2UgJ3NldF9zZWxlY3Rpb24nOiB7XG4gICAgICAgICAgICBjb25zdCB7IHByb3BlcnRpZXMsIG5ld1Byb3BlcnRpZXMgfSA9IG9wO1xuXG4gICAgICAgICAgICBpZiAocHJvcGVydGllcyA9PSBudWxsKSB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICAgICAgLi4ub3AsXG4gICAgICAgICAgICAgICAgICAgIHByb3BlcnRpZXM6IG5ld1Byb3BlcnRpZXMsXG4gICAgICAgICAgICAgICAgICAgIG5ld1Byb3BlcnRpZXM6IG51bGxcbiAgICAgICAgICAgICAgICB9O1xuICAgICAgICAgICAgfSBlbHNlIGlmIChuZXdQcm9wZXJ0aWVzID09IG51bGwpIHtcbiAgICAgICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAgICAgICAuLi5vcCxcbiAgICAgICAgICAgICAgICAgICAgcHJvcGVydGllczogbnVsbCxcbiAgICAgICAgICAgICAgICAgICAgbmV3UHJvcGVydGllczogcHJvcGVydGllc1xuICAgICAgICAgICAgICAgIH07XG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICAgIHJldHVybiB7IC4uLm9wLCBwcm9wZXJ0aWVzOiBuZXdQcm9wZXJ0aWVzLCBuZXdQcm9wZXJ0aWVzOiBwcm9wZXJ0aWVzIH07XG4gICAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICBjYXNlICdzZXRfdmlld3BvcnQnOiB7XG4gICAgICAgICAgICBjb25zdCB7IHByb3BlcnRpZXMsIG5ld1Byb3BlcnRpZXMgfSA9IG9wO1xuICAgICAgICAgICAgaWYgKHByb3BlcnRpZXMgPT0gbnVsbCkge1xuICAgICAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgICAgICAgIC4uLm9wLFxuICAgICAgICAgICAgICAgICAgICBwcm9wZXJ0aWVzOiBuZXdQcm9wZXJ0aWVzLFxuICAgICAgICAgICAgICAgICAgICBuZXdQcm9wZXJ0aWVzOiBuZXdQcm9wZXJ0aWVzXG4gICAgICAgICAgICAgICAgfTtcbiAgICAgICAgICAgIH0gZWxzZSBpZiAobmV3UHJvcGVydGllcyA9PSBudWxsKSB7XG4gICAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICAgICAgLi4ub3AsXG4gICAgICAgICAgICAgICAgICAgIHByb3BlcnRpZXM6IHByb3BlcnRpZXMsXG4gICAgICAgICAgICAgICAgICAgIG5ld1Byb3BlcnRpZXM6IHByb3BlcnRpZXNcbiAgICAgICAgICAgICAgICB9O1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgICByZXR1cm4geyAuLi5vcCwgcHJvcGVydGllczogbmV3UHJvcGVydGllcywgbmV3UHJvcGVydGllczogcHJvcGVydGllcyB9O1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgfVxufTtcblxuZXhwb3J0IGNvbnN0IFBsYWl0T3BlcmF0aW9uOiBQbGFpdE9wZXJhdGlvbkludGVyZmFjZSA9IHtcbiAgICBpc1NldFZpZXdwb3J0T3BlcmF0aW9uLFxuICAgIGludmVyc2Vcbn07XG4iXX0=
84
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"operation.js","sourceRoot":"","sources":["../../../../packages/plait/src/interfaces/operation.ts"],"names":[],"mappings":"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","sourcesContent":["import { PlaitNode } from './node';\nimport { Path } from './path';\nimport { Selection } from './selection';\nimport { Viewport } from './viewport';\n\nexport type InsertNodeOperation = {\n    type: 'insert_node';\n    path: Path;\n    node: PlaitNode;\n};\n\nexport type RemoveNodeOperation = {\n    type: 'remove_node';\n    path: Path;\n    node: PlaitNode;\n};\n\nexport type MoveNodeOperation = {\n    type: 'move_node';\n    path: Path;\n    newPath: Path;\n};\n\nexport type SetViewportOperation = {\n    type: 'set_viewport';\n    properties: Partial<Viewport>;\n    newProperties: Partial<Viewport>;\n};\n\nexport type SetSelectionOperation = {\n    type: 'set_selection';\n    properties: Selection | null;\n    newProperties: Selection | null;\n};\n\nexport type SetNodeOperation = {\n    type: 'set_node';\n    path: Path;\n    properties: Partial<PlaitNode>;\n    newProperties: Partial<PlaitNode>;\n};\n\nexport type PlaitOperation =\n    | InsertNodeOperation\n    | SetViewportOperation\n    | SetSelectionOperation\n    | SetNodeOperation\n    | RemoveNodeOperation\n    | MoveNodeOperation;\n\nexport interface PlaitOperationInterface {\n    inverse: (op: PlaitOperation) => PlaitOperation;\n    isSetViewportOperation: (value: any) => boolean;\n}\n\nexport const isSetViewportOperation = (value: any): value is SetViewportOperation => {\n    return value.type === 'set_viewport';\n};\n\nexport const inverse = (op: PlaitOperation): PlaitOperation => {\n    switch (op.type) {\n        case 'insert_node': {\n            return { ...op, type: 'remove_node' };\n        }\n\n        case 'remove_node': {\n            return { ...op, type: 'insert_node' };\n        }\n\n        case 'move_node': {\n            const { newPath, path } = op;\n\n            // PERF: in this case the move operation is a no-op anyways.\n            if (Path.equals(newPath, path)) {\n                return op;\n            }\n\n            // when operation path is [0,0] -> [0,2], should exec Path.transform to get [0,1] -> [0,0]\n            // shoud not return [0,2] -> [0,0] #WIK-8981\n            // if (Path.isSibling(path, newPath)) {\n            //     return { ...op, path: newPath, newPath: path };\n            // }\n\n            // If the move does not happen within a single parent it is possible\n            // for the move to impact the true path to the location where the node\n            // was removed from and where it was inserted. We have to adjust for this\n            // and find the original path. We can accomplish this (only in non-sibling)\n            // moves by looking at the impact of the move operation on the node\n            // after the original move path.\n            const inversePath = Path.transform(path, op)!;\n            const inverseNewPath = Path.transform(Path.next(path), op)!;\n            return { ...op, path: inversePath, newPath: inverseNewPath };\n        }\n\n        case 'set_node': {\n            const { properties, newProperties } = op;\n            return { ...op, properties: newProperties, newProperties: properties };\n        }\n\n        case 'set_selection': {\n            const { properties, newProperties } = op;\n\n            if (properties == null) {\n                return {\n                    ...op,\n                    properties: newProperties,\n                    newProperties: null\n                };\n            } else if (newProperties == null) {\n                return {\n                    ...op,\n                    properties: null,\n                    newProperties: properties\n                };\n            } else {\n                return { ...op, properties: newProperties, newProperties: properties };\n            }\n        }\n\n        case 'set_viewport': {\n            const { properties, newProperties } = op;\n            if (properties == null) {\n                return {\n                    ...op,\n                    properties: newProperties,\n                    newProperties: newProperties\n                };\n            } else if (newProperties == null) {\n                return {\n                    ...op,\n                    properties: properties,\n                    newProperties: properties\n                };\n            } else {\n                return { ...op, properties: newProperties, newProperties: properties };\n            }\n        }\n    }\n};\n\nexport const PlaitOperation: PlaitOperationInterface = {\n    isSetViewportOperation,\n    inverse\n};\n"]}
@@ -132,8 +132,8 @@ export const Path = {
132
132
  break;
133
133
  }
134
134
  }
135
- return null;
135
+ return p;
136
136
  });
137
137
  }
138
138
  };
139
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"path.js","sourceRoot":"","sources":["../../../../packages/plait/src/interfaces/path.ts"],"names":[],"mappings":"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","sourcesContent":["import produce from 'immer';\nimport { PlaitOperation } from './operation';\n\nexport type Path = number[];\n\nexport const Path = {\n    parent(path: Path): Path {\n        if (path.length === 0) {\n            throw new Error(`Cannot get the parent path of the root path [${path}].`);\n        }\n\n        return path.slice(0, -1);\n    },\n    next(path: Path): Path {\n        if (path.length === 0) {\n            throw new Error(`Cannot get the next path of a root path [${path}], because it has no next index.`);\n        }\n        const last = path[path.length - 1];\n        return path.slice(0, -1).concat(last + 1);\n    },\n    /**\n     * Check if a path is an ancestor of another.\n     */\n\n    isAncestor(path: Path, another: Path): boolean {\n        return path.length < another.length && Path.compare(path, another) === 0;\n    },\n    /**\n     * Compare a path to another, returning an integer indicating whether the path\n     * was before, at, or after the other.\n     *\n     * Note: Two paths of unequal length can still receive a `0` result if one is\n     * directly above or below the other. If you want exact matching, use\n     * [[Path.equals]] instead.\n     */\n    compare(path: Path, another: Path): -1 | 0 | 1 {\n        const min = Math.min(path.length, another.length);\n\n        for (let i = 0; i < min; i++) {\n            if (path[i] < another[i]) return -1;\n            if (path[i] > another[i]) return 1;\n        }\n\n        return 0;\n    },\n\n    /**\n     * Check if a path is exactly equal to another.\n     */\n\n    equals(path: Path, another: Path): boolean {\n        return path.length === another.length && path.every((n, i) => n === another[i]);\n    },\n    /**\n     * Check if a path ends before one of the indexes in another.\n     */\n\n    endsBefore(path: Path, another: Path): boolean {\n        const i = path.length - 1;\n        const as = path.slice(0, i);\n        const bs = another.slice(0, i);\n        const av = path[i];\n        const bv = another[i];\n        return Path.equals(as, bs) && av < bv;\n    },\n    /**\n     * Check if a path is a sibling of another.\n     */\n    isSibling(path: Path, another: Path): boolean {\n        if (path.length !== another.length) {\n            return false;\n        }\n\n        const as = path.slice(0, -1);\n        const bs = another.slice(0, -1);\n        const al = path[path.length - 1];\n        const bl = another[another.length - 1];\n        return al !== bl && Path.equals(as, bs);\n    },\n    transform(path: Path | null, operation: PlaitOperation): Path | null {\n        return produce(path, p => {\n            // PERF: Exit early if the operation is guaranteed not to have an effect.\n            if (!path || path?.length === 0) {\n                return;\n            }\n\n            if (p === null) {\n                return null;\n            }\n\n            switch (operation.type) {\n                case 'insert_node': {\n                    const { path: op } = operation;\n\n                    if (Path.equals(op, p) || Path.endsBefore(op, p) || Path.isAncestor(op, p)) {\n                        p[op.length - 1] += 1;\n                    }\n\n                    break;\n                }\n\n                case 'remove_node': {\n                    const { path: op } = operation;\n\n                    if (Path.equals(op, p) || Path.isAncestor(op, p)) {\n                        return null;\n                    } else if (Path.endsBefore(op, p)) {\n                        p[op.length - 1] -= 1;\n                    }\n\n                    break;\n                }\n\n                case 'move_node': {\n                    const { path: op, newPath: onp } = operation;\n\n                    // If the old and new path are the same, it's a no-op.\n                    if (Path.equals(op, onp)) {\n                        return;\n                    }\n\n                    if (Path.isAncestor(op, p) || Path.equals(op, p)) {\n                        const copy = onp.slice();\n                        // op.length <= onp.length is different for slate\n                        // resolve drag from [0, 0] to [0, 3] issue\n                        if (Path.endsBefore(op, onp) && op.length <= onp.length) {\n                            copy[op.length - 1] -= 1;\n                        }\n\n                        return copy.concat(p.slice(op.length));\n                    } else if (Path.isSibling(op, onp) && (Path.isAncestor(onp, p) || Path.equals(onp, p))) {\n                        if (Path.endsBefore(op, p)) {\n                            p[op.length - 1] -= 1;\n                        } else {\n                            p[op.length - 1] += 1;\n                        }\n                    } else if (Path.endsBefore(onp, p) || Path.equals(onp, p) || Path.isAncestor(onp, p)) {\n                        if (Path.endsBefore(op, p)) {\n                            p[op.length - 1] -= 1;\n                        }\n\n                        p[onp.length - 1] += 1;\n                    } else if (Path.endsBefore(op, p)) {\n                        if (Path.equals(onp, p)) {\n                            p[onp.length - 1] += 1;\n                        }\n\n                        p[op.length - 1] -= 1;\n                    }\n\n                    break;\n                }\n            }\n            return null;\n        });\n    }\n};\n"]}
139
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"path.js","sourceRoot":"","sources":["../../../../packages/plait/src/interfaces/path.ts"],"names":[],"mappings":"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","sourcesContent":["import produce from 'immer';\nimport { PlaitOperation } from './operation';\n\nexport type Path = number[];\n\nexport const Path = {\n    parent(path: Path): Path {\n        if (path.length === 0) {\n            throw new Error(`Cannot get the parent path of the root path [${path}].`);\n        }\n\n        return path.slice(0, -1);\n    },\n    next(path: Path): Path {\n        if (path.length === 0) {\n            throw new Error(`Cannot get the next path of a root path [${path}], because it has no next index.`);\n        }\n        const last = path[path.length - 1];\n        return path.slice(0, -1).concat(last + 1);\n    },\n    /**\n     * Check if a path is an ancestor of another.\n     */\n\n    isAncestor(path: Path, another: Path): boolean {\n        return path.length < another.length && Path.compare(path, another) === 0;\n    },\n    /**\n     * Compare a path to another, returning an integer indicating whether the path\n     * was before, at, or after the other.\n     *\n     * Note: Two paths of unequal length can still receive a `0` result if one is\n     * directly above or below the other. If you want exact matching, use\n     * [[Path.equals]] instead.\n     */\n    compare(path: Path, another: Path): -1 | 0 | 1 {\n        const min = Math.min(path.length, another.length);\n\n        for (let i = 0; i < min; i++) {\n            if (path[i] < another[i]) return -1;\n            if (path[i] > another[i]) return 1;\n        }\n\n        return 0;\n    },\n\n    /**\n     * Check if a path is exactly equal to another.\n     */\n\n    equals(path: Path, another: Path): boolean {\n        return path.length === another.length && path.every((n, i) => n === another[i]);\n    },\n    /**\n     * Check if a path ends before one of the indexes in another.\n     */\n\n    endsBefore(path: Path, another: Path): boolean {\n        const i = path.length - 1;\n        const as = path.slice(0, i);\n        const bs = another.slice(0, i);\n        const av = path[i];\n        const bv = another[i];\n        return Path.equals(as, bs) && av < bv;\n    },\n    /**\n     * Check if a path is a sibling of another.\n     */\n    isSibling(path: Path, another: Path): boolean {\n        if (path.length !== another.length) {\n            return false;\n        }\n\n        const as = path.slice(0, -1);\n        const bs = another.slice(0, -1);\n        const al = path[path.length - 1];\n        const bl = another[another.length - 1];\n        return al !== bl && Path.equals(as, bs);\n    },\n    transform(path: Path | null, operation: PlaitOperation): Path | null {\n        return produce(path, p => {\n            // PERF: Exit early if the operation is guaranteed not to have an effect.\n            if (!path || path?.length === 0) {\n                return;\n            }\n\n            if (p === null) {\n                return null;\n            }\n\n            switch (operation.type) {\n                case 'insert_node': {\n                    const { path: op } = operation;\n\n                    if (Path.equals(op, p) || Path.endsBefore(op, p) || Path.isAncestor(op, p)) {\n                        p[op.length - 1] += 1;\n                    }\n\n                    break;\n                }\n\n                case 'remove_node': {\n                    const { path: op } = operation;\n\n                    if (Path.equals(op, p) || Path.isAncestor(op, p)) {\n                        return null;\n                    } else if (Path.endsBefore(op, p)) {\n                        p[op.length - 1] -= 1;\n                    }\n\n                    break;\n                }\n\n                case 'move_node': {\n                    const { path: op, newPath: onp } = operation;\n\n                    // If the old and new path are the same, it's a no-op.\n                    if (Path.equals(op, onp)) {\n                        return;\n                    }\n\n                    if (Path.isAncestor(op, p) || Path.equals(op, p)) {\n                        const copy = onp.slice();\n                        // op.length <= onp.length is different for slate\n                        // resolve drag from [0, 0] to [0, 3] issue\n                        if (Path.endsBefore(op, onp) && op.length <= onp.length) {\n                            copy[op.length - 1] -= 1;\n                        }\n\n                        return copy.concat(p.slice(op.length));\n                    } else if (Path.isSibling(op, onp) && (Path.isAncestor(onp, p) || Path.equals(onp, p))) {\n                        if (Path.endsBefore(op, p)) {\n                            p[op.length - 1] -= 1;\n                        } else {\n                            p[op.length - 1] += 1;\n                        }\n                    } else if (Path.endsBefore(onp, p) || Path.equals(onp, p) || Path.isAncestor(onp, p)) {\n                        if (Path.endsBefore(op, p)) {\n                            p[op.length - 1] -= 1;\n                        }\n\n                        p[onp.length - 1] += 1;\n                    } else if (Path.endsBefore(op, p)) {\n                        if (Path.equals(onp, p)) {\n                            p[onp.length - 1] += 1;\n                        }\n\n                        p[op.length - 1] -= 1;\n                    }\n\n                    break;\n                }\n            }\n            return p;\n        });\n    }\n};\n"]}
@@ -151,7 +151,7 @@ const Path = {
151
151
  break;
152
152
  }
153
153
  }
154
- return null;
154
+ return p;
155
155
  });
156
156
  }
157
157
  };
@@ -192,11 +192,20 @@ const inverse = (op) => {
192
192
  if (Path.equals(newPath, path)) {
193
193
  return op;
194
194
  }
195
- if (Path.isSibling(path, newPath)) {
196
- const inversePath = Path.transform(path, op);
197
- return Object.assign(Object.assign({}, op), { path: inversePath, newPath: path });
198
- }
199
- return Object.assign(Object.assign({}, op), { path: newPath, newPath: path });
195
+ // when operation path is [0,0] -> [0,2], should exec Path.transform to get [0,1] -> [0,0]
196
+ // shoud not return [0,2] -> [0,0] #WIK-8981
197
+ // if (Path.isSibling(path, newPath)) {
198
+ // return { ...op, path: newPath, newPath: path };
199
+ // }
200
+ // If the move does not happen within a single parent it is possible
201
+ // for the move to impact the true path to the location where the node
202
+ // was removed from and where it was inserted. We have to adjust for this
203
+ // and find the original path. We can accomplish this (only in non-sibling)
204
+ // moves by looking at the impact of the move operation on the node
205
+ // after the original move path.
206
+ const inversePath = Path.transform(path, op);
207
+ const inverseNewPath = Path.transform(Path.next(path), op);
208
+ return Object.assign(Object.assign({}, op), { path: inversePath, newPath: inverseNewPath });
200
209
  }
201
210
  case 'set_node': {
202
211
  const { properties, newProperties } = op;