@ckeditor/ckeditor5-typing 40.0.0 โ†’ 40.1.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.
@@ -1,261 +1,261 @@
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 typing/deleteobserver
7
- */
8
- import { env, keyCodes, isInsideCombinedSymbol, isInsideEmojiSequence, isInsideSurrogatePair } from '@ckeditor/ckeditor5-utils';
9
- import { BubblingEventInfo, DomEventData, Observer } from '@ckeditor/ckeditor5-engine';
10
- const DELETE_CHARACTER = 'character';
11
- const DELETE_WORD = 'word';
12
- const DELETE_CODE_POINT = 'codePoint';
13
- const DELETE_SELECTION = 'selection';
14
- const DELETE_BACKWARD = 'backward';
15
- const DELETE_FORWARD = 'forward';
16
- const DELETE_EVENT_TYPES = {
17
- // --------------------------------------- Backward delete types -----------------------------------------------------
18
- // This happens in Safari on Mac when some content is selected and Ctrl + K is pressed.
19
- deleteContent: {
20
- unit: DELETE_SELECTION,
21
- // According to the Input Events Level 2 spec, this delete type has no direction
22
- // but to keep things simple, let's default to backward.
23
- direction: DELETE_BACKWARD
24
- },
25
- // Chrome and Safari on Mac: Backspace or Ctrl + H
26
- deleteContentBackward: {
27
- // This kind of deletions must be done on the code point-level instead of target range provided by the DOM beforeinput event.
28
- // Take for instance "๐Ÿ‘จโ€๐Ÿ‘ฉโ€๐Ÿ‘งโ€๐Ÿ‘ง", it equals:
29
- //
30
- // * [ "๐Ÿ‘จ", "ZERO WIDTH JOINER", "๐Ÿ‘ฉ", "ZERO WIDTH JOINER", "๐Ÿ‘ง", "ZERO WIDTH JOINER", "๐Ÿ‘ง" ]
31
- // * or simply "\u{1F468}\u200D\u{1F469}\u200D\u{1F467}\u200D\u{1F467}"
32
- //
33
- // The range provided by the browser would cause the entire multi-byte grapheme to disappear while the user
34
- // intention when deleting backwards ("๐Ÿ‘จโ€๐Ÿ‘ฉโ€๐Ÿ‘งโ€๐Ÿ‘ง[]", then backspace) is gradual "decomposition" (first to "๐Ÿ‘จโ€๐Ÿ‘ฉโ€๐Ÿ‘งโ€[]",
35
- // then to "๐Ÿ‘จโ€๐Ÿ‘ฉโ€[]", etc.).
36
- //
37
- // * "๐Ÿ‘จโ€๐Ÿ‘ฉโ€๐Ÿ‘งโ€๐Ÿ‘ง[]" + backward delete (by code point) -> results in "๐Ÿ‘จโ€๐Ÿ‘ฉโ€๐Ÿ‘ง[]", removed the last "๐Ÿ‘ง" ๐Ÿ‘
38
- // * "๐Ÿ‘จโ€๐Ÿ‘ฉโ€๐Ÿ‘งโ€๐Ÿ‘ง[]" + backward delete (by character) -> results in "[]", removed the whole grapheme ๐Ÿ‘Ž
39
- //
40
- // Deleting by code-point is simply a better UX. See "deleteContentForward" to learn more.
41
- unit: DELETE_CODE_POINT,
42
- direction: DELETE_BACKWARD
43
- },
44
- // On Mac: Option + Backspace.
45
- // On iOS: Hold the backspace for a while and the whole words will start to disappear.
46
- deleteWordBackward: {
47
- unit: DELETE_WORD,
48
- direction: DELETE_BACKWARD
49
- },
50
- // Safari on Mac: Cmd + Backspace
51
- deleteHardLineBackward: {
52
- unit: DELETE_SELECTION,
53
- direction: DELETE_BACKWARD
54
- },
55
- // Chrome on Mac: Cmd + Backspace.
56
- deleteSoftLineBackward: {
57
- unit: DELETE_SELECTION,
58
- direction: DELETE_BACKWARD
59
- },
60
- // --------------------------------------- Forward delete types -----------------------------------------------------
61
- // Chrome on Mac: Fn + Backspace or Ctrl + D
62
- // Safari on Mac: Ctrl + K or Ctrl + D
63
- deleteContentForward: {
64
- // Unlike backward delete, this delete must be performed by character instead of by code point, which
65
- // provides the best UX for working with accented letters.
66
- // Take, for example "bฬ‚" ("\u0062\u0302", or [ "LATIN SMALL LETTER B", "COMBINING CIRCUMFLEX ACCENT" ]):
67
- //
68
- // * "bฬ‚[]" + backward delete (by code point) -> results in "b[]", removed the combining mark ๐Ÿ‘
69
- // * "[]bฬ‚" + forward delete (by code point) -> results in "[]^", a bare combining mark does that not make sense when alone ๐Ÿ‘Ž
70
- // * "[]bฬ‚" + forward delete (by character) -> results in "[]", removed both "b" and the combining mark ๐Ÿ‘
71
- //
72
- // See: "deleteContentBackward" to learn more.
73
- unit: DELETE_CHARACTER,
74
- direction: DELETE_FORWARD
75
- },
76
- // On Mac: Fn + Option + Backspace.
77
- deleteWordForward: {
78
- unit: DELETE_WORD,
79
- direction: DELETE_FORWARD
80
- },
81
- // Chrome on Mac: Ctrl + K (you have to disable the Link plugin first, though, because it uses the same keystroke)
82
- // This is weird that it does not work in Safari on Mac despite being listed in the official shortcuts listing
83
- // on Apple's webpage.
84
- deleteHardLineForward: {
85
- unit: DELETE_SELECTION,
86
- direction: DELETE_FORWARD
87
- },
88
- // At this moment there is no known way to trigger this event type but let's keep it for the symmetry with
89
- // deleteSoftLineBackward.
90
- deleteSoftLineForward: {
91
- unit: DELETE_SELECTION,
92
- direction: DELETE_FORWARD
93
- }
94
- };
95
- /**
96
- * Delete observer introduces the {@link module:engine/view/document~Document#event:delete} event.
97
- */
98
- export default class DeleteObserver extends Observer {
99
- /**
100
- * @inheritDoc
101
- */
102
- constructor(view) {
103
- super(view);
104
- const document = view.document;
105
- // It matters how many subsequent deletions were made, e.g. when the backspace key was pressed and held
106
- // by the user for some time. For instance, if such scenario ocurred and the heading the selection was
107
- // anchored to was the only content of the editor, it will not be converted into a paragraph (the user
108
- // wanted to clean it up, not remove it, it's about UX). Check out the DeleteCommand implementation to learn more.
109
- //
110
- // Fun fact: Safari on Mac won't fire beforeinput for backspace in an empty heading (only content).
111
- let sequence = 0;
112
- document.on('keydown', () => {
113
- sequence++;
114
- });
115
- document.on('keyup', () => {
116
- sequence = 0;
117
- });
118
- document.on('beforeinput', (evt, data) => {
119
- if (!this.isEnabled) {
120
- return;
121
- }
122
- const { targetRanges, domEvent, inputType } = data;
123
- const deleteEventSpec = DELETE_EVENT_TYPES[inputType];
124
- if (!deleteEventSpec) {
125
- return;
126
- }
127
- const deleteData = {
128
- direction: deleteEventSpec.direction,
129
- unit: deleteEventSpec.unit,
130
- sequence
131
- };
132
- if (deleteData.unit == DELETE_SELECTION) {
133
- deleteData.selectionToRemove = view.createSelection(targetRanges[0]);
134
- }
135
- // The default deletion unit for deleteContentBackward is a single code point
136
- // but if the browser provides a wider target range then we should use it.
137
- if (inputType === 'deleteContentBackward') {
138
- // On Android, deleteContentBackward has sequence 1 by default.
139
- if (env.isAndroid) {
140
- deleteData.sequence = 1;
141
- }
142
- // The beforeInput event wants more than a single character to be removed.
143
- if (shouldUseTargetRanges(targetRanges)) {
144
- deleteData.unit = DELETE_SELECTION;
145
- deleteData.selectionToRemove = view.createSelection(targetRanges);
146
- }
147
- }
148
- const eventInfo = new BubblingEventInfo(document, 'delete', targetRanges[0]);
149
- document.fire(eventInfo, new DomEventData(view, domEvent, deleteData));
150
- // Stop the beforeinput event if `delete` event was stopped.
151
- // https://github.com/ckeditor/ckeditor5/issues/753
152
- if (eventInfo.stop.called) {
153
- evt.stop();
154
- }
155
- });
156
- // TODO: to be removed when https://bugs.chromium.org/p/chromium/issues/detail?id=1365311 is solved.
157
- if (env.isBlink) {
158
- enableChromeWorkaround(this);
159
- }
160
- }
161
- /**
162
- * @inheritDoc
163
- */
164
- observe() { }
165
- /**
166
- * @inheritDoc
167
- */
168
- stopObserving() { }
169
- }
170
- /**
171
- * Enables workaround for the issue https://github.com/ckeditor/ckeditor5/issues/11904.
172
- */
173
- function enableChromeWorkaround(observer) {
174
- const view = observer.view;
175
- const document = view.document;
176
- let pressedKeyCode = null;
177
- let beforeInputReceived = false;
178
- document.on('keydown', (evt, { keyCode }) => {
179
- pressedKeyCode = keyCode;
180
- beforeInputReceived = false;
181
- });
182
- document.on('keyup', (evt, { keyCode, domEvent }) => {
183
- const selection = document.selection;
184
- const shouldFireDeleteEvent = observer.isEnabled &&
185
- keyCode == pressedKeyCode &&
186
- isDeleteKeyCode(keyCode) &&
187
- !selection.isCollapsed &&
188
- !beforeInputReceived;
189
- pressedKeyCode = null;
190
- if (shouldFireDeleteEvent) {
191
- const targetRange = selection.getFirstRange();
192
- const eventInfo = new BubblingEventInfo(document, 'delete', targetRange);
193
- const deleteData = {
194
- unit: DELETE_SELECTION,
195
- direction: getDeleteDirection(keyCode),
196
- selectionToRemove: selection
197
- };
198
- document.fire(eventInfo, new DomEventData(view, domEvent, deleteData));
199
- }
200
- });
201
- document.on('beforeinput', (evt, { inputType }) => {
202
- const deleteEventSpec = DELETE_EVENT_TYPES[inputType];
203
- const isMatchingBeforeInput = isDeleteKeyCode(pressedKeyCode) &&
204
- deleteEventSpec &&
205
- deleteEventSpec.direction == getDeleteDirection(pressedKeyCode);
206
- if (isMatchingBeforeInput) {
207
- beforeInputReceived = true;
208
- }
209
- }, { priority: 'high' });
210
- document.on('beforeinput', (evt, { inputType, data }) => {
211
- const shouldIgnoreBeforeInput = pressedKeyCode == keyCodes.delete &&
212
- inputType == 'insertText' &&
213
- data == '\x7f'; // Delete character :P
214
- if (shouldIgnoreBeforeInput) {
215
- evt.stop();
216
- }
217
- }, { priority: 'high' });
218
- function isDeleteKeyCode(keyCode) {
219
- return keyCode == keyCodes.backspace || keyCode == keyCodes.delete;
220
- }
221
- function getDeleteDirection(keyCode) {
222
- return keyCode == keyCodes.backspace ? DELETE_BACKWARD : DELETE_FORWARD;
223
- }
224
- }
225
- /**
226
- * Verifies whether the given target ranges cover more than a single character and should be used instead of a single code-point deletion.
227
- */
228
- function shouldUseTargetRanges(targetRanges) {
229
- // The collapsed target range could happen for example while deleting inside an inline filler
230
- // (it's mapped to collapsed position before an inline filler).
231
- if (targetRanges.length != 1 || targetRanges[0].isCollapsed) {
232
- return false;
233
- }
234
- const walker = targetRanges[0].getWalker({
235
- direction: 'backward',
236
- singleCharacters: true,
237
- ignoreElementEnd: true
238
- });
239
- let count = 0;
240
- for (const { nextPosition } of walker) {
241
- // There is some element in the range so count it as a single character.
242
- if (!nextPosition.parent.is('$text')) {
243
- count++;
244
- }
245
- else {
246
- const data = nextPosition.parent.data;
247
- const offset = nextPosition.offset;
248
- // Count combined symbols and emoji sequences as a single character.
249
- if (isInsideSurrogatePair(data, offset) ||
250
- isInsideCombinedSymbol(data, offset) ||
251
- isInsideEmojiSequence(data, offset)) {
252
- continue;
253
- }
254
- count++;
255
- }
256
- if (count > 1) {
257
- return true;
258
- }
259
- }
260
- return false;
261
- }
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 typing/deleteobserver
7
+ */
8
+ import { env, keyCodes, isInsideCombinedSymbol, isInsideEmojiSequence, isInsideSurrogatePair } from '@ckeditor/ckeditor5-utils';
9
+ import { BubblingEventInfo, DomEventData, Observer } from '@ckeditor/ckeditor5-engine';
10
+ const DELETE_CHARACTER = 'character';
11
+ const DELETE_WORD = 'word';
12
+ const DELETE_CODE_POINT = 'codePoint';
13
+ const DELETE_SELECTION = 'selection';
14
+ const DELETE_BACKWARD = 'backward';
15
+ const DELETE_FORWARD = 'forward';
16
+ const DELETE_EVENT_TYPES = {
17
+ // --------------------------------------- Backward delete types -----------------------------------------------------
18
+ // This happens in Safari on Mac when some content is selected and Ctrl + K is pressed.
19
+ deleteContent: {
20
+ unit: DELETE_SELECTION,
21
+ // According to the Input Events Level 2 spec, this delete type has no direction
22
+ // but to keep things simple, let's default to backward.
23
+ direction: DELETE_BACKWARD
24
+ },
25
+ // Chrome and Safari on Mac: Backspace or Ctrl + H
26
+ deleteContentBackward: {
27
+ // This kind of deletions must be done on the code point-level instead of target range provided by the DOM beforeinput event.
28
+ // Take for instance "๐Ÿ‘จโ€๐Ÿ‘ฉโ€๐Ÿ‘งโ€๐Ÿ‘ง", it equals:
29
+ //
30
+ // * [ "๐Ÿ‘จ", "ZERO WIDTH JOINER", "๐Ÿ‘ฉ", "ZERO WIDTH JOINER", "๐Ÿ‘ง", "ZERO WIDTH JOINER", "๐Ÿ‘ง" ]
31
+ // * or simply "\u{1F468}\u200D\u{1F469}\u200D\u{1F467}\u200D\u{1F467}"
32
+ //
33
+ // The range provided by the browser would cause the entire multi-byte grapheme to disappear while the user
34
+ // intention when deleting backwards ("๐Ÿ‘จโ€๐Ÿ‘ฉโ€๐Ÿ‘งโ€๐Ÿ‘ง[]", then backspace) is gradual "decomposition" (first to "๐Ÿ‘จโ€๐Ÿ‘ฉโ€๐Ÿ‘งโ€[]",
35
+ // then to "๐Ÿ‘จโ€๐Ÿ‘ฉโ€[]", etc.).
36
+ //
37
+ // * "๐Ÿ‘จโ€๐Ÿ‘ฉโ€๐Ÿ‘งโ€๐Ÿ‘ง[]" + backward delete (by code point) -> results in "๐Ÿ‘จโ€๐Ÿ‘ฉโ€๐Ÿ‘ง[]", removed the last "๐Ÿ‘ง" ๐Ÿ‘
38
+ // * "๐Ÿ‘จโ€๐Ÿ‘ฉโ€๐Ÿ‘งโ€๐Ÿ‘ง[]" + backward delete (by character) -> results in "[]", removed the whole grapheme ๐Ÿ‘Ž
39
+ //
40
+ // Deleting by code-point is simply a better UX. See "deleteContentForward" to learn more.
41
+ unit: DELETE_CODE_POINT,
42
+ direction: DELETE_BACKWARD
43
+ },
44
+ // On Mac: Option + Backspace.
45
+ // On iOS: Hold the backspace for a while and the whole words will start to disappear.
46
+ deleteWordBackward: {
47
+ unit: DELETE_WORD,
48
+ direction: DELETE_BACKWARD
49
+ },
50
+ // Safari on Mac: Cmd + Backspace
51
+ deleteHardLineBackward: {
52
+ unit: DELETE_SELECTION,
53
+ direction: DELETE_BACKWARD
54
+ },
55
+ // Chrome on Mac: Cmd + Backspace.
56
+ deleteSoftLineBackward: {
57
+ unit: DELETE_SELECTION,
58
+ direction: DELETE_BACKWARD
59
+ },
60
+ // --------------------------------------- Forward delete types -----------------------------------------------------
61
+ // Chrome on Mac: Fn + Backspace or Ctrl + D
62
+ // Safari on Mac: Ctrl + K or Ctrl + D
63
+ deleteContentForward: {
64
+ // Unlike backward delete, this delete must be performed by character instead of by code point, which
65
+ // provides the best UX for working with accented letters.
66
+ // Take, for example "bฬ‚" ("\u0062\u0302", or [ "LATIN SMALL LETTER B", "COMBINING CIRCUMFLEX ACCENT" ]):
67
+ //
68
+ // * "bฬ‚[]" + backward delete (by code point) -> results in "b[]", removed the combining mark ๐Ÿ‘
69
+ // * "[]bฬ‚" + forward delete (by code point) -> results in "[]^", a bare combining mark does that not make sense when alone ๐Ÿ‘Ž
70
+ // * "[]bฬ‚" + forward delete (by character) -> results in "[]", removed both "b" and the combining mark ๐Ÿ‘
71
+ //
72
+ // See: "deleteContentBackward" to learn more.
73
+ unit: DELETE_CHARACTER,
74
+ direction: DELETE_FORWARD
75
+ },
76
+ // On Mac: Fn + Option + Backspace.
77
+ deleteWordForward: {
78
+ unit: DELETE_WORD,
79
+ direction: DELETE_FORWARD
80
+ },
81
+ // Chrome on Mac: Ctrl + K (you have to disable the Link plugin first, though, because it uses the same keystroke)
82
+ // This is weird that it does not work in Safari on Mac despite being listed in the official shortcuts listing
83
+ // on Apple's webpage.
84
+ deleteHardLineForward: {
85
+ unit: DELETE_SELECTION,
86
+ direction: DELETE_FORWARD
87
+ },
88
+ // At this moment there is no known way to trigger this event type but let's keep it for the symmetry with
89
+ // deleteSoftLineBackward.
90
+ deleteSoftLineForward: {
91
+ unit: DELETE_SELECTION,
92
+ direction: DELETE_FORWARD
93
+ }
94
+ };
95
+ /**
96
+ * Delete observer introduces the {@link module:engine/view/document~Document#event:delete} event.
97
+ */
98
+ export default class DeleteObserver extends Observer {
99
+ /**
100
+ * @inheritDoc
101
+ */
102
+ constructor(view) {
103
+ super(view);
104
+ const document = view.document;
105
+ // It matters how many subsequent deletions were made, e.g. when the backspace key was pressed and held
106
+ // by the user for some time. For instance, if such scenario ocurred and the heading the selection was
107
+ // anchored to was the only content of the editor, it will not be converted into a paragraph (the user
108
+ // wanted to clean it up, not remove it, it's about UX). Check out the DeleteCommand implementation to learn more.
109
+ //
110
+ // Fun fact: Safari on Mac won't fire beforeinput for backspace in an empty heading (only content).
111
+ let sequence = 0;
112
+ document.on('keydown', () => {
113
+ sequence++;
114
+ });
115
+ document.on('keyup', () => {
116
+ sequence = 0;
117
+ });
118
+ document.on('beforeinput', (evt, data) => {
119
+ if (!this.isEnabled) {
120
+ return;
121
+ }
122
+ const { targetRanges, domEvent, inputType } = data;
123
+ const deleteEventSpec = DELETE_EVENT_TYPES[inputType];
124
+ if (!deleteEventSpec) {
125
+ return;
126
+ }
127
+ const deleteData = {
128
+ direction: deleteEventSpec.direction,
129
+ unit: deleteEventSpec.unit,
130
+ sequence
131
+ };
132
+ if (deleteData.unit == DELETE_SELECTION) {
133
+ deleteData.selectionToRemove = view.createSelection(targetRanges[0]);
134
+ }
135
+ // The default deletion unit for deleteContentBackward is a single code point
136
+ // but if the browser provides a wider target range then we should use it.
137
+ if (inputType === 'deleteContentBackward') {
138
+ // On Android, deleteContentBackward has sequence 1 by default.
139
+ if (env.isAndroid) {
140
+ deleteData.sequence = 1;
141
+ }
142
+ // The beforeInput event wants more than a single character to be removed.
143
+ if (shouldUseTargetRanges(targetRanges)) {
144
+ deleteData.unit = DELETE_SELECTION;
145
+ deleteData.selectionToRemove = view.createSelection(targetRanges);
146
+ }
147
+ }
148
+ const eventInfo = new BubblingEventInfo(document, 'delete', targetRanges[0]);
149
+ document.fire(eventInfo, new DomEventData(view, domEvent, deleteData));
150
+ // Stop the beforeinput event if `delete` event was stopped.
151
+ // https://github.com/ckeditor/ckeditor5/issues/753
152
+ if (eventInfo.stop.called) {
153
+ evt.stop();
154
+ }
155
+ });
156
+ // TODO: to be removed when https://bugs.chromium.org/p/chromium/issues/detail?id=1365311 is solved.
157
+ if (env.isBlink) {
158
+ enableChromeWorkaround(this);
159
+ }
160
+ }
161
+ /**
162
+ * @inheritDoc
163
+ */
164
+ observe() { }
165
+ /**
166
+ * @inheritDoc
167
+ */
168
+ stopObserving() { }
169
+ }
170
+ /**
171
+ * Enables workaround for the issue https://github.com/ckeditor/ckeditor5/issues/11904.
172
+ */
173
+ function enableChromeWorkaround(observer) {
174
+ const view = observer.view;
175
+ const document = view.document;
176
+ let pressedKeyCode = null;
177
+ let beforeInputReceived = false;
178
+ document.on('keydown', (evt, { keyCode }) => {
179
+ pressedKeyCode = keyCode;
180
+ beforeInputReceived = false;
181
+ });
182
+ document.on('keyup', (evt, { keyCode, domEvent }) => {
183
+ const selection = document.selection;
184
+ const shouldFireDeleteEvent = observer.isEnabled &&
185
+ keyCode == pressedKeyCode &&
186
+ isDeleteKeyCode(keyCode) &&
187
+ !selection.isCollapsed &&
188
+ !beforeInputReceived;
189
+ pressedKeyCode = null;
190
+ if (shouldFireDeleteEvent) {
191
+ const targetRange = selection.getFirstRange();
192
+ const eventInfo = new BubblingEventInfo(document, 'delete', targetRange);
193
+ const deleteData = {
194
+ unit: DELETE_SELECTION,
195
+ direction: getDeleteDirection(keyCode),
196
+ selectionToRemove: selection
197
+ };
198
+ document.fire(eventInfo, new DomEventData(view, domEvent, deleteData));
199
+ }
200
+ });
201
+ document.on('beforeinput', (evt, { inputType }) => {
202
+ const deleteEventSpec = DELETE_EVENT_TYPES[inputType];
203
+ const isMatchingBeforeInput = isDeleteKeyCode(pressedKeyCode) &&
204
+ deleteEventSpec &&
205
+ deleteEventSpec.direction == getDeleteDirection(pressedKeyCode);
206
+ if (isMatchingBeforeInput) {
207
+ beforeInputReceived = true;
208
+ }
209
+ }, { priority: 'high' });
210
+ document.on('beforeinput', (evt, { inputType, data }) => {
211
+ const shouldIgnoreBeforeInput = pressedKeyCode == keyCodes.delete &&
212
+ inputType == 'insertText' &&
213
+ data == '\x7f'; // Delete character :P
214
+ if (shouldIgnoreBeforeInput) {
215
+ evt.stop();
216
+ }
217
+ }, { priority: 'high' });
218
+ function isDeleteKeyCode(keyCode) {
219
+ return keyCode == keyCodes.backspace || keyCode == keyCodes.delete;
220
+ }
221
+ function getDeleteDirection(keyCode) {
222
+ return keyCode == keyCodes.backspace ? DELETE_BACKWARD : DELETE_FORWARD;
223
+ }
224
+ }
225
+ /**
226
+ * Verifies whether the given target ranges cover more than a single character and should be used instead of a single code-point deletion.
227
+ */
228
+ function shouldUseTargetRanges(targetRanges) {
229
+ // The collapsed target range could happen for example while deleting inside an inline filler
230
+ // (it's mapped to collapsed position before an inline filler).
231
+ if (targetRanges.length != 1 || targetRanges[0].isCollapsed) {
232
+ return false;
233
+ }
234
+ const walker = targetRanges[0].getWalker({
235
+ direction: 'backward',
236
+ singleCharacters: true,
237
+ ignoreElementEnd: true
238
+ });
239
+ let count = 0;
240
+ for (const { nextPosition } of walker) {
241
+ // There is some element in the range so count it as a single character.
242
+ if (!nextPosition.parent.is('$text')) {
243
+ count++;
244
+ }
245
+ else {
246
+ const data = nextPosition.parent.data;
247
+ const offset = nextPosition.offset;
248
+ // Count combined symbols and emoji sequences as a single character.
249
+ if (isInsideSurrogatePair(data, offset) ||
250
+ isInsideCombinedSymbol(data, offset) ||
251
+ isInsideEmojiSequence(data, offset)) {
252
+ continue;
253
+ }
254
+ count++;
255
+ }
256
+ if (count > 1) {
257
+ return true;
258
+ }
259
+ }
260
+ return false;
261
+ }
package/src/index.d.ts CHANGED
@@ -1,24 +1,24 @@
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 typing
7
- */
8
- export { default as Typing } from './typing';
9
- export { default as Input } from './input';
10
- export { default as Delete } from './delete';
11
- export { default as TextWatcher } from './textwatcher';
12
- export { default as TwoStepCaretMovement } from './twostepcaretmovement';
13
- export { default as TextTransformation } from './texttransformation';
14
- export { default as inlineHighlight } from './utils/inlinehighlight';
15
- export { default as findAttributeRange, findAttributeRangeBound } from './utils/findattributerange';
16
- export { default as getLastTextLine, type LastTextLineData } from './utils/getlasttextline';
17
- export { default as InsertTextCommand, type InsertTextCommandExecuteEvent } from './inserttextcommand';
18
- export type { default as DeleteCommand } from './deletecommand';
19
- export type { TypingConfig } from './typingconfig';
20
- export type { ViewDocumentDeleteEvent } from './deleteobserver';
21
- export type { ViewDocumentInsertTextEvent, InsertTextEventData } from './inserttextobserver';
22
- export type { TextWatcherMatchedEvent } from './textwatcher';
23
- export type { TextWatcherMatchedDataEvent } from './textwatcher';
24
- import './augmentation';
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 typing
7
+ */
8
+ export { default as Typing } from './typing';
9
+ export { default as Input } from './input';
10
+ export { default as Delete } from './delete';
11
+ export { default as TextWatcher } from './textwatcher';
12
+ export { default as TwoStepCaretMovement } from './twostepcaretmovement';
13
+ export { default as TextTransformation } from './texttransformation';
14
+ export { default as inlineHighlight } from './utils/inlinehighlight';
15
+ export { default as findAttributeRange, findAttributeRangeBound } from './utils/findattributerange';
16
+ export { default as getLastTextLine, type LastTextLineData } from './utils/getlasttextline';
17
+ export { default as InsertTextCommand, type InsertTextCommandExecuteEvent } from './inserttextcommand';
18
+ export type { default as DeleteCommand } from './deletecommand';
19
+ export type { TypingConfig } from './typingconfig';
20
+ export type { ViewDocumentDeleteEvent } from './deleteobserver';
21
+ export type { ViewDocumentInsertTextEvent, InsertTextEventData } from './inserttextobserver';
22
+ export type { TextWatcherMatchedEvent } from './textwatcher';
23
+ export type { TextWatcherMatchedDataEvent } from './textwatcher';
24
+ import './augmentation';
package/src/index.js CHANGED
@@ -1,18 +1,18 @@
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 typing
7
- */
8
- export { default as Typing } from './typing';
9
- export { default as Input } from './input';
10
- export { default as Delete } from './delete';
11
- export { default as TextWatcher } from './textwatcher';
12
- export { default as TwoStepCaretMovement } from './twostepcaretmovement';
13
- export { default as TextTransformation } from './texttransformation';
14
- export { default as inlineHighlight } from './utils/inlinehighlight';
15
- export { default as findAttributeRange, findAttributeRangeBound } from './utils/findattributerange';
16
- export { default as getLastTextLine } from './utils/getlasttextline';
17
- export { default as InsertTextCommand } from './inserttextcommand';
18
- import './augmentation';
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 typing
7
+ */
8
+ export { default as Typing } from './typing';
9
+ export { default as Input } from './input';
10
+ export { default as Delete } from './delete';
11
+ export { default as TextWatcher } from './textwatcher';
12
+ export { default as TwoStepCaretMovement } from './twostepcaretmovement';
13
+ export { default as TextTransformation } from './texttransformation';
14
+ export { default as inlineHighlight } from './utils/inlinehighlight';
15
+ export { default as findAttributeRange, findAttributeRangeBound } from './utils/findattributerange';
16
+ export { default as getLastTextLine } from './utils/getlasttextline';
17
+ export { default as InsertTextCommand } from './inserttextcommand';
18
+ import './augmentation';