2019-05-22 14:16:32 +00:00
|
|
|
/*
|
|
|
|
Copyright 2019 New Vector Ltd
|
|
|
|
Copyright 2019 The Matrix.org Foundation C.I.C.
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2019-05-21 15:59:54 +00:00
|
|
|
import Markdown from '../Markdown';
|
2019-10-01 02:39:58 +00:00
|
|
|
import {makeGenericPermalink} from "../utils/permalinks/Permalinks";
|
2019-05-21 15:59:54 +00:00
|
|
|
|
|
|
|
export function mdSerialize(model) {
|
2019-05-14 09:37:16 +00:00
|
|
|
return model.parts.reduce((html, part) => {
|
|
|
|
switch (part.type) {
|
|
|
|
case "newline":
|
2019-05-21 15:59:54 +00:00
|
|
|
return html + "\n";
|
2019-05-14 09:37:16 +00:00
|
|
|
case "plain":
|
2019-08-21 13:27:50 +00:00
|
|
|
case "command":
|
2019-05-14 09:37:16 +00:00
|
|
|
case "pill-candidate":
|
2019-06-14 16:24:36 +00:00
|
|
|
case "at-room-pill":
|
2019-05-14 09:37:16 +00:00
|
|
|
return html + part.text;
|
|
|
|
case "room-pill":
|
|
|
|
case "user-pill":
|
2020-04-14 23:40:38 +00:00
|
|
|
return html + `[${part.text.replace(/[\\\]]/, c => "\\" + c)}](${makeGenericPermalink(part.resourceId)})`;
|
2019-05-14 09:37:16 +00:00
|
|
|
}
|
|
|
|
}, "");
|
|
|
|
}
|
|
|
|
|
2019-08-20 15:15:12 +00:00
|
|
|
export function htmlSerializeIfNeeded(model, {forceHTML = false} = {}) {
|
2019-05-21 15:59:54 +00:00
|
|
|
const md = mdSerialize(model);
|
|
|
|
const parser = new Markdown(md);
|
2019-07-08 14:55:56 +00:00
|
|
|
if (!parser.isPlainText() || forceHTML) {
|
2019-05-21 15:59:54 +00:00
|
|
|
return parser.toHTML();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-14 09:37:16 +00:00
|
|
|
export function textSerialize(model) {
|
|
|
|
return model.parts.reduce((text, part) => {
|
|
|
|
switch (part.type) {
|
|
|
|
case "newline":
|
|
|
|
return text + "\n";
|
|
|
|
case "plain":
|
2019-08-21 13:27:50 +00:00
|
|
|
case "command":
|
2019-05-14 09:37:16 +00:00
|
|
|
case "pill-candidate":
|
2019-06-14 16:24:36 +00:00
|
|
|
case "at-room-pill":
|
2019-05-14 09:37:16 +00:00
|
|
|
return text + part.text;
|
|
|
|
case "room-pill":
|
|
|
|
case "user-pill":
|
2019-08-30 09:51:29 +00:00
|
|
|
return text + `${part.text}`;
|
2019-05-14 09:37:16 +00:00
|
|
|
}
|
|
|
|
}, "");
|
|
|
|
}
|
2019-08-21 09:26:21 +00:00
|
|
|
|
|
|
|
export function containsEmote(model) {
|
2020-01-21 15:55:21 +00:00
|
|
|
return startsWith(model, "/me ");
|
|
|
|
}
|
|
|
|
|
|
|
|
export function startsWith(model, prefix) {
|
2019-08-21 09:26:21 +00:00
|
|
|
const firstPart = model.parts[0];
|
2019-08-21 13:27:50 +00:00
|
|
|
// part type will be "plain" while editing,
|
|
|
|
// and "command" while composing a message.
|
|
|
|
return firstPart &&
|
|
|
|
(firstPart.type === "plain" || firstPart.type === "command") &&
|
2020-01-21 15:55:21 +00:00
|
|
|
firstPart.text.startsWith(prefix);
|
2019-08-21 09:26:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function stripEmoteCommand(model) {
|
|
|
|
// trim "/me "
|
2020-01-21 15:55:21 +00:00
|
|
|
return stripPrefix(model, "/me ");
|
|
|
|
}
|
|
|
|
|
|
|
|
export function stripPrefix(model, prefix) {
|
2019-08-21 09:26:21 +00:00
|
|
|
model = model.clone();
|
2020-01-21 15:55:21 +00:00
|
|
|
model.removeText({index: 0, offset: 0}, prefix.length);
|
2019-08-21 09:26:21 +00:00
|
|
|
return model;
|
|
|
|
}
|
2019-09-02 15:53:14 +00:00
|
|
|
|
|
|
|
export function unescapeMessage(model) {
|
|
|
|
const {parts} = model;
|
|
|
|
if (parts.length) {
|
|
|
|
const firstPart = parts[0];
|
|
|
|
// only unescape \/ to / at start of editor
|
|
|
|
if (firstPart.type === "plain" && firstPart.text.startsWith("\\/")) {
|
|
|
|
model = model.clone();
|
|
|
|
model.removeText({index: 0, offset: 0}, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return model;
|
|
|
|
}
|