/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the MIT License. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ import * as browser from '../../../../vs/base/browser/browser'; import * as dom from '../../../../vs/base/browser/dom'; import { FastDomNode } from '../../../../vs/base/browser/fastDomNode'; import { IKeyboardEvent } from '../../../../vs/base/browser/keyboardEvent'; import { RunOnceScheduler } from '../../../../vs/base/common/async'; import { Emitter, Event } from '../../../../vs/base/common/event'; import { KeyCode } from '../../../../vs/base/common/keyCodes'; import { Disposable, IDisposable } from '../../../../vs/base/common/lifecycle'; import * as platform from '../../../../vs/base/common/platform'; import * as strings from '../../../../vs/base/common/strings'; import { ITextAreaWrapper, ITypeData, TextAreaState, _debugComposition, } from '../../../../vs/editor/browser/controller/textAreaState'; import { Position } from '../../../../vs/editor/common/core/position'; import { Selection } from '../../../../vs/editor/common/core/selection'; export namespace TextAreaSyntethicEvents { export const Tap = '-monaco-textarea-synthetic-tap'; } export interface ICompositionData { data: string; } export const CopyOptions = { forceCopyWithSyntaxHighlighting: false, }; const enum ReadFromTextArea { Type, Paste, } export interface IPasteData { text: string; metadata: ClipboardStoredMetadata | null; } export interface ClipboardDataToCopy { isFromEmptySelection: boolean; multicursorText: string[] | null | undefined; text: string; html: string | null | undefined; mode: string | null; } export interface ClipboardStoredMetadata { version: 1; isFromEmptySelection: boolean | undefined; multicursorText: string[] | null | undefined; mode: string | null; } export interface ITextAreaInputHost { getDataToCopy(html: boolean): ClipboardDataToCopy; getScreenReaderContent(currentState: TextAreaState): TextAreaState; deduceModelPosition( viewAnchorPosition: Position, deltaOffset: number, lineFeedCnt: number ): Position; } interface CompositionEvent extends UIEvent { readonly data: string; readonly locale: string; } interface InMemoryClipboardMetadata { lastCopiedValue: string; data: ClipboardStoredMetadata; } /** * Every time we write to the clipboard, we record a bit of extra metadata here. * Every time we read from the cipboard, if the text matches our last written text, * we can fetch the previous metadata. */ export class InMemoryClipboardMetadataManager { public static readonly INSTANCE = new InMemoryClipboardMetadataManager(); private _lastState: InMemoryClipboardMetadata | null; constructor() { this._lastState = null; } public set(lastCopiedValue: string, data: ClipboardStoredMetadata): void { this._lastState = { lastCopiedValue, data }; } public get(pastedText: string): ClipboardStoredMetadata | null { if (this._lastState && this._lastState.lastCopiedValue === pastedText) { // match! return this._lastState.data; } this._lastState = null; return null; } } export interface ICompositionStartEvent { revealDeltaColumns: number; } /** * Writes screen reader content to the textarea and is able to analyze its input events to generate: * - onCut * - onPaste * - onType * * Composition events are generated for presentation purposes (composition input is reflected in onType). */ export class TextAreaInput extends Disposable { private _onFocus = this._register(new Emitter()); public readonly onFocus: Event = this._onFocus.event; private _onBlur = this._register(new Emitter()); public readonly onBlur: Event = this._onBlur.event; private _onKeyDown = this._register(new Emitter()); public readonly onKeyDown: Event = this._onKeyDown.event; private _onKeyUp = this._register(new Emitter()); public readonly onKeyUp: Event = this._onKeyUp.event; private _onCut = this._register(new Emitter()); public readonly onCut: Event = this._onCut.event; private _onPaste = this._register(new Emitter()); public readonly onPaste: Event = this._onPaste.event; private _onType = this._register(new Emitter()); public readonly onType: Event = this._onType.event; private _onCompositionStart = this._register( new Emitter() ); public readonly onCompositionStart: Event = this._onCompositionStart.event; private _onCompositionUpdate = this._register( new Emitter() ); public readonly onCompositionUpdate: Event = this._onCompositionUpdate.event; private _onCompositionEnd = this._register(new Emitter()); public readonly onCompositionEnd: Event = this._onCompositionEnd.event; private _onSelectionChangeRequest = this._register(new Emitter()); public readonly onSelectionChangeRequest: Event = this._onSelectionChangeRequest.event; // --- private readonly _host: ITextAreaInputHost; private readonly _textArea: TextAreaWrapper; private readonly _asyncTriggerCut: RunOnceScheduler; private readonly _asyncFocusGainWriteScreenReaderContent: RunOnceScheduler; private _textAreaState: TextAreaState; private _selectionChangeListener: IDisposable | null; private _hasFocus: boolean; private _isDoingComposition: boolean; private _nextCommand: ReadFromTextArea; constructor( host: ITextAreaInputHost, private textArea: FastDomNode ) { super(); this._host = host; this._textArea = this._register(new TextAreaWrapper(textArea)); this._asyncTriggerCut = this._register( new RunOnceScheduler(() => this._onCut.fire(), 0) ); this._asyncFocusGainWriteScreenReaderContent = this._register( new RunOnceScheduler( () => this.writeScreenReaderContent('asyncFocusGain'), 0 ) ); this._textAreaState = TextAreaState.EMPTY; this._selectionChangeListener = null; this.writeScreenReaderContent('ctor'); this._hasFocus = false; this._isDoingComposition = false; this._nextCommand = ReadFromTextArea.Type; let lastKeyDown: IKeyboardEvent | null = null; this._register( dom.addStandardDisposableListener( textArea.domNode, 'keydown', (e: IKeyboardEvent) => { if ( e.keyCode === KeyCode.KEY_IN_COMPOSITION || (this._isDoingComposition && e.keyCode === KeyCode.Backspace) ) { // Stop propagation for keyDown events if the IME is processing key input e.stopPropagation(); } if (e.equals(KeyCode.Escape)) { // Prevent default always for `Esc`, otherwise it will generate a keypress // See https://msdn.microsoft.com/en-us/library/ie/ms536939(v=vs.85).aspx e.preventDefault(); } lastKeyDown = e; this._onKeyDown.fire(e); } ) ); this._register( dom.addStandardDisposableListener( textArea.domNode, 'keyup', (e: IKeyboardEvent) => { this._onKeyUp.fire(e); } ) ); this._register( dom.addDisposableListener( textArea.domNode, 'compositionstart', (e: CompositionEvent) => { if (_debugComposition) { console.log(`[compositionstart]`, e); } if (this._isDoingComposition) { return; } this._isDoingComposition = true; if ( platform.isMacintosh && this._textAreaState.selectionStart === this._textAreaState.selectionEnd && this._textAreaState.selectionStart > 0 && this._textAreaState.value.substr( this._textAreaState.selectionStart - 1, 1 ) === e.data ) { const isArrowKey = lastKeyDown && lastKeyDown.equals(KeyCode.KEY_IN_COMPOSITION) && (lastKeyDown.code === 'ArrowRight' || lastKeyDown.code === 'ArrowLeft'); if (isArrowKey || browser.isFirefox) { // Handling long press case on Chromium/Safari macOS + arrow key => pretend the character was selected // or long press case on Firefox on macOS if (_debugComposition) { console.log( `[compositionstart] Handling long press case on macOS + arrow key or Firefox`, e ); } this._textAreaState = new TextAreaState( this._textAreaState.value, this._textAreaState.selectionStart - 1, this._textAreaState.selectionEnd, this._textAreaState.selectionStartPosition ? new Position( this._textAreaState.selectionStartPosition.lineNumber, this._textAreaState.selectionStartPosition.column - 1 ) : null, this._textAreaState.selectionEndPosition ); this._onCompositionStart.fire({ revealDeltaColumns: -1 }); return; } } if (browser.isAndroid) { // when tapping on the editor, Android enters composition mode to edit the current word // so we cannot clear the textarea on Android and we must pretend the current word was selected this._onCompositionStart.fire({ revealDeltaColumns: -this._textAreaState.selectionStart, }); return; } this._setAndWriteTextAreaState( 'compositionstart', TextAreaState.EMPTY ); this._onCompositionStart.fire({ revealDeltaColumns: 0 }); } ) ); /** * Deduce the typed input from a text area's value and the last observed state. */ const deduceInputFromTextAreaValue = ( couldBeEmojiInput: boolean ): [TextAreaState, ITypeData] => { const oldState = this._textAreaState; const newState = TextAreaState.readFromTextArea(this._textArea); return [ newState, TextAreaState.deduceInput(oldState, newState, couldBeEmojiInput), ]; }; const deduceAndroidCompositionInput = (): [TextAreaState, ITypeData] => { const oldState = this._textAreaState; const newState = TextAreaState.readFromTextArea(this._textArea); return [ newState, TextAreaState.deduceAndroidCompositionInput(oldState, newState), ]; }; /** * Deduce the composition input from a string. */ const deduceComposition = (text: string): [TextAreaState, ITypeData] => { const oldState = this._textAreaState; const newState = TextAreaState.selectedText(text); const typeInput: ITypeData = { text: newState.value, replacePrevCharCnt: oldState.selectionEnd - oldState.selectionStart, replaceNextCharCnt: 0, positionDelta: 0, }; return [newState, typeInput]; }; this._register( dom.addDisposableListener( textArea.domNode, 'compositionupdate', (e: CompositionEvent) => { if (_debugComposition) { console.log(`[compositionupdate]`, e); } if (browser.isAndroid) { // On Android, the data sent with the composition update event is unusable. // For example, if the cursor is in the middle of a word like Mic|osoft // and Microsoft is chosen from the keyboard's suggestions, the e.data will contain "Microsoft". // This is not really usable because it doesn't tell us where the edit began and where it ended. const [newState, typeInput] = deduceAndroidCompositionInput(); this._textAreaState = newState; this._onType.fire(typeInput); this._onCompositionUpdate.fire(e); return; } const [newState, typeInput] = deduceComposition(e.data || ''); this._textAreaState = newState; this._onType.fire(typeInput); this._onCompositionUpdate.fire(e); } ) ); this._register( dom.addDisposableListener( textArea.domNode, 'compositionend', (e: CompositionEvent) => { if (_debugComposition) { console.log(`[compositionend]`, e); } // https://github.com/microsoft/monaco-editor/issues/1663 // On iOS 13.2, Chinese system IME randomly trigger an additional compositionend event with empty data if (!this._isDoingComposition) { return; } this._isDoingComposition = false; if (browser.isAndroid) { // On Android, the data sent with the composition update event is unusable. // For example, if the cursor is in the middle of a word like Mic|osoft // and Microsoft is chosen from the keyboard's suggestions, the e.data will contain "Microsoft". // This is not really usable because it doesn't tell us where the edit began and where it ended. const [newState, typeInput] = deduceAndroidCompositionInput(); this._textAreaState = newState; this._onType.fire(typeInput); this._onCompositionEnd.fire(); return; } const [newState, typeInput] = deduceComposition(e.data || ''); this._textAreaState = newState; this._onType.fire(typeInput); // isChrome: the textarea is not updated correctly when composition ends // isFirefox: the textarea is not updated correctly after inserting emojis // => we cannot assume the text at the end consists only of the composited text if (browser.isChrome || browser.isFirefox) { this._textAreaState = TextAreaState.readFromTextArea( this._textArea ); } this._onCompositionEnd.fire(); } ) ); this._register( dom.addDisposableListener(textArea.domNode, 'input', () => { // Pretend here we touched the text area, as the `input` event will most likely // result in a `selectionchange` event which we want to ignore this._textArea.setIgnoreSelectionChangeTime('received input event'); if (this._isDoingComposition) { return; } const [newState, typeInput] = deduceInputFromTextAreaValue( /*couldBeEmojiInput*/ platform.isMacintosh ); if ( typeInput.replacePrevCharCnt === 0 && typeInput.text.length === 1 && strings.isHighSurrogate(typeInput.text.charCodeAt(0)) ) { // Ignore invalid input but keep it around for next time return; } this._textAreaState = newState; if (this._nextCommand === ReadFromTextArea.Type) { if (typeInput.text !== '' || typeInput.replacePrevCharCnt !== 0) { this._onType.fire(typeInput); } } else { if (typeInput.text !== '' || typeInput.replacePrevCharCnt !== 0) { this._firePaste(typeInput.text, null); } this._nextCommand = ReadFromTextArea.Type; } }) ); // --- Clipboard operations this._register( dom.addDisposableListener( textArea.domNode, 'cut', (e: ClipboardEvent) => { // Pretend here we touched the text area, as the `cut` event will most likely // result in a `selectionchange` event which we want to ignore this._textArea.setIgnoreSelectionChangeTime('received cut event'); this._ensureClipboardGetsEditorSelection(e); this._asyncTriggerCut.schedule(); } ) ); this._register( dom.addDisposableListener( textArea.domNode, 'copy', (e: ClipboardEvent) => { this._ensureClipboardGetsEditorSelection(e); } ) ); this._register( dom.addDisposableListener( textArea.domNode, 'paste', (e: ClipboardEvent) => { // Pretend here we touched the text area, as the `paste` event will most likely // result in a `selectionchange` event which we want to ignore this._textArea.setIgnoreSelectionChangeTime('received paste event'); if (ClipboardEventUtils.canUseTextData(e)) { const [pastePlainText, metadata] = ClipboardEventUtils.getTextData(e); if (pastePlainText !== '') { this._firePaste(pastePlainText, metadata); } } else { if ( this._textArea.getSelectionStart() !== this._textArea.getSelectionEnd() ) { // Clean up the textarea, to get a clean paste this._setAndWriteTextAreaState('paste', TextAreaState.EMPTY); } this._nextCommand = ReadFromTextArea.Paste; } } ) ); this._register( dom.addDisposableListener(textArea.domNode, 'focus', () => { const hadFocus = this._hasFocus; this._setHasFocus(true); if (browser.isSafari && !hadFocus && this._hasFocus) { // When "tabbing into" the textarea, immediately after dispatching the 'focus' event, // Safari will always move the selection at offset 0 in the textarea this._asyncFocusGainWriteScreenReaderContent.schedule(); } }) ); this._register( dom.addDisposableListener(textArea.domNode, 'blur', () => { if (this._isDoingComposition) { // See https://github.com/microsoft/vscode/issues/112621 // where compositionend is not triggered when the editor // is taken off-dom during a composition // Clear the flag to be able to write to the textarea this._isDoingComposition = false; // Clear the textarea to avoid an unwanted cursor type this.writeScreenReaderContent('blurWithoutCompositionEnd'); // Fire artificial composition end this._onCompositionEnd.fire(); } this._setHasFocus(false); }) ); this._register( dom.addDisposableListener( textArea.domNode, TextAreaSyntethicEvents.Tap, () => { if (browser.isAndroid && this._isDoingComposition) { // on Android, tapping does not cancel the current composition, so the // textarea is stuck showing the old composition // Clear the flag to be able to write to the textarea this._isDoingComposition = false; // Clear the textarea to avoid an unwanted cursor type this.writeScreenReaderContent('tapWithoutCompositionEnd'); // Fire artificial composition end this._onCompositionEnd.fire(); } } ) ); } private _installSelectionChangeListener(): IDisposable { // See https://github.com/microsoft/vscode/issues/27216 and https://github.com/microsoft/vscode/issues/98256 // When using a Braille display, it is possible for users to reposition the // system caret. This is reflected in Chrome as a `selectionchange` event. // // The `selectionchange` event appears to be emitted under numerous other circumstances, // so it is quite a challenge to distinguish a `selectionchange` coming in from a user // using a Braille display from all the other cases. // // The problems with the `selectionchange` event are: // * the event is emitted when the textarea is focused programmatically -- textarea.focus() // * the event is emitted when the selection is changed in the textarea programmatically -- textarea.setSelectionRange(...) // * the event is emitted when the value of the textarea is changed programmatically -- textarea.value = '...' // * the event is emitted when tabbing into the textarea // * the event is emitted asynchronously (sometimes with a delay as high as a few tens of ms) // * the event sometimes comes in bursts for a single logical textarea operation // `selectionchange` events often come multiple times for a single logical change // so throttle multiple `selectionchange` events that burst in a short period of time. let previousSelectionChangeEventTime = 0; return dom.addDisposableListener(document, 'selectionchange', (e) => { if (!this._hasFocus) { return; } if (this._isDoingComposition) { return; } if (!browser.isChrome) { // Support only for Chrome until testing happens on other browsers return; } const now = Date.now(); const delta1 = now - previousSelectionChangeEventTime; previousSelectionChangeEventTime = now; if (delta1 < 5) { // received another `selectionchange` event within 5ms of the previous `selectionchange` event // => ignore it return; } const delta2 = now - this._textArea.getIgnoreSelectionChangeTime(); this._textArea.resetSelectionChangeTime(); if (delta2 < 100) { // received a `selectionchange` event within 100ms since we touched the textarea // => ignore it, since we caused it return; } if ( !this._textAreaState.selectionStartPosition || !this._textAreaState.selectionEndPosition ) { // Cannot correlate a position in the textarea with a position in the editor... return; } const newValue = this._textArea.getValue(); if (this._textAreaState.value !== newValue) { // Cannot correlate a position in the textarea with a position in the editor... return; } const newSelectionStart = this._textArea.getSelectionStart(); const newSelectionEnd = this._textArea.getSelectionEnd(); if ( this._textAreaState.selectionStart === newSelectionStart && this._textAreaState.selectionEnd === newSelectionEnd ) { // Nothing to do... return; } const _newSelectionStartPosition = this._textAreaState.deduceEditorPosition(newSelectionStart); const newSelectionStartPosition = this._host.deduceModelPosition( _newSelectionStartPosition[0]!, _newSelectionStartPosition[1], _newSelectionStartPosition[2] ); const _newSelectionEndPosition = this._textAreaState.deduceEditorPosition(newSelectionEnd); const newSelectionEndPosition = this._host.deduceModelPosition( _newSelectionEndPosition[0]!, _newSelectionEndPosition[1], _newSelectionEndPosition[2] ); const newSelection = new Selection( newSelectionStartPosition.lineNumber, newSelectionStartPosition.column, newSelectionEndPosition.lineNumber, newSelectionEndPosition.column ); this._onSelectionChangeRequest.fire(newSelection); }); } public override dispose(): void { super.dispose(); if (this._selectionChangeListener) { this._selectionChangeListener.dispose(); this._selectionChangeListener = null; } } public focusTextArea(): void { // Setting this._hasFocus and writing the screen reader content // will result in a focus() and setSelectionRange() in the textarea this._setHasFocus(true); // If the editor is off DOM, focus cannot be really set, so let's double check that we have managed to set the focus this.refreshFocusState(); } public isFocused(): boolean { return this._hasFocus; } public refreshFocusState(): void { const shadowRoot = dom.getShadowRoot(this.textArea.domNode); if (shadowRoot) { // this._setHasFocus(shadowRoot.activeElement === this.textArea.domNode); } else if (dom.isInDOM(this.textArea.domNode)) { // this._setHasFocus(document.activeElement === this.textArea.domNode); } else { // this._setHasFocus(false); } } private _setHasFocus(newHasFocus: boolean): void { if (this._hasFocus === newHasFocus) { // no change return; } this._hasFocus = newHasFocus; if (this._selectionChangeListener) { this._selectionChangeListener.dispose(); this._selectionChangeListener = null; } if (this._hasFocus) { this._selectionChangeListener = this._installSelectionChangeListener(); } if (this._hasFocus) { this.writeScreenReaderContent('focusgain'); } if (this._hasFocus) { this._onFocus.fire(); } else { this._onBlur.fire(); } } private _setAndWriteTextAreaState( reason: string, textAreaState: TextAreaState ): void { if (!this._hasFocus) { textAreaState = textAreaState.collapseSelection(); } textAreaState.writeToTextArea(reason, this._textArea, this._hasFocus); this._textAreaState = textAreaState; } public writeScreenReaderContent(reason: string): void { if (this._isDoingComposition) { // Do not write to the text area when doing composition return; } this._setAndWriteTextAreaState( reason, this._host.getScreenReaderContent(this._textAreaState) ); } private _ensureClipboardGetsEditorSelection(e: ClipboardEvent): void { const dataToCopy = this._host.getDataToCopy( ClipboardEventUtils.canUseTextData(e) ); const storedMetadata: ClipboardStoredMetadata = { version: 1, isFromEmptySelection: dataToCopy.isFromEmptySelection, multicursorText: dataToCopy.multicursorText, mode: dataToCopy.mode, }; InMemoryClipboardMetadataManager.INSTANCE.set( // When writing "LINE\r\n" to the clipboard and then pasting, // Firefox pastes "LINE\n", so let's work around this quirk browser.isFirefox ? dataToCopy.text.replace(/\r\n/g, '\n') : dataToCopy.text, storedMetadata ); if (!ClipboardEventUtils.canUseTextData(e)) { // Looks like an old browser. The strategy is to place the text // we'd like to be copied to the clipboard in the textarea and select it. this._setAndWriteTextAreaState( 'copy or cut', TextAreaState.selectedText(dataToCopy.text) ); return; } ClipboardEventUtils.setTextData( e, dataToCopy.text, dataToCopy.html, storedMetadata ); } private _firePaste( text: string, metadata: ClipboardStoredMetadata | null ): void { if (!metadata) { // try the in-memory store metadata = InMemoryClipboardMetadataManager.INSTANCE.get(text); } this._onPaste.fire({ text: text, metadata: metadata, }); } } class ClipboardEventUtils { public static canUseTextData(e: ClipboardEvent): boolean { if (e.clipboardData) { return true; } if ((window).clipboardData) { return true; } return false; } public static getTextData( e: ClipboardEvent ): [string, ClipboardStoredMetadata | null] { if (e.clipboardData) { e.preventDefault(); const text = e.clipboardData.getData('text/plain'); let metadata: ClipboardStoredMetadata | null = null; const rawmetadata = e.clipboardData.getData('vscode-editor-data'); if (typeof rawmetadata === 'string') { try { metadata = JSON.parse(rawmetadata); if (metadata.version !== 1) { metadata = null; } } catch (err) { // no problem! } } return [text, metadata]; } if ((window).clipboardData) { e.preventDefault(); const text: string = (window).clipboardData.getData('Text'); return [text, null]; } throw new Error('ClipboardEventUtils.getTextData: Cannot use text data!'); } public static setTextData( e: ClipboardEvent, text: string, html: string | null | undefined, metadata: ClipboardStoredMetadata ): void { if (e.clipboardData) { e.clipboardData.setData('text/plain', text); if (typeof html === 'string') { e.clipboardData.setData('text/html', html); } e.clipboardData.setData('vscode-editor-data', JSON.stringify(metadata)); e.preventDefault(); return; } if ((window).clipboardData) { (window).clipboardData.setData('Text', text); e.preventDefault(); return; } throw new Error('ClipboardEventUtils.setTextData: Cannot use text data!'); } } class TextAreaWrapper extends Disposable implements ITextAreaWrapper { private readonly _actual: FastDomNode; private _ignoreSelectionChangeTime: number; constructor(_textArea: FastDomNode) { super(); this._actual = _textArea; this._ignoreSelectionChangeTime = 0; } public setIgnoreSelectionChangeTime(reason: string): void { this._ignoreSelectionChangeTime = Date.now(); } public getIgnoreSelectionChangeTime(): number { return this._ignoreSelectionChangeTime; } public resetSelectionChangeTime(): void { this._ignoreSelectionChangeTime = 0; } public getValue(): string { // console.log('current value: ' + this._textArea.value); return this._actual.domNode.value; } public setValue(reason: string, value: string): void { const textArea = this._actual.domNode; if (textArea.value === value) { // No change return; } // console.log('reason: ' + reason + ', current value: ' + textArea.value + ' => new value: ' + value); this.setIgnoreSelectionChangeTime('setValue'); textArea.value = value; } public getSelectionStart(): number { return this._actual.domNode.selectionDirection === 'backward' ? this._actual.domNode.selectionEnd : this._actual.domNode.selectionStart; } public getSelectionEnd(): number { return this._actual.domNode.selectionDirection === 'backward' ? this._actual.domNode.selectionStart : this._actual.domNode.selectionEnd; } public setSelectionRange( reason: string, selectionStart: number, selectionEnd: number ): void { const textArea = this._actual.domNode; let activeElement: Element | null = null; const shadowRoot = dom.getShadowRoot(textArea); if (shadowRoot) { activeElement = shadowRoot.activeElement; } else { activeElement = document.activeElement; } const currentIsFocused = activeElement === textArea; const currentSelectionStart = textArea.selectionStart; const currentSelectionEnd = textArea.selectionEnd; if ( currentIsFocused && currentSelectionStart === selectionStart && currentSelectionEnd === selectionEnd ) { // No change // Firefox iframe bug https://github.com/microsoft/monaco-editor/issues/643#issuecomment-367871377 if (browser.isFirefox && window.parent !== window) { textArea.focus(); } return; } // console.log('reason: ' + reason + ', setSelectionRange: ' + selectionStart + ' -> ' + selectionEnd); if (currentIsFocused) { // No need to focus, only need to change the selection range this.setIgnoreSelectionChangeTime('setSelectionRange'); textArea.setSelectionRange(selectionStart, selectionEnd); if (browser.isFirefox && window.parent !== window) { textArea.focus(); } return; } // If the focus is outside the textarea, browsers will try really hard to reveal the textarea. // Here, we try to undo the browser's desperate reveal. try { const scrollState = dom.saveParentsScrollTop(textArea); this.setIgnoreSelectionChangeTime('setSelectionRange'); textArea.focus(); textArea.setSelectionRange(selectionStart, selectionEnd); dom.restoreParentsScrollTop(textArea, scrollState); } catch (e) { // Sometimes IE throws when setting selection (e.g. textarea is off-DOM) } } }