diff --git a/src/RoomInfoListItem.cpp b/src/RoomInfoListItem.cpp
new file mode 100644
index 00000000..172cdb90
--- /dev/null
+++ b/src/RoomInfoListItem.cpp
@@ -0,0 +1,390 @@
+/*
+ * 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 <QDateTime>
+#include <QDebug>
+#include <QMouseEvent>
+#include <QPainter>
+
+#include <variant.hpp>
+
+#include "Cache.h"
+#include "Config.h"
+#include "RoomInfoListItem.h"
+#include "Utils.h"
+#include "ui/Menu.h"
+#include "ui/Ripple.h"
+#include "ui/RippleOverlay.h"
+#include "ui/Theme.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<qreal>(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();
+}
|