import {EditorView} from "@codemirror/view" // import {EditorState} from "@codemirror/state" import Clipboard from "clipboard/dist/clipboard.min"; // Modes import {modes, modeMap, modesAsStreamLanguages} from "./modes"; import {viewer} from "./setups.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(//gi ,'\n'); const content = elem.textContent.trimEnd(); let mode = ''; if (innerCodeElem !== null) { const langName = innerCodeElem.className.replace('language-', ''); mode = getMode(langName, content); } const wrapper = document.createElement('div'); elem.parentNode.insertBefore(wrapper, elem); const cm = new EditorView({ parent: wrapper, doc: content, extensions: viewer(), }); elem.remove(); // TODO - theme: getTheme(), // TODO - mode, addCopyIcon(cm); } /** * Add a button to a CodeMirror instance which copies the contents to the clipboard upon click. * @param cmInstance */ function addCopyIcon(cmInstance) { // TODO // const copyIcon = ``; // const copyButton = document.createElement('div'); // copyButton.classList.add('CodeMirror-copy'); // copyButton.innerHTML = copyIcon; // cmInstance.display.wrapper.appendChild(copyButton); // // const clipboard = new Clipboard(copyButton, { // text: function(trigger) { // return cmInstance.getValue() // } // }); // // clipboard.on('success', event => { // copyButton.classList.add('success'); // setTimeout(() => { // copyButton.classList.remove('success'); // }, 240); // }); } /** * Search for a codemirror code based off a user suggestion * @param {String} suggestion * @param {String} content * @returns {string} */ function getMode(suggestion, content) { suggestion = suggestion.trim().replace(/^\./g, '').toLowerCase(); const modeMapType = typeof modeMap[suggestion]; if (modeMapType === 'undefined') { return ''; } if (modeMapType === 'function') { return modeMap[suggestion](content); } return modeMap[suggestion]; } /** * 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 mode of a codemirror instance. * @param cmInstance * @param modeSuggestion */ export function setMode(cmInstance, modeSuggestion, content) { cmInstance.setOption('mode', getMode(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 * @returns {*} */ export function markdownEditor(elem) { const content = elem.textContent; const config = { value: content, mode: "markdown", lineNumbers: true, lineWrapping: true, theme: getTheme(), scrollPastEnd: true, }; window.$events.emitPublic(elem, 'editor-markdown-cm::pre-init', {config}); return CodeMirror(function (elt) { elem.parentNode.insertBefore(elt, elem); elem.style.display = 'none'; }, config); } /** * Get the 'meta' key dependent on the user's system. * @returns {string} */ export function getMetaKey() { let mac = CodeMirror.keyMap["default"] == CodeMirror.keyMap.macDefault; return mac ? "Cmd" : "Ctrl"; }