matrix-react-sdk
Version:
SDK for matrix.org using React
231 lines (223 loc) • 31.6 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.containsEmote = containsEmote;
exports.htmlSerializeFromMdIfNeeded = htmlSerializeFromMdIfNeeded;
exports.htmlSerializeIfNeeded = htmlSerializeIfNeeded;
exports.mdSerialize = mdSerialize;
exports.startsWith = startsWith;
exports.stripEmoteCommand = stripEmoteCommand;
exports.stripPrefix = stripPrefix;
exports.textSerialize = textSerialize;
exports.unescapeMessage = unescapeMessage;
var _htmlEntities = require("html-entities");
var _escapeHtml = _interopRequireDefault(require("escape-html"));
var _Markdown = _interopRequireDefault(require("../Markdown"));
var _Permalinks = require("../utils/permalinks/Permalinks");
var _SettingsStore = _interopRequireDefault(require("../settings/SettingsStore"));
var _SdkConfig = _interopRequireDefault(require("../SdkConfig"));
var _parts = require("./parts");
/*
Copyright 2024 New Vector Ltd.
Copyright 2019, 2020 The Matrix.org Foundation C.I.C.
Copyright 2019 New Vector Ltd
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
function mdSerialize(model) {
return model.parts.reduce((html, part) => {
switch (part.type) {
case _parts.Type.Newline:
return html + "\n";
case _parts.Type.Plain:
case _parts.Type.Emoji:
case _parts.Type.Command:
case _parts.Type.PillCandidate:
case _parts.Type.AtRoomPill:
return html + part.text;
case _parts.Type.RoomPill:
{
const url = (0, _Permalinks.makeGenericPermalink)(part.resourceId, true);
// Escape square brackets and backslashes
// Here we use the resourceId for compatibility with non-rich text clients
// See https://github.com/vector-im/element-web/issues/16660
const title = part.resourceId.replace(/[[\\\]]/g, c => "\\" + c);
return html + `[${title}](${url})`;
}
case _parts.Type.UserPill:
{
const url = (0, _Permalinks.makeGenericPermalink)(part.resourceId, true);
// Escape square brackets and backslashes; convert newlines to HTML
const title = part.text.replace(/[[\\\]]/g, c => "\\" + c).replace(/\n/g, "<br>");
return html + `[${title}](${url})`;
}
}
}, "");
}
function htmlSerializeIfNeeded(model, {
forceHTML = false,
useMarkdown = true
} = {}) {
if (!useMarkdown) {
return (0, _escapeHtml.default)(textSerialize(model)).replace(/\n/g, "<br/>");
}
const md = mdSerialize(model);
return htmlSerializeFromMdIfNeeded(md, {
forceHTML
});
}
function htmlSerializeFromMdIfNeeded(md, {
forceHTML = false
} = {}) {
// copy of raw input to remove unwanted math later
const orig = md;
if (_SettingsStore.default.getValue("feature_latex_maths")) {
const patternNames = ["tex", "latex"];
const patternTypes = ["display", "inline"];
const patternDefaults = {
tex: {
// detect math with tex delimiters, inline: $...$, display $$...$$
// preferably use negative lookbehinds, not supported in all major browsers:
// const displayPattern = "^(?<!\\\\)\\$\\$(?![ \\t])(([^$]|\\\\\\$)+?)\\$\\$$";
// const inlinePattern = "(?:^|\\s)(?<!\\\\)\\$(?!\\s)(([^$]|\\\\\\$)+?)(?<!\\\\|\\s)\\$";
// conditions for display math detection $$...$$:
// - pattern starts and ends on a new line
// - left delimiter ($$) is not escaped by backslash
display: "(^)\\$\\$(([^$]|\\\\\\$)+?)\\$\\$$",
// conditions for inline math detection $...$:
// - pattern starts at beginning of line, follows whitespace character or punctuation
// - pattern is on a single line
// - left and right delimiters ($) are not escaped by backslashes
// - left delimiter is not followed by whitespace character
// - right delimiter is not prefixed with whitespace character
inline: "(^|\\s|[.,!?:;])(?!\\\\)\\$(?!\\s)(([^$\\n]|\\\\\\$)*([^\\\\\\s\\$]|\\\\\\$)(?:\\\\\\$)?)\\$"
},
latex: {
// detect math with latex delimiters, inline: \(...\), display \[...\]
// conditions for display math detection \[...\]:
// - pattern starts and ends on a new line
// - pattern is not empty
display: "(^)\\\\\\[(?!\\\\\\])(.*?)\\\\\\]$",
// conditions for inline math detection \(...\):
// - pattern starts at beginning of line or is not prefixed with backslash
// - pattern is not empty
inline: "(^|[^\\\\])\\\\\\((?!\\\\\\))(.*?)\\\\\\)"
}
};
patternNames.forEach(function (patternName) {
patternTypes.forEach(function (patternType) {
// get the regex replace pattern from config or use the default
const pattern = _SdkConfig.default.get("latex_maths_delims")?.[patternType]?.["pattern"]?.[patternName] || patternDefaults[patternName][patternType];
md = md.replace(RegExp(pattern, "gms"), function (m, p1, p2) {
const p2e = (0, _htmlEntities.encode)(p2);
switch (patternType) {
case "display":
return `${p1}<div data-mx-maths="${p2e}">\n\n</div>\n\n`;
case "inline":
return `${p1}<span data-mx-maths="${p2e}"></span>`;
}
});
});
});
// make sure div tags always start on a new line, otherwise it will confuse the markdown parser
md = md.replace(/(.)<div/g, function (m, p1) {
return `${p1}\n<div`;
});
}
const parser = new _Markdown.default(md);
if (!parser.isPlainText() || forceHTML) {
// feed Markdown output to HTML parser
const phtml = new DOMParser().parseFromString(parser.toHTML(), "text/html");
if (_SettingsStore.default.getValue("feature_latex_maths")) {
// original Markdown without LaTeX replacements
const parserOrig = new _Markdown.default(orig);
const phtmlOrig = new DOMParser().parseFromString(parserOrig.toHTML(), "text/html");
// since maths delimiters are handled before Markdown,
// code blocks could contain mangled content.
// replace code blocks with original content
[...phtmlOrig.getElementsByTagName("code")].forEach((e, i) => {
phtml.getElementsByTagName("code").item(i).textContent = e.textContent;
});
// add fallback output for latex math, which should not be interpreted as markdown
[...phtml.querySelectorAll("div, span")].forEach((e, i) => {
const tex = e.getAttribute("data-mx-maths");
if (tex) {
e.innerHTML = `<code>${tex}</code>`;
}
});
}
return phtml.body.innerHTML;
}
// ensure removal of escape backslashes in non-Markdown messages
if (md.indexOf("\\") > -1) {
return parser.toPlaintext();
}
}
function textSerialize(model) {
return model.parts.reduce((text, part) => {
switch (part.type) {
case _parts.Type.Newline:
return text + "\n";
case _parts.Type.Plain:
case _parts.Type.Emoji:
case _parts.Type.Command:
case _parts.Type.PillCandidate:
case _parts.Type.AtRoomPill:
return text + part.text;
case _parts.Type.RoomPill:
// Here we use the resourceId for compatibility with non-rich text clients
// See https://github.com/vector-im/element-web/issues/16660
return text + `${part.resourceId}`;
case _parts.Type.UserPill:
return text + `${part.text}`;
}
}, "");
}
function containsEmote(model) {
const hasCommand = startsWith(model, "/me ", false);
const hasArgument = model.parts[0]?.text?.length > 4 || model.parts.length > 1;
return hasCommand && hasArgument;
}
function startsWith(model, prefix, caseSensitive = true) {
const firstPart = model.parts[0];
// part type will be "plain" while editing,
// and "command" while composing a message.
let text = firstPart?.text || "";
if (!caseSensitive) {
prefix = prefix.toLowerCase();
text = text.toLowerCase();
}
return firstPart && (firstPart.type === _parts.Type.Plain || firstPart.type === _parts.Type.Command) && text.startsWith(prefix);
}
function stripEmoteCommand(model) {
// trim "/me "
return stripPrefix(model, "/me ");
}
function stripPrefix(model, prefix) {
model = model.clone();
model.removeText({
index: 0,
offset: 0
}, prefix.length);
return model;
}
function unescapeMessage(model) {
const {
parts
} = model;
if (parts.length) {
const firstPart = parts[0];
// only unescape \/ to / at start of editor
if (firstPart.type === _parts.Type.Plain && firstPart.text.startsWith("\\/")) {
model = model.clone();
model.removeText({
index: 0,
offset: 0
}, 1);
}
}
return model;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_htmlEntities","require","_escapeHtml","_interopRequireDefault","_Markdown","_Permalinks","_SettingsStore","_SdkConfig","_parts","mdSerialize","model","parts","reduce","html","part","type","Type","Newline","Plain","Emoji","Command","PillCandidate","AtRoomPill","text","RoomPill","url","makeGenericPermalink","resourceId","title","replace","c","UserPill","htmlSerializeIfNeeded","forceHTML","useMarkdown","escapeHtml","textSerialize","md","htmlSerializeFromMdIfNeeded","orig","SettingsStore","getValue","patternNames","patternTypes","patternDefaults","tex","display","inline","latex","forEach","patternName","patternType","pattern","SdkConfig","get","RegExp","m","p1","p2","p2e","encode","parser","Markdown","isPlainText","phtml","DOMParser","parseFromString","toHTML","parserOrig","phtmlOrig","getElementsByTagName","e","i","item","textContent","querySelectorAll","getAttribute","innerHTML","body","indexOf","toPlaintext","containsEmote","hasCommand","startsWith","hasArgument","length","prefix","caseSensitive","firstPart","toLowerCase","stripEmoteCommand","stripPrefix","clone","removeText","index","offset","unescapeMessage"],"sources":["../../src/editor/serialize.ts"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2019, 2020 The Matrix.org Foundation C.I.C.\nCopyright 2019 New Vector Ltd\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport { encode } from \"html-entities\";\nimport escapeHtml from \"escape-html\";\n\nimport Markdown from \"../Markdown\";\nimport { makeGenericPermalink } from \"../utils/permalinks/Permalinks\";\nimport EditorModel from \"./model\";\nimport SettingsStore from \"../settings/SettingsStore\";\nimport SdkConfig from \"../SdkConfig\";\nimport { Type } from \"./parts\";\n\nexport function mdSerialize(model: EditorModel): string {\n    return model.parts.reduce((html, part) => {\n        switch (part.type) {\n            case Type.Newline:\n                return html + \"\\n\";\n            case Type.Plain:\n            case Type.Emoji:\n            case Type.Command:\n            case Type.PillCandidate:\n            case Type.AtRoomPill:\n                return html + part.text;\n            case Type.RoomPill: {\n                const url = makeGenericPermalink(part.resourceId, true);\n                // Escape square brackets and backslashes\n                // Here we use the resourceId for compatibility with non-rich text clients\n                // See https://github.com/vector-im/element-web/issues/16660\n                const title = part.resourceId.replace(/[[\\\\\\]]/g, (c) => \"\\\\\" + c);\n                return html + `[${title}](${url})`;\n            }\n            case Type.UserPill: {\n                const url = makeGenericPermalink(part.resourceId, true);\n                // Escape square brackets and backslashes; convert newlines to HTML\n                const title = part.text.replace(/[[\\\\\\]]/g, (c) => \"\\\\\" + c).replace(/\\n/g, \"<br>\");\n                return html + `[${title}](${url})`;\n            }\n        }\n    }, \"\");\n}\n\ninterface ISerializeOpts {\n    forceHTML?: boolean;\n    useMarkdown?: boolean;\n}\n\nexport function htmlSerializeIfNeeded(\n    model: EditorModel,\n    { forceHTML = false, useMarkdown = true }: ISerializeOpts = {},\n): string | undefined {\n    if (!useMarkdown) {\n        return escapeHtml(textSerialize(model)).replace(/\\n/g, \"<br/>\");\n    }\n\n    const md = mdSerialize(model);\n    return htmlSerializeFromMdIfNeeded(md, { forceHTML });\n}\n\nexport function htmlSerializeFromMdIfNeeded(md: string, { forceHTML = false } = {}): string | undefined {\n    // copy of raw input to remove unwanted math later\n    const orig = md;\n\n    if (SettingsStore.getValue(\"feature_latex_maths\")) {\n        const patternNames = [\"tex\", \"latex\"] as const;\n        const patternTypes = [\"display\", \"inline\"] as const;\n        const patternDefaults = {\n            tex: {\n                // detect math with tex delimiters, inline: $...$, display $$...$$\n                // preferably use negative lookbehinds, not supported in all major browsers:\n                // const displayPattern = \"^(?<!\\\\\\\\)\\\\$\\\\$(?![ \\\\t])(([^$]|\\\\\\\\\\\\$)+?)\\\\$\\\\$$\";\n                // const inlinePattern = \"(?:^|\\\\s)(?<!\\\\\\\\)\\\\$(?!\\\\s)(([^$]|\\\\\\\\\\\\$)+?)(?<!\\\\\\\\|\\\\s)\\\\$\";\n\n                // conditions for display math detection $$...$$:\n                // - pattern starts and ends on a new line\n                // - left delimiter ($$) is not escaped by backslash\n                display: \"(^)\\\\$\\\\$(([^$]|\\\\\\\\\\\\$)+?)\\\\$\\\\$$\",\n\n                // conditions for inline math detection $...$:\n                // - pattern starts at beginning of line, follows whitespace character or punctuation\n                // - pattern is on a single line\n                // - left and right delimiters ($) are not escaped by backslashes\n                // - left delimiter is not followed by whitespace character\n                // - right delimiter is not prefixed with whitespace character\n                inline: \"(^|\\\\s|[.,!?:;])(?!\\\\\\\\)\\\\$(?!\\\\s)(([^$\\\\n]|\\\\\\\\\\\\$)*([^\\\\\\\\\\\\s\\\\$]|\\\\\\\\\\\\$)(?:\\\\\\\\\\\\$)?)\\\\$\",\n            },\n            latex: {\n                // detect math with latex delimiters, inline: \\(...\\), display \\[...\\]\n\n                // conditions for display math detection \\[...\\]:\n                // - pattern starts and ends on a new line\n                // - pattern is not empty\n                display: \"(^)\\\\\\\\\\\\[(?!\\\\\\\\\\\\])(.*?)\\\\\\\\\\\\]$\",\n\n                // conditions for inline math detection \\(...\\):\n                // - pattern starts at beginning of line or is not prefixed with backslash\n                // - pattern is not empty\n                inline: \"(^|[^\\\\\\\\])\\\\\\\\\\\\((?!\\\\\\\\\\\\))(.*?)\\\\\\\\\\\\)\",\n            },\n        };\n\n        patternNames.forEach(function (patternName) {\n            patternTypes.forEach(function (patternType) {\n                // get the regex replace pattern from config or use the default\n                const pattern =\n                    SdkConfig.get(\"latex_maths_delims\")?.[patternType]?.[\"pattern\"]?.[patternName] ||\n                    patternDefaults[patternName][patternType];\n\n                md = md.replace(RegExp(pattern, \"gms\"), function (m, p1, p2) {\n                    const p2e = encode(p2);\n                    switch (patternType) {\n                        case \"display\":\n                            return `${p1}<div data-mx-maths=\"${p2e}\">\\n\\n</div>\\n\\n`;\n                        case \"inline\":\n                            return `${p1}<span data-mx-maths=\"${p2e}\"></span>`;\n                    }\n                });\n            });\n        });\n\n        // make sure div tags always start on a new line, otherwise it will confuse the markdown parser\n        md = md.replace(/(.)<div/g, function (m, p1) {\n            return `${p1}\\n<div`;\n        });\n    }\n\n    const parser = new Markdown(md);\n    if (!parser.isPlainText() || forceHTML) {\n        // feed Markdown output to HTML parser\n        const phtml = new DOMParser().parseFromString(parser.toHTML(), \"text/html\");\n\n        if (SettingsStore.getValue(\"feature_latex_maths\")) {\n            // original Markdown without LaTeX replacements\n            const parserOrig = new Markdown(orig);\n            const phtmlOrig = new DOMParser().parseFromString(parserOrig.toHTML(), \"text/html\");\n\n            // since maths delimiters are handled before Markdown,\n            // code blocks could contain mangled content.\n            // replace code blocks with original content\n            [...phtmlOrig.getElementsByTagName(\"code\")].forEach((e, i) => {\n                phtml.getElementsByTagName(\"code\").item(i)!.textContent = e.textContent;\n            });\n\n            // add fallback output for latex math, which should not be interpreted as markdown\n            [...phtml.querySelectorAll(\"div, span\")].forEach((e, i) => {\n                const tex = e.getAttribute(\"data-mx-maths\");\n                if (tex) {\n                    e.innerHTML = `<code>${tex}</code>`;\n                }\n            });\n        }\n        return phtml.body.innerHTML;\n    }\n    // ensure removal of escape backslashes in non-Markdown messages\n    if (md.indexOf(\"\\\\\") > -1) {\n        return parser.toPlaintext();\n    }\n}\n\nexport function textSerialize(model: EditorModel): string {\n    return model.parts.reduce((text, part) => {\n        switch (part.type) {\n            case Type.Newline:\n                return text + \"\\n\";\n            case Type.Plain:\n            case Type.Emoji:\n            case Type.Command:\n            case Type.PillCandidate:\n            case Type.AtRoomPill:\n                return text + part.text;\n            case Type.RoomPill:\n                // Here we use the resourceId for compatibility with non-rich text clients\n                // See https://github.com/vector-im/element-web/issues/16660\n                return text + `${part.resourceId}`;\n            case Type.UserPill:\n                return text + `${part.text}`;\n        }\n    }, \"\");\n}\n\nexport function containsEmote(model: EditorModel): boolean {\n    const hasCommand = startsWith(model, \"/me \", false);\n    const hasArgument = model.parts[0]?.text?.length > 4 || model.parts.length > 1;\n    return hasCommand && hasArgument;\n}\n\nexport function startsWith(model: EditorModel, prefix: string, caseSensitive = true): boolean {\n    const firstPart = model.parts[0];\n    // part type will be \"plain\" while editing,\n    // and \"command\" while composing a message.\n    let text = firstPart?.text || \"\";\n    if (!caseSensitive) {\n        prefix = prefix.toLowerCase();\n        text = text.toLowerCase();\n    }\n\n    return firstPart && (firstPart.type === Type.Plain || firstPart.type === Type.Command) && text.startsWith(prefix);\n}\n\nexport function stripEmoteCommand(model: EditorModel): EditorModel {\n    // trim \"/me \"\n    return stripPrefix(model, \"/me \");\n}\n\nexport function stripPrefix(model: EditorModel, prefix: string): EditorModel {\n    model = model.clone();\n    model.removeText({ index: 0, offset: 0 }, prefix.length);\n    return model;\n}\n\nexport function unescapeMessage(model: EditorModel): EditorModel {\n    const { parts } = model;\n    if (parts.length) {\n        const firstPart = parts[0];\n        // only unescape \\/ to / at start of editor\n        if (firstPart.type === Type.Plain && firstPart.text.startsWith(\"\\\\/\")) {\n            model = model.clone();\n            model.removeText({ index: 0, offset: 0 }, 1);\n        }\n    }\n    return model;\n}\n"],"mappings":"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","ignoreList":[]}