@theia/monaco
Version:
Theia - Monaco Extension
185 lines (163 loc) • 9.34 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 { 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 savedDiffState: monaco.editor.IDiffEditorViewState | null;
protected originalTextModel: monaco.editor.ITextModel;
protected modifiedTextModel: monaco.editor.ITextModel;
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.originalTextModel = originalModel.textEditorModel;
this.modifiedTextModel = modifiedModel.textEditorModel;
this.documents.add(originalModel);
const original = originalModel.textEditorModel;
const modified = modifiedModel.textEditorModel;
this.wordWrapOverride = options?.wordWrapOverride2;
this._diffNavigator = diffNavigatorFactory.createdDiffNavigator(this._diffEditor);
this._diffEditor.setModel({ original, modified });
}
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.originalModel.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 shouldDisplayDirtyDiff(): boolean {
return false;
}
override handleVisibilityChanged(nowVisible: boolean): void {
if (nowVisible) {
this.diffEditor.setModel({ original: this.originalTextModel, modified: this.modifiedTextModel });
this.diffEditor.restoreViewState(this.savedDiffState);
this.diffEditor.focus();
} else {
const originalModel = this.diffEditor.getOriginalEditor().getModel();
if (originalModel) {
this.originalTextModel = originalModel;
}
const modifiedModel = this.diffEditor.getModifiedEditor().getModel();
if (modifiedModel) {
this.modifiedTextModel = modifiedModel;
}
this.savedDiffState = this.diffEditor.saveViewState();
// eslint-disable-next-line no-null/no-null
this.diffEditor.setModel(null);
}
}
}
class EmbeddedDiffEditor extends EmbeddedDiffEditorWidget implements IStandaloneDiffEditor {
protected override _createInnerEditor(instantiationService: IInstantiationService, container: HTMLElement,
options: Readonly<IEditorConstructionOptions>): 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);
}
}