@ckeditor/ckeditor5-engine 38.1.1 → 38.2.0-alpha.1
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/package.json +3 -2
- package/src/controller/datacontroller.d.ts +334 -334
- package/src/controller/datacontroller.js +481 -481
- package/src/controller/editingcontroller.d.ts +98 -98
- package/src/controller/editingcontroller.js +191 -191
- package/src/conversion/conversion.d.ts +478 -478
- package/src/conversion/conversion.js +601 -601
- package/src/conversion/conversionhelpers.d.ts +26 -26
- package/src/conversion/conversionhelpers.js +32 -32
- package/src/conversion/downcastdispatcher.d.ts +547 -547
- package/src/conversion/downcastdispatcher.js +538 -538
- package/src/conversion/downcasthelpers.d.ts +1226 -1226
- package/src/conversion/downcasthelpers.js +2183 -2183
- package/src/conversion/mapper.d.ts +503 -503
- package/src/conversion/mapper.js +536 -536
- package/src/conversion/modelconsumable.d.ts +201 -201
- package/src/conversion/modelconsumable.js +333 -333
- package/src/conversion/upcastdispatcher.d.ts +492 -492
- package/src/conversion/upcastdispatcher.js +460 -460
- package/src/conversion/upcasthelpers.d.ts +499 -499
- package/src/conversion/upcasthelpers.js +950 -950
- package/src/conversion/viewconsumable.d.ts +369 -369
- package/src/conversion/viewconsumable.js +532 -532
- package/src/dataprocessor/basichtmlwriter.d.ts +18 -18
- package/src/dataprocessor/basichtmlwriter.js +19 -19
- package/src/dataprocessor/dataprocessor.d.ts +61 -61
- package/src/dataprocessor/dataprocessor.js +5 -5
- package/src/dataprocessor/htmldataprocessor.d.ts +76 -76
- package/src/dataprocessor/htmldataprocessor.js +96 -96
- package/src/dataprocessor/htmlwriter.d.ts +16 -16
- package/src/dataprocessor/htmlwriter.js +5 -5
- package/src/dataprocessor/xmldataprocessor.d.ts +90 -90
- package/src/dataprocessor/xmldataprocessor.js +108 -108
- package/src/dev-utils/model.d.ts +124 -124
- package/src/dev-utils/model.js +395 -395
- package/src/dev-utils/operationreplayer.d.ts +51 -51
- package/src/dev-utils/operationreplayer.js +112 -112
- package/src/dev-utils/utils.d.ts +37 -37
- package/src/dev-utils/utils.js +73 -73
- package/src/dev-utils/view.d.ts +319 -319
- package/src/dev-utils/view.js +967 -967
- package/src/index.d.ts +113 -113
- package/src/index.js +77 -77
- package/src/model/batch.d.ts +106 -106
- package/src/model/batch.js +96 -96
- package/src/model/differ.d.ts +373 -373
- package/src/model/differ.js +1090 -1090
- package/src/model/document.d.ts +266 -264
- package/src/model/document.js +356 -356
- package/src/model/documentfragment.d.ts +200 -200
- package/src/model/documentfragment.js +306 -306
- package/src/model/documentselection.d.ts +420 -420
- package/src/model/documentselection.js +990 -987
- package/src/model/element.d.ts +165 -165
- package/src/model/element.js +281 -281
- package/src/model/history.d.ts +114 -114
- package/src/model/history.js +207 -207
- package/src/model/item.d.ts +14 -14
- package/src/model/item.js +5 -5
- package/src/model/liveposition.d.ts +77 -77
- package/src/model/liveposition.js +93 -93
- package/src/model/liverange.d.ts +102 -102
- package/src/model/liverange.js +120 -120
- package/src/model/markercollection.d.ts +335 -335
- package/src/model/markercollection.js +403 -403
- package/src/model/model.d.ts +920 -920
- package/src/model/model.js +843 -843
- package/src/model/node.d.ts +256 -256
- package/src/model/node.js +375 -375
- package/src/model/nodelist.d.ts +91 -91
- package/src/model/nodelist.js +163 -163
- package/src/model/operation/attributeoperation.d.ts +103 -103
- package/src/model/operation/attributeoperation.js +148 -148
- package/src/model/operation/detachoperation.d.ts +60 -60
- package/src/model/operation/detachoperation.js +77 -77
- package/src/model/operation/insertoperation.d.ts +90 -90
- package/src/model/operation/insertoperation.js +135 -135
- package/src/model/operation/markeroperation.d.ts +91 -91
- package/src/model/operation/markeroperation.js +107 -107
- package/src/model/operation/mergeoperation.d.ts +100 -100
- package/src/model/operation/mergeoperation.js +167 -167
- package/src/model/operation/moveoperation.d.ts +96 -96
- package/src/model/operation/moveoperation.js +164 -164
- package/src/model/operation/nooperation.d.ts +38 -38
- package/src/model/operation/nooperation.js +48 -48
- package/src/model/operation/operation.d.ts +96 -96
- package/src/model/operation/operation.js +62 -62
- package/src/model/operation/operationfactory.d.ts +18 -18
- package/src/model/operation/operationfactory.js +44 -44
- package/src/model/operation/renameoperation.d.ts +83 -83
- package/src/model/operation/renameoperation.js +115 -115
- package/src/model/operation/rootattributeoperation.d.ts +98 -98
- package/src/model/operation/rootattributeoperation.js +155 -155
- package/src/model/operation/rootoperation.d.ts +80 -80
- package/src/model/operation/rootoperation.js +114 -114
- package/src/model/operation/splitoperation.d.ts +109 -109
- package/src/model/operation/splitoperation.js +194 -194
- package/src/model/operation/transform.d.ts +100 -100
- package/src/model/operation/transform.js +1985 -1985
- package/src/model/operation/utils.d.ts +71 -71
- package/src/model/operation/utils.js +213 -213
- package/src/model/position.d.ts +539 -539
- package/src/model/position.js +979 -979
- package/src/model/range.d.ts +458 -458
- package/src/model/range.js +875 -875
- package/src/model/rootelement.d.ts +54 -54
- package/src/model/rootelement.js +68 -68
- package/src/model/schema.d.ts +1186 -1176
- package/src/model/schema.js +1237 -1237
- package/src/model/selection.d.ts +482 -482
- package/src/model/selection.js +789 -789
- package/src/model/text.d.ts +66 -66
- package/src/model/text.js +85 -85
- package/src/model/textproxy.d.ts +144 -144
- package/src/model/textproxy.js +189 -189
- package/src/model/treewalker.d.ts +186 -186
- package/src/model/treewalker.js +244 -244
- package/src/model/typecheckable.d.ts +285 -285
- package/src/model/typecheckable.js +16 -16
- package/src/model/utils/autoparagraphing.d.ts +37 -37
- package/src/model/utils/autoparagraphing.js +64 -64
- package/src/model/utils/deletecontent.d.ts +58 -58
- package/src/model/utils/deletecontent.js +488 -488
- package/src/model/utils/findoptimalinsertionrange.d.ts +32 -32
- package/src/model/utils/findoptimalinsertionrange.js +57 -57
- package/src/model/utils/getselectedcontent.d.ts +30 -30
- package/src/model/utils/getselectedcontent.js +125 -125
- package/src/model/utils/insertcontent.d.ts +46 -46
- package/src/model/utils/insertcontent.js +705 -705
- package/src/model/utils/insertobject.d.ts +44 -44
- package/src/model/utils/insertobject.js +139 -139
- package/src/model/utils/modifyselection.d.ts +48 -48
- package/src/model/utils/modifyselection.js +186 -186
- package/src/model/utils/selection-post-fixer.d.ts +74 -74
- package/src/model/utils/selection-post-fixer.js +260 -260
- package/src/model/writer.d.ts +851 -851
- package/src/model/writer.js +1306 -1306
- package/src/view/attributeelement.d.ts +108 -108
- package/src/view/attributeelement.js +184 -184
- package/src/view/containerelement.d.ts +49 -49
- package/src/view/containerelement.js +80 -80
- package/src/view/datatransfer.d.ts +79 -79
- package/src/view/datatransfer.js +98 -98
- package/src/view/document.d.ts +184 -184
- package/src/view/document.js +120 -120
- package/src/view/documentfragment.d.ts +149 -149
- package/src/view/documentfragment.js +228 -228
- package/src/view/documentselection.d.ts +306 -306
- package/src/view/documentselection.js +256 -256
- package/src/view/domconverter.d.ts +634 -650
- package/src/view/domconverter.js +1407 -1373
- package/src/view/downcastwriter.d.ts +996 -996
- package/src/view/downcastwriter.js +1696 -1696
- package/src/view/editableelement.d.ts +52 -52
- package/src/view/editableelement.js +61 -61
- package/src/view/element.d.ts +468 -468
- package/src/view/element.js +724 -724
- package/src/view/elementdefinition.d.ts +87 -87
- package/src/view/elementdefinition.js +5 -5
- package/src/view/emptyelement.d.ts +41 -41
- package/src/view/emptyelement.js +73 -73
- package/src/view/filler.d.ts +111 -111
- package/src/view/filler.js +150 -148
- package/src/view/item.d.ts +14 -14
- package/src/view/item.js +5 -5
- package/src/view/matcher.d.ts +486 -486
- package/src/view/matcher.js +507 -507
- package/src/view/node.d.ts +163 -163
- package/src/view/node.js +228 -228
- package/src/view/observer/arrowkeysobserver.d.ts +45 -45
- package/src/view/observer/arrowkeysobserver.js +40 -40
- package/src/view/observer/bubblingemittermixin.d.ts +166 -166
- package/src/view/observer/bubblingemittermixin.js +172 -172
- package/src/view/observer/bubblingeventinfo.d.ts +47 -47
- package/src/view/observer/bubblingeventinfo.js +37 -37
- package/src/view/observer/clickobserver.d.ts +43 -43
- package/src/view/observer/clickobserver.js +29 -29
- package/src/view/observer/compositionobserver.d.ts +82 -82
- package/src/view/observer/compositionobserver.js +60 -60
- package/src/view/observer/domeventdata.d.ts +50 -50
- package/src/view/observer/domeventdata.js +47 -47
- package/src/view/observer/domeventobserver.d.ts +73 -73
- package/src/view/observer/domeventobserver.js +79 -79
- package/src/view/observer/fakeselectionobserver.d.ts +47 -47
- package/src/view/observer/fakeselectionobserver.js +91 -91
- package/src/view/observer/focusobserver.d.ts +82 -82
- package/src/view/observer/focusobserver.js +86 -86
- package/src/view/observer/inputobserver.d.ts +86 -86
- package/src/view/observer/inputobserver.js +164 -164
- package/src/view/observer/keyobserver.d.ts +66 -66
- package/src/view/observer/keyobserver.js +39 -39
- package/src/view/observer/mouseobserver.d.ts +89 -89
- package/src/view/observer/mouseobserver.js +29 -29
- package/src/view/observer/mutationobserver.d.ts +86 -86
- package/src/view/observer/mutationobserver.js +206 -206
- package/src/view/observer/observer.d.ts +89 -89
- package/src/view/observer/observer.js +84 -84
- package/src/view/observer/selectionobserver.d.ts +148 -148
- package/src/view/observer/selectionobserver.js +202 -202
- package/src/view/observer/tabobserver.d.ts +46 -46
- package/src/view/observer/tabobserver.js +42 -42
- package/src/view/placeholder.d.ts +85 -85
- package/src/view/placeholder.js +230 -230
- package/src/view/position.d.ts +189 -189
- package/src/view/position.js +324 -324
- package/src/view/range.d.ts +279 -279
- package/src/view/range.js +430 -430
- package/src/view/rawelement.d.ts +73 -73
- package/src/view/rawelement.js +105 -105
- package/src/view/renderer.d.ts +265 -265
- package/src/view/renderer.js +1000 -1000
- package/src/view/rooteditableelement.d.ts +41 -41
- package/src/view/rooteditableelement.js +69 -69
- package/src/view/selection.d.ts +375 -375
- package/src/view/selection.js +559 -559
- package/src/view/styles/background.d.ts +33 -33
- package/src/view/styles/background.js +74 -74
- package/src/view/styles/border.d.ts +43 -43
- package/src/view/styles/border.js +316 -316
- package/src/view/styles/margin.d.ts +29 -29
- package/src/view/styles/margin.js +34 -34
- package/src/view/styles/padding.d.ts +29 -29
- package/src/view/styles/padding.js +34 -34
- package/src/view/styles/utils.d.ts +93 -93
- package/src/view/styles/utils.js +219 -219
- package/src/view/stylesmap.d.ts +675 -675
- package/src/view/stylesmap.js +766 -766
- package/src/view/text.d.ts +74 -74
- package/src/view/text.js +93 -93
- package/src/view/textproxy.d.ts +97 -97
- package/src/view/textproxy.js +124 -124
- package/src/view/treewalker.d.ts +195 -195
- package/src/view/treewalker.js +327 -327
- package/src/view/typecheckable.d.ts +448 -448
- package/src/view/typecheckable.js +19 -19
- package/src/view/uielement.d.ts +96 -96
- package/src/view/uielement.js +182 -182
- package/src/view/upcastwriter.d.ts +417 -417
- package/src/view/upcastwriter.js +359 -359
- package/src/view/view.d.ts +480 -480
- package/src/view/view.js +534 -534
|
@@ -1,194 +1,194 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license Copyright (c) 2003-2023, 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
|
-
/**
|
|
6
|
-
* @module engine/model/operation/splitoperation
|
|
7
|
-
*/
|
|
8
|
-
import Operation from './operation';
|
|
9
|
-
import MergeOperation from './mergeoperation';
|
|
10
|
-
import Position from '../position';
|
|
11
|
-
import Range from '../range';
|
|
12
|
-
import { _insert, _move } from './utils';
|
|
13
|
-
import { CKEditorError } from '@ckeditor/ckeditor5-utils';
|
|
14
|
-
/**
|
|
15
|
-
* Operation to split {@link module:engine/model/element~Element an element} at given
|
|
16
|
-
* {@link module:engine/model/operation/splitoperation~SplitOperation#splitPosition split position} into two elements,
|
|
17
|
-
* both containing a part of the element's original content.
|
|
18
|
-
*/
|
|
19
|
-
export default class SplitOperation extends Operation {
|
|
20
|
-
/**
|
|
21
|
-
* Creates a split operation.
|
|
22
|
-
*
|
|
23
|
-
* @param splitPosition Position at which an element should be split.
|
|
24
|
-
* @param howMany Total offset size of elements that are in the split element after `position`.
|
|
25
|
-
* @param insertionPosition Position at which the clone of split element (or element from graveyard) will be inserted.
|
|
26
|
-
* @param graveyardPosition Position in the graveyard root before the element which
|
|
27
|
-
* should be used as a parent of the nodes after `position`. If it is not set, a copy of the the `position` parent will be used.
|
|
28
|
-
* @param baseVersion Document {@link module:engine/model/document~Document#version} on which operation
|
|
29
|
-
* can be applied or `null` if the operation operates on detached (non-document) tree.
|
|
30
|
-
*/
|
|
31
|
-
constructor(splitPosition, howMany, insertionPosition, graveyardPosition, baseVersion) {
|
|
32
|
-
super(baseVersion);
|
|
33
|
-
this.splitPosition = splitPosition.clone();
|
|
34
|
-
// Keep position sticking to the next node. This way any new content added at the place where the element is split
|
|
35
|
-
// will be left in the original element.
|
|
36
|
-
this.splitPosition.stickiness = 'toNext';
|
|
37
|
-
this.howMany = howMany;
|
|
38
|
-
this.insertionPosition = insertionPosition;
|
|
39
|
-
this.graveyardPosition = graveyardPosition ? graveyardPosition.clone() : null;
|
|
40
|
-
if (this.graveyardPosition) {
|
|
41
|
-
this.graveyardPosition.stickiness = 'toNext';
|
|
42
|
-
}
|
|
43
|
-
}
|
|
44
|
-
/**
|
|
45
|
-
* @inheritDoc
|
|
46
|
-
*/
|
|
47
|
-
get type() {
|
|
48
|
-
return 'split';
|
|
49
|
-
}
|
|
50
|
-
/**
|
|
51
|
-
* Position inside the new clone of a split element.
|
|
52
|
-
*
|
|
53
|
-
* This is a position where nodes that are after the split position will be moved to.
|
|
54
|
-
*/
|
|
55
|
-
get moveTargetPosition() {
|
|
56
|
-
const path = this.insertionPosition.path.slice();
|
|
57
|
-
path.push(0);
|
|
58
|
-
return new Position(this.insertionPosition.root, path);
|
|
59
|
-
}
|
|
60
|
-
/**
|
|
61
|
-
* Artificial range that contains all the nodes from the split element that will be moved to the new element.
|
|
62
|
-
* The range starts at {@link #splitPosition} and ends in the same parent, at `POSITIVE_INFINITY` offset.
|
|
63
|
-
*/
|
|
64
|
-
get movedRange() {
|
|
65
|
-
const end = this.splitPosition.getShiftedBy(Number.POSITIVE_INFINITY);
|
|
66
|
-
return new Range(this.splitPosition, end);
|
|
67
|
-
}
|
|
68
|
-
/**
|
|
69
|
-
* @inheritDoc
|
|
70
|
-
*/
|
|
71
|
-
get affectedSelectable() {
|
|
72
|
-
// These could be positions but `Selectable` type only supports `Iterable<Range>`.
|
|
73
|
-
const ranges = [
|
|
74
|
-
Range._createFromPositionAndShift(this.splitPosition, 0),
|
|
75
|
-
Range._createFromPositionAndShift(this.insertionPosition, 0)
|
|
76
|
-
];
|
|
77
|
-
if (this.graveyardPosition) {
|
|
78
|
-
ranges.push(Range._createFromPositionAndShift(this.graveyardPosition, 0));
|
|
79
|
-
}
|
|
80
|
-
return ranges;
|
|
81
|
-
}
|
|
82
|
-
/**
|
|
83
|
-
* Creates and returns an operation that has the same parameters as this operation.
|
|
84
|
-
*
|
|
85
|
-
* @returns Clone of this operation.
|
|
86
|
-
*/
|
|
87
|
-
clone() {
|
|
88
|
-
return new SplitOperation(this.splitPosition, this.howMany, this.insertionPosition, this.graveyardPosition, this.baseVersion);
|
|
89
|
-
}
|
|
90
|
-
/**
|
|
91
|
-
* See {@link module:engine/model/operation/operation~Operation#getReversed `Operation#getReversed()`}.
|
|
92
|
-
*/
|
|
93
|
-
getReversed() {
|
|
94
|
-
const graveyard = this.splitPosition.root.document.graveyard;
|
|
95
|
-
const graveyardPosition = new Position(graveyard, [0]);
|
|
96
|
-
return new MergeOperation(this.moveTargetPosition, this.howMany, this.splitPosition, graveyardPosition, this.baseVersion + 1);
|
|
97
|
-
}
|
|
98
|
-
/**
|
|
99
|
-
* @inheritDoc
|
|
100
|
-
* @internal
|
|
101
|
-
*/
|
|
102
|
-
_validate() {
|
|
103
|
-
const element = this.splitPosition.parent;
|
|
104
|
-
const offset = this.splitPosition.offset;
|
|
105
|
-
// Validate whether split operation has correct parameters.
|
|
106
|
-
if (!element || element.maxOffset < offset) {
|
|
107
|
-
/**
|
|
108
|
-
* Split position is invalid.
|
|
109
|
-
*
|
|
110
|
-
* @error split-operation-position-invalid
|
|
111
|
-
*/
|
|
112
|
-
throw new CKEditorError('split-operation-position-invalid', this);
|
|
113
|
-
}
|
|
114
|
-
else if (!element.parent) {
|
|
115
|
-
/**
|
|
116
|
-
* Cannot split root element.
|
|
117
|
-
*
|
|
118
|
-
* @error split-operation-split-in-root
|
|
119
|
-
*/
|
|
120
|
-
throw new CKEditorError('split-operation-split-in-root', this);
|
|
121
|
-
}
|
|
122
|
-
else if (this.howMany != element.maxOffset - this.splitPosition.offset) {
|
|
123
|
-
/**
|
|
124
|
-
* Split operation specifies wrong number of nodes to move.
|
|
125
|
-
*
|
|
126
|
-
* @error split-operation-how-many-invalid
|
|
127
|
-
*/
|
|
128
|
-
throw new CKEditorError('split-operation-how-many-invalid', this);
|
|
129
|
-
}
|
|
130
|
-
else if (this.graveyardPosition && !this.graveyardPosition.nodeAfter) {
|
|
131
|
-
/**
|
|
132
|
-
* Graveyard position invalid.
|
|
133
|
-
*
|
|
134
|
-
* @error split-operation-graveyard-position-invalid
|
|
135
|
-
*/
|
|
136
|
-
throw new CKEditorError('split-operation-graveyard-position-invalid', this);
|
|
137
|
-
}
|
|
138
|
-
}
|
|
139
|
-
/**
|
|
140
|
-
* @inheritDoc
|
|
141
|
-
* @internal
|
|
142
|
-
*/
|
|
143
|
-
_execute() {
|
|
144
|
-
const splitElement = this.splitPosition.parent;
|
|
145
|
-
if (this.graveyardPosition) {
|
|
146
|
-
_move(Range._createFromPositionAndShift(this.graveyardPosition, 1), this.insertionPosition);
|
|
147
|
-
}
|
|
148
|
-
else {
|
|
149
|
-
const newElement = splitElement._clone();
|
|
150
|
-
_insert(this.insertionPosition, newElement);
|
|
151
|
-
}
|
|
152
|
-
const sourceRange = new Range(Position._createAt(splitElement, this.splitPosition.offset), Position._createAt(splitElement, splitElement.maxOffset));
|
|
153
|
-
_move(sourceRange, this.moveTargetPosition);
|
|
154
|
-
}
|
|
155
|
-
/**
|
|
156
|
-
* @inheritDoc
|
|
157
|
-
*/
|
|
158
|
-
toJSON() {
|
|
159
|
-
const json = super.toJSON();
|
|
160
|
-
json.splitPosition = this.splitPosition.toJSON();
|
|
161
|
-
json.insertionPosition = this.insertionPosition.toJSON();
|
|
162
|
-
if (this.graveyardPosition) {
|
|
163
|
-
json.graveyardPosition = this.graveyardPosition.toJSON();
|
|
164
|
-
}
|
|
165
|
-
return json;
|
|
166
|
-
}
|
|
167
|
-
/**
|
|
168
|
-
* @inheritDoc
|
|
169
|
-
*/
|
|
170
|
-
static get className() {
|
|
171
|
-
return 'SplitOperation';
|
|
172
|
-
}
|
|
173
|
-
/**
|
|
174
|
-
* Helper function that returns a default insertion position basing on given `splitPosition`. The default insertion
|
|
175
|
-
* position is after the split element.
|
|
176
|
-
*/
|
|
177
|
-
static getInsertionPosition(splitPosition) {
|
|
178
|
-
const path = splitPosition.path.slice(0, -1);
|
|
179
|
-
path[path.length - 1]++;
|
|
180
|
-
return new Position(splitPosition.root, path, 'toPrevious');
|
|
181
|
-
}
|
|
182
|
-
/**
|
|
183
|
-
* Creates `SplitOperation` object from deserialized object, i.e. from parsed JSON string.
|
|
184
|
-
*
|
|
185
|
-
* @param json Deserialized JSON object.
|
|
186
|
-
* @param document Document on which this operation will be applied.
|
|
187
|
-
*/
|
|
188
|
-
static fromJSON(json, document) {
|
|
189
|
-
const splitPosition = Position.fromJSON(json.splitPosition, document);
|
|
190
|
-
const insertionPosition = Position.fromJSON(json.insertionPosition, document);
|
|
191
|
-
const graveyardPosition = json.graveyardPosition ? Position.fromJSON(json.graveyardPosition, document) : null;
|
|
192
|
-
return new this(splitPosition, json.howMany, insertionPosition, graveyardPosition, json.baseVersion);
|
|
193
|
-
}
|
|
194
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* @license Copyright (c) 2003-2023, 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
|
+
/**
|
|
6
|
+
* @module engine/model/operation/splitoperation
|
|
7
|
+
*/
|
|
8
|
+
import Operation from './operation.js';
|
|
9
|
+
import MergeOperation from './mergeoperation.js';
|
|
10
|
+
import Position from '../position.js';
|
|
11
|
+
import Range from '../range.js';
|
|
12
|
+
import { _insert, _move } from './utils.js';
|
|
13
|
+
import { CKEditorError } from '@ckeditor/ckeditor5-utils';
|
|
14
|
+
/**
|
|
15
|
+
* Operation to split {@link module:engine/model/element~Element an element} at given
|
|
16
|
+
* {@link module:engine/model/operation/splitoperation~SplitOperation#splitPosition split position} into two elements,
|
|
17
|
+
* both containing a part of the element's original content.
|
|
18
|
+
*/
|
|
19
|
+
export default class SplitOperation extends Operation {
|
|
20
|
+
/**
|
|
21
|
+
* Creates a split operation.
|
|
22
|
+
*
|
|
23
|
+
* @param splitPosition Position at which an element should be split.
|
|
24
|
+
* @param howMany Total offset size of elements that are in the split element after `position`.
|
|
25
|
+
* @param insertionPosition Position at which the clone of split element (or element from graveyard) will be inserted.
|
|
26
|
+
* @param graveyardPosition Position in the graveyard root before the element which
|
|
27
|
+
* should be used as a parent of the nodes after `position`. If it is not set, a copy of the the `position` parent will be used.
|
|
28
|
+
* @param baseVersion Document {@link module:engine/model/document~Document#version} on which operation
|
|
29
|
+
* can be applied or `null` if the operation operates on detached (non-document) tree.
|
|
30
|
+
*/
|
|
31
|
+
constructor(splitPosition, howMany, insertionPosition, graveyardPosition, baseVersion) {
|
|
32
|
+
super(baseVersion);
|
|
33
|
+
this.splitPosition = splitPosition.clone();
|
|
34
|
+
// Keep position sticking to the next node. This way any new content added at the place where the element is split
|
|
35
|
+
// will be left in the original element.
|
|
36
|
+
this.splitPosition.stickiness = 'toNext';
|
|
37
|
+
this.howMany = howMany;
|
|
38
|
+
this.insertionPosition = insertionPosition;
|
|
39
|
+
this.graveyardPosition = graveyardPosition ? graveyardPosition.clone() : null;
|
|
40
|
+
if (this.graveyardPosition) {
|
|
41
|
+
this.graveyardPosition.stickiness = 'toNext';
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* @inheritDoc
|
|
46
|
+
*/
|
|
47
|
+
get type() {
|
|
48
|
+
return 'split';
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Position inside the new clone of a split element.
|
|
52
|
+
*
|
|
53
|
+
* This is a position where nodes that are after the split position will be moved to.
|
|
54
|
+
*/
|
|
55
|
+
get moveTargetPosition() {
|
|
56
|
+
const path = this.insertionPosition.path.slice();
|
|
57
|
+
path.push(0);
|
|
58
|
+
return new Position(this.insertionPosition.root, path);
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Artificial range that contains all the nodes from the split element that will be moved to the new element.
|
|
62
|
+
* The range starts at {@link #splitPosition} and ends in the same parent, at `POSITIVE_INFINITY` offset.
|
|
63
|
+
*/
|
|
64
|
+
get movedRange() {
|
|
65
|
+
const end = this.splitPosition.getShiftedBy(Number.POSITIVE_INFINITY);
|
|
66
|
+
return new Range(this.splitPosition, end);
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* @inheritDoc
|
|
70
|
+
*/
|
|
71
|
+
get affectedSelectable() {
|
|
72
|
+
// These could be positions but `Selectable` type only supports `Iterable<Range>`.
|
|
73
|
+
const ranges = [
|
|
74
|
+
Range._createFromPositionAndShift(this.splitPosition, 0),
|
|
75
|
+
Range._createFromPositionAndShift(this.insertionPosition, 0)
|
|
76
|
+
];
|
|
77
|
+
if (this.graveyardPosition) {
|
|
78
|
+
ranges.push(Range._createFromPositionAndShift(this.graveyardPosition, 0));
|
|
79
|
+
}
|
|
80
|
+
return ranges;
|
|
81
|
+
}
|
|
82
|
+
/**
|
|
83
|
+
* Creates and returns an operation that has the same parameters as this operation.
|
|
84
|
+
*
|
|
85
|
+
* @returns Clone of this operation.
|
|
86
|
+
*/
|
|
87
|
+
clone() {
|
|
88
|
+
return new SplitOperation(this.splitPosition, this.howMany, this.insertionPosition, this.graveyardPosition, this.baseVersion);
|
|
89
|
+
}
|
|
90
|
+
/**
|
|
91
|
+
* See {@link module:engine/model/operation/operation~Operation#getReversed `Operation#getReversed()`}.
|
|
92
|
+
*/
|
|
93
|
+
getReversed() {
|
|
94
|
+
const graveyard = this.splitPosition.root.document.graveyard;
|
|
95
|
+
const graveyardPosition = new Position(graveyard, [0]);
|
|
96
|
+
return new MergeOperation(this.moveTargetPosition, this.howMany, this.splitPosition, graveyardPosition, this.baseVersion + 1);
|
|
97
|
+
}
|
|
98
|
+
/**
|
|
99
|
+
* @inheritDoc
|
|
100
|
+
* @internal
|
|
101
|
+
*/
|
|
102
|
+
_validate() {
|
|
103
|
+
const element = this.splitPosition.parent;
|
|
104
|
+
const offset = this.splitPosition.offset;
|
|
105
|
+
// Validate whether split operation has correct parameters.
|
|
106
|
+
if (!element || element.maxOffset < offset) {
|
|
107
|
+
/**
|
|
108
|
+
* Split position is invalid.
|
|
109
|
+
*
|
|
110
|
+
* @error split-operation-position-invalid
|
|
111
|
+
*/
|
|
112
|
+
throw new CKEditorError('split-operation-position-invalid', this);
|
|
113
|
+
}
|
|
114
|
+
else if (!element.parent) {
|
|
115
|
+
/**
|
|
116
|
+
* Cannot split root element.
|
|
117
|
+
*
|
|
118
|
+
* @error split-operation-split-in-root
|
|
119
|
+
*/
|
|
120
|
+
throw new CKEditorError('split-operation-split-in-root', this);
|
|
121
|
+
}
|
|
122
|
+
else if (this.howMany != element.maxOffset - this.splitPosition.offset) {
|
|
123
|
+
/**
|
|
124
|
+
* Split operation specifies wrong number of nodes to move.
|
|
125
|
+
*
|
|
126
|
+
* @error split-operation-how-many-invalid
|
|
127
|
+
*/
|
|
128
|
+
throw new CKEditorError('split-operation-how-many-invalid', this);
|
|
129
|
+
}
|
|
130
|
+
else if (this.graveyardPosition && !this.graveyardPosition.nodeAfter) {
|
|
131
|
+
/**
|
|
132
|
+
* Graveyard position invalid.
|
|
133
|
+
*
|
|
134
|
+
* @error split-operation-graveyard-position-invalid
|
|
135
|
+
*/
|
|
136
|
+
throw new CKEditorError('split-operation-graveyard-position-invalid', this);
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
/**
|
|
140
|
+
* @inheritDoc
|
|
141
|
+
* @internal
|
|
142
|
+
*/
|
|
143
|
+
_execute() {
|
|
144
|
+
const splitElement = this.splitPosition.parent;
|
|
145
|
+
if (this.graveyardPosition) {
|
|
146
|
+
_move(Range._createFromPositionAndShift(this.graveyardPosition, 1), this.insertionPosition);
|
|
147
|
+
}
|
|
148
|
+
else {
|
|
149
|
+
const newElement = splitElement._clone();
|
|
150
|
+
_insert(this.insertionPosition, newElement);
|
|
151
|
+
}
|
|
152
|
+
const sourceRange = new Range(Position._createAt(splitElement, this.splitPosition.offset), Position._createAt(splitElement, splitElement.maxOffset));
|
|
153
|
+
_move(sourceRange, this.moveTargetPosition);
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* @inheritDoc
|
|
157
|
+
*/
|
|
158
|
+
toJSON() {
|
|
159
|
+
const json = super.toJSON();
|
|
160
|
+
json.splitPosition = this.splitPosition.toJSON();
|
|
161
|
+
json.insertionPosition = this.insertionPosition.toJSON();
|
|
162
|
+
if (this.graveyardPosition) {
|
|
163
|
+
json.graveyardPosition = this.graveyardPosition.toJSON();
|
|
164
|
+
}
|
|
165
|
+
return json;
|
|
166
|
+
}
|
|
167
|
+
/**
|
|
168
|
+
* @inheritDoc
|
|
169
|
+
*/
|
|
170
|
+
static get className() {
|
|
171
|
+
return 'SplitOperation';
|
|
172
|
+
}
|
|
173
|
+
/**
|
|
174
|
+
* Helper function that returns a default insertion position basing on given `splitPosition`. The default insertion
|
|
175
|
+
* position is after the split element.
|
|
176
|
+
*/
|
|
177
|
+
static getInsertionPosition(splitPosition) {
|
|
178
|
+
const path = splitPosition.path.slice(0, -1);
|
|
179
|
+
path[path.length - 1]++;
|
|
180
|
+
return new Position(splitPosition.root, path, 'toPrevious');
|
|
181
|
+
}
|
|
182
|
+
/**
|
|
183
|
+
* Creates `SplitOperation` object from deserialized object, i.e. from parsed JSON string.
|
|
184
|
+
*
|
|
185
|
+
* @param json Deserialized JSON object.
|
|
186
|
+
* @param document Document on which this operation will be applied.
|
|
187
|
+
*/
|
|
188
|
+
static fromJSON(json, document) {
|
|
189
|
+
const splitPosition = Position.fromJSON(json.splitPosition, document);
|
|
190
|
+
const insertionPosition = Position.fromJSON(json.insertionPosition, document);
|
|
191
|
+
const graveyardPosition = json.graveyardPosition ? Position.fromJSON(json.graveyardPosition, document) : null;
|
|
192
|
+
return new this(splitPosition, json.howMany, insertionPosition, graveyardPosition, json.baseVersion);
|
|
193
|
+
}
|
|
194
|
+
}
|
|
@@ -1,100 +1,100 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @license Copyright (c) 2003-2023, 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 type Operation from './operation';
|
|
6
|
-
import type Document from '../document';
|
|
7
|
-
/**
|
|
8
|
-
* Transforms operation `a` by operation `b`.
|
|
9
|
-
*
|
|
10
|
-
* @param a Operation to be transformed.
|
|
11
|
-
* @param b Operation to transform by.
|
|
12
|
-
* @param context Transformation context for this transformation.
|
|
13
|
-
* @returns Transformation result.
|
|
14
|
-
*/
|
|
15
|
-
export declare function transform(a: Operation, b: Operation, context?: TransformationContext): Array<Operation>;
|
|
16
|
-
/**
|
|
17
|
-
* Performs a transformation of two sets of operations - `operationsA` and `operationsB`. The transformation is two-way -
|
|
18
|
-
* both transformed `operationsA` and transformed `operationsB` are returned.
|
|
19
|
-
*
|
|
20
|
-
* Note, that the first operation in each set should base on the same document state (
|
|
21
|
-
* {@link module:engine/model/document~Document#version document version}).
|
|
22
|
-
*
|
|
23
|
-
* It is assumed that `operationsA` are "more important" during conflict resolution between two operations.
|
|
24
|
-
*
|
|
25
|
-
* New copies of both passed arrays and operations inside them are returned. Passed arguments are not altered.
|
|
26
|
-
*
|
|
27
|
-
* Base versions of the transformed operations sets are updated accordingly. For example, assume that base versions are `4`
|
|
28
|
-
* and there are `3` operations in `operationsA` and `5` operations in `operationsB`. Then:
|
|
29
|
-
*
|
|
30
|
-
* * transformed `operationsA` will start from base version `9` (`4` base version + `5` operations B),
|
|
31
|
-
* * transformed `operationsB` will start from base version `7` (`4` base version + `3` operations A).
|
|
32
|
-
*
|
|
33
|
-
* If no operation was broken into two during transformation, then both sets will end up with an operation that bases on version `11`:
|
|
34
|
-
*
|
|
35
|
-
* * transformed `operationsA` start from `9` and there are `3` of them, so the last will have `baseVersion` equal to `11`,
|
|
36
|
-
* * transformed `operationsB` start from `7` and there are `5` of them, so the last will have `baseVersion` equal to `11`.
|
|
37
|
-
*
|
|
38
|
-
* @param operationsA
|
|
39
|
-
* @param operationsB
|
|
40
|
-
* @param options Additional transformation options.
|
|
41
|
-
* @param options.document Document which the operations change.
|
|
42
|
-
* @param options.useRelations Whether during transformation relations should be used (used during undo for better conflict resolution).
|
|
43
|
-
* @param options.padWithNoOps Whether additional {@link module:engine/model/operation/nooperation~NoOperation}s
|
|
44
|
-
* should be added to the transformation results to force the same last base version for both transformed sets (in case
|
|
45
|
-
* if some operations got broken into multiple operations during transformation).
|
|
46
|
-
* @param options.forceWeakRemove If set to `false`, remove operation will be always stronger than move operation,
|
|
47
|
-
* so the removed nodes won't end up back in the document root. When set to `true`, context data will be used.
|
|
48
|
-
* @returns Transformation result.
|
|
49
|
-
*/
|
|
50
|
-
export declare function transformSets(operationsA: Array<Operation>, operationsB: Array<Operation>, options: {
|
|
51
|
-
document: Document;
|
|
52
|
-
useRelations?: boolean;
|
|
53
|
-
padWithNoOps?: boolean;
|
|
54
|
-
forceWeakRemove?: boolean;
|
|
55
|
-
}): TransformSetsResult;
|
|
56
|
-
/**
|
|
57
|
-
* The result of {@link module:engine/model/operation/transform~transformSets}.
|
|
58
|
-
*/
|
|
59
|
-
export interface TransformSetsResult {
|
|
60
|
-
/**
|
|
61
|
-
* Transformed `operationsA`.
|
|
62
|
-
*/
|
|
63
|
-
operationsA: Array<Operation>;
|
|
64
|
-
/**
|
|
65
|
-
* Transformed `operationsB`.
|
|
66
|
-
*/
|
|
67
|
-
operationsB: Array<Operation>;
|
|
68
|
-
/**
|
|
69
|
-
* A map that links transformed operations to original operations. The keys are the transformed
|
|
70
|
-
* operations and the values are the original operations from the input (`operationsA` and `operationsB`).
|
|
71
|
-
*/
|
|
72
|
-
originalOperations: Map<Operation, Operation>;
|
|
73
|
-
}
|
|
74
|
-
/**
|
|
75
|
-
* Holds additional contextual information about a transformed pair of operations (`a` and `b`). Those information
|
|
76
|
-
* can be used for better conflict resolving.
|
|
77
|
-
*/
|
|
78
|
-
export type TransformationContext = {
|
|
79
|
-
/**
|
|
80
|
-
* Whether `a` is strong operation in this transformation, or weak.
|
|
81
|
-
*/
|
|
82
|
-
aIsStrong?: boolean;
|
|
83
|
-
/**
|
|
84
|
-
* Whether `a` operation was undone.
|
|
85
|
-
*/
|
|
86
|
-
aWasUndone?: boolean;
|
|
87
|
-
/**
|
|
88
|
-
* Whether `b` operation was undone.
|
|
89
|
-
*/
|
|
90
|
-
bWasUndone?: boolean;
|
|
91
|
-
/**
|
|
92
|
-
* The relation between `a` operation and an operation undone by `b` operation.
|
|
93
|
-
*/
|
|
94
|
-
abRelation?: any;
|
|
95
|
-
/**
|
|
96
|
-
* The relation between `b` operation and an operation undone by `a` operation.
|
|
97
|
-
*/
|
|
98
|
-
baRelation?: any;
|
|
99
|
-
forceWeakRemove?: boolean;
|
|
100
|
-
};
|
|
1
|
+
/**
|
|
2
|
+
* @license Copyright (c) 2003-2023, 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 type Operation from './operation.js';
|
|
6
|
+
import type Document from '../document.js';
|
|
7
|
+
/**
|
|
8
|
+
* Transforms operation `a` by operation `b`.
|
|
9
|
+
*
|
|
10
|
+
* @param a Operation to be transformed.
|
|
11
|
+
* @param b Operation to transform by.
|
|
12
|
+
* @param context Transformation context for this transformation.
|
|
13
|
+
* @returns Transformation result.
|
|
14
|
+
*/
|
|
15
|
+
export declare function transform(a: Operation, b: Operation, context?: TransformationContext): Array<Operation>;
|
|
16
|
+
/**
|
|
17
|
+
* Performs a transformation of two sets of operations - `operationsA` and `operationsB`. The transformation is two-way -
|
|
18
|
+
* both transformed `operationsA` and transformed `operationsB` are returned.
|
|
19
|
+
*
|
|
20
|
+
* Note, that the first operation in each set should base on the same document state (
|
|
21
|
+
* {@link module:engine/model/document~Document#version document version}).
|
|
22
|
+
*
|
|
23
|
+
* It is assumed that `operationsA` are "more important" during conflict resolution between two operations.
|
|
24
|
+
*
|
|
25
|
+
* New copies of both passed arrays and operations inside them are returned. Passed arguments are not altered.
|
|
26
|
+
*
|
|
27
|
+
* Base versions of the transformed operations sets are updated accordingly. For example, assume that base versions are `4`
|
|
28
|
+
* and there are `3` operations in `operationsA` and `5` operations in `operationsB`. Then:
|
|
29
|
+
*
|
|
30
|
+
* * transformed `operationsA` will start from base version `9` (`4` base version + `5` operations B),
|
|
31
|
+
* * transformed `operationsB` will start from base version `7` (`4` base version + `3` operations A).
|
|
32
|
+
*
|
|
33
|
+
* If no operation was broken into two during transformation, then both sets will end up with an operation that bases on version `11`:
|
|
34
|
+
*
|
|
35
|
+
* * transformed `operationsA` start from `9` and there are `3` of them, so the last will have `baseVersion` equal to `11`,
|
|
36
|
+
* * transformed `operationsB` start from `7` and there are `5` of them, so the last will have `baseVersion` equal to `11`.
|
|
37
|
+
*
|
|
38
|
+
* @param operationsA
|
|
39
|
+
* @param operationsB
|
|
40
|
+
* @param options Additional transformation options.
|
|
41
|
+
* @param options.document Document which the operations change.
|
|
42
|
+
* @param options.useRelations Whether during transformation relations should be used (used during undo for better conflict resolution).
|
|
43
|
+
* @param options.padWithNoOps Whether additional {@link module:engine/model/operation/nooperation~NoOperation}s
|
|
44
|
+
* should be added to the transformation results to force the same last base version for both transformed sets (in case
|
|
45
|
+
* if some operations got broken into multiple operations during transformation).
|
|
46
|
+
* @param options.forceWeakRemove If set to `false`, remove operation will be always stronger than move operation,
|
|
47
|
+
* so the removed nodes won't end up back in the document root. When set to `true`, context data will be used.
|
|
48
|
+
* @returns Transformation result.
|
|
49
|
+
*/
|
|
50
|
+
export declare function transformSets(operationsA: Array<Operation>, operationsB: Array<Operation>, options: {
|
|
51
|
+
document: Document;
|
|
52
|
+
useRelations?: boolean;
|
|
53
|
+
padWithNoOps?: boolean;
|
|
54
|
+
forceWeakRemove?: boolean;
|
|
55
|
+
}): TransformSetsResult;
|
|
56
|
+
/**
|
|
57
|
+
* The result of {@link module:engine/model/operation/transform~transformSets}.
|
|
58
|
+
*/
|
|
59
|
+
export interface TransformSetsResult {
|
|
60
|
+
/**
|
|
61
|
+
* Transformed `operationsA`.
|
|
62
|
+
*/
|
|
63
|
+
operationsA: Array<Operation>;
|
|
64
|
+
/**
|
|
65
|
+
* Transformed `operationsB`.
|
|
66
|
+
*/
|
|
67
|
+
operationsB: Array<Operation>;
|
|
68
|
+
/**
|
|
69
|
+
* A map that links transformed operations to original operations. The keys are the transformed
|
|
70
|
+
* operations and the values are the original operations from the input (`operationsA` and `operationsB`).
|
|
71
|
+
*/
|
|
72
|
+
originalOperations: Map<Operation, Operation>;
|
|
73
|
+
}
|
|
74
|
+
/**
|
|
75
|
+
* Holds additional contextual information about a transformed pair of operations (`a` and `b`). Those information
|
|
76
|
+
* can be used for better conflict resolving.
|
|
77
|
+
*/
|
|
78
|
+
export type TransformationContext = {
|
|
79
|
+
/**
|
|
80
|
+
* Whether `a` is strong operation in this transformation, or weak.
|
|
81
|
+
*/
|
|
82
|
+
aIsStrong?: boolean;
|
|
83
|
+
/**
|
|
84
|
+
* Whether `a` operation was undone.
|
|
85
|
+
*/
|
|
86
|
+
aWasUndone?: boolean;
|
|
87
|
+
/**
|
|
88
|
+
* Whether `b` operation was undone.
|
|
89
|
+
*/
|
|
90
|
+
bWasUndone?: boolean;
|
|
91
|
+
/**
|
|
92
|
+
* The relation between `a` operation and an operation undone by `b` operation.
|
|
93
|
+
*/
|
|
94
|
+
abRelation?: any;
|
|
95
|
+
/**
|
|
96
|
+
* The relation between `b` operation and an operation undone by `a` operation.
|
|
97
|
+
*/
|
|
98
|
+
baRelation?: any;
|
|
99
|
+
forceWeakRemove?: boolean;
|
|
100
|
+
};
|