nheko/src/RoomList.cpp

553 lines
17 KiB
C++
Raw Normal View History

2017-04-06 01:06:42 +02:00
/*
* nheko Copyright (C) 2017 Konstantinos Sideris <siderisk@auth.gr>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <limits>
2020-03-14 00:30:50 +01:00
#include <set>
2017-10-28 20:24:42 +02:00
#include <QObject>
2020-01-31 06:12:02 +01:00
#include <QPainter>
#include <QScroller>
2020-05-25 13:03:49 +02:00
#include <QStyle>
#include <QStyleOption>
#include <QTimer>
2017-04-06 01:06:42 +02:00
2018-07-17 15:37:25 +02:00
#include "Logging.h"
#include "MainWindow.h"
2017-04-06 01:06:42 +02:00
#include "RoomInfoListItem.h"
#include "RoomList.h"
#include "UserSettingsPage.h"
2018-04-29 14:42:40 +02:00
#include "Utils.h"
2018-07-17 15:37:25 +02:00
#include "ui/OverlayModal.h"
2017-04-06 01:06:42 +02:00
RoomList::RoomList(QSharedPointer<UserSettings> userSettings, QWidget *parent)
2017-08-20 12:47:22 +02:00
: QWidget(parent)
2017-04-06 01:06:42 +02:00
{
2017-09-10 11:59:21 +02:00
topLayout_ = new QVBoxLayout(this);
topLayout_->setSpacing(0);
topLayout_->setMargin(0);
scrollArea_ = new QScrollArea(this);
scrollArea_->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
scrollArea_->setSizeAdjustPolicy(QAbstractScrollArea::AdjustToContents);
scrollArea_->setWidgetResizable(true);
2018-01-09 14:07:32 +01:00
scrollArea_->setAlignment(Qt::AlignLeading | Qt::AlignTop | Qt::AlignVCenter);
scrollArea_->setAttribute(Qt::WA_AcceptTouchEvents);
2017-09-10 11:59:21 +02:00
QScroller::grabGesture(scrollArea_, QScroller::TouchGesture);
QScroller::grabGesture(scrollArea_, QScroller::LeftMouseButtonGesture);
// The scrollbar on macOS will hide itself when not active so it won't interfere
// with the content.
#if not defined(Q_OS_MAC)
scrollArea_->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
#endif
2017-11-08 22:09:15 +01:00
scrollAreaContents_ = new QWidget(this);
scrollAreaContents_->setObjectName("roomlist_area");
2017-09-10 11:59:21 +02:00
contentsLayout_ = new QVBoxLayout(scrollAreaContents_);
2018-10-07 13:18:44 +02:00
contentsLayout_->setAlignment(Qt::AlignTop);
2017-09-10 11:59:21 +02:00
contentsLayout_->setSpacing(0);
contentsLayout_->setMargin(0);
scrollArea_->setWidget(scrollAreaContents_);
topLayout_->addWidget(scrollArea_);
connect(this, &RoomList::updateRoomAvatarCb, this, &RoomList::updateRoomAvatar);
2020-03-15 20:27:30 +01:00
connect(userSettings.data(),
2020-03-15 19:30:21 +01:00
&UserSettings::roomSortingChanged,
this,
&RoomList::sortRoomsByLastMessage);
2017-04-06 01:06:42 +02:00
}
2017-08-20 12:47:22 +02:00
void
2018-04-21 15:34:50 +02:00
RoomList::addRoom(const QString &room_id, const RoomInfo &info)
{
2020-10-17 00:57:29 +02:00
auto room_item = new RoomInfoListItem(room_id, info, scrollArea_);
2018-04-21 15:34:50 +02:00
room_item->setRoomName(QString::fromStdString(std::move(info.name)));
connect(room_item, &RoomInfoListItem::clicked, this, &RoomList::highlightSelectedRoom);
connect(room_item, &RoomInfoListItem::leaveRoom, this, [](const QString &room_id) {
MainWindow::instance()->openLeaveRoomDialog(room_id);
});
2020-10-17 00:57:29 +02:00
QSharedPointer<RoomInfoListItem> roomWidget(room_item, &QObject::deleteLater);
rooms_.emplace(room_id, roomWidget);
rooms_sort_cache_.push_back(roomWidget);
2018-04-21 15:34:50 +02:00
if (!info.avatar_url.empty())
updateAvatar(room_id, QString::fromStdString(info.avatar_url));
int pos = contentsLayout_->count() - 1;
contentsLayout_->insertWidget(pos, room_item);
}
2017-12-21 23:00:48 +01:00
void
RoomList::updateAvatar(const QString &room_id, const QString &url)
{
emit updateRoomAvatarCb(room_id, url);
2017-12-21 23:00:48 +01:00
}
void
RoomList::removeRoom(const QString &room_id, bool reset)
{
auto roomIt = rooms_.find(room_id);
if (roomIt == rooms_.end()) {
return;
}
for (auto roomSortIt = rooms_sort_cache_.begin(); roomSortIt != rooms_sort_cache_.end();
++roomSortIt) {
if (roomIt->second == *roomSortIt) {
rooms_sort_cache_.erase(roomSortIt);
break;
}
}
2018-01-25 13:34:15 +01:00
rooms_.erase(room_id);
if (rooms_.empty() || !reset)
return;
auto room = firstRoom();
if (room.second.isNull())
return;
room.second->setPressedState(true);
emit roomChanged(room.first);
}
2017-08-20 12:47:22 +02:00
void
RoomList::updateUnreadMessageCount(const QString &roomid, int count, int highlightedCount)
{
if (!roomExists(roomid)) {
nhlog::ui()->warn("updateUnreadMessageCount: unknown room_id {}",
roomid.toStdString());
2017-09-10 11:59:21 +02:00
return;
}
rooms_[roomid]->updateUnreadMessageCount(count, highlightedCount);
2017-09-10 11:59:21 +02:00
calculateUnreadMessageCount();
sortRoomsByLastMessage();
}
2017-08-20 12:47:22 +02:00
void
RoomList::calculateUnreadMessageCount()
{
2017-09-10 11:59:21 +02:00
int total_unread_msgs = 0;
for (const auto &room : rooms_) {
if (!room.second.isNull())
total_unread_msgs += room.second->unreadMessageCount();
}
2017-09-10 11:59:21 +02:00
emit totalUnreadMessageCountUpdated(total_unread_msgs);
}
2017-08-20 12:47:22 +02:00
void
2018-04-21 15:34:50 +02:00
RoomList::initialize(const QMap<QString, RoomInfo> &info)
2017-04-06 01:06:42 +02:00
{
nhlog::ui()->info("initialize room list");
2018-04-21 15:34:50 +02:00
2017-09-10 11:59:21 +02:00
rooms_.clear();
2017-04-06 01:06:42 +02:00
2020-05-03 22:30:51 +02:00
// prevent flickering and save time sorting over and over again
setUpdatesEnabled(false);
2018-04-21 15:34:50 +02:00
for (auto it = info.begin(); it != info.end(); it++) {
if (it.value().is_invite)
addInvitedRoom(it.key(), it.value());
else
addRoom(it.key(), it.value());
2017-09-10 11:59:21 +02:00
}
2017-05-31 18:42:07 +02:00
for (auto it = info.begin(); it != info.end(); it++)
updateRoomDescription(it.key(), it.value().msgInfo);
setUpdatesEnabled(true);
if (rooms_.empty())
2017-09-10 11:59:21 +02:00
return;
2020-02-04 21:16:04 +01:00
sortRoomsByLastMessage();
auto room = firstRoom();
if (room.second.isNull())
return;
room.second->setPressedState(true);
emit roomChanged(room.first);
}
2018-04-22 13:19:05 +02:00
void
RoomList::cleanupInvites(const std::map<QString, bool> &invites)
{
2018-04-27 17:19:43 +02:00
if (invites.size() == 0)
2018-04-22 13:19:05 +02:00
return;
2018-04-27 17:19:43 +02:00
utils::erase_if(rooms_, [invites](auto &room) {
auto room_id = room.first;
auto item = room.second;
if (!item)
return false;
return item->isInvite() && (invites.find(room_id) == invites.end());
2018-04-27 17:19:43 +02:00
});
2018-04-22 13:19:05 +02:00
}
2017-08-20 12:47:22 +02:00
void
2018-04-21 15:34:50 +02:00
RoomList::sync(const std::map<QString, RoomInfo> &info)
{
2018-04-21 15:34:50 +02:00
for (const auto &room : info)
updateRoom(room.first, room.second);
if (!info.empty())
sortRoomsByLastMessage();
2017-04-06 01:06:42 +02:00
}
2017-08-20 12:47:22 +02:00
void
RoomList::highlightSelectedRoom(const QString &room_id)
2017-04-06 01:06:42 +02:00
{
2017-09-10 11:59:21 +02:00
emit roomChanged(room_id);
if (!roomExists(room_id)) {
nhlog::ui()->warn("roomlist: clicked unknown room_id");
2017-09-10 11:59:21 +02:00
return;
}
for (auto const &room : rooms_) {
if (room.second.isNull())
continue;
if (room.first != room_id) {
room.second->setPressedState(false);
2017-09-10 11:59:21 +02:00
} else {
room.second->setPressedState(true);
scrollArea_->ensureWidgetVisible(room.second.data());
2017-09-10 11:59:21 +02:00
}
}
2018-01-09 14:07:32 +01:00
selectedRoom_ = room_id;
2017-04-06 01:06:42 +02:00
}
2020-01-31 01:39:51 +01:00
void
RoomList::nextRoom()
{
for (int ii = 0; ii < contentsLayout_->count() - 1; ++ii) {
auto room = qobject_cast<RoomInfoListItem *>(contentsLayout_->itemAt(ii)->widget());
if (!room)
continue;
if (room->roomId() == selectedRoom_) {
auto nextRoom = qobject_cast<RoomInfoListItem *>(
contentsLayout_->itemAt(ii + 1)->widget());
// Not a room message.
if (!nextRoom || nextRoom->isInvite())
return;
emit roomChanged(nextRoom->roomId());
if (!roomExists(nextRoom->roomId())) {
nhlog::ui()->warn("roomlist: clicked unknown room_id");
return;
}
room->setPressedState(false);
nextRoom->setPressedState(true);
scrollArea_->ensureWidgetVisible(nextRoom);
selectedRoom_ = nextRoom->roomId();
return;
}
}
}
void
RoomList::previousRoom()
{
for (int ii = 1; ii < contentsLayout_->count(); ++ii) {
auto room = qobject_cast<RoomInfoListItem *>(contentsLayout_->itemAt(ii)->widget());
if (!room)
continue;
if (room->roomId() == selectedRoom_) {
auto nextRoom = qobject_cast<RoomInfoListItem *>(
contentsLayout_->itemAt(ii - 1)->widget());
// Not a room message.
if (!nextRoom || nextRoom->isInvite())
return;
emit roomChanged(nextRoom->roomId());
if (!roomExists(nextRoom->roomId())) {
nhlog::ui()->warn("roomlist: clicked unknown room_id");
return;
}
room->setPressedState(false);
nextRoom->setPressedState(true);
scrollArea_->ensureWidgetVisible(nextRoom);
selectedRoom_ = nextRoom->roomId();
return;
}
}
}
2017-08-20 12:47:22 +02:00
void
RoomList::updateRoomAvatar(const QString &roomid, const QString &img)
2017-04-06 01:06:42 +02:00
{
if (!roomExists(roomid)) {
nhlog::ui()->warn("avatar update on non-existent room_id: {}",
roomid.toStdString());
2017-09-10 11:59:21 +02:00
return;
}
2017-04-06 01:06:42 +02:00
rooms_[roomid]->setAvatar(img);
2017-12-21 23:00:48 +01:00
// Used to inform other widgets for the new image data.
emit roomAvatarChanged(roomid, img);
}
2017-08-20 12:47:22 +02:00
void
RoomList::updateRoomDescription(const QString &roomid, const DescInfo &info)
{
if (!roomExists(roomid)) {
nhlog::ui()->warn("description update on non-existent room_id: {}, {}",
roomid.toStdString(),
info.body.toStdString());
2017-09-10 11:59:21 +02:00
return;
}
rooms_[roomid]->setDescriptionMessage(info);
if (underMouse()) {
// When the user hover out of the roomlist a sort will be triggered.
isSortPending_ = true;
return;
}
isSortPending_ = false;
emit sortRoomsByLastMessage();
}
2020-03-14 14:16:08 +01:00
struct room_sort
{
bool operator()(const QSharedPointer<RoomInfoListItem> a,
const QSharedPointer<RoomInfoListItem> b) const
2020-03-14 14:16:08 +01:00
{
2020-03-14 00:30:50 +01:00
// Sort by "importance" (i.e. invites before mentions before
// notifs before new events before old events), then secondly
// by recency.
// Checking importance first
const auto a_importance = a->calculateImportance();
const auto b_importance = b->calculateImportance();
2020-03-14 14:16:08 +01:00
if (a_importance != b_importance) {
2020-03-14 00:30:50 +01:00
return a_importance > b_importance;
}
// Now sort by recency
// Zero if empty, otherwise the time that the event occured
const uint64_t a_recency =
a->lastMessageInfo().userid.isEmpty() ? 0 : a->lastMessageInfo().timestamp;
const uint64_t b_recency =
b->lastMessageInfo().userid.isEmpty() ? 0 : b->lastMessageInfo().timestamp;
2020-03-14 00:30:50 +01:00
return a_recency > b_recency;
}
};
void
RoomList::sortRoomsByLastMessage()
{
isSortPending_ = false;
std::stable_sort(begin(rooms_sort_cache_), end(rooms_sort_cache_), room_sort{});
int newIndex = 0;
for (const auto &roomWidget : rooms_sort_cache_) {
2020-05-01 00:38:07 +02:00
const auto currentIndex = contentsLayout_->indexOf(roomWidget.data());
if (currentIndex != newIndex) {
2020-05-01 00:38:07 +02:00
contentsLayout_->removeWidget(roomWidget.data());
contentsLayout_->insertWidget(newIndex, roomWidget.data());
}
newIndex++;
}
}
void
RoomList::leaveEvent(QEvent *event)
{
if (isSortPending_)
QTimer::singleShot(700, this, &RoomList::sortRoomsByLastMessage);
QWidget::leaveEvent(event);
2017-04-06 01:06:42 +02:00
}
void
RoomList::closeJoinRoomDialog(bool isJoining, QString roomAlias)
{
joinRoomModal_->hide();
if (isJoining)
emit joinRoom(roomAlias);
}
2018-01-09 14:07:32 +01:00
void
RoomList::removeFilter(const std::set<QString> &roomsToHide)
{
setUpdatesEnabled(false);
for (int i = 0; i < contentsLayout_->count(); i++) {
auto widget =
qobject_cast<RoomInfoListItem *>(contentsLayout_->itemAt(i)->widget());
if (widget) {
if (roomsToHide.find(widget->roomId()) == roomsToHide.end())
widget->show();
else
widget->hide();
}
}
setUpdatesEnabled(true);
}
void
RoomList::applyFilter(const std::set<QString> &filter)
2018-01-09 14:07:32 +01:00
{
// Disabling paint updates will resolve issues with screen flickering on big room lists.
setUpdatesEnabled(false);
2018-01-09 14:07:32 +01:00
for (int i = 0; i < contentsLayout_->count(); i++) {
// If filter contains the room for the current RoomInfoListItem,
2018-01-09 14:07:32 +01:00
// show the list item, otherwise hide it
auto listitem =
qobject_cast<RoomInfoListItem *>(contentsLayout_->itemAt(i)->widget());
if (!listitem)
continue;
if (filter.find(listitem->roomId()) != filter.end())
listitem->show();
else
listitem->hide();
2018-01-09 14:07:32 +01:00
}
setUpdatesEnabled(true);
// If the already selected room is part of the group, make sure it's visible.
if (!selectedRoom_.isEmpty() && (filter.find(selectedRoom_) != filter.end()))
return;
selectFirstVisibleRoom();
}
void
RoomList::selectFirstVisibleRoom()
{
for (int i = 0; i < contentsLayout_->count(); i++) {
auto item = qobject_cast<RoomInfoListItem *>(contentsLayout_->itemAt(i)->widget());
if (item && item->isVisible()) {
highlightSelectedRoom(item->roomId());
break;
2018-01-09 14:07:32 +01:00
}
}
}
2017-11-25 21:20:34 +01:00
void
RoomList::paintEvent(QPaintEvent *)
{
QStyleOption opt;
opt.init(this);
QPainter p(this);
style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this);
}
2017-12-19 21:36:12 +01:00
void
2018-04-21 15:34:50 +02:00
RoomList::updateRoom(const QString &room_id, const RoomInfo &info)
2017-12-19 21:36:12 +01:00
{
2018-04-21 15:34:50 +02:00
if (!roomExists(room_id)) {
if (info.is_invite)
addInvitedRoom(room_id, info);
else
addRoom(room_id, info);
return;
2017-12-19 21:36:12 +01:00
}
2018-04-21 15:34:50 +02:00
auto room = rooms_[room_id];
updateAvatar(room_id, QString::fromStdString(info.avatar_url));
room->setRoomName(QString::fromStdString(info.name));
room->setRoomType(info.is_invite);
room->update();
2017-12-19 21:36:12 +01:00
}
void
2018-04-21 15:34:50 +02:00
RoomList::addInvitedRoom(const QString &room_id, const RoomInfo &info)
2017-12-19 21:36:12 +01:00
{
2020-10-17 00:57:29 +02:00
auto room_item = new RoomInfoListItem(room_id, info, scrollArea_);
2018-04-21 15:34:50 +02:00
2017-12-19 21:36:12 +01:00
connect(room_item, &RoomInfoListItem::acceptInvite, this, &RoomList::acceptInvite);
connect(room_item, &RoomInfoListItem::declineInvite, this, &RoomList::declineInvite);
QSharedPointer<RoomInfoListItem> roomWidget(room_item);
rooms_.emplace(room_id, roomWidget);
rooms_sort_cache_.push_back(roomWidget);
2017-12-19 21:36:12 +01:00
2018-04-21 15:34:50 +02:00
updateAvatar(room_id, QString::fromStdString(info.avatar_url));
2017-12-19 21:36:12 +01:00
int pos = contentsLayout_->count() - 1;
contentsLayout_->insertWidget(pos, room_item);
}
std::pair<QString, QSharedPointer<RoomInfoListItem>>
RoomList::firstRoom() const
{
2020-02-04 21:16:04 +01:00
for (int i = 0; i < contentsLayout_->count(); i++) {
auto item = qobject_cast<RoomInfoListItem *>(contentsLayout_->itemAt(i)->widget());
2020-02-04 21:16:04 +01:00
if (item) {
auto topRoom = rooms_.find(item->roomId());
if (topRoom != rooms_.end()) {
return std::pair<QString, QSharedPointer<RoomInfoListItem>>(
item->roomId(), topRoom->second);
}
2020-02-04 21:16:04 +01:00
}
}
2020-02-04 21:16:04 +01:00
return {};
}
void
RoomList::updateReadStatus(const std::map<QString, bool> &status)
{
for (const auto &room : status) {
if (roomExists(room.first)) {
auto item = rooms_.at(room.first);
if (item)
item->setReadState(room.second);
}
}
}