2018-10-07 13:09:47 +02:00
|
|
|
#include "CommunitiesList.h"
|
2018-10-07 13:18:44 +02:00
|
|
|
#include "Cache.h"
|
2018-07-17 15:37:25 +02:00
|
|
|
#include "Logging.h"
|
2018-05-08 17:43:56 +02:00
|
|
|
#include "MatrixClient.h"
|
2020-01-31 06:12:02 +01:00
|
|
|
#include "Splitter.h"
|
|
|
|
|
|
|
|
#include <mtx/responses/groups.hpp>
|
2020-10-27 17:45:28 +01:00
|
|
|
#include <nlohmann/json.hpp>
|
2018-01-09 14:07:32 +01:00
|
|
|
|
|
|
|
#include <QLabel>
|
|
|
|
|
2018-05-08 17:43:56 +02:00
|
|
|
CommunitiesList::CommunitiesList(QWidget *parent)
|
2018-01-09 14:07:32 +01:00
|
|
|
: QWidget(parent)
|
|
|
|
{
|
|
|
|
QSizePolicy sizePolicy(QSizePolicy::Fixed, QSizePolicy::Expanding);
|
|
|
|
sizePolicy.setHorizontalStretch(0);
|
|
|
|
sizePolicy.setVerticalStretch(1);
|
|
|
|
setSizePolicy(sizePolicy);
|
|
|
|
|
|
|
|
topLayout_ = new QVBoxLayout(this);
|
|
|
|
topLayout_->setSpacing(0);
|
|
|
|
topLayout_->setMargin(0);
|
|
|
|
|
2020-01-31 06:12:02 +01:00
|
|
|
const auto sideBarSizes = splitter::calculateSidebarSizes(QFont{});
|
2018-10-07 13:09:47 +02:00
|
|
|
setFixedWidth(sideBarSizes.groups);
|
2018-01-09 14:07:32 +01:00
|
|
|
|
|
|
|
scrollArea_ = new QScrollArea(this);
|
|
|
|
scrollArea_->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
|
|
scrollArea_->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
|
|
scrollArea_->setSizeAdjustPolicy(QAbstractScrollArea::AdjustToContents);
|
|
|
|
scrollArea_->setWidgetResizable(true);
|
|
|
|
scrollArea_->setAlignment(Qt::AlignLeading | Qt::AlignTop | Qt::AlignVCenter);
|
|
|
|
|
2020-02-21 02:39:22 +01:00
|
|
|
contentsLayout_ = new QVBoxLayout();
|
2018-01-09 14:07:32 +01:00
|
|
|
contentsLayout_->setSpacing(0);
|
|
|
|
contentsLayout_->setMargin(0);
|
|
|
|
|
2018-04-28 14:27:12 +02:00
|
|
|
addGlobalItem();
|
2018-01-09 14:07:32 +01:00
|
|
|
contentsLayout_->addStretch(1);
|
|
|
|
|
2020-02-21 02:39:22 +01:00
|
|
|
scrollArea_->setLayout(contentsLayout_);
|
2018-01-09 14:07:32 +01:00
|
|
|
topLayout_->addWidget(scrollArea_);
|
|
|
|
|
2018-06-09 15:03:14 +02:00
|
|
|
connect(
|
|
|
|
this, &CommunitiesList::avatarRetrieved, this, &CommunitiesList::updateCommunityAvatar);
|
2018-01-09 14:07:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2018-07-14 11:08:16 +02:00
|
|
|
CommunitiesList::setCommunities(const mtx::responses::JoinedGroups &response)
|
2018-01-09 14:07:32 +01:00
|
|
|
{
|
2018-09-28 14:40:51 +02:00
|
|
|
// remove all non-tag communities
|
|
|
|
auto it = communities_.begin();
|
|
|
|
while (it != communities_.end()) {
|
|
|
|
if (it->second->is_tag()) {
|
|
|
|
++it;
|
|
|
|
} else {
|
|
|
|
it = communities_.erase(it);
|
|
|
|
}
|
|
|
|
}
|
2018-01-09 14:07:32 +01:00
|
|
|
|
2018-04-28 14:27:12 +02:00
|
|
|
addGlobalItem();
|
2018-01-09 14:07:32 +01:00
|
|
|
|
2018-07-14 11:08:16 +02:00
|
|
|
for (const auto &group : response.groups)
|
|
|
|
addCommunity(group);
|
2018-01-09 14:07:32 +01:00
|
|
|
|
2018-04-28 14:27:12 +02:00
|
|
|
communities_["world"]->setPressedState(true);
|
2018-01-09 14:07:32 +01:00
|
|
|
emit communityChanged("world");
|
2018-09-28 14:40:51 +02:00
|
|
|
sortEntries();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CommunitiesList::syncTags(const std::map<QString, RoomInfo> &info)
|
|
|
|
{
|
|
|
|
for (const auto &room : info)
|
|
|
|
setTagsForRoom(room.first, room.second.tags);
|
|
|
|
sortEntries();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CommunitiesList::setTagsForRoom(const QString &room_id, const std::vector<std::string> &tags)
|
|
|
|
{
|
|
|
|
// create missing tag if any
|
|
|
|
for (const auto &tag : tags) {
|
|
|
|
// filter out tags we should ignore according to the spec
|
|
|
|
// https://matrix.org/docs/spec/client_server/r0.4.0.html#id154
|
|
|
|
// nheko currently does not make use of internal tags
|
|
|
|
// so we ignore any tag containig a `.` (which would indicate a tag
|
|
|
|
// in the form `tld.domain.*`) except for `m.*` and `u.*`.
|
|
|
|
if (tag.find(".") != ::std::string::npos && tag.compare(0, 2, "m.") &&
|
|
|
|
tag.compare(0, 2, "u."))
|
|
|
|
continue;
|
|
|
|
QString name = QString("tag:") + QString::fromStdString(tag);
|
|
|
|
if (!communityExists(name)) {
|
|
|
|
addCommunity(std::string("tag:") + tag);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// update membership of the room for all tags
|
|
|
|
auto it = communities_.begin();
|
|
|
|
while (it != communities_.end()) {
|
|
|
|
// Skip if the community is not a tag
|
|
|
|
if (!it->second->is_tag()) {
|
|
|
|
++it;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// insert or remove the room from the tag as appropriate
|
|
|
|
std::string current_tag =
|
2020-12-25 04:11:47 +01:00
|
|
|
it->first.right(static_cast<int>(it->first.size() - strlen("tag:")))
|
|
|
|
.toStdString();
|
2018-09-28 14:40:51 +02:00
|
|
|
if (std::find(tags.begin(), tags.end(), current_tag) != tags.end()) {
|
|
|
|
// the room has this tag
|
|
|
|
it->second->addRoom(room_id);
|
|
|
|
} else {
|
|
|
|
// the room does not have this tag
|
|
|
|
it->second->delRoom(room_id);
|
|
|
|
}
|
|
|
|
// Check if the tag is now empty, if yes delete it
|
|
|
|
if (it->second->rooms().empty()) {
|
|
|
|
it = communities_.erase(it);
|
|
|
|
} else {
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
2018-01-09 14:07:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2018-07-14 11:08:16 +02:00
|
|
|
CommunitiesList::addCommunity(const std::string &group_id)
|
2018-01-09 14:07:32 +01:00
|
|
|
{
|
2018-07-14 11:08:16 +02:00
|
|
|
const auto id = QString::fromStdString(group_id);
|
2018-01-09 14:07:32 +01:00
|
|
|
|
2018-07-14 11:08:16 +02:00
|
|
|
CommunitiesListItem *list_item = new CommunitiesListItem(id, scrollArea_);
|
|
|
|
communities_.emplace(id, QSharedPointer<CommunitiesListItem>(list_item));
|
2018-01-09 14:07:32 +01:00
|
|
|
contentsLayout_->insertWidget(contentsLayout_->count() - 1, list_item);
|
|
|
|
|
2020-11-18 13:29:28 +01:00
|
|
|
connect(list_item,
|
|
|
|
&CommunitiesListItem::clicked,
|
|
|
|
this,
|
|
|
|
&CommunitiesList::highlightSelectedCommunity);
|
2021-01-23 00:30:45 +01:00
|
|
|
connect(list_item, &CommunitiesListItem::isDisabledChanged, this, [this]() {
|
|
|
|
for (const auto &community : communities_) {
|
|
|
|
if (community.second->isPressed()) {
|
|
|
|
emit highlightSelectedCommunity(community.first);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2020-11-18 13:29:28 +01:00
|
|
|
|
|
|
|
if (group_id.empty() || group_id.front() != '+')
|
|
|
|
return;
|
|
|
|
|
|
|
|
nhlog::ui()->debug("Add community: {}", group_id);
|
|
|
|
|
2018-07-14 11:08:16 +02:00
|
|
|
connect(this,
|
|
|
|
&CommunitiesList::groupProfileRetrieved,
|
|
|
|
this,
|
|
|
|
[this](const QString &id, const mtx::responses::GroupProfile &profile) {
|
|
|
|
if (communities_.find(id) == communities_.end())
|
|
|
|
return;
|
|
|
|
|
|
|
|
communities_.at(id)->setName(QString::fromStdString(profile.name));
|
|
|
|
|
|
|
|
if (!profile.avatar_url.empty())
|
|
|
|
fetchCommunityAvatar(id,
|
|
|
|
QString::fromStdString(profile.avatar_url));
|
|
|
|
});
|
|
|
|
connect(this,
|
|
|
|
&CommunitiesList::groupRoomsRetrieved,
|
|
|
|
this,
|
2021-01-23 00:30:45 +01:00
|
|
|
[this](const QString &id, const std::set<QString> &rooms) {
|
|
|
|
nhlog::ui()->info(
|
|
|
|
"Fetched rooms for {}: {}", id.toStdString(), rooms.size());
|
2018-07-14 11:08:16 +02:00
|
|
|
if (communities_.find(id) == communities_.end())
|
|
|
|
return;
|
|
|
|
|
|
|
|
communities_.at(id)->setRooms(rooms);
|
|
|
|
});
|
|
|
|
|
2018-07-15 18:09:08 +02:00
|
|
|
http::client()->group_profile(
|
2018-07-14 11:08:16 +02:00
|
|
|
group_id, [id, this](const mtx::responses::GroupProfile &res, mtx::http::RequestErr err) {
|
|
|
|
if (err) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
emit groupProfileRetrieved(id, res);
|
|
|
|
});
|
|
|
|
|
2018-07-15 18:09:08 +02:00
|
|
|
http::client()->group_rooms(
|
2018-07-14 11:08:16 +02:00
|
|
|
group_id, [id, this](const nlohmann::json &res, mtx::http::RequestErr err) {
|
|
|
|
if (err) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-01-23 00:30:45 +01:00
|
|
|
std::set<QString> room_ids;
|
2018-07-14 11:08:16 +02:00
|
|
|
for (const auto &room : res.at("chunk"))
|
2021-01-23 00:30:45 +01:00
|
|
|
room_ids.emplace(QString::fromStdString(room.at("room_id")));
|
2018-07-14 11:08:16 +02:00
|
|
|
|
|
|
|
emit groupRoomsRetrieved(id, room_ids);
|
|
|
|
});
|
2018-01-09 14:07:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CommunitiesList::updateCommunityAvatar(const QString &community_id, const QPixmap &img)
|
|
|
|
{
|
2018-01-24 19:46:37 +01:00
|
|
|
if (!communityExists(community_id)) {
|
2020-01-31 06:12:02 +01:00
|
|
|
nhlog::ui()->warn("Avatar update on nonexistent community {}",
|
|
|
|
community_id.toStdString());
|
2018-01-09 14:07:32 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-07-14 11:08:16 +02:00
|
|
|
communities_.at(community_id)->setAvatar(img.toImage());
|
2018-01-09 14:07:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
CommunitiesList::highlightSelectedCommunity(const QString &community_id)
|
|
|
|
{
|
2018-01-24 19:46:37 +01:00
|
|
|
if (!communityExists(community_id)) {
|
2020-01-31 06:12:02 +01:00
|
|
|
nhlog::ui()->debug("CommunitiesList: clicked unknown community");
|
2018-01-09 14:07:32 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-01-24 19:46:37 +01:00
|
|
|
emit communityChanged(community_id);
|
|
|
|
|
|
|
|
for (const auto &community : communities_) {
|
|
|
|
if (community.first != community_id) {
|
|
|
|
community.second->setPressedState(false);
|
2018-01-09 14:07:32 +01:00
|
|
|
} else {
|
2018-01-24 19:46:37 +01:00
|
|
|
community.second->setPressedState(true);
|
|
|
|
scrollArea_->ensureWidgetVisible(community.second.data());
|
2018-01-09 14:07:32 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-06-09 15:03:14 +02:00
|
|
|
|
|
|
|
void
|
|
|
|
CommunitiesList::fetchCommunityAvatar(const QString &id, const QString &avatarUrl)
|
|
|
|
{
|
2019-12-15 02:56:04 +01:00
|
|
|
auto savedImgData = cache::image(avatarUrl);
|
2018-06-09 15:03:14 +02:00
|
|
|
if (!savedImgData.isNull()) {
|
|
|
|
QPixmap pix;
|
|
|
|
pix.loadFromData(savedImgData);
|
|
|
|
emit avatarRetrieved(id, pix);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-06-10 19:03:45 +02:00
|
|
|
if (avatarUrl.isEmpty())
|
|
|
|
return;
|
|
|
|
|
2018-06-09 15:03:14 +02:00
|
|
|
mtx::http::ThumbOpts opts;
|
|
|
|
opts.mxc_url = avatarUrl.toStdString();
|
2018-07-15 18:09:08 +02:00
|
|
|
http::client()->get_thumbnail(
|
2018-06-09 15:03:14 +02:00
|
|
|
opts, [this, opts, id](const std::string &res, mtx::http::RequestErr err) {
|
|
|
|
if (err) {
|
2018-06-14 01:28:35 +02:00
|
|
|
nhlog::net()->warn("failed to download avatar: {} - ({} {})",
|
2018-06-14 08:36:41 +02:00
|
|
|
opts.mxc_url,
|
|
|
|
mtx::errors::to_string(err->matrix_error.errcode),
|
|
|
|
err->matrix_error.error);
|
2018-06-09 15:03:14 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-12-15 02:56:04 +01:00
|
|
|
cache::saveImage(opts.mxc_url, res);
|
2018-06-09 15:03:14 +02:00
|
|
|
|
2020-12-25 01:08:06 +01:00
|
|
|
auto data = QByteArray(res.data(), (int)res.size());
|
2018-06-09 15:03:14 +02:00
|
|
|
|
|
|
|
QPixmap pix;
|
|
|
|
pix.loadFromData(data);
|
|
|
|
|
|
|
|
emit avatarRetrieved(id, pix);
|
|
|
|
});
|
|
|
|
}
|
2018-07-14 11:08:16 +02:00
|
|
|
|
2021-01-23 00:30:45 +01:00
|
|
|
std::set<QString>
|
2018-07-14 11:08:16 +02:00
|
|
|
CommunitiesList::roomList(const QString &id) const
|
|
|
|
{
|
|
|
|
if (communityExists(id))
|
|
|
|
return communities_.at(id)->rooms();
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
2018-09-28 14:40:51 +02:00
|
|
|
|
2020-05-18 03:30:04 +02:00
|
|
|
std::vector<std::string>
|
|
|
|
CommunitiesList::currentTags() const
|
|
|
|
{
|
|
|
|
std::vector<std::string> tags;
|
|
|
|
for (auto &entry : communities_) {
|
|
|
|
CommunitiesListItem *item = entry.second.data();
|
|
|
|
if (item->is_tag())
|
|
|
|
tags.push_back(entry.first.mid(4).toStdString());
|
|
|
|
}
|
|
|
|
return tags;
|
|
|
|
}
|
|
|
|
|
2021-01-23 00:30:45 +01:00
|
|
|
std::set<QString>
|
|
|
|
CommunitiesList::hiddenTagsAndCommunities() const
|
|
|
|
{
|
|
|
|
std::set<QString> hiddenTags;
|
|
|
|
for (auto &entry : communities_) {
|
|
|
|
if (entry.second->isDisabled())
|
|
|
|
hiddenTags.insert(entry.first);
|
|
|
|
}
|
|
|
|
|
|
|
|
return hiddenTags;
|
|
|
|
}
|
|
|
|
|
2018-09-28 14:40:51 +02:00
|
|
|
void
|
|
|
|
CommunitiesList::sortEntries()
|
|
|
|
{
|
|
|
|
std::vector<CommunitiesListItem *> header;
|
|
|
|
std::vector<CommunitiesListItem *> communities;
|
|
|
|
std::vector<CommunitiesListItem *> tags;
|
|
|
|
std::vector<CommunitiesListItem *> footer;
|
|
|
|
// remove all the contents and sort them in the 4 vectors
|
|
|
|
for (auto &entry : communities_) {
|
|
|
|
CommunitiesListItem *item = entry.second.data();
|
|
|
|
contentsLayout_->removeWidget(item);
|
|
|
|
// world is handled separately
|
|
|
|
if (entry.first == "world")
|
|
|
|
continue;
|
|
|
|
// sort the rest
|
|
|
|
if (item->is_tag())
|
|
|
|
if (entry.first == "tag:m.favourite")
|
|
|
|
header.push_back(item);
|
|
|
|
else if (entry.first == "tag:m.lowpriority")
|
|
|
|
footer.push_back(item);
|
|
|
|
else
|
|
|
|
tags.push_back(item);
|
|
|
|
else
|
|
|
|
communities.push_back(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
// now there remains only the stretch in the layout, remove it
|
|
|
|
QLayoutItem *stretch = contentsLayout_->itemAt(0);
|
|
|
|
contentsLayout_->removeItem(stretch);
|
|
|
|
|
|
|
|
contentsLayout_->addWidget(communities_["world"].data());
|
|
|
|
|
|
|
|
auto insert_widgets = [this](auto &vec) {
|
|
|
|
for (auto item : vec)
|
|
|
|
contentsLayout_->addWidget(item);
|
|
|
|
};
|
|
|
|
insert_widgets(header);
|
|
|
|
insert_widgets(communities);
|
|
|
|
insert_widgets(tags);
|
|
|
|
insert_widgets(footer);
|
|
|
|
|
|
|
|
contentsLayout_->addItem(stretch);
|
|
|
|
}
|