handsontable 0.0.0-next-8dc7078-20240322 → 0.0.0-next-28fc088-20240325

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of handsontable might be problematic. Click here for more details.

Files changed (98) hide show
  1. package/3rdparty/walkontable/src/cell/range.d.ts +2 -1
  2. package/3rdparty/walkontable/src/cell/range.js +22 -5
  3. package/3rdparty/walkontable/src/cell/range.mjs +22 -5
  4. package/base.js +2 -2
  5. package/base.mjs +2 -2
  6. package/core/viewportScroll/index.js +4 -1
  7. package/core/viewportScroll/index.mjs +4 -1
  8. package/core/viewportScroll/scrollStrategies/focusScroll.js +15 -0
  9. package/core/viewportScroll/scrollStrategies/focusScroll.mjs +11 -0
  10. package/core.d.ts +4 -3
  11. package/core.js +76 -21
  12. package/core.mjs +76 -21
  13. package/dist/handsontable.css +2 -2
  14. package/dist/handsontable.full.css +2 -2
  15. package/dist/handsontable.full.js +2649 -1389
  16. package/dist/handsontable.full.min.css +2 -2
  17. package/dist/handsontable.full.min.js +71 -71
  18. package/dist/handsontable.js +2653 -1393
  19. package/dist/handsontable.min.css +2 -2
  20. package/dist/handsontable.min.js +19 -19
  21. package/editorManager.js +12 -8
  22. package/editorManager.mjs +12 -8
  23. package/focusManager.js +7 -1
  24. package/focusManager.mjs +7 -1
  25. package/helpers/mixed.js +1 -1
  26. package/helpers/mixed.mjs +1 -1
  27. package/package.json +1 -1
  28. package/pluginHooks.d.ts +4 -0
  29. package/pluginHooks.js +98 -37
  30. package/pluginHooks.mjs +98 -37
  31. package/plugins/collapsibleColumns/collapsibleColumns.js +9 -3
  32. package/plugins/collapsibleColumns/collapsibleColumns.mjs +9 -3
  33. package/plugins/columnSorting/columnSorting.js +8 -2
  34. package/plugins/columnSorting/columnSorting.mjs +8 -2
  35. package/plugins/contextMenu/menu/defaultShortcutsList.js +26 -10
  36. package/plugins/contextMenu/menu/defaultShortcutsList.mjs +26 -10
  37. package/plugins/mergeCells/calculations/selection.js +1 -70
  38. package/plugins/mergeCells/calculations/selection.mjs +1 -70
  39. package/plugins/mergeCells/cellsCollection.js +116 -0
  40. package/plugins/mergeCells/cellsCollection.mjs +116 -0
  41. package/plugins/mergeCells/contextMenuItem/toggleMerge.js +11 -1
  42. package/plugins/mergeCells/contextMenuItem/toggleMerge.mjs +11 -1
  43. package/plugins/mergeCells/focusOrder.js +305 -0
  44. package/plugins/mergeCells/focusOrder.mjs +300 -0
  45. package/plugins/mergeCells/mergeCells.js +336 -192
  46. package/plugins/mergeCells/mergeCells.mjs +336 -192
  47. package/plugins/multiColumnSorting/multiColumnSorting.js +8 -2
  48. package/plugins/multiColumnSorting/multiColumnSorting.mjs +8 -2
  49. package/plugins/nestedHeaders/nestedHeaders.js +1 -0
  50. package/plugins/nestedHeaders/nestedHeaders.mjs +1 -0
  51. package/plugins/nestedRows/nestedRows.js +9 -3
  52. package/plugins/nestedRows/nestedRows.mjs +9 -3
  53. package/renderers/checkboxRenderer/checkboxRenderer.js +8 -5
  54. package/renderers/checkboxRenderer/checkboxRenderer.mjs +8 -5
  55. package/selection/highlight/visualSelection.js +2 -0
  56. package/selection/highlight/visualSelection.mjs +2 -0
  57. package/selection/selection.js +209 -40
  58. package/selection/selection.mjs +208 -39
  59. package/selection/transformation.js +83 -32
  60. package/selection/transformation.mjs +83 -32
  61. package/shortcutContexts/commands/editor/closeAndSave.js +2 -2
  62. package/shortcutContexts/commands/editor/closeAndSave.mjs +2 -2
  63. package/shortcutContexts/commands/editor/open.js +18 -3
  64. package/shortcutContexts/commands/editor/open.mjs +18 -3
  65. package/shortcutContexts/commands/extendCellsSelection/down.js +1 -1
  66. package/shortcutContexts/commands/extendCellsSelection/down.mjs +1 -1
  67. package/shortcutContexts/commands/extendCellsSelection/left.js +1 -1
  68. package/shortcutContexts/commands/extendCellsSelection/left.mjs +1 -1
  69. package/shortcutContexts/commands/extendCellsSelection/right.js +1 -1
  70. package/shortcutContexts/commands/extendCellsSelection/right.mjs +1 -1
  71. package/shortcutContexts/commands/extendCellsSelection/toColumns.js +1 -1
  72. package/shortcutContexts/commands/extendCellsSelection/toColumns.mjs +1 -1
  73. package/shortcutContexts/commands/extendCellsSelection/toMostBottom.js +3 -1
  74. package/shortcutContexts/commands/extendCellsSelection/toMostBottom.mjs +3 -1
  75. package/shortcutContexts/commands/extendCellsSelection/toMostInlineEnd.js +9 -3
  76. package/shortcutContexts/commands/extendCellsSelection/toMostInlineEnd.mjs +9 -3
  77. package/shortcutContexts/commands/extendCellsSelection/toMostInlineStart.js +10 -3
  78. package/shortcutContexts/commands/extendCellsSelection/toMostInlineStart.mjs +10 -3
  79. package/shortcutContexts/commands/extendCellsSelection/toMostLeft.js +3 -1
  80. package/shortcutContexts/commands/extendCellsSelection/toMostLeft.mjs +3 -1
  81. package/shortcutContexts/commands/extendCellsSelection/toMostRight.js +3 -1
  82. package/shortcutContexts/commands/extendCellsSelection/toMostRight.mjs +3 -1
  83. package/shortcutContexts/commands/extendCellsSelection/toMostTop.js +3 -1
  84. package/shortcutContexts/commands/extendCellsSelection/toMostTop.mjs +3 -1
  85. package/shortcutContexts/commands/extendCellsSelection/toRows.js +1 -1
  86. package/shortcutContexts/commands/extendCellsSelection/toRows.mjs +1 -1
  87. package/shortcutContexts/commands/extendCellsSelection/up.js +1 -1
  88. package/shortcutContexts/commands/extendCellsSelection/up.mjs +1 -1
  89. package/shortcutContexts/commands/moveCellSelection/inlineEnd.js +6 -1
  90. package/shortcutContexts/commands/moveCellSelection/inlineEnd.mjs +6 -1
  91. package/shortcutContexts/commands/moveCellSelection/inlineStart.js +6 -1
  92. package/shortcutContexts/commands/moveCellSelection/inlineStart.mjs +6 -1
  93. package/shortcutContexts/grid.js +2 -2
  94. package/shortcutContexts/grid.mjs +2 -2
  95. package/shortcuts/context.js +2 -1
  96. package/shortcuts/context.mjs +2 -1
  97. package/utils/dataStructures/linkedList.js +6 -1
  98. package/utils/dataStructures/linkedList.mjs +6 -1
@@ -36,74 +36,6 @@ class SelectionCalculations {
36
36
  this.hot = plugin.hot;
37
37
  }
38
38
 
39
- /**
40
- * "Snap" the delta value according to defined merged cells. (In other words, compensate the rowspan -
41
- * e.g. Going up with `delta.row = -1` over a merged cell with `rowspan = 3`, `delta.row` should change to `-3`.).
42
- *
43
- * @param {object} delta The delta object containing `row` and `col` properties.
44
- * @param {CellRange} selectionRange The selection range.
45
- * @param {object} mergedCell A merged cell object.
46
- */
47
- snapDelta(delta, selectionRange, mergedCell) {
48
- const cellCoords = selectionRange.to;
49
- const newRow = cellCoords.row + delta.row;
50
- const newColumn = cellCoords.col + delta.col;
51
- if (delta.row) {
52
- this.jumpOverMergedCell(delta, mergedCell, newRow);
53
- } else if (delta.col) {
54
- this.jumpOverMergedCell(delta, mergedCell, newColumn);
55
- }
56
- }
57
-
58
- /**
59
- * "Jump" over the merged cell (compensate for the indexes within the merged cell to get past it).
60
- *
61
- * @private
62
- * @param {object} delta The delta object.
63
- * @param {MergedCellCoords} mergedCell The merge cell object.
64
- * @param {number} newIndex New row/column index, created with the delta.
65
- */
66
- jumpOverMergedCell(delta, mergedCell, newIndex) {
67
- let flatDelta = delta.row || delta.col;
68
- let includesIndex = null;
69
- let firstIndex = null;
70
- let lastIndex = null;
71
- if (delta.row) {
72
- includesIndex = mergedCell.includesVertically(newIndex);
73
- firstIndex = mergedCell.row;
74
- lastIndex = mergedCell.getLastRow();
75
- } else if (delta.col) {
76
- includesIndex = mergedCell.includesHorizontally(newIndex);
77
- firstIndex = mergedCell.col;
78
- lastIndex = mergedCell.getLastColumn();
79
- }
80
- if (flatDelta === 0) {
81
- return;
82
- } else if (flatDelta > 0) {
83
- if (includesIndex && newIndex !== firstIndex) {
84
- flatDelta += lastIndex - newIndex + 1;
85
- }
86
- } else if (includesIndex && newIndex !== lastIndex) {
87
- flatDelta -= newIndex - firstIndex + 1;
88
- }
89
- if (delta.row) {
90
- delta.row = flatDelta;
91
- } else if (delta.col) {
92
- delta.col = flatDelta;
93
- }
94
- }
95
-
96
- /**
97
- * Get a selection range with `to` property incremented by the provided delta.
98
- *
99
- * @param {CellRange} oldSelectionRange The base selection range.
100
- * @param {object} delta The delta object with `row` and `col` properties.
101
- * @returns {CellRange} A new `CellRange` object.
102
- */
103
- getUpdatedSelectionRange(oldSelectionRange, delta) {
104
- return this.hot._createCellRange(oldSelectionRange.highlight, oldSelectionRange.from, this.hot._createCellCoords(oldSelectionRange.to.row + delta.row, oldSelectionRange.to.col + delta.col));
105
- }
106
-
107
39
  /**
108
40
  * Generate an additional class name for the entirely-selected merged cells.
109
41
  *
@@ -136,8 +68,7 @@ class SelectionCalculations {
136
68
  const fullMergeAreaWithinSelection = startRow <= mergedCell.row && startColumn <= mergedCell.col && endRow >= mergeRowEnd && endColumn >= mergeColumnEnd;
137
69
  if (fullMergeAreaWithinSelection) {
138
70
  return `${this.fullySelectedMergedCellClassName}-${layerLevel}`;
139
- } else if (this.plugin.selectionCalculations.isMergeCellFullySelected(mergedCell, this.plugin.hot.getSelectedRange())) {
140
- // eslint-disable-line max-len
71
+ } else if (this.isMergeCellFullySelected(mergedCell, this.plugin.hot.getSelectedRange())) {
141
72
  return `${this.fullySelectedMergedCellClassName}-multiple`;
142
73
  }
143
74
  }
@@ -33,74 +33,6 @@ class SelectionCalculations {
33
33
  this.hot = plugin.hot;
34
34
  }
35
35
 
36
- /**
37
- * "Snap" the delta value according to defined merged cells. (In other words, compensate the rowspan -
38
- * e.g. Going up with `delta.row = -1` over a merged cell with `rowspan = 3`, `delta.row` should change to `-3`.).
39
- *
40
- * @param {object} delta The delta object containing `row` and `col` properties.
41
- * @param {CellRange} selectionRange The selection range.
42
- * @param {object} mergedCell A merged cell object.
43
- */
44
- snapDelta(delta, selectionRange, mergedCell) {
45
- const cellCoords = selectionRange.to;
46
- const newRow = cellCoords.row + delta.row;
47
- const newColumn = cellCoords.col + delta.col;
48
- if (delta.row) {
49
- this.jumpOverMergedCell(delta, mergedCell, newRow);
50
- } else if (delta.col) {
51
- this.jumpOverMergedCell(delta, mergedCell, newColumn);
52
- }
53
- }
54
-
55
- /**
56
- * "Jump" over the merged cell (compensate for the indexes within the merged cell to get past it).
57
- *
58
- * @private
59
- * @param {object} delta The delta object.
60
- * @param {MergedCellCoords} mergedCell The merge cell object.
61
- * @param {number} newIndex New row/column index, created with the delta.
62
- */
63
- jumpOverMergedCell(delta, mergedCell, newIndex) {
64
- let flatDelta = delta.row || delta.col;
65
- let includesIndex = null;
66
- let firstIndex = null;
67
- let lastIndex = null;
68
- if (delta.row) {
69
- includesIndex = mergedCell.includesVertically(newIndex);
70
- firstIndex = mergedCell.row;
71
- lastIndex = mergedCell.getLastRow();
72
- } else if (delta.col) {
73
- includesIndex = mergedCell.includesHorizontally(newIndex);
74
- firstIndex = mergedCell.col;
75
- lastIndex = mergedCell.getLastColumn();
76
- }
77
- if (flatDelta === 0) {
78
- return;
79
- } else if (flatDelta > 0) {
80
- if (includesIndex && newIndex !== firstIndex) {
81
- flatDelta += lastIndex - newIndex + 1;
82
- }
83
- } else if (includesIndex && newIndex !== lastIndex) {
84
- flatDelta -= newIndex - firstIndex + 1;
85
- }
86
- if (delta.row) {
87
- delta.row = flatDelta;
88
- } else if (delta.col) {
89
- delta.col = flatDelta;
90
- }
91
- }
92
-
93
- /**
94
- * Get a selection range with `to` property incremented by the provided delta.
95
- *
96
- * @param {CellRange} oldSelectionRange The base selection range.
97
- * @param {object} delta The delta object with `row` and `col` properties.
98
- * @returns {CellRange} A new `CellRange` object.
99
- */
100
- getUpdatedSelectionRange(oldSelectionRange, delta) {
101
- return this.hot._createCellRange(oldSelectionRange.highlight, oldSelectionRange.from, this.hot._createCellCoords(oldSelectionRange.to.row + delta.row, oldSelectionRange.to.col + delta.col));
102
- }
103
-
104
36
  /**
105
37
  * Generate an additional class name for the entirely-selected merged cells.
106
38
  *
@@ -133,8 +65,7 @@ class SelectionCalculations {
133
65
  const fullMergeAreaWithinSelection = startRow <= mergedCell.row && startColumn <= mergedCell.col && endRow >= mergeRowEnd && endColumn >= mergeColumnEnd;
134
66
  if (fullMergeAreaWithinSelection) {
135
67
  return `${this.fullySelectedMergedCellClassName}-${layerLevel}`;
136
- } else if (this.plugin.selectionCalculations.isMergeCellFullySelected(mergedCell, this.plugin.hot.getSelectedRange())) {
137
- // eslint-disable-line max-len
68
+ } else if (this.isMergeCellFullySelected(mergedCell, this.plugin.hot.getSelectedRange())) {
138
69
  return `${this.fullySelectedMergedCellClassName}-multiple`;
139
70
  }
140
71
  }
@@ -3,6 +3,7 @@
3
3
  exports.__esModule = true;
4
4
  require("core-js/modules/es.error.cause.js");
5
5
  require("core-js/modules/es.array.push.js");
6
+ require("core-js/modules/es.array.unscopables.flat-map.js");
6
7
  var _cellCoords = _interopRequireDefault(require("./cellCoords"));
7
8
  var _number = require("../../helpers/number");
8
9
  var _console = require("../../helpers/console");
@@ -10,17 +11,30 @@ var _array = require("../../helpers/array");
10
11
  var _utils = require("./utils");
11
12
  var _templateLiteralTag = require("../../helpers/templateLiteralTag");
12
13
  function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
14
+ function _classPrivateMethodInitSpec(obj, privateSet) { _checkPrivateRedeclaration(obj, privateSet); privateSet.add(obj); }
15
+ function _checkPrivateRedeclaration(obj, privateCollection) { if (privateCollection.has(obj)) { throw new TypeError("Cannot initialize the same private elements twice on an object"); } }
13
16
  function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
14
17
  function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : String(i); }
15
18
  function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
19
+ function _assertClassBrand(e, t, n) { if ("function" == typeof e ? e === t : e.has(t)) return arguments.length < 3 ? t : n; throw new TypeError("Private element is not present on this object"); }
16
20
  /**
17
21
  * Defines a container object for the merged cells.
18
22
  *
19
23
  * @private
20
24
  * @class MergedCellsCollection
21
25
  */
26
+ var _MergedCellsCollection_brand = /*#__PURE__*/new WeakSet();
22
27
  class MergedCellsCollection {
23
28
  constructor(mergeCellsPlugin) {
29
+ /**
30
+ * Gets the list of the indexes that do not intersect with other merged cells within the provided range.
31
+ *
32
+ * @param {CellRange} range The range to search within.
33
+ * @param {'row' | 'col'} axis The axis to search within.
34
+ * @param {number} scanDirection The direction to scan the range. `1` for forward, `-1` for backward.
35
+ * @returns {number[]}
36
+ */
37
+ _classPrivateMethodInitSpec(this, _MergedCellsCollection_brand);
24
38
  /**
25
39
  * Reference to the Merge Cells plugin.
26
40
  *
@@ -259,6 +273,81 @@ class MergedCellsCollection {
259
273
  return this.hot._createCellCoords(firstRenderableRow, firstRenderableColumn);
260
274
  }
261
275
 
276
+ /**
277
+ * Gets the start-most visual column index that do not intersect with other merged cells within the provided range.
278
+ *
279
+ * @param {CellRange} range The range to search within.
280
+ * @param {number} visualColumnIndex The visual column index to start the search from.
281
+ * @returns {number}
282
+ */
283
+ getStartMostColumnIndex(range, visualColumnIndex) {
284
+ const indexes = _assertClassBrand(_MergedCellsCollection_brand, this, _getNonIntersectingIndexes).call(this, range, 'col', -1);
285
+ let startMostIndex = visualColumnIndex;
286
+ for (let i = 0; i < indexes.length; i++) {
287
+ if (indexes[i] <= visualColumnIndex) {
288
+ startMostIndex = indexes[i];
289
+ break;
290
+ }
291
+ }
292
+ return startMostIndex;
293
+ }
294
+
295
+ /**
296
+ * Gets the end-most visual column index that do not intersect with other merged cells within the provided range.
297
+ *
298
+ * @param {CellRange} range The range to search within.
299
+ * @param {number} visualColumnIndex The visual column index to start the search from.
300
+ * @returns {number}
301
+ */
302
+ getEndMostColumnIndex(range, visualColumnIndex) {
303
+ const indexes = _assertClassBrand(_MergedCellsCollection_brand, this, _getNonIntersectingIndexes).call(this, range, 'col', 1);
304
+ let endMostIndex = visualColumnIndex;
305
+ for (let i = 0; i < indexes.length; i++) {
306
+ if (indexes[i] >= visualColumnIndex) {
307
+ endMostIndex = indexes[i];
308
+ break;
309
+ }
310
+ }
311
+ return endMostIndex;
312
+ }
313
+
314
+ /**
315
+ * Gets the top-most visual row index that do not intersect with other merged cells within the provided range.
316
+ *
317
+ * @param {CellRange} range The range to search within.
318
+ * @param {number} visualRowIndex The visual row index to start the search from.
319
+ * @returns {number}
320
+ */
321
+ getTopMostRowIndex(range, visualRowIndex) {
322
+ const indexes = _assertClassBrand(_MergedCellsCollection_brand, this, _getNonIntersectingIndexes).call(this, range, 'row', -1);
323
+ let topMostIndex = visualRowIndex;
324
+ for (let i = 0; i < indexes.length; i++) {
325
+ if (indexes[i] <= visualRowIndex) {
326
+ topMostIndex = indexes[i];
327
+ break;
328
+ }
329
+ }
330
+ return topMostIndex;
331
+ }
332
+
333
+ /**
334
+ * Gets the bottom-most visual row index that do not intersect with other merged cells within the provided range.
335
+ *
336
+ * @param {CellRange} range The range to search within.
337
+ * @param {number} visualRowIndex The visual row index to start the search from.
338
+ * @returns {number}
339
+ */
340
+ getBottomMostRowIndex(range, visualRowIndex) {
341
+ const indexes = _assertClassBrand(_MergedCellsCollection_brand, this, _getNonIntersectingIndexes).call(this, range, 'row', 1);
342
+ let bottomMostIndex = visualRowIndex;
343
+ for (let i = 0; i < indexes.length; i++) {
344
+ if (indexes[i] >= visualRowIndex) {
345
+ bottomMostIndex = indexes[i];
346
+ break;
347
+ }
348
+ }
349
+ return bottomMostIndex;
350
+ }
262
351
  /**
263
352
  * Shift the merged cell in the direction and by an offset defined in the arguments.
264
353
  *
@@ -294,4 +383,31 @@ class MergedCellsCollection {
294
383
  });
295
384
  }
296
385
  }
386
+ function _getNonIntersectingIndexes(range, axis) {
387
+ let scanDirection = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 1;
388
+ const indexes = new Map();
389
+ const from = scanDirection === 1 ? range.getTopStartCorner() : range.getBottomEndCorner();
390
+ const to = scanDirection === 1 ? range.getBottomEndCorner() : range.getTopStartCorner();
391
+ for (let row = from.row; scanDirection === 1 ? row <= to.row : row >= to.row; row += scanDirection) {
392
+ for (let column = from.col; scanDirection === 1 ? column <= to.col : column >= to.col; column += scanDirection) {
393
+ const index = axis === 'row' ? row : column;
394
+ const mergedCell = this.get(row, column);
395
+ let lastIndex = index;
396
+ if (mergedCell) {
397
+ lastIndex = scanDirection === 1 ? mergedCell[axis] + mergedCell[`${axis}span`] - 1 : mergedCell[axis];
398
+ }
399
+ if (!indexes.has(index)) {
400
+ indexes.set(index, new Set());
401
+ }
402
+ indexes.get(index).add(lastIndex);
403
+ }
404
+ }
405
+ return Array.from(new Set(Array.from(indexes.entries()).filter(_ref => {
406
+ let [, set] = _ref;
407
+ return set.size === 1;
408
+ }).flatMap(_ref2 => {
409
+ let [, set] = _ref2;
410
+ return Array.from(set);
411
+ })));
412
+ }
297
413
  var _default = exports.default = MergedCellsCollection;
@@ -1,8 +1,12 @@
1
1
  import "core-js/modules/es.error.cause.js";
2
2
  import "core-js/modules/es.array.push.js";
3
+ import "core-js/modules/es.array.unscopables.flat-map.js";
4
+ function _classPrivateMethodInitSpec(obj, privateSet) { _checkPrivateRedeclaration(obj, privateSet); privateSet.add(obj); }
5
+ function _checkPrivateRedeclaration(obj, privateCollection) { if (privateCollection.has(obj)) { throw new TypeError("Cannot initialize the same private elements twice on an object"); } }
3
6
  function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
4
7
  function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : String(i); }
5
8
  function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
9
+ function _assertClassBrand(e, t, n) { if ("function" == typeof e ? e === t : e.has(t)) return arguments.length < 3 ? t : n; throw new TypeError("Private element is not present on this object"); }
6
10
  import MergedCellCoords from "./cellCoords.mjs";
7
11
  import { rangeEach, rangeEachReverse } from "../../helpers/number.mjs";
8
12
  import { warn } from "../../helpers/console.mjs";
@@ -15,8 +19,18 @@ import { toSingleLine } from "../../helpers/templateLiteralTag.mjs";
15
19
  * @private
16
20
  * @class MergedCellsCollection
17
21
  */
22
+ var _MergedCellsCollection_brand = /*#__PURE__*/new WeakSet();
18
23
  class MergedCellsCollection {
19
24
  constructor(mergeCellsPlugin) {
25
+ /**
26
+ * Gets the list of the indexes that do not intersect with other merged cells within the provided range.
27
+ *
28
+ * @param {CellRange} range The range to search within.
29
+ * @param {'row' | 'col'} axis The axis to search within.
30
+ * @param {number} scanDirection The direction to scan the range. `1` for forward, `-1` for backward.
31
+ * @returns {number[]}
32
+ */
33
+ _classPrivateMethodInitSpec(this, _MergedCellsCollection_brand);
20
34
  /**
21
35
  * Reference to the Merge Cells plugin.
22
36
  *
@@ -255,6 +269,81 @@ class MergedCellsCollection {
255
269
  return this.hot._createCellCoords(firstRenderableRow, firstRenderableColumn);
256
270
  }
257
271
 
272
+ /**
273
+ * Gets the start-most visual column index that do not intersect with other merged cells within the provided range.
274
+ *
275
+ * @param {CellRange} range The range to search within.
276
+ * @param {number} visualColumnIndex The visual column index to start the search from.
277
+ * @returns {number}
278
+ */
279
+ getStartMostColumnIndex(range, visualColumnIndex) {
280
+ const indexes = _assertClassBrand(_MergedCellsCollection_brand, this, _getNonIntersectingIndexes).call(this, range, 'col', -1);
281
+ let startMostIndex = visualColumnIndex;
282
+ for (let i = 0; i < indexes.length; i++) {
283
+ if (indexes[i] <= visualColumnIndex) {
284
+ startMostIndex = indexes[i];
285
+ break;
286
+ }
287
+ }
288
+ return startMostIndex;
289
+ }
290
+
291
+ /**
292
+ * Gets the end-most visual column index that do not intersect with other merged cells within the provided range.
293
+ *
294
+ * @param {CellRange} range The range to search within.
295
+ * @param {number} visualColumnIndex The visual column index to start the search from.
296
+ * @returns {number}
297
+ */
298
+ getEndMostColumnIndex(range, visualColumnIndex) {
299
+ const indexes = _assertClassBrand(_MergedCellsCollection_brand, this, _getNonIntersectingIndexes).call(this, range, 'col', 1);
300
+ let endMostIndex = visualColumnIndex;
301
+ for (let i = 0; i < indexes.length; i++) {
302
+ if (indexes[i] >= visualColumnIndex) {
303
+ endMostIndex = indexes[i];
304
+ break;
305
+ }
306
+ }
307
+ return endMostIndex;
308
+ }
309
+
310
+ /**
311
+ * Gets the top-most visual row index that do not intersect with other merged cells within the provided range.
312
+ *
313
+ * @param {CellRange} range The range to search within.
314
+ * @param {number} visualRowIndex The visual row index to start the search from.
315
+ * @returns {number}
316
+ */
317
+ getTopMostRowIndex(range, visualRowIndex) {
318
+ const indexes = _assertClassBrand(_MergedCellsCollection_brand, this, _getNonIntersectingIndexes).call(this, range, 'row', -1);
319
+ let topMostIndex = visualRowIndex;
320
+ for (let i = 0; i < indexes.length; i++) {
321
+ if (indexes[i] <= visualRowIndex) {
322
+ topMostIndex = indexes[i];
323
+ break;
324
+ }
325
+ }
326
+ return topMostIndex;
327
+ }
328
+
329
+ /**
330
+ * Gets the bottom-most visual row index that do not intersect with other merged cells within the provided range.
331
+ *
332
+ * @param {CellRange} range The range to search within.
333
+ * @param {number} visualRowIndex The visual row index to start the search from.
334
+ * @returns {number}
335
+ */
336
+ getBottomMostRowIndex(range, visualRowIndex) {
337
+ const indexes = _assertClassBrand(_MergedCellsCollection_brand, this, _getNonIntersectingIndexes).call(this, range, 'row', 1);
338
+ let bottomMostIndex = visualRowIndex;
339
+ for (let i = 0; i < indexes.length; i++) {
340
+ if (indexes[i] >= visualRowIndex) {
341
+ bottomMostIndex = indexes[i];
342
+ break;
343
+ }
344
+ }
345
+ return bottomMostIndex;
346
+ }
258
347
  /**
259
348
  * Shift the merged cell in the direction and by an offset defined in the arguments.
260
349
  *
@@ -290,4 +379,31 @@ class MergedCellsCollection {
290
379
  });
291
380
  }
292
381
  }
382
+ function _getNonIntersectingIndexes(range, axis) {
383
+ let scanDirection = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 1;
384
+ const indexes = new Map();
385
+ const from = scanDirection === 1 ? range.getTopStartCorner() : range.getBottomEndCorner();
386
+ const to = scanDirection === 1 ? range.getBottomEndCorner() : range.getTopStartCorner();
387
+ for (let row = from.row; scanDirection === 1 ? row <= to.row : row >= to.row; row += scanDirection) {
388
+ for (let column = from.col; scanDirection === 1 ? column <= to.col : column >= to.col; column += scanDirection) {
389
+ const index = axis === 'row' ? row : column;
390
+ const mergedCell = this.get(row, column);
391
+ let lastIndex = index;
392
+ if (mergedCell) {
393
+ lastIndex = scanDirection === 1 ? mergedCell[axis] + mergedCell[`${axis}span`] - 1 : mergedCell[axis];
394
+ }
395
+ if (!indexes.has(index)) {
396
+ indexes.set(index, new Set());
397
+ }
398
+ indexes.get(index).add(lastIndex);
399
+ }
400
+ }
401
+ return Array.from(new Set(Array.from(indexes.entries()).filter(_ref => {
402
+ let [, set] = _ref;
403
+ return set.size === 1;
404
+ }).flatMap(_ref2 => {
405
+ let [, set] = _ref2;
406
+ return Array.from(set);
407
+ })));
408
+ }
293
409
  export default MergedCellsCollection;
@@ -25,7 +25,17 @@ function toggleMergeItem(plugin) {
25
25
  return this.getTranslatedPhrase(C.CONTEXTMENU_ITEMS_MERGE_CELLS);
26
26
  },
27
27
  callback() {
28
- plugin.toggleMergeOnSelection();
28
+ const currentRange = this.getSelectedRangeLast();
29
+ if (!currentRange) {
30
+ return;
31
+ }
32
+ currentRange.setDirection(this.isRtl() ? 'NE-SW' : 'NW-SE');
33
+ const {
34
+ from,
35
+ to
36
+ } = currentRange;
37
+ plugin.toggleMerge(currentRange);
38
+ this.selectCell(from.row, from.col, to.row, to.col, false);
29
39
  },
30
40
  disabled() {
31
41
  const sel = this.getSelectedLast();
@@ -18,7 +18,17 @@ export default function toggleMergeItem(plugin) {
18
18
  return this.getTranslatedPhrase(C.CONTEXTMENU_ITEMS_MERGE_CELLS);
19
19
  },
20
20
  callback() {
21
- plugin.toggleMergeOnSelection();
21
+ const currentRange = this.getSelectedRangeLast();
22
+ if (!currentRange) {
23
+ return;
24
+ }
25
+ currentRange.setDirection(this.isRtl() ? 'NE-SW' : 'NW-SE');
26
+ const {
27
+ from,
28
+ to
29
+ } = currentRange;
30
+ plugin.toggleMerge(currentRange);
31
+ this.selectCell(from.row, from.col, to.row, to.col, false);
22
32
  },
23
33
  disabled() {
24
34
  const sel = this.getSelectedLast();