@theia/monaco
Version:
Theia - Monaco Extension
111 lines (101 loc) • 5.58 kB
text/typescript
// *****************************************************************************
// Copyright (C) 2022 Ericsson 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 { inject, injectable, postConstruct } from '@theia/core/shared/inversify';
import { ILanguageService } from '@theia/monaco-editor-core/esm/vs/editor/common/languages/language';
import { MarkdownRenderer as CodeMarkdownRenderer, IMarkdownRendererOptions }
from '@theia/monaco-editor-core/esm/vs/editor/browser/widget/markdownRenderer/browser/markdownRenderer';
import { StandaloneServices } from '@theia/monaco-editor-core/esm/vs/editor/standalone/browser/standaloneServices';
import * as monaco from '@theia/monaco-editor-core';
import { OpenerService, PreferenceService, WidgetOpenerOptions, open } from '@theia/core/lib/browser';
import { IOpenerService, OpenExternalOptions, OpenInternalOptions } from '@theia/monaco-editor-core/esm/vs/platform/opener/common/opener';
import { HttpOpenHandlerOptions } from '@theia/core/lib/browser/http-open-handler';
import { URI } from '@theia/core/lib/common/uri';
import { MarkdownRenderer, MarkdownRenderOptions, MarkdownRenderResult } from '@theia/core/lib/browser/markdown-rendering/markdown-renderer';
import { MarkedOptions, MarkdownRenderOptions as MonacoMarkdownRenderOptions } from '@theia/monaco-editor-core/esm/vs/base/browser/markdownRenderer';
import { MarkdownString } from '@theia/core/lib/common/markdown-rendering';
import { DisposableStore } from '@theia/monaco-editor-core/esm/vs/base/common/lifecycle';
import { DisposableCollection, DisposableGroup } from '@theia/core';
()
export class MonacoMarkdownRenderer implements MarkdownRenderer {
(OpenerService) protected readonly openerService: OpenerService;
(PreferenceService) protected readonly preferences: PreferenceService;
protected delegate: CodeMarkdownRenderer;
protected _openerService: OpenerService | undefined;
render(markdown: MarkdownString | undefined, options?: MarkdownRenderOptions, markedOptions?: MarkedOptions): MarkdownRenderResult {
return this.delegate.render(markdown, this.transformOptions(options), markedOptions);
}
protected transformOptions(options?: MarkdownRenderOptions): MonacoMarkdownRenderOptions | undefined {
if (!options?.actionHandler) {
return options as MarkdownRenderOptions & { actionHandler: undefined } | undefined;
}
const monacoActionHandler: MonacoMarkdownRenderOptions['actionHandler'] = {
disposables: this.toDisposableStore(options.actionHandler.disposables),
callback: (content, e) => options.actionHandler!.callback(content, e?.browserEvent)
};
return { ...options, actionHandler: monacoActionHandler };
}
protected toDisposableStore(current: DisposableGroup): DisposableStore {
if (current instanceof DisposableStore) {
return current;
} else if (current instanceof DisposableCollection) {
const store = new DisposableStore();
current['disposables'].forEach(disposable => store.add(disposable));
return store;
} else {
return new DisposableStore();
}
}
()
protected init(): void {
const languages = StandaloneServices.get(ILanguageService);
const openerService = StandaloneServices.get(IOpenerService);
openerService.registerOpener({
open: (u, options) => this.interceptOpen(u, options)
});
const that = this;
const prefs = new class implements IMarkdownRendererOptions {
get codeBlockFontFamily(): string | undefined {
return that.preferences.get<string>('editor.fontFamily');
}
};
this.delegate = new CodeMarkdownRenderer(prefs, languages, openerService);
}
protected async interceptOpen(monacoUri: monaco.Uri | string, monacoOptions?: OpenInternalOptions | OpenExternalOptions): Promise<boolean> {
let options = undefined;
if (monacoOptions) {
if ('openToSide' in monacoOptions && monacoOptions.openToSide) {
options = Object.assign(options || {}, <WidgetOpenerOptions>{
widgetOptions: {
mode: 'split-right'
}
});
}
if ('openExternal' in monacoOptions && monacoOptions.openExternal) {
options = Object.assign(options || {}, <HttpOpenHandlerOptions>{
openExternal: true
});
}
}
const uri = new URI(monacoUri.toString());
try {
await open(this.openerService, uri, options);
return true;
} catch (e) {
console.error(`Fail to open '${uri.toString()}':`, e);
return false;
}
}
}