/* * 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 "Avatar.h" #include "ChatPage.h" #include "Config.h" #include "timeline/TimelineItem.h" #include "timeline/widgets/AudioItem.h" #include "timeline/widgets/FileItem.h" #include "timeline/widgets/ImageItem.h" #include "timeline/widgets/VideoItem.h" constexpr const static char *CHECKMARK = "✓"; void TimelineItem::init() { userAvatar_ = nullptr; timestamp_ = nullptr; userName_ = nullptr; body_ = nullptr; font_.setPixelSize(conf::fontSize); QFontMetrics fm(font_); receiptsMenu_ = new QMenu(this); showReadReceipts_ = new QAction("Read receipts", this); receiptsMenu_->addAction(showReadReceipts_); connect(showReadReceipts_, &QAction::triggered, this, [this]() { if (!event_id_.isEmpty()) ChatPage::instance()->showReadReceipts(event_id_); }); topLayout_ = new QHBoxLayout(this); mainLayout_ = new QVBoxLayout; messageLayout_ = new QHBoxLayout; topLayout_->setContentsMargins(conf::timeline::msgMargin, conf::timeline::msgMargin, 0, 0); topLayout_->setSpacing(0); topLayout_->addLayout(mainLayout_, 1); mainLayout_->setContentsMargins(conf::timeline::headerLeftMargin, 0, 0, 0); mainLayout_->setSpacing(0); QFont checkmarkFont; checkmarkFont.setPixelSize(conf::timeline::fonts::timestamp); // Setting fixed width for checkmark because systems may have a differing width for a // space and the Unicode checkmark. checkmark_ = new QLabel(" ", this); checkmark_->setFont(checkmarkFont); checkmark_->setFixedWidth(QFontMetrics{checkmarkFont}.width(CHECKMARK)); } /* * For messages created locally. */ TimelineItem::TimelineItem(mtx::events::MessageType ty, const QString &userid, QString body, bool withSender, QWidget *parent) : QWidget(parent) { init(); auto displayName = TimelineViewManager::displayName(userid); auto timestamp = QDateTime::currentDateTime(); if (ty == mtx::events::MessageType::Emote) { body = QString("* %1 %2").arg(displayName).arg(body); descriptionMsg_ = {"", userid, body, utils::descriptiveTime(timestamp), timestamp}; } else { descriptionMsg_ = { "You: ", userid, body, utils::descriptiveTime(timestamp), timestamp}; } body = body.toHtmlEscaped(); body.replace(conf::strings::url_regex, conf::strings::url_html); body.replace("\n", "
"); generateTimestamp(timestamp); messageLayout_->setContentsMargins(0, 0, 20, 4); messageLayout_->setSpacing(20); if (withSender) { generateBody(displayName, body); setupAvatarLayout(displayName); messageLayout_->addLayout(headerLayout_, 1); AvatarProvider::resolve(userid, [this](const QImage &img) { setUserAvatar(img); }); } else { generateBody(body); setupSimpleLayout(); messageLayout_->addWidget(body_, 1); } messageLayout_->addWidget(checkmark_); messageLayout_->addWidget(timestamp_); mainLayout_->addLayout(messageLayout_); } TimelineItem::TimelineItem(ImageItem *image, const QString &userid, bool withSender, QWidget *parent) : QWidget{parent} { init(); setupLocalWidgetLayout(image, userid, "sent an image", withSender); } TimelineItem::TimelineItem(FileItem *file, const QString &userid, bool withSender, QWidget *parent) : QWidget{parent} { init(); setupLocalWidgetLayout(file, userid, "sent a file", withSender); } TimelineItem::TimelineItem(AudioItem *audio, const QString &userid, bool withSender, QWidget *parent) : QWidget{parent} { init(); setupLocalWidgetLayout(audio, userid, "sent an audio clip", withSender); } TimelineItem::TimelineItem(VideoItem *video, const QString &userid, bool withSender, QWidget *parent) : QWidget{parent} { init(); setupLocalWidgetLayout(video, userid, "sent a video clip", withSender); } TimelineItem::TimelineItem(ImageItem *image, const mtx::events::RoomEvent &event, bool with_sender, QWidget *parent) : QWidget(parent) { setupWidgetLayout, ImageItem>( image, event, " sent an image", with_sender); } TimelineItem::TimelineItem(FileItem *file, const mtx::events::RoomEvent &event, bool with_sender, QWidget *parent) : QWidget(parent) { setupWidgetLayout, FileItem>( file, event, " sent a file", with_sender); } TimelineItem::TimelineItem(AudioItem *audio, const mtx::events::RoomEvent &event, bool with_sender, QWidget *parent) : QWidget(parent) { setupWidgetLayout, AudioItem>( audio, event, " sent an audio clip", with_sender); } TimelineItem::TimelineItem(VideoItem *video, const mtx::events::RoomEvent &event, bool with_sender, QWidget *parent) : QWidget(parent) { setupWidgetLayout, VideoItem>( video, event, " sent a video clip", with_sender); } /* * Used to display remote notice messages. */ TimelineItem::TimelineItem(const mtx::events::RoomEvent &event, bool with_sender, QWidget *parent) : QWidget(parent) { init(); event_id_ = QString::fromStdString(event.event_id); const auto sender = QString::fromStdString(event.sender); const auto timestamp = QDateTime::fromMSecsSinceEpoch(event.origin_server_ts); auto body = QString::fromStdString(event.content.body).trimmed().toHtmlEscaped(); descriptionMsg_ = {TimelineViewManager::displayName(sender), sender, " sent a notification", utils::descriptiveTime(timestamp), timestamp}; generateTimestamp(timestamp); body.replace(conf::strings::url_regex, conf::strings::url_html); body.replace("\n", "
"); body = "" + body + ""; messageLayout_->setContentsMargins(0, 0, 20, 4); messageLayout_->setSpacing(20); if (with_sender) { auto displayName = TimelineViewManager::displayName(sender); generateBody(displayName, body); setupAvatarLayout(displayName); messageLayout_->addLayout(headerLayout_, 1); AvatarProvider::resolve(sender, [this](const QImage &img) { setUserAvatar(img); }); } else { generateBody(body); setupSimpleLayout(); messageLayout_->addWidget(body_, 1); } messageLayout_->addWidget(checkmark_); messageLayout_->addWidget(timestamp_); mainLayout_->addLayout(messageLayout_); } /* * Used to display remote emote messages. */ TimelineItem::TimelineItem(const mtx::events::RoomEvent &event, bool with_sender, QWidget *parent) : QWidget(parent) { init(); event_id_ = QString::fromStdString(event.event_id); const auto sender = QString::fromStdString(event.sender); auto body = QString::fromStdString(event.content.body).trimmed(); auto timestamp = QDateTime::fromMSecsSinceEpoch(event.origin_server_ts); auto displayName = TimelineViewManager::displayName(sender); auto emoteMsg = QString("* %1 %2").arg(displayName).arg(body); descriptionMsg_ = {"", sender, emoteMsg, utils::descriptiveTime(timestamp), timestamp}; generateTimestamp(timestamp); emoteMsg = emoteMsg.toHtmlEscaped(); emoteMsg.replace(conf::strings::url_regex, conf::strings::url_html); emoteMsg.replace("\n", "
"); messageLayout_->setContentsMargins(0, 0, 20, 4); messageLayout_->setSpacing(20); if (with_sender) { generateBody(displayName, emoteMsg); setupAvatarLayout(displayName); messageLayout_->addLayout(headerLayout_, 1); AvatarProvider::resolve(sender, [this](const QImage &img) { setUserAvatar(img); }); } else { generateBody(emoteMsg); setupSimpleLayout(); messageLayout_->addWidget(body_, 1); } messageLayout_->addWidget(checkmark_); messageLayout_->addWidget(timestamp_); mainLayout_->addLayout(messageLayout_); } /* * Used to display remote text messages. */ TimelineItem::TimelineItem(const mtx::events::RoomEvent &event, bool with_sender, QWidget *parent) : QWidget(parent) { init(); event_id_ = QString::fromStdString(event.event_id); const auto sender = QString::fromStdString(event.sender); auto body = QString::fromStdString(event.content.body).trimmed(); auto timestamp = QDateTime::fromMSecsSinceEpoch(event.origin_server_ts); auto displayName = TimelineViewManager::displayName(sender); QSettings settings; descriptionMsg_ = {sender == settings.value("auth/user_id") ? "You" : displayName, sender, QString(": %1").arg(body), utils::descriptiveTime(timestamp), timestamp}; generateTimestamp(timestamp); body = body.toHtmlEscaped(); body.replace(conf::strings::url_regex, conf::strings::url_html); body.replace("\n", "
"); messageLayout_->setContentsMargins(0, 0, 20, 4); messageLayout_->setSpacing(20); if (with_sender) { generateBody(displayName, body); setupAvatarLayout(displayName); messageLayout_->addLayout(headerLayout_, 1); AvatarProvider::resolve(sender, [this](const QImage &img) { setUserAvatar(img); }); } else { generateBody(body); setupSimpleLayout(); messageLayout_->addWidget(body_, 1); } messageLayout_->addWidget(checkmark_); messageLayout_->addWidget(timestamp_); mainLayout_->addLayout(messageLayout_); } void TimelineItem::markReceived() { checkmark_->setText(CHECKMARK); checkmark_->setAlignment(Qt::AlignTop); } // Only the body is displayed. void TimelineItem::generateBody(const QString &body) { QString content("%1"); body_ = new QLabel(this); body_->setFont(font_); body_->setWordWrap(true); body_->setText(content.arg(replaceEmoji(body))); body_->setMargin(0); body_->setTextInteractionFlags(Qt::TextSelectableByMouse | Qt::TextBrowserInteraction); body_->setOpenExternalLinks(true); } // The username/timestamp is displayed along with the message body. void TimelineItem::generateBody(const QString &userid, const QString &body) { auto sender = userid; if (userid.startsWith("@")) { // TODO: Fix this by using a UserId type. if (userid.split(":")[0].split("@").size() > 1) sender = userid.split(":")[0].split("@")[1]; } QFont usernameFont = font_; usernameFont.setWeight(60); QFontMetrics fm(usernameFont); userName_ = new QLabel(this); userName_->setFont(usernameFont); userName_->setText(fm.elidedText(sender, Qt::ElideRight, 500)); if (body.isEmpty()) return; body_ = new QLabel(this); body_->setFont(font_); body_->setWordWrap(true); body_->setText(QString("%1").arg(replaceEmoji(body))); body_->setTextInteractionFlags(Qt::TextSelectableByMouse | Qt::TextBrowserInteraction); body_->setOpenExternalLinks(true); body_->setMargin(0); } void TimelineItem::generateTimestamp(const QDateTime &time) { QFont timestampFont; timestampFont.setPixelSize(conf::timeline::fonts::timestamp); QFontMetrics fm(timestampFont); int topMargin = QFontMetrics(font_).ascent() - fm.ascent(); timestamp_ = new QLabel(this); timestamp_->setAlignment(Qt::AlignTop); timestamp_->setFont(timestampFont); timestamp_->setText( QString(" %1 ").arg(time.toString("HH:mm"))); timestamp_->setContentsMargins(0, topMargin, 0, 0); timestamp_->setStyleSheet( QString("font-size: %1px;").arg(conf::timeline::fonts::timestamp)); } QString TimelineItem::replaceEmoji(const QString &body) { QString fmtBody = ""; QVector utf32_string = body.toUcs4(); for (auto &code : utf32_string) { // TODO: Be more precise here. if (code > 9000) fmtBody += QString("") .arg(conf::emojiSize) + QString::fromUcs4(&code, 1) + ""; else fmtBody += QString::fromUcs4(&code, 1); } return fmtBody; } void TimelineItem::setupAvatarLayout(const QString &userName) { topLayout_->setContentsMargins(conf::timeline::msgMargin, conf::timeline::msgMargin, 0, 0); userAvatar_ = new Avatar(this); userAvatar_->setLetter(QChar(userName[0]).toUpper()); userAvatar_->setSize(conf::timeline::avatarSize); // TODO: The provided user name should be a UserId class if (userName[0] == '@' && userName.size() > 1) userAvatar_->setLetter(QChar(userName[1]).toUpper()); sideLayout_ = new QVBoxLayout; sideLayout_->setMargin(0); sideLayout_->setSpacing(0); sideLayout_->addWidget(userAvatar_); sideLayout_->addStretch(1); topLayout_->insertLayout(0, sideLayout_); headerLayout_ = new QVBoxLayout; headerLayout_->setMargin(0); headerLayout_->setSpacing(0); headerLayout_->addWidget(userName_); headerLayout_->addWidget(body_); } void TimelineItem::setupSimpleLayout() { topLayout_->setContentsMargins(conf::timeline::avatarSize + conf::timeline::msgMargin + 1, conf::timeline::msgMargin / 3, 0, 0); } void TimelineItem::setUserAvatar(const QImage &avatar) { if (userAvatar_ == nullptr) return; userAvatar_->setImage(avatar); } void TimelineItem::contextMenuEvent(QContextMenuEvent *event) { if (receiptsMenu_) receiptsMenu_->exec(event->globalPos()); } void TimelineItem::paintEvent(QPaintEvent *) { QStyleOption opt; opt.init(this); QPainter p(this); style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this); }