9874a53206
Split out legacy modes to their own dynamically imported bundle to reduce main code bundle size.
207 lines
No EOL
5.8 KiB
JavaScript
207 lines
No EOL
5.8 KiB
JavaScript
import {EditorView, keymap} from "@codemirror/view";
|
|
import {copyTextToClipboard} from "../services/clipboard.js"
|
|
|
|
// Modes
|
|
import {viewer, editor} from "./setups.js";
|
|
import {createView, updateViewLanguage} from "./views.js";
|
|
|
|
/**
|
|
* Highlight pre elements on a page
|
|
*/
|
|
export function highlight() {
|
|
const codeBlocks = document.querySelectorAll('.page-content pre, .comment-box .content pre');
|
|
for (const codeBlock of codeBlocks) {
|
|
highlightElem(codeBlock);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Highlight all code blocks within the given parent element
|
|
* @param {HTMLElement} parent
|
|
*/
|
|
export function highlightWithin(parent) {
|
|
const codeBlocks = parent.querySelectorAll('pre');
|
|
for (const codeBlock of codeBlocks) {
|
|
highlightElem(codeBlock);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Add code highlighting to a single element.
|
|
* @param {HTMLElement} elem
|
|
*/
|
|
function highlightElem(elem) {
|
|
const innerCodeElem = elem.querySelector('code[class^=language-]');
|
|
elem.innerHTML = elem.innerHTML.replace(/<br\s*[\/]?>/gi ,'\n');
|
|
const content = elem.textContent.trimEnd();
|
|
|
|
let langName = '';
|
|
if (innerCodeElem !== null) {
|
|
langName = innerCodeElem.className.replace('language-', '');
|
|
}
|
|
|
|
const wrapper = document.createElement('div');
|
|
elem.parentNode.insertBefore(wrapper, elem);
|
|
|
|
const ev = createView({
|
|
parent: wrapper,
|
|
doc: content,
|
|
extensions: viewer(),
|
|
});
|
|
setMode(ev, langName, content);
|
|
|
|
elem.remove();
|
|
|
|
addCopyIcon(ev);
|
|
}
|
|
|
|
/**
|
|
* Add a button to a CodeMirror instance which copies the contents to the clipboard upon click.
|
|
* @param {EditorView} editorView
|
|
*/
|
|
function addCopyIcon(editorView) {
|
|
const copyIcon = `<svg viewBox="0 0 24 24" width="16" height="16" xmlns="http://www.w3.org/2000/svg"><path d="M0 0h24v24H0z" fill="none"/><path d="M16 1H4c-1.1 0-2 .9-2 2v14h2V3h12V1zm3 4H8c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h11c1.1 0 2-.9 2-2V7c0-1.1-.9-2-2-2zm0 16H8V7h11v14z"/></svg>`;
|
|
const copyButton = document.createElement('button');
|
|
copyButton.setAttribute('type', 'button')
|
|
copyButton.classList.add('cm-copy-button');
|
|
copyButton.innerHTML = copyIcon;
|
|
editorView.dom.appendChild(copyButton);
|
|
|
|
copyButton.addEventListener('click', event => {
|
|
copyTextToClipboard(editorView.state.doc.toString());
|
|
copyButton.classList.add('success');
|
|
setTimeout(() => {
|
|
copyButton.classList.remove('success');
|
|
}, 240);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Ge the theme to use for CodeMirror instances.
|
|
* @returns {*|string}
|
|
*/
|
|
function getTheme() {
|
|
const darkMode = document.documentElement.classList.contains('dark-mode');
|
|
return window.codeTheme || (darkMode ? 'darcula' : 'default');
|
|
}
|
|
|
|
/**
|
|
* Create a CodeMirror instance for showing inside the WYSIWYG editor.
|
|
* Manages a textarea element to hold code content.
|
|
* @param {HTMLElement} cmContainer
|
|
* @param {String} content
|
|
* @param {String} language
|
|
* @returns {{wrap: Element, editor: *}}
|
|
*/
|
|
export function wysiwygView(cmContainer, content, language) {
|
|
return CodeMirror(cmContainer, {
|
|
value: content,
|
|
mode: getMode(language, content),
|
|
lineNumbers: true,
|
|
lineWrapping: false,
|
|
theme: getTheme(),
|
|
readOnly: true
|
|
});
|
|
}
|
|
|
|
|
|
/**
|
|
* Create a CodeMirror instance to show in the WYSIWYG pop-up editor
|
|
* @param {HTMLElement} elem
|
|
* @param {String} modeSuggestion
|
|
* @returns {*}
|
|
*/
|
|
export function popupEditor(elem, modeSuggestion) {
|
|
const content = elem.textContent;
|
|
|
|
return CodeMirror(function(elt) {
|
|
elem.parentNode.insertBefore(elt, elem);
|
|
elem.style.display = 'none';
|
|
}, {
|
|
value: content,
|
|
mode: getMode(modeSuggestion, content),
|
|
lineNumbers: true,
|
|
lineWrapping: false,
|
|
theme: getTheme()
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Create an inline editor to replace the given textarea.
|
|
* @param {HTMLTextAreaElement} textArea
|
|
* @param {String} mode
|
|
* @returns {CodeMirror3}
|
|
*/
|
|
export function inlineEditor(textArea, mode) {
|
|
return CodeMirror.fromTextArea(textArea, {
|
|
mode: getMode(mode, textArea.value),
|
|
lineNumbers: true,
|
|
lineWrapping: false,
|
|
theme: getTheme(),
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Set the language mode of a codemirror EditorView.
|
|
*
|
|
* @param {EditorView} ev
|
|
* @param {string} modeSuggestion
|
|
* @param {string} content
|
|
*/
|
|
export function setMode(ev, modeSuggestion, content) {
|
|
updateViewLanguage(ev, modeSuggestion, content);
|
|
}
|
|
|
|
/**
|
|
* Set the content of a cm instance.
|
|
* @param cmInstance
|
|
* @param codeContent
|
|
*/
|
|
export function setContent(cmInstance, codeContent) {
|
|
cmInstance.setValue(codeContent);
|
|
setTimeout(() => {
|
|
updateLayout(cmInstance);
|
|
}, 10);
|
|
}
|
|
|
|
/**
|
|
* Update the layout (codemirror refresh) of a cm instance.
|
|
* @param cmInstance
|
|
*/
|
|
export function updateLayout(cmInstance) {
|
|
cmInstance.refresh();
|
|
}
|
|
|
|
/**
|
|
* Get a CodeMirror instance to use for the markdown editor.
|
|
* @param {HTMLElement} elem
|
|
* @param {function} onChange
|
|
* @param {object} domEventHandlers
|
|
* @param {Array} keyBindings
|
|
* @returns {*}
|
|
*/
|
|
export function markdownEditor(elem, onChange, domEventHandlers, keyBindings) {
|
|
const content = elem.textContent;
|
|
const config = {
|
|
parent: elem.parentNode,
|
|
doc: content,
|
|
extensions: [
|
|
...editor('markdown'),
|
|
EditorView.updateListener.of((v) => {
|
|
onChange(v);
|
|
}),
|
|
EditorView.domEventHandlers(domEventHandlers),
|
|
keymap.of(keyBindings),
|
|
],
|
|
};
|
|
|
|
// Emit a pre-event public event to allow tweaking of the configure before view creation.
|
|
window.$events.emitPublic(elem, 'editor-markdown-cm::pre-init', {cmEditorViewConfig: config});
|
|
|
|
// Create editor view, hide original input
|
|
const ev = createView(config);
|
|
setMode(ev, 'markdown', '');
|
|
elem.style.display = 'none';
|
|
|
|
return ev;
|
|
} |