2019-05-07 18:31:37 +03:00
|
|
|
/*
|
|
|
|
Copyright 2019 New Vector Ltd
|
2019-05-22 17:16:32 +03:00
|
|
|
Copyright 2019 The Matrix.org Foundation C.I.C.
|
2019-05-07 18:31:37 +03:00
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import { MATRIXTO_URL_PATTERN } from '../linkify-matrix';
|
2019-05-13 20:20:21 +03:00
|
|
|
import { PlainPart, UserPillPart, RoomPillPart, NewlinePart } from "./parts";
|
2019-05-07 18:31:37 +03:00
|
|
|
|
|
|
|
function parseHtmlMessage(html) {
|
|
|
|
const REGEX_MATRIXTO = new RegExp(MATRIXTO_URL_PATTERN);
|
2019-05-08 12:13:36 +03:00
|
|
|
// no nodes from parsing here should be inserted in the document,
|
|
|
|
// as scripts in event handlers, etc would be executed then.
|
|
|
|
// we're only taking text, so that is fine
|
2019-05-07 18:31:37 +03:00
|
|
|
const nodes = Array.from(new DOMParser().parseFromString(html, "text/html").body.childNodes);
|
|
|
|
const parts = nodes.map(n => {
|
|
|
|
switch (n.nodeType) {
|
|
|
|
case Node.TEXT_NODE:
|
|
|
|
return new PlainPart(n.nodeValue);
|
|
|
|
case Node.ELEMENT_NODE:
|
|
|
|
switch (n.nodeName) {
|
|
|
|
case "MX-REPLY":
|
|
|
|
return null;
|
|
|
|
case "A": {
|
|
|
|
const {href} = n;
|
|
|
|
const pillMatch = REGEX_MATRIXTO.exec(href) || [];
|
|
|
|
const resourceId = pillMatch[1]; // The room/user ID
|
|
|
|
const prefix = pillMatch[2]; // The first character of prefix
|
|
|
|
switch (prefix) {
|
2019-05-09 16:01:17 +03:00
|
|
|
case "@": return new UserPillPart(resourceId, n.textContent);
|
|
|
|
case "#": return new RoomPillPart(resourceId, n.textContent);
|
|
|
|
default: return new PlainPart(n.textContent);
|
2019-05-07 18:31:37 +03:00
|
|
|
}
|
|
|
|
}
|
2019-05-13 20:20:21 +03:00
|
|
|
case "BR":
|
|
|
|
return new NewlinePart("\n");
|
2019-05-07 18:31:37 +03:00
|
|
|
default:
|
2019-05-09 16:01:17 +03:00
|
|
|
return new PlainPart(n.textContent);
|
2019-05-07 18:31:37 +03:00
|
|
|
}
|
2019-05-09 16:01:17 +03:00
|
|
|
default:
|
|
|
|
return null;
|
2019-05-07 18:31:37 +03:00
|
|
|
}
|
|
|
|
}).filter(p => !!p);
|
|
|
|
return parts;
|
|
|
|
}
|
|
|
|
|
2019-05-14 12:53:42 +03:00
|
|
|
export function parseEvent(event) {
|
2019-05-07 18:31:37 +03:00
|
|
|
const content = event.getContent();
|
|
|
|
if (content.format === "org.matrix.custom.html") {
|
2019-05-15 13:49:43 +03:00
|
|
|
return parseHtmlMessage(content.formatted_body || "");
|
2019-05-07 18:31:37 +03:00
|
|
|
} else {
|
2019-05-15 13:49:43 +03:00
|
|
|
const body = content.body || "";
|
|
|
|
const lines = body.split("\n");
|
2019-05-14 12:37:40 +03:00
|
|
|
const parts = lines.reduce((parts, line, i) => {
|
|
|
|
const isLast = i === lines.length - 1;
|
|
|
|
const text = new PlainPart(line);
|
|
|
|
const newLine = !isLast && new NewlinePart("\n");
|
|
|
|
if (newLine) {
|
|
|
|
return parts.concat(text, newLine);
|
|
|
|
} else {
|
|
|
|
return parts.concat(text);
|
|
|
|
}
|
|
|
|
}, []);
|
|
|
|
return parts;
|
2019-05-07 18:31:37 +03:00
|
|
|
}
|
|
|
|
}
|