1189 lines
41 KiB
C++
1189 lines
41 KiB
C++
// SPDX-FileCopyrightText: 2018-2020 Black Hat <bhat@encom.eu.org>
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
#include "neochatroom.h"
|
|
|
|
#include <QFileInfo>
|
|
#include <QMetaObject>
|
|
#include <QMimeDatabase>
|
|
#include <QTextDocument>
|
|
#include <functional>
|
|
|
|
#include <QMediaMetaData>
|
|
#include <QMediaPlayer>
|
|
|
|
#include <qcoro/qcorosignal.h>
|
|
#include <qcoro/task.h>
|
|
|
|
#include <connection.h>
|
|
#include <csapi/pushrules.h>
|
|
#include <csapi/redaction.h>
|
|
#include <csapi/report_content.h>
|
|
#include <csapi/room_state.h>
|
|
#include <csapi/typing.h>
|
|
#include <events/encryptionevent.h>
|
|
#include <events/reactionevent.h>
|
|
#include <events/redactionevent.h>
|
|
#include <events/roomavatarevent.h>
|
|
#include <events/roomcanonicalaliasevent.h>
|
|
#include <events/roommemberevent.h>
|
|
#include <events/roompowerlevelsevent.h>
|
|
#include <events/simplestateevents.h>
|
|
#include <jobs/downloadfilejob.h>
|
|
#include <qt_connection_util.h>
|
|
|
|
#include "controller.h"
|
|
#include "joinrulesevent.h"
|
|
#include "neochatconfig.h"
|
|
#include "neochatuser.h"
|
|
#include "notificationsmanager.h"
|
|
#include "stickerevent.h"
|
|
#include "utils.h"
|
|
|
|
#include <KLocalizedString>
|
|
|
|
using namespace Quotient;
|
|
|
|
NeoChatRoom::NeoChatRoom(Connection *connection, QString roomId, JoinState joinState)
|
|
: Room(connection, std::move(roomId), joinState)
|
|
{
|
|
connect(connection, &Connection::accountDataChanged, this, &NeoChatRoom::updatePushNotificationState);
|
|
connect(this, &NeoChatRoom::notificationCountChanged, this, &NeoChatRoom::countChanged);
|
|
connect(this, &NeoChatRoom::highlightCountChanged, this, &NeoChatRoom::countChanged);
|
|
connect(this, &Room::fileTransferCompleted, this, [this] {
|
|
setFileUploadingProgress(0);
|
|
setHasFileUploading(false);
|
|
});
|
|
|
|
connect(this, &Room::aboutToAddHistoricalMessages, this, &NeoChatRoom::readMarkerLoadedChanged);
|
|
|
|
connect(this, &Quotient::Room::eventsHistoryJobChanged, this, &NeoChatRoom::lastActiveTimeChanged);
|
|
|
|
connect(this, &Room::joinStateChanged, this, [this](JoinState oldState, JoinState newState) {
|
|
if (oldState == JoinState::Invite && newState != JoinState::Invite) {
|
|
Q_EMIT isInviteChanged();
|
|
}
|
|
});
|
|
connect(this, &Room::displaynameChanged, this, &NeoChatRoom::displayNameChanged);
|
|
|
|
connectSingleShot(this, &Room::baseStateLoaded, this, [this]() {
|
|
Q_EMIT canEncryptRoomChanged();
|
|
if (this->joinState() != JoinState::Invite) {
|
|
return;
|
|
}
|
|
const QString senderId = getCurrentState<RoomMemberEvent>(localUser()->id())->senderId();
|
|
QImage avatar_image;
|
|
if (!user(senderId)->avatarUrl(this).isEmpty()) {
|
|
avatar_image = user(senderId)->avatar(128, this);
|
|
} else {
|
|
qWarning() << "using this room's avatar";
|
|
avatar_image = avatar(128);
|
|
}
|
|
NotificationsManager::instance().postInviteNotification(this, htmlSafeDisplayName(), htmlSafeMemberName(senderId), avatar_image);
|
|
});
|
|
connect(this, &Room::changed, this, [this] {
|
|
Q_EMIT canEncryptRoomChanged();
|
|
});
|
|
}
|
|
|
|
void NeoChatRoom::uploadFile(const QUrl &url, const QString &body)
|
|
{
|
|
doUploadFile(url, body);
|
|
}
|
|
|
|
QCoro::Task<void> NeoChatRoom::doUploadFile(QUrl url, QString body)
|
|
{
|
|
if (url.isEmpty()) {
|
|
co_return;
|
|
}
|
|
|
|
auto mime = QMimeDatabase().mimeTypeForUrl(url);
|
|
QFileInfo fileInfo(url.toLocalFile());
|
|
EventContent::TypedBase *content;
|
|
if (mime.name().startsWith("image/")) {
|
|
QImage image(url.toLocalFile());
|
|
content = new EventContent::ImageContent(url, fileInfo.size(), mime, image.size(), fileInfo.fileName());
|
|
} else if (mime.name().startsWith("audio/")) {
|
|
content = new EventContent::AudioContent(url, fileInfo.size(), mime, fileInfo.fileName());
|
|
} else if (mime.name().startsWith("video/")) {
|
|
QMediaPlayer player;
|
|
#if QT_VERSION > QT_VERSION_CHECK(6, 0, 0)
|
|
player.setSource(url);
|
|
#else
|
|
player.setMedia(url);
|
|
#endif
|
|
co_await qCoro(&player, &QMediaPlayer::mediaStatusChanged);
|
|
#if QT_VERSION > QT_VERSION_CHECK(6, 0, 0)
|
|
auto resolution = player.metaData().value(QMediaMetaData::Resolution).toSize();
|
|
#else
|
|
auto resolution = player.metaData(QMediaMetaData::Resolution).toSize();
|
|
#endif
|
|
content = new EventContent::VideoContent(url, fileInfo.size(), mime, resolution, fileInfo.fileName());
|
|
} else {
|
|
content = new EventContent::FileContent(url, fileInfo.size(), mime, fileInfo.fileName());
|
|
}
|
|
#ifdef QUOTIENT_07
|
|
QString txnId = postFile(body.isEmpty() ? url.fileName() : body, content);
|
|
#else
|
|
QString txnId = postFile(body.isEmpty() ? url.fileName() : body, url, false);
|
|
#endif
|
|
setHasFileUploading(true);
|
|
#ifdef QUOTIENT_07
|
|
connect(this, &Room::fileTransferCompleted, [this, txnId](const QString &id, FileSourceInfo) {
|
|
#else
|
|
connect(this, &Room::fileTransferCompleted, [this, txnId](const QString &id, const QUrl & /*localFile*/, const QUrl & /*mxcUrl*/) {
|
|
#endif
|
|
if (id == txnId) {
|
|
setFileUploadingProgress(0);
|
|
setHasFileUploading(false);
|
|
}
|
|
});
|
|
connect(this, &Room::fileTransferFailed, [this, txnId](const QString &id, const QString & /*error*/) {
|
|
if (id == txnId) {
|
|
setFileUploadingProgress(0);
|
|
setHasFileUploading(false);
|
|
}
|
|
});
|
|
connect(this, &Room::fileTransferProgress, [this, txnId](const QString &id, qint64 progress, qint64 total) {
|
|
if (id == txnId) {
|
|
setFileUploadingProgress(int(float(progress) / float(total) * 100));
|
|
}
|
|
});
|
|
}
|
|
|
|
void NeoChatRoom::acceptInvitation()
|
|
{
|
|
connection()->joinRoom(id());
|
|
}
|
|
|
|
void NeoChatRoom::forget()
|
|
{
|
|
connection()->forgetRoom(id());
|
|
}
|
|
|
|
QVariantList NeoChatRoom::getUsersTyping() const
|
|
{
|
|
auto users = usersTyping();
|
|
users.removeAll(localUser());
|
|
QVariantList userVariants;
|
|
for (User *user : users) {
|
|
userVariants.append(QVariantMap{
|
|
{"id", user->id()},
|
|
{"avatarMediaId", user->avatarMediaId(this)},
|
|
{"displayName", user->displayname(this)},
|
|
{"display", user->name()},
|
|
});
|
|
}
|
|
return userVariants;
|
|
}
|
|
|
|
void NeoChatRoom::sendTypingNotification(bool isTyping)
|
|
{
|
|
connection()->callApi<SetTypingJob>(BackgroundRequest, localUser()->id(), id(), isTyping, 10000);
|
|
}
|
|
|
|
const RoomMessageEvent *NeoChatRoom::lastEvent(bool ignoreStateEvent) const
|
|
{
|
|
for (auto timelineItem = messageEvents().rbegin(); timelineItem < messageEvents().rend(); timelineItem++) {
|
|
const RoomEvent *event = timelineItem->get();
|
|
|
|
if (is<RedactionEvent>(*event) || is<ReactionEvent>(*event)) {
|
|
continue;
|
|
}
|
|
if (event->isRedacted()) {
|
|
continue;
|
|
}
|
|
|
|
if (event->isStateEvent()
|
|
&& (ignoreStateEvent || !NeoChatConfig::self()->showLeaveJoinEvent() || static_cast<const StateEventBase &>(*event).repeatsState())) {
|
|
continue;
|
|
}
|
|
|
|
if (auto roomEvent = eventCast<const RoomMessageEvent>(event)) {
|
|
if (!roomEvent->replacedEvent().isEmpty() && roomEvent->replacedEvent() != roomEvent->id()) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (connection()->isIgnored(user(event->senderId()))) {
|
|
continue;
|
|
}
|
|
|
|
if (auto lastEvent = eventCast<const RoomMessageEvent>(event)) {
|
|
return lastEvent;
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
bool NeoChatRoom::lastEventIsSpoiler() const
|
|
{
|
|
if (auto event = lastEvent()) {
|
|
if (auto e = eventCast<const RoomMessageEvent>(event)) {
|
|
if (e->hasTextContent() && e->content() && e->mimeType().name() == "text/html") {
|
|
auto htmlBody = static_cast<const Quotient::EventContent::TextContent *>(e->content())->body;
|
|
return htmlBody.contains("data-mx-spoiler");
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
QString NeoChatRoom::lastEventToString() const
|
|
{
|
|
if (auto event = lastEvent()) {
|
|
return roomMembername(event->senderId()) + (event->isStateEvent() ? " " : ": ") + eventToString(*event);
|
|
}
|
|
return QLatin1String("");
|
|
}
|
|
|
|
bool NeoChatRoom::isEventHighlighted(const RoomEvent *e) const
|
|
{
|
|
return highlights.contains(e);
|
|
}
|
|
|
|
void NeoChatRoom::checkForHighlights(const Quotient::TimelineItem &ti)
|
|
{
|
|
auto localUserId = localUser()->id();
|
|
if (ti->senderId() == localUserId) {
|
|
return;
|
|
}
|
|
if (auto *e = ti.viewAs<RoomMessageEvent>()) {
|
|
const auto &text = e->plainBody();
|
|
if (text.contains(localUserId) || text.contains(roomMembername(localUserId))) {
|
|
highlights.insert(e);
|
|
}
|
|
}
|
|
}
|
|
|
|
void NeoChatRoom::onAddNewTimelineEvents(timeline_iter_t from)
|
|
{
|
|
std::for_each(from, messageEvents().cend(), [this](const TimelineItem &ti) {
|
|
checkForHighlights(ti);
|
|
});
|
|
}
|
|
|
|
void NeoChatRoom::onAddHistoricalTimelineEvents(rev_iter_t from)
|
|
{
|
|
std::for_each(from, messageEvents().crend(), [this](const TimelineItem &ti) {
|
|
checkForHighlights(ti);
|
|
});
|
|
}
|
|
|
|
void NeoChatRoom::onRedaction(const RoomEvent &prevEvent, const RoomEvent & /*after*/)
|
|
{
|
|
if (const auto &e = eventCast<const ReactionEvent>(&prevEvent)) {
|
|
if (auto relatedEventId = e->relation().eventId; !relatedEventId.isEmpty()) {
|
|
Q_EMIT updatedEvent(relatedEventId);
|
|
}
|
|
}
|
|
}
|
|
|
|
void NeoChatRoom::countChanged()
|
|
{
|
|
if (displayed() && !hasUnreadMessages()) {
|
|
resetNotificationCount();
|
|
resetHighlightCount();
|
|
}
|
|
}
|
|
|
|
QDateTime NeoChatRoom::lastActiveTime()
|
|
{
|
|
if (timelineSize() == 0) {
|
|
return QDateTime();
|
|
}
|
|
|
|
if (auto event = lastEvent(true)) {
|
|
return event->originTimestamp();
|
|
}
|
|
|
|
// no message found, take last event
|
|
return messageEvents().rbegin()->get()->originTimestamp();
|
|
}
|
|
|
|
QString NeoChatRoom::subtitleText()
|
|
{
|
|
QString subtitle = this->lastEventToString().size() == 0 ? this->topic() : this->lastEventToString();
|
|
|
|
subtitle
|
|
// replace blockquote, i.e. '> text'
|
|
.replace(QRegularExpression("(\r\n\t|\n|\r\t|)> "), " ")
|
|
// replace headings, i.e. "# text"
|
|
.replace(QRegularExpression("(\r\n\t|\n|\r\t|)\\#{1,6} "), " ")
|
|
// replace newlines
|
|
.replace(QRegularExpression("(\r\n\t|\n|\r\t)"), " ")
|
|
// replace '**text**' and '__text__'
|
|
.replace(QRegularExpression("(\\*\\*|__)(?=\\S)([^\\r]*\\S)\\1"), "\\2")
|
|
// replace '*text*' and '_text_'
|
|
.replace(QRegularExpression("(\\*|_)(?=\\S)([^\\r]*\\S)\\1"), "\\2")
|
|
// replace '~~text~~'
|
|
.replace(QRegularExpression("~~(.*)~~"), "\\1")
|
|
// replace '~text~'
|
|
.replace(QRegularExpression("~(.*)~"), "\\1")
|
|
// replace '<del>text</del>'
|
|
.replace(QRegularExpression("<del>(.*)</del>"), "\\1")
|
|
// replace '```code```'
|
|
.replace(QRegularExpression("```([^```]+)```"), "\\1")
|
|
// replace '`code`'
|
|
.replace(QRegularExpression("`([^`]+)`"), "\\1");
|
|
|
|
return subtitle.size() > 0 ? subtitle : QStringLiteral(" ");
|
|
}
|
|
|
|
int NeoChatRoom::savedTopVisibleIndex() const
|
|
{
|
|
return firstDisplayedMarker() == historyEdge() ? 0 : int(firstDisplayedMarker() - messageEvents().rbegin());
|
|
}
|
|
|
|
int NeoChatRoom::savedBottomVisibleIndex() const
|
|
{
|
|
return lastDisplayedMarker() == historyEdge() ? 0 : int(lastDisplayedMarker() - messageEvents().rbegin());
|
|
}
|
|
|
|
void NeoChatRoom::saveViewport(int topIndex, int bottomIndex)
|
|
{
|
|
if (topIndex == -1 || bottomIndex == -1 || (bottomIndex == savedBottomVisibleIndex() && (bottomIndex == 0 || topIndex == savedTopVisibleIndex()))) {
|
|
return;
|
|
}
|
|
if (bottomIndex == 0) {
|
|
setFirstDisplayedEventId({});
|
|
setLastDisplayedEventId({});
|
|
return;
|
|
}
|
|
setFirstDisplayedEvent(maxTimelineIndex() - topIndex);
|
|
setLastDisplayedEvent(maxTimelineIndex() - bottomIndex);
|
|
}
|
|
|
|
QVariantList NeoChatRoom::getUsers(const QString &keyword, int limit) const
|
|
{
|
|
const auto userList = users();
|
|
QVariantList matchedList;
|
|
int count = 0;
|
|
for (const auto u : userList) {
|
|
if (u->displayname(this).contains(keyword, Qt::CaseInsensitive)) {
|
|
NeoChatUser user(u->id(), u->connection());
|
|
QVariantMap userVariant{{QStringLiteral("id"), user.id()},
|
|
{QStringLiteral("displayName"), user.displayname(this)},
|
|
{QStringLiteral("avatarMediaId"), user.avatarMediaId(this)},
|
|
{QStringLiteral("color"), user.color()}};
|
|
|
|
matchedList.append(QVariant::fromValue(userVariant));
|
|
count++;
|
|
if (count == limit) { // -1 is infinite
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return matchedList;
|
|
}
|
|
|
|
QVariantMap NeoChatRoom::getUser(const QString &userID) const
|
|
{
|
|
NeoChatUser user(userID, connection());
|
|
return QVariantMap{{QStringLiteral("id"), user.id()},
|
|
{QStringLiteral("displayName"), user.displayname(this)},
|
|
{QStringLiteral("avatarMediaId"), user.avatarMediaId(this)},
|
|
{QStringLiteral("color"), user.color()}};
|
|
}
|
|
|
|
QUrl NeoChatRoom::urlToMxcUrl(const QUrl &mxcUrl)
|
|
{
|
|
#ifdef QUOTIENT_07
|
|
return connection()->makeMediaUrl(mxcUrl);
|
|
#else
|
|
return DownloadFileJob::makeRequestUrl(connection()->homeserver(), mxcUrl);
|
|
#endif
|
|
}
|
|
|
|
QString NeoChatRoom::avatarMediaId() const
|
|
{
|
|
if (const auto avatar = Room::avatarMediaId(); !avatar.isEmpty()) {
|
|
return avatar;
|
|
}
|
|
|
|
// Use the first (excluding self) user's avatar for direct chats
|
|
const auto dcUsers = directChatUsers();
|
|
for (const auto u : dcUsers) {
|
|
if (u != localUser()) {
|
|
return u->avatarMediaId(this);
|
|
}
|
|
}
|
|
|
|
return {};
|
|
}
|
|
|
|
QString NeoChatRoom::eventToString(const RoomEvent &evt, Qt::TextFormat format, bool removeReply) const
|
|
{
|
|
const bool prettyPrint = (format == Qt::RichText);
|
|
|
|
using namespace Quotient;
|
|
#ifdef QUOTIENT_07
|
|
return switchOnType(
|
|
#else
|
|
return visit(
|
|
#endif
|
|
evt,
|
|
[this, prettyPrint, removeReply](const RoomMessageEvent &e) {
|
|
using namespace MessageEventContent;
|
|
|
|
// 1. prettyPrint/HTML
|
|
if (prettyPrint && e.hasTextContent() && e.mimeType().name() != "text/plain") {
|
|
auto htmlBody = static_cast<const TextContent *>(e.content())->body;
|
|
if (removeReply) {
|
|
htmlBody.remove(utils::removeRichReplyRegex);
|
|
}
|
|
htmlBody.replace(utils::userPillRegExp, R"(<b class="user-pill">\1</b>)");
|
|
htmlBody.replace(utils::strikethroughRegExp, "<s>\\1</s>");
|
|
|
|
auto url = connection()->homeserver();
|
|
auto base = url.scheme() + QStringLiteral("://") + url.host() + (url.port() != -1 ? ':' + QString::number(url.port()) : QString());
|
|
htmlBody.replace(utils::mxcImageRegExp, QStringLiteral(R"(<img \1 src="%1/_matrix/media/r0/download/\2/\3" \4 > )").arg(base));
|
|
|
|
return htmlBody;
|
|
}
|
|
|
|
if (e.hasFileContent()) {
|
|
auto fileCaption = e.content()->fileInfo()->originalName.toHtmlEscaped();
|
|
if (fileCaption.isEmpty()) {
|
|
fileCaption = prettyPrint ? Quotient::prettyPrint(e.plainBody()) : e.plainBody();
|
|
} else if (e.content()->fileInfo()->originalName != e.plainBody()) {
|
|
fileCaption = e.plainBody() + " | " + fileCaption;
|
|
}
|
|
return !fileCaption.isEmpty() ? fileCaption : i18n("a file");
|
|
}
|
|
|
|
// 2. prettyPrint/text 3. plainText/HTML 4. plainText/text
|
|
QString plainBody;
|
|
if (e.hasTextContent() && e.content() && e.mimeType().name() == "text/plain") { // 2/4
|
|
plainBody = static_cast<const TextContent *>(e.content())->body;
|
|
} else { // 3
|
|
plainBody = e.plainBody();
|
|
}
|
|
|
|
if (prettyPrint) {
|
|
if (removeReply) {
|
|
plainBody.remove(utils::removeReplyRegex);
|
|
}
|
|
return Quotient::prettyPrint(plainBody);
|
|
}
|
|
if (removeReply) {
|
|
return plainBody.remove(utils::removeReplyRegex);
|
|
}
|
|
return plainBody;
|
|
},
|
|
[](const StickerEvent &e) {
|
|
return e.body();
|
|
},
|
|
[this](const RoomMemberEvent &e) {
|
|
// FIXME: Rewind to the name that was at the time of this event
|
|
auto subjectName = this->htmlSafeMemberName(e.userId());
|
|
if (e.membership() == MembershipType::Leave) {
|
|
auto displayName = e.prevContent()->displayName;
|
|
#ifdef QUOTIENT_07
|
|
if (displayName) {
|
|
subjectName = sanitized(*displayName).toHtmlEscaped();
|
|
#else
|
|
if (displayName.isEmpty()) {
|
|
subjectName = sanitized(displayName).toHtmlEscaped();
|
|
#endif
|
|
}
|
|
}
|
|
|
|
// The below code assumes senderName output in AuthorRole
|
|
switch (e.membership()) {
|
|
case MembershipType::Invite:
|
|
if (e.repeatsState()) {
|
|
auto text = i18n("reinvited %1 to the room", subjectName);
|
|
if (!e.reason().isEmpty()) {
|
|
text += i18nc("Optional reason for an invitation", ": %1") + e.reason().toHtmlEscaped();
|
|
}
|
|
return text;
|
|
}
|
|
Q_FALLTHROUGH();
|
|
case MembershipType::Join: {
|
|
QString text{};
|
|
// Part 1: invites and joins
|
|
if (e.repeatsState()) {
|
|
text = i18n("joined the room (repeated)");
|
|
} else if (e.changesMembership()) {
|
|
text = e.membership() == MembershipType::Invite ? i18n("invited %1 to the room", subjectName) : i18n("joined the room");
|
|
}
|
|
if (!text.isEmpty()) {
|
|
if (!e.reason().isEmpty()) {
|
|
text += i18n(": %1", e.reason().toHtmlEscaped());
|
|
}
|
|
return text;
|
|
}
|
|
// Part 2: profile changes of joined members
|
|
if (e.isRename() && NeoChatConfig::self()->showRename()) {
|
|
if (e.displayName().isEmpty()) {
|
|
text = i18nc("their refers to a singular user", "cleared their display name");
|
|
} else {
|
|
text = i18nc("their refers to a singular user", "changed their display name to %1", e.displayName().toHtmlEscaped());
|
|
}
|
|
}
|
|
if (e.isAvatarUpdate() && NeoChatConfig::self()->showAvatarUpdate()) {
|
|
if (!text.isEmpty()) {
|
|
text += i18n(" and ");
|
|
}
|
|
if (e.avatarUrl().isEmpty()) {
|
|
text += i18nc("their refers to a singular user", "cleared their avatar");
|
|
#ifdef QUOTIENT_07
|
|
} else if (!e.prevContent()->avatarUrl) {
|
|
#else
|
|
} else if (e.prevContent()->avatarUrl.isEmpty()) {
|
|
#endif
|
|
text += i18n("set an avatar");
|
|
} else {
|
|
text += i18nc("their refers to a singular user", "updated their avatar");
|
|
}
|
|
}
|
|
return text;
|
|
}
|
|
case MembershipType::Leave:
|
|
if (e.prevContent() && e.prevContent()->membership == MembershipType::Invite) {
|
|
return (e.senderId() != e.userId()) ? i18n("withdrew %1's invitation", subjectName) : i18n("rejected the invitation");
|
|
}
|
|
|
|
if (e.prevContent() && e.prevContent()->membership == MembershipType::Ban) {
|
|
return (e.senderId() != e.userId()) ? i18n("unbanned %1", subjectName) : i18n("self-unbanned");
|
|
}
|
|
return (e.senderId() != e.userId())
|
|
? i18n("has put %1 out of the room: %2", subjectName, e.contentJson()["reason"_ls].toString().toHtmlEscaped())
|
|
: i18n("left the room");
|
|
case MembershipType::Ban:
|
|
if (e.senderId() != e.userId()) {
|
|
if (e.reason().isEmpty()) {
|
|
return i18n("banned %1 from the room", subjectName);
|
|
} else {
|
|
return i18n("banned %1 from the room: %2", subjectName, e.reason().toHtmlEscaped());
|
|
}
|
|
} else {
|
|
return i18n("self-banned from the room");
|
|
}
|
|
case MembershipType::Knock:
|
|
return i18n("requested an invite");
|
|
default:;
|
|
}
|
|
return i18n("made something unknown");
|
|
},
|
|
[](const RoomCanonicalAliasEvent &e) {
|
|
return (e.alias().isEmpty()) ? i18n("cleared the room main alias") : i18n("set the room main alias to: %1", e.alias());
|
|
},
|
|
[](const RoomNameEvent &e) {
|
|
return (e.name().isEmpty()) ? i18n("cleared the room name") : i18n("set the room name to: %1", e.name().toHtmlEscaped());
|
|
},
|
|
[prettyPrint](const RoomTopicEvent &e) {
|
|
return (e.topic().isEmpty()) ? i18n("cleared the topic") : i18n("set the topic to: %1", prettyPrint ? Quotient::prettyPrint(e.topic()) : e.topic());
|
|
},
|
|
[](const RoomAvatarEvent &) {
|
|
return i18n("changed the room avatar");
|
|
},
|
|
[](const EncryptionEvent &) {
|
|
return i18n("activated End-to-End Encryption");
|
|
},
|
|
[](const RoomCreateEvent &e) {
|
|
return e.isUpgrade() ? i18n("upgraded the room to version %1", e.version().isEmpty() ? "1" : e.version().toHtmlEscaped())
|
|
: i18n("created the room, version %1", e.version().isEmpty() ? "1" : e.version().toHtmlEscaped());
|
|
},
|
|
[](const RoomPowerLevelsEvent &) {
|
|
return i18nc("'power level' means permission level", "changed the power levels for this room");
|
|
},
|
|
[](const StateEventBase &e) {
|
|
if (e.matrixType() == QLatin1String("m.room.server_acl")) {
|
|
return i18n("changed the server access control lists for this room");
|
|
}
|
|
if (e.matrixType() == QLatin1String("im.vector.modular.widgets")) {
|
|
if (e.fullJson()["unsigned"]["prev_content"].toObject().isEmpty()) {
|
|
return i18nc("[User] added <name> widget", "added %1 widget", e.contentJson()["name"].toString());
|
|
}
|
|
if (e.contentJson().isEmpty()) {
|
|
return i18nc("[User] removed <name> widget", "removed %1 widget", e.fullJson()["unsigned"]["prev_content"]["name"].toString());
|
|
}
|
|
return i18nc("[User] configured <name> widget", "configured %1 widget", e.contentJson()["name"].toString());
|
|
}
|
|
return e.stateKey().isEmpty() ? i18n("updated %1 state", e.matrixType())
|
|
: i18n("updated %1 state for %2", e.matrixType(), e.stateKey().toHtmlEscaped());
|
|
},
|
|
i18n("Unknown event"));
|
|
}
|
|
|
|
void NeoChatRoom::changeAvatar(const QUrl &localFile)
|
|
{
|
|
const auto job = connection()->uploadFile(localFile.toLocalFile());
|
|
#ifdef QUOTIENT_07
|
|
if (isJobPending(job)) {
|
|
#else
|
|
if (isJobRunning(job)) {
|
|
#endif
|
|
connect(job, &BaseJob::success, this, [this, job] {
|
|
#ifdef QUOTIENT_07
|
|
connection()->callApi<SetRoomStateWithKeyJob>(id(), "m.room.avatar", QString(), QJsonObject{{"url", job->contentUri().toString()}});
|
|
#else
|
|
connection()->callApi<SetRoomStateWithKeyJob>(id(), "m.room.avatar", QString(), QJsonObject{{"url", job->contentUri()}});
|
|
#endif
|
|
});
|
|
}
|
|
}
|
|
|
|
void NeoChatRoom::addLocalAlias(const QString &alias)
|
|
{
|
|
auto a = aliases();
|
|
if (a.contains(alias)) {
|
|
return;
|
|
}
|
|
|
|
a += alias;
|
|
|
|
setLocalAliases(a);
|
|
}
|
|
|
|
void NeoChatRoom::removeLocalAlias(const QString &alias)
|
|
{
|
|
auto a = aliases();
|
|
if (!a.contains(alias)) {
|
|
return;
|
|
}
|
|
|
|
a.removeAll(alias);
|
|
|
|
setLocalAliases(a);
|
|
}
|
|
|
|
QString msgTypeToString(MessageEventType msgType)
|
|
{
|
|
switch (msgType) {
|
|
case MessageEventType::Text:
|
|
return "m.text";
|
|
case MessageEventType::File:
|
|
return "m.file";
|
|
case MessageEventType::Audio:
|
|
return "m.audio";
|
|
case MessageEventType::Emote:
|
|
return "m.emote";
|
|
case MessageEventType::Image:
|
|
return "m.image";
|
|
case MessageEventType::Video:
|
|
return "m.video";
|
|
case MessageEventType::Notice:
|
|
return "m.notice";
|
|
case MessageEventType::Location:
|
|
return "m.location";
|
|
default:
|
|
return "m.text";
|
|
}
|
|
}
|
|
|
|
void NeoChatRoom::postMessage(const QString &rawText, const QString &text, MessageEventType type, const QString &replyEventId, const QString &relateToEventId)
|
|
{
|
|
postHtmlMessage(rawText, text, type, replyEventId, relateToEventId);
|
|
}
|
|
|
|
void NeoChatRoom::postHtmlMessage(const QString &text, const QString &html, MessageEventType type, const QString &replyEventId, const QString &relateToEventId)
|
|
{
|
|
bool isReply = !replyEventId.isEmpty();
|
|
bool isEdit = !relateToEventId.isEmpty();
|
|
const auto replyIt = findInTimeline(replyEventId);
|
|
if (replyIt == historyEdge()) {
|
|
isReply = false;
|
|
}
|
|
|
|
if (isEdit) {
|
|
QJsonObject json{
|
|
{"type", "m.room.message"},
|
|
{"msgtype", msgTypeToString(type)},
|
|
{"body", "* " + text},
|
|
{"format", "org.matrix.custom.html"},
|
|
{"formatted_body", html},
|
|
{"m.new_content", QJsonObject{{"body", text}, {"msgtype", msgTypeToString(type)}, {"format", "org.matrix.custom.html"}, {"formatted_body", html}}},
|
|
{"m.relates_to", QJsonObject{{"rel_type", "m.replace"}, {"event_id", relateToEventId}}}};
|
|
|
|
postJson("m.room.message", json);
|
|
return;
|
|
}
|
|
|
|
if (isReply) {
|
|
const auto &replyEvt = **replyIt;
|
|
|
|
// clang-format off
|
|
QJsonObject json{
|
|
{"msgtype", msgTypeToString(type)},
|
|
{"body", "> <" + replyEvt.senderId() + "> " + eventToString(replyEvt) + "\n\n" + text},
|
|
{"format", "org.matrix.custom.html"},
|
|
{"m.relates_to",
|
|
QJsonObject {
|
|
{"m.in_reply_to",
|
|
QJsonObject {
|
|
{"event_id", replyEventId}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
{"formatted_body",
|
|
"<mx-reply><blockquote><a href=\"https://matrix.to/#/" +
|
|
id() + "/" +
|
|
replyEventId +
|
|
"\">In reply to</a> <a href=\"https://matrix.to/#/" +
|
|
replyEvt.senderId() + "\">" + replyEvt.senderId() +
|
|
"</a><br>" + eventToString(replyEvt, Qt::RichText) +
|
|
"</blockquote></mx-reply>" + html
|
|
}
|
|
};
|
|
// clang-format on
|
|
|
|
postJson("m.room.message", json);
|
|
|
|
return;
|
|
}
|
|
|
|
Room::postHtmlMessage(text, html, type);
|
|
}
|
|
|
|
void NeoChatRoom::toggleReaction(const QString &eventId, const QString &reaction)
|
|
{
|
|
if (eventId.isEmpty() || reaction.isEmpty()) {
|
|
return;
|
|
}
|
|
|
|
const auto eventIt = findInTimeline(eventId);
|
|
if (eventIt == historyEdge()) {
|
|
return;
|
|
}
|
|
|
|
const auto &evt = **eventIt;
|
|
|
|
QStringList redactEventIds; // What if there are multiple reaction events?
|
|
|
|
const auto &annotations = relatedEvents(evt, EventRelation::Annotation());
|
|
if (!annotations.isEmpty()) {
|
|
for (const auto &a : annotations) {
|
|
if (auto e = eventCast<const ReactionEvent>(a)) {
|
|
if (e->relation().key != reaction) {
|
|
continue;
|
|
}
|
|
|
|
if (e->senderId() == localUser()->id()) {
|
|
redactEventIds.push_back(e->id());
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!redactEventIds.isEmpty()) {
|
|
for (const auto &redactEventId : redactEventIds) {
|
|
redactEvent(redactEventId);
|
|
}
|
|
} else {
|
|
postReaction(eventId, reaction);
|
|
}
|
|
}
|
|
|
|
bool NeoChatRoom::containsUser(const QString &userID) const
|
|
{
|
|
auto u = Room::user(userID);
|
|
|
|
if (!u) {
|
|
return false;
|
|
}
|
|
|
|
return Room::memberJoinState(u) != JoinState::Leave;
|
|
}
|
|
|
|
bool NeoChatRoom::canSendEvent(const QString &eventType) const
|
|
{
|
|
auto plEvent = getCurrentState<RoomPowerLevelsEvent>();
|
|
auto pl = plEvent->powerLevelForEvent(eventType);
|
|
auto currentPl = plEvent->powerLevelForUser(localUser()->id());
|
|
|
|
return currentPl >= pl;
|
|
}
|
|
|
|
bool NeoChatRoom::canSendState(const QString &eventType) const
|
|
{
|
|
auto plEvent = getCurrentState<RoomPowerLevelsEvent>();
|
|
auto pl = plEvent->powerLevelForState(eventType);
|
|
auto currentPl = plEvent->powerLevelForUser(localUser()->id());
|
|
|
|
return currentPl >= pl;
|
|
}
|
|
|
|
bool NeoChatRoom::readMarkerLoaded() const
|
|
{
|
|
const auto it = findInTimeline(readMarkerEventId());
|
|
return it != historyEdge();
|
|
}
|
|
|
|
bool NeoChatRoom::isInvite() const
|
|
{
|
|
return joinState() == JoinState::Invite;
|
|
}
|
|
|
|
bool NeoChatRoom::isUserBanned(const QString &user) const
|
|
{
|
|
return getCurrentState<RoomMemberEvent>(user)->membership() == MembershipType::Ban;
|
|
}
|
|
|
|
QString NeoChatRoom::htmlSafeName() const
|
|
{
|
|
return name().toHtmlEscaped();
|
|
}
|
|
|
|
QString NeoChatRoom::htmlSafeDisplayName() const
|
|
{
|
|
return displayName().toHtmlEscaped();
|
|
}
|
|
|
|
void NeoChatRoom::deleteMessagesByUser(const QString &user)
|
|
{
|
|
doDeleteMessagesByUser(user);
|
|
}
|
|
|
|
QString NeoChatRoom::joinRule() const
|
|
{
|
|
return getCurrentState<JoinRulesEvent>()->joinRule();
|
|
}
|
|
|
|
QCoro::Task<void> NeoChatRoom::doDeleteMessagesByUser(const QString &user)
|
|
{
|
|
QStringList events;
|
|
for (const auto &event : messageEvents()) {
|
|
if (event->senderId() == user && !event->isRedacted() && !event.viewAs<RedactionEvent>() && !event->isStateEvent()) {
|
|
events += event->id();
|
|
}
|
|
}
|
|
for (const auto &e : events) {
|
|
auto job = connection()->callApi<RedactEventJob>(id(), QUrl::toPercentEncoding(e), connection()->generateTxnId());
|
|
co_await qCoro(job, &BaseJob::finished);
|
|
if (job->error() != BaseJob::Success) {
|
|
qWarning() << "Error: \"" << job->error() << "\" while deleting messages. Aborting";
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void NeoChatRoom::clearInvitationNotification()
|
|
{
|
|
NotificationsManager::instance().clearInvitationNotification(id());
|
|
}
|
|
|
|
bool NeoChatRoom::isSpace()
|
|
{
|
|
const auto creationEvent = this->creation();
|
|
if (!creationEvent) {
|
|
return false;
|
|
}
|
|
|
|
#ifdef QUOTIENT_07
|
|
return creationEvent->roomType() == RoomType::Space;
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
void NeoChatRoom::setPushNotificationState(PushNotificationState::State state)
|
|
{
|
|
// The caller should never try to set the state to unknown.
|
|
// It exists only as a default state to diable the settings options until the actual state is retrieved from the server.
|
|
if (state == PushNotificationState::Unknown) {
|
|
Q_ASSERT(false);
|
|
return;
|
|
}
|
|
|
|
/**
|
|
* This stops updatePushNotificationState from temporarily changing
|
|
* m_pushNotificationStateUpdating to default after the exisitng rules are deleted but
|
|
* before a new rule is added.
|
|
* The value is set to false after the rule enable job is successful.
|
|
*/
|
|
m_pushNotificationStateUpdating = true;
|
|
|
|
/**
|
|
* First remove any exisiting room rules of the wrong type.
|
|
* Note to prevent race conditions any rule that is going ot be overridden later is not removed.
|
|
* If the default push notification state is chosen any exisiting rule needs to be removed.
|
|
*/
|
|
QJsonObject accountData = connection()->accountDataJson("m.push_rules");
|
|
|
|
// For default and mute check for a room rule and remove if found.
|
|
if (state == PushNotificationState::Default || state == PushNotificationState::Mute) {
|
|
QJsonArray roomRuleArray = accountData["global"].toObject()["room"].toArray();
|
|
for (const auto &i : roomRuleArray) {
|
|
QJsonObject roomRule = i.toObject();
|
|
if (roomRule["rule_id"] == id()) {
|
|
Controller::instance().activeConnection()->callApi<DeletePushRuleJob>("global", "room", id());
|
|
}
|
|
}
|
|
}
|
|
|
|
// For default, all and @mentions and keywords check for an override rule and remove if found.
|
|
if (state == PushNotificationState::Default || state == PushNotificationState::All || state == PushNotificationState::MentionKeyword) {
|
|
QJsonArray overrideRuleArray = accountData["global"].toObject()["override"].toArray();
|
|
for (const auto &i : overrideRuleArray) {
|
|
QJsonObject overrideRule = i.toObject();
|
|
if (overrideRule["rule_id"] == id()) {
|
|
Controller::instance().activeConnection()->callApi<DeletePushRuleJob>("global", "override", id());
|
|
}
|
|
}
|
|
}
|
|
|
|
if (state == PushNotificationState::Mute) {
|
|
/**
|
|
* To mute a room an override rule with "don't notify is set".
|
|
*
|
|
* Setup the rule action to "don't notify" to stop all room notifications
|
|
* see https://spec.matrix.org/v1.3/client-server-api/#actions
|
|
*
|
|
* "actions": [
|
|
* "don't_notify"
|
|
* ]
|
|
*/
|
|
const QVector<QVariant> actions = {"dont_notify"};
|
|
/**
|
|
* Setup the push condition to get all events for the current room
|
|
* see https://spec.matrix.org/v1.3/client-server-api/#conditions-1
|
|
*
|
|
* "conditions": [
|
|
* {
|
|
* "key": "type",
|
|
* "kind": "event_match",
|
|
* "pattern": "room_id"
|
|
* }
|
|
* ]
|
|
*/
|
|
PushCondition pushCondition;
|
|
pushCondition.kind = "event_match";
|
|
pushCondition.key = "room_id";
|
|
pushCondition.pattern = id();
|
|
const QVector<PushCondition> conditions = {pushCondition};
|
|
|
|
// Add new override rule and make sure it's enabled
|
|
auto job = Controller::instance().activeConnection()->callApi<SetPushRuleJob>("global", "override", id(), actions, "", "", conditions, "");
|
|
connect(job, &BaseJob::success, this, [this]() {
|
|
auto enableJob = Controller::instance().activeConnection()->callApi<SetPushRuleEnabledJob>("global", "override", id(), true);
|
|
connect(enableJob, &BaseJob::success, this, [this]() {
|
|
m_pushNotificationStateUpdating = false;
|
|
});
|
|
});
|
|
} else if (state == PushNotificationState::MentionKeyword) {
|
|
/**
|
|
* To only get notifcations for @ mentions and keywords a room rule with "don't_notify" is set.
|
|
*
|
|
* Note - This works becuase a default override rule which catches all user mentions will
|
|
* take precedent and notify. See https://spec.matrix.org/v1.3/client-server-api/#default-override-rules. Any keywords will also have a similar override
|
|
* rule.
|
|
*
|
|
* Setup the rule action to "don't notify" to stop all room event notifications
|
|
* see https://spec.matrix.org/v1.3/client-server-api/#actions
|
|
*
|
|
* "actions": [
|
|
* "don't_notify"
|
|
* ]
|
|
*/
|
|
const QVector<QVariant> actions = {"dont_notify"};
|
|
// No conditions for a room rule
|
|
const QVector<PushCondition> conditions;
|
|
|
|
auto setJob = Controller::instance().activeConnection()->callApi<SetPushRuleJob>("global", "room", id(), actions, "", "", conditions, "");
|
|
connect(setJob, &BaseJob::success, this, [this]() {
|
|
auto enableJob = Controller::instance().activeConnection()->callApi<SetPushRuleEnabledJob>("global", "room", id(), true);
|
|
connect(enableJob, &BaseJob::success, this, [this]() {
|
|
m_pushNotificationStateUpdating = false;
|
|
});
|
|
});
|
|
} else if (state == PushNotificationState::All) {
|
|
/**
|
|
* To send a notification for all room messages a room rule with "notify" is set.
|
|
*
|
|
* Setup the rule action to "notify" so all room events give notifications.
|
|
* Tweeks is also set to follow default sound settings
|
|
* see https://spec.matrix.org/v1.3/client-server-api/#actions
|
|
*
|
|
* "actions": [
|
|
* "notify",
|
|
* {
|
|
* "set_tweek": "sound",
|
|
* "value": "default",
|
|
* }
|
|
* ]
|
|
*/
|
|
QJsonObject tweaks;
|
|
tweaks.insert("set_tweak", "sound");
|
|
tweaks.insert("value", "default");
|
|
const QVector<QVariant> actions = {"notify", tweaks};
|
|
// No conditions for a room rule
|
|
const QVector<PushCondition> conditions;
|
|
|
|
// Add new room rule and make sure enabled
|
|
auto setJob = Controller::instance().activeConnection()->callApi<SetPushRuleJob>("global", "room", id(), actions, "", "", conditions, "");
|
|
connect(setJob, &BaseJob::success, this, [this]() {
|
|
auto enableJob = Controller::instance().activeConnection()->callApi<SetPushRuleEnabledJob>("global", "room", id(), true);
|
|
connect(enableJob, &BaseJob::success, this, [this]() {
|
|
m_pushNotificationStateUpdating = false;
|
|
});
|
|
});
|
|
}
|
|
|
|
m_currentPushNotificationState = state;
|
|
Q_EMIT pushNotificationStateChanged(m_currentPushNotificationState);
|
|
|
|
}
|
|
|
|
void NeoChatRoom::updatePushNotificationState(QString type)
|
|
{
|
|
if (type != "m.push_rules" || m_pushNotificationStateUpdating) {
|
|
return;
|
|
}
|
|
|
|
QJsonObject accountData = connection()->accountDataJson("m.push_rules");
|
|
|
|
// First look for a room rule with the room id
|
|
QJsonArray roomRuleArray = accountData["global"].toObject()["room"].toArray();
|
|
for (const auto &i : roomRuleArray) {
|
|
QJsonObject roomRule = i.toObject();
|
|
if (roomRule["rule_id"] == id()) {
|
|
QString notifyAction = roomRule["actions"].toArray()[0].toString();
|
|
if (notifyAction == "notify") {
|
|
m_currentPushNotificationState = PushNotificationState::All;
|
|
Q_EMIT pushNotificationStateChanged(m_currentPushNotificationState);
|
|
return;
|
|
} else if (notifyAction == "dont_notify") {
|
|
m_currentPushNotificationState = PushNotificationState::MentionKeyword;
|
|
Q_EMIT pushNotificationStateChanged(m_currentPushNotificationState);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Check for an override rule with the room id
|
|
QJsonArray overrideRuleArray = accountData["global"].toObject()["override"].toArray();
|
|
for (const auto &i : overrideRuleArray) {
|
|
QJsonObject overrideRule = i.toObject();
|
|
if (overrideRule["rule_id"] == id()) {
|
|
QString notifyAction = overrideRule["actions"].toArray()[0].toString();
|
|
if (notifyAction == "dont_notify") {
|
|
m_currentPushNotificationState = PushNotificationState::Mute;
|
|
Q_EMIT pushNotificationStateChanged(m_currentPushNotificationState);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// If neither a room or override rule exist for the room then the setting must be default
|
|
m_currentPushNotificationState = PushNotificationState::Default;
|
|
Q_EMIT pushNotificationStateChanged(m_currentPushNotificationState);
|
|
}
|
|
|
|
void NeoChatRoom::reportEvent(const QString &eventId, const QString &reason)
|
|
{
|
|
auto job = connection()->callApi<ReportContentJob>(id(), eventId, -50, reason);
|
|
connect(job, &BaseJob::finished, this, [this, job]() {
|
|
if (job->error() == BaseJob::Success) {
|
|
Q_EMIT showMessage(Positive, i18n("Report sent successfully."));
|
|
Q_EMIT showMessage(MessageType::Positive, i18n("Report sent successfully."));
|
|
}
|
|
});
|
|
}
|
|
|
|
QString NeoChatRoom::chatBoxText() const
|
|
{
|
|
return m_chatBoxText;
|
|
}
|
|
|
|
void NeoChatRoom::setChatBoxText(const QString &text)
|
|
{
|
|
m_chatBoxText = text;
|
|
Q_EMIT chatBoxTextChanged();
|
|
}
|
|
|
|
QString NeoChatRoom::chatBoxReplyId() const
|
|
{
|
|
return m_chatBoxReplyId;
|
|
}
|
|
|
|
void NeoChatRoom::setChatBoxReplyId(const QString &replyId)
|
|
{
|
|
m_chatBoxReplyId = replyId;
|
|
Q_EMIT chatBoxReplyIdChanged();
|
|
}
|
|
|
|
QString NeoChatRoom::chatBoxEditId() const
|
|
{
|
|
return m_chatBoxEditId;
|
|
}
|
|
|
|
void NeoChatRoom::setChatBoxEditId(const QString &editId)
|
|
{
|
|
m_chatBoxEditId = editId;
|
|
Q_EMIT chatBoxEditIdChanged();
|
|
}
|
|
|
|
NeoChatUser *NeoChatRoom::chatBoxReplyUser() const
|
|
{
|
|
if (m_chatBoxReplyId.isEmpty()) {
|
|
return nullptr;
|
|
}
|
|
return static_cast<NeoChatUser *>(user((*findInTimeline(m_chatBoxReplyId))->senderId()));
|
|
}
|
|
|
|
QString NeoChatRoom::chatBoxReplyMessage() const
|
|
{
|
|
if (m_chatBoxReplyId.isEmpty()) {
|
|
return {};
|
|
}
|
|
return eventToString(*static_cast<const RoomMessageEvent *>(&**findInTimeline(m_chatBoxReplyId)));
|
|
}
|
|
|
|
NeoChatUser *NeoChatRoom::chatBoxEditUser() const
|
|
{
|
|
if (m_chatBoxEditId.isEmpty()) {
|
|
return nullptr;
|
|
}
|
|
return static_cast<NeoChatUser *>(user((*findInTimeline(m_chatBoxEditId))->senderId()));
|
|
}
|
|
|
|
QString NeoChatRoom::chatBoxEditMessage() const
|
|
{
|
|
if (m_chatBoxEditId.isEmpty()) {
|
|
return {};
|
|
}
|
|
return eventToString(*static_cast<const RoomMessageEvent *>(&**findInTimeline(m_chatBoxEditId)));
|
|
}
|
|
|
|
QString NeoChatRoom::chatBoxAttachmentPath() const
|
|
{
|
|
return m_chatBoxAttachmentPath;
|
|
}
|
|
|
|
void NeoChatRoom::setChatBoxAttachmentPath(const QString &attachmentPath)
|
|
{
|
|
m_chatBoxAttachmentPath = attachmentPath;
|
|
Q_EMIT chatBoxAttachmentPathChanged();
|
|
}
|
|
|
|
QVector<Mention> *NeoChatRoom::mentions()
|
|
{
|
|
return &m_mentions;
|
|
}
|
|
|
|
QString NeoChatRoom::savedText() const
|
|
{
|
|
return m_savedText;
|
|
}
|
|
|
|
void NeoChatRoom::setSavedText(const QString &savedText)
|
|
{
|
|
m_savedText = savedText;
|
|
}
|
|
|
|
bool NeoChatRoom::canEncryptRoom() const
|
|
{
|
|
#ifdef QUOTIENT_07
|
|
#ifdef Quotient_E2EE_ENABLED
|
|
return !usesEncryption() && canSendState("m.room.encryption");
|
|
#endif
|
|
#endif
|
|
return false;
|
|
}
|