Merge pull request #1541 from NepNep21/ignore-users
Support (un)ignoring users (#546)
This commit is contained in:
commit
a583de297c
@ -777,6 +777,7 @@ set(QML_SOURCES
|
|||||||
resources/qml/dialogs/AllowedRoomsSettingsDialog.qml
|
resources/qml/dialogs/AllowedRoomsSettingsDialog.qml
|
||||||
resources/qml/dialogs/RoomSettings.qml
|
resources/qml/dialogs/RoomSettings.qml
|
||||||
resources/qml/dialogs/UserProfile.qml
|
resources/qml/dialogs/UserProfile.qml
|
||||||
|
resources/qml/dialogs/IgnoredUsers.qml
|
||||||
resources/qml/emoji/StickerPicker.qml
|
resources/qml/emoji/StickerPicker.qml
|
||||||
resources/qml/pages/LoginPage.qml
|
resources/qml/pages/LoginPage.qml
|
||||||
resources/qml/pages/RegisterPage.qml
|
resources/qml/pages/RegisterPage.qml
|
||||||
|
83
resources/qml/dialogs/IgnoredUsers.qml
Normal file
83
resources/qml/dialogs/IgnoredUsers.qml
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
// SPDX-FileCopyrightText: Nheko Contributors
|
||||||
|
//
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
import QtQml 2.15
|
||||||
|
import QtQuick 2.15
|
||||||
|
import QtQuick.Controls 2.15
|
||||||
|
import QtQuick.Layouts 2.15
|
||||||
|
import QtQuick.Window 2.15
|
||||||
|
import im.nheko 1.0
|
||||||
|
|
||||||
|
Window {
|
||||||
|
id: ignoredUsers
|
||||||
|
|
||||||
|
title: qsTr("Ignored users")
|
||||||
|
flags: Qt.WindowCloseButtonHint | Qt.WindowTitleHint
|
||||||
|
height: 650
|
||||||
|
width: 420
|
||||||
|
minimumHeight: 420
|
||||||
|
color: palette.window
|
||||||
|
|
||||||
|
ListView {
|
||||||
|
id: view
|
||||||
|
anchors.fill: parent
|
||||||
|
spacing: Nheko.paddingMedium
|
||||||
|
footerPositioning: ListView.OverlayFooter
|
||||||
|
|
||||||
|
model: TimelineManager.ignoredUsers
|
||||||
|
header: ColumnLayout {
|
||||||
|
Text {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.maximumWidth: view.width
|
||||||
|
wrapMode: Text.Wrap
|
||||||
|
color: palette.text
|
||||||
|
text: qsTr("Ignoring a user hides their messages (they can still see yours!).")
|
||||||
|
}
|
||||||
|
|
||||||
|
Item { Layout.preferredHeight: Nheko.paddingLarge }
|
||||||
|
}
|
||||||
|
delegate: RowLayout {
|
||||||
|
property var profile: TimelineManager.getGlobalUserProfile(modelData)
|
||||||
|
|
||||||
|
width: view.width
|
||||||
|
|
||||||
|
Avatar {
|
||||||
|
enabled: false
|
||||||
|
displayName: profile.displayName
|
||||||
|
userid: profile.userid
|
||||||
|
url: profile.avatarUrl.replace("mxc://", "image://MxcImage/")
|
||||||
|
}
|
||||||
|
|
||||||
|
Text {
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.alignment: Qt.AlignLeft
|
||||||
|
elide: Text.ElideRight
|
||||||
|
color: palette.text
|
||||||
|
text: modelData
|
||||||
|
}
|
||||||
|
|
||||||
|
ImageButton {
|
||||||
|
Layout.preferredHeight: 24
|
||||||
|
Layout.preferredWidth: 24
|
||||||
|
image: ":/icons/icons/ui/dismiss.svg"
|
||||||
|
hoverEnabled: true
|
||||||
|
ToolTip.visible: hovered
|
||||||
|
ToolTip.text: qsTr("Stop Ignoring.")
|
||||||
|
onClicked: profile.ignored = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
footer: DialogButtonBox {
|
||||||
|
z: 2
|
||||||
|
width: view.width
|
||||||
|
alignment: Qt.AlignRight
|
||||||
|
standardButtons: DialogButtonBox.Ok
|
||||||
|
onAccepted: ignoredUsers.close()
|
||||||
|
|
||||||
|
background: Rectangle {
|
||||||
|
anchors.fill: parent
|
||||||
|
color: palette.window
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -289,6 +289,18 @@ ApplicationWindow {
|
|||||||
visible: !profile.isGlobalUserProfile && profile.room.permissions.canBan()
|
visible: !profile.isGlobalUserProfile && profile.room.permissions.canBan()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ImageButton {
|
||||||
|
Layout.preferredHeight: 24
|
||||||
|
Layout.preferredWidth: 24
|
||||||
|
image: ":/icons/icons/ui/volume-off-indicator.svg"
|
||||||
|
hoverEnabled: true
|
||||||
|
ToolTip.visible: hovered
|
||||||
|
ToolTip.text: profile.ignored ? qsTr("Unignore the user.") : qsTr("Ignore the user.")
|
||||||
|
buttonTextColor: profile.ignored ? Nheko.theme.red : palette.buttonText
|
||||||
|
onClicked: profile.ignored = !profile.ignored
|
||||||
|
visible: !profile.isSelf
|
||||||
|
}
|
||||||
|
|
||||||
ImageButton {
|
ImageButton {
|
||||||
Layout.preferredHeight: 24
|
Layout.preferredHeight: 24
|
||||||
Layout.preferredWidth: 24
|
Layout.preferredWidth: 24
|
||||||
@ -298,7 +310,6 @@ ApplicationWindow {
|
|||||||
ToolTip.text: qsTr("Refresh device list.")
|
ToolTip.text: qsTr("Refresh device list.")
|
||||||
onClicked: profile.refreshDevices()
|
onClicked: profile.refreshDevices()
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TabBar {
|
TabBar {
|
||||||
|
@ -233,6 +233,24 @@ Rectangle {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
DelegateChoice {
|
||||||
|
roleValue: UserSettingsModel.ManageIgnoredUsers
|
||||||
|
Button {
|
||||||
|
text: qsTr("MANAGE")
|
||||||
|
onClicked: {
|
||||||
|
var dialog = ignoredUsersDialog.createObject();
|
||||||
|
dialog.show();
|
||||||
|
destroyOnClose(dialog);
|
||||||
|
}
|
||||||
|
|
||||||
|
Component {
|
||||||
|
id: ignoredUsersDialog
|
||||||
|
|
||||||
|
IgnoredUsers {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
DelegateChoice {
|
DelegateChoice {
|
||||||
Text {
|
Text {
|
||||||
text: model.value
|
text: model.value
|
||||||
|
@ -6,6 +6,9 @@
|
|||||||
#include <QInputDialog>
|
#include <QInputDialog>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
#include <unordered_set>
|
||||||
|
|
||||||
#include <mtx/responses.hpp>
|
#include <mtx/responses.hpp>
|
||||||
|
|
||||||
#include "AvatarProvider.h"
|
#include "AvatarProvider.h"
|
||||||
@ -775,6 +778,23 @@ ChatPage::handleSyncResponse(const mtx::responses::Sync &res, const std::string
|
|||||||
// Ensure that we have enough one-time keys available.
|
// Ensure that we have enough one-time keys available.
|
||||||
ensureOneTimeKeyCount(res.device_one_time_keys_count, res.device_unused_fallback_key_types);
|
ensureOneTimeKeyCount(res.device_one_time_keys_count, res.device_unused_fallback_key_types);
|
||||||
|
|
||||||
|
std::optional<mtx::events::account_data::IgnoredUsers> oldIgnoredUsers;
|
||||||
|
if (auto ignoreEv = std::ranges::find_if(
|
||||||
|
res.account_data.events,
|
||||||
|
[](const mtx::events::collections::RoomAccountDataEvents &e) {
|
||||||
|
return std::holds_alternative<
|
||||||
|
mtx::events::AccountDataEvent<mtx::events::account_data::IgnoredUsers>>(e);
|
||||||
|
});
|
||||||
|
ignoreEv != res.account_data.events.end()) {
|
||||||
|
if (auto oldEv = cache::client()->getAccountData(mtx::events::EventType::IgnoredUsers))
|
||||||
|
oldIgnoredUsers =
|
||||||
|
std::get<mtx::events::AccountDataEvent<mtx::events::account_data::IgnoredUsers>>(
|
||||||
|
*oldEv)
|
||||||
|
.content;
|
||||||
|
else
|
||||||
|
oldIgnoredUsers = mtx::events::account_data::IgnoredUsers{};
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: fine grained error handling
|
// TODO: fine grained error handling
|
||||||
try {
|
try {
|
||||||
cache::client()->saveState(res);
|
cache::client()->saveState(res);
|
||||||
@ -783,6 +803,36 @@ ChatPage::handleSyncResponse(const mtx::responses::Sync &res, const std::string
|
|||||||
auto updates = cache::getRoomInfo(cache::client()->roomsWithStateUpdates(res));
|
auto updates = cache::getRoomInfo(cache::client()->roomsWithStateUpdates(res));
|
||||||
|
|
||||||
emit syncUI(std::move(res));
|
emit syncUI(std::move(res));
|
||||||
|
|
||||||
|
// if the ignored users changed, clear timeline of all affected rooms.
|
||||||
|
if (oldIgnoredUsers) {
|
||||||
|
if (auto newEv =
|
||||||
|
cache::client()->getAccountData(mtx::events::EventType::IgnoredUsers)) {
|
||||||
|
std::vector<mtx::events::account_data::IgnoredUser> changedUsers{};
|
||||||
|
std::ranges::set_symmetric_difference(
|
||||||
|
oldIgnoredUsers->users,
|
||||||
|
std::get<mtx::events::AccountDataEvent<mtx::events::account_data::IgnoredUsers>>(
|
||||||
|
*newEv)
|
||||||
|
.content.users,
|
||||||
|
std::back_inserter(changedUsers),
|
||||||
|
{},
|
||||||
|
&mtx::events::account_data::IgnoredUser::id,
|
||||||
|
&mtx::events::account_data::IgnoredUser::id);
|
||||||
|
|
||||||
|
std::unordered_set<std::string> roomsToReload;
|
||||||
|
for (const auto &user : changedUsers) {
|
||||||
|
auto commonRooms = cache::client()->getCommonRooms(user.id);
|
||||||
|
for (const auto &room : commonRooms)
|
||||||
|
roomsToReload.insert(room.first);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const auto &room : roomsToReload) {
|
||||||
|
if (auto model =
|
||||||
|
view_manager_->rooms()->getRoomById(QString::fromStdString(room)))
|
||||||
|
model->clearTimeline();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
} catch (const lmdb::map_full_error &e) {
|
} catch (const lmdb::map_full_error &e) {
|
||||||
nhlog::db()->error("lmdb is full: {}", e.what());
|
nhlog::db()->error("lmdb is full: {}", e.what());
|
||||||
cache::deleteOldData();
|
cache::deleteOldData();
|
||||||
|
@ -1042,6 +1042,8 @@ UserSettingsModel::data(const QModelIndex &index, int role) const
|
|||||||
return tr("Read receipts");
|
return tr("Read receipts");
|
||||||
case HiddenTimelineEvents:
|
case HiddenTimelineEvents:
|
||||||
return tr("Hidden events");
|
return tr("Hidden events");
|
||||||
|
case IgnoredUsers:
|
||||||
|
return tr("Ignored users");
|
||||||
case DesktopNotifications:
|
case DesktopNotifications:
|
||||||
return tr("Desktop notifications");
|
return tr("Desktop notifications");
|
||||||
case AlertOnNotification:
|
case AlertOnNotification:
|
||||||
@ -1485,6 +1487,8 @@ UserSettingsModel::data(const QModelIndex &index, int role) const
|
|||||||
return tr("Regularly redact expired events as specified in the event expiration "
|
return tr("Regularly redact expired events as specified in the event expiration "
|
||||||
"configuration. Since this is currently not executed server side, you need "
|
"configuration. Since this is currently not executed server side, you need "
|
||||||
"to have one client running this regularly.");
|
"to have one client running this regularly.");
|
||||||
|
case IgnoredUsers:
|
||||||
|
return tr("Manage your ignored users.");
|
||||||
}
|
}
|
||||||
} else if (role == Type) {
|
} else if (role == Type) {
|
||||||
switch (index.row()) {
|
switch (index.row()) {
|
||||||
@ -1571,6 +1575,8 @@ UserSettingsModel::data(const QModelIndex &index, int role) const
|
|||||||
return KeyStatus;
|
return KeyStatus;
|
||||||
case HiddenTimelineEvents:
|
case HiddenTimelineEvents:
|
||||||
return ConfigureHiddenEvents;
|
return ConfigureHiddenEvents;
|
||||||
|
case IgnoredUsers:
|
||||||
|
return ManageIgnoredUsers;
|
||||||
}
|
}
|
||||||
} else if (role == ValueLowerBound) {
|
} else if (role == ValueLowerBound) {
|
||||||
switch (index.row()) {
|
switch (index.row()) {
|
||||||
|
@ -508,6 +508,7 @@ class UserSettingsModel : public QAbstractListModel
|
|||||||
MessageVisibilitySection,
|
MessageVisibilitySection,
|
||||||
ExpireEvents,
|
ExpireEvents,
|
||||||
HiddenTimelineEvents,
|
HiddenTimelineEvents,
|
||||||
|
IgnoredUsers,
|
||||||
|
|
||||||
NotificationsSection,
|
NotificationsSection,
|
||||||
DesktopNotifications,
|
DesktopNotifications,
|
||||||
@ -566,6 +567,7 @@ public:
|
|||||||
SessionKeyImportExport,
|
SessionKeyImportExport,
|
||||||
XSignKeysRequestDownload,
|
XSignKeysRequestDownload,
|
||||||
ConfigureHiddenEvents,
|
ConfigureHiddenEvents,
|
||||||
|
ManageIgnoredUsers,
|
||||||
};
|
};
|
||||||
Q_ENUM(Types);
|
Q_ENUM(Types);
|
||||||
|
|
||||||
|
@ -18,8 +18,6 @@
|
|||||||
#include "CacheStructs.h"
|
#include "CacheStructs.h"
|
||||||
#include "EventStore.h"
|
#include "EventStore.h"
|
||||||
#include "InputBar.h"
|
#include "InputBar.h"
|
||||||
#include "InviteesModel.h"
|
|
||||||
#include "MemberList.h"
|
|
||||||
#include "Permissions.h"
|
#include "Permissions.h"
|
||||||
#include "ReadReceiptsModel.h"
|
#include "ReadReceiptsModel.h"
|
||||||
#include "ui/RoomSummary.h"
|
#include "ui/RoomSummary.h"
|
||||||
|
@ -12,6 +12,7 @@
|
|||||||
#include <QString>
|
#include <QString>
|
||||||
|
|
||||||
#include "Cache.h"
|
#include "Cache.h"
|
||||||
|
#include "Cache_p.h"
|
||||||
#include "ChatPage.h"
|
#include "ChatPage.h"
|
||||||
#include "CombinedImagePackModel.h"
|
#include "CombinedImagePackModel.h"
|
||||||
#include "CommandCompleter.h"
|
#include "CommandCompleter.h"
|
||||||
@ -210,6 +211,7 @@ TimelineViewManager::sync(const mtx::responses::Sync &sync_)
|
|||||||
this->rooms_->sync(sync_);
|
this->rooms_->sync(sync_);
|
||||||
this->communities_->sync(sync_);
|
this->communities_->sync(sync_);
|
||||||
this->presenceEmitter->sync(sync_.presence);
|
this->presenceEmitter->sync(sync_.presence);
|
||||||
|
this->processIgnoredUsers(sync_.account_data);
|
||||||
|
|
||||||
if (isInitialSync_) {
|
if (isInitialSync_) {
|
||||||
this->isInitialSync_ = false;
|
this->isInitialSync_ = false;
|
||||||
@ -560,3 +562,41 @@ TimelineViewManager::fixImageRendering(QQuickTextDocument *t, QQuickItem *i)
|
|||||||
QObject::connect(t->textDocument(), SIGNAL(imagesLoaded()), i, SLOT(updateWholeDocument()));
|
QObject::connect(t->textDocument(), SIGNAL(imagesLoaded()), i, SLOT(updateWholeDocument()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
using IgnoredUsers = mtx::events::EphemeralEvent<mtx::events::account_data::IgnoredUsers>;
|
||||||
|
|
||||||
|
static QVector<QString>
|
||||||
|
convertIgnoredToQt(const IgnoredUsers &ev)
|
||||||
|
{
|
||||||
|
QVector<QString> users;
|
||||||
|
for (const mtx::events::account_data::IgnoredUser &user : ev.content.users) {
|
||||||
|
users.push_back(QString::fromStdString(user.id));
|
||||||
|
}
|
||||||
|
|
||||||
|
return users;
|
||||||
|
}
|
||||||
|
|
||||||
|
QVector<QString>
|
||||||
|
TimelineViewManager::getIgnoredUsers()
|
||||||
|
{
|
||||||
|
const auto cache = cache::client()->getAccountData(mtx::events::EventType::IgnoredUsers);
|
||||||
|
if (!cache) {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
return convertIgnoredToQt(std::get<IgnoredUsers>(*cache));
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
TimelineViewManager::processIgnoredUsers(const mtx::responses::AccountData &data)
|
||||||
|
{
|
||||||
|
for (const mtx::events::collections::RoomAccountDataEvents::variant &ev : data.events) {
|
||||||
|
if (!std::holds_alternative<IgnoredUsers>(ev)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const auto &ignoredEv = std::get<IgnoredUsers>(ev);
|
||||||
|
|
||||||
|
emit this->ignoredUsersChanged(convertIgnoredToQt(ignoredEv));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
@ -11,7 +11,8 @@
|
|||||||
#include <mtx/common.hpp>
|
#include <mtx/common.hpp>
|
||||||
#include <mtx/responses/messages.hpp>
|
#include <mtx/responses/messages.hpp>
|
||||||
|
|
||||||
#include "ReadReceiptsModel.h"
|
#include "InviteesModel.h"
|
||||||
|
#include "MemberList.h"
|
||||||
#include "timeline/CommunitiesModel.h"
|
#include "timeline/CommunitiesModel.h"
|
||||||
#include "timeline/PresenceEmitter.h"
|
#include "timeline/PresenceEmitter.h"
|
||||||
#include "timeline/RoomlistModel.h"
|
#include "timeline/RoomlistModel.h"
|
||||||
@ -39,6 +40,7 @@ class TimelineViewManager final : public QObject
|
|||||||
Q_PROPERTY(
|
Q_PROPERTY(
|
||||||
bool isInitialSync MEMBER isInitialSync_ READ isInitialSync NOTIFY initialSyncChanged)
|
bool isInitialSync MEMBER isInitialSync_ READ isInitialSync NOTIFY initialSyncChanged)
|
||||||
Q_PROPERTY(bool isConnected READ isConnected NOTIFY isConnectedChanged)
|
Q_PROPERTY(bool isConnected READ isConnected NOTIFY isConnectedChanged)
|
||||||
|
Q_PROPERTY(QVector<QString> ignoredUsers READ getIgnoredUsers NOTIFY ignoredUsersChanged)
|
||||||
|
|
||||||
public:
|
public:
|
||||||
TimelineViewManager(CallManager *callManager, ChatPage *parent = nullptr);
|
TimelineViewManager(CallManager *callManager, ChatPage *parent = nullptr);
|
||||||
@ -62,6 +64,10 @@ public:
|
|||||||
return instance_;
|
return instance_;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static TimelineViewManager *instance() { return TimelineViewManager::instance_; }
|
||||||
|
|
||||||
|
QVector<QString> getIgnoredUsers();
|
||||||
|
|
||||||
void sync(const mtx::responses::Sync &sync_);
|
void sync(const mtx::responses::Sync &sync_);
|
||||||
|
|
||||||
VerificationManager *verificationManager() { return verificationManager_; }
|
VerificationManager *verificationManager() { return verificationManager_; }
|
||||||
@ -113,6 +119,7 @@ signals:
|
|||||||
QString url,
|
QString url,
|
||||||
double originalWidth,
|
double originalWidth,
|
||||||
double proportionalHeight);
|
double proportionalHeight);
|
||||||
|
void ignoredUsersChanged(const QVector<QString> &ignoredUsers);
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void updateReadReceipts(const QString &room_id, const std::vector<QString> &event_ids);
|
void updateReadReceipts(const QString &room_id, const std::vector<QString> &event_ids);
|
||||||
@ -154,4 +161,6 @@ private:
|
|||||||
QHash<QPair<QString, quint64>, QColor> userColors;
|
QHash<QPair<QString, quint64>, QColor> userColors;
|
||||||
|
|
||||||
inline static TimelineViewManager *instance_ = nullptr;
|
inline static TimelineViewManager *instance_ = nullptr;
|
||||||
|
|
||||||
|
void processIgnoredUsers(const mtx::responses::AccountData &data);
|
||||||
};
|
};
|
||||||
|
@ -11,11 +11,11 @@
|
|||||||
#include "Cache_p.h"
|
#include "Cache_p.h"
|
||||||
#include "ChatPage.h"
|
#include "ChatPage.h"
|
||||||
#include "Logging.h"
|
#include "Logging.h"
|
||||||
|
#include "MainWindow.h"
|
||||||
|
#include "MatrixClient.h"
|
||||||
#include "UserProfile.h"
|
#include "UserProfile.h"
|
||||||
#include "Utils.h"
|
#include "Utils.h"
|
||||||
#include "encryption/DeviceVerificationFlow.h"
|
|
||||||
#include "encryption/VerificationManager.h"
|
#include "encryption/VerificationManager.h"
|
||||||
#include "mtx/responses/crypto.hpp"
|
|
||||||
#include "timeline/TimelineModel.h"
|
#include "timeline/TimelineModel.h"
|
||||||
#include "timeline/TimelineViewManager.h"
|
#include "timeline/TimelineViewManager.h"
|
||||||
#include "ui/UIA.h"
|
#include "ui/UIA.h"
|
||||||
@ -64,6 +64,19 @@ UserProfile::UserProfile(const QString &roomid,
|
|||||||
new RoomInfoModel(cache::client()->getCommonRooms(userid.toStdString()), this);
|
new RoomInfoModel(cache::client()->getCommonRooms(userid.toStdString()), this);
|
||||||
else
|
else
|
||||||
sharedRooms_ = new RoomInfoModel({}, this);
|
sharedRooms_ = new RoomInfoModel({}, this);
|
||||||
|
|
||||||
|
connect(ChatPage::instance(), &ChatPage::syncUI, this, [this](const mtx::responses::Sync &res) {
|
||||||
|
if (auto ignoreEv = std::ranges::find_if(
|
||||||
|
res.account_data.events,
|
||||||
|
[](const mtx::events::collections::RoomAccountDataEvents &e) {
|
||||||
|
return std::holds_alternative<
|
||||||
|
mtx::events::AccountDataEvent<mtx::events::account_data::IgnoredUsers>>(e);
|
||||||
|
});
|
||||||
|
ignoreEv != res.account_data.events.end()) {
|
||||||
|
// doesn't matter much if it was actually us
|
||||||
|
emit ignoredChanged();
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
QHash<int, QByteArray>
|
QHash<int, QByteArray>
|
||||||
@ -224,6 +237,49 @@ UserProfile::refreshDevices()
|
|||||||
fetchDeviceList(this->userid_);
|
fetchDeviceList(this->userid_);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
UserProfile::ignored() const
|
||||||
|
{
|
||||||
|
auto old = TimelineViewManager::instance()->getIgnoredUsers();
|
||||||
|
return old.contains(userid_);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
UserProfile::setIgnored(bool ignore)
|
||||||
|
{
|
||||||
|
auto old = TimelineViewManager::instance()->getIgnoredUsers();
|
||||||
|
if (ignore) {
|
||||||
|
if (old.contains(userid_)) {
|
||||||
|
emit ignoredChanged();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
old.append(userid_);
|
||||||
|
} else {
|
||||||
|
if (!old.contains(userid_)) {
|
||||||
|
emit ignoredChanged();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
old.removeAll(userid_);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<mtx::events::account_data::IgnoredUser> content;
|
||||||
|
for (const QString &item : std::as_const(old)) {
|
||||||
|
content.emplace_back(item.toStdString());
|
||||||
|
}
|
||||||
|
|
||||||
|
mtx::events::account_data::IgnoredUsers payload{.users{content}};
|
||||||
|
|
||||||
|
auto userid = userid_;
|
||||||
|
|
||||||
|
http::client()->put_account_data(payload, [userid](mtx::http::RequestErr e) {
|
||||||
|
if (e) {
|
||||||
|
MainWindow::instance()->showNotification(
|
||||||
|
tr("Failed to ignore \"%1\": %2")
|
||||||
|
.arg(userid, QString::fromStdString(e->matrix_error.error)));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
UserProfile::fetchDeviceList(const QString &userID)
|
UserProfile::fetchDeviceList(const QString &userID)
|
||||||
{
|
{
|
||||||
@ -345,10 +401,6 @@ UserProfile::banUser()
|
|||||||
ChatPage::instance()->banUser(roomid_, this->userid_, QLatin1String(""));
|
ChatPage::instance()->banUser(roomid_, this->userid_, QLatin1String(""));
|
||||||
}
|
}
|
||||||
|
|
||||||
// void ignoreUser(){
|
|
||||||
|
|
||||||
// }
|
|
||||||
|
|
||||||
void
|
void
|
||||||
UserProfile::kickUser()
|
UserProfile::kickUser()
|
||||||
{
|
{
|
||||||
|
@ -157,6 +157,7 @@ class UserProfile final : public QObject
|
|||||||
Q_PROPERTY(int userVerified READ getUserStatus NOTIFY userStatusChanged)
|
Q_PROPERTY(int userVerified READ getUserStatus NOTIFY userStatusChanged)
|
||||||
Q_PROPERTY(bool isLoading READ isLoading NOTIFY loadingChanged)
|
Q_PROPERTY(bool isLoading READ isLoading NOTIFY loadingChanged)
|
||||||
Q_PROPERTY(bool userVerificationEnabled READ userVerificationEnabled NOTIFY userStatusChanged)
|
Q_PROPERTY(bool userVerificationEnabled READ userVerificationEnabled NOTIFY userStatusChanged)
|
||||||
|
Q_PROPERTY(bool ignored READ ignored WRITE setIgnored NOTIFY ignoredChanged)
|
||||||
Q_PROPERTY(bool isSelf READ isSelf CONSTANT)
|
Q_PROPERTY(bool isSelf READ isSelf CONSTANT)
|
||||||
Q_PROPERTY(TimelineModel *room READ room CONSTANT)
|
Q_PROPERTY(TimelineModel *room READ room CONSTANT)
|
||||||
public:
|
public:
|
||||||
@ -184,7 +185,6 @@ public:
|
|||||||
Q_INVOKABLE void refreshDevices();
|
Q_INVOKABLE void refreshDevices();
|
||||||
Q_INVOKABLE void banUser();
|
Q_INVOKABLE void banUser();
|
||||||
Q_INVOKABLE void signOutDevice(const QString &deviceID);
|
Q_INVOKABLE void signOutDevice(const QString &deviceID);
|
||||||
// Q_INVOKABLE void ignoreUser();
|
|
||||||
Q_INVOKABLE void kickUser();
|
Q_INVOKABLE void kickUser();
|
||||||
Q_INVOKABLE void startChat();
|
Q_INVOKABLE void startChat();
|
||||||
Q_INVOKABLE void startChat(bool encryptionEnabled);
|
Q_INVOKABLE void startChat(bool encryptionEnabled);
|
||||||
@ -193,6 +193,9 @@ public:
|
|||||||
Q_INVOKABLE void changeAvatar();
|
Q_INVOKABLE void changeAvatar();
|
||||||
Q_INVOKABLE void openGlobalProfile();
|
Q_INVOKABLE void openGlobalProfile();
|
||||||
|
|
||||||
|
void setIgnored(bool ignored);
|
||||||
|
bool ignored() const;
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void userStatusChanged();
|
void userStatusChanged();
|
||||||
void loadingChanged();
|
void loadingChanged();
|
||||||
@ -201,6 +204,7 @@ signals:
|
|||||||
void displayError(const QString &errorMessage);
|
void displayError(const QString &errorMessage);
|
||||||
void globalUsernameRetrieved(const QString &globalUser);
|
void globalUsernameRetrieved(const QString &globalUser);
|
||||||
void devicesChanged();
|
void devicesChanged();
|
||||||
|
void ignoredChanged();
|
||||||
|
|
||||||
// internal
|
// internal
|
||||||
void verificationStatiChanged();
|
void verificationStatiChanged();
|
||||||
|
Loading…
Reference in New Issue
Block a user