UNPKG

monaco-editor-core

Version:

A browser based code editor

184 lines (183 loc) • 9.03 kB
/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the MIT License. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) { var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d; if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc); else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r; return c > 3 && r && Object.defineProperty(target, key, r), r; }; var __param = (this && this.__param) || function (paramIndex, decorator) { return function (target, key) { decorator(target, key, paramIndex); } }; var MessageController_1; import { renderMarkdown } from '../../../../base/browser/markdownRenderer.js'; import { alert } from '../../../../base/browser/ui/aria/aria.js'; import { Event } from '../../../../base/common/event.js'; import { isMarkdownString } from '../../../../base/common/htmlContent.js'; import { DisposableStore, MutableDisposable } from '../../../../base/common/lifecycle.js'; import './messageController.css'; import { EditorCommand, registerEditorCommand, registerEditorContribution } from '../../../browser/editorExtensions.js'; import { Range } from '../../../common/core/range.js'; import { openLinkFromMarkdown } from '../../../browser/widget/markdownRenderer/browser/markdownRenderer.js'; import * as nls from '../../../../nls.js'; import { IContextKeyService, RawContextKey } from '../../../../platform/contextkey/common/contextkey.js'; import { IOpenerService } from '../../../../platform/opener/common/opener.js'; import * as dom from '../../../../base/browser/dom.js'; let MessageController = class MessageController { static { MessageController_1 = this; } static { this.ID = 'editor.contrib.messageController'; } static { this.MESSAGE_VISIBLE = new RawContextKey('messageVisible', false, nls.localize('messageVisible', 'Whether the editor is currently showing an inline message')); } static get(editor) { return editor.getContribution(MessageController_1.ID); } constructor(editor, contextKeyService, _openerService) { this._openerService = _openerService; this._messageWidget = new MutableDisposable(); this._messageListeners = new DisposableStore(); this._mouseOverMessage = false; this._editor = editor; this._visible = MessageController_1.MESSAGE_VISIBLE.bindTo(contextKeyService); } dispose() { this._message?.dispose(); this._messageListeners.dispose(); this._messageWidget.dispose(); this._visible.reset(); } showMessage(message, position) { alert(isMarkdownString(message) ? message.value : message); this._visible.set(true); this._messageWidget.clear(); this._messageListeners.clear(); this._message = isMarkdownString(message) ? renderMarkdown(message, { actionHandler: { callback: (url) => { this.closeMessage(); openLinkFromMarkdown(this._openerService, url, isMarkdownString(message) ? message.isTrusted : undefined); }, disposables: this._messageListeners }, }) : undefined; this._messageWidget.value = new MessageWidget(this._editor, position, typeof message === 'string' ? message : this._message.element); // close on blur (debounced to allow to tab into the message), cursor, model change, dispose this._messageListeners.add(Event.debounce(this._editor.onDidBlurEditorText, (last, event) => event, 0)(() => { if (this._mouseOverMessage) { return; // override when mouse over message } if (this._messageWidget.value && dom.isAncestor(dom.getActiveElement(), this._messageWidget.value.getDomNode())) { return; // override when focus is inside the message } this.closeMessage(); })); this._messageListeners.add(this._editor.onDidChangeCursorPosition(() => this.closeMessage())); this._messageListeners.add(this._editor.onDidDispose(() => this.closeMessage())); this._messageListeners.add(this._editor.onDidChangeModel(() => this.closeMessage())); this._messageListeners.add(dom.addDisposableListener(this._messageWidget.value.getDomNode(), dom.EventType.MOUSE_ENTER, () => this._mouseOverMessage = true, true)); this._messageListeners.add(dom.addDisposableListener(this._messageWidget.value.getDomNode(), dom.EventType.MOUSE_LEAVE, () => this._mouseOverMessage = false, true)); // close on mouse move let bounds; this._messageListeners.add(this._editor.onMouseMove(e => { // outside the text area if (!e.target.position) { return; } if (!bounds) { // define bounding box around position and first mouse occurance bounds = new Range(position.lineNumber - 3, 1, e.target.position.lineNumber + 3, 1); } else if (!bounds.containsPosition(e.target.position)) { // check if position is still in bounds this.closeMessage(); } })); } closeMessage() { this._visible.reset(); this._messageListeners.clear(); if (this._messageWidget.value) { this._messageListeners.add(MessageWidget.fadeOut(this._messageWidget.value)); } } }; MessageController = MessageController_1 = __decorate([ __param(1, IContextKeyService), __param(2, IOpenerService) ], MessageController); export { MessageController }; const MessageCommand = EditorCommand.bindToContribution(MessageController.get); registerEditorCommand(new MessageCommand({ id: 'leaveEditorMessage', precondition: MessageController.MESSAGE_VISIBLE, handler: c => c.closeMessage(), kbOpts: { weight: 100 /* KeybindingWeight.EditorContrib */ + 30, primary: 9 /* KeyCode.Escape */ } })); class MessageWidget { static fadeOut(messageWidget) { const dispose = () => { messageWidget.dispose(); clearTimeout(handle); messageWidget.getDomNode().removeEventListener('animationend', dispose); }; const handle = setTimeout(dispose, 110); messageWidget.getDomNode().addEventListener('animationend', dispose); messageWidget.getDomNode().classList.add('fadeOut'); return { dispose }; } constructor(editor, { lineNumber, column }, text) { // Editor.IContentWidget.allowEditorOverflow this.allowEditorOverflow = true; this.suppressMouseDown = false; this._editor = editor; this._editor.revealLinesInCenterIfOutsideViewport(lineNumber, lineNumber, 0 /* ScrollType.Smooth */); this._position = { lineNumber, column }; this._domNode = document.createElement('div'); this._domNode.classList.add('monaco-editor-overlaymessage'); this._domNode.style.marginLeft = '-6px'; const anchorTop = document.createElement('div'); anchorTop.classList.add('anchor', 'top'); this._domNode.appendChild(anchorTop); const message = document.createElement('div'); if (typeof text === 'string') { message.classList.add('message'); message.textContent = text; } else { text.classList.add('message'); message.appendChild(text); } this._domNode.appendChild(message); const anchorBottom = document.createElement('div'); anchorBottom.classList.add('anchor', 'below'); this._domNode.appendChild(anchorBottom); this._editor.addContentWidget(this); this._domNode.classList.add('fadeIn'); } dispose() { this._editor.removeContentWidget(this); } getId() { return 'messageoverlay'; } getDomNode() { return this._domNode; } getPosition() { return { position: this._position, preference: [ 1 /* ContentWidgetPositionPreference.ABOVE */, 2 /* ContentWidgetPositionPreference.BELOW */, ], positionAffinity: 1 /* PositionAffinity.Right */, }; } afterRender(position) { this._domNode.classList.toggle('below', position === 2 /* ContentWidgetPositionPreference.BELOW */); } } registerEditorContribution(MessageController.ID, MessageController, 4 /* EditorContributionInstantiation.Lazy */);