2021-03-05 02:35:15 +03:00
|
|
|
// SPDX-FileCopyrightText: 2021 Nheko Contributors
|
2022-01-01 06:57:53 +03:00
|
|
|
// SPDX-FileCopyrightText: 2022 Nheko Contributors
|
2021-03-05 02:35:15 +03:00
|
|
|
//
|
|
|
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
|
2021-02-10 18:52:42 +03:00
|
|
|
#include "RoomSettings.h"
|
|
|
|
|
2021-02-13 21:29:42 +03:00
|
|
|
#include <QApplication>
|
2021-02-11 21:09:11 +03:00
|
|
|
#include <QFileDialog>
|
2021-02-13 21:29:42 +03:00
|
|
|
#include <QHBoxLayout>
|
2021-02-11 21:09:11 +03:00
|
|
|
#include <QImageReader>
|
|
|
|
#include <QMimeDatabase>
|
|
|
|
#include <QStandardPaths>
|
2021-02-12 10:18:12 +03:00
|
|
|
#include <QVBoxLayout>
|
2022-01-11 09:38:27 +03:00
|
|
|
#include <algorithm>
|
2021-02-10 18:52:42 +03:00
|
|
|
#include <mtx/responses/common.hpp>
|
|
|
|
#include <mtx/responses/media.hpp>
|
|
|
|
|
|
|
|
#include "Cache.h"
|
2022-01-11 09:38:27 +03:00
|
|
|
#include "Cache_p.h"
|
2021-02-12 10:18:12 +03:00
|
|
|
#include "Config.h"
|
2021-02-13 21:29:42 +03:00
|
|
|
#include "Logging.h"
|
2021-02-11 17:24:09 +03:00
|
|
|
#include "MatrixClient.h"
|
|
|
|
#include "Utils.h"
|
2022-01-11 09:38:27 +03:00
|
|
|
#include "mtx/events/event_type.hpp"
|
|
|
|
#include "mtx/events/nheko_extensions/hidden_events.hpp"
|
|
|
|
#include "mtxclient/http/client.hpp"
|
2021-02-13 21:29:42 +03:00
|
|
|
#include "ui/TextField.h"
|
2021-02-11 17:24:09 +03:00
|
|
|
|
|
|
|
using namespace mtx::events;
|
2021-02-10 18:52:42 +03:00
|
|
|
|
2021-02-12 10:18:12 +03:00
|
|
|
EditModal::EditModal(const QString &roomId, QWidget *parent)
|
|
|
|
: QWidget(parent)
|
|
|
|
, roomId_{roomId}
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
setAutoFillBackground(true);
|
|
|
|
setAttribute(Qt::WA_DeleteOnClose, true);
|
|
|
|
setWindowFlags(Qt::Tool | Qt::WindowStaysOnTopHint);
|
|
|
|
setWindowModality(Qt::WindowModal);
|
|
|
|
|
|
|
|
QFont largeFont;
|
|
|
|
largeFont.setPointSizeF(largeFont.pointSizeF() * 1.4);
|
|
|
|
setMinimumWidth(conf::window::minModalWidth);
|
|
|
|
setSizePolicy(QSizePolicy::Maximum, QSizePolicy::Maximum);
|
|
|
|
|
|
|
|
auto layout = new QVBoxLayout(this);
|
|
|
|
|
|
|
|
applyBtn_ = new QPushButton(tr("Apply"), this);
|
|
|
|
cancelBtn_ = new QPushButton(tr("Cancel"), this);
|
|
|
|
cancelBtn_->setDefault(true);
|
|
|
|
|
|
|
|
auto btnLayout = new QHBoxLayout;
|
|
|
|
btnLayout->addStretch(1);
|
|
|
|
btnLayout->setSpacing(15);
|
|
|
|
btnLayout->addWidget(cancelBtn_);
|
|
|
|
btnLayout->addWidget(applyBtn_);
|
|
|
|
|
|
|
|
nameInput_ = new TextField(this);
|
|
|
|
nameInput_->setLabel(tr("Name").toUpper());
|
|
|
|
topicInput_ = new TextField(this);
|
|
|
|
topicInput_->setLabel(tr("Topic").toUpper());
|
|
|
|
|
|
|
|
errorField_ = new QLabel(this);
|
|
|
|
errorField_->setWordWrap(true);
|
|
|
|
errorField_->hide();
|
|
|
|
|
|
|
|
layout->addWidget(nameInput_);
|
|
|
|
layout->addWidget(topicInput_);
|
|
|
|
layout->addLayout(btnLayout, 1);
|
|
|
|
|
|
|
|
auto labelLayout = new QHBoxLayout;
|
|
|
|
labelLayout->setAlignment(Qt::AlignHCenter);
|
|
|
|
labelLayout->addWidget(errorField_);
|
|
|
|
layout->addLayout(labelLayout);
|
|
|
|
|
|
|
|
connect(applyBtn_, &QPushButton::clicked, this, &EditModal::applyClicked);
|
|
|
|
connect(cancelBtn_, &QPushButton::clicked, this, &EditModal::close);
|
|
|
|
|
|
|
|
auto window = QApplication::activeWindow();
|
|
|
|
|
|
|
|
if (window != nullptr) {
|
|
|
|
auto center = window->frameGeometry().center();
|
|
|
|
move(center.x() - (width() * 0.5), center.y() - (height() * 0.5));
|
|
|
|
}
|
2021-02-12 10:18:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2021-02-13 16:38:52 +03:00
|
|
|
EditModal::topicEventSent(const QString &topic)
|
2021-02-12 10:18:12 +03:00
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
errorField_->hide();
|
|
|
|
emit topicChanged(topic);
|
|
|
|
close();
|
2021-02-12 10:18:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
EditModal::nameEventSent(const QString &name)
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
errorField_->hide();
|
|
|
|
emit nameChanged(name);
|
|
|
|
close();
|
2021-02-12 10:18:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
EditModal::error(const QString &msg)
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
errorField_->setText(msg);
|
|
|
|
errorField_->show();
|
2021-02-12 10:18:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
EditModal::applyClicked()
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
// Check if the values are changed from the originals.
|
|
|
|
auto newName = nameInput_->text().trimmed();
|
|
|
|
auto newTopic = topicInput_->text().trimmed();
|
2021-02-12 10:18:12 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
errorField_->hide();
|
2021-02-12 10:18:12 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
if (newName == initialName_ && newTopic == initialTopic_) {
|
|
|
|
close();
|
|
|
|
return;
|
|
|
|
}
|
2021-02-12 10:18:12 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
using namespace mtx::events;
|
|
|
|
auto proxy = std::make_shared<ThreadProxy>();
|
|
|
|
connect(proxy.get(), &ThreadProxy::topicEventSent, this, &EditModal::topicEventSent);
|
|
|
|
connect(proxy.get(), &ThreadProxy::nameEventSent, this, &EditModal::nameEventSent);
|
|
|
|
connect(proxy.get(), &ThreadProxy::error, this, &EditModal::error);
|
2021-02-12 10:18:12 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
if (newName != initialName_ && !newName.isEmpty()) {
|
|
|
|
state::Name body;
|
|
|
|
body.name = newName.toStdString();
|
|
|
|
|
|
|
|
http::client()->send_state_event(
|
|
|
|
roomId_.toStdString(),
|
|
|
|
body,
|
|
|
|
[proxy, newName](const mtx::responses::EventId &, mtx::http::RequestErr err) {
|
|
|
|
if (err) {
|
|
|
|
emit proxy->error(QString::fromStdString(err->matrix_error.error));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
emit proxy->nameEventSent(newName);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (newTopic != initialTopic_ && !newTopic.isEmpty()) {
|
|
|
|
state::Topic body;
|
|
|
|
body.topic = newTopic.toStdString();
|
|
|
|
|
|
|
|
http::client()->send_state_event(
|
|
|
|
roomId_.toStdString(),
|
|
|
|
body,
|
|
|
|
[proxy, newTopic](const mtx::responses::EventId &, mtx::http::RequestErr err) {
|
|
|
|
if (err) {
|
|
|
|
emit proxy->error(QString::fromStdString(err->matrix_error.error));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
emit proxy->topicEventSent(newTopic);
|
|
|
|
});
|
|
|
|
}
|
2021-02-12 10:18:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
EditModal::setFields(const QString &roomName, const QString &roomTopic)
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
initialName_ = roomName;
|
|
|
|
initialTopic_ = roomTopic;
|
2021-02-12 10:18:12 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
nameInput_->setText(roomName);
|
|
|
|
topicInput_->setText(roomTopic);
|
2021-02-12 10:18:12 +03:00
|
|
|
}
|
|
|
|
|
2021-02-10 18:52:42 +03:00
|
|
|
RoomSettings::RoomSettings(QString roomid, QObject *parent)
|
2021-02-14 08:56:10 +03:00
|
|
|
: QObject(parent)
|
|
|
|
, roomid_{std::move(roomid)}
|
2021-02-10 18:52:42 +03:00
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
retrieveRoomInfo();
|
|
|
|
|
|
|
|
// get room setting notifications
|
|
|
|
http::client()->get_pushrules(
|
|
|
|
"global",
|
|
|
|
"override",
|
|
|
|
roomid_.toStdString(),
|
|
|
|
[this](const mtx::pushrules::PushRule &rule, mtx::http::RequestErr &err) {
|
|
|
|
if (err) {
|
|
|
|
if (err->status_code == 404)
|
|
|
|
http::client()->get_pushrules(
|
|
|
|
"global",
|
|
|
|
"room",
|
|
|
|
roomid_.toStdString(),
|
|
|
|
[this](const mtx::pushrules::PushRule &rule, mtx::http::RequestErr &err) {
|
|
|
|
if (err) {
|
|
|
|
notifications_ = 2; // all messages
|
|
|
|
emit notificationsChanged();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rule.enabled) {
|
|
|
|
notifications_ = 1; // mentions only
|
|
|
|
emit notificationsChanged();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rule.enabled) {
|
|
|
|
notifications_ = 0; // muted
|
|
|
|
emit notificationsChanged();
|
|
|
|
} else {
|
|
|
|
notifications_ = 2; // all messages
|
|
|
|
emit notificationsChanged();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
// access rules
|
|
|
|
if (info_.join_rule == state::JoinRule::Public) {
|
|
|
|
if (info_.guest_access) {
|
|
|
|
accessRules_ = 0;
|
|
|
|
} else {
|
|
|
|
accessRules_ = 1;
|
2021-02-11 17:24:09 +03:00
|
|
|
}
|
2021-09-18 01:22:33 +03:00
|
|
|
} else if (info_.join_rule == state::JoinRule::Invite) {
|
|
|
|
accessRules_ = 2;
|
|
|
|
} else if (info_.join_rule == state::JoinRule::Knock) {
|
|
|
|
accessRules_ = 3;
|
|
|
|
} else if (info_.join_rule == state::JoinRule::Restricted) {
|
|
|
|
accessRules_ = 4;
|
|
|
|
}
|
|
|
|
emit accessJoinRulesChanged();
|
2022-01-11 09:38:27 +03:00
|
|
|
|
|
|
|
// Get room's hidden events and store it in member variable.
|
|
|
|
using mtx::events::EventType;
|
|
|
|
if (auto hiddenEvents =
|
|
|
|
cache::client()->getAccountData(EventType::NhekoHiddenEvents, roomid_.toStdString())) {
|
|
|
|
if (auto tmp = std::get_if<mtx::events::EphemeralEvent<
|
|
|
|
mtx::events::account_data::nheko_extensions::HiddenEvents>>(&*hiddenEvents)) {
|
|
|
|
const auto &types = tmp->content.hidden_event_types;
|
|
|
|
auto is_hidden{[&types](EventType searchFor) {
|
|
|
|
return std::find_if(types.begin(), types.end(), [&searchFor](const auto curType) {
|
|
|
|
return curType == searchFor;
|
|
|
|
}) != types.end();
|
|
|
|
}};
|
|
|
|
|
|
|
|
hiddenEvents_ = {is_hidden(EventType::RoomMember),
|
|
|
|
is_hidden(EventType::RoomPowerLevels),
|
|
|
|
is_hidden(EventType::Sticker)};
|
|
|
|
}
|
|
|
|
}
|
2021-02-10 18:52:42 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QString
|
|
|
|
RoomSettings::roomName() const
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
return utils::replaceEmoji(QString::fromStdString(info_.name).toHtmlEscaped());
|
2021-02-13 16:38:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QString
|
|
|
|
RoomSettings::roomTopic() const
|
|
|
|
{
|
2021-12-29 08:01:38 +03:00
|
|
|
return utils::replaceEmoji(
|
|
|
|
utils::linkifyMessage(QString::fromStdString(info_.topic)
|
|
|
|
.toHtmlEscaped()
|
|
|
|
.replace(QLatin1String("\n"), QLatin1String("<br>"))));
|
2021-02-10 18:52:42 +03:00
|
|
|
}
|
|
|
|
|
2021-02-11 18:53:33 +03:00
|
|
|
QString
|
|
|
|
RoomSettings::roomId() const
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
return roomid_;
|
2021-02-11 18:53:33 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QString
|
|
|
|
RoomSettings::roomVersion() const
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
return QString::fromStdString(info_.version);
|
2021-02-11 18:53:33 +03:00
|
|
|
}
|
|
|
|
|
2021-02-11 21:09:11 +03:00
|
|
|
bool
|
|
|
|
RoomSettings::isLoading() const
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
return isLoading_;
|
2021-02-11 21:09:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
QString
|
|
|
|
RoomSettings::roomAvatarUrl()
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
return QString::fromStdString(info_.avatar_url);
|
2021-02-11 21:09:11 +03:00
|
|
|
}
|
|
|
|
|
2021-02-11 18:53:33 +03:00
|
|
|
int
|
|
|
|
RoomSettings::memberCount() const
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
return info_.member_count;
|
2021-02-11 18:53:33 +03:00
|
|
|
}
|
|
|
|
|
2021-02-10 18:52:42 +03:00
|
|
|
void
|
|
|
|
RoomSettings::retrieveRoomInfo()
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
try {
|
|
|
|
usesEncryption_ = cache::isRoomEncrypted(roomid_.toStdString());
|
|
|
|
info_ = cache::singleRoomInfo(roomid_.toStdString());
|
|
|
|
} catch (const lmdb::error &) {
|
|
|
|
nhlog::db()->warn("failed to retrieve room info from cache: {}", roomid_.toStdString());
|
|
|
|
}
|
2021-02-11 17:24:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
RoomSettings::notifications()
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
return notifications_;
|
2021-02-11 17:24:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
RoomSettings::accessJoinRules()
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
return accessRules_;
|
2021-02-11 17:24:09 +03:00
|
|
|
}
|
|
|
|
|
2022-01-11 09:38:27 +03:00
|
|
|
bool
|
|
|
|
RoomSettings::eventHidden(int index)
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
// Is empty if there are no preferences stored for this room.
|
|
|
|
if (!hiddenEvents_.empty()) {
|
|
|
|
return hiddenEvents_.at(index);
|
|
|
|
}
|
|
|
|
} catch (...) {
|
|
|
|
nhlog::db()->warn("Failed to retrieve hidden event setting at {}", index);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-02-11 17:24:09 +03:00
|
|
|
void
|
|
|
|
RoomSettings::enableEncryption()
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
if (usesEncryption_)
|
|
|
|
return;
|
|
|
|
|
|
|
|
const auto room_id = roomid_.toStdString();
|
|
|
|
http::client()->enable_encryption(
|
|
|
|
room_id, [room_id, this](const mtx::responses::EventId &, mtx::http::RequestErr err) {
|
|
|
|
if (err) {
|
|
|
|
int status_code = static_cast<int>(err->status_code);
|
|
|
|
nhlog::net()->warn("failed to enable encryption in room ({}): {} {}",
|
|
|
|
room_id,
|
|
|
|
err->matrix_error.error,
|
|
|
|
status_code);
|
|
|
|
emit displayError(tr("Failed to enable encryption: %1")
|
|
|
|
.arg(QString::fromStdString(err->matrix_error.error)));
|
|
|
|
usesEncryption_ = false;
|
|
|
|
emit encryptionChanged();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
nhlog::net()->info("enabled encryption on room ({})", room_id);
|
|
|
|
});
|
|
|
|
|
|
|
|
usesEncryption_ = true;
|
|
|
|
emit encryptionChanged();
|
2021-02-11 17:24:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
RoomSettings::canChangeJoinRules() const
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
try {
|
|
|
|
return cache::hasEnoughPowerLevel(
|
|
|
|
{EventType::RoomJoinRules}, roomid_.toStdString(), utils::localUser().toStdString());
|
|
|
|
} catch (const lmdb::error &e) {
|
|
|
|
nhlog::db()->warn("lmdb error: {}", e.what());
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2021-02-11 17:24:09 +03:00
|
|
|
}
|
|
|
|
|
2021-02-11 21:09:11 +03:00
|
|
|
bool
|
|
|
|
RoomSettings::canChangeNameAndTopic() const
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
try {
|
|
|
|
return cache::hasEnoughPowerLevel({EventType::RoomName, EventType::RoomTopic},
|
|
|
|
roomid_.toStdString(),
|
|
|
|
utils::localUser().toStdString());
|
|
|
|
} catch (const lmdb::error &e) {
|
|
|
|
nhlog::db()->warn("lmdb error: {}", e.what());
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2021-02-11 21:09:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
RoomSettings::canChangeAvatar() const
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
try {
|
|
|
|
return cache::hasEnoughPowerLevel(
|
|
|
|
{EventType::RoomAvatar}, roomid_.toStdString(), utils::localUser().toStdString());
|
|
|
|
} catch (const lmdb::error &e) {
|
|
|
|
nhlog::db()->warn("lmdb error: {}", e.what());
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2021-02-11 21:09:11 +03:00
|
|
|
}
|
|
|
|
|
2021-02-11 17:24:09 +03:00
|
|
|
bool
|
|
|
|
RoomSettings::isEncryptionEnabled() const
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
return usesEncryption_;
|
2021-02-11 17:24:09 +03:00
|
|
|
}
|
|
|
|
|
2021-08-18 00:31:25 +03:00
|
|
|
bool
|
|
|
|
RoomSettings::supportsKnocking() const
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
return info_.version != "" && info_.version != "1" && info_.version != "2" &&
|
|
|
|
info_.version != "3" && info_.version != "4" && info_.version != "5" &&
|
|
|
|
info_.version != "6";
|
2021-08-18 00:31:25 +03:00
|
|
|
}
|
|
|
|
bool
|
|
|
|
RoomSettings::supportsRestricted() const
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
return info_.version != "" && info_.version != "1" && info_.version != "2" &&
|
|
|
|
info_.version != "3" && info_.version != "4" && info_.version != "5" &&
|
|
|
|
info_.version != "6" && info_.version != "7";
|
2021-08-18 00:31:25 +03:00
|
|
|
}
|
|
|
|
|
2021-02-12 10:18:12 +03:00
|
|
|
void
|
|
|
|
RoomSettings::openEditModal()
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
retrieveRoomInfo();
|
|
|
|
|
|
|
|
auto modal = new EditModal(roomid_);
|
|
|
|
modal->setFields(QString::fromStdString(info_.name), QString::fromStdString(info_.topic));
|
|
|
|
modal->raise();
|
|
|
|
modal->show();
|
|
|
|
connect(modal, &EditModal::nameChanged, this, [this](const QString &newName) {
|
|
|
|
info_.name = newName.toStdString();
|
|
|
|
emit roomNameChanged();
|
|
|
|
});
|
|
|
|
|
|
|
|
connect(modal, &EditModal::topicChanged, this, [this](const QString &newTopic) {
|
|
|
|
info_.topic = newTopic.toStdString();
|
|
|
|
emit roomTopicChanged();
|
|
|
|
});
|
2021-02-12 10:18:12 +03:00
|
|
|
}
|
|
|
|
|
2022-01-11 09:38:27 +03:00
|
|
|
void
|
|
|
|
RoomSettings::saveHiddenEventsSettings(const bool toggleRoomMember,
|
|
|
|
const bool toggleRoomPowerLevels,
|
|
|
|
const bool toggleSticker)
|
|
|
|
{
|
|
|
|
const auto roomid = roomid_.toStdString();
|
|
|
|
nhlog::ui()->debug("Setting events to hidden in room {}: m.room.member={}, "
|
|
|
|
"m.room.power_levels={}, m.sticker={}",
|
|
|
|
roomid,
|
|
|
|
toggleRoomMember,
|
|
|
|
toggleRoomPowerLevels,
|
|
|
|
toggleSticker);
|
|
|
|
|
|
|
|
mtx::events::account_data::nheko_extensions::HiddenEvents hiddenEvents;
|
|
|
|
hiddenEvents.hidden_event_types = {
|
|
|
|
EventType::Reaction, EventType::CallCandidates, EventType::Unsupported};
|
|
|
|
if (toggleRoomMember) {
|
|
|
|
hiddenEvents.hidden_event_types.emplace_back(mtx::events::EventType::RoomMember);
|
|
|
|
}
|
|
|
|
if (toggleRoomPowerLevels) {
|
|
|
|
hiddenEvents.hidden_event_types.emplace_back(mtx::events::EventType::RoomPowerLevels);
|
|
|
|
}
|
|
|
|
if (toggleSticker) {
|
|
|
|
hiddenEvents.hidden_event_types.emplace_back(mtx::events::EventType::Sticker);
|
|
|
|
}
|
|
|
|
http::client()->put_room_account_data(roomid, hiddenEvents, [&roomid](mtx::http::RequestErr e) {
|
|
|
|
if (e) {
|
|
|
|
nhlog::net()->error("Failed to update room account data in {}: {}", roomid, *e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-02-11 17:24:09 +03:00
|
|
|
void
|
|
|
|
RoomSettings::changeNotifications(int currentIndex)
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
notifications_ = currentIndex;
|
|
|
|
|
|
|
|
std::string room_id = roomid_.toStdString();
|
|
|
|
if (notifications_ == 0) {
|
|
|
|
// mute room
|
|
|
|
// delete old rule first, then add new rule
|
|
|
|
mtx::pushrules::PushRule rule;
|
|
|
|
rule.actions = {mtx::pushrules::actions::dont_notify{}};
|
|
|
|
mtx::pushrules::PushCondition condition;
|
|
|
|
condition.kind = "event_match";
|
|
|
|
condition.key = "room_id";
|
|
|
|
condition.pattern = room_id;
|
|
|
|
rule.conditions = {condition};
|
|
|
|
|
|
|
|
http::client()->put_pushrules(
|
|
|
|
"global", "override", room_id, rule, [room_id](mtx::http::RequestErr &err) {
|
|
|
|
if (err)
|
|
|
|
nhlog::net()->error("failed to set pushrule for room {}: {} {}",
|
|
|
|
room_id,
|
|
|
|
static_cast<int>(err->status_code),
|
|
|
|
err->matrix_error.error);
|
|
|
|
http::client()->delete_pushrules(
|
|
|
|
"global", "room", room_id, [room_id](mtx::http::RequestErr &) {});
|
|
|
|
});
|
|
|
|
} else if (notifications_ == 1) {
|
|
|
|
// mentions only
|
|
|
|
// delete old rule first, then add new rule
|
|
|
|
mtx::pushrules::PushRule rule;
|
|
|
|
rule.actions = {mtx::pushrules::actions::dont_notify{}};
|
|
|
|
http::client()->put_pushrules(
|
|
|
|
"global", "room", room_id, rule, [room_id](mtx::http::RequestErr &err) {
|
|
|
|
if (err)
|
|
|
|
nhlog::net()->error("failed to set pushrule for room {}: {} {}",
|
|
|
|
room_id,
|
|
|
|
static_cast<int>(err->status_code),
|
|
|
|
err->matrix_error.error);
|
|
|
|
http::client()->delete_pushrules(
|
|
|
|
"global", "override", room_id, [room_id](mtx::http::RequestErr &) {});
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
// all messages
|
|
|
|
http::client()->delete_pushrules(
|
|
|
|
"global", "override", room_id, [room_id](mtx::http::RequestErr &) {
|
|
|
|
http::client()->delete_pushrules(
|
|
|
|
"global", "room", room_id, [room_id](mtx::http::RequestErr &) {});
|
|
|
|
});
|
|
|
|
}
|
2021-02-11 17:24:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
RoomSettings::changeAccessRules(int index)
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
using namespace mtx::events::state;
|
|
|
|
|
|
|
|
auto guest_access = [](int index) -> state::GuestAccess {
|
|
|
|
state::GuestAccess event;
|
|
|
|
|
|
|
|
if (index == 0)
|
|
|
|
event.guest_access = state::AccessState::CanJoin;
|
|
|
|
else
|
|
|
|
event.guest_access = state::AccessState::Forbidden;
|
|
|
|
|
|
|
|
return event;
|
|
|
|
}(index);
|
|
|
|
|
|
|
|
auto join_rule = [](int index) -> state::JoinRules {
|
|
|
|
state::JoinRules event;
|
|
|
|
|
|
|
|
switch (index) {
|
|
|
|
case 0:
|
|
|
|
case 1:
|
|
|
|
event.join_rule = state::JoinRule::Public;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
event.join_rule = state::JoinRule::Invite;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
event.join_rule = state::JoinRule::Knock;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
event.join_rule = state::JoinRule::Restricted;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
event.join_rule = state::JoinRule::Invite;
|
|
|
|
}
|
2021-02-11 17:24:09 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
return event;
|
|
|
|
}(index);
|
2021-02-11 17:24:09 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
updateAccessRules(roomid_.toStdString(), join_rule, guest_access);
|
2021-02-11 17:24:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
RoomSettings::updateAccessRules(const std::string &room_id,
|
|
|
|
const mtx::events::state::JoinRules &join_rule,
|
|
|
|
const mtx::events::state::GuestAccess &guest_access)
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
isLoading_ = true;
|
|
|
|
emit loadingChanged();
|
|
|
|
|
|
|
|
http::client()->send_state_event(
|
|
|
|
room_id,
|
|
|
|
join_rule,
|
|
|
|
[this, room_id, guest_access](const mtx::responses::EventId &, mtx::http::RequestErr err) {
|
|
|
|
if (err) {
|
|
|
|
nhlog::net()->warn("failed to send m.room.join_rule: {} {}",
|
|
|
|
static_cast<int>(err->status_code),
|
|
|
|
err->matrix_error.error);
|
|
|
|
emit displayError(QString::fromStdString(err->matrix_error.error));
|
|
|
|
isLoading_ = false;
|
|
|
|
emit loadingChanged();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
http::client()->send_state_event(
|
|
|
|
room_id,
|
|
|
|
guest_access,
|
|
|
|
[this](const mtx::responses::EventId &, mtx::http::RequestErr err) {
|
|
|
|
if (err) {
|
|
|
|
nhlog::net()->warn("failed to send m.room.guest_access: {} {}",
|
|
|
|
static_cast<int>(err->status_code),
|
|
|
|
err->matrix_error.error);
|
|
|
|
emit displayError(QString::fromStdString(err->matrix_error.error));
|
|
|
|
}
|
2021-02-11 17:24:09 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
isLoading_ = false;
|
|
|
|
emit loadingChanged();
|
|
|
|
});
|
|
|
|
});
|
2021-02-11 21:09:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
RoomSettings::stopLoading()
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
isLoading_ = false;
|
|
|
|
emit loadingChanged();
|
2021-02-11 21:09:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
RoomSettings::avatarChanged()
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
retrieveRoomInfo();
|
|
|
|
emit avatarUrlChanged();
|
2021-02-11 21:09:11 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
RoomSettings::updateAvatar()
|
|
|
|
{
|
2021-09-18 01:22:33 +03:00
|
|
|
const QString picturesFolder =
|
|
|
|
QStandardPaths::writableLocation(QStandardPaths::PicturesLocation);
|
|
|
|
const QString fileName = QFileDialog::getOpenFileName(
|
|
|
|
nullptr, tr("Select an avatar"), picturesFolder, tr("All Files (*)"));
|
|
|
|
|
|
|
|
if (fileName.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
QMimeDatabase db;
|
|
|
|
QMimeType mime = db.mimeTypeForFile(fileName, QMimeDatabase::MatchContent);
|
|
|
|
|
2021-12-29 06:28:08 +03:00
|
|
|
const auto format = mime.name().split(QStringLiteral("/"))[0];
|
2021-09-18 01:22:33 +03:00
|
|
|
|
|
|
|
QFile file{fileName, this};
|
2021-12-29 06:28:08 +03:00
|
|
|
if (format != QLatin1String("image")) {
|
2021-09-18 01:22:33 +03:00
|
|
|
emit displayError(tr("The selected file is not an image"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!file.open(QIODevice::ReadOnly)) {
|
|
|
|
emit displayError(tr("Error while reading file: %1").arg(file.errorString()));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
isLoading_ = true;
|
|
|
|
emit loadingChanged();
|
|
|
|
|
|
|
|
// Events emitted from the http callbacks (different threads) will
|
|
|
|
// be queued back into the UI thread through this proxy object.
|
|
|
|
auto proxy = std::make_shared<ThreadProxy>();
|
|
|
|
connect(proxy.get(), &ThreadProxy::error, this, &RoomSettings::displayError);
|
|
|
|
connect(proxy.get(), &ThreadProxy::stopLoading, this, &RoomSettings::stopLoading);
|
|
|
|
|
|
|
|
const auto bin = file.peek(file.size());
|
|
|
|
const auto payload = std::string(bin.data(), bin.size());
|
|
|
|
const auto dimensions = QImageReader(&file).size();
|
|
|
|
|
|
|
|
// First we need to create a new mxc URI
|
|
|
|
// (i.e upload media to the Matrix content repository) for the new avatar.
|
|
|
|
http::client()->upload(
|
|
|
|
payload,
|
|
|
|
mime.name().toStdString(),
|
|
|
|
QFileInfo(fileName).fileName().toStdString(),
|
|
|
|
[proxy = std::move(proxy),
|
|
|
|
dimensions,
|
|
|
|
payload,
|
|
|
|
mimetype = mime.name().toStdString(),
|
|
|
|
size = payload.size(),
|
|
|
|
room_id = roomid_.toStdString(),
|
|
|
|
content = std::move(bin)](const mtx::responses::ContentURI &res, mtx::http::RequestErr err) {
|
|
|
|
if (err) {
|
|
|
|
emit proxy->stopLoading();
|
|
|
|
emit proxy->error(tr("Failed to upload image: %s")
|
|
|
|
.arg(QString::fromStdString(err->matrix_error.error)));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
using namespace mtx::events;
|
|
|
|
state::Avatar avatar_event;
|
|
|
|
avatar_event.image_info.w = dimensions.width();
|
|
|
|
avatar_event.image_info.h = dimensions.height();
|
|
|
|
avatar_event.image_info.mimetype = mimetype;
|
|
|
|
avatar_event.image_info.size = size;
|
|
|
|
avatar_event.url = res.content_uri;
|
|
|
|
|
|
|
|
http::client()->send_state_event(
|
|
|
|
room_id,
|
|
|
|
avatar_event,
|
|
|
|
[content = std::move(content),
|
|
|
|
proxy = std::move(proxy)](const mtx::responses::EventId &, mtx::http::RequestErr err) {
|
|
|
|
if (err) {
|
|
|
|
emit proxy->error(tr("Failed to upload image: %s")
|
2021-02-11 21:09:11 +03:00
|
|
|
.arg(QString::fromStdString(err->matrix_error.error)));
|
2021-09-18 01:22:33 +03:00
|
|
|
return;
|
|
|
|
}
|
2021-02-11 17:24:09 +03:00
|
|
|
|
2021-09-18 01:22:33 +03:00
|
|
|
emit proxy->stopLoading();
|
|
|
|
});
|
|
|
|
});
|
2021-02-22 23:35:11 +03:00
|
|
|
}
|