File indexing completed on 2025-10-19 04:49:14
0001 /* 0002 SPDX-FileCopyrightText: 2020-2024 Laurent Montel <montel@kde.org> 0003 0004 SPDX-License-Identifier: LGPL-2.0-or-later 0005 */ 0006 0007 #include "messagelistview.h" 0008 #include "administratordialog/moderationconsole/moderationmessageinfodialog.h" 0009 #include "chat/followmessagejob.h" 0010 #include "chat/unfollowmessagejob.h" 0011 0012 #include "moderation/moderationdismissreportsjob.h" 0013 0014 #include "connection.h" 0015 #include "delegate/messagelistdelegate.h" 0016 #include "dialogs/directchannelinfodialog.h" 0017 #include "dialogs/reportmessagedialog.h" 0018 #include "moderation/moderationreportsjob.h" 0019 #include "rocketchataccount.h" 0020 #include "room.h" 0021 #include "roomutil.h" 0022 #include "ruqola.h" 0023 #include "ruqola_translatemessage_debug.h" 0024 #include "ruqolawidgets_debug.h" 0025 #include "selectedmessagebackgroundanimation.h" 0026 #include "threadwidget/threadmessagedialog.h" 0027 0028 #include <KLocalizedString> 0029 #include <KMessageBox> 0030 0031 #include <QApplication> 0032 #include <QClipboard> 0033 #include <QIcon> 0034 #include <QKeyEvent> 0035 #include <QMenu> 0036 #include <QPainter> 0037 #include <QScrollBar> 0038 0039 #include "config-ruqola.h" 0040 0041 #if HAVE_TEXT_TRANSLATOR 0042 #include "translatetext/translatetextjob.h" 0043 #include "translatetext/translatorenginemanager.h" 0044 #include <texttranslator_version.h> 0045 0046 #include <TextTranslator/TranslatorMenu> 0047 #endif 0048 0049 MessageListView::MessageListView(RocketChatAccount *account, Mode mode, QWidget *parent) 0050 : MessageListViewBase(parent) 0051 , mMode(mode) 0052 , mMessageListDelegate(new MessageListDelegate(account, this)) 0053 , mCurrentRocketChatAccount(account) 0054 { 0055 if (mCurrentRocketChatAccount) { 0056 mMessageListDelegate->setRocketChatAccount(mCurrentRocketChatAccount); 0057 } 0058 mMessageListDelegate->setShowThreadContext(mMode != Mode::ThreadEditing); 0059 mMessageListDelegate->setEnableEmojiMenu(mMode != Mode::Moderation); 0060 setItemDelegate(mMessageListDelegate); 0061 0062 connect(verticalScrollBar(), &QScrollBar::valueChanged, this, &MessageListView::slotVerticalScrollbarChanged); 0063 0064 // ensure the scrolling behavior isn't jumpy 0065 // we always single step by roughly one line 0066 const auto lineHeight = fontMetrics().height() + 10; 0067 verticalScrollBar()->setSingleStep(lineHeight); 0068 // the page step depends on the height of the viewport and needs to be reset when the range changes 0069 // as Qt would otherwise overwrite it internally. We apparently need a queued connection too to ensure our value is set 0070 connect(verticalScrollBar(), &QScrollBar::rangeChanged, this, &MessageListView::updateVerticalPageStep, Qt::QueuedConnection); 0071 updateVerticalPageStep(); 0072 connect(mMessageListDelegate, &MessageListDelegate::showUserInfo, this, &MessageListView::slotShowUserInfo); 0073 connect(mMessageListDelegate, &MessageListDelegate::startPrivateConversation, this, &MessageListView::slotStartPrivateConversation); 0074 connect(mMessageListDelegate, &MessageListDelegate::updateView, this, &MessageListView::slotUpdateView); 0075 connect(this, &MessageListView::needToClearSizeHintCache, mMessageListDelegate, &MessageListDelegate::clearSizeHintCache); 0076 } 0077 0078 MessageListView::~MessageListView() = default; 0079 0080 void MessageListView::wheelEvent(QWheelEvent *e) 0081 { 0082 const int y = e->angleDelta().y(); 0083 if (y > 0) { 0084 if (verticalScrollBar()->value() <= 1) { 0085 Q_EMIT loadHistoryRequested(); 0086 } 0087 } 0088 MessageListViewBase::wheelEvent(e); 0089 } 0090 0091 void MessageListView::paintEvent(QPaintEvent *e) 0092 { 0093 if (mRoom && (mRoom->numberMessages() == 0)) { 0094 QPainter p(viewport()); 0095 0096 QFont font = p.font(); 0097 font.setItalic(true); 0098 p.setFont(font); 0099 if (mRoom->channelType() == Room::RoomType::Direct) { 0100 p.drawText(QRect(0, 0, width(), height()), Qt::AlignHCenter | Qt::AlignTop, i18n("You have joined a new direct message")); 0101 } else { 0102 p.drawText(QRect(0, 0, width(), height()), Qt::AlignHCenter | Qt::AlignTop, i18n("Start of conversation")); 0103 } 0104 } else { 0105 QListView::paintEvent(e); 0106 } 0107 } 0108 0109 void MessageListView::slotUpdateView() 0110 { 0111 viewport()->update(); 0112 } 0113 0114 void MessageListView::setRoom(Room *room) 0115 { 0116 if (mRoom) { 0117 disconnect(mRoom, &Room::lastSeenChanged, this, &MessageListView::slotUpdateView); 0118 mMessageListDelegate->clearSelection(); 0119 } 0120 mRoom = room; 0121 if (mRoom) { 0122 connect(mRoom, &Room::lastSeenChanged, this, &MessageListView::slotUpdateView); 0123 } 0124 } 0125 0126 void MessageListView::slotVerticalScrollbarChanged(int value) 0127 { 0128 if (value == 0) { 0129 Q_EMIT loadHistoryRequested(); 0130 // Perhaps finding a better method. 0131 verticalScrollBar()->setValue(1); // If we are at 0 we can't continue to load history 0132 } 0133 } 0134 0135 void MessageListView::goToMessage(const QString &messageId) 0136 { 0137 auto messageModel = qobject_cast<MessagesModel *>(model()); 0138 Q_ASSERT(messageModel); 0139 const QModelIndex index = messageModel->indexForMessage(messageId); 0140 if (index.isValid()) { 0141 scrollTo(index); 0142 } else { 0143 qCWarning(RUQOLAWIDGETS_LOG) << "Message not found:" << messageId; 0144 } 0145 } 0146 0147 void MessageListView::setChannelSelected(Room *room) 0148 { 0149 auto oldModel = qobject_cast<MessagesModel *>(model()); 0150 if (oldModel) { 0151 oldModel->deactivate(); 0152 } 0153 setRoom(room); 0154 const QString roomId = room->roomId(); 0155 mCurrentRocketChatAccount->switchingToRoom(roomId); 0156 MessagesModel *model = mCurrentRocketChatAccount->messageModelForRoom(roomId); 0157 setModel(model); 0158 model->activate(); 0159 } 0160 0161 void MessageListView::setModel(QAbstractItemModel *newModel) 0162 { 0163 QAbstractItemModel *oldModel = model(); 0164 if (oldModel) { 0165 disconnect(oldModel, nullptr, this, nullptr); 0166 } 0167 QListView::setModel(newModel); 0168 connect(newModel, &QAbstractItemModel::rowsAboutToBeInserted, this, &MessageListView::checkIfAtBottom); 0169 connect(newModel, &QAbstractItemModel::rowsAboutToBeRemoved, this, &MessageListView::checkIfAtBottom); 0170 connect(newModel, &QAbstractItemModel::modelAboutToBeReset, this, &MessageListView::checkIfAtBottom); 0171 // Connect to rangeChanged rather than rowsInserted/rowsRemoved/modelReset. 0172 // This way it also catches the case of an item changing height (e.g. after async image loading) 0173 connect(verticalScrollBar(), &QScrollBar::rangeChanged, this, &MessageListView::maybeScrollToBottom); 0174 0175 connect(newModel, &QAbstractItemModel::rowsInserted, this, &MessageListView::modelChanged); 0176 connect(newModel, &QAbstractItemModel::rowsRemoved, this, &MessageListView::modelChanged); 0177 connect(newModel, &QAbstractItemModel::modelReset, this, &MessageListView::modelChanged); 0178 // Clear document cache when message is updated otherwise image description is not up to date 0179 connect(newModel, &QAbstractItemModel::dataChanged, this, [this](const QModelIndex &topLeft, const QModelIndex &, const QVector<int> &roles) { 0180 if (roles.contains(MessagesModel::OriginalMessageOrAttachmentDescription)) { 0181 const Message *message = topLeft.data(MessagesModel::MessagePointer).value<Message *>(); 0182 if (message) { 0183 mMessageListDelegate->removeMessageCache(message); 0184 } 0185 } else if (roles.contains(MessagesModel::DisplayUrlPreview) || roles.contains(MessagesModel::DisplayAttachment)) { 0186 const Message *message = topLeft.data(MessagesModel::MessagePointer).value<Message *>(); 0187 if (message) { 0188 mMessageListDelegate->removeSizeHintCache(message->messageId()); 0189 } 0190 } 0191 }); 0192 0193 scrollToBottom(); 0194 } 0195 0196 void MessageListView::handleKeyPressEvent(QKeyEvent *ev) 0197 { 0198 const int key = ev->key(); 0199 if (key == Qt::Key_Up || key == Qt::Key_Down || key == Qt::Key_PageDown || key == Qt::Key_PageUp) { 0200 // QListView/QAIV PageUp/PageDown moves the current item, first inside visible bounds 0201 // before it triggers scrolling around. Let's just let the scrollarea handle it, 0202 // since we don't show the current item. 0203 QAbstractScrollArea::keyPressEvent(ev); 0204 ev->accept(); 0205 } else if (ev->modifiers() & Qt::ControlModifier) { 0206 if (key == Qt::Key_Home) { 0207 scrollToTop(); 0208 ev->accept(); 0209 } else if (key == Qt::Key_End) { 0210 scrollToBottom(); 0211 ev->accept(); 0212 } 0213 } 0214 } 0215 0216 void MessageListView::createTranslorMenu() 0217 { 0218 #if HAVE_TEXT_TRANSLATOR 0219 if (!mTranslatorMenu) { 0220 mTranslatorMenu = new TextTranslator::TranslatorMenu(this); 0221 connect(mTranslatorMenu, &TextTranslator::TranslatorMenu::translate, this, &MessageListView::slotTranslate); 0222 connect(Ruqola::self(), &Ruqola::translatorMenuChanged, this, [this]() { 0223 TranslatorEngineManager::self()->translatorConfigChanged(); 0224 mTranslatorMenu->updateMenu(); 0225 }); 0226 } 0227 #endif 0228 } 0229 0230 void MessageListView::contextMenuEvent(QContextMenuEvent *event) 0231 { 0232 // if (!mRoom) { 0233 // return; 0234 // } 0235 const QModelIndex index = indexAt(event->pos()); 0236 if (!index.isValid()) { 0237 if (Ruqola::self()->debug()) { 0238 QMenu menu(this); 0239 addDebugMenu(menu, index); 0240 menu.exec(event->globalPos()); 0241 } 0242 return; 0243 } 0244 0245 auto options = listViewOptions(); 0246 options.rect = visualRect(index); 0247 options.index = index; 0248 MessageListDelegate::MenuInfo info; 0249 info.editMode = (mMode == Mode::Editing); 0250 info.globalPos = event->globalPos(); 0251 info.pos = viewport()->mapFromGlobal(event->globalPos()); 0252 info.roomType = mRoom ? mRoom->channelType() : Room::RoomType::Unknown; 0253 if (mMessageListDelegate->contextMenu(options, index, info)) { 0254 return; 0255 } 0256 const auto messageType = index.data(MessagesModel::MessageType).value<Message::MessageType>(); 0257 const bool isSystemMessage = (messageType == Message::System) || (messageType == Message::Information) || (messageType == Message::VideoConference); 0258 QMenu menu(this); 0259 if (isSystemMessage) { 0260 if (Ruqola::self()->debug()) { 0261 addDebugMenu(menu, index); 0262 menu.exec(event->globalPos()); 0263 } 0264 return; 0265 } 0266 mMessageListDelegate->attachmentContextMenu(options, index, info, &menu); 0267 const bool isNotOwnerOfMessage = (index.data(MessagesModel::UserId).toString() != mCurrentRocketChatAccount->userId()); 0268 0269 auto copyAction = new QAction(QIcon::fromTheme(QStringLiteral("edit-copy")), i18n("Copy Message"), &menu); 0270 copyAction->setShortcut(QKeySequence::Copy); 0271 connect(copyAction, &QAction::triggered, this, [this, index]() { 0272 copyMessageToClipboard(index); 0273 }); 0274 QAction *setPinnedMessage = nullptr; 0275 if (mCurrentRocketChatAccount->allowMessagePinningEnabled() && mRoom && mRoom->allowToPinMessage()) { 0276 const bool isPinned = index.data(MessagesModel::Pinned).toBool(); 0277 setPinnedMessage = new QAction(QIcon::fromTheme(QStringLiteral("pin")), isPinned ? i18n("Unpin Message") : i18n("Pin Message"), &menu); 0278 connect(setPinnedMessage, &QAction::triggered, this, [this, isPinned, index]() { 0279 slotSetPinnedMessage(index, isPinned); 0280 }); 0281 } 0282 QAction *setAsFavoriteAction = nullptr; 0283 if (mCurrentRocketChatAccount->allowMessageStarringEnabled()) { 0284 const bool isStarred = index.data(MessagesModel::Starred).toBool(); 0285 setAsFavoriteAction = 0286 new QAction(QIcon::fromTheme(QStringLiteral("favorite")), isStarred ? i18n("Remove as Favorite") : i18n("Set as Favorite"), &menu); 0287 connect(setAsFavoriteAction, &QAction::triggered, this, [this, isStarred, index]() { 0288 slotSetAsFavorite(index, isStarred); 0289 }); 0290 } 0291 QAction *deleteAction = nullptr; 0292 if (index.data(MessagesModel::CanDeleteMessage).toBool()) { 0293 deleteAction = new QAction(QIcon::fromTheme(QStringLiteral("edit-delete")), i18n("Delete"), &menu); 0294 connect(deleteAction, &QAction::triggered, this, [this, index]() { 0295 slotDeleteMessage(index); 0296 }); 0297 } 0298 0299 auto selectAllAction = new QAction(i18n("Select All"), &menu); 0300 connect(selectAllAction, &QAction::triggered, this, [this, index]() { 0301 slotSelectAll(index); 0302 }); 0303 0304 auto markMessageAsUnReadAction = new QAction(i18n("Mark Message As Unread"), &menu); 0305 connect(markMessageAsUnReadAction, &QAction::triggered, this, [this, index]() { 0306 slotMarkMessageAsUnread(index); 0307 }); 0308 0309 auto showFullThreadAction = new QAction(i18n("Show Full Thread"), &menu); 0310 connect(showFullThreadAction, &QAction::triggered, this, [this, index]() { 0311 slotShowFullThread(index); 0312 }); 0313 0314 auto editAction = new QAction(QIcon::fromTheme(QStringLiteral("document-edit")), i18n("Edit"), &menu); 0315 connect(editAction, &QAction::triggered, this, [this, index]() { 0316 slotEditMessage(index); 0317 }); 0318 0319 auto quoteAction = new QAction(QIcon::fromTheme(QStringLiteral("format-text-blockquote")), i18n("Quote"), &menu); 0320 connect(quoteAction, &QAction::triggered, this, [this, index]() { 0321 slotQuoteMessage(index); 0322 }); 0323 0324 auto copyLinkToMessageAction = new QAction(i18n("Copy Link To Message"), &menu); // TODO add icon 0325 connect(copyLinkToMessageAction, &QAction::triggered, this, [this, index]() { 0326 slotCopyLinkToMessage(index); 0327 }); 0328 0329 const Message *message = index.data(MessagesModel::MessagePointer).value<Message *>(); 0330 0331 const QString threadMessageId = index.data(MessagesModel::ThreadMessageId).toString(); 0332 const bool messageIsFollowing = threadMessageId.isEmpty() ? message->replies().contains(mCurrentRocketChatAccount->userId()) 0333 : index.data(MessagesModel::ThreadMessageFollowed).toBool(); 0334 0335 const auto followingToMessageAction = 0336 new QAction(messageIsFollowing ? QIcon::fromTheme(QStringLiteral("notifications-disabled")) : QIcon::fromTheme(QStringLiteral("notifications")), 0337 messageIsFollowing ? i18n("Unfollow Message") : i18n("Follow Message"), 0338 &menu); 0339 connect(followingToMessageAction, &QAction::triggered, this, [this, index, messageIsFollowing]() { 0340 slotFollowMessage(index, messageIsFollowing); 0341 }); 0342 0343 auto copyUrlAction = [&]() -> QAction * { 0344 auto options = listViewOptions(); 0345 options.rect = visualRect(index); 0346 options.index = index; 0347 const QString url = mMessageListDelegate->urlAt(options, index, viewport()->mapFromGlobal(event->globalPos())); 0348 if (url.isEmpty() || url.startsWith(QStringLiteral("ruqola:/"))) 0349 return nullptr; 0350 auto action = new QAction(QIcon::fromTheme(QStringLiteral("edit-copy")), i18n("Copy URL"), &menu); 0351 connect(action, &QAction::triggered, this, [url]() { 0352 QGuiApplication::clipboard()->setText(url); 0353 }); 0354 return action; 0355 }(); 0356 0357 auto userInfoActions = [&]() -> QList<QAction *> { 0358 QList<QAction *> listActions; 0359 auto options = listViewOptions(); 0360 options.rect = visualRect(index); 0361 options.index = index; 0362 QString url = mMessageListDelegate->urlAt(options, index, viewport()->mapFromGlobal(event->globalPos())); 0363 if (url.isEmpty()) 0364 return {}; 0365 if (url.startsWith(QLatin1String("ruqola:/user/"))) { 0366 url.remove(QStringLiteral("ruqola:/user/")); 0367 if (!RoomUtil::validUser(url)) { 0368 return {}; 0369 } 0370 } else { 0371 return {}; 0372 } 0373 auto action = new QAction(QIcon::fromTheme(QStringLiteral("documentinfo")), i18n("User Info"), &menu); 0374 connect(action, &QAction::triggered, this, [url, this]() { 0375 slotShowUserInfo(url); 0376 }); 0377 listActions.append(action); 0378 if (info.editMode) { 0379 if (info.roomType != Room::RoomType::Direct) { 0380 if (mCurrentRocketChatAccount->hasPermission(QStringLiteral("create-d"))) { 0381 auto startPrivateConversationAction = new QAction(i18n("Start a Private Conversation"), &menu); 0382 connect(startPrivateConversationAction, &QAction::triggered, this, [this, url]() { 0383 slotStartPrivateConversation(url); 0384 }); 0385 auto separator = new QAction(&menu); 0386 separator->setSeparator(true); 0387 listActions.append(separator); 0388 listActions.append(startPrivateConversationAction); 0389 } 0390 } 0391 } 0392 return listActions; 0393 }(); 0394 0395 switch (mMode) { 0396 case Mode::Editing: { 0397 auto startDiscussion = new QAction(i18n("Start a Discussion"), &menu); 0398 connect(startDiscussion, &QAction::triggered, this, [this, index]() { 0399 slotStartDiscussion(index); 0400 }); 0401 menu.addAction(startDiscussion); 0402 menu.addSeparator(); 0403 if (mCurrentRocketChatAccount->threadsEnabled()) { 0404 auto replyInThreadAction = new QAction(i18n("Reply in Thread"), &menu); 0405 connect(replyInThreadAction, &QAction::triggered, this, [this, index]() { 0406 slotReplyInThread(index); 0407 }); 0408 menu.addAction(replyInThreadAction); 0409 0410 const QString threadMessageId = index.data(MessagesModel::ThreadMessageId).toString(); 0411 const int threadMessageCount = index.data(MessagesModel::ThreadCount).toInt(); 0412 if (!threadMessageId.isEmpty() || threadMessageCount > 0) { 0413 menu.addSeparator(); 0414 menu.addAction(showFullThreadAction); 0415 } 0416 } 0417 menu.addSeparator(); 0418 menu.addAction(quoteAction); 0419 menu.addSeparator(); 0420 if (setPinnedMessage) { 0421 menu.addAction(setPinnedMessage); 0422 } 0423 if (setAsFavoriteAction) { 0424 menu.addAction(setAsFavoriteAction); 0425 } 0426 menu.addSeparator(); 0427 0428 if (index.data(MessagesModel::CanEditMessage).toBool()) { 0429 menu.addAction(editAction); 0430 menu.addSeparator(); 0431 } 0432 menu.addAction(copyAction); 0433 if (copyUrlAction) { 0434 menu.addAction(copyUrlAction); 0435 } 0436 menu.addAction(copyLinkToMessageAction); 0437 menu.addSeparator(); 0438 menu.addAction(selectAllAction); 0439 0440 menu.addSeparator(); 0441 if (isNotOwnerOfMessage) { 0442 menu.addAction(markMessageAsUnReadAction); 0443 } 0444 0445 menu.addSeparator(); 0446 menu.addAction(followingToMessageAction); 0447 0448 #if HAVE_TEXT_TRANSLATOR 0449 createTranslorMenu(); 0450 if (!mTranslatorMenu->isEmpty()) { 0451 menu.addSeparator(); 0452 mTranslatorMenu->setModelIndex(index); 0453 menu.addMenu(mTranslatorMenu->menu()); 0454 } 0455 #endif 0456 0457 if (deleteAction) { 0458 menu.addSeparator(); 0459 menu.addAction(deleteAction); 0460 } 0461 if (mCurrentRocketChatAccount->hasAutotranslateSupport() || !message->localTranslation().isEmpty()) { 0462 createSeparator(menu); 0463 const bool isTranslated = message->showTranslatedMessage(); 0464 auto translateAction = new QAction(isTranslated ? i18n("Show Original Message") : i18n("Translate Message"), &menu); 0465 connect(translateAction, &QAction::triggered, this, [this, index, isTranslated]() { 0466 slotTranslateMessage(index, !isTranslated); 0467 }); 0468 menu.addAction(translateAction); 0469 } 0470 break; 0471 } 0472 case Mode::ThreadEditing: { 0473 if (setPinnedMessage) { 0474 menu.addAction(setPinnedMessage); 0475 } 0476 if (setAsFavoriteAction) { 0477 menu.addAction(setAsFavoriteAction); 0478 } 0479 0480 menu.addSeparator(); 0481 menu.addAction(quoteAction); 0482 menu.addSeparator(); 0483 menu.addAction(copyAction); 0484 if (copyUrlAction) { 0485 menu.addAction(copyUrlAction); 0486 } 0487 menu.addAction(copyLinkToMessageAction); 0488 menu.addSeparator(); 0489 menu.addAction(selectAllAction); 0490 if (isNotOwnerOfMessage) { 0491 menu.addAction(markMessageAsUnReadAction); 0492 menu.addSeparator(); 0493 } 0494 if (index.data(MessagesModel::CanEditMessage).toBool()) { 0495 menu.addSeparator(); 0496 menu.addAction(editAction); 0497 } 0498 0499 if (deleteAction) { 0500 menu.addSeparator(); 0501 menu.addAction(deleteAction); 0502 } 0503 break; 0504 } 0505 case Mode::Moderation: { 0506 auto showReportInfo = new QAction(i18n("View Reports"), &menu); // Add icon 0507 connect(showReportInfo, &QAction::triggered, this, [this, message]() { 0508 const auto messageId = message->messageId(); 0509 const auto job = new RocketChatRestApi::ModerationReportsJob(this); 0510 job->setMessageId(messageId); 0511 mCurrentRocketChatAccount->restApi()->initializeRestApiJob(job); 0512 connect(job, &RocketChatRestApi::ModerationReportsJob::moderationReportsDone, this, [this](const QJsonObject &obj) { 0513 ModerationReportInfos infos; 0514 infos.parseModerationReportInfos(obj); 0515 slotShowReportInfo(std::move(infos)); 0516 }); 0517 if (!job->start()) { 0518 qCWarning(RUQOLAWIDGETS_LOG) << "Impossible to start ModerationReportInfoJob job"; 0519 } 0520 }); 0521 menu.addAction(showReportInfo); 0522 menu.addSeparator(); 0523 if (copyUrlAction) { 0524 menu.addAction(copyUrlAction); 0525 } 0526 menu.addSeparator(); 0527 menu.addAction(selectAllAction); 0528 menu.addSeparator(); 0529 auto dismissReports = new QAction(i18n("Dismiss Reports"), &menu); // Add icon 0530 connect(dismissReports, &QAction::triggered, this, [this, message]() { 0531 const auto messageId = message->messageId(); 0532 const auto job = new RocketChatRestApi::ModerationDismissReportsJob(this); 0533 job->setMessageId(messageId); 0534 mCurrentRocketChatAccount->restApi()->initializeRestApiJob(job); 0535 connect(job, &RocketChatRestApi::ModerationDismissReportsJob::moderationDismissReportsDone, this, []() { 0536 // TODO 0537 qDebug() << " RocketChatRestApi::ModerationDismissReportsJob::moderationDismissReportsDone "; 0538 // TODO update element! 0539 }); 0540 if (!job->start()) { 0541 qCWarning(RUQOLAWIDGETS_LOG) << "Impossible to start ModerationReportInfoJob job"; 0542 } 0543 }); 0544 0545 break; 0546 } 0547 case Mode::Viewing: { 0548 #if 0 0549 if (setPinnedMessage) { 0550 menu.addAction(setPinnedMessage); 0551 } 0552 #endif 0553 if (setAsFavoriteAction) { 0554 menu.addAction(setAsFavoriteAction); 0555 menu.addSeparator(); 0556 } 0557 menu.addAction(copyAction); 0558 if (copyUrlAction) { 0559 menu.addAction(copyUrlAction); 0560 } 0561 menu.addAction(copyLinkToMessageAction); 0562 menu.addSeparator(); 0563 menu.addAction(selectAllAction); 0564 #if 0 0565 createTranslorMenu(); 0566 if (!mTranslatorMenu->isEmpty()) { 0567 menu.addSeparator(); 0568 mTranslatorMenu->setModelIndex(index); 0569 menu.addMenu(mTranslatorMenu->menu()); 0570 } 0571 #endif 0572 menu.addSeparator(); 0573 auto goToMessageAction = new QAction(i18n("Go to Message"), &menu); // Add icon 0574 connect(goToMessageAction, &QAction::triggered, this, [this, index, message]() { 0575 const QString messageId = message->messageId(); 0576 const QString messageDateTimeUtc = index.data(MessagesModel::DateTimeUtc).toString(); 0577 Q_EMIT goToMessageRequested(messageId, messageDateTimeUtc); 0578 }); 0579 menu.addAction(goToMessageAction); 0580 break; 0581 } 0582 } 0583 0584 if (mMessageListDelegate->hasSelection()) { 0585 addTextPlugins(&menu, mMessageListDelegate->selectedText()); 0586 } 0587 0588 #if HAVE_TEXT_TO_SPEECH 0589 createSeparator(menu); 0590 auto speakAction = menu.addAction(QIcon::fromTheme(QStringLiteral("preferences-desktop-text-to-speech")), i18n("Speak Text")); 0591 connect(speakAction, &QAction::triggered, this, [this, index]() { 0592 slotTextToSpeech(index); 0593 }); 0594 #endif 0595 0596 if (mMode != Mode::Moderation && isNotOwnerOfMessage) { 0597 createSeparator(menu); 0598 auto reportMessageAction = new QAction(QIcon::fromTheme(QStringLiteral("messagebox_warning")), i18n("Report Message"), &menu); 0599 connect(reportMessageAction, &QAction::triggered, this, [this, index]() { 0600 slotReportMessage(index); 0601 }); 0602 menu.addAction(reportMessageAction); 0603 } 0604 if (!userInfoActions.isEmpty()) { 0605 menu.addSeparator(); 0606 for (auto action : userInfoActions) { 0607 menu.addAction(action); 0608 } 0609 } 0610 if (Ruqola::self()->debug()) { 0611 addDebugMenu(menu, index); 0612 } 0613 if (!menu.actions().isEmpty()) { 0614 menu.exec(event->globalPos()); 0615 } 0616 } 0617 0618 void MessageListView::addDebugMenu(QMenu &menu, const QModelIndex &index) 0619 { 0620 if (!mRoom) { 0621 return; 0622 } 0623 createSeparator(menu); 0624 if (index.isValid()) { 0625 auto debugMessageAction = new QAction(QStringLiteral("Dump Message"), &menu); // Don't translate it. 0626 connect(debugMessageAction, &QAction::triggered, this, [this, index]() { 0627 slotDebugMessage(index); 0628 }); 0629 menu.addAction(debugMessageAction); 0630 createSeparator(menu); 0631 } 0632 auto debugRoomAction = new QAction(QStringLiteral("Dump Room"), &menu); // Don't translate it. 0633 connect(debugRoomAction, &QAction::triggered, this, [this]() { 0634 // Dump info about room => don't use qCDebug here. 0635 qDebug() << " mRoom " << *mRoom; 0636 }); 0637 menu.addAction(debugRoomAction); 0638 } 0639 0640 bool MessageListView::maybeStartDrag(QMouseEvent *event, const QStyleOptionViewItem &option, const QModelIndex &index) 0641 { 0642 return mMessageListDelegate->maybeStartDrag(event, option, index); 0643 } 0644 0645 bool MessageListView::mouseEvent(QMouseEvent *event, const QStyleOptionViewItem &option, const QModelIndex &index) 0646 { 0647 return mMessageListDelegate->mouseEvent(event, option, index); 0648 } 0649 0650 void MessageListView::createSeparator(QMenu &menu) 0651 { 0652 if (!menu.isEmpty()) { 0653 menu.addSeparator(); 0654 } 0655 } 0656 0657 void MessageListView::slotSelectAll(const QModelIndex &index) 0658 { 0659 mMessageListDelegate->selectAll(listViewOptions(), index); 0660 } 0661 0662 void MessageListView::slotTranslateMessage(const QModelIndex &index, bool checked) 0663 { 0664 auto model = const_cast<QAbstractItemModel *>(index.model()); 0665 model->setData(index, checked, MessagesModel::ShowTranslatedMessage); 0666 } 0667 0668 void MessageListView::slotDebugMessage(const QModelIndex &index) 0669 { 0670 const Message *message = index.data(MessagesModel::MessagePointer).value<Message *>(); 0671 // Show debug output. 0672 qDebug() << " message " << *message << " MessageConvertedText " << index.data(MessagesModel::MessageConvertedText).toString(); 0673 } 0674 0675 void MessageListView::setCurrentRocketChatAccount(RocketChatAccount *currentRocketChatAccount) 0676 { 0677 mCurrentRocketChatAccount = currentRocketChatAccount; 0678 mMessageListDelegate->setRocketChatAccount(mCurrentRocketChatAccount); 0679 } 0680 0681 void MessageListView::slotFollowMessage(const QModelIndex &index, bool messageIsFollowing) 0682 { 0683 const QString messageId = index.data(MessagesModel::MessageId).toString(); 0684 if (messageIsFollowing) { 0685 auto job = new RocketChatRestApi::UnFollowMessageJob(this); 0686 job->setMessageId(messageId); 0687 mCurrentRocketChatAccount->restApi()->initializeRestApiJob(job); 0688 // connect(job, &RocketChatRestApi::FollowMessageJob::followMessageDone, this, &UsersInRoleWidget::slotAddUsersToRoleDone); 0689 if (!job->start()) { 0690 qCWarning(RUQOLAWIDGETS_LOG) << "Impossible to start UnFollowMessageJob job"; 0691 } 0692 } else { 0693 auto job = new RocketChatRestApi::FollowMessageJob(this); 0694 job->setMessageId(messageId); 0695 mCurrentRocketChatAccount->restApi()->initializeRestApiJob(job); 0696 // connect(job, &RocketChatRestApi::UnFollowMessageJob::followMessageDone, this, &UsersInRoleWidget::slotAddUsersToRoleDone); 0697 if (!job->start()) { 0698 qCWarning(RUQOLAWIDGETS_LOG) << "Impossible to start FollowMessageJob job"; 0699 } 0700 } 0701 } 0702 0703 void MessageListView::slotCopyLinkToMessage(const QModelIndex &index) 0704 { 0705 const QString messageId = index.data(MessagesModel::MessageId).toString(); 0706 const QString permalink = generatePermalink(messageId); 0707 QClipboard *clip = QApplication::clipboard(); 0708 clip->setText(permalink, QClipboard::Clipboard); 0709 } 0710 0711 QString MessageListView::generatePermalink(const QString &messageId) const 0712 { 0713 if (!mRoom) { 0714 return {}; 0715 } 0716 QString permalink = mCurrentRocketChatAccount->serverUrl() + QLatin1Char('/') + RoomUtil::generatePermalink(messageId, mRoom->name(), mRoom->channelType()); 0717 if (!permalink.startsWith(QStringLiteral("https://"))) { 0718 permalink.prepend(QStringLiteral("https://")); 0719 } 0720 return permalink; 0721 } 0722 0723 void MessageListView::slotQuoteMessage(const QModelIndex &index) 0724 { 0725 const QString messageId = index.data(MessagesModel::MessageId).toString(); 0726 QString text = index.data(MessagesModel::OriginalMessage).toString(); 0727 const QString permalink = generatePermalink(messageId); 0728 // qDebug() << " permalink " << permalink; 0729 if (text.length() > 80) { 0730 text = text.left(80) + QStringLiteral("..."); 0731 } 0732 Q_EMIT quoteMessageRequested(permalink, text); 0733 } 0734 0735 void MessageListView::slotEditMessage(const QModelIndex &index) 0736 { 0737 const QString text = index.data(MessagesModel::OriginalMessageOrAttachmentDescription).toString(); 0738 const QString messageId = index.data(MessagesModel::MessageId).toString(); 0739 Q_EMIT editMessageRequested(messageId, text); 0740 } 0741 0742 void MessageListView::slotShowFullThread(const QModelIndex &index) 0743 { 0744 const Message *message = index.data(MessagesModel::MessagePointer).value<Message *>(); 0745 const QString threadMessageId = message->threadMessageId(); 0746 QString threadMessagePreview = index.data(MessagesModel::ThreadMessagePreview).toString(); 0747 const bool threadIsFollowing = threadMessageId.isEmpty() ? message->replies().contains(mCurrentRocketChatAccount->userId()) 0748 : index.data(MessagesModel::ThreadMessageFollowed).toBool(); 0749 QString messageId = threadMessageId; 0750 if (threadMessageId.isEmpty()) { 0751 messageId = message->messageId(); 0752 if (threadMessagePreview.isEmpty()) { 0753 threadMessagePreview = index.data(MessagesModel::MessageConvertedText).toString(); 0754 } 0755 } 0756 auto dlg = new ThreadMessageDialog(mCurrentRocketChatAccount, this); 0757 ThreadMessageWidget::ThreadMessageInfo info; 0758 info.threadMessageId = messageId; 0759 info.threadMessagePreview = threadMessagePreview; 0760 info.threadIsFollowing = threadIsFollowing; 0761 info.room = mRoom; 0762 const Message tm = index.data(MessagesModel::ThreadMessage).value<Message>(); 0763 info.messageThread = tm; 0764 dlg->setThreadMessageInfo(info); 0765 dlg->show(); 0766 } 0767 0768 void MessageListView::slotMarkMessageAsUnread(const QModelIndex &index) 0769 { 0770 const QString messageId = index.data(MessagesModel::MessageId).toString(); 0771 mCurrentRocketChatAccount->markMessageAsUnReadFrom(messageId); 0772 } 0773 0774 void MessageListView::slotDeleteMessage(const QModelIndex &index) 0775 { 0776 if (KMessageBox::ButtonCode::PrimaryAction 0777 == KMessageBox::questionTwoActions(this, 0778 i18n("Do you want to delete this message?"), 0779 i18nc("@title", "Delete Message"), 0780 KStandardGuiItem::del(), 0781 KStandardGuiItem::cancel())) { 0782 const QString messageId = index.data(MessagesModel::MessageId).toString(); 0783 mCurrentRocketChatAccount->deleteMessage(messageId, mRoom->roomId()); 0784 } 0785 } 0786 0787 void MessageListView::slotTextToSpeech(const QModelIndex &index) 0788 { 0789 QString message = mMessageListDelegate->selectedText(); 0790 if (message.isEmpty()) { 0791 message = index.data(MessagesModel::OriginalMessage).toString(); 0792 } 0793 if (!message.isEmpty()) { 0794 Q_EMIT textToSpeech(message); 0795 } 0796 } 0797 0798 void MessageListView::slotReportMessage(const QModelIndex &index) 0799 { 0800 QPointer<ReportMessageDialog> dlg = new ReportMessageDialog(this); 0801 const QString message = index.data(MessagesModel::OriginalMessage).toString(); 0802 dlg->setPreviewMessage(message); 0803 if (dlg->exec()) { 0804 const QString messageId = index.data(MessagesModel::MessageId).toString(); 0805 mCurrentRocketChatAccount->reportMessage(messageId, dlg->message()); 0806 } 0807 delete dlg; 0808 } 0809 0810 void MessageListView::slotSetAsFavorite(const QModelIndex &index, bool isStarred) 0811 { 0812 const QString messageId = index.data(MessagesModel::MessageId).toString(); 0813 mCurrentRocketChatAccount->starMessage(messageId, !isStarred); 0814 } 0815 0816 void MessageListView::slotSetPinnedMessage(const QModelIndex &index, bool isPinned) 0817 { 0818 const QString messageId = index.data(MessagesModel::MessageId).toString(); 0819 mCurrentRocketChatAccount->pinMessage(messageId, !isPinned); 0820 } 0821 0822 void MessageListView::slotStartPrivateConversation(const QString &userName) 0823 { 0824 Q_EMIT createPrivateConversation(userName); 0825 } 0826 0827 void MessageListView::slotStartDiscussion(const QModelIndex &index) 0828 { 0829 const QString message = index.data(MessagesModel::OriginalMessage).toString(); 0830 const QString messageId = index.data(MessagesModel::MessageId).toString(); 0831 Q_EMIT createNewDiscussion(messageId, message); 0832 } 0833 0834 QString MessageListView::selectedText(const QModelIndex &index) 0835 { 0836 QString message = mMessageListDelegate->selectedText(); 0837 if (message.isEmpty()) { 0838 if (!index.isValid()) { 0839 return {}; 0840 } 0841 message = index.data(MessagesModel::OriginalMessage).toString(); 0842 } 0843 return message; 0844 } 0845 0846 void MessageListView::clearTextDocumentCache() 0847 { 0848 mMessageListDelegate->clearTextDocumentCache(); 0849 } 0850 0851 void MessageListView::scrollTo(const QModelIndex &index, QAbstractItemView::ScrollHint hint) 0852 { 0853 disconnect(verticalScrollBar(), &QScrollBar::valueChanged, this, &MessageListView::slotVerticalScrollbarChanged); 0854 QListView::scrollTo(index, hint); 0855 connect(verticalScrollBar(), &QScrollBar::valueChanged, this, &MessageListView::slotVerticalScrollbarChanged); 0856 addSelectedMessageBackgroundAnimation(index); 0857 } 0858 0859 void MessageListView::addSelectedMessageBackgroundAnimation(const QModelIndex &index) 0860 { 0861 auto messageModel = qobject_cast<MessagesModel *>(model()); 0862 if (messageModel) { 0863 auto animation = new SelectedMessageBackgroundAnimation(messageModel, this); 0864 animation->setModelIndex(index); 0865 animation->start(); 0866 } else { 0867 qCWarning(RUQOLAWIDGETS_LOG) << " message model empty"; 0868 } 0869 } 0870 0871 void MessageListView::setSearchText(const QString &str) 0872 { 0873 mMessageListDelegate->setSearchText(str); 0874 } 0875 0876 MessageListView::Mode MessageListView::mode() const 0877 { 0878 return mMode; 0879 } 0880 0881 void MessageListView::slotReplyInThread(const QModelIndex &index) 0882 { 0883 const QString messageId = index.data(MessagesModel::MessageId).toString(); 0884 const QString threadPreview = index.data(MessagesModel::OriginalMessage).toString(); 0885 Q_EMIT replyInThreadRequested(messageId, threadPreview); 0886 } 0887 0888 void MessageListView::slotShowUserInfo(const QString &userName) 0889 { 0890 DirectChannelInfoDialog dlg(mCurrentRocketChatAccount, this); 0891 dlg.setUserName(userName); 0892 dlg.setRoles(mCurrentRocketChatAccount->roleInfo()); 0893 dlg.exec(); 0894 } 0895 0896 void MessageListView::slotTranslate(const QString &from, const QString &to, const QPersistentModelIndex &modelIndex) 0897 { 0898 #if HAVE_TEXT_TRANSLATOR 0899 if (modelIndex.isValid()) { 0900 const QString originalMessage = modelIndex.data(MessagesModel::OriginalMessage).toString(); 0901 if (!originalMessage.isEmpty()) { 0902 qCDebug(RUQOLA_TRANSLATEMESSAGE_LOG) << " originalMessage " << originalMessage; 0903 qCDebug(RUQOLA_TRANSLATEMESSAGE_LOG) << " from " << from << " to " << to; 0904 TranslateTextJob::TranslateInfo info; 0905 info.from = from; 0906 info.to = to; 0907 info.inputText = originalMessage; 0908 auto job = new TranslateTextJob(this); 0909 job->setInfo(info); 0910 connect(job, &TranslateTextJob::translateDone, this, [this, modelIndex, job](const QString &str) { 0911 auto messageModel = qobject_cast<MessagesModel *>(model()); 0912 qCDebug(RUQOLA_TRANSLATEMESSAGE_LOG) << " modelIndex " << modelIndex; 0913 // qCDebug(RUQOLA_TRANSLATEMESSAGE_LOG) << " messageModel " << messageModel; 0914 messageModel->setData(modelIndex, str, MessagesModel::LocalTranslation); 0915 qCDebug(RUQOLA_TRANSLATEMESSAGE_LOG) << " translated string :" << str; 0916 job->deleteLater(); 0917 }); 0918 connect(job, &TranslateTextJob::translateFailed, this, [this, job](const QString &errorMessage) { 0919 KMessageBox::error(this, errorMessage, i18n("Translator Error")); 0920 job->deleteLater(); 0921 }); 0922 job->translate(); 0923 } 0924 } 0925 #else 0926 Q_UNUSED(from) 0927 Q_UNUSED(to) 0928 Q_UNUSED(modelIndex) 0929 #endif 0930 } 0931 0932 void MessageListView::slotShowReportInfo(const ModerationReportInfos &info) 0933 { 0934 ModerationMessageInfoDialog dlg(mCurrentRocketChatAccount, this); 0935 dlg.setReportInfos(info); 0936 dlg.exec(); 0937 } 0938 #include "moc_messagelistview.cpp"