UNPKG

matrix-react-sdk

Version:
231 lines (223 loc) 31.6 kB
"use strict"; 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":[]}