@ckeditor/ckeditor5-undo 0.0.0-internal-20241017.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.
Files changed (287) hide show
  1. package/CHANGELOG.md +4 -0
  2. package/LICENSE.md +17 -0
  3. package/README.md +22 -0
  4. package/ckeditor5-metadata.json +23 -0
  5. package/dist/augmentation.d.ts +20 -0
  6. package/dist/basecommand.d.ts +76 -0
  7. package/dist/index-content.css +4 -0
  8. package/dist/index-editor.css +4 -0
  9. package/dist/index.css +4 -0
  10. package/dist/index.d.ts +17 -0
  11. package/dist/index.js +547 -0
  12. package/dist/index.js.map +1 -0
  13. package/dist/redocommand.d.ts +31 -0
  14. package/dist/translations/ar.d.ts +8 -0
  15. package/dist/translations/ar.js +5 -0
  16. package/dist/translations/ar.umd.js +11 -0
  17. package/dist/translations/ast.d.ts +8 -0
  18. package/dist/translations/ast.js +5 -0
  19. package/dist/translations/ast.umd.js +11 -0
  20. package/dist/translations/az.d.ts +8 -0
  21. package/dist/translations/az.js +5 -0
  22. package/dist/translations/az.umd.js +11 -0
  23. package/dist/translations/bg.d.ts +8 -0
  24. package/dist/translations/bg.js +5 -0
  25. package/dist/translations/bg.umd.js +11 -0
  26. package/dist/translations/bn.d.ts +8 -0
  27. package/dist/translations/bn.js +5 -0
  28. package/dist/translations/bn.umd.js +11 -0
  29. package/dist/translations/ca.d.ts +8 -0
  30. package/dist/translations/ca.js +5 -0
  31. package/dist/translations/ca.umd.js +11 -0
  32. package/dist/translations/cs.d.ts +8 -0
  33. package/dist/translations/cs.js +5 -0
  34. package/dist/translations/cs.umd.js +11 -0
  35. package/dist/translations/da.d.ts +8 -0
  36. package/dist/translations/da.js +5 -0
  37. package/dist/translations/da.umd.js +11 -0
  38. package/dist/translations/de-ch.d.ts +8 -0
  39. package/dist/translations/de-ch.js +5 -0
  40. package/dist/translations/de-ch.umd.js +11 -0
  41. package/dist/translations/de.d.ts +8 -0
  42. package/dist/translations/de.js +5 -0
  43. package/dist/translations/de.umd.js +11 -0
  44. package/dist/translations/el.d.ts +8 -0
  45. package/dist/translations/el.js +5 -0
  46. package/dist/translations/el.umd.js +11 -0
  47. package/dist/translations/en-au.d.ts +8 -0
  48. package/dist/translations/en-au.js +5 -0
  49. package/dist/translations/en-au.umd.js +11 -0
  50. package/dist/translations/en-gb.d.ts +8 -0
  51. package/dist/translations/en-gb.js +5 -0
  52. package/dist/translations/en-gb.umd.js +11 -0
  53. package/dist/translations/en.d.ts +8 -0
  54. package/dist/translations/en.js +5 -0
  55. package/dist/translations/en.umd.js +11 -0
  56. package/dist/translations/eo.d.ts +8 -0
  57. package/dist/translations/eo.js +5 -0
  58. package/dist/translations/eo.umd.js +11 -0
  59. package/dist/translations/es.d.ts +8 -0
  60. package/dist/translations/es.js +5 -0
  61. package/dist/translations/es.umd.js +11 -0
  62. package/dist/translations/et.d.ts +8 -0
  63. package/dist/translations/et.js +5 -0
  64. package/dist/translations/et.umd.js +11 -0
  65. package/dist/translations/eu.d.ts +8 -0
  66. package/dist/translations/eu.js +5 -0
  67. package/dist/translations/eu.umd.js +11 -0
  68. package/dist/translations/fa.d.ts +8 -0
  69. package/dist/translations/fa.js +5 -0
  70. package/dist/translations/fa.umd.js +11 -0
  71. package/dist/translations/fi.d.ts +8 -0
  72. package/dist/translations/fi.js +5 -0
  73. package/dist/translations/fi.umd.js +11 -0
  74. package/dist/translations/fr.d.ts +8 -0
  75. package/dist/translations/fr.js +5 -0
  76. package/dist/translations/fr.umd.js +11 -0
  77. package/dist/translations/gl.d.ts +8 -0
  78. package/dist/translations/gl.js +5 -0
  79. package/dist/translations/gl.umd.js +11 -0
  80. package/dist/translations/he.d.ts +8 -0
  81. package/dist/translations/he.js +5 -0
  82. package/dist/translations/he.umd.js +11 -0
  83. package/dist/translations/hi.d.ts +8 -0
  84. package/dist/translations/hi.js +5 -0
  85. package/dist/translations/hi.umd.js +11 -0
  86. package/dist/translations/hr.d.ts +8 -0
  87. package/dist/translations/hr.js +5 -0
  88. package/dist/translations/hr.umd.js +11 -0
  89. package/dist/translations/hu.d.ts +8 -0
  90. package/dist/translations/hu.js +5 -0
  91. package/dist/translations/hu.umd.js +11 -0
  92. package/dist/translations/id.d.ts +8 -0
  93. package/dist/translations/id.js +5 -0
  94. package/dist/translations/id.umd.js +11 -0
  95. package/dist/translations/it.d.ts +8 -0
  96. package/dist/translations/it.js +5 -0
  97. package/dist/translations/it.umd.js +11 -0
  98. package/dist/translations/ja.d.ts +8 -0
  99. package/dist/translations/ja.js +5 -0
  100. package/dist/translations/ja.umd.js +11 -0
  101. package/dist/translations/km.d.ts +8 -0
  102. package/dist/translations/km.js +5 -0
  103. package/dist/translations/km.umd.js +11 -0
  104. package/dist/translations/kn.d.ts +8 -0
  105. package/dist/translations/kn.js +5 -0
  106. package/dist/translations/kn.umd.js +11 -0
  107. package/dist/translations/ko.d.ts +8 -0
  108. package/dist/translations/ko.js +5 -0
  109. package/dist/translations/ko.umd.js +11 -0
  110. package/dist/translations/ku.d.ts +8 -0
  111. package/dist/translations/ku.js +5 -0
  112. package/dist/translations/ku.umd.js +11 -0
  113. package/dist/translations/lt.d.ts +8 -0
  114. package/dist/translations/lt.js +5 -0
  115. package/dist/translations/lt.umd.js +11 -0
  116. package/dist/translations/lv.d.ts +8 -0
  117. package/dist/translations/lv.js +5 -0
  118. package/dist/translations/lv.umd.js +11 -0
  119. package/dist/translations/ms.d.ts +8 -0
  120. package/dist/translations/ms.js +5 -0
  121. package/dist/translations/ms.umd.js +11 -0
  122. package/dist/translations/nb.d.ts +8 -0
  123. package/dist/translations/nb.js +5 -0
  124. package/dist/translations/nb.umd.js +11 -0
  125. package/dist/translations/ne.d.ts +8 -0
  126. package/dist/translations/ne.js +5 -0
  127. package/dist/translations/ne.umd.js +11 -0
  128. package/dist/translations/nl.d.ts +8 -0
  129. package/dist/translations/nl.js +5 -0
  130. package/dist/translations/nl.umd.js +11 -0
  131. package/dist/translations/no.d.ts +8 -0
  132. package/dist/translations/no.js +5 -0
  133. package/dist/translations/no.umd.js +11 -0
  134. package/dist/translations/pl.d.ts +8 -0
  135. package/dist/translations/pl.js +5 -0
  136. package/dist/translations/pl.umd.js +11 -0
  137. package/dist/translations/pt-br.d.ts +8 -0
  138. package/dist/translations/pt-br.js +5 -0
  139. package/dist/translations/pt-br.umd.js +11 -0
  140. package/dist/translations/pt.d.ts +8 -0
  141. package/dist/translations/pt.js +5 -0
  142. package/dist/translations/pt.umd.js +11 -0
  143. package/dist/translations/ro.d.ts +8 -0
  144. package/dist/translations/ro.js +5 -0
  145. package/dist/translations/ro.umd.js +11 -0
  146. package/dist/translations/ru.d.ts +8 -0
  147. package/dist/translations/ru.js +5 -0
  148. package/dist/translations/ru.umd.js +11 -0
  149. package/dist/translations/si.d.ts +8 -0
  150. package/dist/translations/si.js +5 -0
  151. package/dist/translations/si.umd.js +11 -0
  152. package/dist/translations/sk.d.ts +8 -0
  153. package/dist/translations/sk.js +5 -0
  154. package/dist/translations/sk.umd.js +11 -0
  155. package/dist/translations/sq.d.ts +8 -0
  156. package/dist/translations/sq.js +5 -0
  157. package/dist/translations/sq.umd.js +11 -0
  158. package/dist/translations/sr-latn.d.ts +8 -0
  159. package/dist/translations/sr-latn.js +5 -0
  160. package/dist/translations/sr-latn.umd.js +11 -0
  161. package/dist/translations/sr.d.ts +8 -0
  162. package/dist/translations/sr.js +5 -0
  163. package/dist/translations/sr.umd.js +11 -0
  164. package/dist/translations/sv.d.ts +8 -0
  165. package/dist/translations/sv.js +5 -0
  166. package/dist/translations/sv.umd.js +11 -0
  167. package/dist/translations/th.d.ts +8 -0
  168. package/dist/translations/th.js +5 -0
  169. package/dist/translations/th.umd.js +11 -0
  170. package/dist/translations/ti.d.ts +8 -0
  171. package/dist/translations/ti.js +5 -0
  172. package/dist/translations/ti.umd.js +11 -0
  173. package/dist/translations/tk.d.ts +8 -0
  174. package/dist/translations/tk.js +5 -0
  175. package/dist/translations/tk.umd.js +11 -0
  176. package/dist/translations/tr.d.ts +8 -0
  177. package/dist/translations/tr.js +5 -0
  178. package/dist/translations/tr.umd.js +11 -0
  179. package/dist/translations/tt.d.ts +8 -0
  180. package/dist/translations/tt.js +5 -0
  181. package/dist/translations/tt.umd.js +11 -0
  182. package/dist/translations/ug.d.ts +8 -0
  183. package/dist/translations/ug.js +5 -0
  184. package/dist/translations/ug.umd.js +11 -0
  185. package/dist/translations/uk.d.ts +8 -0
  186. package/dist/translations/uk.js +5 -0
  187. package/dist/translations/uk.umd.js +11 -0
  188. package/dist/translations/ur.d.ts +8 -0
  189. package/dist/translations/ur.js +5 -0
  190. package/dist/translations/ur.umd.js +11 -0
  191. package/dist/translations/uz.d.ts +8 -0
  192. package/dist/translations/uz.js +5 -0
  193. package/dist/translations/uz.umd.js +11 -0
  194. package/dist/translations/vi.d.ts +8 -0
  195. package/dist/translations/vi.js +5 -0
  196. package/dist/translations/vi.umd.js +11 -0
  197. package/dist/translations/zh-cn.d.ts +8 -0
  198. package/dist/translations/zh-cn.js +5 -0
  199. package/dist/translations/zh-cn.umd.js +11 -0
  200. package/dist/translations/zh.d.ts +8 -0
  201. package/dist/translations/zh.js +5 -0
  202. package/dist/translations/zh.umd.js +11 -0
  203. package/dist/undo.d.ts +125 -0
  204. package/dist/undocommand.d.ts +41 -0
  205. package/dist/undoediting.d.ts +45 -0
  206. package/dist/undoui.d.ts +42 -0
  207. package/lang/contexts.json +4 -0
  208. package/lang/translations/ar.po +26 -0
  209. package/lang/translations/ast.po +26 -0
  210. package/lang/translations/az.po +26 -0
  211. package/lang/translations/bg.po +26 -0
  212. package/lang/translations/bn.po +26 -0
  213. package/lang/translations/ca.po +26 -0
  214. package/lang/translations/cs.po +26 -0
  215. package/lang/translations/da.po +26 -0
  216. package/lang/translations/de-ch.po +26 -0
  217. package/lang/translations/de.po +26 -0
  218. package/lang/translations/el.po +26 -0
  219. package/lang/translations/en-au.po +26 -0
  220. package/lang/translations/en-gb.po +26 -0
  221. package/lang/translations/en.po +26 -0
  222. package/lang/translations/eo.po +26 -0
  223. package/lang/translations/es.po +26 -0
  224. package/lang/translations/et.po +26 -0
  225. package/lang/translations/eu.po +26 -0
  226. package/lang/translations/fa.po +26 -0
  227. package/lang/translations/fi.po +26 -0
  228. package/lang/translations/fr.po +26 -0
  229. package/lang/translations/gl.po +26 -0
  230. package/lang/translations/he.po +26 -0
  231. package/lang/translations/hi.po +26 -0
  232. package/lang/translations/hr.po +26 -0
  233. package/lang/translations/hu.po +26 -0
  234. package/lang/translations/id.po +26 -0
  235. package/lang/translations/it.po +26 -0
  236. package/lang/translations/ja.po +26 -0
  237. package/lang/translations/km.po +26 -0
  238. package/lang/translations/kn.po +26 -0
  239. package/lang/translations/ko.po +26 -0
  240. package/lang/translations/ku.po +26 -0
  241. package/lang/translations/lt.po +26 -0
  242. package/lang/translations/lv.po +26 -0
  243. package/lang/translations/ms.po +26 -0
  244. package/lang/translations/nb.po +26 -0
  245. package/lang/translations/ne.po +26 -0
  246. package/lang/translations/nl.po +26 -0
  247. package/lang/translations/no.po +26 -0
  248. package/lang/translations/pl.po +26 -0
  249. package/lang/translations/pt-br.po +26 -0
  250. package/lang/translations/pt.po +26 -0
  251. package/lang/translations/ro.po +26 -0
  252. package/lang/translations/ru.po +26 -0
  253. package/lang/translations/si.po +26 -0
  254. package/lang/translations/sk.po +26 -0
  255. package/lang/translations/sq.po +26 -0
  256. package/lang/translations/sr-latn.po +26 -0
  257. package/lang/translations/sr.po +26 -0
  258. package/lang/translations/sv.po +26 -0
  259. package/lang/translations/th.po +26 -0
  260. package/lang/translations/ti.po +26 -0
  261. package/lang/translations/tk.po +26 -0
  262. package/lang/translations/tr.po +26 -0
  263. package/lang/translations/tt.po +26 -0
  264. package/lang/translations/ug.po +26 -0
  265. package/lang/translations/uk.po +26 -0
  266. package/lang/translations/ur.po +26 -0
  267. package/lang/translations/uz.po +26 -0
  268. package/lang/translations/vi.po +26 -0
  269. package/lang/translations/zh-cn.po +26 -0
  270. package/lang/translations/zh.po +26 -0
  271. package/package.json +39 -0
  272. package/src/augmentation.d.ts +16 -0
  273. package/src/augmentation.js +5 -0
  274. package/src/basecommand.d.ts +72 -0
  275. package/src/basecommand.js +192 -0
  276. package/src/index.d.ts +13 -0
  277. package/src/index.js +11 -0
  278. package/src/redocommand.d.ts +27 -0
  279. package/src/redocommand.js +40 -0
  280. package/src/undo.d.ts +121 -0
  281. package/src/undo.js +127 -0
  282. package/src/undocommand.d.ts +37 -0
  283. package/src/undocommand.js +44 -0
  284. package/src/undoediting.d.ts +41 -0
  285. package/src/undoediting.js +102 -0
  286. package/src/undoui.d.ts +38 -0
  287. package/src/undoui.js +79 -0
package/dist/index.js ADDED
@@ -0,0 +1,547 @@
1
+ /**
2
+ * @license Copyright (c) 2003-2024, CKSource Holding sp. z o.o. All rights reserved.
3
+ * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
4
+ */
5
+ import { Command, Plugin, icons } from '@ckeditor/ckeditor5-core/dist/index.js';
6
+ import { transformSets, NoOperation } from '@ckeditor/ckeditor5-engine/dist/index.js';
7
+ import { ButtonView, MenuBarMenuListItemButtonView } from '@ckeditor/ckeditor5-ui/dist/index.js';
8
+
9
+ /**
10
+ * Base class for the undo feature commands: {@link module:undo/undocommand~UndoCommand} and {@link module:undo/redocommand~RedoCommand}.
11
+ */ class BaseCommand extends Command {
12
+ /**
13
+ * Stack of items stored by the command. These are pairs of:
14
+ *
15
+ * * {@link module:engine/model/batch~Batch batch} saved by the command,
16
+ * * {@link module:engine/model/selection~Selection selection} state at the moment of saving the batch.
17
+ */ _stack = [];
18
+ /**
19
+ * Stores all batches that were created by this command.
20
+ *
21
+ * @internal
22
+ */ _createdBatches = new WeakSet();
23
+ /**
24
+ * @inheritDoc
25
+ */ constructor(editor){
26
+ super(editor);
27
+ // Refresh state, so the command is inactive right after initialization.
28
+ this.refresh();
29
+ // This command should not depend on selection change.
30
+ this._isEnabledBasedOnSelection = false;
31
+ // Set the transparent batch for the `editor.data.set()` call if the
32
+ // batch type is not set already.
33
+ this.listenTo(editor.data, 'set', (evt, data)=>{
34
+ // Create a shallow copy of the options to not change the original args.
35
+ // And make sure that an object is assigned to data[ 1 ].
36
+ data[1] = {
37
+ ...data[1]
38
+ };
39
+ const options = data[1];
40
+ // If batch type is not set, default to non-undoable batch.
41
+ if (!options.batchType) {
42
+ options.batchType = {
43
+ isUndoable: false
44
+ };
45
+ }
46
+ }, {
47
+ priority: 'high'
48
+ });
49
+ // Clear the stack for the `transparent` batches.
50
+ this.listenTo(editor.data, 'set', (evt, data)=>{
51
+ // We can assume that the object exists and it has a `batchType` property.
52
+ // It was ensured with a higher priority listener before.
53
+ const options = data[1];
54
+ if (!options.batchType.isUndoable) {
55
+ this.clearStack();
56
+ }
57
+ });
58
+ }
59
+ /**
60
+ * @inheritDoc
61
+ */ refresh() {
62
+ this.isEnabled = this._stack.length > 0;
63
+ }
64
+ /**
65
+ * Returns all batches created by this command.
66
+ */ get createdBatches() {
67
+ return this._createdBatches;
68
+ }
69
+ /**
70
+ * Stores a batch in the command, together with the selection state of the {@link module:engine/model/document~Document document}
71
+ * created by the editor which this command is registered to.
72
+ *
73
+ * @param batch The batch to add.
74
+ */ addBatch(batch) {
75
+ const docSelection = this.editor.model.document.selection;
76
+ const selection = {
77
+ ranges: docSelection.hasOwnRange ? Array.from(docSelection.getRanges()) : [],
78
+ isBackward: docSelection.isBackward
79
+ };
80
+ this._stack.push({
81
+ batch,
82
+ selection
83
+ });
84
+ this.refresh();
85
+ }
86
+ /**
87
+ * Removes all items from the stack.
88
+ */ clearStack() {
89
+ this._stack = [];
90
+ this.refresh();
91
+ }
92
+ /**
93
+ * Restores the {@link module:engine/model/document~Document#selection document selection} state after a batch was undone.
94
+ *
95
+ * @param ranges Ranges to be restored.
96
+ * @param isBackward A flag describing whether the restored range was selected forward or backward.
97
+ * @param operations Operations which has been applied since selection has been stored.
98
+ */ _restoreSelection(ranges, isBackward, operations) {
99
+ const model = this.editor.model;
100
+ const document = model.document;
101
+ // This will keep the transformed selection ranges.
102
+ const selectionRanges = [];
103
+ // Transform all ranges from the restored selection.
104
+ const transformedRangeGroups = ranges.map((range)=>range.getTransformedByOperations(operations));
105
+ const allRanges = transformedRangeGroups.flat();
106
+ for (const rangeGroup of transformedRangeGroups){
107
+ // While transforming there could appear ranges that are contained by other ranges, we shall ignore them.
108
+ const transformed = rangeGroup.filter((range)=>range.root != document.graveyard).filter((range)=>!isRangeContainedByAnyOtherRange(range, allRanges));
109
+ // All the transformed ranges ended up in graveyard.
110
+ if (!transformed.length) {
111
+ continue;
112
+ }
113
+ // After the range got transformed, we have an array of ranges. Some of those
114
+ // ranges may be "touching" -- they can be next to each other and could be merged.
115
+ normalizeRanges(transformed);
116
+ // For each `range` from `ranges`, we take only one transformed range.
117
+ // This is because we want to prevent situation where single-range selection
118
+ // got transformed to multi-range selection.
119
+ selectionRanges.push(transformed[0]);
120
+ }
121
+ // @if CK_DEBUG_ENGINE // console.log( `Restored selection by undo: ${ selectionRanges.join( ', ' ) }` );
122
+ // `selectionRanges` may be empty if all ranges ended up in graveyard. If that is the case, do not restore selection.
123
+ if (selectionRanges.length) {
124
+ model.change((writer)=>{
125
+ writer.setSelection(selectionRanges, {
126
+ backward: isBackward
127
+ });
128
+ });
129
+ }
130
+ }
131
+ /**
132
+ * Undoes a batch by reversing that batch, transforming reversed batch and finally applying it.
133
+ * This is a helper method for {@link #execute}.
134
+ *
135
+ * @param batchToUndo The batch to be undone.
136
+ * @param undoingBatch The batch that will contain undoing changes.
137
+ */ _undo(batchToUndo, undoingBatch) {
138
+ const model = this.editor.model;
139
+ const document = model.document;
140
+ // All changes done by the command execution will be saved as one batch.
141
+ this._createdBatches.add(undoingBatch);
142
+ const operationsToUndo = batchToUndo.operations.slice().filter((operation)=>operation.isDocumentOperation);
143
+ operationsToUndo.reverse();
144
+ // We will process each operation from `batchToUndo`, in reverse order. If there were operations A, B and C in undone batch,
145
+ // we need to revert them in reverse order, so first C' (reversed C), then B', then A'.
146
+ for (const operationToUndo of operationsToUndo){
147
+ const nextBaseVersion = operationToUndo.baseVersion + 1;
148
+ const historyOperations = Array.from(document.history.getOperations(nextBaseVersion));
149
+ const transformedSets = transformSets([
150
+ operationToUndo.getReversed()
151
+ ], historyOperations, {
152
+ useRelations: true,
153
+ document: this.editor.model.document,
154
+ padWithNoOps: false,
155
+ forceWeakRemove: true
156
+ });
157
+ const reversedOperations = transformedSets.operationsA;
158
+ // After reversed operation has been transformed by all history operations, apply it.
159
+ for (let operation of reversedOperations){
160
+ // Do not apply any operation on non-editable space.
161
+ const affectedSelectable = operation.affectedSelectable;
162
+ if (affectedSelectable && !model.canEditAt(affectedSelectable)) {
163
+ operation = new NoOperation(operation.baseVersion);
164
+ }
165
+ // Before applying, add the operation to the `undoingBatch`.
166
+ undoingBatch.addOperation(operation);
167
+ model.applyOperation(operation);
168
+ document.history.setOperationAsUndone(operationToUndo, operation);
169
+ }
170
+ }
171
+ }
172
+ }
173
+ /**
174
+ * Normalizes list of ranges by joining intersecting or "touching" ranges.
175
+ *
176
+ * @param ranges Ranges to be normalized.
177
+ */ function normalizeRanges(ranges) {
178
+ ranges.sort((a, b)=>a.start.isBefore(b.start) ? -1 : 1);
179
+ for(let i = 1; i < ranges.length; i++){
180
+ const previousRange = ranges[i - 1];
181
+ const joinedRange = previousRange.getJoined(ranges[i], true);
182
+ if (joinedRange) {
183
+ // Replace the ranges on the list with the new joined range.
184
+ i--;
185
+ ranges.splice(i, 2, joinedRange);
186
+ }
187
+ }
188
+ }
189
+ function isRangeContainedByAnyOtherRange(range, ranges) {
190
+ return ranges.some((otherRange)=>otherRange !== range && otherRange.containsRange(range, true));
191
+ }
192
+
193
+ /**
194
+ * The undo command stores {@link module:engine/model/batch~Batch batches} applied to the
195
+ * {@link module:engine/model/document~Document document} and is able to undo a batch by reversing it and transforming by
196
+ * batches from {@link module:engine/model/document~Document#history history} that happened after the reversed batch.
197
+ *
198
+ * The undo command also takes care of restoring the {@link module:engine/model/document~Document#selection document selection}.
199
+ */ class UndoCommand extends BaseCommand {
200
+ /**
201
+ * Executes the command. This method reverts a {@link module:engine/model/batch~Batch batch} added to the command's stack, transforms
202
+ * and applies the reverted version on the {@link module:engine/model/document~Document document} and removes the batch from the stack.
203
+ * Then, it restores the {@link module:engine/model/document~Document#selection document selection}.
204
+ *
205
+ * @fires execute
206
+ * @fires revert
207
+ * @param batch A batch that should be undone. If not set, the last added batch will be undone.
208
+ */ execute(batch = null) {
209
+ // If batch is not given, set `batchIndex` to the last index in command stack.
210
+ const batchIndex = batch ? this._stack.findIndex((a)=>a.batch == batch) : this._stack.length - 1;
211
+ const item = this._stack.splice(batchIndex, 1)[0];
212
+ const undoingBatch = this.editor.model.createBatch({
213
+ isUndo: true
214
+ });
215
+ // All changes have to be done in one `enqueueChange` callback so other listeners will not
216
+ // step between consecutive operations, or won't do changes to the document before selection is properly restored.
217
+ this.editor.model.enqueueChange(undoingBatch, ()=>{
218
+ this._undo(item.batch, undoingBatch);
219
+ const operations = this.editor.model.document.history.getOperations(item.batch.baseVersion);
220
+ this._restoreSelection(item.selection.ranges, item.selection.isBackward, operations);
221
+ });
222
+ // Firing `revert` event after the change block to make sure that it includes all changes from post-fixers
223
+ // and make sure that the selection is "stabilized" (the selection range is saved after undo is executed and then
224
+ // restored on redo, so it is important that the selection range is saved after post-fixers are done).
225
+ this.fire('revert', item.batch, undoingBatch);
226
+ this.refresh();
227
+ }
228
+ }
229
+
230
+ /**
231
+ * The redo command stores {@link module:engine/model/batch~Batch batches} that were used to undo a batch by
232
+ * {@link module:undo/undocommand~UndoCommand}. It is able to redo a previously undone batch by reversing the undoing
233
+ * batches created by `UndoCommand`. The reversed batch is transformed by all the batches from
234
+ * {@link module:engine/model/document~Document#history history} that happened after the reversed undo batch.
235
+ *
236
+ * The redo command also takes care of restoring the {@link module:engine/model/document~Document#selection document selection}.
237
+ */ class RedoCommand extends BaseCommand {
238
+ /**
239
+ * Executes the command. This method reverts the last {@link module:engine/model/batch~Batch batch} added to
240
+ * the command's stack, applies the reverted and transformed version on the
241
+ * {@link module:engine/model/document~Document document} and removes the batch from the stack.
242
+ * Then, it restores the {@link module:engine/model/document~Document#selection document selection}.
243
+ *
244
+ * @fires execute
245
+ */ execute() {
246
+ const item = this._stack.pop();
247
+ const redoingBatch = this.editor.model.createBatch({
248
+ isUndo: true
249
+ });
250
+ // All changes have to be done in one `enqueueChange` callback so other listeners will not step between consecutive
251
+ // operations, or won't do changes to the document before selection is properly restored.
252
+ this.editor.model.enqueueChange(redoingBatch, ()=>{
253
+ const lastOperation = item.batch.operations[item.batch.operations.length - 1];
254
+ const nextBaseVersion = lastOperation.baseVersion + 1;
255
+ const operations = this.editor.model.document.history.getOperations(nextBaseVersion);
256
+ this._restoreSelection(item.selection.ranges, item.selection.isBackward, operations);
257
+ this._undo(item.batch, redoingBatch);
258
+ });
259
+ this.refresh();
260
+ }
261
+ }
262
+
263
+ /**
264
+ * The undo engine feature.
265
+ *
266
+ * It introduces the `'undo'` and `'redo'` commands to the editor.
267
+ */ class UndoEditing extends Plugin {
268
+ /**
269
+ * The command that manages the undo {@link module:engine/model/batch~Batch batches} stack (history).
270
+ * Created and registered during the {@link #init feature initialization}.
271
+ */ _undoCommand;
272
+ /**
273
+ * The command that manages the redo {@link module:engine/model/batch~Batch batches} stack (history).
274
+ * Created and registered during the {@link #init feature initialization}.
275
+ */ _redoCommand;
276
+ /**
277
+ * Keeps track of which batches were registered in undo.
278
+ */ _batchRegistry = new WeakSet();
279
+ /**
280
+ * @inheritDoc
281
+ */ static get pluginName() {
282
+ return 'UndoEditing';
283
+ }
284
+ /**
285
+ * @inheritDoc
286
+ */ static get isOfficialPlugin() {
287
+ return true;
288
+ }
289
+ /**
290
+ * @inheritDoc
291
+ */ init() {
292
+ const editor = this.editor;
293
+ const t = editor.t;
294
+ // Create commands.
295
+ this._undoCommand = new UndoCommand(editor);
296
+ this._redoCommand = new RedoCommand(editor);
297
+ // Register command to the editor.
298
+ editor.commands.add('undo', this._undoCommand);
299
+ editor.commands.add('redo', this._redoCommand);
300
+ this.listenTo(editor.model, 'applyOperation', (evt, args)=>{
301
+ const operation = args[0];
302
+ // Do not register batch if the operation is not a document operation.
303
+ // This prevents from creating empty undo steps, where all operations where non-document operations.
304
+ // Non-document operations creates and alters content in detached tree fragments (for example, document fragments).
305
+ // Most of time this is preparing data before it is inserted into actual tree (for example during copy & paste).
306
+ // Such operations should not be reversed.
307
+ if (!operation.isDocumentOperation) {
308
+ return;
309
+ }
310
+ const batch = operation.batch;
311
+ const isRedoBatch = this._redoCommand.createdBatches.has(batch);
312
+ const isUndoBatch = this._undoCommand.createdBatches.has(batch);
313
+ const wasProcessed = this._batchRegistry.has(batch);
314
+ // Skip the batch if it was already processed.
315
+ if (wasProcessed) {
316
+ return;
317
+ }
318
+ // Add the batch to the registry so it will not be processed again.
319
+ this._batchRegistry.add(batch);
320
+ if (!batch.isUndoable) {
321
+ return;
322
+ }
323
+ if (isRedoBatch) {
324
+ // If this batch comes from `redoCommand`, add it to the `undoCommand` stack.
325
+ this._undoCommand.addBatch(batch);
326
+ } else if (!isUndoBatch) {
327
+ // If the batch comes neither from `redoCommand` nor from `undoCommand` then it is a new, regular batch.
328
+ // Add the batch to the `undoCommand` stack and clear the `redoCommand` stack.
329
+ this._undoCommand.addBatch(batch);
330
+ this._redoCommand.clearStack();
331
+ }
332
+ }, {
333
+ priority: 'highest'
334
+ });
335
+ this.listenTo(this._undoCommand, 'revert', (evt, undoneBatch, undoingBatch)=>{
336
+ this._redoCommand.addBatch(undoingBatch);
337
+ });
338
+ editor.keystrokes.set('CTRL+Z', 'undo');
339
+ editor.keystrokes.set('CTRL+Y', 'redo');
340
+ editor.keystrokes.set('CTRL+SHIFT+Z', 'redo');
341
+ // Add the information about the keystrokes to the accessibility database.
342
+ editor.accessibility.addKeystrokeInfos({
343
+ keystrokes: [
344
+ {
345
+ label: t('Undo'),
346
+ keystroke: 'CTRL+Z'
347
+ },
348
+ {
349
+ label: t('Redo'),
350
+ keystroke: [
351
+ [
352
+ 'CTRL+Y'
353
+ ],
354
+ [
355
+ 'CTRL+SHIFT+Z'
356
+ ]
357
+ ]
358
+ }
359
+ ]
360
+ });
361
+ }
362
+ }
363
+
364
+ /**
365
+ * The undo UI feature. It introduces the `'undo'` and `'redo'` buttons to the editor.
366
+ */ class UndoUI extends Plugin {
367
+ /**
368
+ * @inheritDoc
369
+ */ static get pluginName() {
370
+ return 'UndoUI';
371
+ }
372
+ /**
373
+ * @inheritDoc
374
+ */ static get isOfficialPlugin() {
375
+ return true;
376
+ }
377
+ /**
378
+ * @inheritDoc
379
+ */ init() {
380
+ const editor = this.editor;
381
+ const locale = editor.locale;
382
+ const t = editor.t;
383
+ const localizedUndoIcon = locale.uiLanguageDirection == 'ltr' ? icons.undo : icons.redo;
384
+ const localizedRedoIcon = locale.uiLanguageDirection == 'ltr' ? icons.redo : icons.undo;
385
+ this._addButtonsToFactory('undo', t('Undo'), 'CTRL+Z', localizedUndoIcon);
386
+ this._addButtonsToFactory('redo', t('Redo'), 'CTRL+Y', localizedRedoIcon);
387
+ }
388
+ /**
389
+ * Creates a button for the specified command.
390
+ *
391
+ * @param name Command name.
392
+ * @param label Button label.
393
+ * @param keystroke Command keystroke.
394
+ * @param Icon Source of the icon.
395
+ */ _addButtonsToFactory(name, label, keystroke, Icon) {
396
+ const editor = this.editor;
397
+ editor.ui.componentFactory.add(name, ()=>{
398
+ const buttonView = this._createButton(ButtonView, name, label, keystroke, Icon);
399
+ buttonView.set({
400
+ tooltip: true
401
+ });
402
+ return buttonView;
403
+ });
404
+ editor.ui.componentFactory.add('menuBar:' + name, ()=>{
405
+ return this._createButton(MenuBarMenuListItemButtonView, name, label, keystroke, Icon);
406
+ });
407
+ }
408
+ /**
409
+ * TODO
410
+ */ _createButton(ButtonClass, name, label, keystroke, Icon) {
411
+ const editor = this.editor;
412
+ const locale = editor.locale;
413
+ const command = editor.commands.get(name);
414
+ const view = new ButtonClass(locale);
415
+ view.set({
416
+ label,
417
+ icon: Icon,
418
+ keystroke
419
+ });
420
+ view.bind('isEnabled').to(command, 'isEnabled');
421
+ this.listenTo(view, 'execute', ()=>{
422
+ editor.execute(name);
423
+ editor.editing.view.focus();
424
+ });
425
+ return view;
426
+ }
427
+ }
428
+
429
+ /**
430
+ * The undo feature.
431
+ *
432
+ * This is a "glue" plugin which loads the {@link module:undo/undoediting~UndoEditing undo editing feature}
433
+ * and the {@link module:undo/undoui~UndoUI undo UI feature}.
434
+ *
435
+ * Below is an explanation of the undo mechanism working together with {@link module:engine/model/history~History History}:
436
+ *
437
+ * Whenever an {@link module:engine/model/operation/operation~Operation operation} is applied to the
438
+ * {@link module:engine/model/document~Document document}, it is saved to `History` as is.
439
+ * The {@link module:engine/model/batch~Batch batch} that owns that operation is also saved, in
440
+ * {@link module:undo/undocommand~UndoCommand}, together with the selection that was present in the document before the
441
+ * operation was applied. A batch is saved instead of the operation because changes are undone batch-by-batch, not operation-by-operation
442
+ * and a batch is seen as one undo step.
443
+ *
444
+ * After changes happen to the document, the `History` and `UndoCommand` stack can be represented as follows:
445
+ *
446
+ * ```
447
+ * History Undo stack
448
+ * ============== ==================================
449
+ * [operation A1] [ batch A ]
450
+ * [operation B1] [ batch B ]
451
+ * [operation B2] [ batch C ]
452
+ * [operation C1]
453
+ * [operation C2]
454
+ * [operation B3]
455
+ * [operation C3]
456
+ * ```
457
+ *
458
+ * Where operations starting with the same letter are from same batch.
459
+ *
460
+ * Undoing a batch means that a set of operations which will reverse the effects of that batch needs to be generated.
461
+ * For example, if a batch added several letters, undoing the batch should remove them. It is important to apply undoing
462
+ * operations in the reversed order, so if a batch has operation `X`, `Y`, `Z`, reversed operations `Zr`, `Yr` and `Xr`
463
+ * need to be applied. Otherwise reversed operation `Xr` would operate on a wrong document state, because operation `X`
464
+ * does not know that operations `Y` and `Z` happened.
465
+ *
466
+ * After operations from an undone batch got {@link module:engine/model/operation/operation~Operation#getReversed reversed},
467
+ * one needs to make sure if they are ready to be applied. In the scenario above, operation `C3` is the last operation and `C3r`
468
+ * bases on up-to-date document state, so it can be applied to the document.
469
+ *
470
+ * ```
471
+ * History Undo stack
472
+ * ================= ==================================
473
+ * [ operation A1 ] [ batch A ]
474
+ * [ operation B1 ] [ batch B ]
475
+ * [ operation B2 ] [ processing undoing batch C ]
476
+ * [ operation C1 ]
477
+ * [ operation C2 ]
478
+ * [ operation B3 ]
479
+ * [ operation C3 ]
480
+ * [ operation C3r ]
481
+ * ```
482
+ *
483
+ * Next is operation `C2`, reversed to `C2r`. `C2r` bases on `C2`, so it bases on the wrong document state. It needs to be
484
+ * transformed by operations from history that happened after it, so it "knows" about them. Let us assume that `C2' = C2r * B3 * C3 * C3r`,
485
+ * where `*` means "transformed by". Rest of operations from that batch are processed in the same fashion.
486
+ *
487
+ * ```
488
+ * History Undo stack Redo stack
489
+ * ================= ================================== ==================================
490
+ * [ operation A1 ] [ batch A ] [ batch Cr ]
491
+ * [ operation B1 ] [ batch B ]
492
+ * [ operation B2 ]
493
+ * [ operation C1 ]
494
+ * [ operation C2 ]
495
+ * [ operation B3 ]
496
+ * [ operation C3 ]
497
+ * [ operation C3r ]
498
+ * [ operation C2' ]
499
+ * [ operation C1' ]
500
+ * ```
501
+ *
502
+ * Selective undo works on the same basis, however, instead of undoing the last batch in the undo stack, any batch can be undone.
503
+ * The same algorithm applies: operations from a batch (i.e. `A1`) are reversed and then transformed by operations stored in history.
504
+ *
505
+ * Redo also is very similar to undo. It has its own stack that is filled with undoing (reversed batches). Operations from
506
+ * the batch that is re-done are reversed-back, transformed in proper order and applied to the document.
507
+ *
508
+ * ```
509
+ * History Undo stack Redo stack
510
+ * ================= ================================== ==================================
511
+ * [ operation A1 ] [ batch A ]
512
+ * [ operation B1 ] [ batch B ]
513
+ * [ operation B2 ] [ batch Crr ]
514
+ * [ operation C1 ]
515
+ * [ operation C2 ]
516
+ * [ operation B3 ]
517
+ * [ operation C3 ]
518
+ * [ operation C3r ]
519
+ * [ operation C2' ]
520
+ * [ operation C1' ]
521
+ * [ operation C1'r]
522
+ * [ operation C2'r]
523
+ * [ operation C3rr]
524
+ * ```
525
+ */ class Undo extends Plugin {
526
+ /**
527
+ * @inheritDoc
528
+ */ static get requires() {
529
+ return [
530
+ UndoEditing,
531
+ UndoUI
532
+ ];
533
+ }
534
+ /**
535
+ * @inheritDoc
536
+ */ static get pluginName() {
537
+ return 'Undo';
538
+ }
539
+ /**
540
+ * @inheritDoc
541
+ */ static get isOfficialPlugin() {
542
+ return true;
543
+ }
544
+ }
545
+
546
+ export { Undo, UndoEditing, UndoUI };
547
+ //# sourceMappingURL=index.js.map