thelounge/client/js/libs/handlebars/parse.js

146 lines
4.1 KiB
JavaScript
Raw Normal View History

"use strict";
const parseStyle = require("./ircmessageparser/parseStyle");
const findChannels = require("./ircmessageparser/findChannels");
const findLinks = require("./ircmessageparser/findLinks");
2017-08-23 16:19:04 +02:00
const findEmoji = require("./ircmessageparser/findEmoji");
2017-11-14 23:36:45 +01:00
const findNames = require("./ircmessageparser/findNames");
const merge = require("./ircmessageparser/merge");
2017-11-14 23:36:45 +01:00
const colorClass = require("./colorClass");
2018-03-09 23:00:16 +01:00
const emojiMap = require("../fullnamemap.json");
2018-07-11 20:00:12 +02:00
const LinkPreviewToggle = require("../../../components/LinkPreviewToggle.vue").default;
// Create an HTML `span` with styling information for a given fragment
2018-07-12 10:41:40 +02:00
function createFragment(fragment, createElement) {
const classes = [];
if (fragment.bold) {
classes.push("irc-bold");
}
if (fragment.textColor !== undefined) {
classes.push("irc-fg" + fragment.textColor);
}
if (fragment.bgColor !== undefined) {
classes.push("irc-bg" + fragment.bgColor);
}
if (fragment.italic) {
classes.push("irc-italic");
}
if (fragment.underline) {
classes.push("irc-underline");
}
if (fragment.strikethrough) {
classes.push("irc-strikethrough");
}
if (fragment.monospace) {
classes.push("irc-monospace");
}
if (classes.length === 0 && !fragment.hexColor) {
return fragment.text;
}
return createElement("span", {
class: classes,
style: {
2018-07-12 10:26:12 +02:00
"color": `#${fragment.hexColor}`,
"background-color": fragment.hexBgColor ? `#${fragment.hexBgColor}` : null,
},
}, fragment.text);
}
2017-11-14 23:36:45 +01:00
// Transform an IRC message potentially filled with styling control codes, URLs,
// nicknames, and channels into a string of HTML elements to display on the client.
2018-07-19 19:44:24 +02:00
module.exports = function parse(createElement, text, message = undefined, network = undefined) {
// Extract the styling information and get the plain text version from it
const styleFragments = parseStyle(text);
const cleanText = styleFragments.map((fragment) => fragment.text).join("");
// On the plain text, find channels and URLs, returned as "parts". Parts are
// arrays of objects containing start and end markers, as well as metadata
// depending on what was found (channel or link).
2018-07-19 19:44:24 +02:00
const channelPrefixes = network ? network.serverOptions.CHANTYPES : ["#", "&"];
const userModes = network ? network.serverOptions.PREFIX : ["!", "@", "%", "+"];
const channelParts = findChannels(cleanText, channelPrefixes, userModes);
const linkParts = findLinks(cleanText);
2017-08-23 16:19:04 +02:00
const emojiParts = findEmoji(cleanText);
2018-07-11 20:00:12 +02:00
const nameParts = findNames(cleanText, message ? (message.users || []) : []);
const parts = channelParts
.concat(linkParts)
2017-08-23 16:19:04 +02:00
.concat(emojiParts)
2018-04-19 18:00:46 +02:00
.concat(nameParts);
2017-11-14 23:36:45 +01:00
// Merge the styling information with the channels / URLs / nicks / text objects and
// generate HTML strings with the resulting fragments
2018-04-19 18:01:20 +02:00
return merge(parts, styleFragments, cleanText).map((textPart) => {
2018-07-12 10:41:40 +02:00
const fragments = textPart.fragments.map((fragment) => createFragment(fragment, createElement));
// Wrap these potentially styled fragments with links and channel buttons
if (textPart.link) {
2018-07-12 10:41:40 +02:00
const preview = message && message.previews.find((p) => p.link === textPart.link);
const link = createElement("a", {
attrs: {
href: textPart.link,
target: "_blank",
rel: "noopener",
},
}, fragments);
if (!preview) {
return link;
2018-03-09 23:00:16 +01:00
}
2018-07-12 10:41:40 +02:00
return [link, createElement(LinkPreviewToggle, {
class: ["toggle-button", "toggle-preview"],
props: {
link: preview,
},
}, fragments)];
} else if (textPart.channel) {
return createElement("span", {
class: [
"inline-channel",
],
attrs: {
"role": "button",
"tabindex": 0,
"data-chan": textPart.channel,
},
}, fragments);
} else if (textPart.emoji) {
2018-08-31 12:33:16 +02:00
const title = emojiMap[textPart.emoji] ? `Emoji: ${emojiMap[textPart.emoji]}` : null;
2018-07-12 10:41:40 +02:00
return createElement("span", {
class: [
"emoji",
],
attrs: {
"role": "img",
2018-08-31 12:33:16 +02:00
"aria-label": title,
"title": title,
2018-07-12 10:41:40 +02:00
},
}, fragments);
2017-11-14 23:36:45 +01:00
} else if (textPart.nick) {
2018-07-12 10:41:40 +02:00
return createElement("span", {
class: [
"user",
colorClass(textPart.nick),
],
attrs: {
"role": "button",
"data-name": textPart.nick,
},
}, fragments);
}
return fragments;
2018-07-12 10:41:40 +02:00
});
};