@theia/monaco
Version:
Theia - Monaco Extension
180 lines (156 loc) • 8.99 kB
text/typescript
// *****************************************************************************
// Copyright (C) 2018 TypeFox and others.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// http://www.eclipse.org/legal/epl-2.0.
//
// This Source Code may also be made available under the following Secondary
// Licenses when the conditions for such availability set forth in the Eclipse
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
// with the GNU Classpath Exception which is available at
// https://www.gnu.org/software/classpath/license.html.
//
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
// *****************************************************************************
import URI from '@theia/core/lib/common/uri';
import { Disposable } from '@theia/core/lib/common';
import { Dimension, DiffNavigator, DeltaDecorationParams } from '@theia/editor/lib/browser';
import { MonacoEditorModel } from './monaco-editor-model';
import { EditorServiceOverrides, MonacoEditor, MonacoEditorServices } from './monaco-editor';
import { MonacoDiffNavigatorFactory } from './monaco-diff-navigator-factory';
import { DiffUris } from '@theia/core/lib/browser/diff-uris';
import * as monaco from '@theia/monaco-editor-core';
import { ICodeEditor, IDiffEditorConstructionOptions } from '@theia/monaco-editor-core/esm/vs/editor/browser/editorBrowser';
import { IActionDescriptor, IStandaloneCodeEditor, IStandaloneDiffEditor, StandaloneCodeEditor, StandaloneDiffEditor2 }
from '@theia/monaco-editor-core/esm/vs/editor/standalone/browser/standaloneCodeEditor';
import { IEditorConstructionOptions } from '@theia/monaco-editor-core/esm/vs/editor/browser/config/editorConfiguration';
import { ICodeEditorWidgetOptions } from '@theia/monaco-editor-core/esm/vs/editor/browser/widget/codeEditor/codeEditorWidget';
import { EmbeddedDiffEditorWidget } from '@theia/monaco-editor-core/esm/vs/editor/browser/widget/diffEditor/embeddedDiffEditorWidget';
import { IInstantiationService } from '@theia/monaco-editor-core/esm/vs/platform/instantiation/common/instantiation';
import { ContextKeyValue, IContextKey } from '@theia/monaco-editor-core/esm/vs/platform/contextkey/common/contextkey';
import { IDisposable } from '@theia/monaco-editor-core/esm/vs/base/common/lifecycle';
import { ICommandHandler } from '@theia/monaco-editor-core/esm/vs/platform/commands/common/commands';
import { EditorContextKeys } from '@theia/monaco-editor-core/esm/vs/editor/common/editorContextKeys';
import { IEditorOptions } from '@theia/monaco-editor-core/esm/vs/editor/common/config/editorOptions';
import { ILineChange } from '@theia/monaco-editor-core/esm/vs/editor/common/diff/legacyLinesDiffComputer';
export namespace MonacoDiffEditor {
export interface IOptions extends MonacoEditor.ICommonOptions, IDiffEditorConstructionOptions {
}
}
export class MonacoDiffEditor extends MonacoEditor {
protected _diffEditor: IStandaloneDiffEditor;
protected _diffNavigator: DiffNavigator;
protected readonly diffEditorModel: monaco.editor.IDiffEditorModel;
constructor(
uri: URI,
node: HTMLElement,
readonly originalModel: MonacoEditorModel,
readonly modifiedModel: MonacoEditorModel,
services: MonacoEditorServices,
protected readonly diffNavigatorFactory: MonacoDiffNavigatorFactory,
options?: MonacoDiffEditor.IOptions,
override?: EditorServiceOverrides,
parentEditor?: MonacoEditor
) {
super(uri, modifiedModel, node, services, options, override, parentEditor);
this.diffEditorModel = { original: this.originalModel.textEditorModel, modified: this.modifiedModel.textEditorModel };
this.documents.add(originalModel);
this.wordWrapOverride = options?.wordWrapOverride2;
this._diffNavigator = diffNavigatorFactory.createdDiffNavigator(this._diffEditor);
}
get diffEditor(): monaco.editor.IStandaloneDiffEditor {
return this._diffEditor as unknown as monaco.editor.IStandaloneDiffEditor;
}
get diffNavigator(): DiffNavigator {
return this._diffNavigator;
}
get diffInformation(): ILineChange[] {
return this._diffEditor.getLineChanges() || [];
}
protected override create(options?: IDiffEditorConstructionOptions, override?: EditorServiceOverrides): Disposable {
options = { ...options, fixedOverflowWidgets: true };
const instantiator = this.getInstantiatorWithOverrides(override);
/**
* @monaco-uplift. Should be guaranteed to work.
* Incomparable enums prevent TypeScript from believing that public IStandaloneDiffEditor is satisfied by private StandaloneDiffEditor
*/
this._diffEditor = this.parentEditor ?
instantiator.createInstance(EmbeddedDiffEditor, this.node, options, {}, this.parentEditor.getControl() as unknown as ICodeEditor) :
instantiator.createInstance(StandaloneDiffEditor2, this.node, options);
this.editor = this._diffEditor.getModifiedEditor() as unknown as monaco.editor.IStandaloneCodeEditor;
return this._diffEditor;
}
protected wordWrapOverride: IEditorOptions['wordWrapOverride2'];
protected lastReachedSideBySideBreakpoint = true;
protected override resize(dimension: Dimension | null): void {
if (this.node) {
const layoutSize = this.computeLayoutSize(this.node, dimension);
this._diffEditor.layout(layoutSize);
// Workaround for https://github.com/microsoft/vscode/issues/217386#issuecomment-2711750462
const leftEditor = this._diffEditor.getOriginalEditor();
const hasReachedSideBySideBreakpoint = leftEditor.contextKeyService
.getContextKeyValue(EditorContextKeys.diffEditorRenderSideBySideInlineBreakpointReached.key);
if (hasReachedSideBySideBreakpoint !== this.lastReachedSideBySideBreakpoint) {
leftEditor.updateOptions({ wordWrapOverride2: this.wordWrapOverride ?? hasReachedSideBySideBreakpoint ? 'off' : 'inherit' });
}
this.lastReachedSideBySideBreakpoint = !!hasReachedSideBySideBreakpoint;
}
}
override isActionSupported(id: string): boolean {
const action = this._diffEditor.getSupportedActions().find(a => a.id === id);
return !!action && action.isSupported() && super.isActionSupported(id);
}
override deltaDecorations(params: DeltaDecorationParams): string[] {
console.warn('`deltaDecorations` should be called on either the original, or the modified editor.');
return [];
}
override getResourceUri(): URI {
return new URI(this.modifiedModel.uri);
}
override createMoveToUri(resourceUri: URI): URI {
const [left, right] = DiffUris.decode(this.uri);
return DiffUris.encode(left.withPath(resourceUri.path), right.withPath(resourceUri.path));
}
override handleVisibilityChanged(nowVisible: boolean): void {
const isFirstShow = nowVisible && !this.savedViewState;
super.handleVisibilityChanged(nowVisible);
if (isFirstShow) {
this._diffEditor.revealFirstDiff();
}
}
override readonly onShouldDisplayDirtyDiffChanged = undefined;
override shouldDisplayDirtyDiff(): boolean {
return false;
}
override setShouldDisplayDirtyDiff(value: boolean): void {
// no op
}
protected override get baseEditor(): monaco.editor.IEditor {
return this.diffEditor;
}
protected override get baseModel(): monaco.editor.IEditorModel {
return this.diffEditorModel;
}
}
class EmbeddedDiffEditor extends EmbeddedDiffEditorWidget implements IStandaloneDiffEditor {
protected override _createInnerEditor(instantiationService: IInstantiationService, container: HTMLElement,
options: Readonly<IEditorConstructionOptions>, _editorWidgetOptions: ICodeEditorWidgetOptions): StandaloneCodeEditor {
return instantiationService.createInstance(StandaloneCodeEditor, container, options);
}
override getOriginalEditor(): IStandaloneCodeEditor {
return super.getOriginalEditor() as IStandaloneCodeEditor;
}
override getModifiedEditor(): IStandaloneCodeEditor {
return super.getModifiedEditor() as IStandaloneCodeEditor;
}
addCommand(keybinding: number, handler: ICommandHandler, context?: string): string | null {
return this.getModifiedEditor().addCommand(keybinding, handler, context);
}
createContextKey<T extends ContextKeyValue = ContextKeyValue>(key: string, defaultValue: T): IContextKey<T> {
return this.getModifiedEditor().createContextKey(key, defaultValue);
}
addAction(descriptor: IActionDescriptor): IDisposable {
return this.getModifiedEditor().addAction(descriptor);
}
}