UNPKG

handsontable

Version:

Handsontable is a JavaScript Spreadsheet Component available for React, Angular and Vue.

1,212 lines (1,008 loc) • 58.7 kB
"use strict"; function _typeof(obj) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (obj) { return typeof obj; } : function (obj) { return obj && "function" == typeof Symbol && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }, _typeof(obj); } require("core-js/modules/es.object.set-prototype-of.js"); require("core-js/modules/es.object.get-prototype-of.js"); require("core-js/modules/es.reflect.construct.js"); require("core-js/modules/es.reflect.get.js"); require("core-js/modules/es.object.get-own-property-descriptor.js"); require("core-js/modules/es.symbol.js"); require("core-js/modules/es.symbol.description.js"); require("core-js/modules/es.symbol.iterator.js"); require("core-js/modules/es.array.from.js"); require("core-js/modules/es.array.slice.js"); require("core-js/modules/es.function.name.js"); require("core-js/modules/es.regexp.exec.js"); exports.__esModule = true; exports.PLUGIN_PRIORITY = exports.PLUGIN_KEY = exports.MergeCells = void 0; require("core-js/modules/es.array.iterator.js"); require("core-js/modules/es.object.to-string.js"); require("core-js/modules/es.string.iterator.js"); require("core-js/modules/es.weak-map.js"); require("core-js/modules/web.dom-collections.iterator.js"); require("core-js/modules/es.array.filter.js"); require("core-js/modules/es.array.splice.js"); require("core-js/modules/es.array.concat.js"); require("core-js/modules/es.array.includes.js"); require("core-js/modules/es.string.includes.js"); var _base = require("../base"); var _pluginHooks = _interopRequireDefault(require("../../pluginHooks")); var _event = require("../../helpers/dom/event"); var _src = require("../../3rdparty/walkontable/src"); var _cellsCollection = _interopRequireDefault(require("./cellsCollection")); var _cellCoords = _interopRequireDefault(require("./cellCoords")); var _autofill = _interopRequireDefault(require("./calculations/autofill")); var _selection = _interopRequireDefault(require("./calculations/selection")); var _toggleMerge = _interopRequireDefault(require("./contextMenuItem/toggleMerge")); var _array = require("../../helpers/array"); var _object = require("../../helpers/object"); var _console = require("../../helpers/console"); var _number = require("../../helpers/number"); var _utils = require("./utils"); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); } function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } function _iterableToArrayLimit(arr, i) { var _i = arr == null ? null : typeof Symbol !== "undefined" && arr[Symbol.iterator] || arr["@@iterator"]; if (_i == null) return; var _arr = []; var _n = true; var _d = false; var _s, _e; try { for (_i = _i.call(arr); !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"] != null) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; } function _toConsumableArray(arr) { return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread(); } function _nonIterableSpread() { throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); } function _iterableToArray(iter) { if (typeof Symbol !== "undefined" && iter[Symbol.iterator] != null || iter["@@iterator"] != null) return Array.from(iter); } function _arrayWithoutHoles(arr) { if (Array.isArray(arr)) return _arrayLikeToArray(arr); } function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } function _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); Object.defineProperty(Constructor, "prototype", { writable: false }); return Constructor; } function _get() { if (typeof Reflect !== "undefined" && Reflect.get) { _get = Reflect.get; } else { _get = function _get(target, property, receiver) { var base = _superPropBase(target, property); if (!base) return; var desc = Object.getOwnPropertyDescriptor(base, property); if (desc.get) { return desc.get.call(arguments.length < 3 ? target : receiver); } return desc.value; }; } return _get.apply(this, arguments); } function _superPropBase(object, property) { while (!Object.prototype.hasOwnProperty.call(object, property)) { object = _getPrototypeOf(object); if (object === null) break; } return object; } function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function"); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, writable: true, configurable: true } }); Object.defineProperty(subClass, "prototype", { writable: false }); if (superClass) _setPrototypeOf(subClass, superClass); } function _setPrototypeOf(o, p) { _setPrototypeOf = Object.setPrototypeOf || function _setPrototypeOf(o, p) { o.__proto__ = p; return o; }; return _setPrototypeOf(o, p); } function _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = _getPrototypeOf(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = _getPrototypeOf(this).constructor; result = Reflect.construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return _possibleConstructorReturn(this, result); }; } function _possibleConstructorReturn(self, call) { if (call && (_typeof(call) === "object" || typeof call === "function")) { return call; } else if (call !== void 0) { throw new TypeError("Derived constructors may only return object or undefined"); } return _assertThisInitialized(self); } function _assertThisInitialized(self) { if (self === void 0) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return self; } function _isNativeReflectConstruct() { if (typeof Reflect === "undefined" || !Reflect.construct) return false; if (Reflect.construct.sham) return false; if (typeof Proxy === "function") return true; try { Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); return true; } catch (e) { return false; } } function _getPrototypeOf(o) { _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf : function _getPrototypeOf(o) { return o.__proto__ || Object.getPrototypeOf(o); }; return _getPrototypeOf(o); } _pluginHooks.default.getSingleton().register('beforeMergeCells'); _pluginHooks.default.getSingleton().register('afterMergeCells'); _pluginHooks.default.getSingleton().register('beforeUnmergeCells'); _pluginHooks.default.getSingleton().register('afterUnmergeCells'); var PLUGIN_KEY = 'mergeCells'; exports.PLUGIN_KEY = PLUGIN_KEY; var PLUGIN_PRIORITY = 150; exports.PLUGIN_PRIORITY = PLUGIN_PRIORITY; var privatePool = new WeakMap(); /** * @plugin MergeCells * @class MergeCells * * @description * Plugin, which allows merging cells in the table (using the initial configuration, API or context menu). * * @example * * ```js * const hot = new Handsontable(document.getElementById('example'), { * data: getData(), * mergeCells: [ * {row: 0, col: 3, rowspan: 3, colspan: 3}, * {row: 2, col: 6, rowspan: 2, colspan: 2}, * {row: 4, col: 8, rowspan: 3, colspan: 3} * ], * ``` */ var MergeCells = /*#__PURE__*/function (_BasePlugin) { _inherits(MergeCells, _BasePlugin); var _super = _createSuper(MergeCells); function MergeCells(hotInstance) { var _this; _classCallCheck(this, MergeCells); _this = _super.call(this, hotInstance); privatePool.set(_assertThisInitialized(_this), { lastDesiredCoords: null }); /** * A container for all the merged cells. * * @private * @type {MergedCellsCollection} */ _this.mergedCellsCollection = null; /** * Instance of the class responsible for all the autofill-related calculations. * * @private * @type {AutofillCalculations} */ _this.autofillCalculations = null; /** * Instance of the class responsible for the selection-related calculations. * * @private * @type {SelectionCalculations} */ _this.selectionCalculations = null; return _this; } /** * Checks if the plugin is enabled in the handsontable settings. This method is executed in {@link Hooks#beforeInit} * hook and if it returns `true` than the {@link MergeCells#enablePlugin} method is called. * * @returns {boolean} */ _createClass(MergeCells, [{ key: "isEnabled", value: function isEnabled() { return !!this.hot.getSettings()[PLUGIN_KEY]; } /** * Enables the plugin functionality for this Handsontable instance. */ }, { key: "enablePlugin", value: function enablePlugin() { var _this2 = this; if (this.enabled) { return; } this.mergedCellsCollection = new _cellsCollection.default(this); this.autofillCalculations = new _autofill.default(this); this.selectionCalculations = new _selection.default(this); this.addHook('afterInit', function () { return _this2.onAfterInit.apply(_this2, arguments); }); this.addHook('beforeKeyDown', function () { return _this2.onBeforeKeyDown.apply(_this2, arguments); }); this.addHook('modifyTransformStart', function () { return _this2.onModifyTransformStart.apply(_this2, arguments); }); this.addHook('afterModifyTransformStart', function () { return _this2.onAfterModifyTransformStart.apply(_this2, arguments); }); this.addHook('modifyTransformEnd', function () { return _this2.onModifyTransformEnd.apply(_this2, arguments); }); this.addHook('modifyGetCellCoords', function () { return _this2.onModifyGetCellCoords.apply(_this2, arguments); }); this.addHook('beforeSetRangeStart', function () { return _this2.onBeforeSetRangeStart.apply(_this2, arguments); }); this.addHook('beforeSetRangeStartOnly', function () { return _this2.onBeforeSetRangeStart.apply(_this2, arguments); }); this.addHook('beforeSetRangeEnd', function () { return _this2.onBeforeSetRangeEnd.apply(_this2, arguments); }); this.addHook('afterIsMultipleSelection', function () { return _this2.onAfterIsMultipleSelection.apply(_this2, arguments); }); this.addHook('afterRenderer', function () { return _this2.onAfterRenderer.apply(_this2, arguments); }); this.addHook('afterContextMenuDefaultOptions', function () { return _this2.addMergeActionsToContextMenu.apply(_this2, arguments); }); this.addHook('afterGetCellMeta', function () { return _this2.onAfterGetCellMeta.apply(_this2, arguments); }); this.addHook('afterViewportRowCalculatorOverride', function () { return _this2.onAfterViewportRowCalculatorOverride.apply(_this2, arguments); }); this.addHook('afterViewportColumnCalculatorOverride', function () { return _this2.onAfterViewportColumnCalculatorOverride.apply(_this2, arguments); }); this.addHook('modifyAutofillRange', function () { return _this2.onModifyAutofillRange.apply(_this2, arguments); }); this.addHook('afterCreateCol', function () { return _this2.onAfterCreateCol.apply(_this2, arguments); }); this.addHook('afterRemoveCol', function () { return _this2.onAfterRemoveCol.apply(_this2, arguments); }); this.addHook('afterCreateRow', function () { return _this2.onAfterCreateRow.apply(_this2, arguments); }); this.addHook('afterRemoveRow', function () { return _this2.onAfterRemoveRow.apply(_this2, arguments); }); this.addHook('afterChange', function () { return _this2.onAfterChange.apply(_this2, arguments); }); this.addHook('beforeDrawBorders', function () { return _this2.onBeforeDrawAreaBorders.apply(_this2, arguments); }); this.addHook('afterDrawSelection', function () { return _this2.onAfterDrawSelection.apply(_this2, arguments); }); this.addHook('beforeRemoveCellClassNames', function () { return _this2.onBeforeRemoveCellClassNames.apply(_this2, arguments); }); this.addHook('beforeUndoStackChange', function (action, source) { if (source === 'MergeCells') { return false; } }); _get(_getPrototypeOf(MergeCells.prototype), "enablePlugin", this).call(this); } /** * Disables the plugin functionality for this Handsontable instance. */ }, { key: "disablePlugin", value: function disablePlugin() { this.clearCollections(); this.hot.render(); _get(_getPrototypeOf(MergeCells.prototype), "disablePlugin", this).call(this); } /** * Updates the plugin state. This method is executed when {@link Core#updateSettings} is invoked. */ }, { key: "updatePlugin", value: function updatePlugin() { var settings = this.hot.getSettings()[PLUGIN_KEY]; this.disablePlugin(); this.enablePlugin(); this.generateFromSettings(settings); _get(_getPrototypeOf(MergeCells.prototype), "updatePlugin", this).call(this); } /** * Validates a single setting object, represented by a single merged cell information object. * * @private * @param {object} setting An object with `row`, `col`, `rowspan` and `colspan` properties. * @returns {boolean} */ }, { key: "validateSetting", value: function validateSetting(setting) { var valid = true; if (!setting) { return false; } if (_cellCoords.default.containsNegativeValues(setting)) { (0, _console.warn)(_cellCoords.default.NEGATIVE_VALUES_WARNING(setting)); valid = false; } else if (_cellCoords.default.isOutOfBounds(setting, this.hot.countRows(), this.hot.countCols())) { (0, _console.warn)(_cellCoords.default.IS_OUT_OF_BOUNDS_WARNING(setting)); valid = false; } else if (_cellCoords.default.isSingleCell(setting)) { (0, _console.warn)(_cellCoords.default.IS_SINGLE_CELL(setting)); valid = false; } else if (_cellCoords.default.containsZeroSpan(setting)) { (0, _console.warn)(_cellCoords.default.ZERO_SPAN_WARNING(setting)); valid = false; } return valid; } /** * Generates the merged cells from the settings provided to the plugin. * * @private * @param {Array|boolean} settings The settings provided to the plugin. */ }, { key: "generateFromSettings", value: function generateFromSettings(settings) { var _this3 = this; if (Array.isArray(settings)) { var _this$hot; var populationArgumentsList = []; (0, _array.arrayEach)(settings, function (setting) { if (!_this3.validateSetting(setting)) { return; } var highlight = new _src.CellCoords(setting.row, setting.col); var rangeEnd = new _src.CellCoords(setting.row + setting.rowspan - 1, setting.col + setting.colspan - 1); var mergeRange = new _src.CellRange(highlight, highlight, rangeEnd); populationArgumentsList.push(_this3.mergeRange(mergeRange, true, true)); }); // remove 'empty' setting objects, caused by improper merge range declarations populationArgumentsList = populationArgumentsList.filter(function (value) { return value !== true; }); var bulkPopulationData = this.getBulkCollectionData(populationArgumentsList); (_this$hot = this.hot).populateFromArray.apply(_this$hot, _toConsumableArray(bulkPopulationData)); } } /** * Generates a bulk set of all the data to be populated to fill the data "under" the added merged cells. * * @private * @param {Array} populationArgumentsList Array in a form of `[row, column, dataUnderCollection]`. * @returns {Array} Array in a form of `[row, column, dataOfAllCollections]`. */ }, { key: "getBulkCollectionData", value: function getBulkCollectionData(populationArgumentsList) { var _this$hot2; var populationDataRange = this.getBulkCollectionDataRange(populationArgumentsList); var dataAtRange = (_this$hot2 = this.hot).getData.apply(_this$hot2, _toConsumableArray(populationDataRange)); var newDataAtRange = dataAtRange.splice(0); (0, _array.arrayEach)(populationArgumentsList, function (mergedCellArguments) { var _mergedCellArguments = _slicedToArray(mergedCellArguments, 3), mergedCellRowIndex = _mergedCellArguments[0], mergedCellColumnIndex = _mergedCellArguments[1], mergedCellData = _mergedCellArguments[2]; (0, _array.arrayEach)(mergedCellData, function (mergedCellRow, rowIndex) { (0, _array.arrayEach)(mergedCellRow, function (mergedCellElement, columnIndex) { newDataAtRange[mergedCellRowIndex - populationDataRange[0] + rowIndex][mergedCellColumnIndex - populationDataRange[1] + columnIndex] = mergedCellElement; // eslint-disable-line max-len }); }); }); return [populationDataRange[0], populationDataRange[1], newDataAtRange]; } /** * Gets the range of combined data ranges provided in a form of an array of arrays ([row, column, dataUnderCollection]). * * @private * @param {Array} populationArgumentsList Array containing argument lists for the `populateFromArray` method - row, column and data for population. * @returns {Array[]} Start and end coordinates of the merged cell range. (in a form of [rowIndex, columnIndex]). */ }, { key: "getBulkCollectionDataRange", value: function getBulkCollectionDataRange(populationArgumentsList) { var start = [0, 0]; var end = [0, 0]; var mergedCellRow = null; var mergedCellColumn = null; var mergedCellData = null; (0, _array.arrayEach)(populationArgumentsList, function (mergedCellArguments) { mergedCellRow = mergedCellArguments[0]; mergedCellColumn = mergedCellArguments[1]; mergedCellData = mergedCellArguments[2]; start[0] = Math.min(mergedCellRow, start[0]); start[1] = Math.min(mergedCellColumn, start[1]); end[0] = Math.max(mergedCellRow + mergedCellData.length - 1, end[0]); end[1] = Math.max(mergedCellColumn + mergedCellData[0].length - 1, end[1]); }); return [].concat(start, end); } /** * Clears the merged cells from the merged cell container. */ }, { key: "clearCollections", value: function clearCollections() { this.mergedCellsCollection.clear(); } /** * Returns `true` if a range is mergeable. * * @private * @param {object} newMergedCellInfo Merged cell information object to test. * @param {boolean} [auto=false] `true` if triggered at initialization. * @returns {boolean} */ }, { key: "canMergeRange", value: function canMergeRange(newMergedCellInfo) { var auto = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false; return auto ? true : this.validateSetting(newMergedCellInfo); } /** * Merge or unmerge, based on last selected range. * * @private */ }, { key: "toggleMergeOnSelection", value: function toggleMergeOnSelection() { var currentRange = this.hot.getSelectedRangeLast(); if (!currentRange) { return; } currentRange.setDirection('NW-SE'); var from = currentRange.from, to = currentRange.to; this.toggleMerge(currentRange); this.hot.selectCell(from.row, from.col, to.row, to.col, false); } /** * Merges the selection provided as a cell range. * * @param {CellRange} [cellRange] Selection cell range. */ }, { key: "mergeSelection", value: function mergeSelection() { var cellRange = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : this.hot.getSelectedRangeLast(); if (!cellRange) { return; } cellRange.setDirection('NW-SE'); var from = cellRange.from, to = cellRange.to; this.unmergeRange(cellRange, true); this.mergeRange(cellRange); this.hot.selectCell(from.row, from.col, to.row, to.col, false); } /** * Unmerges the selection provided as a cell range. * * @param {CellRange} [cellRange] Selection cell range. */ }, { key: "unmergeSelection", value: function unmergeSelection() { var cellRange = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : this.hot.getSelectedRangeLast(); if (!cellRange) { return; } var from = cellRange.from, to = cellRange.to; this.unmergeRange(cellRange, true); this.hot.selectCell(from.row, from.col, to.row, to.col, false); } /** * Merges cells in the provided cell range. * * @private * @param {CellRange} cellRange Cell range to merge. * @param {boolean} [auto=false] `true` if is called automatically, e.g. At initialization. * @param {boolean} [preventPopulation=false] `true`, if the method should not run `populateFromArray` at the end, but rather return its arguments. * @returns {Array|boolean} Returns an array of [row, column, dataUnderCollection] if preventPopulation is set to true. If the the merging process went successful, it returns `true`, otherwise - `false`. * @fires Hooks#beforeMergeCells * @fires Hooks#afterMergeCells */ }, { key: "mergeRange", value: function mergeRange(cellRange) { var _this4 = this; var auto = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false; var preventPopulation = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false; var topLeft = cellRange.getTopLeftCorner(); var bottomRight = cellRange.getBottomRightCorner(); var mergeParent = { row: topLeft.row, col: topLeft.col, rowspan: bottomRight.row - topLeft.row + 1, colspan: bottomRight.col - topLeft.col + 1 }; var clearedData = []; var populationInfo = null; if (!this.canMergeRange(mergeParent, auto)) { return false; } this.hot.runHooks('beforeMergeCells', cellRange, auto); (0, _number.rangeEach)(0, mergeParent.rowspan - 1, function (i) { (0, _number.rangeEach)(0, mergeParent.colspan - 1, function (j) { var clearedValue = null; if (!clearedData[i]) { clearedData[i] = []; } if (i === 0 && j === 0) { clearedValue = _this4.hot.getDataAtCell(mergeParent.row, mergeParent.col); } else { _this4.hot.setCellMeta(mergeParent.row + i, mergeParent.col + j, 'hidden', true); } clearedData[i][j] = clearedValue; }); }); this.hot.setCellMeta(mergeParent.row, mergeParent.col, 'spanned', true); var mergedCellAdded = this.mergedCellsCollection.add(mergeParent); if (mergedCellAdded) { if (preventPopulation) { populationInfo = [mergeParent.row, mergeParent.col, clearedData]; } else { this.hot.populateFromArray(mergeParent.row, mergeParent.col, clearedData, void 0, void 0, this.pluginName); } this.hot.runHooks('afterMergeCells', cellRange, mergeParent, auto); return populationInfo; } return true; } /** * Unmerges the selection provided as a cell range. If no cell range is provided, it uses the current selection. * * @private * @param {CellRange} cellRange Selection cell range. * @param {boolean} [auto=false] `true` if called automatically by the plugin. * * @fires Hooks#beforeUnmergeCells * @fires Hooks#afterUnmergeCells */ }, { key: "unmergeRange", value: function unmergeRange(cellRange) { var _this5 = this; var auto = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false; var mergedCells = this.mergedCellsCollection.getWithinRange(cellRange); if (!mergedCells) { return; } this.hot.runHooks('beforeUnmergeCells', cellRange, auto); (0, _array.arrayEach)(mergedCells, function (currentCollection) { _this5.mergedCellsCollection.remove(currentCollection.row, currentCollection.col); (0, _number.rangeEach)(0, currentCollection.rowspan - 1, function (i) { (0, _number.rangeEach)(0, currentCollection.colspan - 1, function (j) { _this5.hot.removeCellMeta(currentCollection.row + i, currentCollection.col + j, 'hidden'); }); }); _this5.hot.removeCellMeta(currentCollection.row, currentCollection.col, 'spanned'); }); this.hot.runHooks('afterUnmergeCells', cellRange, auto); this.hot.render(); } /** * Merges or unmerges, based on the cell range provided as `cellRange`. * * @private * @param {CellRange} cellRange The cell range to merge or unmerged. */ }, { key: "toggleMerge", value: function toggleMerge(cellRange) { var mergedCell = this.mergedCellsCollection.get(cellRange.from.row, cellRange.from.col); var mergedCellCoversWholeRange = mergedCell.row === cellRange.from.row && mergedCell.col === cellRange.from.col && mergedCell.row + mergedCell.rowspan - 1 === cellRange.to.row && mergedCell.col + mergedCell.colspan - 1 === cellRange.to.col; if (mergedCellCoversWholeRange) { this.unmergeRange(cellRange); } else { this.mergeSelection(cellRange); } } /** * Merges the specified range. * * @param {number} startRow Start row of the merged cell. * @param {number} startColumn Start column of the merged cell. * @param {number} endRow End row of the merged cell. * @param {number} endColumn End column of the merged cell. * @fires Hooks#beforeMergeCells * @fires Hooks#afterMergeCells */ }, { key: "merge", value: function merge(startRow, startColumn, endRow, endColumn) { var start = new _src.CellCoords(startRow, startColumn); var end = new _src.CellCoords(endRow, endColumn); this.mergeRange(new _src.CellRange(start, start, end)); } /** * Unmerges the merged cell in the provided range. * * @param {number} startRow Start row of the merged cell. * @param {number} startColumn Start column of the merged cell. * @param {number} endRow End row of the merged cell. * @param {number} endColumn End column of the merged cell. * @fires Hooks#beforeUnmergeCells * @fires Hooks#afterUnmergeCells */ }, { key: "unmerge", value: function unmerge(startRow, startColumn, endRow, endColumn) { var start = new _src.CellCoords(startRow, startColumn); var end = new _src.CellCoords(endRow, endColumn); this.unmergeRange(new _src.CellRange(start, start, end)); } /** * `afterInit` hook callback. * * @private */ }, { key: "onAfterInit", value: function onAfterInit() { this.generateFromSettings(this.hot.getSettings()[PLUGIN_KEY]); this.hot.render(); } /** * `beforeKeyDown` hook callback. * * @private * @param {KeyboardEvent} event The `keydown` event object. */ }, { key: "onBeforeKeyDown", value: function onBeforeKeyDown(event) { var ctrlDown = (event.ctrlKey || event.metaKey) && !event.altKey; if (ctrlDown && event.keyCode === 77) { // CTRL + M this.toggleMerge(this.hot.getSelectedRangeLast()); this.hot.render(); (0, _event.stopImmediatePropagation)(event); } } /** * Modifies the information on whether the current selection contains multiple cells. The `afterIsMultipleSelection` hook callback. * * @private * @param {boolean} isMultiple Determines whether the current selection contains multiple cells. * @returns {boolean} */ }, { key: "onAfterIsMultipleSelection", value: function onAfterIsMultipleSelection(isMultiple) { if (isMultiple) { var mergedCells = this.mergedCellsCollection.mergedCells; var selectionRange = this.hot.getSelectedRangeLast(); for (var group = 0; group < mergedCells.length; group += 1) { if (selectionRange.from.row === mergedCells[group].row && selectionRange.from.col === mergedCells[group].col && selectionRange.to.row === mergedCells[group].row + mergedCells[group].rowspan - 1 && selectionRange.to.col === mergedCells[group].col + mergedCells[group].colspan - 1) { return false; } } } return isMultiple; } /** * `modifyTransformStart` hook callback. * * @private * @param {object} delta The transformation delta. */ }, { key: "onModifyTransformStart", value: function onModifyTransformStart(delta) { var priv = privatePool.get(this); var currentlySelectedRange = this.hot.getSelectedRangeLast(); var newDelta = { row: delta.row, col: delta.col }; var nextPosition = null; var currentPosition = new _src.CellCoords(currentlySelectedRange.highlight.row, currentlySelectedRange.highlight.col); var mergedParent = this.mergedCellsCollection.get(currentPosition.row, currentPosition.col); if (!priv.lastDesiredCoords) { priv.lastDesiredCoords = new _src.CellCoords(null, null); } if (mergedParent) { // only merge selected var mergeTopLeft = new _src.CellCoords(mergedParent.row, mergedParent.col); var mergeBottomRight = new _src.CellCoords(mergedParent.row + mergedParent.rowspan - 1, mergedParent.col + mergedParent.colspan - 1); var mergeRange = new _src.CellRange(mergeTopLeft, mergeTopLeft, mergeBottomRight); if (!mergeRange.includes(priv.lastDesiredCoords)) { priv.lastDesiredCoords = new _src.CellCoords(null, null); // reset outdated version of lastDesiredCoords } newDelta.row = priv.lastDesiredCoords.row ? priv.lastDesiredCoords.row - currentPosition.row : newDelta.row; newDelta.col = priv.lastDesiredCoords.col ? priv.lastDesiredCoords.col - currentPosition.col : newDelta.col; if (delta.row > 0) { // moving down newDelta.row = mergedParent.row + mergedParent.rowspan - 1 - currentPosition.row + delta.row; } else if (delta.row < 0) { // moving up newDelta.row = currentPosition.row - mergedParent.row + delta.row; } if (delta.col > 0) { // moving right newDelta.col = mergedParent.col + mergedParent.colspan - 1 - currentPosition.col + delta.col; } else if (delta.col < 0) { // moving left newDelta.col = currentPosition.col - mergedParent.col + delta.col; } } nextPosition = new _src.CellCoords(currentlySelectedRange.highlight.row + newDelta.row, currentlySelectedRange.highlight.col + newDelta.col); var nextPositionMergedCell = this.mergedCellsCollection.get(nextPosition.row, nextPosition.col); if (nextPositionMergedCell) { // skipping the invisible cells in the merge range var firstRenderableCoords = this.mergedCellsCollection.getFirstRenderableCoords(nextPositionMergedCell.row, nextPositionMergedCell.col); priv.lastDesiredCoords = nextPosition; newDelta = { row: firstRenderableCoords.row - currentPosition.row, col: firstRenderableCoords.col - currentPosition.col }; } if (newDelta.row !== 0) { delta.row = newDelta.row; } if (newDelta.col !== 0) { delta.col = newDelta.col; } } /** * `modifyTransformEnd` hook callback. Needed to handle "jumping over" merged merged cells, while selecting. * * @private * @param {object} delta The transformation delta. */ }, { key: "onModifyTransformEnd", value: function onModifyTransformEnd(delta) { var _this6 = this; var currentSelectionRange = this.hot.getSelectedRangeLast(); var newDelta = (0, _object.clone)(delta); var newSelectionRange = this.selectionCalculations.getUpdatedSelectionRange(currentSelectionRange, delta); var tempDelta = (0, _object.clone)(newDelta); var mergedCellsWithinRange = this.mergedCellsCollection.getWithinRange(newSelectionRange, true); do { tempDelta = (0, _object.clone)(newDelta); this.selectionCalculations.getUpdatedSelectionRange(currentSelectionRange, newDelta); (0, _array.arrayEach)(mergedCellsWithinRange, function (mergedCell) { _this6.selectionCalculations.snapDelta(newDelta, currentSelectionRange, mergedCell); }); } while (newDelta.row !== tempDelta.row || newDelta.col !== tempDelta.col); delta.row = newDelta.row; delta.col = newDelta.col; } /** * `modifyGetCellCoords` hook callback. Swaps the `getCell` coords with the merged parent coords. * * @private * @param {number} row Row index. * @param {number} column Visual column index. * @returns {Array|undefined} Visual coordinates of the merge. */ }, { key: "onModifyGetCellCoords", value: function onModifyGetCellCoords(row, column) { if (row < 0 || column < 0) { return; } var mergeParent = this.mergedCellsCollection.get(row, column); if (!mergeParent) { return; } var mergeRow = mergeParent.row, mergeColumn = mergeParent.col, colspan = mergeParent.colspan, rowspan = mergeParent.rowspan; return [// Most top-left merged cell coords. mergeRow, mergeColumn, // Most bottom-right merged cell coords. mergeRow + rowspan - 1, mergeColumn + colspan - 1]; } /** * `afterContextMenuDefaultOptions` hook callback. * * @private * @param {object} defaultOptions The default context menu options. */ }, { key: "addMergeActionsToContextMenu", value: function addMergeActionsToContextMenu(defaultOptions) { defaultOptions.items.push({ name: '---------' }, (0, _toggleMerge.default)(this)); } /** * `afterRenderer` hook callback. * * @private * @param {HTMLElement} TD The cell to be modified. * @param {number} row Row index. * @param {number} col Visual column index. */ }, { key: "onAfterRenderer", value: function onAfterRenderer(TD, row, col) { var mergedCell = this.mergedCellsCollection.get(row, col); // We shouldn't override data in the collection. var mergedCellCopy = (0, _object.isObject)(mergedCell) ? (0, _object.clone)(mergedCell) : void 0; if ((0, _object.isObject)(mergedCellCopy)) { var _this$hot3 = this.hot, rowMapper = _this$hot3.rowIndexMapper, columnMapper = _this$hot3.columnIndexMapper; var mergeRow = mergedCellCopy.row, mergeColumn = mergedCellCopy.col, colspan = mergedCellCopy.colspan, rowspan = mergedCellCopy.rowspan; var _this$translateMerged = this.translateMergedCellToRenderable(mergeRow, rowspan, mergeColumn, colspan), _this$translateMerged2 = _slicedToArray(_this$translateMerged, 2), lastMergedRowIndex = _this$translateMerged2[0], lastMergedColumnIndex = _this$translateMerged2[1]; var renderedRowIndex = rowMapper.getRenderableFromVisualIndex(row); var renderedColumnIndex = columnMapper.getRenderableFromVisualIndex(col); var maxRowSpan = lastMergedRowIndex - renderedRowIndex + 1; // Number of rendered columns. var maxColSpan = lastMergedColumnIndex - renderedColumnIndex + 1; // Number of rendered columns. // We just try to determine some values basing on the actual number of rendered indexes (some columns may be hidden). mergedCellCopy.row = rowMapper.getFirstNotHiddenIndex(mergedCellCopy.row, 1); // We just try to determine some values basing on the actual number of rendered indexes (some columns may be hidden). mergedCellCopy.col = columnMapper.getFirstNotHiddenIndex(mergedCellCopy.col, 1); // The `rowSpan` property for a `TD` element should be at most equal to number of rendered rows in the merge area. mergedCellCopy.rowspan = Math.min(mergedCellCopy.rowspan, maxRowSpan); // The `colSpan` property for a `TD` element should be at most equal to number of rendered columns in the merge area. mergedCellCopy.colspan = Math.min(mergedCellCopy.colspan, maxColSpan); } (0, _utils.applySpanProperties)(TD, mergedCellCopy, row, col); } /** * `beforeSetRangeStart` and `beforeSetRangeStartOnly` hook callback. * A selection within merge area should be rewritten to the start of merge area. * * @private * @param {object} coords Cell coords. */ }, { key: "onBeforeSetRangeStart", value: function onBeforeSetRangeStart(coords) { // TODO: It is a workaround, but probably this hook may be needed. Every selection on the merge area // could set start point of the selection to the start of the merge area. However, logic inside `expandByRange` need // an initial start point. Click on the merge cell when there are some hidden indexes break the logic in some cases. // Please take a look at #7010 for more information. I'm not sure if selection directions are calculated properly // and what was idea for flipping direction inside `expandByRange` method. if (this.mergedCellsCollection.isFirstRenderableMergedCell(coords.row, coords.col)) { var mergeParent = this.mergedCellsCollection.get(coords.row, coords.col); var _ref = [mergeParent.row, mergeParent.col]; coords.row = _ref[0]; coords.col = _ref[1]; } } /** * `beforeSetRangeEnd` hook callback. * While selecting cells with keyboard or mouse, make sure that rectangular area is expanded to the extent of the merged cell. * * Note: Please keep in mind that callback may modify both start and end range coordinates by the reference. * * @private * @param {object} coords Cell coords. */ }, { key: "onBeforeSetRangeEnd", value: function onBeforeSetRangeEnd(coords) { var selRange = this.hot.getSelectedRangeLast(); selRange.highlight = new _src.CellCoords(selRange.highlight.row, selRange.highlight.col); // clone in case we will modify its reference selRange.to = coords; var rangeExpanded = false; if (this.hot.selection.isSelectedByColumnHeader() || this.hot.selection.isSelectedByRowHeader()) { return; } do { rangeExpanded = false; for (var i = 0; i < this.mergedCellsCollection.mergedCells.length; i += 1) { var cellInfo = this.mergedCellsCollection.mergedCells[i]; var mergedCellRange = cellInfo.getRange(); if (selRange.expandByRange(mergedCellRange)) { coords.row = selRange.to.row; coords.col = selRange.to.col; rangeExpanded = true; } } } while (rangeExpanded); } /** * The `afterGetCellMeta` hook callback. * * @private * @param {number} row Row index. * @param {number} col Column index. * @param {object} cellProperties The cell properties object. */ }, { key: "onAfterGetCellMeta", value: function onAfterGetCellMeta(row, col, cellProperties) { var mergeParent = this.mergedCellsCollection.get(row, col); if (mergeParent) { if (mergeParent.row !== row || mergeParent.col !== col) { cellProperties.copyable = false; } else { cellProperties.rowspan = mergeParent.rowspan; cellProperties.colspan = mergeParent.colspan; } } } /** * `afterViewportRowCalculatorOverride` hook callback. * * @private * @param {object} calc The row calculator object. */ }, { key: "onAfterViewportRowCalculatorOverride", value: function onAfterViewportRowCalculatorOverride(calc) { var nrOfColumns = this.hot.countCols(); this.modifyViewportRowStart(calc, nrOfColumns); this.modifyViewportRowEnd(calc, nrOfColumns); } /** * Modify viewport start when needed. We extend viewport when merged cells aren't fully visible. * * @private * @param {object} calc The row calculator object. * @param {number} nrOfColumns Number of visual columns. */ }, { key: "modifyViewportRowStart", value: function modifyViewportRowStart(calc, nrOfColumns) { var rowMapper = this.hot.rowIndexMapper; var visualStartRow = rowMapper.getVisualFromRenderableIndex(calc.startRow); for (var visualColumnIndex = 0; visualColumnIndex < nrOfColumns; visualColumnIndex += 1) { var mergeParentForViewportStart = this.mergedCellsCollection.get(visualStartRow, visualColumnIndex); if ((0, _object.isObject)(mergeParentForViewportStart)) { var renderableIndexAtMergeStart = rowMapper.getRenderableFromVisualIndex(rowMapper.getFirstNotHiddenIndex(mergeParentForViewportStart.row, 1)); // Merge start is out of the viewport (i.e. when we scrolled to the bottom and we can see just part of a merge). if (renderableIndexAtMergeStart < calc.startRow) { // We extend viewport when some rows have been merged. calc.startRow = renderableIndexAtMergeStart; // We are looking for next merges inside already extended viewport (starting again from row equal to 0). this.modifyViewportRowStart(calc, nrOfColumns); // recursively search upwards return; // Finish the current loop. Everything will be checked from the beginning by above recursion. } } } } /** * Modify viewport end when needed. We extend viewport when merged cells aren't fully visible. * * @private * @param {object} calc The row calculator object. * @param {number} nrOfColumns Number of visual columns. */ }, { key: "modifyViewportRowEnd", value: function modifyViewportRowEnd(calc, nrOfColumns) { var rowMapper = this.hot.rowIndexMapper; var visualEndRow = rowMapper.getVisualFromRenderableIndex(calc.endRow); for (var visualColumnIndex = 0; visualColumnIndex < nrOfColumns; visualColumnIndex += 1) { var mergeParentForViewportEnd = this.mergedCellsCollection.get(visualEndRow, visualColumnIndex); if ((0, _object.isObject)(mergeParentForViewportEnd)) { var mergeEnd = mergeParentForViewportEnd.row + mergeParentForViewportEnd.rowspan - 1; var renderableIndexAtMergeEnd = rowMapper.getRenderableFromVisualIndex(rowMapper.getFirstNotHiddenIndex(mergeEnd, -1)); // Merge end is out of the viewport. if (renderableIndexAtMergeEnd > calc.endRow) { // We extend the viewport when some rows have been merged. calc.endRow = renderableIndexAtMergeEnd; // We are looking for next merges inside already extended viewport (starting again from row equal to 0). this.modifyViewportRowEnd(calc, nrOfColumns); // recursively search upwards return; // Finish the current loop. Everything will be checked from the beginning by above recursion. } } } } /** * `afterViewportColumnCalculatorOverride` hook callback. * * @private * @param {object} calc The column calculator object. */ }, { key: "onAfterViewportColumnCalculatorOverride", value: function onAfterViewportColumnCalculatorOverride(calc) { var nrOfRows = this.hot.countRows(); this.modifyViewportColumnStart(calc, nrOfRows); this.modifyViewportColumnEnd(calc, nrOfRows); } /** * Modify viewport start when needed. We extend viewport when merged cells aren't fully visible. * * @private * @param {object} calc The column calculator object. * @param {number} nrOfRows Number of visual rows. */ }, { key: "modifyViewportColumnStart", value: function modifyViewportColumnStart(calc, nrOfRows) { var columnMapper = this.hot.columnIndexMapper; var visualStartCol = columnMapper.getVisualFromRenderableIndex(calc.startColumn); for (var visualRowIndex = 0; visualRowIndex < nrOfRows; visualRowIndex += 1) { var mergeParentForViewportStart = this.mergedCellsCollection.get(visualRowIndex, visualStartCol); if ((0, _object.isObject)(mergeParentForViewportStart)) { var renderableIndexAtMergeStart = columnMapper.getRenderableFromVisualIndex(columnMapper.getFirstNotHiddenIndex(mergeParentForViewportStart.col, 1)); // Merge start is out of the viewport (i.e. when we scrolled to the right and we can see just part of a merge). if (renderableIndexAtMergeStart < calc.startColumn) { // We extend viewport when some columns have been merged. calc.startColumn = renderableIndexAtMergeStart; // We are looking for next merges inside already extended viewport (starting again from column equal to 0). this.modifyViewportColumnStart(calc, nrOfRows); // recursively search upwards return; // Finish the current loop. Everything will be checked from the beginning by above recursion. } } } } /** * Modify viewport end when needed. We extend viewport when merged cells aren't fully visible. * * @private * @param {object} calc The column calculator object. * @param {number} nrOfRows Number of visual rows. */ }, { key: "modifyViewportColumnEnd", value: function modifyViewportColumnEnd(calc, nrOfRows) { var columnMapper = this.hot.columnIndexMapper; var visualEndCol = columnMapper.getVisualFromRenderableIndex(calc.endColumn); for (var visualRowIndex = 0; visualRowIndex < nrOfRows; visualRowIndex += 1) { var mergeParentForViewportEnd = this.mergedCellsCollection.get(visualRowIndex, visualEndCol); if ((0, _object.isObject)(mergeParentForViewportEnd)) { var mergeEnd = mergeParentForViewportEnd.col + mergeParentForViewportEnd.colspan - 1; var renderableIndexAtMergeEnd = columnMapper.getRenderableFromVisualIndex(columnMapper.getFirstNotHiddenIndex(mergeEnd, -1)); // Merge end is out of the viewport. if (renderableIndexAtMergeEnd > calc.endColumn) { // We extend the viewport when some columns have been merged. calc.endColumn = renderableIndexAtMergeEnd; // We are looking for next merges inside already extended viewport (starting again from column equal to 0). this.modifyViewportColumnEnd(calc, nrOfRows); // recursively search upwards return; // Finish the current loop. Everything will be checked from the beginning by above recursion. } } } } /** * Translates merged cell coordinates to renderable indexes. * * @private * @param {number} parentRow Visual row index. * @param {number} rowspan Rowspan which describes shift which will be applied to parent row * to calculate renderable index which points to the most bottom * index position. Pass rowspan as `0` to calculate the most top * index position. * @param {number} parentColumn Visual column index. * @param {number} colspan Colspan which describes shift which will be applied to parent column * to calculate renderable index which points to the most right * index position. Pass colspan as `0` to calculate the most left * index position.