@ckeditor/ckeditor5-engine 40.0.0 → 40.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +39 -39
- package/LICENSE.md +3 -3
- package/package.json +2 -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 +562 -562
- package/src/conversion/downcastdispatcher.js +548 -547
- package/src/conversion/downcasthelpers.d.ts +1226 -1226
- package/src/conversion/downcasthelpers.js +2178 -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 +536 -532
- package/src/dataprocessor/basichtmlwriter.d.ts +18 -18
- package/src/dataprocessor/basichtmlwriter.js +20 -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 +114 -114
- package/src/index.js +78 -78
- package/src/model/batch.d.ts +106 -106
- package/src/model/batch.js +96 -96
- package/src/model/differ.d.ts +387 -387
- package/src/model/differ.js +1149 -1149
- package/src/model/document.d.ts +272 -272
- package/src/model/document.js +360 -361
- 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 +993 -993
- 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 +919 -919
- package/src/model/model.js +842 -842
- 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 +59 -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 +76 -76
- package/src/model/operation/rootoperation.js +90 -90
- 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 +217 -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 +60 -60
- package/src/model/rootelement.js +74 -74
- package/src/model/schema.d.ts +1186 -1186
- package/src/model/schema.js +1242 -1242
- 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 +63 -63
- 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 +122 -120
- package/src/view/documentfragment.d.ts +153 -149
- package/src/view/documentfragment.js +234 -228
- package/src/view/documentselection.d.ts +306 -306
- package/src/view/documentselection.js +256 -256
- package/src/view/domconverter.d.ts +652 -640
- package/src/view/domconverter.js +1473 -1450
- package/src/view/downcastwriter.d.ts +996 -996
- package/src/view/downcastwriter.js +1696 -1696
- package/src/view/editableelement.d.ts +62 -62
- package/src/view/editableelement.js +62 -62
- 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 -150
- 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 +96 -96
- package/src/view/placeholder.js +267 -267
- 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 -999
- 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 +765 -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 +183 -182
- package/src/view/upcastwriter.d.ts +417 -417
- package/src/view/upcastwriter.js +359 -359
- package/src/view/view.d.ts +487 -487
- package/src/view/view.js +546 -546
package/src/view/treewalker.js
CHANGED
|
@@ -1,327 +1,327 @@
|
|
|
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/view/treewalker
|
|
7
|
-
*/
|
|
8
|
-
import Element from './element';
|
|
9
|
-
import Text from './text';
|
|
10
|
-
import TextProxy from './textproxy';
|
|
11
|
-
import Position from './position';
|
|
12
|
-
import { CKEditorError } from '@ckeditor/ckeditor5-utils';
|
|
13
|
-
/**
|
|
14
|
-
* Position iterator class. It allows to iterate forward and backward over the document.
|
|
15
|
-
*/
|
|
16
|
-
export default class TreeWalker {
|
|
17
|
-
/**
|
|
18
|
-
* Creates a range iterator. All parameters are optional, but you have to specify either `boundaries` or `startPosition`.
|
|
19
|
-
*
|
|
20
|
-
* @param options Object with configuration.
|
|
21
|
-
*/
|
|
22
|
-
constructor(options = {}) {
|
|
23
|
-
if (!options.boundaries && !options.startPosition) {
|
|
24
|
-
/**
|
|
25
|
-
* Neither boundaries nor starting position have been defined.
|
|
26
|
-
*
|
|
27
|
-
* @error view-tree-walker-no-start-position
|
|
28
|
-
*/
|
|
29
|
-
throw new CKEditorError('view-tree-walker-no-start-position', null);
|
|
30
|
-
}
|
|
31
|
-
if (options.direction && options.direction != 'forward' && options.direction != 'backward') {
|
|
32
|
-
/**
|
|
33
|
-
* Only `backward` and `forward` direction allowed.
|
|
34
|
-
*
|
|
35
|
-
* @error view-tree-walker-unknown-direction
|
|
36
|
-
*/
|
|
37
|
-
throw new CKEditorError('view-tree-walker-unknown-direction', options.startPosition, { direction: options.direction });
|
|
38
|
-
}
|
|
39
|
-
this.boundaries = options.boundaries || null;
|
|
40
|
-
if (options.startPosition) {
|
|
41
|
-
this._position = Position._createAt(options.startPosition);
|
|
42
|
-
}
|
|
43
|
-
else {
|
|
44
|
-
this._position = Position._createAt(options.boundaries[options.direction == 'backward' ? 'end' : 'start']);
|
|
45
|
-
}
|
|
46
|
-
this.direction = options.direction || 'forward';
|
|
47
|
-
this.singleCharacters = !!options.singleCharacters;
|
|
48
|
-
this.shallow = !!options.shallow;
|
|
49
|
-
this.ignoreElementEnd = !!options.ignoreElementEnd;
|
|
50
|
-
this._boundaryStartParent = this.boundaries ? this.boundaries.start.parent : null;
|
|
51
|
-
this._boundaryEndParent = this.boundaries ? this.boundaries.end.parent : null;
|
|
52
|
-
}
|
|
53
|
-
/**
|
|
54
|
-
* Iterable interface.
|
|
55
|
-
*/
|
|
56
|
-
[Symbol.iterator]() {
|
|
57
|
-
return this;
|
|
58
|
-
}
|
|
59
|
-
/**
|
|
60
|
-
* Iterator position. If start position is not defined then position depends on {@link #direction}. If direction is
|
|
61
|
-
* `'forward'` position starts form the beginning, when direction is `'backward'` position starts from the end.
|
|
62
|
-
*/
|
|
63
|
-
get position() {
|
|
64
|
-
return this._position;
|
|
65
|
-
}
|
|
66
|
-
/**
|
|
67
|
-
* Moves {@link #position} in the {@link #direction} skipping values as long as the callback function returns `true`.
|
|
68
|
-
*
|
|
69
|
-
* For example:
|
|
70
|
-
*
|
|
71
|
-
* ```ts
|
|
72
|
-
* walker.skip( value => value.type == 'text' ); // <p>{}foo</p> -> <p>foo[]</p>
|
|
73
|
-
* walker.skip( value => true ); // Move the position to the end: <p>{}foo</p> -> <p>foo</p>[]
|
|
74
|
-
* walker.skip( value => false ); // Do not move the position.
|
|
75
|
-
* ```
|
|
76
|
-
*
|
|
77
|
-
* @param skip Callback function. Gets {@link module:engine/view/treewalker~TreeWalkerValue} and should
|
|
78
|
-
* return `true` if the value should be skipped or `false` if not.
|
|
79
|
-
*/
|
|
80
|
-
skip(skip) {
|
|
81
|
-
let nextResult;
|
|
82
|
-
let prevPosition;
|
|
83
|
-
do {
|
|
84
|
-
prevPosition = this.position;
|
|
85
|
-
nextResult = this.next();
|
|
86
|
-
} while (!nextResult.done && skip(nextResult.value));
|
|
87
|
-
if (!nextResult.done) {
|
|
88
|
-
this._position = prevPosition;
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
/**
|
|
92
|
-
* Gets the next tree walker's value.
|
|
93
|
-
*
|
|
94
|
-
* @returns Object implementing iterator interface, returning
|
|
95
|
-
* information about taken step.
|
|
96
|
-
*/
|
|
97
|
-
next() {
|
|
98
|
-
if (this.direction == 'forward') {
|
|
99
|
-
return this._next();
|
|
100
|
-
}
|
|
101
|
-
else {
|
|
102
|
-
return this._previous();
|
|
103
|
-
}
|
|
104
|
-
}
|
|
105
|
-
/**
|
|
106
|
-
* Makes a step forward in view. Moves the {@link #position} to the next position and returns the encountered value.
|
|
107
|
-
*/
|
|
108
|
-
_next() {
|
|
109
|
-
let position = this.position.clone();
|
|
110
|
-
const previousPosition = this.position;
|
|
111
|
-
const parent = position.parent;
|
|
112
|
-
// We are at the end of the root.
|
|
113
|
-
if (parent.parent === null && position.offset === parent.childCount) {
|
|
114
|
-
return { done: true, value: undefined };
|
|
115
|
-
}
|
|
116
|
-
// We reached the walker boundary.
|
|
117
|
-
if (parent === this._boundaryEndParent && position.offset == this.boundaries.end.offset) {
|
|
118
|
-
return { done: true, value: undefined };
|
|
119
|
-
}
|
|
120
|
-
// Get node just after current position.
|
|
121
|
-
let node;
|
|
122
|
-
// Text is a specific parent because it contains string instead of child nodes.
|
|
123
|
-
if (parent instanceof Text) {
|
|
124
|
-
if (position.isAtEnd) {
|
|
125
|
-
// Prevent returning "elementEnd" for Text node. Skip that value and return the next walker step.
|
|
126
|
-
this._position = Position._createAfter(parent);
|
|
127
|
-
return this._next();
|
|
128
|
-
}
|
|
129
|
-
node = parent.data[position.offset];
|
|
130
|
-
}
|
|
131
|
-
else {
|
|
132
|
-
node = parent.getChild(position.offset);
|
|
133
|
-
}
|
|
134
|
-
if (node instanceof Element) {
|
|
135
|
-
if (!this.shallow) {
|
|
136
|
-
position = new Position(node, 0);
|
|
137
|
-
}
|
|
138
|
-
else {
|
|
139
|
-
// We are past the walker boundaries.
|
|
140
|
-
if (this.boundaries && this.boundaries.end.isBefore(position)) {
|
|
141
|
-
return { done: true, value: undefined };
|
|
142
|
-
}
|
|
143
|
-
position.offset++;
|
|
144
|
-
}
|
|
145
|
-
this._position = position;
|
|
146
|
-
return this._formatReturnValue('elementStart', node, previousPosition, position, 1);
|
|
147
|
-
}
|
|
148
|
-
if (node instanceof Text) {
|
|
149
|
-
if (this.singleCharacters) {
|
|
150
|
-
position = new Position(node, 0);
|
|
151
|
-
this._position = position;
|
|
152
|
-
return this._next();
|
|
153
|
-
}
|
|
154
|
-
let charactersCount = node.data.length;
|
|
155
|
-
let item;
|
|
156
|
-
// If text stick out of walker range, we need to cut it and wrap in TextProxy.
|
|
157
|
-
if (node == this._boundaryEndParent) {
|
|
158
|
-
charactersCount = this.boundaries.end.offset;
|
|
159
|
-
item = new TextProxy(node, 0, charactersCount);
|
|
160
|
-
position = Position._createAfter(item);
|
|
161
|
-
}
|
|
162
|
-
else {
|
|
163
|
-
item = new TextProxy(node, 0, node.data.length);
|
|
164
|
-
// If not just keep moving forward.
|
|
165
|
-
position.offset++;
|
|
166
|
-
}
|
|
167
|
-
this._position = position;
|
|
168
|
-
return this._formatReturnValue('text', item, previousPosition, position, charactersCount);
|
|
169
|
-
}
|
|
170
|
-
if (typeof node == 'string') {
|
|
171
|
-
let textLength;
|
|
172
|
-
if (this.singleCharacters) {
|
|
173
|
-
textLength = 1;
|
|
174
|
-
}
|
|
175
|
-
else {
|
|
176
|
-
// Check if text stick out of walker range.
|
|
177
|
-
const endOffset = parent === this._boundaryEndParent ? this.boundaries.end.offset : parent.data.length;
|
|
178
|
-
textLength = endOffset - position.offset;
|
|
179
|
-
}
|
|
180
|
-
const textProxy = new TextProxy(parent, position.offset, textLength);
|
|
181
|
-
position.offset += textLength;
|
|
182
|
-
this._position = position;
|
|
183
|
-
return this._formatReturnValue('text', textProxy, previousPosition, position, textLength);
|
|
184
|
-
}
|
|
185
|
-
// `node` is not set, we reached the end of current `parent`.
|
|
186
|
-
position = Position._createAfter(parent);
|
|
187
|
-
this._position = position;
|
|
188
|
-
if (this.ignoreElementEnd) {
|
|
189
|
-
return this._next();
|
|
190
|
-
}
|
|
191
|
-
return this._formatReturnValue('elementEnd', parent, previousPosition, position);
|
|
192
|
-
}
|
|
193
|
-
/**
|
|
194
|
-
* Makes a step backward in view. Moves the {@link #position} to the previous position and returns the encountered value.
|
|
195
|
-
*/
|
|
196
|
-
_previous() {
|
|
197
|
-
let position = this.position.clone();
|
|
198
|
-
const previousPosition = this.position;
|
|
199
|
-
const parent = position.parent;
|
|
200
|
-
// We are at the beginning of the root.
|
|
201
|
-
if (parent.parent === null && position.offset === 0) {
|
|
202
|
-
return { done: true, value: undefined };
|
|
203
|
-
}
|
|
204
|
-
// We reached the walker boundary.
|
|
205
|
-
if (parent == this._boundaryStartParent && position.offset == this.boundaries.start.offset) {
|
|
206
|
-
return { done: true, value: undefined };
|
|
207
|
-
}
|
|
208
|
-
// Get node just before current position.
|
|
209
|
-
let node;
|
|
210
|
-
// Text {@link module:engine/view/text~Text} element is a specific parent because contains string instead of child nodes.
|
|
211
|
-
if (parent instanceof Text) {
|
|
212
|
-
if (position.isAtStart) {
|
|
213
|
-
// Prevent returning "elementStart" for Text node. Skip that value and return the next walker step.
|
|
214
|
-
this._position = Position._createBefore(parent);
|
|
215
|
-
return this._previous();
|
|
216
|
-
}
|
|
217
|
-
node = parent.data[position.offset - 1];
|
|
218
|
-
}
|
|
219
|
-
else {
|
|
220
|
-
node = parent.getChild(position.offset - 1);
|
|
221
|
-
}
|
|
222
|
-
if (node instanceof Element) {
|
|
223
|
-
if (this.shallow) {
|
|
224
|
-
position.offset--;
|
|
225
|
-
this._position = position;
|
|
226
|
-
return this._formatReturnValue('elementStart', node, previousPosition, position, 1);
|
|
227
|
-
}
|
|
228
|
-
position = new Position(node, node.childCount);
|
|
229
|
-
this._position = position;
|
|
230
|
-
if (this.ignoreElementEnd) {
|
|
231
|
-
return this._previous();
|
|
232
|
-
}
|
|
233
|
-
return this._formatReturnValue('elementEnd', node, previousPosition, position);
|
|
234
|
-
}
|
|
235
|
-
if (node instanceof Text) {
|
|
236
|
-
if (this.singleCharacters) {
|
|
237
|
-
position = new Position(node, node.data.length);
|
|
238
|
-
this._position = position;
|
|
239
|
-
return this._previous();
|
|
240
|
-
}
|
|
241
|
-
let charactersCount = node.data.length;
|
|
242
|
-
let item;
|
|
243
|
-
// If text stick out of walker range, we need to cut it and wrap in TextProxy.
|
|
244
|
-
if (node == this._boundaryStartParent) {
|
|
245
|
-
const offset = this.boundaries.start.offset;
|
|
246
|
-
item = new TextProxy(node, offset, node.data.length - offset);
|
|
247
|
-
charactersCount = item.data.length;
|
|
248
|
-
position = Position._createBefore(item);
|
|
249
|
-
}
|
|
250
|
-
else {
|
|
251
|
-
item = new TextProxy(node, 0, node.data.length);
|
|
252
|
-
// If not just keep moving backward.
|
|
253
|
-
position.offset--;
|
|
254
|
-
}
|
|
255
|
-
this._position = position;
|
|
256
|
-
return this._formatReturnValue('text', item, previousPosition, position, charactersCount);
|
|
257
|
-
}
|
|
258
|
-
if (typeof node == 'string') {
|
|
259
|
-
let textLength;
|
|
260
|
-
if (!this.singleCharacters) {
|
|
261
|
-
// Check if text stick out of walker range.
|
|
262
|
-
const startOffset = parent === this._boundaryStartParent ? this.boundaries.start.offset : 0;
|
|
263
|
-
textLength = position.offset - startOffset;
|
|
264
|
-
}
|
|
265
|
-
else {
|
|
266
|
-
textLength = 1;
|
|
267
|
-
}
|
|
268
|
-
position.offset -= textLength;
|
|
269
|
-
const textProxy = new TextProxy(parent, position.offset, textLength);
|
|
270
|
-
this._position = position;
|
|
271
|
-
return this._formatReturnValue('text', textProxy, previousPosition, position, textLength);
|
|
272
|
-
}
|
|
273
|
-
// `node` is not set, we reached the beginning of current `parent`.
|
|
274
|
-
position = Position._createBefore(parent);
|
|
275
|
-
this._position = position;
|
|
276
|
-
return this._formatReturnValue('elementStart', parent, previousPosition, position, 1);
|
|
277
|
-
}
|
|
278
|
-
/**
|
|
279
|
-
* Format returned data and adjust `previousPosition` and `nextPosition` if reach the bound of the {@link module:engine/view/text~Text}.
|
|
280
|
-
*
|
|
281
|
-
* @param type Type of step.
|
|
282
|
-
* @param item Item between old and new position.
|
|
283
|
-
* @param previousPosition Previous position of iterator.
|
|
284
|
-
* @param nextPosition Next position of iterator.
|
|
285
|
-
* @param length Length of the item.
|
|
286
|
-
*/
|
|
287
|
-
_formatReturnValue(type, item, previousPosition, nextPosition, length) {
|
|
288
|
-
// Text is a specific parent, because contains string instead of children.
|
|
289
|
-
// Walker doesn't enter to the Text except situations when walker is iterating over every single character,
|
|
290
|
-
// or the bound starts/ends inside the Text. So when the position is at the beginning or at the end of the Text
|
|
291
|
-
// we move it just before or just after Text.
|
|
292
|
-
if (item instanceof TextProxy) {
|
|
293
|
-
// Position is at the end of Text.
|
|
294
|
-
if (item.offsetInText + item.data.length == item.textNode.data.length) {
|
|
295
|
-
if (this.direction == 'forward' && !(this.boundaries && this.boundaries.end.isEqual(this.position))) {
|
|
296
|
-
nextPosition = Position._createAfter(item.textNode);
|
|
297
|
-
// When we change nextPosition of returned value we need also update walker current position.
|
|
298
|
-
this._position = nextPosition;
|
|
299
|
-
}
|
|
300
|
-
else {
|
|
301
|
-
previousPosition = Position._createAfter(item.textNode);
|
|
302
|
-
}
|
|
303
|
-
}
|
|
304
|
-
// Position is at the begining ot the text.
|
|
305
|
-
if (item.offsetInText === 0) {
|
|
306
|
-
if (this.direction == 'backward' && !(this.boundaries && this.boundaries.start.isEqual(this.position))) {
|
|
307
|
-
nextPosition = Position._createBefore(item.textNode);
|
|
308
|
-
// When we change nextPosition of returned value we need also update walker current position.
|
|
309
|
-
this._position = nextPosition;
|
|
310
|
-
}
|
|
311
|
-
else {
|
|
312
|
-
previousPosition = Position._createBefore(item.textNode);
|
|
313
|
-
}
|
|
314
|
-
}
|
|
315
|
-
}
|
|
316
|
-
return {
|
|
317
|
-
done: false,
|
|
318
|
-
value: {
|
|
319
|
-
type,
|
|
320
|
-
item,
|
|
321
|
-
previousPosition,
|
|
322
|
-
nextPosition,
|
|
323
|
-
length
|
|
324
|
-
}
|
|
325
|
-
};
|
|
326
|
-
}
|
|
327
|
-
}
|
|
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/view/treewalker
|
|
7
|
+
*/
|
|
8
|
+
import Element from './element';
|
|
9
|
+
import Text from './text';
|
|
10
|
+
import TextProxy from './textproxy';
|
|
11
|
+
import Position from './position';
|
|
12
|
+
import { CKEditorError } from '@ckeditor/ckeditor5-utils';
|
|
13
|
+
/**
|
|
14
|
+
* Position iterator class. It allows to iterate forward and backward over the document.
|
|
15
|
+
*/
|
|
16
|
+
export default class TreeWalker {
|
|
17
|
+
/**
|
|
18
|
+
* Creates a range iterator. All parameters are optional, but you have to specify either `boundaries` or `startPosition`.
|
|
19
|
+
*
|
|
20
|
+
* @param options Object with configuration.
|
|
21
|
+
*/
|
|
22
|
+
constructor(options = {}) {
|
|
23
|
+
if (!options.boundaries && !options.startPosition) {
|
|
24
|
+
/**
|
|
25
|
+
* Neither boundaries nor starting position have been defined.
|
|
26
|
+
*
|
|
27
|
+
* @error view-tree-walker-no-start-position
|
|
28
|
+
*/
|
|
29
|
+
throw new CKEditorError('view-tree-walker-no-start-position', null);
|
|
30
|
+
}
|
|
31
|
+
if (options.direction && options.direction != 'forward' && options.direction != 'backward') {
|
|
32
|
+
/**
|
|
33
|
+
* Only `backward` and `forward` direction allowed.
|
|
34
|
+
*
|
|
35
|
+
* @error view-tree-walker-unknown-direction
|
|
36
|
+
*/
|
|
37
|
+
throw new CKEditorError('view-tree-walker-unknown-direction', options.startPosition, { direction: options.direction });
|
|
38
|
+
}
|
|
39
|
+
this.boundaries = options.boundaries || null;
|
|
40
|
+
if (options.startPosition) {
|
|
41
|
+
this._position = Position._createAt(options.startPosition);
|
|
42
|
+
}
|
|
43
|
+
else {
|
|
44
|
+
this._position = Position._createAt(options.boundaries[options.direction == 'backward' ? 'end' : 'start']);
|
|
45
|
+
}
|
|
46
|
+
this.direction = options.direction || 'forward';
|
|
47
|
+
this.singleCharacters = !!options.singleCharacters;
|
|
48
|
+
this.shallow = !!options.shallow;
|
|
49
|
+
this.ignoreElementEnd = !!options.ignoreElementEnd;
|
|
50
|
+
this._boundaryStartParent = this.boundaries ? this.boundaries.start.parent : null;
|
|
51
|
+
this._boundaryEndParent = this.boundaries ? this.boundaries.end.parent : null;
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Iterable interface.
|
|
55
|
+
*/
|
|
56
|
+
[Symbol.iterator]() {
|
|
57
|
+
return this;
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* Iterator position. If start position is not defined then position depends on {@link #direction}. If direction is
|
|
61
|
+
* `'forward'` position starts form the beginning, when direction is `'backward'` position starts from the end.
|
|
62
|
+
*/
|
|
63
|
+
get position() {
|
|
64
|
+
return this._position;
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* Moves {@link #position} in the {@link #direction} skipping values as long as the callback function returns `true`.
|
|
68
|
+
*
|
|
69
|
+
* For example:
|
|
70
|
+
*
|
|
71
|
+
* ```ts
|
|
72
|
+
* walker.skip( value => value.type == 'text' ); // <p>{}foo</p> -> <p>foo[]</p>
|
|
73
|
+
* walker.skip( value => true ); // Move the position to the end: <p>{}foo</p> -> <p>foo</p>[]
|
|
74
|
+
* walker.skip( value => false ); // Do not move the position.
|
|
75
|
+
* ```
|
|
76
|
+
*
|
|
77
|
+
* @param skip Callback function. Gets {@link module:engine/view/treewalker~TreeWalkerValue} and should
|
|
78
|
+
* return `true` if the value should be skipped or `false` if not.
|
|
79
|
+
*/
|
|
80
|
+
skip(skip) {
|
|
81
|
+
let nextResult;
|
|
82
|
+
let prevPosition;
|
|
83
|
+
do {
|
|
84
|
+
prevPosition = this.position;
|
|
85
|
+
nextResult = this.next();
|
|
86
|
+
} while (!nextResult.done && skip(nextResult.value));
|
|
87
|
+
if (!nextResult.done) {
|
|
88
|
+
this._position = prevPosition;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
/**
|
|
92
|
+
* Gets the next tree walker's value.
|
|
93
|
+
*
|
|
94
|
+
* @returns Object implementing iterator interface, returning
|
|
95
|
+
* information about taken step.
|
|
96
|
+
*/
|
|
97
|
+
next() {
|
|
98
|
+
if (this.direction == 'forward') {
|
|
99
|
+
return this._next();
|
|
100
|
+
}
|
|
101
|
+
else {
|
|
102
|
+
return this._previous();
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* Makes a step forward in view. Moves the {@link #position} to the next position and returns the encountered value.
|
|
107
|
+
*/
|
|
108
|
+
_next() {
|
|
109
|
+
let position = this.position.clone();
|
|
110
|
+
const previousPosition = this.position;
|
|
111
|
+
const parent = position.parent;
|
|
112
|
+
// We are at the end of the root.
|
|
113
|
+
if (parent.parent === null && position.offset === parent.childCount) {
|
|
114
|
+
return { done: true, value: undefined };
|
|
115
|
+
}
|
|
116
|
+
// We reached the walker boundary.
|
|
117
|
+
if (parent === this._boundaryEndParent && position.offset == this.boundaries.end.offset) {
|
|
118
|
+
return { done: true, value: undefined };
|
|
119
|
+
}
|
|
120
|
+
// Get node just after current position.
|
|
121
|
+
let node;
|
|
122
|
+
// Text is a specific parent because it contains string instead of child nodes.
|
|
123
|
+
if (parent instanceof Text) {
|
|
124
|
+
if (position.isAtEnd) {
|
|
125
|
+
// Prevent returning "elementEnd" for Text node. Skip that value and return the next walker step.
|
|
126
|
+
this._position = Position._createAfter(parent);
|
|
127
|
+
return this._next();
|
|
128
|
+
}
|
|
129
|
+
node = parent.data[position.offset];
|
|
130
|
+
}
|
|
131
|
+
else {
|
|
132
|
+
node = parent.getChild(position.offset);
|
|
133
|
+
}
|
|
134
|
+
if (node instanceof Element) {
|
|
135
|
+
if (!this.shallow) {
|
|
136
|
+
position = new Position(node, 0);
|
|
137
|
+
}
|
|
138
|
+
else {
|
|
139
|
+
// We are past the walker boundaries.
|
|
140
|
+
if (this.boundaries && this.boundaries.end.isBefore(position)) {
|
|
141
|
+
return { done: true, value: undefined };
|
|
142
|
+
}
|
|
143
|
+
position.offset++;
|
|
144
|
+
}
|
|
145
|
+
this._position = position;
|
|
146
|
+
return this._formatReturnValue('elementStart', node, previousPosition, position, 1);
|
|
147
|
+
}
|
|
148
|
+
if (node instanceof Text) {
|
|
149
|
+
if (this.singleCharacters) {
|
|
150
|
+
position = new Position(node, 0);
|
|
151
|
+
this._position = position;
|
|
152
|
+
return this._next();
|
|
153
|
+
}
|
|
154
|
+
let charactersCount = node.data.length;
|
|
155
|
+
let item;
|
|
156
|
+
// If text stick out of walker range, we need to cut it and wrap in TextProxy.
|
|
157
|
+
if (node == this._boundaryEndParent) {
|
|
158
|
+
charactersCount = this.boundaries.end.offset;
|
|
159
|
+
item = new TextProxy(node, 0, charactersCount);
|
|
160
|
+
position = Position._createAfter(item);
|
|
161
|
+
}
|
|
162
|
+
else {
|
|
163
|
+
item = new TextProxy(node, 0, node.data.length);
|
|
164
|
+
// If not just keep moving forward.
|
|
165
|
+
position.offset++;
|
|
166
|
+
}
|
|
167
|
+
this._position = position;
|
|
168
|
+
return this._formatReturnValue('text', item, previousPosition, position, charactersCount);
|
|
169
|
+
}
|
|
170
|
+
if (typeof node == 'string') {
|
|
171
|
+
let textLength;
|
|
172
|
+
if (this.singleCharacters) {
|
|
173
|
+
textLength = 1;
|
|
174
|
+
}
|
|
175
|
+
else {
|
|
176
|
+
// Check if text stick out of walker range.
|
|
177
|
+
const endOffset = parent === this._boundaryEndParent ? this.boundaries.end.offset : parent.data.length;
|
|
178
|
+
textLength = endOffset - position.offset;
|
|
179
|
+
}
|
|
180
|
+
const textProxy = new TextProxy(parent, position.offset, textLength);
|
|
181
|
+
position.offset += textLength;
|
|
182
|
+
this._position = position;
|
|
183
|
+
return this._formatReturnValue('text', textProxy, previousPosition, position, textLength);
|
|
184
|
+
}
|
|
185
|
+
// `node` is not set, we reached the end of current `parent`.
|
|
186
|
+
position = Position._createAfter(parent);
|
|
187
|
+
this._position = position;
|
|
188
|
+
if (this.ignoreElementEnd) {
|
|
189
|
+
return this._next();
|
|
190
|
+
}
|
|
191
|
+
return this._formatReturnValue('elementEnd', parent, previousPosition, position);
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* Makes a step backward in view. Moves the {@link #position} to the previous position and returns the encountered value.
|
|
195
|
+
*/
|
|
196
|
+
_previous() {
|
|
197
|
+
let position = this.position.clone();
|
|
198
|
+
const previousPosition = this.position;
|
|
199
|
+
const parent = position.parent;
|
|
200
|
+
// We are at the beginning of the root.
|
|
201
|
+
if (parent.parent === null && position.offset === 0) {
|
|
202
|
+
return { done: true, value: undefined };
|
|
203
|
+
}
|
|
204
|
+
// We reached the walker boundary.
|
|
205
|
+
if (parent == this._boundaryStartParent && position.offset == this.boundaries.start.offset) {
|
|
206
|
+
return { done: true, value: undefined };
|
|
207
|
+
}
|
|
208
|
+
// Get node just before current position.
|
|
209
|
+
let node;
|
|
210
|
+
// Text {@link module:engine/view/text~Text} element is a specific parent because contains string instead of child nodes.
|
|
211
|
+
if (parent instanceof Text) {
|
|
212
|
+
if (position.isAtStart) {
|
|
213
|
+
// Prevent returning "elementStart" for Text node. Skip that value and return the next walker step.
|
|
214
|
+
this._position = Position._createBefore(parent);
|
|
215
|
+
return this._previous();
|
|
216
|
+
}
|
|
217
|
+
node = parent.data[position.offset - 1];
|
|
218
|
+
}
|
|
219
|
+
else {
|
|
220
|
+
node = parent.getChild(position.offset - 1);
|
|
221
|
+
}
|
|
222
|
+
if (node instanceof Element) {
|
|
223
|
+
if (this.shallow) {
|
|
224
|
+
position.offset--;
|
|
225
|
+
this._position = position;
|
|
226
|
+
return this._formatReturnValue('elementStart', node, previousPosition, position, 1);
|
|
227
|
+
}
|
|
228
|
+
position = new Position(node, node.childCount);
|
|
229
|
+
this._position = position;
|
|
230
|
+
if (this.ignoreElementEnd) {
|
|
231
|
+
return this._previous();
|
|
232
|
+
}
|
|
233
|
+
return this._formatReturnValue('elementEnd', node, previousPosition, position);
|
|
234
|
+
}
|
|
235
|
+
if (node instanceof Text) {
|
|
236
|
+
if (this.singleCharacters) {
|
|
237
|
+
position = new Position(node, node.data.length);
|
|
238
|
+
this._position = position;
|
|
239
|
+
return this._previous();
|
|
240
|
+
}
|
|
241
|
+
let charactersCount = node.data.length;
|
|
242
|
+
let item;
|
|
243
|
+
// If text stick out of walker range, we need to cut it and wrap in TextProxy.
|
|
244
|
+
if (node == this._boundaryStartParent) {
|
|
245
|
+
const offset = this.boundaries.start.offset;
|
|
246
|
+
item = new TextProxy(node, offset, node.data.length - offset);
|
|
247
|
+
charactersCount = item.data.length;
|
|
248
|
+
position = Position._createBefore(item);
|
|
249
|
+
}
|
|
250
|
+
else {
|
|
251
|
+
item = new TextProxy(node, 0, node.data.length);
|
|
252
|
+
// If not just keep moving backward.
|
|
253
|
+
position.offset--;
|
|
254
|
+
}
|
|
255
|
+
this._position = position;
|
|
256
|
+
return this._formatReturnValue('text', item, previousPosition, position, charactersCount);
|
|
257
|
+
}
|
|
258
|
+
if (typeof node == 'string') {
|
|
259
|
+
let textLength;
|
|
260
|
+
if (!this.singleCharacters) {
|
|
261
|
+
// Check if text stick out of walker range.
|
|
262
|
+
const startOffset = parent === this._boundaryStartParent ? this.boundaries.start.offset : 0;
|
|
263
|
+
textLength = position.offset - startOffset;
|
|
264
|
+
}
|
|
265
|
+
else {
|
|
266
|
+
textLength = 1;
|
|
267
|
+
}
|
|
268
|
+
position.offset -= textLength;
|
|
269
|
+
const textProxy = new TextProxy(parent, position.offset, textLength);
|
|
270
|
+
this._position = position;
|
|
271
|
+
return this._formatReturnValue('text', textProxy, previousPosition, position, textLength);
|
|
272
|
+
}
|
|
273
|
+
// `node` is not set, we reached the beginning of current `parent`.
|
|
274
|
+
position = Position._createBefore(parent);
|
|
275
|
+
this._position = position;
|
|
276
|
+
return this._formatReturnValue('elementStart', parent, previousPosition, position, 1);
|
|
277
|
+
}
|
|
278
|
+
/**
|
|
279
|
+
* Format returned data and adjust `previousPosition` and `nextPosition` if reach the bound of the {@link module:engine/view/text~Text}.
|
|
280
|
+
*
|
|
281
|
+
* @param type Type of step.
|
|
282
|
+
* @param item Item between old and new position.
|
|
283
|
+
* @param previousPosition Previous position of iterator.
|
|
284
|
+
* @param nextPosition Next position of iterator.
|
|
285
|
+
* @param length Length of the item.
|
|
286
|
+
*/
|
|
287
|
+
_formatReturnValue(type, item, previousPosition, nextPosition, length) {
|
|
288
|
+
// Text is a specific parent, because contains string instead of children.
|
|
289
|
+
// Walker doesn't enter to the Text except situations when walker is iterating over every single character,
|
|
290
|
+
// or the bound starts/ends inside the Text. So when the position is at the beginning or at the end of the Text
|
|
291
|
+
// we move it just before or just after Text.
|
|
292
|
+
if (item instanceof TextProxy) {
|
|
293
|
+
// Position is at the end of Text.
|
|
294
|
+
if (item.offsetInText + item.data.length == item.textNode.data.length) {
|
|
295
|
+
if (this.direction == 'forward' && !(this.boundaries && this.boundaries.end.isEqual(this.position))) {
|
|
296
|
+
nextPosition = Position._createAfter(item.textNode);
|
|
297
|
+
// When we change nextPosition of returned value we need also update walker current position.
|
|
298
|
+
this._position = nextPosition;
|
|
299
|
+
}
|
|
300
|
+
else {
|
|
301
|
+
previousPosition = Position._createAfter(item.textNode);
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
// Position is at the begining ot the text.
|
|
305
|
+
if (item.offsetInText === 0) {
|
|
306
|
+
if (this.direction == 'backward' && !(this.boundaries && this.boundaries.start.isEqual(this.position))) {
|
|
307
|
+
nextPosition = Position._createBefore(item.textNode);
|
|
308
|
+
// When we change nextPosition of returned value we need also update walker current position.
|
|
309
|
+
this._position = nextPosition;
|
|
310
|
+
}
|
|
311
|
+
else {
|
|
312
|
+
previousPosition = Position._createBefore(item.textNode);
|
|
313
|
+
}
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
return {
|
|
317
|
+
done: false,
|
|
318
|
+
value: {
|
|
319
|
+
type,
|
|
320
|
+
item,
|
|
321
|
+
previousPosition,
|
|
322
|
+
nextPosition,
|
|
323
|
+
length
|
|
324
|
+
}
|
|
325
|
+
};
|
|
326
|
+
}
|
|
327
|
+
}
|