123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371 |
- import { addClass, getCaretPosition, getComputedStyle, getCssTransform, getScrollableElement, getScrollbarWidth, innerWidth, offset, resetCssTransform, setCaretPosition, hasVerticalScrollbar, hasHorizontalScrollbar } from './../helpers/dom/element';
- import autoResize from './../../lib/autoResize/autoResize';
- import BaseEditor, { EditorState } from './_baseEditor';
- import EventManager from './../eventManager';
- import { KEY_CODES } from './../helpers/unicode';
- import { stopPropagation, stopImmediatePropagation, isImmediatePropagationStopped } from './../helpers/dom/event';
- var TextEditor = BaseEditor.prototype.extend();
- /**
- * @private
- * @editor TextEditor
- * @class TextEditor
- * @dependencies autoResize
- */
- TextEditor.prototype.init = function () {
- var that = this;
- this.createElements();
- this.eventManager = new EventManager(this);
- this.bindEvents();
- this.autoResize = autoResize();
- this.instance.addHook('afterDestroy', function () {
- that.destroy();
- });
- };
- TextEditor.prototype.getValue = function () {
- return this.TEXTAREA.value;
- };
- TextEditor.prototype.setValue = function (newValue) {
- this.TEXTAREA.value = newValue;
- };
- var onBeforeKeyDown = function onBeforeKeyDown(event) {
- var instance = this,
- that = instance.getActiveEditor(),
- ctrlDown;
- // catch CTRL but not right ALT (which in some systems triggers ALT+CTRL)
- ctrlDown = (event.ctrlKey || event.metaKey) && !event.altKey;
- // Process only events that have been fired in the editor
- if (event.target !== that.TEXTAREA || isImmediatePropagationStopped(event)) {
- return;
- }
- if (event.keyCode === 17 || event.keyCode === 224 || event.keyCode === 91 || event.keyCode === 93) {
- // when CTRL or its equivalent is pressed and cell is edited, don't prepare selectable text in textarea
- stopImmediatePropagation(event);
- return;
- }
- switch (event.keyCode) {
- case KEY_CODES.ARROW_RIGHT:
- if (that.isInFullEditMode()) {
- if (!that.isWaiting() && !that.allowKeyEventPropagation || !that.isWaiting() && that.allowKeyEventPropagation && !that.allowKeyEventPropagation(event.keyCode)) {
- stopImmediatePropagation(event);
- }
- }
- break;
- case KEY_CODES.ARROW_LEFT:
- if (that.isInFullEditMode()) {
- if (!that.isWaiting() && !that.allowKeyEventPropagation || !that.isWaiting() && that.allowKeyEventPropagation && !that.allowKeyEventPropagation(event.keyCode)) {
- stopImmediatePropagation(event);
- }
- }
- break;
- case KEY_CODES.ARROW_UP:
- case KEY_CODES.ARROW_DOWN:
- if (that.isInFullEditMode()) {
- if (!that.isWaiting() && !that.allowKeyEventPropagation || !that.isWaiting() && that.allowKeyEventPropagation && !that.allowKeyEventPropagation(event.keyCode)) {
- stopImmediatePropagation(event);
- }
- }
- break;
- case KEY_CODES.ENTER:
- var selected = that.instance.getSelected();
- var isMultipleSelection = !(selected[0] === selected[2] && selected[1] === selected[3]);
- if (ctrlDown && !isMultipleSelection || event.altKey) {
- // if ctrl+enter or alt+enter, add new line
- if (that.isOpened()) {
- var caretPosition = getCaretPosition(that.TEXTAREA),
- value = that.getValue();
- var newValue = value.slice(0, caretPosition) + '\n' + value.slice(caretPosition);
- that.setValue(newValue);
- setCaretPosition(that.TEXTAREA, caretPosition + 1);
- } else {
- that.beginEditing(that.originalValue + '\n');
- }
- stopImmediatePropagation(event);
- }
- event.preventDefault(); // don't add newline to field
- break;
- case KEY_CODES.A:
- case KEY_CODES.X:
- case KEY_CODES.C:
- case KEY_CODES.V:
- if (ctrlDown) {
- stopImmediatePropagation(event); // CTRL+A, CTRL+C, CTRL+V, CTRL+X should only work locally when cell is edited (not in table context)
- }
- break;
- case KEY_CODES.BACKSPACE:
- case KEY_CODES.DELETE:
- case KEY_CODES.HOME:
- case KEY_CODES.END:
- stopImmediatePropagation(event); // backspace, delete, home, end should only work locally when cell is edited (not in table context)
- break;
- default:
- break;
- }
- if ([KEY_CODES.ARROW_UP, KEY_CODES.ARROW_RIGHT, KEY_CODES.ARROW_DOWN, KEY_CODES.ARROW_LEFT].indexOf(event.keyCode) === -1) {
- that.autoResize.resize(String.fromCharCode(event.keyCode));
- }
- };
- TextEditor.prototype.open = function () {
- this.refreshDimensions(); // need it instantly, to prevent https://github.com/handsontable/handsontable/issues/348
- this.instance.addHook('beforeKeyDown', onBeforeKeyDown);
- };
- TextEditor.prototype.close = function (tdOutside) {
- this.textareaParentStyle.display = 'none';
- this.autoResize.unObserve();
- if (document.activeElement === this.TEXTAREA) {
- this.instance.listen(); // don't refocus the table if user focused some cell outside of HT on purpose
- }
- this.instance.removeHook('beforeKeyDown', onBeforeKeyDown);
- };
- TextEditor.prototype.focus = function () {
- this.TEXTAREA.focus();
- setCaretPosition(this.TEXTAREA, this.TEXTAREA.value.length);
- };
- TextEditor.prototype.createElements = function () {
- // this.$body = $(document.body);
- this.TEXTAREA = document.createElement('TEXTAREA');
- addClass(this.TEXTAREA, 'handsontableInput');
- this.textareaStyle = this.TEXTAREA.style;
- this.textareaStyle.width = 0;
- this.textareaStyle.height = 0;
- this.TEXTAREA_PARENT = document.createElement('DIV');
- addClass(this.TEXTAREA_PARENT, 'handsontableInputHolder');
- this.textareaParentStyle = this.TEXTAREA_PARENT.style;
- this.textareaParentStyle.top = 0;
- this.textareaParentStyle.left = 0;
- this.textareaParentStyle.display = 'none';
- this.TEXTAREA_PARENT.appendChild(this.TEXTAREA);
- this.instance.rootElement.appendChild(this.TEXTAREA_PARENT);
- var that = this;
- this.instance._registerTimeout(setTimeout(function () {
- that.refreshDimensions();
- }, 0));
- };
- TextEditor.prototype.getEditedCell = function () {
- var editorSection = this.checkEditorSection(),
- editedCell;
- switch (editorSection) {
- case 'top':
- editedCell = this.instance.view.wt.wtOverlays.topOverlay.clone.wtTable.getCell({
- row: this.row,
- col: this.col
- });
- this.textareaParentStyle.zIndex = 101;
- break;
- case 'top-left-corner':
- editedCell = this.instance.view.wt.wtOverlays.topLeftCornerOverlay.clone.wtTable.getCell({
- row: this.row,
- col: this.col
- });
- this.textareaParentStyle.zIndex = 103;
- break;
- case 'bottom-left-corner':
- editedCell = this.instance.view.wt.wtOverlays.bottomLeftCornerOverlay.clone.wtTable.getCell({
- row: this.row,
- col: this.col
- });
- this.textareaParentStyle.zIndex = 103;
- break;
- case 'left':
- editedCell = this.instance.view.wt.wtOverlays.leftOverlay.clone.wtTable.getCell({
- row: this.row,
- col: this.col
- });
- this.textareaParentStyle.zIndex = 102;
- break;
- case 'bottom':
- editedCell = this.instance.view.wt.wtOverlays.bottomOverlay.clone.wtTable.getCell({
- row: this.row,
- col: this.col
- });
- this.textareaParentStyle.zIndex = 102;
- break;
- default:
- editedCell = this.instance.getCell(this.row, this.col);
- this.textareaParentStyle.zIndex = '';
- break;
- }
- return editedCell != -1 && editedCell != -2 ? editedCell : void 0;
- };
- TextEditor.prototype.refreshValue = function () {
- var sourceData = this.instance.getSourceDataAtCell(this.row, this.prop);
- this.originalValue = sourceData;
- this.setValue(sourceData);
- this.refreshDimensions();
- };
- TextEditor.prototype.refreshDimensions = function () {
- if (this.state !== EditorState.EDITING) {
- return;
- }
- this.TD = this.getEditedCell();
- // TD is outside of the viewport.
- if (!this.TD) {
- this.close(true);
- return;
- }
- var currentOffset = offset(this.TD),
- containerOffset = offset(this.instance.rootElement),
- scrollableContainer = getScrollableElement(this.TD),
- totalRowsCount = this.instance.countRows(),
- // If colHeaders is disabled, cells in the first row have border-top
- editTopModifier = currentOffset.top === containerOffset.top ? 0 : 1,
- editTop = currentOffset.top - containerOffset.top - editTopModifier - (scrollableContainer.scrollTop || 0),
- editLeft = currentOffset.left - containerOffset.left - 1 - (scrollableContainer.scrollLeft || 0),
- settings = this.instance.getSettings(),
- rowHeadersCount = this.instance.hasRowHeaders(),
- colHeadersCount = this.instance.hasColHeaders(),
- editorSection = this.checkEditorSection(),
- backgroundColor = this.TD.style.backgroundColor,
- cssTransformOffset;
- // TODO: Refactor this to the new instance.getCell method (from #ply-59), after 0.12.1 is released
- switch (editorSection) {
- case 'top':
- cssTransformOffset = getCssTransform(this.instance.view.wt.wtOverlays.topOverlay.clone.wtTable.holder.parentNode);
- break;
- case 'left':
- cssTransformOffset = getCssTransform(this.instance.view.wt.wtOverlays.leftOverlay.clone.wtTable.holder.parentNode);
- break;
- case 'top-left-corner':
- cssTransformOffset = getCssTransform(this.instance.view.wt.wtOverlays.topLeftCornerOverlay.clone.wtTable.holder.parentNode);
- break;
- case 'bottom-left-corner':
- cssTransformOffset = getCssTransform(this.instance.view.wt.wtOverlays.bottomLeftCornerOverlay.clone.wtTable.holder.parentNode);
- break;
- case 'bottom':
- cssTransformOffset = getCssTransform(this.instance.view.wt.wtOverlays.bottomOverlay.clone.wtTable.holder.parentNode);
- break;
- default:
- break;
- }
- if (colHeadersCount && this.instance.getSelected()[0] === 0 || settings.fixedRowsBottom && this.instance.getSelected()[0] === totalRowsCount - settings.fixedRowsBottom) {
- editTop += 1;
- }
- if (this.instance.getSelected()[1] === 0) {
- editLeft += 1;
- }
- if (cssTransformOffset && cssTransformOffset != -1) {
- this.textareaParentStyle[cssTransformOffset[0]] = cssTransformOffset[1];
- } else {
- resetCssTransform(this.TEXTAREA_PARENT);
- }
- this.textareaParentStyle.top = editTop + 'px';
- this.textareaParentStyle.left = editLeft + 'px';
- var firstRowOffset = this.instance.view.wt.wtViewport.rowsRenderCalculator.startPosition;
- var firstColumnOffset = this.instance.view.wt.wtViewport.columnsRenderCalculator.startPosition;
- var horizontalScrollPosition = this.instance.view.wt.wtOverlays.leftOverlay.getScrollPosition();
- var verticalScrollPosition = this.instance.view.wt.wtOverlays.topOverlay.getScrollPosition();
- var scrollbarWidth = getScrollbarWidth();
- var cellTopOffset = this.TD.offsetTop + firstRowOffset - verticalScrollPosition;
- var cellLeftOffset = this.TD.offsetLeft + firstColumnOffset - horizontalScrollPosition;
- var width = innerWidth(this.TD) - 8;
- var actualVerticalScrollbarWidth = hasVerticalScrollbar(scrollableContainer) ? scrollbarWidth : 0;
- var actualHorizontalScrollbarWidth = hasHorizontalScrollbar(scrollableContainer) ? scrollbarWidth : 0;
- var maxWidth = this.instance.view.maximumVisibleElementWidth(cellLeftOffset) - 9 - actualVerticalScrollbarWidth;
- var height = this.TD.scrollHeight + 1;
- var maxHeight = Math.max(this.instance.view.maximumVisibleElementHeight(cellTopOffset) - actualHorizontalScrollbarWidth, 23);
- var cellComputedStyle = getComputedStyle(this.TD);
- this.TEXTAREA.style.fontSize = cellComputedStyle.fontSize;
- this.TEXTAREA.style.fontFamily = cellComputedStyle.fontFamily;
- this.TEXTAREA.style.backgroundColor = ''; // RESET STYLE
- this.TEXTAREA.style.backgroundColor = backgroundColor ? backgroundColor : getComputedStyle(this.TEXTAREA).backgroundColor;
- this.autoResize.init(this.TEXTAREA, {
- minHeight: Math.min(height, maxHeight),
- maxHeight: maxHeight, // TEXTAREA should never be wider than visible part of the viewport (should not cover the scrollbar)
- minWidth: Math.min(width, maxWidth),
- maxWidth: maxWidth // TEXTAREA should never be wider than visible part of the viewport (should not cover the scrollbar)
- }, true);
- this.textareaParentStyle.display = 'block';
- };
- TextEditor.prototype.bindEvents = function () {
- var editor = this;
- this.eventManager.addEventListener(this.TEXTAREA, 'cut', function (event) {
- stopPropagation(event);
- });
- this.eventManager.addEventListener(this.TEXTAREA, 'paste', function (event) {
- stopPropagation(event);
- });
- this.instance.addHook('afterScrollHorizontally', function () {
- editor.refreshDimensions();
- });
- this.instance.addHook('afterScrollVertically', function () {
- editor.refreshDimensions();
- });
- this.instance.addHook('afterColumnResize', function () {
- editor.refreshDimensions();
- editor.focus();
- });
- this.instance.addHook('afterRowResize', function () {
- editor.refreshDimensions();
- editor.focus();
- });
- this.instance.addHook('afterDestroy', function () {
- editor.eventManager.destroy();
- });
- };
- TextEditor.prototype.destroy = function () {
- this.eventManager.destroy();
- };
- export default TextEditor;
|