UNPKG

monaco-editor

Version:
161 lines (158 loc) 8.23 kB
import { commonPrefixLength, commonSuffixLength } from '../../../../../base/common/strings.js'; /*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the MIT License. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ class TextAreaState { static { this.EMPTY = new TextAreaState('', 0, 0, null, undefined); } constructor(value, /** the offset where selection starts inside `value` */ selectionStart, /** the offset where selection ends inside `value` */ selectionEnd, /** the editor range in the view coordinate system that matches the selection inside `value` */ selection, /** the visible line count (wrapped, not necessarily matching \n characters) for the text in `value` before `selectionStart` */ newlineCountBeforeSelection) { this.value = value; this.selectionStart = selectionStart; this.selectionEnd = selectionEnd; this.selection = selection; this.newlineCountBeforeSelection = newlineCountBeforeSelection; } toString() { return `[ <${this.value}>, selectionStart: ${this.selectionStart}, selectionEnd: ${this.selectionEnd}]`; } static readFromTextArea(textArea, previousState) { const value = textArea.getValue(); const selectionStart = textArea.getSelectionStart(); const selectionEnd = textArea.getSelectionEnd(); let newlineCountBeforeSelection = undefined; if (previousState) { const valueBeforeSelectionStart = value.substring(0, selectionStart); const previousValueBeforeSelectionStart = previousState.value.substring(0, previousState.selectionStart); if (valueBeforeSelectionStart === previousValueBeforeSelectionStart) { newlineCountBeforeSelection = previousState.newlineCountBeforeSelection; } } return new TextAreaState(value, selectionStart, selectionEnd, null, newlineCountBeforeSelection); } collapseSelection() { if (this.selectionStart === this.value.length) { return this; } return new TextAreaState(this.value, this.value.length, this.value.length, null, undefined); } isWrittenToTextArea(textArea, select) { const valuesEqual = this.value === textArea.getValue(); if (!select) { return valuesEqual; } const selectionsEqual = this.selectionStart === textArea.getSelectionStart() && this.selectionEnd === textArea.getSelectionEnd(); return selectionsEqual && valuesEqual; } writeToTextArea(reason, textArea, select) { textArea.setValue(reason, this.value); if (select) { textArea.setSelectionRange(reason, this.selectionStart, this.selectionEnd); } } deduceEditorPosition(offset) { if (offset <= this.selectionStart) { const str = this.value.substring(offset, this.selectionStart); return this._finishDeduceEditorPosition(this.selection?.getStartPosition() ?? null, str, -1); } if (offset >= this.selectionEnd) { const str = this.value.substring(this.selectionEnd, offset); return this._finishDeduceEditorPosition(this.selection?.getEndPosition() ?? null, str, 1); } const str1 = this.value.substring(this.selectionStart, offset); if (str1.indexOf(String.fromCharCode(8230)) === -1) { return this._finishDeduceEditorPosition(this.selection?.getStartPosition() ?? null, str1, 1); } const str2 = this.value.substring(offset, this.selectionEnd); return this._finishDeduceEditorPosition(this.selection?.getEndPosition() ?? null, str2, -1); } _finishDeduceEditorPosition(anchor, deltaText, signum) { let lineFeedCnt = 0; let lastLineFeedIndex = -1; while ((lastLineFeedIndex = deltaText.indexOf('\n', lastLineFeedIndex + 1)) !== -1) { lineFeedCnt++; } return [anchor, signum * deltaText.length, lineFeedCnt]; } static deduceInput(previousState, currentState, couldBeEmojiInput) { if (!previousState) { // This is the EMPTY state return { text: '', replacePrevCharCnt: 0, replaceNextCharCnt: 0, positionDelta: 0 }; } const prefixLength = Math.min(commonPrefixLength(previousState.value, currentState.value), previousState.selectionStart, currentState.selectionStart); const suffixLength = Math.min(commonSuffixLength(previousState.value, currentState.value), previousState.value.length - previousState.selectionEnd, currentState.value.length - currentState.selectionEnd); previousState.value.substring(prefixLength, previousState.value.length - suffixLength); const currentValue = currentState.value.substring(prefixLength, currentState.value.length - suffixLength); const previousSelectionStart = previousState.selectionStart - prefixLength; const previousSelectionEnd = previousState.selectionEnd - prefixLength; const currentSelectionStart = currentState.selectionStart - prefixLength; const currentSelectionEnd = currentState.selectionEnd - prefixLength; if (currentSelectionStart === currentSelectionEnd) { // no current selection const replacePreviousCharacters = (previousState.selectionStart - prefixLength); return { text: currentValue, replacePrevCharCnt: replacePreviousCharacters, replaceNextCharCnt: 0, positionDelta: 0 }; } // there is a current selection => composition case const replacePreviousCharacters = previousSelectionEnd - previousSelectionStart; return { text: currentValue, replacePrevCharCnt: replacePreviousCharacters, replaceNextCharCnt: 0, positionDelta: 0 }; } static deduceAndroidCompositionInput(previousState, currentState) { if (!previousState) { // This is the EMPTY state return { text: '', replacePrevCharCnt: 0, replaceNextCharCnt: 0, positionDelta: 0 }; } if (previousState.value === currentState.value) { return { text: '', replacePrevCharCnt: 0, replaceNextCharCnt: 0, positionDelta: currentState.selectionEnd - previousState.selectionEnd }; } const prefixLength = Math.min(commonPrefixLength(previousState.value, currentState.value), previousState.selectionEnd); const suffixLength = Math.min(commonSuffixLength(previousState.value, currentState.value), previousState.value.length - previousState.selectionEnd); const previousValue = previousState.value.substring(prefixLength, previousState.value.length - suffixLength); const currentValue = currentState.value.substring(prefixLength, currentState.value.length - suffixLength); previousState.selectionStart - prefixLength; const previousSelectionEnd = previousState.selectionEnd - prefixLength; currentState.selectionStart - prefixLength; const currentSelectionEnd = currentState.selectionEnd - prefixLength; return { text: currentValue, replacePrevCharCnt: previousSelectionEnd, replaceNextCharCnt: previousValue.length - previousSelectionEnd, positionDelta: currentSelectionEnd - currentValue.length }; } static fromScreenReaderContentState(screenReaderContentState) { return new TextAreaState(screenReaderContentState.value, screenReaderContentState.selectionStart, screenReaderContentState.selectionEnd, screenReaderContentState.selection, screenReaderContentState.newlineCountBeforeSelection); } } export { TextAreaState };