/* * nheko Copyright (C) 2017 Konstantinos Sideris * * 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 . */ #include #include #include #include #include "Cache.h" #include "MainWindow.h" #include "MatrixClient.h" #include "OverlayModal.h" #include "RoomInfoListItem.h" #include "RoomList.h" #include "RoomSettings.h" #include "RoomState.h" #include "UserSettingsPage.h" RoomList::RoomList(QSharedPointer client, QSharedPointer userSettings, QWidget *parent) : QWidget(parent) , client_(client) , userSettings_{userSettings} { setStyleSheet("border: none;"); topLayout_ = new QVBoxLayout(this); topLayout_->setSpacing(0); topLayout_->setMargin(0); 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); scrollAreaContents_ = new QWidget(this); contentsLayout_ = new QVBoxLayout(scrollAreaContents_); contentsLayout_->setSpacing(0); contentsLayout_->setMargin(0); contentsLayout_->addStretch(1); scrollArea_->setWidget(scrollAreaContents_); topLayout_->addWidget(scrollArea_); connect(client_.data(), &MatrixClient::roomAvatarRetrieved, this, [=](const QString &room_id, const QPixmap &img, const QString &url, const QByteArray &data) { if (!cache_.isNull()) cache_->saveImage(url, data); updateRoomAvatar(room_id, img); }); } RoomList::~RoomList() {} void RoomList::clear() { rooms_.clear(); } void RoomList::addRoom(const QMap> &settings, const RoomState &state, const QString &room_id) { RoomInfoListItem *room_item = new RoomInfoListItem(settings[room_id], state, room_id, scrollArea_); connect(room_item, &RoomInfoListItem::clicked, this, &RoomList::highlightSelectedRoom); connect(room_item, &RoomInfoListItem::leaveRoom, this, &RoomList::openLeaveRoomDialog); rooms_.insert(room_id, QSharedPointer(room_item)); if (!state.getAvatar().toString().isEmpty()) updateAvatar(room_id, state.getAvatar().toString()); int pos = contentsLayout_->count() - 1; contentsLayout_->insertWidget(pos, room_item); } void RoomList::updateAvatar(const QString &room_id, const QString &url) { if (url.isEmpty()) return; QByteArray savedImgData; if (!cache_.isNull()) savedImgData = cache_->image(url); if (savedImgData.isEmpty()) { client_->fetchRoomAvatar(room_id, url); } else { QPixmap img; img.loadFromData(savedImgData); updateRoomAvatar(room_id, img); } } void RoomList::removeRoom(const QString &room_id, bool reset) { rooms_.remove(room_id); if (rooms_.isEmpty() || !reset) return; auto first_room = rooms_.first(); first_room->setPressedState(true); emit roomChanged(rooms_.firstKey()); } void RoomList::updateUnreadMessageCount(const QString &roomid, int count) { if (!rooms_.contains(roomid)) { qWarning() << "UpdateUnreadMessageCount: Unknown roomid"; return; } rooms_[roomid]->updateUnreadMessageCount(count); calculateUnreadMessageCount(); } void RoomList::calculateUnreadMessageCount() { int total_unread_msgs = 0; for (const auto &room : rooms_) total_unread_msgs += room->unreadMessageCount(); emit totalUnreadMessageCountUpdated(total_unread_msgs); } void RoomList::setInitialRooms(const QMap> &settings, const QMap &states) { rooms_.clear(); if (settings.size() != states.size()) { qWarning() << "Initializing room list"; qWarning() << "Different number of room states and room settings"; return; } for (auto it = states.constBegin(); it != states.constEnd(); ++it) { const auto room_id = it.key(); const auto state = it.value(); addRoom(settings, state, room_id); } if (rooms_.isEmpty()) return; setFilterRooms(filterRooms_); auto first_room = rooms_.first(); first_room->setPressedState(true); emit roomChanged(rooms_.firstKey()); } void RoomList::openLeaveRoomDialog(const QString &room_id) { if (leaveRoomDialog_.isNull()) { leaveRoomDialog_ = QSharedPointer(new dialogs::LeaveRoom(this)); connect(leaveRoomDialog_.data(), &dialogs::LeaveRoom::closing, this, [=](bool leaving) { closeLeaveRoomDialog(leaving, room_id); }); } if (leaveRoomModal_.isNull()) { leaveRoomModal_ = QSharedPointer( new OverlayModal(MainWindow::instance(), leaveRoomDialog_.data())); leaveRoomModal_->setDuration(0); leaveRoomModal_->setColor(QColor(30, 30, 30, 170)); } leaveRoomModal_->fadeIn(); } void RoomList::sync(const QMap &states, QMap> &settings) { for (auto it = states.constBegin(); it != states.constEnd(); ++it) { auto room_id = it.key(); auto state = it.value(); if (!rooms_.contains(room_id)) { settings.insert(room_id, QSharedPointer(new RoomSettings(room_id))); addRoom(settings, state, room_id); } auto room = rooms_[room_id]; auto current_avatar = room->state().getAvatar(); auto new_avatar = state.getAvatar(); if (current_avatar != new_avatar && !new_avatar.toString().isEmpty()) updateAvatar(room_id, new_avatar.toString()); room->setState(state); } } void RoomList::clearRoomMessageCount(const QString &room_id) { if (!rooms_.contains(room_id)) return; auto room = rooms_[room_id]; room->clearUnreadMessageCount(); calculateUnreadMessageCount(); } void RoomList::highlightSelectedRoom(const QString &room_id) { emit roomChanged(room_id); if (!rooms_.contains(room_id)) { qDebug() << "RoomList: clicked unknown roomid"; return; } clearRoomMessageCount(room_id); calculateUnreadMessageCount(); for (auto it = rooms_.constBegin(); it != rooms_.constEnd(); ++it) { if (it.key() != room_id) { it.value()->setPressedState(false); } else { it.value()->setPressedState(true); scrollArea_->ensureWidgetVisible( qobject_cast(it.value().data())); } } selectedRoom_ = room_id; } void RoomList::updateRoomAvatar(const QString &roomid, const QPixmap &img) { if (!rooms_.contains(roomid)) { qWarning() << "Avatar update on non existent room" << roomid; return; } rooms_.value(roomid)->setAvatar(img.toImage()); // Used to inform other widgets for the new image data. emit roomAvatarChanged(roomid, img); } void RoomList::updateRoomDescription(const QString &roomid, const DescInfo &info) { if (!rooms_.contains(roomid)) { qWarning() << "Description update on non existent room" << roomid << info.body; return; } rooms_.value(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(); } void RoomList::sortRoomsByLastMessage() { if (!userSettings_->isOrderingEnabled()) return; isSortPending_ = false; std::multimap> times; for (int ii = 0; ii < contentsLayout_->count(); ++ii) { auto room = qobject_cast(contentsLayout_->itemAt(ii)->widget()); if (!room) continue; // Not a room message. if (room->lastMessageInfo().userid.isEmpty()) times.emplace(0, room); else times.emplace(room->lastMessageInfo().datetime.toMSecsSinceEpoch(), room); } for (auto it = times.cbegin(); it != times.cend(); ++it) { const auto roomWidget = it->second; const auto currentIndex = contentsLayout_->indexOf(roomWidget); const auto newIndex = std::distance(times.cbegin(), it); if (currentIndex == newIndex) continue; contentsLayout_->removeWidget(roomWidget); contentsLayout_->insertWidget(newIndex, roomWidget); } } void RoomList::leaveEvent(QEvent *event) { if (isSortPending_) QTimer::singleShot(700, this, &RoomList::sortRoomsByLastMessage); QWidget::leaveEvent(event); } void RoomList::closeJoinRoomDialog(bool isJoining, QString roomAlias) { joinRoomModal_->fadeOut(); if (isJoining) { client_->joinRoom(roomAlias); } } void RoomList::closeLeaveRoomDialog(bool leaving, const QString &room_id) { leaveRoomModal_->fadeOut(); if (leaving) { client_->leaveRoom(room_id); } } void RoomList::setFilterRooms(bool filterRooms) { filterRooms_ = filterRooms; for (int i = 0; i < contentsLayout_->count(); i++) { // If roomFilter_ contains the room for the current RoomInfoListItem, // show the list item, otherwise hide it RoomInfoListItem *listitem = (RoomInfoListItem *)contentsLayout_->itemAt(i)->widget(); if (listitem != nullptr) { if (!filterRooms) { contentsLayout_->itemAt(i)->widget()->show(); } else if (roomFilter_.contains(listitem->roomId())) { contentsLayout_->itemAt(i)->widget()->show(); } else { contentsLayout_->itemAt(i)->widget()->hide(); } } } if (filterRooms_ && !roomFilter_.contains(selectedRoom_)) { RoomInfoListItem *firstVisibleRoom = nullptr; for (int i = 0; i < contentsLayout_->count(); i++) { QWidget *item = contentsLayout_->itemAt(i)->widget(); if (item != nullptr && item->isVisible()) { firstVisibleRoom = (RoomInfoListItem *)item; break; } } if (firstVisibleRoom != nullptr) { highlightSelectedRoom(firstVisibleRoom->roomId()); } } else { scrollArea_->ensureWidgetVisible( qobject_cast(rooms_.value(selectedRoom_).data())); } } void RoomList::paintEvent(QPaintEvent *) { QStyleOption opt; opt.init(this); QPainter p(this); style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this); } void RoomList::syncInvites(const std::map &rooms) { for (auto it = rooms.cbegin(); it != rooms.cend(); ++it) { const auto room_id = QString::fromStdString(it->first); if (!rooms_.contains(room_id)) addInvitedRoom(room_id, it->second); } } void RoomList::setRoomFilter(QList room_ids) { roomFilter_ = room_ids; setFilterRooms(true); } void RoomList::addInvitedRoom(const QString &room_id, const mtx::responses::InvitedRoom &room) { auto room_item = new RoomInfoListItem(room_id, room, scrollArea_); connect(room_item, &RoomInfoListItem::acceptInvite, this, &RoomList::acceptInvite); connect(room_item, &RoomInfoListItem::declineInvite, this, &RoomList::declineInvite); rooms_.insert(room_id, QSharedPointer(room_item)); updateAvatar(room_id, QString::fromStdString(room.avatar())); int pos = contentsLayout_->count() - 1; contentsLayout_->insertWidget(pos, room_item); }