FIX: adds katex support for chat (#88)
Also drops jquery support and uses `decorateCookedElement` instead of `decorateCooked` for posts.
This commit is contained in:
parent
35e80a0007
commit
686736d3c2
|
@ -1,72 +1,84 @@
|
||||||
import $ from "jquery";
|
|
||||||
import loadScript from "discourse/lib/load-script";
|
import loadScript from "discourse/lib/load-script";
|
||||||
import { withPluginApi } from "discourse/lib/plugin-api";
|
import { withPluginApi } from "discourse/lib/plugin-api";
|
||||||
|
|
||||||
function ensureKaTeX() {
|
async function ensureKaTeX() {
|
||||||
return loadScript("/plugins/discourse-math/katex/katex.min.js").then(() => {
|
try {
|
||||||
return loadScript("/plugins/discourse-math/katex/katex.min.css", {
|
await loadScript("/plugins/discourse-math/katex/katex.min.js");
|
||||||
|
await loadScript("/plugins/discourse-math/katex/katex.min.css", {
|
||||||
css: true,
|
css: true,
|
||||||
})
|
});
|
||||||
.then(() => {
|
await loadScript("/plugins/discourse-math/katex/mhchem.min.js");
|
||||||
return loadScript("/plugins/discourse-math/katex/mhchem.min.js");
|
await loadScript("/plugins/discourse-math/katex/copy-tex.min.js");
|
||||||
})
|
} catch (e) {
|
||||||
.then(() => {
|
// eslint-disable-next-line no-console
|
||||||
return loadScript("/plugins/discourse-math/katex/copy-tex.min.js");
|
console.error("Failed to load KaTeX dependencies.", e);
|
||||||
});
|
}
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function decorate(elem, katexOpts) {
|
function decorate(elem, katexOpts) {
|
||||||
const $elem = $(elem);
|
|
||||||
katexOpts["displayMode"] = elem.tagName === "DIV";
|
katexOpts["displayMode"] = elem.tagName === "DIV";
|
||||||
|
|
||||||
if ($elem.data("applied-katex")) {
|
if (elem.dataset.appliedKatex) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
$elem.data("applied-katex", true);
|
|
||||||
|
|
||||||
if ($elem.hasClass("math")) {
|
elem.dataset.appliedKatex = true;
|
||||||
const tag = elem.tagName === "DIV" ? "div" : "span";
|
|
||||||
const displayClass = tag === "div" ? "block-math" : "inline-math";
|
if (!elem.classList.contains("math")) {
|
||||||
const text = $elem.text();
|
return;
|
||||||
$elem.addClass(`math-container ${displayClass} katex-math`).text("");
|
|
||||||
window.katex.render(text, elem, katexOpts);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const tag = elem.tagName === "DIV" ? "div" : "span";
|
||||||
|
const displayClass = tag === "div" ? "block-math" : "inline-math";
|
||||||
|
const text = elem.textContent;
|
||||||
|
elem.classList.add("math-container", displayClass, "katex-math");
|
||||||
|
elem.textContent = "";
|
||||||
|
window.katex.render(text, elem, katexOpts);
|
||||||
}
|
}
|
||||||
|
|
||||||
function katex($elem) {
|
async function katex(elem) {
|
||||||
if (!$elem || !$elem.find) {
|
if (!elem) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const mathElems = $elem.find(".math");
|
const mathElems = elem.querySelectorAll(".math");
|
||||||
|
if (!mathElems.length > 0) {
|
||||||
if (mathElems.length > 0) {
|
return;
|
||||||
ensureKaTeX().then(() => {
|
|
||||||
// enable persistent macros with are disabled by default: https://katex.org/docs/api.html#persistent-macros
|
|
||||||
// also enable equation labelling and referencing which are disabled by default
|
|
||||||
// both of these are enabled in mathjax by default, so now the katex implementation is (more) mathjax compatible
|
|
||||||
const katexOpts = {
|
|
||||||
trust: (context) => ["\\htmlId", "\\href"].includes(context.command),
|
|
||||||
macros: {
|
|
||||||
"\\eqref": "\\href{###1}{(\\text{#1})}",
|
|
||||||
"\\ref": "\\href{###1}{\\text{#1}}",
|
|
||||||
"\\label": "\\htmlId{#1}{}",
|
|
||||||
},
|
|
||||||
displayMode: false,
|
|
||||||
};
|
|
||||||
mathElems.each((idx, elem) => decorate(elem, katexOpts));
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
await ensureKaTeX();
|
||||||
|
|
||||||
|
// enable persistent macros with are disabled by default: https://katex.org/docs/api.html#persistent-macros
|
||||||
|
// also enable equation labelling and referencing which are disabled by default
|
||||||
|
// both of these are enabled in mathjax by default, so now the katex implementation is (more) mathjax compatible
|
||||||
|
const katexOpts = {
|
||||||
|
trust: (context) => ["\\htmlId", "\\href"].includes(context.command),
|
||||||
|
macros: {
|
||||||
|
"\\eqref": "\\href{###1}{(\\text{#1})}",
|
||||||
|
"\\ref": "\\href{###1}{\\text{#1}}",
|
||||||
|
"\\label": "\\htmlId{#1}{}",
|
||||||
|
},
|
||||||
|
displayMode: false,
|
||||||
|
};
|
||||||
|
mathElems.forEach((mathElem) => decorate(mathElem, katexOpts));
|
||||||
}
|
}
|
||||||
|
|
||||||
function initializeMath(api) {
|
function initializeMath(api) {
|
||||||
api.decorateCooked(
|
api.decorateCookedElement(
|
||||||
function (elem) {
|
function (elem) {
|
||||||
katex(elem);
|
katex(elem);
|
||||||
},
|
},
|
||||||
{ id: "katex" }
|
{ id: "katex" }
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (api.decorateChatMessage) {
|
||||||
|
api.decorateChatMessage(
|
||||||
|
(element) => {
|
||||||
|
katex(element);
|
||||||
|
},
|
||||||
|
{ id: "katex-chat" }
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
|
Loading…
Reference in New Issue