@ckeditor/ckeditor5-table 36.0.1 → 37.0.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.
Files changed (170) hide show
  1. package/README.md +1 -1
  2. package/build/table.js +1 -1
  3. package/ckeditor5-metadata.json +6 -6
  4. package/package.json +34 -29
  5. package/src/augmentation.d.ts +76 -0
  6. package/src/augmentation.js +5 -0
  7. package/src/commands/insertcolumncommand.d.ts +55 -0
  8. package/src/commands/insertcolumncommand.js +45 -60
  9. package/src/commands/insertrowcommand.d.ts +54 -0
  10. package/src/commands/insertrowcommand.js +44 -59
  11. package/src/commands/inserttablecommand.d.ts +44 -0
  12. package/src/commands/inserttablecommand.js +50 -68
  13. package/src/commands/mergecellcommand.d.ts +68 -0
  14. package/src/commands/mergecellcommand.js +169 -244
  15. package/src/commands/mergecellscommand.d.ts +28 -0
  16. package/src/commands/mergecellscommand.js +72 -101
  17. package/src/commands/removecolumncommand.d.ts +29 -0
  18. package/src/commands/removecolumncommand.js +88 -102
  19. package/src/commands/removerowcommand.d.ts +29 -0
  20. package/src/commands/removerowcommand.js +63 -80
  21. package/src/commands/selectcolumncommand.d.ts +33 -0
  22. package/src/commands/selectcolumncommand.js +41 -54
  23. package/src/commands/selectrowcommand.d.ts +33 -0
  24. package/src/commands/selectrowcommand.js +38 -48
  25. package/src/commands/setheadercolumncommand.d.ts +50 -0
  26. package/src/commands/setheadercolumncommand.js +48 -73
  27. package/src/commands/setheaderrowcommand.d.ts +53 -0
  28. package/src/commands/setheaderrowcommand.js +56 -85
  29. package/src/commands/splitcellcommand.d.ts +43 -0
  30. package/src/commands/splitcellcommand.js +35 -49
  31. package/src/converters/downcast.d.ts +63 -0
  32. package/src/converters/downcast.js +98 -130
  33. package/src/converters/table-caption-post-fixer.d.ts +20 -0
  34. package/src/converters/table-caption-post-fixer.js +36 -52
  35. package/src/converters/table-cell-paragraph-post-fixer.d.ts +32 -0
  36. package/src/converters/table-cell-paragraph-post-fixer.js +88 -119
  37. package/src/converters/table-cell-refresh-handler.d.ts +18 -0
  38. package/src/converters/table-cell-refresh-handler.js +29 -48
  39. package/src/converters/table-headings-refresh-handler.d.ts +17 -0
  40. package/src/converters/table-headings-refresh-handler.js +35 -54
  41. package/src/converters/table-layout-post-fixer.d.ts +226 -0
  42. package/src/converters/table-layout-post-fixer.js +276 -313
  43. package/src/converters/tableproperties.d.ts +54 -0
  44. package/src/converters/tableproperties.js +136 -168
  45. package/src/converters/upcasttable.d.ts +49 -0
  46. package/src/converters/upcasttable.js +199 -251
  47. package/src/index.d.ts +60 -0
  48. package/src/index.js +1 -2
  49. package/src/plaintableoutput.d.ts +25 -0
  50. package/src/plaintableoutput.js +107 -135
  51. package/src/table.d.ts +33 -0
  52. package/src/table.js +12 -88
  53. package/src/tablecaption/tablecaptionediting.d.ts +63 -0
  54. package/src/tablecaption/tablecaptionediting.js +104 -135
  55. package/src/tablecaption/tablecaptionui.d.ts +21 -0
  56. package/src/tablecaption/tablecaptionui.js +42 -58
  57. package/src/tablecaption/toggletablecaptioncommand.d.ts +68 -0
  58. package/src/tablecaption/toggletablecaptioncommand.js +77 -92
  59. package/src/tablecaption/utils.d.ts +42 -0
  60. package/src/tablecaption/utils.js +35 -61
  61. package/src/tablecaption.d.ts +22 -0
  62. package/src/tablecaption.js +12 -19
  63. package/src/tablecellproperties/commands/tablecellbackgroundcolorcommand.d.ts +32 -0
  64. package/src/tablecellproperties/commands/tablecellbackgroundcolorcommand.js +14 -20
  65. package/src/tablecellproperties/commands/tablecellbordercolorcommand.d.ts +37 -0
  66. package/src/tablecellproperties/commands/tablecellbordercolorcommand.js +27 -37
  67. package/src/tablecellproperties/commands/tablecellborderstylecommand.d.ts +37 -0
  68. package/src/tablecellproperties/commands/tablecellborderstylecommand.js +27 -37
  69. package/src/tablecellproperties/commands/tablecellborderwidthcommand.d.ts +51 -0
  70. package/src/tablecellproperties/commands/tablecellborderwidthcommand.js +42 -53
  71. package/src/tablecellproperties/commands/tablecellheightcommand.d.ts +46 -0
  72. package/src/tablecellproperties/commands/tablecellheightcommand.js +29 -36
  73. package/src/tablecellproperties/commands/tablecellhorizontalalignmentcommand.d.ts +32 -0
  74. package/src/tablecellproperties/commands/tablecellhorizontalalignmentcommand.js +14 -20
  75. package/src/tablecellproperties/commands/tablecellpaddingcommand.d.ts +51 -0
  76. package/src/tablecellproperties/commands/tablecellpaddingcommand.js +42 -53
  77. package/src/tablecellproperties/commands/tablecellpropertycommand.d.ts +62 -0
  78. package/src/tablecellproperties/commands/tablecellpropertycommand.js +77 -122
  79. package/src/tablecellproperties/commands/tablecellverticalalignmentcommand.d.ts +40 -0
  80. package/src/tablecellproperties/commands/tablecellverticalalignmentcommand.js +14 -20
  81. package/src/tablecellproperties/tablecellpropertiesediting.d.ts +41 -0
  82. package/src/tablecellproperties/tablecellpropertiesediting.js +193 -236
  83. package/src/tablecellproperties/tablecellpropertiesui.d.ts +111 -0
  84. package/src/tablecellproperties/tablecellpropertiesui.js +302 -456
  85. package/src/tablecellproperties/ui/tablecellpropertiesview.d.ts +227 -0
  86. package/src/tablecellproperties/ui/tablecellpropertiesview.js +509 -844
  87. package/src/tablecellproperties.d.ts +28 -0
  88. package/src/tablecellproperties.js +12 -98
  89. package/src/tablecellwidth/commands/tablecellwidthcommand.d.ts +46 -0
  90. package/src/tablecellwidth/commands/tablecellwidthcommand.js +29 -35
  91. package/src/tablecellwidth/tablecellwidthediting.d.ts +28 -0
  92. package/src/tablecellwidth/tablecellwidthediting.js +25 -38
  93. package/src/tableclipboard.d.ts +63 -0
  94. package/src/tableclipboard.js +429 -568
  95. package/src/tablecolumnresize/constants.d.ts +20 -0
  96. package/src/tablecolumnresize/constants.js +0 -10
  97. package/src/tablecolumnresize/converters.d.ts +18 -0
  98. package/src/tablecolumnresize/converters.js +35 -119
  99. package/src/tablecolumnresize/tablecolumnresizeediting.d.ts +137 -0
  100. package/src/tablecolumnresize/tablecolumnresizeediting.js +545 -711
  101. package/src/tablecolumnresize/tablewidthscommand.d.ts +38 -0
  102. package/src/tablecolumnresize/tablewidthscommand.js +61 -0
  103. package/src/tablecolumnresize/utils.d.ts +141 -0
  104. package/src/tablecolumnresize/utils.js +256 -233
  105. package/src/tablecolumnresize.d.ts +24 -0
  106. package/src/tablecolumnresize.js +12 -19
  107. package/src/tableconfig.d.ts +331 -0
  108. package/src/tableconfig.js +5 -0
  109. package/src/tableediting.d.ts +97 -0
  110. package/src/tableediting.js +157 -176
  111. package/src/tablekeyboard.d.ts +63 -0
  112. package/src/tablekeyboard.js +261 -344
  113. package/src/tablemouse/mouseeventsobserver.d.ts +62 -0
  114. package/src/tablemouse/mouseeventsobserver.js +13 -50
  115. package/src/tablemouse.d.ts +46 -0
  116. package/src/tablemouse.js +154 -202
  117. package/src/tableproperties/commands/tablealignmentcommand.d.ts +32 -0
  118. package/src/tableproperties/commands/tablealignmentcommand.js +14 -20
  119. package/src/tableproperties/commands/tablebackgroundcolorcommand.d.ts +32 -0
  120. package/src/tableproperties/commands/tablebackgroundcolorcommand.js +14 -20
  121. package/src/tableproperties/commands/tablebordercolorcommand.d.ts +37 -0
  122. package/src/tableproperties/commands/tablebordercolorcommand.js +27 -37
  123. package/src/tableproperties/commands/tableborderstylecommand.d.ts +37 -0
  124. package/src/tableproperties/commands/tableborderstylecommand.js +27 -37
  125. package/src/tableproperties/commands/tableborderwidthcommand.d.ts +51 -0
  126. package/src/tableproperties/commands/tableborderwidthcommand.js +42 -53
  127. package/src/tableproperties/commands/tableheightcommand.d.ts +46 -0
  128. package/src/tableproperties/commands/tableheightcommand.js +29 -33
  129. package/src/tableproperties/commands/tablepropertycommand.d.ts +61 -0
  130. package/src/tableproperties/commands/tablepropertycommand.js +68 -112
  131. package/src/tableproperties/commands/tablewidthcommand.d.ts +46 -0
  132. package/src/tableproperties/commands/tablewidthcommand.js +29 -33
  133. package/src/tableproperties/tablepropertiesediting.d.ts +39 -0
  134. package/src/tableproperties/tablepropertiesediting.js +163 -210
  135. package/src/tableproperties/tablepropertiesui.d.ts +113 -0
  136. package/src/tableproperties/tablepropertiesui.js +293 -439
  137. package/src/tableproperties/ui/tablepropertiesview.d.ts +203 -0
  138. package/src/tableproperties/ui/tablepropertiesview.js +427 -718
  139. package/src/tableproperties.d.ts +28 -0
  140. package/src/tableproperties.js +12 -95
  141. package/src/tableselection.d.ts +106 -0
  142. package/src/tableselection.js +279 -376
  143. package/src/tabletoolbar.d.ts +31 -0
  144. package/src/tabletoolbar.js +38 -92
  145. package/src/tableui.d.ts +53 -0
  146. package/src/tableui.js +281 -338
  147. package/src/tableutils.d.ts +448 -0
  148. package/src/tableutils.js +1015 -1229
  149. package/src/tablewalker.d.ts +323 -0
  150. package/src/tablewalker.js +308 -548
  151. package/src/ui/colorinputview.d.ts +143 -0
  152. package/src/ui/colorinputview.js +229 -366
  153. package/src/ui/formrowview.d.ts +61 -0
  154. package/src/ui/formrowview.js +38 -84
  155. package/src/ui/inserttableview.d.ts +77 -0
  156. package/src/ui/inserttableview.js +152 -242
  157. package/src/utils/common.d.ts +42 -0
  158. package/src/utils/common.js +33 -57
  159. package/src/utils/structure.d.ts +245 -0
  160. package/src/utils/structure.js +261 -379
  161. package/src/utils/table-properties.d.ts +67 -0
  162. package/src/utils/table-properties.js +60 -81
  163. package/src/utils/ui/contextualballoon.d.ts +34 -0
  164. package/src/utils/ui/contextualballoon.js +70 -89
  165. package/src/utils/ui/table-properties.d.ts +193 -0
  166. package/src/utils/ui/table-properties.js +259 -319
  167. package/src/utils/ui/widget.d.ts +16 -0
  168. package/src/utils/ui/widget.js +24 -46
  169. package/src/tablecolumnresize/tablecolumnwidthscommand.js +0 -55
  170. package/src/tablecolumnresize/tablewidthresizecommand.js +0 -65
@@ -2,16 +2,10 @@
2
2
  * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.
3
3
  * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
4
4
  */
5
-
6
- /**
7
- * @module table/commands/mergecellcommand
8
- */
9
-
10
5
  import { Command } from 'ckeditor5/src/core';
11
6
  import TableWalker from '../tablewalker';
12
7
  import { isHeadingColumnCell } from '../utils/common';
13
8
  import { removeEmptyRowsColumns } from '../utils/structure';
14
-
15
9
  /**
16
10
  * The merge cell command.
17
11
  *
@@ -22,252 +16,183 @@ import { removeEmptyRowsColumns } from '../utils/structure';
22
16
  *
23
17
  * For example, to merge with a cell to the right:
24
18
  *
25
- * editor.execute( 'mergeTableCellRight' );
19
+ * ```ts
20
+ * editor.execute( 'mergeTableCellRight' );
21
+ * ```
26
22
  *
27
23
  * **Note**: If a table cell has a different [`rowspan`](https://www.w3.org/TR/html50/tabular-data.html#attr-tdth-rowspan)
28
24
  * (for `'mergeTableCellRight'` and `'mergeTableCellLeft'`) or [`colspan`](https://www.w3.org/TR/html50/tabular-data.html#attr-tdth-colspan)
29
25
  * (for `'mergeTableCellUp'` and `'mergeTableCellDown'`), the command will be disabled.
30
- *
31
- * @extends module:core/command~Command
32
26
  */
33
27
  export default class MergeCellCommand extends Command {
34
- /**
35
- * Creates a new `MergeCellCommand` instance.
36
- *
37
- * @param {module:core/editor/editor~Editor} editor The editor on which this command will be used.
38
- * @param {Object} options
39
- * @param {String} options.direction Indicates which cell to merge with the currently selected one.
40
- * Possible values are: `'left'`, `'right'`, `'up'` and `'down'`.
41
- */
42
- constructor( editor, options ) {
43
- super( editor );
44
-
45
- /**
46
- * The direction that indicates which cell will be merged with the currently selected one.
47
- *
48
- * @readonly
49
- * @member {String} #direction
50
- */
51
- this.direction = options.direction;
52
-
53
- /**
54
- * Whether the merge is horizontal (left/right) or vertical (up/down).
55
- *
56
- * @readonly
57
- * @member {Boolean} #isHorizontal
58
- */
59
- this.isHorizontal = this.direction == 'right' || this.direction == 'left';
60
- }
61
-
62
- /**
63
- * @inheritDoc
64
- */
65
- refresh() {
66
- const cellToMerge = this._getMergeableCell();
67
-
68
- this.value = cellToMerge;
69
- this.isEnabled = !!cellToMerge;
70
- }
71
-
72
- /**
73
- * Executes the command.
74
- *
75
- * Depending on the command's {@link #direction} value, it will merge the cell that is to the `'left'`, `'right'`, `'up'` or `'down'`.
76
- *
77
- * @fires execute
78
- */
79
- execute() {
80
- const model = this.editor.model;
81
- const doc = model.document;
82
- const tableUtils = this.editor.plugins.get( 'TableUtils' );
83
- const tableCell = tableUtils.getTableCellsContainingSelection( doc.selection )[ 0 ];
84
-
85
- const cellToMerge = this.value;
86
- const direction = this.direction;
87
-
88
- model.change( writer => {
89
- const isMergeNext = direction == 'right' || direction == 'down';
90
-
91
- // The merge mechanism is always the same so sort cells to be merged.
92
- const cellToExpand = isMergeNext ? tableCell : cellToMerge;
93
- const cellToRemove = isMergeNext ? cellToMerge : tableCell;
94
-
95
- // Cache the parent of cell to remove for later check.
96
- const removedTableCellRow = cellToRemove.parent;
97
-
98
- mergeTableCells( cellToRemove, cellToExpand, writer );
99
-
100
- const spanAttribute = this.isHorizontal ? 'colspan' : 'rowspan';
101
- const cellSpan = parseInt( tableCell.getAttribute( spanAttribute ) || 1 );
102
- const cellToMergeSpan = parseInt( cellToMerge.getAttribute( spanAttribute ) || 1 );
103
-
104
- // Update table cell span attribute and merge set selection on merged contents.
105
- writer.setAttribute( spanAttribute, cellSpan + cellToMergeSpan, cellToExpand );
106
- writer.setSelection( writer.createRangeIn( cellToExpand ) );
107
-
108
- const tableUtils = this.editor.plugins.get( 'TableUtils' );
109
- const table = removedTableCellRow.findAncestor( 'table' );
110
-
111
- // Remove empty rows and columns after merging.
112
- removeEmptyRowsColumns( table, tableUtils );
113
- } );
114
- }
115
-
116
- /**
117
- * Returns a cell that can be merged with the current cell depending on the command's direction.
118
- *
119
- * @returns {module:engine/model/element~Element|undefined}
120
- * @private
121
- */
122
- _getMergeableCell() {
123
- const model = this.editor.model;
124
- const doc = model.document;
125
- const tableUtils = this.editor.plugins.get( 'TableUtils' );
126
- const tableCell = tableUtils.getTableCellsContainingSelection( doc.selection )[ 0 ];
127
-
128
- if ( !tableCell ) {
129
- return;
130
- }
131
-
132
- // First get the cell on proper direction.
133
- const cellToMerge = this.isHorizontal ?
134
- getHorizontalCell( tableCell, this.direction, tableUtils ) :
135
- getVerticalCell( tableCell, this.direction, tableUtils );
136
-
137
- if ( !cellToMerge ) {
138
- return;
139
- }
140
-
141
- // If found check if the span perpendicular to merge direction is equal on both cells.
142
- const spanAttribute = this.isHorizontal ? 'rowspan' : 'colspan';
143
- const span = parseInt( tableCell.getAttribute( spanAttribute ) || 1 );
144
-
145
- const cellToMergeSpan = parseInt( cellToMerge.getAttribute( spanAttribute ) || 1 );
146
-
147
- if ( cellToMergeSpan === span ) {
148
- return cellToMerge;
149
- }
150
- }
28
+ /**
29
+ * Creates a new `MergeCellCommand` instance.
30
+ *
31
+ * @param editor The editor on which this command will be used.
32
+ * @param options.direction Indicates which cell to merge with the currently selected one.
33
+ * Possible values are: `'left'`, `'right'`, `'up'` and `'down'`.
34
+ */
35
+ constructor(editor, options) {
36
+ super(editor);
37
+ this.direction = options.direction;
38
+ this.isHorizontal = this.direction == 'right' || this.direction == 'left';
39
+ }
40
+ /**
41
+ * @inheritDoc
42
+ */
43
+ refresh() {
44
+ const cellToMerge = this._getMergeableCell();
45
+ this.value = cellToMerge;
46
+ this.isEnabled = !!cellToMerge;
47
+ }
48
+ /**
49
+ * Executes the command.
50
+ *
51
+ * Depending on the command's {@link #direction} value, it will merge the cell that is to the `'left'`, `'right'`, `'up'` or `'down'`.
52
+ *
53
+ * @fires execute
54
+ */
55
+ execute() {
56
+ const model = this.editor.model;
57
+ const doc = model.document;
58
+ const tableUtils = this.editor.plugins.get('TableUtils');
59
+ const tableCell = tableUtils.getTableCellsContainingSelection(doc.selection)[0];
60
+ const cellToMerge = this.value;
61
+ const direction = this.direction;
62
+ model.change(writer => {
63
+ const isMergeNext = direction == 'right' || direction == 'down';
64
+ // The merge mechanism is always the same so sort cells to be merged.
65
+ const cellToExpand = (isMergeNext ? tableCell : cellToMerge);
66
+ const cellToRemove = (isMergeNext ? cellToMerge : tableCell);
67
+ // Cache the parent of cell to remove for later check.
68
+ const removedTableCellRow = cellToRemove.parent;
69
+ mergeTableCells(cellToRemove, cellToExpand, writer);
70
+ const spanAttribute = this.isHorizontal ? 'colspan' : 'rowspan';
71
+ const cellSpan = parseInt(tableCell.getAttribute(spanAttribute) || '1');
72
+ const cellToMergeSpan = parseInt(cellToMerge.getAttribute(spanAttribute) || '1');
73
+ // Update table cell span attribute and merge set selection on merged contents.
74
+ writer.setAttribute(spanAttribute, cellSpan + cellToMergeSpan, cellToExpand);
75
+ writer.setSelection(writer.createRangeIn(cellToExpand));
76
+ const tableUtils = this.editor.plugins.get('TableUtils');
77
+ const table = removedTableCellRow.findAncestor('table');
78
+ // Remove empty rows and columns after merging.
79
+ removeEmptyRowsColumns(table, tableUtils);
80
+ });
81
+ }
82
+ /**
83
+ * Returns a cell that can be merged with the current cell depending on the command's direction.
84
+ */
85
+ _getMergeableCell() {
86
+ const model = this.editor.model;
87
+ const doc = model.document;
88
+ const tableUtils = this.editor.plugins.get('TableUtils');
89
+ const tableCell = tableUtils.getTableCellsContainingSelection(doc.selection)[0];
90
+ if (!tableCell) {
91
+ return;
92
+ }
93
+ // First get the cell on proper direction.
94
+ const cellToMerge = this.isHorizontal ?
95
+ getHorizontalCell(tableCell, this.direction, tableUtils) :
96
+ getVerticalCell(tableCell, this.direction, tableUtils);
97
+ if (!cellToMerge) {
98
+ return;
99
+ }
100
+ // If found check if the span perpendicular to merge direction is equal on both cells.
101
+ const spanAttribute = this.isHorizontal ? 'rowspan' : 'colspan';
102
+ const span = parseInt(tableCell.getAttribute(spanAttribute) || '1');
103
+ const cellToMergeSpan = parseInt(cellToMerge.getAttribute(spanAttribute) || '1');
104
+ if (cellToMergeSpan === span) {
105
+ return cellToMerge;
106
+ }
107
+ }
151
108
  }
152
-
153
- // Returns the cell that can be merged horizontally.
154
- //
155
- // @param {module:engine/model/element~Element} tableCell
156
- // @param {String} direction
157
- // @param {module:table/tableutils~TableUtils} tableUtils
158
- // @returns {module:engine/model/node~Node|null}
159
- function getHorizontalCell( tableCell, direction, tableUtils ) {
160
- const tableRow = tableCell.parent;
161
- const table = tableRow.parent;
162
- const horizontalCell = direction == 'right' ? tableCell.nextSibling : tableCell.previousSibling;
163
- const hasHeadingColumns = ( table.getAttribute( 'headingColumns' ) || 0 ) > 0;
164
-
165
- if ( !horizontalCell ) {
166
- return;
167
- }
168
-
169
- // Sort cells:
170
- const cellOnLeft = direction == 'right' ? tableCell : horizontalCell;
171
- const cellOnRight = direction == 'right' ? horizontalCell : tableCell;
172
-
173
- // Get their column indexes:
174
- const { column: leftCellColumn } = tableUtils.getCellLocation( cellOnLeft );
175
- const { column: rightCellColumn } = tableUtils.getCellLocation( cellOnRight );
176
-
177
- const leftCellSpan = parseInt( cellOnLeft.getAttribute( 'colspan' ) || 1 );
178
-
179
- const isCellOnLeftInHeadingColumn = isHeadingColumnCell( tableUtils, cellOnLeft, table );
180
- const isCellOnRightInHeadingColumn = isHeadingColumnCell( tableUtils, cellOnRight, table );
181
-
182
- // We cannot merge heading columns cells with regular cells.
183
- if ( hasHeadingColumns && isCellOnLeftInHeadingColumn != isCellOnRightInHeadingColumn ) {
184
- return;
185
- }
186
-
187
- // The cell on the right must have index that is distant to the cell on the left by the left cell's width (colspan).
188
- const cellsAreTouching = leftCellColumn + leftCellSpan === rightCellColumn;
189
-
190
- // If the right cell's column index is different it means that there are rowspanned cells between them.
191
- return cellsAreTouching ? horizontalCell : undefined;
109
+ /**
110
+ * Returns the cell that can be merged horizontally.
111
+ */
112
+ function getHorizontalCell(tableCell, direction, tableUtils) {
113
+ const tableRow = tableCell.parent;
114
+ const table = tableRow.parent;
115
+ const horizontalCell = direction == 'right' ? tableCell.nextSibling : tableCell.previousSibling;
116
+ const hasHeadingColumns = (table.getAttribute('headingColumns') || 0) > 0;
117
+ if (!horizontalCell) {
118
+ return;
119
+ }
120
+ // Sort cells:
121
+ const cellOnLeft = (direction == 'right' ? tableCell : horizontalCell);
122
+ const cellOnRight = (direction == 'right' ? horizontalCell : tableCell);
123
+ // Get their column indexes:
124
+ const { column: leftCellColumn } = tableUtils.getCellLocation(cellOnLeft);
125
+ const { column: rightCellColumn } = tableUtils.getCellLocation(cellOnRight);
126
+ const leftCellSpan = parseInt(cellOnLeft.getAttribute('colspan') || '1');
127
+ const isCellOnLeftInHeadingColumn = isHeadingColumnCell(tableUtils, cellOnLeft);
128
+ const isCellOnRightInHeadingColumn = isHeadingColumnCell(tableUtils, cellOnRight);
129
+ // We cannot merge heading columns cells with regular cells.
130
+ if (hasHeadingColumns && isCellOnLeftInHeadingColumn != isCellOnRightInHeadingColumn) {
131
+ return;
132
+ }
133
+ // The cell on the right must have index that is distant to the cell on the left by the left cell's width (colspan).
134
+ const cellsAreTouching = leftCellColumn + leftCellSpan === rightCellColumn;
135
+ // If the right cell's column index is different it means that there are rowspanned cells between them.
136
+ return cellsAreTouching ? horizontalCell : undefined;
192
137
  }
193
-
194
- // Returns the cell that can be merged vertically.
195
- //
196
- // @param {module:engine/model/element~Element} tableCell
197
- // @param {String} direction
198
- // @param {module:table/tableutils~TableUtils} tableUtils
199
- // @returns {module:engine/model/node~Node|null}
200
- function getVerticalCell( tableCell, direction, tableUtils ) {
201
- const tableRow = tableCell.parent;
202
- const table = tableRow.parent;
203
-
204
- const rowIndex = table.getChildIndex( tableRow );
205
-
206
- // Don't search for mergeable cell if direction points out of the table.
207
- if ( ( direction == 'down' && rowIndex === tableUtils.getRows( table ) - 1 ) || ( direction == 'up' && rowIndex === 0 ) ) {
208
- return;
209
- }
210
-
211
- const rowspan = parseInt( tableCell.getAttribute( 'rowspan' ) || 1 );
212
- const headingRows = table.getAttribute( 'headingRows' ) || 0;
213
-
214
- const isMergeWithBodyCell = direction == 'down' && ( rowIndex + rowspan ) === headingRows;
215
- const isMergeWithHeadCell = direction == 'up' && rowIndex === headingRows;
216
-
217
- // Don't search for mergeable cell if direction points out of the current table section.
218
- if ( headingRows && ( isMergeWithBodyCell || isMergeWithHeadCell ) ) {
219
- return;
220
- }
221
-
222
- const currentCellRowSpan = parseInt( tableCell.getAttribute( 'rowspan' ) || 1 );
223
- const rowOfCellToMerge = direction == 'down' ? rowIndex + currentCellRowSpan : rowIndex;
224
-
225
- const tableMap = [ ...new TableWalker( table, { endRow: rowOfCellToMerge } ) ];
226
-
227
- const currentCellData = tableMap.find( value => value.cell === tableCell );
228
- const mergeColumn = currentCellData.column;
229
-
230
- const cellToMergeData = tableMap.find( ( { row, cellHeight, column } ) => {
231
- if ( column !== mergeColumn ) {
232
- return false;
233
- }
234
-
235
- if ( direction == 'down' ) {
236
- // If merging a cell below the mergeRow is already calculated.
237
- return row === rowOfCellToMerge;
238
- } else {
239
- // If merging a cell above calculate if it spans to mergeRow.
240
- return rowOfCellToMerge === row + cellHeight;
241
- }
242
- } );
243
-
244
- return cellToMergeData && cellToMergeData.cell;
138
+ /**
139
+ * Returns the cell that can be merged vertically.
140
+ */
141
+ function getVerticalCell(tableCell, direction, tableUtils) {
142
+ const tableRow = tableCell.parent;
143
+ const table = tableRow.parent;
144
+ const rowIndex = table.getChildIndex(tableRow);
145
+ // Don't search for mergeable cell if direction points out of the table.
146
+ if ((direction == 'down' && rowIndex === tableUtils.getRows(table) - 1) || (direction == 'up' && rowIndex === 0)) {
147
+ return null;
148
+ }
149
+ const rowspan = parseInt(tableCell.getAttribute('rowspan') || '1');
150
+ const headingRows = table.getAttribute('headingRows') || 0;
151
+ const isMergeWithBodyCell = direction == 'down' && (rowIndex + rowspan) === headingRows;
152
+ const isMergeWithHeadCell = direction == 'up' && rowIndex === headingRows;
153
+ // Don't search for mergeable cell if direction points out of the current table section.
154
+ if (headingRows && (isMergeWithBodyCell || isMergeWithHeadCell)) {
155
+ return null;
156
+ }
157
+ const currentCellRowSpan = parseInt(tableCell.getAttribute('rowspan') || '1');
158
+ const rowOfCellToMerge = direction == 'down' ? rowIndex + currentCellRowSpan : rowIndex;
159
+ const tableMap = [...new TableWalker(table, { endRow: rowOfCellToMerge })];
160
+ const currentCellData = tableMap.find(value => value.cell === tableCell);
161
+ const mergeColumn = currentCellData.column;
162
+ const cellToMergeData = tableMap.find(({ row, cellHeight, column }) => {
163
+ if (column !== mergeColumn) {
164
+ return false;
165
+ }
166
+ if (direction == 'down') {
167
+ // If merging a cell below the mergeRow is already calculated.
168
+ return row === rowOfCellToMerge;
169
+ }
170
+ else {
171
+ // If merging a cell above calculate if it spans to mergeRow.
172
+ return rowOfCellToMerge === row + cellHeight;
173
+ }
174
+ });
175
+ return cellToMergeData && cellToMergeData.cell ? cellToMergeData.cell : null;
245
176
  }
246
-
247
- // Merges two table cells. It will ensure that after merging cells with an empty paragraph, the resulting table cell will only have one
248
- // paragraph. If one of the merged table cells is empty, the merged table cell will have the contents of the non-empty table cell.
249
- // If both are empty, the merged table cell will have only one empty paragraph.
250
- //
251
- // @param {module:engine/model/element~Element} cellToRemove
252
- // @param {module:engine/model/element~Element} cellToExpand
253
- // @param {module:engine/model/writer~Writer} writer
254
- function mergeTableCells( cellToRemove, cellToExpand, writer ) {
255
- if ( !isEmpty( cellToRemove ) ) {
256
- if ( isEmpty( cellToExpand ) ) {
257
- writer.remove( writer.createRangeIn( cellToExpand ) );
258
- }
259
-
260
- writer.move( writer.createRangeIn( cellToRemove ), writer.createPositionAt( cellToExpand, 'end' ) );
261
- }
262
-
263
- // Remove merged table cell.
264
- writer.remove( cellToRemove );
177
+ /**
178
+ * Merges two table cells. It will ensure that after merging cells with an empty paragraph, the resulting table cell will only have one
179
+ * paragraph. If one of the merged table cells is empty, the merged table cell will have the contents of the non-empty table cell.
180
+ * If both are empty, the merged table cell will have only one empty paragraph.
181
+ */
182
+ function mergeTableCells(cellToRemove, cellToExpand, writer) {
183
+ if (!isEmpty(cellToRemove)) {
184
+ if (isEmpty(cellToExpand)) {
185
+ writer.remove(writer.createRangeIn(cellToExpand));
186
+ }
187
+ writer.move(writer.createRangeIn(cellToRemove), writer.createPositionAt(cellToExpand, 'end'));
188
+ }
189
+ // Remove merged table cell.
190
+ writer.remove(cellToRemove);
265
191
  }
266
-
267
- // Checks if the passed table cell contains an empty paragraph.
268
- //
269
- // @param {module:engine/model/element~Element} tableCell
270
- // @returns {Boolean}
271
- function isEmpty( tableCell ) {
272
- return tableCell.childCount == 1 && tableCell.getChild( 0 ).is( 'element', 'paragraph' ) && tableCell.getChild( 0 ).isEmpty;
192
+ /**
193
+ * Checks if the passed table cell contains an empty paragraph.
194
+ */
195
+ function isEmpty(tableCell) {
196
+ const firstTableChild = tableCell.getChild(0);
197
+ return tableCell.childCount == 1 && firstTableChild.is('element', 'paragraph') && firstTableChild.isEmpty;
273
198
  }
@@ -0,0 +1,28 @@
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 { Command } from 'ckeditor5/src/core';
6
+ /**
7
+ * The merge cells command.
8
+ *
9
+ * The command is registered by {@link module:table/tableediting~TableEditing} as the `'mergeTableCells'` editor command.
10
+ *
11
+ * For example, to merge selected table cells:
12
+ *
13
+ * ```ts
14
+ * editor.execute( 'mergeTableCells' );
15
+ * ```
16
+ */
17
+ export default class MergeCellsCommand extends Command {
18
+ /**
19
+ * @inheritDoc
20
+ */
21
+ refresh(): void;
22
+ /**
23
+ * Executes the command.
24
+ *
25
+ * @fires execute
26
+ */
27
+ execute(): void;
28
+ }