From c35085d1d5b438810f5c00bc64c8264d90d33ca0 Mon Sep 17 00:00:00 2001 From: Henry Heino <46334387+personalizedrefrigerator@users.noreply.github.com> Date: Sat, 9 Mar 2024 02:48:22 -0800 Subject: [PATCH] Desktop: Improve beta editor support for the Rich Markdown plugin (#9935) --- .eslintignore | 1 + .gitignore | 1 + packages/app-desktop/app.ts | 4 +- .../NoteBody/CodeMirror/v5/Editor.tsx | 2 +- .../NoteBody/CodeMirror/v6/Editor.tsx | 6 ++ .../CodeMirror5BuiltInOptions.ts | 83 +++++++++++++++++++ .../CodeMirror5Emulation.ts | 46 ++++++++-- .../CodeMirror5Emulation/Decorator.ts | 3 + .../CodeMirror/markdown/decoratorExtension.ts | 12 +-- .../CodeMirror/pluginApi/PluginLoader.ts | 2 +- packages/editor/CodeMirror/theme.ts | 6 +- packages/tools/cspell/dictionary4.txt | 3 +- 12 files changed, 152 insertions(+), 17 deletions(-) create mode 100644 packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5BuiltInOptions.ts diff --git a/.eslintignore b/.eslintignore index facf53b2b..618ef565f 100644 --- a/.eslintignore +++ b/.eslintignore @@ -604,6 +604,7 @@ packages/default-plugins/utils/getCurrentCommitHash.js packages/default-plugins/utils/getPathToPatchFileFor.js packages/default-plugins/utils/readRepositoryJson.js packages/default-plugins/utils/waitForCliInput.js +packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5BuiltInOptions.js packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5Emulation.test.js packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5Emulation.js packages/editor/CodeMirror/CodeMirror5Emulation/Decorator.js diff --git a/.gitignore b/.gitignore index 32ddcc56e..665c9ffed 100644 --- a/.gitignore +++ b/.gitignore @@ -584,6 +584,7 @@ packages/default-plugins/utils/getCurrentCommitHash.js packages/default-plugins/utils/getPathToPatchFileFor.js packages/default-plugins/utils/readRepositoryJson.js packages/default-plugins/utils/waitForCliInput.js +packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5BuiltInOptions.js packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5Emulation.test.js packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5Emulation.js packages/editor/CodeMirror/CodeMirror5Emulation/Decorator.js diff --git a/packages/app-desktop/app.ts b/packages/app-desktop/app.ts index 23d7635b4..a75a95315 100644 --- a/packages/app-desktop/app.ts +++ b/packages/app-desktop/app.ts @@ -201,8 +201,10 @@ class Application extends BaseApplication { // The '*' and '!important' parts are necessary to make sure Russian text is displayed properly // https://github.com/laurent22/joplin/issues/155 + // + // Note: Be careful about the specificity here. Incorrect specificity can break monospaced fonts in tables. - const css = `.CodeMirror *, .cm-editor .cm-content { font-family: ${fontFamilies.join(', ')} !important; }`; + const css = `.CodeMirror5 *, .cm-editor .cm-content { font-family: ${fontFamilies.join(', ')} !important; }`; const styleTag = document.createElement('style'); styleTag.type = 'text/css'; styleTag.appendChild(document.createTextNode(css)); diff --git a/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/v5/Editor.tsx b/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/v5/Editor.tsx index 9f3bcd360..063f16c60 100644 --- a/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/v5/Editor.tsx +++ b/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/v5/Editor.tsx @@ -283,7 +283,7 @@ function Editor(props: EditorProps, ref: any) { } }, [pluginOptions, editor]); - return
; + return
; } export default forwardRef(Editor); diff --git a/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/v6/Editor.tsx b/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/v6/Editor.tsx index ee063c856..19fe5bb12 100644 --- a/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/v6/Editor.tsx +++ b/packages/app-desktop/gui/NoteEditor/NoteBody/CodeMirror/v6/Editor.tsx @@ -98,6 +98,12 @@ const Editor = (props: Props, ref: ForwardedRef) => { const editor = createEditor(editorContainerRef.current, editorProps); editor.addStyles({ '.cm-scroller': { overflow: 'auto' }, + '&.CodeMirror': { + height: 'unset', + background: 'unset', + overflow: 'unset', + direction: 'unset', + }, }); setEditor(editor); diff --git a/packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5BuiltInOptions.ts b/packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5BuiltInOptions.ts new file mode 100644 index 000000000..15d32e231 --- /dev/null +++ b/packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5BuiltInOptions.ts @@ -0,0 +1,83 @@ +import { Compartment, Extension, RangeSetBuilder, StateEffect } from '@codemirror/state'; +import { Decoration, DecorationSet, EditorView, ViewPlugin, ViewUpdate } from '@codemirror/view'; + +const activeLineDecoration = Decoration.line({ class: 'CodeMirror-activeline CodeMirror-activeline-background' }); + +const optionToExtension: Record = { + 'styleActiveLine': [ + ViewPlugin.fromClass(class { + public decorations: DecorationSet; + + public constructor(view: EditorView) { + this.updateDecorations(view); + } + + public update(update: ViewUpdate) { + this.updateDecorations(update.view); + } + + private updateDecorations(view: EditorView) { + const builder = new RangeSetBuilder(); + let lastLine = -1; + + for (const selection of view.state.selection.ranges) { + const startLine = selection.from; + const line = view.state.doc.lineAt(startLine); + + if (line.number !== lastLine) { + builder.add(line.from, line.from, activeLineDecoration); + } + + lastLine = line.number; + } + + this.decorations = builder.finish(); + } + }, { + decorations: plugin => plugin.decorations, + }), + EditorView.baseTheme({ + '&dark .CodeMirror-activeline-background': { + background: '#3304', + color: 'white', + }, + '&light .CodeMirror-activeline-background': { + background: '#7ff4', + color: 'black', + }, + }), + ], +}; + +// Maps several CM5 options to CM6 extensions +export default class CodeMirror5BuiltInOptions { + private activeOptions: string[] = []; + private extensionCompartment: Compartment = new Compartment(); + + public constructor(private editor: EditorView) { + editor.dispatch({ + effects: StateEffect.appendConfig.of(this.extensionCompartment.of([])), + }); + } + + private updateExtensions() { + const extensions = this.activeOptions.map(option => optionToExtension[option]); + this.editor.dispatch({ + effects: this.extensionCompartment.reconfigure(extensions), + }); + } + + public supportsOption(option: string) { + return optionToExtension.hasOwnProperty(option); + } + + public setOption(optionName: string, value: boolean) { + this.activeOptions = this.activeOptions.filter(other => other !== optionName); + + if (value) { + this.activeOptions.push(optionName); + } + + this.updateExtensions(); + } +} diff --git a/packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5Emulation.ts b/packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5Emulation.ts index f9dd4232c..9b1af2bf4 100644 --- a/packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5Emulation.ts +++ b/packages/editor/CodeMirror/CodeMirror5Emulation/CodeMirror5Emulation.ts @@ -8,6 +8,7 @@ import { StateEffect } from '@codemirror/state'; import { StreamParser } from '@codemirror/language'; import Decorator, { LineWidgetOptions, MarkTextOptions } from './Decorator'; import insertLineAfter from '../editorCommands/insertLineAfter'; +import CodeMirror5BuiltInOptions from './CodeMirror5BuiltInOptions'; const { pregQuote } = require('@joplin/lib/string-utils-common'); @@ -52,6 +53,7 @@ export default class CodeMirror5Emulation extends BaseCodeMirror5Emulation { private _options: Record = Object.create(null); private _decorator: Decorator; private _decoratorExtension: Extension; + private _builtInOptions: CodeMirror5BuiltInOptions; // Used by some plugins to store state. public state: Record = Object.create(null); @@ -70,6 +72,7 @@ export default class CodeMirror5Emulation extends BaseCodeMirror5Emulation { const { decorator, extension: decoratorExtension } = Decorator.create(editor); this._decorator = decorator; this._decoratorExtension = decoratorExtension; + this._builtInOptions = new CodeMirror5BuiltInOptions(editor); editor.dispatch({ effects: StateEffect.appendConfig.of(this.makeCM6Extensions()), @@ -129,10 +132,8 @@ export default class CodeMirror5Emulation extends BaseCodeMirror5Emulation { return { dom }; }), - // Note: We can allow legacy CM5 CSS to apply to the editor - // with a line similar to the following: - // EditorView.editorAttributes.of({ class: 'CodeMirror' }), - // Many of these styles, however, don't work well with CodeMirror 6. + // Allows legacy CM5 CSS to apply to the editor: + EditorView.editorAttributes.of({ class: 'CodeMirror' }), ]; } @@ -316,6 +317,8 @@ export default class CodeMirror5Emulation extends BaseCodeMirror5Emulation { const oldValue = this._options[name].value; this._options[name].value = value; this._options[name].onUpdate(this, value, oldValue); + } else if (this._builtInOptions.supportsOption(name)) { + this._builtInOptions.setOption(name, value); } else { super.setOption(name, value); } @@ -329,6 +332,20 @@ export default class CodeMirror5Emulation extends BaseCodeMirror5Emulation { } } + public override coordsChar(coords: { left: number; top: number }, mode?: 'div' | 'local'): DocumentPosition { + // codemirror-vim's API only supports "div" mode. Thus, we convert + // local to div: + if (mode !== 'div') { + const bbox = this.editor.contentDOM.getBoundingClientRect(); + coords = { + left: coords.left - bbox.left, + top: coords.top - bbox.top, + }; + } + + return super.coordsChar(coords, 'div'); + } + // codemirror-vim's API doesn't match the API docs here -- it expects addOverlay // to return a SearchQuery. As such, this override returns "any". public override addOverlay(modeObject: OverlayType): any { @@ -353,7 +370,26 @@ export default class CodeMirror5Emulation extends BaseCodeMirror5Emulation { } public addLineWidget(lineNumber: number, node: HTMLElement, options: LineWidgetOptions) { - this._decorator.addLineWidget(lineNumber, node, options); + return this._decorator.addLineWidget(lineNumber, node, options); + } + + public addWidget(pos: DocumentPosition, node: HTMLElement) { + if (node.parentElement) { + node.remove(); + } + + const loc = posFromDocumentPosition(this.editor.state.doc, pos); + const screenCoords = this.editor.coordsAtPos(loc); + const bbox = this.editor.contentDOM.getBoundingClientRect(); + + node.style.position = 'absolute'; + + const left = screenCoords.left - bbox.left; + node.style.left = `${left}px`; + node.style.maxWidth = `${bbox.width - left}px`; + node.style.top = `${screenCoords.top + this.editor.scrollDOM.scrollTop}px`; + + this.editor.scrollDOM.appendChild(node); } public markText(from: DocumentPosition, to: DocumentPosition, options?: MarkTextOptions) { diff --git a/packages/editor/CodeMirror/CodeMirror5Emulation/Decorator.ts b/packages/editor/CodeMirror/CodeMirror5Emulation/Decorator.ts index c5185b2fb..cd5d111e3 100644 --- a/packages/editor/CodeMirror/CodeMirror5Emulation/Decorator.ts +++ b/packages/editor/CodeMirror/CodeMirror5Emulation/Decorator.ts @@ -68,6 +68,9 @@ class WidgetDecorationWrapper extends WidgetType { container.classList.add(this.options.className); } + // Applies margins and related CSS: + container.classList.add('cm-line'); + return container; } } diff --git a/packages/editor/CodeMirror/markdown/decoratorExtension.ts b/packages/editor/CodeMirror/markdown/decoratorExtension.ts index 9eaee4686..a9627f79a 100644 --- a/packages/editor/CodeMirror/markdown/decoratorExtension.ts +++ b/packages/editor/CodeMirror/markdown/decoratorExtension.ts @@ -49,27 +49,27 @@ const blockQuoteDecoration = Decoration.line({ }); const header1LineDecoration = Decoration.line({ - attributes: { class: 'cm-h1 cm-headerLine' }, + attributes: { class: 'cm-h1 cm-headerLine cm-header' }, }); const header2LineDecoration = Decoration.line({ - attributes: { class: 'cm-h2 cm-headerLine' }, + attributes: { class: 'cm-h2 cm-headerLine cm-header' }, }); const header3LineDecoration = Decoration.line({ - attributes: { class: 'cm-h3 cm-headerLine' }, + attributes: { class: 'cm-h3 cm-headerLine cm-header' }, }); const header4LineDecoration = Decoration.line({ - attributes: { class: 'cm-h4 cm-headerLine' }, + attributes: { class: 'cm-h4 cm-headerLine cm-header' }, }); const header5LineDecoration = Decoration.line({ - attributes: { class: 'cm-h5 cm-headerLine' }, + attributes: { class: 'cm-h5 cm-headerLine cm-header' }, }); const header6LineDecoration = Decoration.line({ - attributes: { class: 'cm-h6 cm-headerLine' }, + attributes: { class: 'cm-h6 cm-headerLine cm-header' }, }); const tableHeaderDecoration = Decoration.line({ diff --git a/packages/editor/CodeMirror/pluginApi/PluginLoader.ts b/packages/editor/CodeMirror/pluginApi/PluginLoader.ts index 97ac246ab..d821997ec 100644 --- a/packages/editor/CodeMirror/pluginApi/PluginLoader.ts +++ b/packages/editor/CodeMirror/pluginApi/PluginLoader.ts @@ -132,7 +132,7 @@ export default class PluginLoader { pluginId: plugin.pluginId, contentScriptId: plugin.contentScriptId, }; - const loadedPlugin = exports.default(context); + const loadedPlugin = exports.default(context) ?? {}; loadedPlugin.plugin?.(this.editor); diff --git a/packages/editor/CodeMirror/theme.ts b/packages/editor/CodeMirror/theme.ts index c80851c80..fed60baea 100644 --- a/packages/editor/CodeMirror/theme.ts +++ b/packages/editor/CodeMirror/theme.ts @@ -85,10 +85,12 @@ const createTheme = (theme: EditorTheme): Extension[] => { }; const codeMirrorTheme = EditorView.theme({ - '&': baseGlobalStyle, + // Include &.CodeMirror to handle the case where additional CodeMirror 5 styles + // need to be overridden. + '&, &.CodeMirror': baseGlobalStyle, // These must be !important or more specific than CodeMirror's built-ins - '.cm-content': { + '& .cm-content': { fontFamily: theme.fontFamily, ...baseContentStyle, paddingBottom: theme.isDesktop ? '400px' : undefined, diff --git a/packages/tools/cspell/dictionary4.txt b/packages/tools/cspell/dictionary4.txt index f9f2b1ac0..0b649d1f6 100644 --- a/packages/tools/cspell/dictionary4.txt +++ b/packages/tools/cspell/dictionary4.txt @@ -91,4 +91,5 @@ activatable titlewrapper notyf Notyf -Prec \ No newline at end of file +activeline +Prec