import {addScript} from "../util/addScript"; import {Constants} from "../../constants"; import {focusByOffset} from "../util/selection"; import {setCodeTheme} from "../../util/assets"; export const highlightRender = (element: Element, cdn = Constants.PROTYLE_CDN) => { let codeElements: NodeListOf; let isPreview = false; if (element.classList.contains("code-block")) { // 编辑器内代码块编辑渲染 codeElements = element.querySelectorAll('[spellcheck="false"]'); } else { if (element.classList.contains("item__readme")) { // bazaar reademe codeElements = element.querySelectorAll("pre code"); codeElements.forEach(item => { item.parentElement.setAttribute("lineNumber", "false"); }); } else if (element.classList.contains("b3-typography")) { // preview & export html markdown codeElements = element.querySelectorAll(".code-block code"); isPreview = true; } else { codeElements = element.querySelectorAll('.code-block [spellcheck="false"]'); } } if (codeElements.length === 0) { return; } setCodeTheme(cdn); addScript(`${cdn}/js/highlight.js/highlight.min.js?v=11.5.0`, "protyleHljsScript").then(() => { addScript(`${cdn}/js/highlight.js/third-languages.js?v=1.0.0`, "protyleHljsThirdScript").then(() => { codeElements.forEach((block: HTMLElement) => { if (block.getAttribute("data-render") === "true") { return; } const wbrElement = block.querySelector("wbr"); let startIndex = 0; if (wbrElement) { let previousSibling = wbrElement.previousSibling; while (previousSibling) { startIndex += previousSibling.textContent.length; while (!previousSibling.previousSibling && previousSibling.parentElement.tagName !== "DIV") { // 高亮 span 中输入 previousSibling = previousSibling.parentElement; } previousSibling = previousSibling.previousSibling; } wbrElement.remove(); } let language; if (isPreview) { language = block.parentElement.getAttribute("data-language"); // preview } else if (block.previousElementSibling) { language = block.previousElementSibling.firstElementChild.textContent; } else { // bazaar readme language = block.className.replace("language-", ""); } if (!hljs.getLanguage(language)) { language = "plaintext"; } block.classList.add("hljs"); // TODO 等待讨论是否需要渲染 if (!hasClosestByAttribute(block, "id", "searchPreview", true) || isPreview) { block.innerHTML = hljs.highlight( block.textContent + (block.textContent.endsWith("\n") ? "" : "\n"), // https://github.com/siyuan-note/siyuan/issues/4609 { language, ignoreIllegals: true }).value; // } block.setAttribute("data-render", "true"); const autoEnter = block.parentElement.getAttribute("linewrap"); const ligatures = block.parentElement.getAttribute("ligatures"); const lineNumber = block.parentElement.getAttribute("linenumber"); if (autoEnter === "true" || (autoEnter !== "false" && window.siyuan.config.editor.codeLineWrap)) { block.style.setProperty("white-space", "pre-wrap"); block.style.setProperty("word-break", "break-all"); } else { block.style.setProperty("white-space", "pre",); block.style.setProperty("word-break", "initial",); } if (ligatures === "true" || (ligatures !== "false" && window.siyuan.config.editor.codeLigatures)) { block.style.fontVariantLigatures = "normal"; } else { block.style.fontVariantLigatures = "none"; } const languageElement = block.parentElement.querySelector(".protyle-action__language") as HTMLElement; if (!isPreview && (lineNumber === "true" || (lineNumber !== "false" && window.siyuan.config.editor.codeSyntaxHighlightLineNum))) { // 需要先添加 class 以防止抖动 https://ld246.com/article/1648116585443 block.classList.add("protyle-linenumber"); setTimeout(() => { // windows 需等待字体下载完成再计算,否则导致不换行,高度计算错误 lineNumberRender(block); }, 20); if (languageElement) { languageElement.style.marginLeft = "3.6em"; } } else if (block.nextElementSibling?.classList.contains("protyle-linenumber__rows")) { block.classList.remove("protyle-linenumber"); block.nextElementSibling.remove(); if (languageElement) { languageElement.style.marginLeft = ""; } } if (wbrElement && getSelection().rangeCount > 0) { focusByOffset(block, startIndex, startIndex); } }); }); }); }; export const lineNumberRender = (block: HTMLElement) => { if (block.parentElement.getAttribute("lineNumber") === "false") { return; } if (block.nextElementSibling && block.nextElementSibling.clientHeight === block.clientHeight) { return; } block.classList.add("protyle-linenumber"); const lineNumberTemp = document.createElement("div"); lineNumberTemp.className = "hljs protyle-linenumber"; lineNumberTemp.setAttribute("style", `padding-top:0 !important;padding-bottom:0 !important;min-height:auto !important;white-space:${block.style.whiteSpace};word-break:${block.style.wordBreak};font-variant-ligatures:${block.style.fontVariantLigatures};`); lineNumberTemp.setAttribute("contenteditable", "true"); block.insertAdjacentElement("afterend", lineNumberTemp); let lineNumberHTML = ""; const lineList = block.textContent.split(/\r\n|\r|\n|\u2028|\u2029/g); if (lineList[lineList.length - 1] === "" && lineList.length > 1) { lineList.pop(); } const isWrap = block.style.wordBreak === "break-all"; lineList.map((line) => { let lineHeight = ""; if (isWrap) { lineNumberTemp.textContent = line || "\n"; const height = lineNumberTemp.getBoundingClientRect().height; lineHeight = ` style="height:${height}px;"`; } lineNumberHTML += ``; }); lineNumberTemp.remove(); if (block.nextElementSibling?.classList.contains("protyle-linenumber__rows")) { block.nextElementSibling.innerHTML = lineNumberHTML; } else { block.insertAdjacentHTML("afterend", `${lineNumberHTML}`); } };