matrix-react-sdk
Version:
SDK for matrix.org using React
212 lines (204 loc) • 33.1 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.addReplyToMessageContent = addReplyToMessageContent;
exports.getNestedReplyText = getNestedReplyText;
exports.getParentEventId = getParentEventId;
exports.makeReplyMixIn = makeReplyMixIn;
exports.shouldDisplayReply = shouldDisplayReply;
exports.stripHTMLReply = stripHTMLReply;
exports.stripPlainReply = stripPlainReply;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _matrix = require("matrix-js-sdk/src/matrix");
var _sanitizeHtml = _interopRequireDefault(require("sanitize-html"));
var _escapeHtml = _interopRequireDefault(require("escape-html"));
var _UrlUtils = require("./UrlUtils");
var _Permalinks = require("./permalinks/Permalinks");
var _location = require("./location");
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { (0, _defineProperty2.default)(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } /*
* Copyright 2024 New Vector Ltd.
* Copyright 2023 The Matrix.org Foundation C.I.C.
* Copyright 2021 Šimon Brandner <simon.bra.ag@gmail.com>
*
* SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
* Please see LICENSE files in the repository root for full details.
*/
function getParentEventId(ev) {
if (!ev || ev.isRedacted()) return;
if (ev.replyEventId) {
return ev.replyEventId;
}
}
// Part of Replies fallback support
function stripPlainReply(body) {
// Removes lines beginning with `> ` until you reach one that doesn't.
const lines = body.split("\n");
while (lines.length && lines[0].startsWith("> ")) lines.shift();
// Reply fallback has a blank line after it, so remove it to prevent leading newline
if (lines[0] === "") lines.shift();
return lines.join("\n");
}
// Part of Replies fallback support - MUST NOT BE RENDERED DIRECTLY - UNSAFE HTML
function stripHTMLReply(html) {
// Sanitize the original HTML for inclusion in <mx-reply>. We allow
// any HTML, since the original sender could use special tags that we
// don't recognize, but want to pass along to any recipients who do
// recognize them -- recipients should be sanitizing before displaying
// anyways. However, we sanitize to 1) remove any mx-reply, so that we
// don't generate a nested mx-reply, and 2) make sure that the HTML is
// properly formatted (e.g. tags are closed where necessary)
return (0, _sanitizeHtml.default)(html, {
allowedTags: false,
// false means allow everything
allowedAttributes: false,
allowVulnerableTags: true,
// silence xss warning, we won't be rendering directly this, so it is safe to do
// we somehow can't allow all schemes, so we allow all that we
// know of and mxc (for img tags)
allowedSchemes: [..._UrlUtils.PERMITTED_URL_SCHEMES, "mxc"],
exclusiveFilter: frame => frame.tag === "mx-reply"
});
}
// Part of Replies fallback support
function getNestedReplyText(ev, permalinkCreator) {
if (!ev) return null;
let {
body,
formatted_body: html,
msgtype
} = ev.getContent();
if (getParentEventId(ev)) {
if (body) body = stripPlainReply(body);
}
if (!body) body = ""; // Always ensure we have a body, for reasons.
if (html) {
// sanitize the HTML before we put it in an <mx-reply>
html = stripHTMLReply(html);
} else {
// Escape the body to use as HTML below.
// We also run a nl2br over the result to fix the fallback representation. We do this
// after converting the text to safe HTML to avoid user-provided BR's from being converted.
html = (0, _escapeHtml.default)(body).replace(/\n/g, "<br/>");
}
// dev note: do not rely on `body` being safe for HTML usage below.
const evLink = permalinkCreator?.forEvent(ev.getId());
const userLink = (0, _Permalinks.makeUserPermalink)(ev.getSender());
const mxid = ev.getSender();
if (_matrix.M_BEACON_INFO.matches(ev.getType())) {
const aTheir = (0, _location.isSelfLocation)(ev.getContent()) ? "their" : "a";
return {
html: `<mx-reply><blockquote><a href="${evLink}">In reply to</a> <a href="${userLink}">${mxid}</a>` + `<br>shared ${aTheir} live location.</blockquote></mx-reply>`,
body: `> <${mxid}> shared ${aTheir} live location.\n\n`
};
}
if (_matrix.M_POLL_START.matches(ev.getType())) {
const extensibleEvent = ev.unstableExtensibleEvent;
const question = extensibleEvent?.question?.text;
return {
html: `<mx-reply><blockquote><a href="${evLink}">In reply to</a> <a href="${userLink}">${mxid}</a>` + `<br>Poll: ${question}</blockquote></mx-reply>`,
body: `> <${mxid}> started poll: ${question}\n\n`
};
}
if (_matrix.M_POLL_END.matches(ev.getType())) {
return {
html: `<mx-reply><blockquote><a href="${evLink}">In reply to</a> <a href="${userLink}">${mxid}</a>` + `<br>Ended poll</blockquote></mx-reply>`,
body: `> <${mxid}>Ended poll\n\n`
};
}
// This fallback contains text that is explicitly EN.
switch (msgtype) {
case _matrix.MsgType.Text:
case _matrix.MsgType.Notice:
{
html = `<mx-reply><blockquote><a href="${evLink}">In reply to</a> <a href="${userLink}">${mxid}</a>` + `<br>${html}</blockquote></mx-reply>`;
const lines = body.trim().split("\n");
if (lines.length > 0) {
lines[0] = `<${mxid}> ${lines[0]}`;
body = lines.map(line => `> ${line}`).join("\n") + "\n\n";
}
break;
}
case _matrix.MsgType.Image:
html = `<mx-reply><blockquote><a href="${evLink}">In reply to</a> <a href="${userLink}">${mxid}</a>` + `<br>sent an image.</blockquote></mx-reply>`;
body = `> <${mxid}> sent an image.\n\n`;
break;
case _matrix.MsgType.Video:
html = `<mx-reply><blockquote><a href="${evLink}">In reply to</a> <a href="${userLink}">${mxid}</a>` + `<br>sent a video.</blockquote></mx-reply>`;
body = `> <${mxid}> sent a video.\n\n`;
break;
case _matrix.MsgType.Audio:
html = `<mx-reply><blockquote><a href="${evLink}">In reply to</a> <a href="${userLink}">${mxid}</a>` + `<br>sent an audio file.</blockquote></mx-reply>`;
body = `> <${mxid}> sent an audio file.\n\n`;
break;
case _matrix.MsgType.File:
html = `<mx-reply><blockquote><a href="${evLink}">In reply to</a> <a href="${userLink}">${mxid}</a>` + `<br>sent a file.</blockquote></mx-reply>`;
body = `> <${mxid}> sent a file.\n\n`;
break;
case _matrix.MsgType.Location:
{
const aTheir = (0, _location.isSelfLocation)(ev.getContent()) ? "their" : "a";
html = `<mx-reply><blockquote><a href="${evLink}">In reply to</a> <a href="${userLink}">${mxid}</a>` + `<br>shared ${aTheir} location.</blockquote></mx-reply>`;
body = `> <${mxid}> shared ${aTheir} location.\n\n`;
break;
}
case _matrix.MsgType.Emote:
{
html = `<mx-reply><blockquote><a href="${evLink}">In reply to</a> * ` + `<a href="${userLink}">${mxid}</a><br>${html}</blockquote></mx-reply>`;
const lines = body.trim().split("\n");
if (lines.length > 0) {
lines[0] = `* <${mxid}> ${lines[0]}`;
body = lines.map(line => `> ${line}`).join("\n") + "\n\n";
}
break;
}
default:
return null;
}
return {
body,
html
};
}
function makeReplyMixIn(ev) {
if (!ev) return {};
const mixin = {
"m.in_reply_to": {
event_id: ev.getId()
}
};
if (ev.threadRootId) {
mixin.is_falling_back = false;
}
return mixin;
}
function shouldDisplayReply(event) {
if (event.isRedacted()) {
return false;
}
const inReplyTo = event.getWireContent()?.["m.relates_to"]?.["m.in_reply_to"];
if (!inReplyTo) {
return false;
}
const relation = event.getRelation();
if (relation?.rel_type === _matrix.THREAD_RELATION_TYPE.name && relation?.is_falling_back) {
return false;
}
return !!inReplyTo.event_id;
}
function addReplyToMessageContent(content, replyToEvent, opts) {
content["m.relates_to"] = _objectSpread(_objectSpread({}, content["m.relates_to"] || {}), makeReplyMixIn(replyToEvent));
if (opts.includeLegacyFallback) {
// Part of Replies fallback support - prepend the text we're sending with the text we're replying to
const nestedReply = getNestedReplyText(replyToEvent, opts.permalinkCreator);
if (nestedReply) {
if (content.formatted_body) {
content.formatted_body = nestedReply.html + content.formatted_body;
}
content.body = nestedReply.body + content.body;
}
}
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_matrix","require","_sanitizeHtml","_interopRequireDefault","_escapeHtml","_UrlUtils","_Permalinks","_location","ownKeys","e","r","t","Object","keys","getOwnPropertySymbols","o","filter","getOwnPropertyDescriptor","enumerable","push","apply","_objectSpread","arguments","length","forEach","_defineProperty2","default","getOwnPropertyDescriptors","defineProperties","defineProperty","getParentEventId","ev","isRedacted","replyEventId","stripPlainReply","body","lines","split","startsWith","shift","join","stripHTMLReply","html","sanitizeHtml","allowedTags","allowedAttributes","allowVulnerableTags","allowedSchemes","PERMITTED_URL_SCHEMES","exclusiveFilter","frame","tag","getNestedReplyText","permalinkCreator","formatted_body","msgtype","getContent","escapeHtml","replace","evLink","forEvent","getId","userLink","makeUserPermalink","getSender","mxid","M_BEACON_INFO","matches","getType","aTheir","isSelfLocation","M_POLL_START","extensibleEvent","unstableExtensibleEvent","question","text","M_POLL_END","MsgType","Text","Notice","trim","map","line","Image","Video","Audio","File","Location","Emote","makeReplyMixIn","mixin","event_id","threadRootId","is_falling_back","shouldDisplayReply","event","inReplyTo","getWireContent","relation","getRelation","rel_type","THREAD_RELATION_TYPE","name","addReplyToMessageContent","content","replyToEvent","opts","includeLegacyFallback","nestedReply"],"sources":["../../src/utils/Reply.ts"],"sourcesContent":["/*\n * Copyright 2024 New Vector Ltd.\n * Copyright 2023 The Matrix.org Foundation C.I.C.\n * Copyright 2021 Šimon Brandner <simon.bra.ag@gmail.com>\n *\n * SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\n * Please see LICENSE files in the repository root for full details.\n */\n\nimport {\n    IContent,\n    IEventRelation,\n    MatrixEvent,\n    MsgType,\n    THREAD_RELATION_TYPE,\n    M_BEACON_INFO,\n    M_POLL_END,\n    M_POLL_START,\n} from \"matrix-js-sdk/src/matrix\";\nimport sanitizeHtml from \"sanitize-html\";\nimport escapeHtml from \"escape-html\";\nimport { PollStartEvent } from \"matrix-js-sdk/src/extensible_events_v1/PollStartEvent\";\n\nimport { PERMITTED_URL_SCHEMES } from \"./UrlUtils\";\nimport { makeUserPermalink, RoomPermalinkCreator } from \"./permalinks/Permalinks\";\nimport { isSelfLocation } from \"./location\";\n\nexport function getParentEventId(ev?: MatrixEvent): string | undefined {\n    if (!ev || ev.isRedacted()) return;\n    if (ev.replyEventId) {\n        return ev.replyEventId;\n    }\n}\n\n// Part of Replies fallback support\nexport function stripPlainReply(body: string): string {\n    // Removes lines beginning with `> ` until you reach one that doesn't.\n    const lines = body.split(\"\\n\");\n    while (lines.length && lines[0].startsWith(\"> \")) lines.shift();\n    // Reply fallback has a blank line after it, so remove it to prevent leading newline\n    if (lines[0] === \"\") lines.shift();\n    return lines.join(\"\\n\");\n}\n\n// Part of Replies fallback support - MUST NOT BE RENDERED DIRECTLY - UNSAFE HTML\nexport function stripHTMLReply(html: string): string {\n    // Sanitize the original HTML for inclusion in <mx-reply>.  We allow\n    // any HTML, since the original sender could use special tags that we\n    // don't recognize, but want to pass along to any recipients who do\n    // recognize them -- recipients should be sanitizing before displaying\n    // anyways.  However, we sanitize to 1) remove any mx-reply, so that we\n    // don't generate a nested mx-reply, and 2) make sure that the HTML is\n    // properly formatted (e.g. tags are closed where necessary)\n    return sanitizeHtml(html, {\n        allowedTags: false, // false means allow everything\n        allowedAttributes: false,\n        allowVulnerableTags: true, // silence xss warning, we won't be rendering directly this, so it is safe to do\n        // we somehow can't allow all schemes, so we allow all that we\n        // know of and mxc (for img tags)\n        allowedSchemes: [...PERMITTED_URL_SCHEMES, \"mxc\"],\n        exclusiveFilter: (frame) => frame.tag === \"mx-reply\",\n    });\n}\n\n// Part of Replies fallback support\nexport function getNestedReplyText(\n    ev: MatrixEvent,\n    permalinkCreator?: RoomPermalinkCreator,\n): { body: string; html: string } | null {\n    if (!ev) return null;\n\n    let {\n        body,\n        formatted_body: html,\n        msgtype,\n    } = ev.getContent<{\n        body: string;\n        msgtype?: string;\n        formatted_body?: string;\n    }>();\n    if (getParentEventId(ev)) {\n        if (body) body = stripPlainReply(body);\n    }\n\n    if (!body) body = \"\"; // Always ensure we have a body, for reasons.\n\n    if (html) {\n        // sanitize the HTML before we put it in an <mx-reply>\n        html = stripHTMLReply(html);\n    } else {\n        // Escape the body to use as HTML below.\n        // We also run a nl2br over the result to fix the fallback representation. We do this\n        // after converting the text to safe HTML to avoid user-provided BR's from being converted.\n        html = escapeHtml(body).replace(/\\n/g, \"<br/>\");\n    }\n\n    // dev note: do not rely on `body` being safe for HTML usage below.\n\n    const evLink = permalinkCreator?.forEvent(ev.getId()!);\n    const userLink = makeUserPermalink(ev.getSender()!);\n    const mxid = ev.getSender();\n\n    if (M_BEACON_INFO.matches(ev.getType())) {\n        const aTheir = isSelfLocation(ev.getContent()) ? \"their\" : \"a\";\n        return {\n            html:\n                `<mx-reply><blockquote><a href=\"${evLink}\">In reply to</a> <a href=\"${userLink}\">${mxid}</a>` +\n                `<br>shared ${aTheir} live location.</blockquote></mx-reply>`,\n            body: `> <${mxid}> shared ${aTheir} live location.\\n\\n`,\n        };\n    }\n\n    if (M_POLL_START.matches(ev.getType())) {\n        const extensibleEvent = ev.unstableExtensibleEvent as PollStartEvent;\n        const question = extensibleEvent?.question?.text;\n        return {\n            html:\n                `<mx-reply><blockquote><a href=\"${evLink}\">In reply to</a> <a href=\"${userLink}\">${mxid}</a>` +\n                `<br>Poll: ${question}</blockquote></mx-reply>`,\n            body: `> <${mxid}> started poll: ${question}\\n\\n`,\n        };\n    }\n    if (M_POLL_END.matches(ev.getType())) {\n        return {\n            html:\n                `<mx-reply><blockquote><a href=\"${evLink}\">In reply to</a> <a href=\"${userLink}\">${mxid}</a>` +\n                `<br>Ended poll</blockquote></mx-reply>`,\n            body: `> <${mxid}>Ended poll\\n\\n`,\n        };\n    }\n\n    // This fallback contains text that is explicitly EN.\n    switch (msgtype) {\n        case MsgType.Text:\n        case MsgType.Notice: {\n            html =\n                `<mx-reply><blockquote><a href=\"${evLink}\">In reply to</a> <a href=\"${userLink}\">${mxid}</a>` +\n                `<br>${html}</blockquote></mx-reply>`;\n            const lines = body.trim().split(\"\\n\");\n            if (lines.length > 0) {\n                lines[0] = `<${mxid}> ${lines[0]}`;\n                body = lines.map((line) => `> ${line}`).join(\"\\n\") + \"\\n\\n\";\n            }\n            break;\n        }\n        case MsgType.Image:\n            html =\n                `<mx-reply><blockquote><a href=\"${evLink}\">In reply to</a> <a href=\"${userLink}\">${mxid}</a>` +\n                `<br>sent an image.</blockquote></mx-reply>`;\n            body = `> <${mxid}> sent an image.\\n\\n`;\n            break;\n        case MsgType.Video:\n            html =\n                `<mx-reply><blockquote><a href=\"${evLink}\">In reply to</a> <a href=\"${userLink}\">${mxid}</a>` +\n                `<br>sent a video.</blockquote></mx-reply>`;\n            body = `> <${mxid}> sent a video.\\n\\n`;\n            break;\n        case MsgType.Audio:\n            html =\n                `<mx-reply><blockquote><a href=\"${evLink}\">In reply to</a> <a href=\"${userLink}\">${mxid}</a>` +\n                `<br>sent an audio file.</blockquote></mx-reply>`;\n            body = `> <${mxid}> sent an audio file.\\n\\n`;\n            break;\n        case MsgType.File:\n            html =\n                `<mx-reply><blockquote><a href=\"${evLink}\">In reply to</a> <a href=\"${userLink}\">${mxid}</a>` +\n                `<br>sent a file.</blockquote></mx-reply>`;\n            body = `> <${mxid}> sent a file.\\n\\n`;\n            break;\n        case MsgType.Location: {\n            const aTheir = isSelfLocation(ev.getContent()) ? \"their\" : \"a\";\n            html =\n                `<mx-reply><blockquote><a href=\"${evLink}\">In reply to</a> <a href=\"${userLink}\">${mxid}</a>` +\n                `<br>shared ${aTheir} location.</blockquote></mx-reply>`;\n            body = `> <${mxid}> shared ${aTheir} location.\\n\\n`;\n            break;\n        }\n        case MsgType.Emote: {\n            html =\n                `<mx-reply><blockquote><a href=\"${evLink}\">In reply to</a> * ` +\n                `<a href=\"${userLink}\">${mxid}</a><br>${html}</blockquote></mx-reply>`;\n            const lines = body.trim().split(\"\\n\");\n            if (lines.length > 0) {\n                lines[0] = `* <${mxid}> ${lines[0]}`;\n                body = lines.map((line) => `> ${line}`).join(\"\\n\") + \"\\n\\n\";\n            }\n            break;\n        }\n        default:\n            return null;\n    }\n\n    return { body, html };\n}\n\nexport function makeReplyMixIn(ev?: MatrixEvent): IEventRelation {\n    if (!ev) return {};\n\n    const mixin: IEventRelation = {\n        \"m.in_reply_to\": {\n            event_id: ev.getId(),\n        },\n    };\n\n    if (ev.threadRootId) {\n        mixin.is_falling_back = false;\n    }\n\n    return mixin;\n}\n\nexport function shouldDisplayReply(event: MatrixEvent): boolean {\n    if (event.isRedacted()) {\n        return false;\n    }\n\n    const inReplyTo = event.getWireContent()?.[\"m.relates_to\"]?.[\"m.in_reply_to\"];\n    if (!inReplyTo) {\n        return false;\n    }\n\n    const relation = event.getRelation();\n    if (relation?.rel_type === THREAD_RELATION_TYPE.name && relation?.is_falling_back) {\n        return false;\n    }\n\n    return !!inReplyTo.event_id;\n}\n\ninterface AddReplyOpts {\n    permalinkCreator?: RoomPermalinkCreator;\n    includeLegacyFallback: false;\n}\n\ninterface IncludeLegacyFeedbackOpts {\n    permalinkCreator?: RoomPermalinkCreator;\n    includeLegacyFallback: true;\n}\n\nexport function addReplyToMessageContent(\n    content: IContent,\n    replyToEvent: MatrixEvent,\n    opts: AddReplyOpts | IncludeLegacyFeedbackOpts,\n): void {\n    content[\"m.relates_to\"] = {\n        ...(content[\"m.relates_to\"] || {}),\n        ...makeReplyMixIn(replyToEvent),\n    };\n\n    if (opts.includeLegacyFallback) {\n        // Part of Replies fallback support - prepend the text we're sending with the text we're replying to\n        const nestedReply = getNestedReplyText(replyToEvent, opts.permalinkCreator);\n        if (nestedReply) {\n            if (content.formatted_body) {\n                content.formatted_body = nestedReply.html + content.formatted_body;\n            }\n            content.body = nestedReply.body + content.body;\n        }\n    }\n}\n"],"mappings":"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","ignoreList":[]}