/* * 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 #include "Cache.h" #include "Config.h" #include "Menu.h" #include "Ripple.h" #include "RippleOverlay.h" #include "RoomInfoListItem.h" #include "Theme.h" #include "Utils.h" constexpr int MaxUnreadCountDisplayed = 99; constexpr int Padding = 9; constexpr int IconSize = 44; constexpr int MaxHeight = IconSize + 2 * Padding; constexpr int InviteBtnX = IconSize + 2 * Padding; constexpr int InviteBtnY = IconSize / 2 + Padding + Padding / 3; void RoomInfoListItem::init(QWidget *parent) { setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed); setMouseTracking(true); setAttribute(Qt::WA_Hover); setFixedHeight(MaxHeight); QPainterPath path; path.addRect(0, 0, parent->width(), height()); ripple_overlay_ = new RippleOverlay(this); ripple_overlay_->setClipPath(path); ripple_overlay_->setClipping(true); font_.setPixelSize(conf::fontSize - 1); usernameFont_ = font_; bubbleFont_ = font_; bubbleFont_.setPixelSize(conf::roomlist::fonts::bubble); unreadCountFont_.setPixelSize(conf::roomlist::fonts::badge); unreadCountFont_.setBold(true); bubbleDiameter_ = QFontMetrics(unreadCountFont_).averageCharWidth() * 3; timestampFont_ = font_; timestampFont_.setPixelSize(conf::roomlist::fonts::timestamp); timestampFont_.setBold(false); headingFont_ = font_; headingFont_.setPixelSize(conf::roomlist::fonts::heading); headingFont_.setWeight(60); menu_ = new Menu(this); leaveRoom_ = new QAction(tr("Leave room"), this); connect(leaveRoom_, &QAction::triggered, this, [this]() { emit leaveRoom(roomId_); }); menu_->addAction(leaveRoom_); } RoomInfoListItem::RoomInfoListItem(QString room_id, RoomInfo info, QWidget *parent) : QWidget(parent) , roomType_{info.is_invite ? RoomType::Invited : RoomType::Joined} , roomId_(std::move(room_id)) , roomName_{QString::fromStdString(std::move(info.name))} , isPressed_(false) , unreadMsgCount_(0) { init(parent); // HACK // We use fake message info with an old date to pin // the invite events to the top. // // State events in invited rooms don't contain timestamp info, // so we can't use them for sorting. if (roomType_ == RoomType::Invited) lastMsgInfo_ = {"-", "-", "-", "-", QDateTime::currentDateTime().addYears(10)}; } void RoomInfoListItem::resizeEvent(QResizeEvent *) { // Update ripple's clipping path. QPainterPath path; path.addRect(0, 0, width(), height()); if (width() > ui::sidebar::SmallSize) setToolTip(""); else setToolTip(roomName_); ripple_overlay_->setClipPath(path); ripple_overlay_->setClipping(true); } void RoomInfoListItem::paintEvent(QPaintEvent *event) { Q_UNUSED(event); QPainter p(this); p.setRenderHint(QPainter::TextAntialiasing); p.setRenderHint(QPainter::SmoothPixmapTransform); p.setRenderHint(QPainter::Antialiasing); QFontMetrics metrics(font_); QPen titlePen(titleColor_); QPen subtitlePen(subtitleColor_); if (isPressed_) { p.fillRect(rect(), highlightedBackgroundColor_); titlePen.setColor(highlightedTitleColor_); subtitlePen.setColor(highlightedSubtitleColor_); } else if (underMouse()) { p.fillRect(rect(), hoverBackgroundColor_); } else { p.fillRect(rect(), backgroundColor_); } QRect avatarRegion(Padding, Padding, IconSize, IconSize); // Description line with the default font. int bottom_y = MaxHeight - Padding - metrics.ascent() / 2; if (width() > ui::sidebar::SmallSize) { p.setFont(headingFont_); p.setPen(titlePen); const int msgStampWidth = QFontMetrics(timestampFont_).width(lastMsgInfo_.timestamp) + 4; // We use the full width of the widget if there is no unread msg bubble. const int bottomLineWidthLimit = (unreadMsgCount_ > 0) ? msgStampWidth : 0; // Name line. QFontMetrics fontNameMetrics(headingFont_); int top_y = 2 * Padding + fontNameMetrics.ascent() / 2; const auto name = metrics.elidedText(roomName(), Qt::ElideRight, (width() - IconSize - 2 * Padding - msgStampWidth) * 0.8); p.drawText(QPoint(2 * Padding + IconSize, top_y), name); if (roomType_ == RoomType::Joined) { p.setFont(font_); p.setPen(subtitlePen); // The limit is the space between the end of the avatar and the start of the // timestamp. int usernameLimit = std::max(0, width() - 3 * Padding - msgStampWidth - IconSize - 20); auto userName = metrics.elidedText(lastMsgInfo_.username, Qt::ElideRight, usernameLimit); p.setFont(usernameFont_); p.drawText(QPoint(2 * Padding + IconSize, bottom_y), userName); int nameWidth = QFontMetrics(usernameFont_).width(userName); p.setFont(font_); // The limit is the space between the end of the username and the start of // the timestamp. int descriptionLimit = std::max( 0, width() - 3 * Padding - bottomLineWidthLimit - IconSize - nameWidth - 5); auto description = metrics.elidedText(lastMsgInfo_.body, Qt::ElideRight, descriptionLimit); p.drawText(QPoint(2 * Padding + IconSize + nameWidth, bottom_y), description); // We show the last message timestamp. p.save(); if (isPressed_) p.setPen(QPen(highlightedTimestampColor_)); else p.setPen(QPen(timestampColor_)); p.setFont(timestampFont_); p.drawText(QPoint(width() - Padding - msgStampWidth, top_y), lastMsgInfo_.timestamp); p.restore(); } else { int btnWidth = (width() - IconSize - 6 * Padding) / 2; acceptBtnRegion_ = QRectF(InviteBtnX, InviteBtnY, btnWidth, 20); declineBtnRegion_ = QRectF(InviteBtnX + btnWidth + 2 * Padding, InviteBtnY, btnWidth, 20); QPainterPath acceptPath; acceptPath.addRoundedRect(acceptBtnRegion_, 10, 10); p.setPen(Qt::NoPen); p.fillPath(acceptPath, btnColor_); p.drawPath(acceptPath); QPainterPath declinePath; declinePath.addRoundedRect(declineBtnRegion_, 10, 10); p.setPen(Qt::NoPen); p.fillPath(declinePath, btnColor_); p.drawPath(declinePath); p.setPen(QPen(btnTextColor_)); p.setFont(font_); p.drawText(acceptBtnRegion_, Qt::AlignCenter, tr("Accept")); p.drawText(declineBtnRegion_, Qt::AlignCenter, tr("Decline")); } } p.setPen(Qt::NoPen); // We using the first letter of room's name. if (roomAvatar_.isNull()) { QBrush brush; brush.setStyle(Qt::SolidPattern); brush.setColor(avatarBgColor()); p.setPen(Qt::NoPen); p.setBrush(brush); p.drawEllipse(avatarRegion.center(), IconSize / 2, IconSize / 2); p.setFont(bubbleFont_); p.setPen(avatarFgColor()); p.setBrush(Qt::NoBrush); p.drawText( avatarRegion.translated(0, -1), Qt::AlignCenter, utils::firstChar(roomName())); } else { p.save(); QPainterPath path; path.addEllipse(Padding, Padding, IconSize, IconSize); p.setClipPath(path); p.drawPixmap(avatarRegion, roomAvatar_); p.restore(); } if (unreadMsgCount_ > 0) { QBrush brush; brush.setStyle(Qt::SolidPattern); brush.setColor(bubbleBgColor()); if (isPressed_) brush.setColor(bubbleFgColor()); p.setBrush(brush); p.setPen(Qt::NoPen); p.setFont(unreadCountFont_); // Extra space on the x-axis to accomodate the extra character space // inside the bubble. const int x_width = unreadMsgCount_ > MaxUnreadCountDisplayed ? QFontMetrics(p.font()).averageCharWidth() : 0; QRectF r(width() - bubbleDiameter_ - Padding - x_width, bottom_y - bubbleDiameter_ / 2 - 5, bubbleDiameter_ + x_width, bubbleDiameter_); if (width() == ui::sidebar::SmallSize) r = QRectF(width() - bubbleDiameter_ - 5, height() - bubbleDiameter_ - 5, bubbleDiameter_ + x_width, bubbleDiameter_); p.setPen(Qt::NoPen); p.drawEllipse(r); p.setPen(QPen(bubbleFgColor())); if (isPressed_) p.setPen(QPen(bubbleBgColor())); auto countTxt = unreadMsgCount_ > MaxUnreadCountDisplayed ? QString("99+") : QString::number(unreadMsgCount_); p.setBrush(Qt::NoBrush); p.drawText(r.translated(0, -0.5), Qt::AlignCenter, countTxt); } } void RoomInfoListItem::updateUnreadMessageCount(int count) { unreadMsgCount_ = count; update(); } void RoomInfoListItem::setPressedState(bool state) { if (isPressed_ != state) { isPressed_ = state; update(); } } void RoomInfoListItem::contextMenuEvent(QContextMenuEvent *event) { Q_UNUSED(event); if (roomType_ == RoomType::Invited) return; menu_->popup(event->globalPos()); } void RoomInfoListItem::mousePressEvent(QMouseEvent *event) { if (event->buttons() == Qt::RightButton) { QWidget::mousePressEvent(event); return; } if (roomType_ == RoomType::Invited) { const auto point = event->pos(); if (acceptBtnRegion_.contains(point)) emit acceptInvite(roomId_); if (declineBtnRegion_.contains(point)) emit declineInvite(roomId_); return; } emit clicked(roomId_); setPressedState(true); // Ripple on mouse position by default. QPoint pos = event->pos(); qreal radiusEndValue = static_cast(width()) / 3; Ripple *ripple = new Ripple(pos); ripple->setRadiusEndValue(radiusEndValue); ripple->setOpacityStartValue(0.15); ripple->setColor(QColor("white")); ripple->radiusAnimation()->setDuration(200); ripple->opacityAnimation()->setDuration(400); ripple_overlay_->addRipple(ripple); } void RoomInfoListItem::setAvatar(const QImage &img) { roomAvatar_ = utils::scaleImageToPixmap(img, IconSize); update(); } void RoomInfoListItem::setDescriptionMessage(const DescInfo &info) { lastMsgInfo_ = info; update(); }