2021-03-05 00:35:15 +01:00
|
|
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
|
2018-01-04 23:27:32 +01:00
|
|
|
#pragma once
|
|
|
|
|
2019-12-14 17:08:36 +01:00
|
|
|
#include <variant>
|
2018-09-01 12:35:10 +02:00
|
|
|
|
2019-07-29 00:40:23 +02:00
|
|
|
#include <QCoreApplication>
|
2018-01-04 23:27:32 +01:00
|
|
|
#include <QDateTime>
|
2018-05-11 15:00:14 +02:00
|
|
|
#include <QPixmap>
|
2018-01-04 23:27:32 +01:00
|
|
|
#include <mtx/events/collections.hpp>
|
2018-09-06 21:34:41 +02:00
|
|
|
#include <mtx/events/common.hpp>
|
2018-01-04 23:27:32 +01:00
|
|
|
|
2019-01-18 18:17:25 +01:00
|
|
|
#include <qmath.h>
|
|
|
|
|
2020-01-31 06:12:02 +01:00
|
|
|
struct DescInfo;
|
|
|
|
|
2019-12-15 03:34:17 +01:00
|
|
|
namespace cache {
|
|
|
|
// Forward declarations to prevent dependency on Cache.h, since this header is included often!
|
|
|
|
QString
|
|
|
|
displayName(const QString &room_id, const QString &user_id);
|
|
|
|
}
|
|
|
|
|
2018-10-01 16:56:46 +02:00
|
|
|
class QComboBox;
|
|
|
|
|
2019-06-14 04:33:04 +02:00
|
|
|
// Contains information about related events for
|
|
|
|
// outgoing messages
|
|
|
|
struct RelatedInfo
|
|
|
|
{
|
2019-07-05 03:20:19 +02:00
|
|
|
using MsgType = mtx::events::MessageType;
|
|
|
|
MsgType type;
|
|
|
|
QString room;
|
2020-01-21 03:36:26 +01:00
|
|
|
QString quoted_body, quoted_formatted_body;
|
2019-06-14 04:33:04 +02:00
|
|
|
std::string related_event;
|
|
|
|
QString quoted_user;
|
|
|
|
};
|
|
|
|
|
2018-01-04 23:27:32 +01:00
|
|
|
namespace utils {
|
|
|
|
|
|
|
|
using TimelineEvent = mtx::events::collections::TimelineEvents;
|
|
|
|
|
2020-05-19 21:04:38 +02:00
|
|
|
bool
|
|
|
|
codepointIsEmoji(uint code);
|
|
|
|
|
2019-07-26 23:44:44 +02:00
|
|
|
QString
|
|
|
|
replaceEmoji(const QString &body);
|
|
|
|
|
2018-07-20 15:15:50 +02:00
|
|
|
QString
|
|
|
|
localUser();
|
|
|
|
|
2018-07-22 18:48:58 +02:00
|
|
|
float
|
|
|
|
scaleFactor();
|
|
|
|
|
|
|
|
void
|
|
|
|
setScaleFactor(float factor);
|
|
|
|
|
2018-07-22 15:36:25 +02:00
|
|
|
//! Whether or not we should respond to key requests for the given room.
|
|
|
|
bool
|
|
|
|
respondsToKeyRequests(const QString &roomId);
|
|
|
|
bool
|
|
|
|
respondsToKeyRequests(const std::string &roomId);
|
|
|
|
|
|
|
|
void
|
|
|
|
setKeyRequestsPreference(QString roomId, bool value);
|
|
|
|
|
2018-01-04 23:27:32 +01:00
|
|
|
//! Human friendly timestamp representation.
|
|
|
|
QString
|
|
|
|
descriptiveTime(const QDateTime &then);
|
|
|
|
|
|
|
|
//! Generate a message description from the event to be displayed
|
|
|
|
//! in the RoomList.
|
|
|
|
DescInfo
|
2020-11-25 23:43:31 +01:00
|
|
|
getMessageDescription(const TimelineEvent &event,
|
|
|
|
const QString &localUser,
|
|
|
|
const QString &displayName);
|
2018-01-12 09:21:53 +01:00
|
|
|
|
|
|
|
//! Get the first character of a string, taking into account that
|
|
|
|
//! surrogate pairs might be in use.
|
|
|
|
QString
|
|
|
|
firstChar(const QString &input);
|
2018-02-18 21:52:31 +01:00
|
|
|
|
|
|
|
//! Get a human readable file size with the appropriate units attached.
|
|
|
|
QString
|
2018-02-19 21:09:21 +01:00
|
|
|
humanReadableFileSize(uint64_t bytes);
|
2018-02-19 22:32:37 +01:00
|
|
|
|
2018-05-05 15:38:41 +02:00
|
|
|
QString
|
|
|
|
event_body(const mtx::events::collections::TimelineEvents &event);
|
|
|
|
|
2018-04-29 14:42:40 +02:00
|
|
|
//! Match widgets/events with a description message.
|
|
|
|
template<class T>
|
|
|
|
QString
|
2019-07-29 00:40:23 +02:00
|
|
|
messageDescription(const QString &username = "",
|
|
|
|
const QString &body = "",
|
|
|
|
const bool isLocal = false)
|
2018-04-29 14:42:40 +02:00
|
|
|
{
|
2020-07-11 01:19:48 +02:00
|
|
|
using Audio = mtx::events::RoomEvent<mtx::events::msg::Audio>;
|
|
|
|
using Emote = mtx::events::RoomEvent<mtx::events::msg::Emote>;
|
|
|
|
using File = mtx::events::RoomEvent<mtx::events::msg::File>;
|
|
|
|
using Image = mtx::events::RoomEvent<mtx::events::msg::Image>;
|
|
|
|
using Notice = mtx::events::RoomEvent<mtx::events::msg::Notice>;
|
|
|
|
using Sticker = mtx::events::Sticker;
|
|
|
|
using Text = mtx::events::RoomEvent<mtx::events::msg::Text>;
|
|
|
|
using Video = mtx::events::RoomEvent<mtx::events::msg::Video>;
|
|
|
|
using CallInvite = mtx::events::RoomEvent<mtx::events::msg::CallInvite>;
|
|
|
|
using CallAnswer = mtx::events::RoomEvent<mtx::events::msg::CallAnswer>;
|
|
|
|
using CallHangUp = mtx::events::RoomEvent<mtx::events::msg::CallHangUp>;
|
|
|
|
using Encrypted = mtx::events::EncryptedEvent<mtx::events::msg::Encrypted>;
|
2018-04-29 14:42:40 +02:00
|
|
|
|
2019-11-09 03:06:10 +01:00
|
|
|
if (std::is_same<T, Audio>::value) {
|
2019-11-02 18:17:06 +01:00
|
|
|
if (isLocal)
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"You sent an audio clip");
|
|
|
|
else
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"%1 sent an audio clip")
|
|
|
|
.arg(username);
|
2019-11-09 03:06:10 +01:00
|
|
|
} else if (std::is_same<T, Image>::value) {
|
2019-11-02 18:17:06 +01:00
|
|
|
if (isLocal)
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"You sent an image");
|
|
|
|
else
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"%1 sent an image")
|
|
|
|
.arg(username);
|
2019-11-09 03:06:10 +01:00
|
|
|
} else if (std::is_same<T, File>::value) {
|
2019-11-02 18:17:06 +01:00
|
|
|
if (isLocal)
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"You sent a file");
|
|
|
|
else
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"%1 sent a file")
|
|
|
|
.arg(username);
|
2019-11-09 03:06:10 +01:00
|
|
|
} else if (std::is_same<T, Video>::value) {
|
2019-11-02 18:17:06 +01:00
|
|
|
if (isLocal)
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"You sent a video");
|
|
|
|
else
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"%1 sent a video")
|
|
|
|
.arg(username);
|
2019-11-09 03:06:10 +01:00
|
|
|
} else if (std::is_same<T, Sticker>::value) {
|
2019-11-02 18:17:06 +01:00
|
|
|
if (isLocal)
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"You sent a sticker");
|
|
|
|
else
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"%1 sent a sticker")
|
|
|
|
.arg(username);
|
2019-07-29 00:40:23 +02:00
|
|
|
} else if (std::is_same<T, Notice>::value) {
|
2019-11-02 18:17:06 +01:00
|
|
|
if (isLocal)
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"You sent a notification");
|
|
|
|
else
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"%1 sent a notification")
|
|
|
|
.arg(username);
|
2019-07-29 00:40:23 +02:00
|
|
|
} else if (std::is_same<T, Text>::value) {
|
2019-11-02 18:17:06 +01:00
|
|
|
if (isLocal)
|
|
|
|
return QCoreApplication::translate("message-description sent:", "You: %1")
|
|
|
|
.arg(body);
|
|
|
|
else
|
|
|
|
return QCoreApplication::translate("message-description sent:", "%1: %2")
|
|
|
|
.arg(username)
|
|
|
|
.arg(body);
|
2019-07-29 00:40:23 +02:00
|
|
|
} else if (std::is_same<T, Emote>::value) {
|
2018-04-29 14:42:40 +02:00
|
|
|
return QString("* %1 %2").arg(username).arg(body);
|
2019-07-29 00:40:23 +02:00
|
|
|
} else if (std::is_same<T, Encrypted>::value) {
|
2019-11-02 18:17:06 +01:00
|
|
|
if (isLocal)
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"You sent an encrypted message");
|
|
|
|
else
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"%1 sent an encrypted message")
|
|
|
|
.arg(username);
|
2020-07-11 01:19:48 +02:00
|
|
|
} else if (std::is_same<T, CallInvite>::value) {
|
|
|
|
if (isLocal)
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"You placed a call");
|
|
|
|
else
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"%1 placed a call")
|
|
|
|
.arg(username);
|
|
|
|
} else if (std::is_same<T, CallAnswer>::value) {
|
|
|
|
if (isLocal)
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"You answered a call");
|
|
|
|
else
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"%1 answered a call")
|
|
|
|
.arg(username);
|
|
|
|
} else if (std::is_same<T, CallHangUp>::value) {
|
|
|
|
if (isLocal)
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"You ended a call");
|
|
|
|
else
|
|
|
|
return QCoreApplication::translate("message-description sent:",
|
|
|
|
"%1 ended a call")
|
|
|
|
.arg(username);
|
2019-07-29 00:40:23 +02:00
|
|
|
} else {
|
|
|
|
return QCoreApplication::translate("utils", "Unknown Message Type");
|
|
|
|
}
|
2018-04-29 14:42:40 +02:00
|
|
|
}
|
|
|
|
|
2018-02-19 22:32:37 +01:00
|
|
|
//! Scale down an image to fit to the given width & height limitations.
|
2018-08-01 20:10:03 +02:00
|
|
|
QPixmap
|
|
|
|
scaleDown(uint64_t maxWidth, uint64_t maxHeight, const QPixmap &source);
|
2018-03-24 22:16:15 +01:00
|
|
|
|
2018-04-27 17:19:43 +02:00
|
|
|
//! Delete items in a container based on a predicate.
|
|
|
|
template<typename ContainerT, typename PredicateT>
|
|
|
|
void
|
|
|
|
erase_if(ContainerT &items, const PredicateT &predicate)
|
|
|
|
{
|
|
|
|
for (auto it = items.begin(); it != items.end();) {
|
|
|
|
if (predicate(*it))
|
|
|
|
it = items.erase(it);
|
|
|
|
else
|
|
|
|
++it;
|
|
|
|
}
|
2018-04-28 15:17:36 +02:00
|
|
|
}
|
2018-04-27 17:19:43 +02:00
|
|
|
|
2018-05-05 15:38:41 +02:00
|
|
|
template<class T>
|
|
|
|
QString
|
|
|
|
message_body(const mtx::events::collections::TimelineEvents &event)
|
|
|
|
{
|
2019-12-14 17:08:36 +01:00
|
|
|
return QString::fromStdString(std::get<T>(event).content.body);
|
2018-05-05 15:38:41 +02:00
|
|
|
}
|
|
|
|
|
2018-03-24 22:16:15 +01:00
|
|
|
//! Calculate the Levenshtein distance between two strings with character skipping.
|
|
|
|
int
|
|
|
|
levenshtein_distance(const std::string &s1, const std::string &s2);
|
2018-05-11 15:00:14 +02:00
|
|
|
|
|
|
|
QPixmap
|
|
|
|
scaleImageToPixmap(const QImage &img, int size);
|
2018-07-15 19:05:31 +02:00
|
|
|
|
|
|
|
//! Convert a Content Matrix URI to an HTTP link.
|
|
|
|
QString
|
|
|
|
mxcToHttp(const QUrl &url, const QString &server, int port);
|
2018-08-21 08:22:51 +02:00
|
|
|
|
|
|
|
//! Convert a ed25519 fingerprint into a human readable form
|
|
|
|
QString
|
|
|
|
humanReadableFingerprint(const std::string &ed25519);
|
|
|
|
|
|
|
|
QString
|
|
|
|
humanReadableFingerprint(const QString &ed25519);
|
2018-09-06 21:34:41 +02:00
|
|
|
|
|
|
|
//! Retrieve the message body taking into account the `formatted_body` field.
|
|
|
|
//! If the `format` of the message is not supported we fallback to `body`.
|
|
|
|
template<typename RoomMessageT>
|
|
|
|
QString
|
2018-09-07 13:52:29 +02:00
|
|
|
getMessageBody(const RoomMessageT &event)
|
2018-09-06 21:34:41 +02:00
|
|
|
{
|
|
|
|
if (event.content.format.empty())
|
2018-09-13 10:02:54 +02:00
|
|
|
return QString::fromStdString(event.content.body).toHtmlEscaped();
|
2018-09-06 21:34:41 +02:00
|
|
|
|
2020-01-29 23:39:05 +01:00
|
|
|
if (event.content.format != mtx::common::FORMAT_MSG_TYPE)
|
2018-09-13 10:02:54 +02:00
|
|
|
return QString::fromStdString(event.content.body).toHtmlEscaped();
|
2018-09-06 21:34:41 +02:00
|
|
|
|
|
|
|
return QString::fromStdString(event.content.formatted_body);
|
|
|
|
}
|
2018-09-07 13:52:29 +02:00
|
|
|
|
|
|
|
//! Replace raw URLs in text with HTML link tags.
|
|
|
|
QString
|
|
|
|
linkifyMessage(const QString &body);
|
2018-09-07 19:05:30 +02:00
|
|
|
|
|
|
|
//! Convert the input markdown text to html.
|
2018-09-12 13:20:12 +02:00
|
|
|
QString
|
2018-09-07 19:05:30 +02:00
|
|
|
markdownToHtml(const QString &text);
|
2018-09-19 21:42:26 +02:00
|
|
|
|
2020-01-14 17:47:30 +01:00
|
|
|
//! Escape every html tag, that was not whitelisted
|
|
|
|
QString
|
|
|
|
escapeBlacklistedHtml(const QString &data);
|
|
|
|
|
2019-06-14 04:33:04 +02:00
|
|
|
//! Generate a Rich Reply quote message
|
|
|
|
QString
|
|
|
|
getFormattedQuoteBody(const RelatedInfo &related, const QString &html);
|
|
|
|
|
2019-07-05 03:20:19 +02:00
|
|
|
//! Get the body for the quote, depending on the event type.
|
|
|
|
QString
|
|
|
|
getQuoteBody(const RelatedInfo &related);
|
|
|
|
|
2018-09-21 09:55:24 +02:00
|
|
|
//! Retrieve the color of the links based on the current theme.
|
2018-09-19 21:42:26 +02:00
|
|
|
QString
|
|
|
|
linkColor();
|
2018-09-21 09:55:24 +02:00
|
|
|
|
2019-01-18 18:17:25 +01:00
|
|
|
//! Returns the hash code of the input QString
|
2019-06-27 14:11:02 +02:00
|
|
|
uint32_t
|
2019-01-18 18:17:25 +01:00
|
|
|
hashQString(const QString &input);
|
|
|
|
|
|
|
|
//! Generate a color (matching #RRGGBB) that has an acceptable contrast to background that is based
|
|
|
|
//! on the input string.
|
|
|
|
QString
|
|
|
|
generateContrastingHexColor(const QString &input, const QString &background);
|
|
|
|
|
|
|
|
//! Given two luminance values, compute the contrast ratio between them.
|
|
|
|
qreal
|
|
|
|
computeContrast(const qreal &one, const qreal &two);
|
|
|
|
|
|
|
|
//! Compute the luminance of a single color. Based on https://stackoverflow.com/a/9733420
|
|
|
|
qreal
|
|
|
|
luminance(const QColor &col);
|
2019-01-18 05:09:42 +01:00
|
|
|
|
2018-09-21 09:55:24 +02:00
|
|
|
//! Center a widget in relation to another widget.
|
|
|
|
void
|
|
|
|
centerWidget(QWidget *widget, QWidget *parent);
|
2018-10-01 16:56:46 +02:00
|
|
|
|
|
|
|
void
|
|
|
|
restoreCombobox(QComboBox *combo, const QString &value);
|
2020-04-26 11:26:51 +02:00
|
|
|
|
|
|
|
//! Read image respecting exif orientation
|
|
|
|
QImage
|
2021-03-02 01:45:16 +01:00
|
|
|
readImage(const QByteArray &data);
|
2021-02-20 19:05:21 +01:00
|
|
|
|
|
|
|
bool
|
|
|
|
isReply(const mtx::events::collections::TimelineEvents &e);
|
2018-09-01 12:35:10 +02:00
|
|
|
}
|