/* This file is part of KMail, the KDE mail client. Copyright (c) 2002 Don Sanders Copyright (c) 2011-2015 Montel Laurent KMail is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License, version 2, as published by the Free Software Foundation. KMail 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, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ // // A toplevel KMainWindow derived class for displaying // single messages or single message parts. // // Could be extended to include support for normal main window // widgets like a toolbar. #include "kmreadermainwin.h" #include "kmreaderwin.h" #include #include #include #include #include #include #include #include #include #include #include "kmail_debug.h" #include #include #include #include #include #include "kmcommands.h" #include #include #include "kmmainwidget.h" #include "messageviewer/viewer/csshelper.h" #include "customtemplatesmenu.h" #include "messageactions.h" #include "util.h" #include "kernel/mailkernel.h" #include "foldercollection.h" #include #include #include #include #include #include #include #include #include #include #include "pimcommon/texttospeech/texttospeech.h" #include "messagecore/helpers/messagehelpers.h" #include using namespace MailCommon; KMReaderMainWin::KMReaderMainWin(MessageViewer::Viewer::DisplayFormatMessage format, bool htmlLoadExtOverride, char *name) : KMail::SecondaryWindow(name ? name : "readerwindow#") { mReaderWin = new KMReaderWin(this, this, actionCollection()); //mReaderWin->setShowCompleteMessage( true ); mReaderWin->setDisplayFormatMessageOverwrite(format); mReaderWin->setHtmlLoadExtOverride(htmlLoadExtOverride); mReaderWin->setDecryptMessageOverwrite(true); initKMReaderMainWin(); } KMReaderMainWin::KMReaderMainWin(char *name) : KMail::SecondaryWindow(name ? name : "readerwindow#") { mReaderWin = new KMReaderWin(this, this, actionCollection()); initKMReaderMainWin(); } KMReaderMainWin::KMReaderMainWin(KMime::Content *aMsgPart, MessageViewer::Viewer::DisplayFormatMessage format, const QString &encoding, char *name) : KMail::SecondaryWindow(name ? name : "readerwindow#") { mReaderWin = new KMReaderWin(this, this, actionCollection()); mReaderWin->setOverrideEncoding(encoding); mReaderWin->setDisplayFormatMessageOverwrite(format); mReaderWin->setMsgPart(aMsgPart); initKMReaderMainWin(); } void KMReaderMainWin::initKMReaderMainWin() { setCentralWidget(mReaderWin); setupAccel(); setupGUI(Keys | StatusBar | Create, QStringLiteral("kmreadermainwin.rc")); mMsgActions->setupForwardingActionsList(this); applyMainWindowSettings(KMKernel::self()->config()->group("Separate Reader Window")); if (! mReaderWin->message().isValid()) { menuBar()->hide(); toolBar(QStringLiteral("mainToolBar"))->hide(); } connect(kmkernel, SIGNAL(configChanged()), this, SLOT(slotConfigChanged())); connect(mReaderWin, SIGNAL(showStatusBarMessage(QString)), statusBar(), SLOT(showMessage(QString))); } KMReaderMainWin::~KMReaderMainWin() { KConfigGroup grp(KMKernel::self()->config()->group("Separate Reader Window")); saveMainWindowSettings(grp); } void KMReaderMainWin::setUseFixedFont(bool useFixedFont) { mReaderWin->setUseFixedFont(useFixedFont); } void KMReaderMainWin::showMessage(const QString &encoding, const Akonadi::Item &msg, const Akonadi::Collection &parentCollection) { mParentCollection = parentCollection; mReaderWin->setOverrideEncoding(encoding); mReaderWin->setMessage(msg, MessageViewer::Viewer::Force); KMime::Message::Ptr message = MessageCore::Util::message(msg); QString caption; if (message) { caption = message->subject()->asUnicodeString(); } if (mParentCollection.isValid()) { caption += QLatin1String(" - "); caption += MailCommon::Util::fullCollectionPath(mParentCollection); } if (!caption.isEmpty()) { setCaption(caption); } mMsg = msg; mMsgActions->setCurrentMessage(msg); const bool canChange = mParentCollection.isValid() ? (bool)(mParentCollection.rights() & Akonadi::Collection::CanDeleteItem) : false; mTrashAction->setEnabled(canChange); menuBar()->show(); toolBar(QStringLiteral("mainToolBar"))->show(); } void KMReaderMainWin::showMessage(const QString &encoding, KMime::Message::Ptr message) { if (!message) { return; } Akonadi::Item item; item.setPayload(message); Akonadi::MessageFlags::copyMessageFlags(*message, item); item.setMimeType(KMime::Message::mimeType()); mMsg = item; mMsgActions->setCurrentMessage(item); mReaderWin->setOverrideEncoding(encoding); mReaderWin->setMessage(message); setCaption(message->subject()->asUnicodeString()); mTrashAction->setEnabled(false); menuBar()->show(); toolBar(QStringLiteral("mainToolBar"))->show(); } void KMReaderMainWin::slotReplyOrForwardFinished() { if (GlobalSettings::self()->closeAfterReplyOrForward()) { close(); } } Akonadi::Collection KMReaderMainWin::parentCollection() const { if (mParentCollection.isValid()) { return mParentCollection; } else { return mMsg.parentCollection(); } } void KMReaderMainWin::slotTrashMsg() { if (!mMsg.isValid()) { return; } KMTrashMsgCommand *command = new KMTrashMsgCommand(parentCollection(), mMsg, -1); command->start(); close(); } void KMReaderMainWin::slotForwardInlineMsg() { if (!mReaderWin->message().hasPayload()) { return; } KMCommand *command = Q_NULLPTR; const Akonadi::Collection parentCol = mReaderWin->message().parentCollection(); if (parentCol.isValid()) { QSharedPointer fd = FolderCollection::forCollection(parentCol, false); if (fd) command = new KMForwardCommand(this, mReaderWin->message(), fd->identity()); else { command = new KMForwardCommand(this, mReaderWin->message()); } } else { command = new KMForwardCommand(this, mReaderWin->message()); } connect(command, &KMTrashMsgCommand::completed, this, &KMReaderMainWin::slotReplyOrForwardFinished); command->start(); } void KMReaderMainWin::slotForwardAttachedMsg() { if (!mReaderWin->message().hasPayload()) { return; } KMCommand *command = Q_NULLPTR; const Akonadi::Collection parentCol = mReaderWin->message().parentCollection(); if (parentCol.isValid()) { QSharedPointer fd = FolderCollection::forCollection(parentCol, false); if (fd) command = new KMForwardAttachedCommand(this, mReaderWin->message(), fd->identity()); else { command = new KMForwardAttachedCommand(this, mReaderWin->message()); } } else { command = new KMForwardAttachedCommand(this, mReaderWin->message()); } connect(command, &KMTrashMsgCommand::completed, this, &KMReaderMainWin::slotReplyOrForwardFinished); command->start(); } void KMReaderMainWin::slotRedirectMsg() { if (!mReaderWin->message().hasPayload()) { return; } KMCommand *command = new KMRedirectCommand(this, mReaderWin->message()); connect(command, &KMTrashMsgCommand::completed, this, &KMReaderMainWin::slotReplyOrForwardFinished); command->start(); } void KMReaderMainWin::slotCustomReplyToMsg(const QString &tmpl) { if (!mReaderWin->message().hasPayload()) { return; } KMCommand *command = new KMReplyCommand(this, mReaderWin->message(), MessageComposer::ReplySmart, mReaderWin->copyText(), false, tmpl); connect(command, &KMTrashMsgCommand::completed, this, &KMReaderMainWin::slotReplyOrForwardFinished); command->start(); } void KMReaderMainWin::slotCustomReplyAllToMsg(const QString &tmpl) { if (!mReaderWin->message().hasPayload()) { return; } KMCommand *command = new KMReplyCommand(this, mReaderWin->message(), MessageComposer::ReplyAll, mReaderWin->copyText(), false, tmpl); connect(command, &KMTrashMsgCommand::completed, this, &KMReaderMainWin::slotReplyOrForwardFinished); command->start(); } void KMReaderMainWin::slotCustomForwardMsg(const QString &tmpl) { if (!mReaderWin->message().hasPayload()) { return; } KMCommand *command = new KMForwardCommand(this, mReaderWin->message(), 0, tmpl); connect(command, &KMTrashMsgCommand::completed, this, &KMReaderMainWin::slotReplyOrForwardFinished); command->start(); } void KMReaderMainWin::slotConfigChanged() { //readConfig(); mMsgActions->setupForwardActions(actionCollection()); mMsgActions->setupForwardingActionsList(this); } void KMReaderMainWin::setupAccel() { if (!kmkernel->xmlGuiInstanceName().isEmpty()) { setComponentName(kmkernel->xmlGuiInstanceName(), i18n("kmail2")); } mMsgActions = new KMail::MessageActions(actionCollection(), this); mMsgActions->setMessageView(mReaderWin); connect(mMsgActions, &KMail::MessageActions::replyActionFinished, this, &KMReaderMainWin::slotReplyOrForwardFinished); //----- File Menu mSaveAtmAction = new QAction(QIcon::fromTheme(QStringLiteral("mail-attachment")), i18n("Save A&ttachments..."), actionCollection()); connect(mSaveAtmAction, &QAction::triggered, mReaderWin->viewer(), &MessageViewer::Viewer::slotAttachmentSaveAll); mTrashAction = new QAction(QIcon::fromTheme(QStringLiteral("user-trash")), i18n("&Move to Trash"), this); mTrashAction->setIconText(i18nc("@action:intoolbar Move to Trash", "Trash")); KMail::Util::addQActionHelpText(mTrashAction, i18n("Move message to trashcan")); actionCollection()->addAction(QStringLiteral("move_to_trash"), mTrashAction); actionCollection()->setDefaultShortcut(mTrashAction, QKeySequence(Qt::Key_Delete)); connect(mTrashAction, &QAction::triggered, this, &KMReaderMainWin::slotTrashMsg); QAction *closeAction = KStandardAction::close(this, SLOT(close()), actionCollection()); QList closeShortcut = closeAction->shortcuts(); closeAction->setShortcuts(closeShortcut << QKeySequence(Qt::Key_Escape)); //----- Message Menu mFontAction = new KFontAction(i18n("Select Font"), this); actionCollection()->addAction(QStringLiteral("text_font"), mFontAction); mFontAction->setFont(mReaderWin->cssHelper()->bodyFont().family()); connect(mFontAction, static_cast(&KFontAction::triggered), this, &KMReaderMainWin::slotFontAction); mFontSizeAction = new KFontSizeAction(i18n("Select Size"), this); mFontSizeAction->setFontSize(mReaderWin->cssHelper()->bodyFont().pointSize()); actionCollection()->addAction(QStringLiteral("text_size"), mFontSizeAction); connect(mFontSizeAction, &KFontSizeAction::fontSizeChanged, this, &KMReaderMainWin::slotSizeAction); connect(mReaderWin->viewer(), &MessageViewer::Viewer::popupMenu, this, &KMReaderMainWin::slotMessagePopup); connect(mReaderWin->viewer(), &MessageViewer::Viewer::itemRemoved, this, &QWidget::close); setStandardToolBarMenuEnabled(true); KStandardAction::configureToolbars(this, SLOT(slotEditToolbars()), actionCollection()); connect(mReaderWin->viewer(), &MessageViewer::Viewer::moveMessageToTrash, this, &KMReaderMainWin::slotTrashMsg); } QAction *KMReaderMainWin::copyActionMenu(QMenu *menu) { KMMainWidget *mainwin = kmkernel->getKMMainWidget(); if (mainwin) { KActionMenu *action = new KActionMenu(menu); action->setIcon(QIcon::fromTheme(QStringLiteral("edit-copy"))); action->setText(i18n("Copy Message To...")); mainwin->standardMailActionManager()->standardActionManager()->createActionFolderMenu(action->menu(), Akonadi::StandardActionManager::CopyItemToMenu); connect(action->menu(), &QMenu::triggered, this, &KMReaderMainWin::slotCopyItem); return action; } return Q_NULLPTR; } QAction *KMReaderMainWin::moveActionMenu(QMenu *menu) { KMMainWidget *mainwin = kmkernel->getKMMainWidget(); if (mainwin) { KActionMenu *action = new KActionMenu(menu); action->setText(i18n("Move Message To...")); mainwin->standardMailActionManager()->standardActionManager()->createActionFolderMenu(action->menu(), Akonadi::StandardActionManager::MoveItemToMenu); connect(action->menu(), SIGNAL(triggered(QAction*)), SLOT(slotMoveItem(QAction*))); return action; } return Q_NULLPTR; } void KMReaderMainWin::slotMoveItem(QAction *action) { if (action) { const QModelIndex index = action->data().value(); const Akonadi::Collection collection = index.data(Akonadi::EntityTreeModel::CollectionRole).value(); copyOrMoveItem(collection, true); } } void KMReaderMainWin::copyOrMoveItem(const Akonadi::Collection &collection, bool move) { if (mMsg.isValid()) { if (move) { Akonadi::ItemMoveJob *job = new Akonadi::ItemMoveJob(mMsg, collection, this); connect(job, SIGNAL(result(KJob*)), this, SLOT(slotCopyMoveResult(KJob*))); } else { Akonadi::ItemCopyJob *job = new Akonadi::ItemCopyJob(mMsg, collection, this); connect(job, SIGNAL(result(KJob*)), this, SLOT(slotCopyMoveResult(KJob*))); } } else { Akonadi::ItemCreateJob *job = new Akonadi::ItemCreateJob(mMsg, collection, this); connect(job, SIGNAL(result(KJob*)), this, SLOT(slotCopyMoveResult(KJob*))); } } void KMReaderMainWin::slotCopyItem(QAction *action) { if (action) { const QModelIndex index = action->data().value(); const Akonadi::Collection collection = index.data(Akonadi::EntityTreeModel::CollectionRole).value(); copyOrMoveItem(collection, false); } } void KMReaderMainWin::slotCopyMoveResult(KJob *job) { if (job->error()) { KMessageBox::sorry(this, i18n("Cannot copy item. %1", job->errorString())); } } void KMReaderMainWin::slotMessagePopup(const Akonadi::Item &aMsg , const QUrl &aUrl, const QUrl &imageUrl, const QPoint &aPoint) { mMsg = aMsg; const QString email = KEmailAddress::firstEmailAddress(aUrl.path()).toLower(); if (aUrl.scheme() == QLatin1String("mailto") && !email.isEmpty()) { Akonadi::ContactSearchJob *job = new Akonadi::ContactSearchJob(this); job->setLimit(1); job->setQuery(Akonadi::ContactSearchJob::Email, email, Akonadi::ContactSearchJob::ExactMatch); job->setProperty("msg", QVariant::fromValue(mMsg)); job->setProperty("point", aPoint); job->setProperty("imageUrl", imageUrl); job->setProperty("url", aUrl); connect(job, &Akonadi::ItemCopyJob::result, this, &KMReaderMainWin::slotContactSearchJobForMessagePopupDone); } else { showMessagePopup(mMsg, aUrl, imageUrl, aPoint, false, false); } } void KMReaderMainWin::slotContactSearchJobForMessagePopupDone(KJob *job) { const Akonadi::ContactSearchJob *searchJob = qobject_cast(job); const bool contactAlreadyExists = !searchJob->contacts().isEmpty(); const Akonadi::Item::List listContact = searchJob->items(); const bool uniqueContactFound = (listContact.count() == 1); if (uniqueContactFound) { mReaderWin->setContactItem(listContact.first(), searchJob->contacts().at(0)); } else { mReaderWin->clearContactItem(); } const Akonadi::Item msg = job->property("msg").value(); const QPoint aPoint = job->property("point").toPoint(); const QUrl imageUrl = job->property("imageUrl").toUrl(); const QUrl url = job->property("url").toUrl(); showMessagePopup(msg, url, imageUrl, aPoint, contactAlreadyExists, uniqueContactFound); } void KMReaderMainWin::showMessagePopup(const Akonadi::Item &msg , const QUrl &url, const QUrl &imageUrl, const QPoint &aPoint, bool contactAlreadyExists, bool uniqueContactFound) { QMenu *menu = Q_NULLPTR; bool urlMenuAdded = false; bool copyAdded = false; const bool messageHasPayload = msg.hasPayload(); if (!url.isEmpty()) { if (url.scheme() == QLatin1String("mailto")) { // popup on a mailto URL menu = new QMenu; menu->addAction(mReaderWin->mailToComposeAction()); if (messageHasPayload) { menu->addAction(mReaderWin->mailToReplyAction()); menu->addAction(mReaderWin->mailToForwardAction()); menu->addSeparator(); } if (contactAlreadyExists) { if (uniqueContactFound) { menu->addAction(mReaderWin->editContactAction()); } else { menu->addAction(mReaderWin->openAddrBookAction()); } } else { menu->addAction(mReaderWin->addAddrBookAction()); menu->addAction(mReaderWin->addToExistingContactAction()); } menu->addSeparator(); menu->addMenu(mReaderWin->viewHtmlOption()); menu->addSeparator(); menu->addAction(mReaderWin->copyURLAction()); menu->addSeparator(); menu->addAction(mReaderWin->shareServiceUrlMenu()); copyAdded = true; urlMenuAdded = true; } else if (url.scheme() != QLatin1String("attachment")) { // popup on a not-mailto URL menu = new QMenu; menu->addAction(mReaderWin->urlOpenAction()); menu->addAction(mReaderWin->addBookmarksAction()); menu->addAction(mReaderWin->urlSaveAsAction()); menu->addAction(mReaderWin->copyURLAction()); menu->addSeparator(); menu->addAction(mReaderWin->shareServiceUrlMenu()); if (mReaderWin->isAShortUrl(url)) { menu->addSeparator(); menu->addAction(mReaderWin->expandShortUrlAction()); } if (!imageUrl.isEmpty()) { menu->addSeparator(); menu->addAction(mReaderWin->copyImageLocation()); menu->addAction(mReaderWin->downloadImageToDiskAction()); menu->addAction(mReaderWin->shareImage()); if (mReaderWin->adblockEnabled()) { menu->addSeparator(); menu->addAction(mReaderWin->blockImage()); } } urlMenuAdded = true; } } const QString selectedText(mReaderWin->copyText()); if (!selectedText.isEmpty()) { if (!menu) { menu = new QMenu; } if (urlMenuAdded) { menu->addSeparator(); } if (messageHasPayload) { menu->addAction(mMsgActions->replyMenu()); menu->addSeparator(); menu->addAction(mMsgActions->mailingListActionMenu()); menu->addSeparator(); } if (!copyAdded) { menu->addAction(mReaderWin->copyAction()); } menu->addAction(mReaderWin->selectAllAction()); menu->addSeparator(); mMsgActions->addWebShortcutsMenu(menu, selectedText); menu->addSeparator(); menu->addAction(mReaderWin->translateAction()); if (PimCommon::TextToSpeech::self()->isReady()) { menu->addSeparator(); menu->addAction(mReaderWin->speakTextAction()); } } else if (!urlMenuAdded) { if (!menu) { menu = new QMenu; } // popup somewhere else (i.e., not a URL) on the message if (messageHasPayload) { bool replyForwardMenu = false; Akonadi::Collection col = parentCollection(); if (col.isValid()) { if (!(CommonKernel->folderIsSentMailFolder(col) || CommonKernel->folderIsDrafts(col) || CommonKernel->folderIsTemplates(col))) { replyForwardMenu = true; } } else if (messageHasPayload) { replyForwardMenu = true; } if (replyForwardMenu) { // add the reply and forward actions only if we are not in a sent-mail, // templates or drafts folder menu->addAction(mMsgActions->replyMenu()); menu->addAction(mMsgActions->forwardMenu()); menu->addSeparator(); } menu->addAction(copyActionMenu(menu)); menu->addAction(moveActionMenu(menu)); menu->addSeparator(); menu->addAction(mMsgActions->mailingListActionMenu()); menu->addSeparator(); if (!imageUrl.isEmpty()) { menu->addSeparator(); menu->addAction(mReaderWin->copyImageLocation()); menu->addAction(mReaderWin->downloadImageToDiskAction()); menu->addAction(mReaderWin->shareImage()); menu->addSeparator(); if (mReaderWin->adblockEnabled()) { menu->addAction(mReaderWin->blockImage()); menu->addSeparator(); } } menu->addAction(mReaderWin->viewSourceAction()); menu->addAction(mReaderWin->toggleFixFontAction()); if (!mReaderWin->mimePartTreeIsEmpty()) { menu->addAction(mReaderWin->toggleMimePartTreeAction()); } menu->addSeparator(); if (mMsgActions->printPreviewAction()) { menu->addAction(mMsgActions->printPreviewAction()); } menu->addAction(mMsgActions->printAction()); menu->addAction(mReaderWin->saveAsAction()); menu->addAction(mSaveAtmAction); if (msg.isValid()) { menu->addSeparator(); menu->addAction(mReaderWin->createTodoAction()); menu->addAction(mReaderWin->createEventAction()); menu->addSeparator(); menu->addAction(mReaderWin->saveMessageDisplayFormatAction()); menu->addAction(mReaderWin->resetMessageDisplayFormatAction()); } } else { menu->addAction(mReaderWin->toggleFixFontAction()); if (!mReaderWin->mimePartTreeIsEmpty()) { menu->addAction(mReaderWin->toggleMimePartTreeAction()); } } if (mReaderWin->adblockEnabled()) { menu->addSeparator(); menu->addAction(mReaderWin->openBlockableItems()); } if (msg.isValid()) { menu->addAction(mMsgActions->addFollowupReminderAction()); } if (msg.isValid()) { menu->addSeparator(); menu->addAction(mMsgActions->addFollowupReminderAction()); } if (kmkernel->allowToDebugBalooSupport()) { menu->addSeparator(); menu->addAction(mMsgActions->debugBalooAction()); } } if (menu) { KAcceleratorManager::manage(menu); menu->exec(aPoint, Q_NULLPTR); delete menu; } } void KMReaderMainWin::slotFontAction(const QString &font) { QFont f(mReaderWin->cssHelper()->bodyFont()); f.setFamily(font); changeFont(f); } void KMReaderMainWin::slotSizeAction(int size) { QFont f(mReaderWin->cssHelper()->bodyFont()); f.setPointSize(size); changeFont(f); } void KMReaderMainWin::changeFont(const QFont &f) { mReaderWin->cssHelper()->setBodyFont(f); mReaderWin->cssHelper()->setPrintFont(f); mReaderWin->update(); } void KMReaderMainWin::slotEditToolbars() { KConfigGroup grp(KMKernel::self()->config(), "ReaderWindow"); saveMainWindowSettings(grp); KEditToolBar dlg(guiFactory(), this); connect(&dlg, &KEditToolBar::newToolBarConfig, this, &KMReaderMainWin::slotUpdateToolbars); dlg.exec(); } void KMReaderMainWin::slotUpdateToolbars() { createGUI(QStringLiteral("kmreadermainwin.rc")); applyMainWindowSettings(KConfigGroup(KMKernel::self()->config(), "ReaderWindow")); }