/* -*- mode: C++; c-file-style: "gnu" -*- This file is part of KMail, the KDE mail client. Copyright (c) 2002 Don Sanders Based on the work of Stefan Taferner 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 */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace KMime; using KMime::Types::AddrSpecList; #include "progressmanager.h" using KPIM::ProgressManager; #include "globalsettings.h" #include "kcursorsaver.h" #include "broadcaststatus.h" using KPIM::BroadcastStatus; #include "kmfoldermgr.h" #include "kmfolderdialog.h" #include "accountmanager.h" using KMail::AccountManager; #include "kmfilter.h" #include "kmfoldertree.h" #include "kmreadermainwin.h" #include "kmfoldercachedimap.h" #include "kmfolderimap.h" #include "kmacctcachedimap.h" #include "composer.h" #include "folderselectiondialog.h" #include "kmfiltermgr.h" #include "messagesender.h" #include "kmaddrbook.h" #include "kmversion.h" #include "searchwindow.h" using KMail::SearchWindow; #include "kmacctfolder.h" #include "undostack.h" #include "kmcommands.h" #include "kmmainwin.h" #include "kmsystemtray.h" #include "kmmessagetag.h" #include "imapaccountbase.h" using KMail::ImapAccountBase; #include "vacation.h" using KMail::Vacation; #include "favoritefolderview.h" #include "subscriptiondialog.h" using KMail::SubscriptionDialog; #include "localsubscriptiondialog.h" using KMail::LocalSubscriptionDialog; #include "attachmentstrategy.h" using KMail::AttachmentStrategy; #include "headerstrategy.h" using KMail::HeaderStrategy; #include "headerstyle.h" using KMail::HeaderStyle; #include "folderjob.h" using KMail::FolderJob; #include "mailinglist-magic.h" #include "antispamwizard.h" using KMail::AntiSpamWizard; #include "filterlogdlg.h" using KMail::FilterLogDialog; #include using KMail::HeaderListQuickSearch; #include "kmheaders.h" #include "mailinglistpropertiesdialog.h" #include "templateparser.h" using KMail::TemplateParser; #include "statusbarlabel.h" #include "archivefolderdialog.h" #include "folderutil.h" #if !defined(NDEBUG) #include "sievedebugdialog.h" using KMail::SieveDebugDialog; #endif #include "managesievescriptsdialog.h" #include "customtemplatesmenu.h" #include #include #include "kmmainwidget.moc" K_GLOBAL_STATIC( KMMainWidget::PtrList, theMainWidgetList ) static const int defaultMinimumWidth = 400; static const int defaultMinimumHeight = 300; //----------------------------------------------------------------------------- KMMainWidget::KMMainWidget( QWidget *parent, KXMLGUIClient *aGUIClient, KActionCollection *actionCollection, KConfig *config ) : QWidget( parent ), mFavoritesCheckMailAction( 0 ), mQuickSearchLine( 0 ), mFavoriteFolderView( 0 ), mMsgView( 0 ), mSplitter1( 0 ), mSplitter2( 0 ), mFolderViewSplitter( 0 ), mArchiveFolderAction( 0 ), mShowBusySplashTimer( 0 ), mShowingOfflineScreen( false ), mMsgActions( 0 ), mVacationIndicatorActive( false ) { // must be the first line of the constructor: mStartupDone = false; mWasEverShown = false; mSearchWin = 0; mIntegrated = true; mFolder = 0; mTemplateFolder = 0; mFolderThreadPref = false; mFolderThreadSubjPref = true; mReaderWindowActive = true; mReaderWindowBelow = true; mFolderHtmlPref = false; mFolderHtmlLoadExtPref = false; mSystemTray = 0; mDestructed = false; mMessageTagToggleMapper = 0; mActionCollection = actionCollection; mTopLayout = new QVBoxLayout( this ); mTopLayout->setMargin( 0 ); mJob = 0; mConfig = config; mGUIClient = aGUIClient; mOpenedImapFolder = false; mCustomTemplateMenus = 0; // FIXME This should become a line separator as soon as the API // is extended in kdelibs. mToolbarActionSeparator = new QAction( this ); mToolbarActionSeparator->setSeparator( true ); mMessageTagToolbarActionSeparator = new QAction( this ); mMessageTagToolbarActionSeparator->setSeparator( true ); theMainWidgetList->append( this ); setMinimumSize( defaultMinimumWidth, defaultMinimumHeight ); readPreConfig(); createWidgets(); setupActions(); readConfig(); QTimer::singleShot( 0, this, SLOT( slotShowStartupFolder() )); connect( kmkernel->acctMgr(), SIGNAL( checkedMail( bool, bool, const QMap & ) ), this, SLOT( slotMailChecked( bool, bool, const QMap & ) ) ); connect( kmkernel->acctMgr(), SIGNAL( accountAdded( KMAccount* ) ), this, SLOT( initializeIMAPActions() ) ); connect( kmkernel->acctMgr(), SIGNAL( accountRemoved( KMAccount* ) ), this, SLOT( initializeIMAPActions() ) ); connect( kmkernel, SIGNAL( configChanged() ), this, SLOT( slotConfigChanged() ) ); // display the full path to the folder in the caption connect( mFolderTree, SIGNAL( currentChanged(Q3ListViewItem*) ), this, SLOT( slotChangeCaption(Q3ListViewItem*) ) ); connect( mFolderTree, SIGNAL( selectionChanged() ), SLOT( updateFolderMenu() ) ); connect( kmkernel->folderMgr(), SIGNAL( folderRemoved(KMFolder*) ), this, SLOT( slotFolderRemoved(KMFolder*) ) ); connect( kmkernel->imapFolderMgr(), SIGNAL( folderRemoved(KMFolder*) ), this, SLOT( slotFolderRemoved(KMFolder*) ) ); connect( kmkernel->dimapFolderMgr(), SIGNAL( folderRemoved(KMFolder*) ), this, SLOT( slotFolderRemoved(KMFolder*) ) ); connect( kmkernel->searchFolderMgr(), SIGNAL( folderRemoved(KMFolder*) ), this, SLOT( slotFolderRemoved(KMFolder*) ) ); connect( kmkernel, SIGNAL( onlineStatusChanged( GlobalSettings::EnumNetworkState::type ) ), this, SLOT( slotUpdateOnlineStatus( GlobalSettings::EnumNetworkState::type ) ) ); toggleSystemTray(); KMainWindow *mainWin = dynamic_cast(topLevelWidget()); KStatusBar *sb = mainWin ? mainWin->statusBar() : 0; mVacationScriptIndicator = new KMail::StatusBarLabel( sb ); mVacationScriptIndicator->hide(); connect( mVacationScriptIndicator, SIGNAL(clicked()), SLOT(slotEditVacation()) ); if ( GlobalSettings::checkOutOfOfficeOnStartup() ) QTimer::singleShot( 0, this, SLOT(slotCheckVacation()) ); // must be the last line of the constructor: mStartupDone = true; } //----------------------------------------------------------------------------- //The kernel may have already been deleted when this method is called, //perform all cleanup that requires the kernel in destruct() KMMainWidget::~KMMainWidget() { theMainWidgetList->removeAll( this ); qDeleteAll( mFilterCommands ); destruct(); } //----------------------------------------------------------------------------- //This method performs all cleanup that requires the kernel to exist. void KMMainWidget::destruct() { if ( mDestructed ) return; if ( mSearchWin ) mSearchWin->close(); writeConfig(); writeFolderConfig(); deleteWidgets(); delete mSystemTray; delete mCustomTemplateMenus; mSystemTray = 0; mCustomTemplateMenus = 0; mDestructed = true; } //----------------------------------------------------------------------------- void KMMainWidget::readPreConfig() { const KConfigGroup geometry( KMKernel::config(), "Geometry" ); const KConfigGroup general( KMKernel::config(), "General" ); const KConfigGroup reader( KMKernel::config(), "Reader" ); mLongFolderList = geometry.readEntry( "FolderList", "long" ) != "short"; mReaderWindowActive = geometry.readEntry( "readerWindowMode", "below" ) != "hide"; mReaderWindowBelow = geometry.readEntry( "readerWindowMode", "below" ) == "below"; mThreadPref = geometry.readEntry( "nestedMessages", false ); mHtmlPref = reader.readEntry( "htmlMail", false ); mHtmlLoadExtPref = reader.readEntry( "htmlLoadExternal", false ); mEnableFavoriteFolderView = GlobalSettings::self()->enableFavoriteFolderView(); mEnableFolderQuickSearch = GlobalSettings::self()->enableFolderQuickSearch(); mEnableQuickSearch = GlobalSettings::self()->quickSearchActive(); } //----------------------------------------------------------------------------- void KMMainWidget::readFolderConfig() { if ( !mFolder ) return; KConfig *config = KMKernel::config(); KConfigGroup group( config, "Folder-" + mFolder->idString() ); mFolderThreadPref = group.readEntry( "threadMessagesOverride", false ); mFolderThreadSubjPref = group.readEntry( "threadMessagesBySubject", true ); mFolderHtmlPref = group.readEntry( "htmlMailOverride", false ); mFolderHtmlLoadExtPref = group.readEntry( "htmlLoadExternalOverride", false ); } //----------------------------------------------------------------------------- void KMMainWidget::writeFolderConfig() { if ( !mFolder ) return; KConfig *config = KMKernel::config(); KConfigGroup group( config, "Folder-" + mFolder->idString() ); group.writeEntry( "threadMessagesOverride", mFolderThreadPref ); group.writeEntry( "threadMessagesBySubject", mFolderThreadSubjPref ); group.writeEntry( "htmlMailOverride", mFolderHtmlPref ); group.writeEntry( "htmlLoadExternalOverride", mFolderHtmlLoadExtPref ); } //----------------------------------------------------------------------------- void KMMainWidget::layoutSplitters() { // This function can only be called when the old splitters are already deleted assert( !mSplitter1 ); assert( !mSplitter2 ); // For some reason, this is necessary here so that the copy action still // works after changing the folder layout. if ( mMsgView ) disconnect( mMsgView->copyAction(), SIGNAL(triggered(bool) ), mMsgView, SLOT( slotCopySelectedText() ) ); // If long folder list is enabled, the splitters are: // Splitter 1: FolderView vs (HeaderAndSearch vs MessageViewer) // Splitter 2: HeaderAndSearch vs MessageViewer // // If long folder list is disabled, the splitters are: // Splitter 1: (FolderView vs HeaderAndSearch) vs MessageViewer // Splitter 2: FolderView vs HeaderAndSearch // The folder view is both the folder tree and the favorite folder view, if // enabled const bool opaqueResize = KGlobalSettings::opaqueResize(); bool readerWindowAtSide = !mReaderWindowBelow && mReaderWindowActive; bool readerWindowBelow = mReaderWindowBelow && mReaderWindowActive; // // Create the splitters // mSplitter1 = new QSplitter( this ); mSplitter1->setObjectName( "splitter1" ); mSplitter1->setOpaqueResize( opaqueResize ); mSplitter1->setChildrenCollapsible( false ); mSplitter2 = new QSplitter( mSplitter1 ); mSplitter2->setObjectName( "splitter2" ); mSplitter2->setOpaqueResize( opaqueResize ); mSplitter2->setChildrenCollapsible( false ); mSplitter1->addWidget( mSplitter2 ); // // Set the layout of the splitters and calculate the widget's parents // QSplitter *folderViewParent, *folderTreeParent, *messageViewerParent; if ( mLongFolderList ) { mSplitter1->setOrientation( Qt::Horizontal ); Qt::Orientation splitter2orientation; if ( !readerWindowAtSide ) splitter2orientation = Qt::Vertical; else splitter2orientation = Qt::Horizontal; mSplitter2->setOrientation( splitter2orientation ); folderViewParent = mSplitter1; messageViewerParent = mSplitter2; } else { Qt::Orientation splitter1orientation; if ( !readerWindowAtSide ) splitter1orientation = Qt::Vertical; else splitter1orientation = Qt::Horizontal; mSplitter1->setOrientation( splitter1orientation ); mSplitter2->setOrientation( Qt::Horizontal ); folderViewParent = mSplitter2; messageViewerParent = mSplitter1; } // // Add the widgets to the splitters and set the parents calculated above // int folderTreePosition = 0; if ( mFavoriteFolderView ) { mFolderViewSplitter = new QSplitter( Qt::Vertical, folderViewParent ); mFolderViewSplitter->setOpaqueResize( opaqueResize ); mFolderViewSplitter->setChildrenCollapsible( false ); folderTreeParent = mFolderViewSplitter; mFolderViewSplitter->addWidget( mFavoriteFolderView ); mFavoriteFolderView->setParent( mFolderViewSplitter ); folderViewParent->insertWidget( 0, mFolderViewSplitter ); folderTreePosition = 1; } else folderTreeParent = folderViewParent; folderTreeParent->insertWidget( folderTreePosition, mSearchAndTree ); mSplitter2->addWidget( mSearchAndHeaders ); if ( mMsgView ) { messageViewerParent->addWidget( mMsgView ); mMsgView->setParent( messageViewerParent ); } mSearchAndTree->setParent( folderTreeParent ); mSearchAndHeaders->setParent( mSplitter2 ); // // Set the stretch factors // mSplitter1->setStretchFactor( 0, 0 ); mSplitter2->setStretchFactor( 0, 0 ); mSplitter1->setStretchFactor( 1, 1 ); mSplitter2->setStretchFactor( 1, 1 ); if ( mFavoriteFolderView ) { mFolderViewSplitter->setStretchFactor( 0, 0 ); mFolderViewSplitter->setStretchFactor( 1, 1 ); } // Because the reader windows's width increases a tiny bit after each restart // in short folder list mode with message window at side, disable the stretching // as a workaround here if ( readerWindowAtSide && !mLongFolderList ) { mSplitter1->setStretchFactor( 0, 1 ); mSplitter1->setStretchFactor( 1, 0 ); } // // Set the sizes of the splitters to the values stored in the config // QList splitter1Sizes; QList splitter2Sizes; const int folderViewWidth = GlobalSettings::self()->folderViewWidth(); int headerHeight = GlobalSettings::self()->searchAndHeaderHeight(); const int messageViewerWidth = GlobalSettings::self()->readerWindowWidth(); int headerWidth = GlobalSettings::self()->searchAndHeaderWidth(); int messageViewerHeight = GlobalSettings::self()->readerWindowHeight(); // If the message viewer was hidden before, make sure it is not zero height if ( messageViewerHeight < 10 && readerWindowBelow ) { headerHeight /= 2; messageViewerHeight = headerHeight; } if ( mLongFolderList ) { if ( !readerWindowAtSide ) { splitter1Sizes << folderViewWidth << headerWidth; splitter2Sizes << headerHeight << messageViewerHeight; } else { splitter1Sizes << folderViewWidth << ( headerWidth + messageViewerWidth ); splitter2Sizes << headerWidth << messageViewerWidth; } } else { if ( !readerWindowAtSide ) { splitter1Sizes << headerHeight << messageViewerHeight; splitter2Sizes << folderViewWidth << headerWidth; } else { splitter1Sizes << headerWidth << messageViewerWidth; splitter2Sizes << folderViewWidth << ( headerWidth + messageViewerWidth ); } } mSplitter1->setSizes( splitter1Sizes ); mSplitter2->setSizes( splitter2Sizes ); if ( mFolderViewSplitter ) { QList splitterSizes; int ffvHeight = GlobalSettings::self()->favoriteFolderViewHeight(); int ftHeight = GlobalSettings::self()->folderTreeHeight(); splitterSizes << ffvHeight << ftHeight; mFolderViewSplitter->setSizes( splitterSizes ); } // // Now add the splitters to the main layout // mTopLayout->addWidget( mSplitter1 ); // Make the copy action work, see disconnect comment above if ( mMsgView ) connect( mMsgView->copyAction(), SIGNAL( triggered(bool) ), mMsgView, SLOT( slotCopySelectedText() ) ); } //----------------------------------------------------------------------------- void KMMainWidget::readConfig() { KConfig *config = KMKernel::config(); bool oldLongFolderList = mLongFolderList; bool oldReaderWindowActive = mReaderWindowActive; bool oldReaderWindowBelow = mReaderWindowBelow; bool oldFavoriteFolderView = mEnableFavoriteFolderView; bool oldFolderQuickSearch = mEnableFolderQuickSearch; bool oldQuickSearch = mEnableQuickSearch; // on startup, the layout is always new and we need to relayout the widgets bool layoutChanged = !mStartupDone; if ( mStartupDone ) { writeConfig(); readPreConfig(); mHeaders->refreshNestedState(); layoutChanged = ( oldLongFolderList != mLongFolderList ) || ( oldReaderWindowActive != mReaderWindowActive ) || ( oldReaderWindowBelow != mReaderWindowBelow ) || ( oldFavoriteFolderView != mEnableFavoriteFolderView ) || ( oldFolderQuickSearch != mEnableFolderQuickSearch ) || ( oldQuickSearch != mEnableQuickSearch ); if( layoutChanged ) { deleteWidgets(); createWidgets(); } } { // area for config group "Geometry" KConfigGroup group(config, "Geometry"); // Set the proper column sizes for the folder tree if ( layoutChanged ) { /** unread / total columns * as we have some dependencies in this widget * it's better to manage these here */ // The columns are shown by default. const int unreadColumn = group.readEntry( "UnreadColumn", -1 ); const int totalColumn = group.readEntry( "TotalColumn", 2 ); const int sizeColumn = group.readEntry( "SizeColumn", -1 ); /* we need to _activate_ them in the correct order * this is ugly because we can't use header()->moveSection * but otherwise the restoreLayout from KMFolderTree * doesn't know that to do */ if ( unreadColumn == 1 ) mFolderTree->addUnreadColumn( i18n("Unread"), 70 ); else if ( totalColumn == 1 ) mFolderTree->addTotalColumn( i18n("Total"), 70 ); else if ( sizeColumn == 1 ) mFolderTree->addSizeColumn( i18n("Size"), 70 ); if ( unreadColumn == 2 ) mFolderTree->addUnreadColumn( i18n("Unread"), 70 ); else if ( totalColumn == 2 ) mFolderTree->addTotalColumn( i18n("Total"), 70 ); else if ( sizeColumn == 2 ) mFolderTree->addSizeColumn( i18n("Size"), 70 ); if ( unreadColumn == 3 ) mFolderTree->addUnreadColumn( i18n("Unread"), 70 ); else if ( totalColumn == 3 ) mFolderTree->addTotalColumn( i18n("Total"), 70 ); else if ( sizeColumn == 3 ) mFolderTree->addSizeColumn( i18n("Size"), 70 ); slotFolderTreeColumnsChanged(); mFolderTree->updatePopup(); } } { // Read the config of the folder views and the header if ( mMsgView ) mMsgView->readConfig(); mHeaders->readConfig(); mHeaders->restoreColumnLayout( KMKernel::config(), "Header-Geometry" ); mFolderTree->readConfig(); mFolderTree->reload(); if ( mFavoriteFolderView ) mFavoriteFolderView->readConfig(); mFavoritesCheckMailAction->setEnabled( GlobalSettings::self()->enableFavoriteFolderView() ); } { // area for config group "General" KConfigGroup group( config, "General" ); mBeepOnNew = group.readEntry( "beep-on-mail", false ); mConfirmEmpty = group.readEntry( "confirm-before-empty", true ); // startup-Folder, defaults to system-inbox mStartupFolder = group.readEntry( "startupFolder", kmkernel->inboxFolder()->idString() ); if ( !mStartupDone ) { // check mail on startup if ( group.readEntry( "checkmail-startup", false ) ) // do it after building the kmmainwin, so that the progressdialog is available QTimer::singleShot( 0, this, SLOT( slotCheckMail() ) ); } } if ( layoutChanged ) { layoutSplitters(); } updateMessageMenu(); updateFileMenu(); toggleSystemTray(); } //----------------------------------------------------------------------------- void KMMainWidget::writeConfig() { KConfig *config = KMKernel::config(); KConfigGroup geometry( config, "Geometry" ); KConfigGroup general( config, "General" ); mFolderTree->writeConfig(); // Don't save the sizes of all the widgets when we were never shown. // This can happen in Kontact, where the KMail plugin is automatically // loaded, but not necessarily shown. // This prevents invalid sizes from being saved if ( mWasEverShown ) { GlobalSettings::self()->setFolderViewWidth( mFolderTree->width() ); // The height of the header widget can be 0, this happens when the user // did not switch to the header widget onced and the "Welcome to KMail" // HTML widget was shown the whole time int headersHeight = mSearchAndHeaders->height(); if ( headersHeight == 0 ) headersHeight = height() / 2; GlobalSettings::self()->setSearchAndHeaderHeight( headersHeight ); GlobalSettings::self()->setSearchAndHeaderWidth( mSearchAndHeaders->width() ); if ( mFavoriteFolderView ) { GlobalSettings::self()->setFavoriteFolderViewHeight( mFavoriteFolderView->height() ); GlobalSettings::self()->setFolderTreeHeight( mFolderTree->height() ); mFavoriteFolderView->writeConfig(); if ( !mLongFolderList ) GlobalSettings::self()->setFolderViewHeight( mFolderViewSplitter->height() ); } else if ( !mLongFolderList ) GlobalSettings::self()->setFolderTreeHeight( mFolderTree->height() ); if ( mMsgView ) { if ( !mReaderWindowBelow ) GlobalSettings::self()->setReaderWindowWidth( mMsgView->width() ); mMsgView->writeConfig(); GlobalSettings::self()->setReaderWindowHeight( mMsgView->width() ); } } // save the state of the unread/total-columns geometry.writeEntry( "UnreadColumn", mFolderTree->unreadIndex() ); geometry.writeEntry( "TotalColumn", mFolderTree->totalIndex() ); geometry.writeEntry( "SizeColumn", mFolderTree->sizeIndex() ); } //----------------------------------------------------------------------------- void KMMainWidget::deleteWidgets() { // Simply delete the top splitter, which always is mSplitter1, regardless // of the layout. This deletes all children. delete mSplitter1; mSearchAndHeaders = 0; mSearchToolBar = 0; mHeaders = 0; mQuickSearchLine = 0; mMsgView = 0; mSearchAndTree = 0; mFolderTree = 0; mFavoriteFolderView = 0; mFolderViewSplitter = 0; mSplitter1 = 0; mSplitter2 = 0; } //----------------------------------------------------------------------------- void KMMainWidget::createWidgets() { // Note that all widgets we create in this function have the parent 'this'. // They will be properly reparented in layoutSplitters() // // Create header view and search bar // mSearchAndHeaders = new KVBox( this ); mSearchToolBar = new QWidget( mSearchAndHeaders); mSearchToolBar->setObjectName( "search toolbar" ); mSearchToolBar->setLayout( new QHBoxLayout() ); mSearchToolBar->layout()->setSpacing( KDialog::spacingHint() ); mSearchToolBar->layout()->setMargin( 0 ); mHeaders = new KMHeaders( this, mSearchAndHeaders ); mHeaders->setObjectName( "headers" ); mQuickSearchLine = new HeaderListQuickSearch( mSearchToolBar, mHeaders ); mQuickSearchLine->setObjectName( "headers quick search line" ); connect( mQuickSearchLine, SIGNAL( requestFullSearch() ), this, SLOT( slotRequestFullSearchFromQuickSearch() ) ); connect( mHeaders, SIGNAL( messageListUpdated() ), mQuickSearchLine, SLOT( updateSearch() ) ); if ( !GlobalSettings::self()->quickSearchActive() ) mSearchToolBar->hide(); connect( mHeaders, SIGNAL( activated(KMMessage*) ), this, SLOT( slotMsgActivated(KMMessage*) ) ); connect( mHeaders, SIGNAL( selectionChanged() ), SLOT( startUpdateMessageActionsTimer() ) ); mPreviousMessageAction = new KAction( i18n( "Extend Selection to Previous Message" ), this ); mPreviousMessageAction->setShortcut( QKeySequence( Qt::SHIFT + Qt::Key_Left ) ); actionCollection()->addAction( "previous_message", mPreviousMessageAction ); connect( mPreviousMessageAction, SIGNAL( triggered( bool ) ), mHeaders, SLOT( selectPrevMessage() ) ); mNextMessageAction = new KAction( i18n( "Extend Selection to Next Message" ), this ); mNextMessageAction->setShortcut( QKeySequence( Qt::SHIFT + Qt::Key_Right ) ); actionCollection()->addAction( "next_message", mNextMessageAction ); connect( mNextMessageAction, SIGNAL( triggered( bool ) ), mHeaders, SLOT( selectNextMessage() ) ); // // Create the reader window // if ( mReaderWindowActive ) { mMsgView = new KMReaderWin( this, this, actionCollection(), 0 ); if ( mMsgActions ) { mMsgActions->setMessageView( mMsgView ); } connect( mMsgView, SIGNAL( replaceMsgByUnencryptedVersion() ), this, SLOT( slotReplaceMsgByUnencryptedVersion() ) ); connect( mMsgView, SIGNAL( popupMenu(KMMessage&,const KUrl&,const QPoint&) ), this, SLOT( slotMsgPopup(KMMessage&,const KUrl&,const QPoint&) ) ); connect( mMsgView, SIGNAL( urlClicked(const KUrl&,int) ), mMsgView, SLOT( slotUrlClicked() ) ); connect( mHeaders, SIGNAL( maybeDeleting() ), mMsgView, SLOT( clearCache() ) ); connect( mMsgView, SIGNAL( noDrag() ), mHeaders, SLOT( slotNoDrag() ) ); connect( mHeaders, SIGNAL( selected(KMMessage*) ), this, SLOT( slotMsgSelected(KMMessage*) ) ); } // // Create the folder tree // the "folder tree" consists of a quicksearch input field and the tree itself // mSearchAndTree = new QWidget( this ); QVBoxLayout *vboxlayout = new QVBoxLayout; vboxlayout->setMargin(0); mFolderQuickSearch = new K3ListViewSearchLine( mSearchAndTree ); mFolderTree = new KMFolderTree( this, mSearchAndTree, "folderTree" ); mFolderQuickSearch->setListView( mFolderTree ); mFolderQuickSearch->setClickMessage( i18n( "Search" ) ); vboxlayout->addWidget( mFolderQuickSearch ); vboxlayout->addWidget( mFolderTree ); mSearchAndTree->setLayout( vboxlayout ); if ( !GlobalSettings::enableFolderQuickSearch() ) { mFolderQuickSearch->hide(); } connect( mFolderTree, SIGNAL( folderSelected(KMFolder*) ), this, SLOT( folderSelected(KMFolder*) ) ); connect( mFolderTree, SIGNAL( folderSelected( KMFolder* ) ), mQuickSearchLine, SLOT( reset() ) ); connect( mFolderTree, SIGNAL( folderSelectedUnread(KMFolder*) ), this, SLOT( folderSelectedUnread(KMFolder*) ) ); connect( mFolderTree, SIGNAL( folderDrop(KMFolder*) ), this, SLOT( slotMoveMsgToFolder(KMFolder*) ) ); connect( mFolderTree, SIGNAL( folderDropCopy(KMFolder*) ), this, SLOT( slotCopyMsgToFolder(KMFolder*) ) ); connect( mFolderTree, SIGNAL( columnsChanged() ), this, SLOT( slotFolderTreeColumnsChanged() ) ); // // Create the favorite folder view // if ( mEnableFavoriteFolderView ) { mFavoriteFolderView = new KMail::FavoriteFolderView( this, this ); if ( mFavoritesCheckMailAction ) connect( mFavoritesCheckMailAction, SIGNAL(triggered(bool)), mFavoriteFolderView, SLOT( checkMail() ) ); connect( mFolderTree, SIGNAL( folderSelected(KMFolder*) ), mFavoriteFolderView, SLOT( folderTreeSelectionChanged(KMFolder*) ) ); connect( mFavoriteFolderView, SIGNAL( folderDrop(KMFolder*) ), SLOT( slotMoveMsgToFolder(KMFolder*) ) ); connect( mFavoriteFolderView, SIGNAL( folderDropCopy(KMFolder*) ), SLOT( slotCopyMsgToFolder(KMFolder*) ) ); } // // Create all kinds of actions // { mRemoveDuplicatesAction = new KAction( i18n("Remove Duplicate Messages"), this ); actionCollection()->addAction( "remove_duplicate_messages", mRemoveDuplicatesAction ); connect( mRemoveDuplicatesAction, SIGNAL( triggered( bool ) ), SLOT( removeDuplicates() ) ); mRemoveDuplicatesAction->setShortcut( QKeySequence( Qt::CTRL + Qt::Key_Asterisk ) ); } { mMoveMsgToFolderAction = new KAction( i18n("Move Message to Folder"), this ); mMoveMsgToFolderAction->setShortcut( QKeySequence( Qt::Key_M ) ); actionCollection()->addAction( "move_message_to_folder", mMoveMsgToFolderAction ); connect( mMoveMsgToFolderAction, SIGNAL( triggered( bool ) ), SLOT( slotMoveMsg() ) ); } { KAction *action = new KAction( i18n("Copy Message to Folder"), this ); actionCollection()->addAction( "copy_message_to_folder", action ); connect( action, SIGNAL( triggered( bool ) ), SLOT( slotCopyMsg() ) ); action->setShortcut( QKeySequence( Qt::Key_C ) ); } { KAction *action = new KAction( i18n("Jump to Folder..."), this ); actionCollection()->addAction( "jump_to_folder", action ); connect( action, SIGNAL( triggered ( bool ) ), SLOT( slotJumpToFolder() ) ); action->setShortcut( QKeySequence( Qt::Key_J ) ); } { KAction *action = new KAction(i18n("Abort Current Operation"), this); actionCollection()->addAction("cancel", action ); connect( action, SIGNAL( triggered( bool ) ), ProgressManager::instance(), SLOT( slotAbortAll() ) ); action->setShortcut( QKeySequence( Qt::Key_Escape ) ); } { KAction *action = new KAction(i18n("Focus on Next Folder"), this); actionCollection()->addAction("inc_current_folder", action ); connect( action, SIGNAL( triggered( bool ) ), mFolderTree, SLOT( incCurrentFolder() ) ); action->setShortcut( QKeySequence( Qt::CTRL+Qt::Key_Right ) ); } { KAction *action = new KAction(i18n("Focus on Previous Folder"), this); actionCollection()->addAction("dec_current_folder", action ); connect( action, SIGNAL( triggered( bool ) ), mFolderTree, SLOT( decCurrentFolder() ) ); action->setShortcut( QKeySequence( Qt::CTRL+Qt::Key_Left ) ); } { KAction *action = new KAction(i18n("Select Folder with Focus"), this); actionCollection()->addAction("select_current_folder", action ); connect( action, SIGNAL( triggered( bool ) ), mFolderTree, SLOT( selectCurrentFolder() ) ); action->setShortcut( QKeySequence( Qt::CTRL+Qt::Key_Space ) ); } { KAction *action = new KAction(i18n("Focus on Next Message"), this); actionCollection()->addAction("inc_current_message", action ); connect( action, SIGNAL( triggered( bool) ), mHeaders, SLOT( incCurrentMessage() ) ); action->setShortcut( QKeySequence( Qt::ALT+Qt::Key_Right ) ); } { KAction *action = new KAction(i18n("Focus on Previous Message"), this); actionCollection()->addAction("dec_current_message", action ); connect( action, SIGNAL( triggered( bool ) ), mHeaders, SLOT( decCurrentMessage() ) ); action->setShortcut( QKeySequence( Qt::ALT+Qt::Key_Left ) ); } { KAction *action = new KAction(i18n("Select Message with Focus"), this); actionCollection()->addAction( "select_current_message", action ); connect( action, SIGNAL( triggered( bool ) ), mHeaders, SLOT( selectCurrentMessage() ) ); action->setShortcut( QKeySequence( Qt::ALT+Qt::Key_Space ) ); } connect( kmkernel->outboxFolder(), SIGNAL( msgRemoved(int, const QString&) ), SLOT( startUpdateMessageActionsTimer() ) ); connect( kmkernel->outboxFolder(), SIGNAL( msgAdded(int) ), SLOT( startUpdateMessageActionsTimer() ) ); } //------------------------------------------------------------------------- void KMMainWidget::slotSearch() { if(!mSearchWin) { mSearchWin = new SearchWindow(this, mFolder); mSearchWin->setModal( false ); mSearchWin->setObjectName( "Search" ); connect(mSearchWin, SIGNAL(destroyed()), this, SLOT(slotSearchClosed())); } else { mSearchWin->activateFolder(mFolder); } mSearchWin->show(); KWindowSystem::activateWindow( mSearchWin->winId() ); } //------------------------------------------------------------------------- void KMMainWidget::slotSearchClosed() { mSearchWin = 0; } //------------------------------------------------------------------------- void KMMainWidget::slotFind() { if( mMsgView ) mMsgView->slotFind(); } //----------------------------------------------------------------------------- void KMMainWidget::slotHelp() { KToolInvocation::invokeHelp(); } //----------------------------------------------------------------------------- void KMMainWidget::slotFilter() { kmkernel->filterMgr()->openDialog( this ); } //----------------------------------------------------------------------------- void KMMainWidget::slotPopFilter() { kmkernel->popFilterMgr()->openDialog( this ); } void KMMainWidget::slotManageSieveScripts() { if ( !kmkernel->askToGoOnline() ) { return; } KMail::ManageSieveScriptsDialog * dlg = new KMail::ManageSieveScriptsDialog( this ); dlg->show(); } //----------------------------------------------------------------------------- void KMMainWidget::slotAddrBook() { KPIM::KAddrBookExternal::openAddressBook(this); } //----------------------------------------------------------------------------- void KMMainWidget::slotImport() { KRun::runCommand("kmailcvt", topLevelWidget()); } //----------------------------------------------------------------------------- void KMMainWidget::slotCheckMail() { if ( !kmkernel->askToGoOnline() ) { return; } kmkernel->acctMgr()->checkMail(true); } //----------------------------------------------------------------------------- void KMMainWidget::slotCheckOneAccount( QAction* item ) { if ( ! item ) { return; } if ( !kmkernel->askToGoOnline() ) { return; } KMAccount* t = kmkernel->acctMgr()->findByName( item->data().toString() ); if ( t ) { kmkernel->acctMgr()->singleCheckMail( t ); } else { kDebug() <<" - account with name '" << item->data().toString() <<"' not found"; } } //----------------------------------------------------------------------------- void KMMainWidget::slotMailChecked( bool newMail, bool sendOnCheck, const QMap & newInFolder ) { const bool sendOnAll = GlobalSettings::self()->sendOnCheck() == GlobalSettings::EnumSendOnCheck::SendOnAllChecks; const bool sendOnManual = GlobalSettings::self()->sendOnCheck() == GlobalSettings::EnumSendOnCheck::SendOnManualChecks; if( !kmkernel->isOffline() && ( sendOnAll || (sendOnManual && sendOnCheck ) ) ) slotSendQueued(); if ( !newMail || newInFolder.isEmpty() ) return; QDBusMessage message = QDBusMessage::createSignal( "/KMail", "org.kde.kmail.kmail", "unreadCountChanged" ); QDBusConnection::sessionBus().send( message ); // build summary for new mail message bool showNotification = false; QString summary; QStringList keys( newInFolder.keys() ); keys.sort(); for ( QStringList::const_iterator it = keys.begin(); it != keys.end(); ++it ) { kDebug() << newInFolder.find( *it ).value() << "new message(s) in" << *it; KMFolder *folder = kmkernel->findFolderById( *it ); if ( folder && !folder->ignoreNewMail() ) { showNotification = true; if ( GlobalSettings::self()->verboseNewMailNotification() ) { summary += "
" + i18np( "1 new message in %2", "%1 new messages in %2", newInFolder.find( *it ).value(), folder->prettyUrl() ); } } } // update folder menus in case some mail got filtered to trash/current folder // and we can enable "empty trash/move all to trash" action etc. updateFolderMenu(); if ( !showNotification ) return; if ( GlobalSettings::self()->verboseNewMailNotification() ) { summary = i18nc( "%1 is a list of the number of new messages per folder", "New mail arrived
%1", summary ); } else { summary = i18n( "New mail arrived" ); } if(kmkernel->xmlGuiInstance().isValid()) { KNotification::event( "new-mail-arrived", summary,QPixmap(),topLevelWidget(), KNotification::CloseOnTimeout, kmkernel->xmlGuiInstance() ); } else KNotification::event( "new-mail-arrived", summary,QPixmap(),topLevelWidget(), KNotification::CloseOnTimeout, kmkernel->xmlGuiInstance() ); if (mBeepOnNew) { KNotification::beep(); } } //----------------------------------------------------------------------------- void KMMainWidget::slotCompose() { KMail::Composer * win; KMMessage* msg = new KMMessage; if ( mFolder ) { msg->initHeader( mFolder->identity() ); TemplateParser parser( msg, TemplateParser::NewMessage, QString(), false, false, false ); parser.process( NULL, mFolder ); win = KMail::makeComposer( msg, mFolder->identity() ); } else { msg->initHeader(); TemplateParser parser( msg, TemplateParser::NewMessage, QString(), false, false, false ); parser.process( NULL, NULL ); win = KMail::makeComposer( msg ); } win->show(); } //----------------------------------------------------------------------------- // TODO: do we want the list sorted alphabetically? void KMMainWidget::slotShowNewFromTemplate() { if ( mFolder ) { const KPIMIdentities::Identity & ident = kmkernel->identityManager()->identityForUoidOrDefault( mFolder->identity() ); mTemplateFolder = kmkernel->folderMgr()->findIdString( ident.templates() ); } if ( !mTemplateFolder ) mTemplateFolder = kmkernel->templatesFolder(); if ( !mTemplateFolder ) return; mTemplateMenu->menu()->clear(); for ( int idx = 0; idxcount(); ++idx ) { KMMsgBase *mb = mTemplateFolder->getMsgBase( idx ); QString subj = mb->subject(); if ( subj.isEmpty() ) subj = i18n("No Subject"); QAction *templateAction = mTemplateMenu->menu()->addAction( KStringHandler::rsqueeze( subj.replace( "&", "&&" ) ) ); templateAction->setData( idx ); } // If there are no templates available, add a menu entry which informs // the user about this. if ( mTemplateMenu->menu()->actions().isEmpty() ) { QAction *noAction = mTemplateMenu->menu()->addAction( i18n( "(no templates)" ) ); noAction->setEnabled( false ); } } //----------------------------------------------------------------------------- void KMMainWidget::slotNewFromTemplate( QAction *action ) { if ( !mTemplateFolder ) return; newFromTemplate( mTemplateFolder->getMsg( action->data().toInt() ) ); } //----------------------------------------------------------------------------- void KMMainWidget::newFromTemplate( KMMessage *msg ) { if ( !msg ) return; KMCommand *command = new KMUseTemplateCommand( this, msg ); command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotPostToML() { if ( mFolder && mFolder->isMailingListEnabled() ) { KMCommand *command = new KMMailingListPostCommand( this, mFolder ); command->start(); } else slotCompose(); } //----------------------------------------------------------------------------- void KMMainWidget::slotFolderMailingListProperties() { if (!mFolderTree) return; KMFolderTreeItem *item = static_cast( mFolderTree->currentItem() ); if ( !item ) return; KMFolder* folder = item->folder(); if ( folder ) { ( new KMail::MailingListFolderPropertiesDialog( this, folder ) )->show(); } } //----------------------------------------------------------------------------- void KMMainWidget::slotFolderShortcutCommand() { if (!mFolderTree) return; KMFolderTreeItem *item = static_cast( mFolderTree->currentItem() ); if ( item ) item->assignShortcut(); } //----------------------------------------------------------------------------- void KMMainWidget::slotModifyFolder() { if (!mFolderTree) return; KMFolderTreeItem *item = static_cast( mFolderTree->currentItem() ); if ( item ) modifyFolder( item ); } //----------------------------------------------------------------------------- void KMMainWidget::modifyFolder( KMFolderTreeItem* folderItem ) { KMFolder* folder = folderItem->folder(); KMFolderTree* folderTree = static_cast( folderItem->listView() ); KMFolderDialog props( folder, folder->parent(), folderTree, i18n("Properties of Folder %1", folder->label() ) ); props.exec(); updateFolderMenu(); //Kolab issue 2152 if ( mSystemTray ) mSystemTray->foldersChanged(); } //----------------------------------------------------------------------------- void KMMainWidget::slotExpireFolder() { QString str; bool canBeExpired = true; if (!mFolder) return; if (!mFolder->isAutoExpire()) { canBeExpired = false; } else if (mFolder->getUnreadExpireUnits()==expireNever && mFolder->getReadExpireUnits()==expireNever) { canBeExpired = false; } if (!canBeExpired) { str = i18n("This folder does not have any expiry options set"); KMessageBox::information(this, str); return; } KConfig *config = KMKernel::config(); KConfigGroup group(config, "General"); if (group.readEntry("warn-before-expire", true ) ) { str = i18n("Are you sure you want to expire the folder %1?", Qt::escape( mFolder->label() )); if (KMessageBox::warningContinueCancel(this, str, i18n("Expire Folder"), KGuiItem(i18n("&Expire"))) != KMessageBox::Continue) return; } mFolder->expireOldMessages( true /*immediate*/); } //----------------------------------------------------------------------------- void KMMainWidget::slotEmptyFolder() { QString str; if (!mFolder) return; bool isTrash = kmkernel->folderIsTrash(mFolder); if (mConfirmEmpty) { QString title = (isTrash) ? i18n("Empty Trash") : i18n("Move to Trash"); QString text = (isTrash) ? i18n("Are you sure you want to empty the trash folder?") : i18n("Are you sure you want to move all messages from " "folder %1 to the trash?", Qt::escape( mFolder->label() ) ); if (KMessageBox::warningContinueCancel(this, text, title, KGuiItem( title, "user-trash")) != KMessageBox::Continue) return; } KCursorSaver busy(KBusyPtr::busy()); slotMarkAll(); if (isTrash) { /* Don't ask for confirmation again when deleting, the user has already confirmed. */ slotDeleteMsg( false ); } else slotTrashMsg(); if (mMsgView) mMsgView->clearCache(); if ( !isTrash ) BroadcastStatus::instance()->setStatusMsg(i18n("Moved all messages to the trash")); updateMessageActions(); // Disable empty trash/move all to trash action - we've just deleted/moved // all folder contents. mEmptyFolderAction->setEnabled( false ); } //----------------------------------------------------------------------------- void KMMainWidget::slotArchiveFolder() { KMail::ArchiveFolderDialog archiveDialog; archiveDialog.setFolder( mFolder ); archiveDialog.exec(); } //----------------------------------------------------------------------------- void KMMainWidget::slotRemoveFolder() { QString str; QDir dir; if ( !mFolder ) return; if ( mFolder->isSystemFolder() ) return; if ( mFolder->isReadOnly() ) return; QString title; if ( mFolder->folderType() == KMFolderTypeSearch ) { title = i18n("Delete Search"); str = i18n("Are you sure you want to delete the search %1?
" "Any messages it shows will still be available in their original folder.
", Qt::escape( mFolder->label() ) ); } else { title = i18n("Delete Folder"); if ( mFolder->count() == 0 ) { if ( !mFolder->child() || mFolder->child()->isEmpty() ) { str = i18n("Are you sure you want to delete the empty folder " "%1?", Qt::escape( mFolder->label() ) ); } else { str = i18n("Are you sure you want to delete the empty folder " "%1 and all its subfolders? Those subfolders might " "not be empty and their contents will be discarded as well. " "

Beware that discarded messages are not saved " "into your Trash folder and are permanently deleted.", Qt::escape( mFolder->label() ) ); } } else { if ( !mFolder->child() || mFolder->child()->isEmpty() ) { str = i18n("Are you sure you want to delete the folder " "%1, discarding its contents? " "

Beware that discarded messages are not saved " "into your Trash folder and are permanently deleted.", Qt::escape( mFolder->label() ) ); } else { str = i18n("Are you sure you want to delete the folder %1 " "and all its subfolders, discarding their contents? " "

Beware that discarded messages are not saved " "into your Trash folder and are permanently deleted.", Qt::escape( mFolder->label() ) ); } } } if ( KMessageBox::warningContinueCancel( this, str, title, KGuiItem( i18n("&Delete"), "edit-delete" ), KStandardGuiItem::cancel(), "", KMessageBox::Notify | KMessageBox::Dangerous ) == KMessageBox::Continue ) { KMail::FolderUtil::deleteFolder( mFolder, this ); } } //----------------------------------------------------------------------------- void KMMainWidget::slotMarkAllAsRead() { if (!mFolder) return; mFolder->markUnreadAsRead(); } //----------------------------------------------------------------------------- void KMMainWidget::slotCompactFolder() { if (mFolder) { int idx = mHeaders->currentItemIndex(); KCursorSaver busy(KBusyPtr::busy()); mFolder->compact( KMFolder::CompactNow ); // setCurrentItemByIndex will override the statusbar message, so save/restore it QString statusMsg = BroadcastStatus::instance()->statusMsg(); mHeaders->setCurrentItemByIndex(idx); BroadcastStatus::instance()->setStatusMsg( statusMsg ); } } //----------------------------------------------------------------------------- void KMMainWidget::slotRefreshFolder() { if (mFolder) { if ( mFolder->folderType() == KMFolderTypeImap || mFolder->folderType() == KMFolderTypeCachedImap ) { if ( !kmkernel->askToGoOnline() ) { return; } } if (mFolder->folderType() == KMFolderTypeImap) { KMFolderImap *imap = static_cast(mFolder->storage()); imap->getAndCheckFolder(); } else if ( mFolder->folderType() == KMFolderTypeCachedImap ) { KMFolderCachedImap* f = static_cast( mFolder->storage() ); f->account()->processNewMailInFolder( mFolder ); } } } void KMMainWidget::slotTroubleshootFolder() { if (mFolder) { if ( mFolder->folderType() == KMFolderTypeCachedImap ) { KMFolderCachedImap* f = static_cast( mFolder->storage() ); f->slotTroubleshoot(); } } } void KMMainWidget::slotTroubleshootMaildir() { if ( !mFolder || !mFolder->folderType() == KMFolderTypeMaildir ) return; KMFolderMaildir* f = static_cast( mFolder->storage() ); if ( KMessageBox::warningContinueCancel( this, i18nc( "@info", "You are about to recreate the index for folder %1." "This will destroy all message status information." "Are you sure you want to continue?", mFolder->label() ), i18nc( "@title", "Really recreate index?" ), KGuiItem( i18nc( "@action:button", "Recreate Index" ) ), KStandardGuiItem::cancel(), QString(), KMessageBox::Notify | KMessageBox::Dangerous ) == KMessageBox::Continue ) { f->createIndexFromContents(); KMessageBox::information( this, i18n( "The index of folder %1 has been recreated.", mFolder->label() ), i18n( "Index recreated" ) ); } } void KMMainWidget::slotInvalidateIMAPFolders() { if ( KMessageBox::warningContinueCancel( this, i18n("Are you sure you want to refresh the IMAP cache?\n" "This will remove all changes that you have done " "locally to your IMAP folders."), i18n("Refresh IMAP Cache"), KGuiItem(i18n("&Refresh")) ) == KMessageBox::Continue ) kmkernel->acctMgr()->invalidateIMAPFolders(); } //----------------------------------------------------------------------------- void KMMainWidget::slotExpireAll() { KConfig *config = KMKernel::config(); int ret = 0; KConfigGroup group(config, "General"); if (group.readEntry("warn-before-expire", true ) ) { ret = KMessageBox::warningContinueCancel(KMainWindow::memberList().first(), i18n("Are you sure you want to expire all old messages?"), i18n("Expire Old Messages?"), KGuiItem(i18n("Expire"))); if (ret != KMessageBox::Continue) { return; } } kmkernel->expireAllFoldersNow(); } //----------------------------------------------------------------------------- void KMMainWidget::slotCompactAll() { KCursorSaver busy(KBusyPtr::busy()); kmkernel->compactAllFolders(); } //----------------------------------------------------------------------------- void KMMainWidget::slotOverrideHtml() { if( mHtmlPref == mFolderHtmlPref ) { int result = KMessageBox::warningContinueCancel( this, // the warning text is taken from configuredialog.cpp: i18n( "Use of HTML in mail will make you more vulnerable to " "\"spam\" and may increase the likelihood that your system will be " "compromised by other present and anticipated security exploits." ), i18n( "Security Warning" ), KGuiItem(i18n( "Use HTML" )), KStandardGuiItem::cancel(), "OverrideHtmlWarning", false); if( result == KMessageBox::Cancel ) { mPreferHtmlAction->setChecked( false ); return; } } mFolderHtmlPref = !mFolderHtmlPref; if (mMsgView) { mMsgView->setHtmlOverride(mFolderHtmlPref); mMsgView->update( true ); } } //----------------------------------------------------------------------------- void KMMainWidget::slotOverrideHtmlLoadExt() { if( mHtmlLoadExtPref == mFolderHtmlLoadExtPref ) { int result = KMessageBox::warningContinueCancel( this, // the warning text is taken from configuredialog.cpp: i18n( "Loading external references in html mail will make you more vulnerable to " "\"spam\" and may increase the likelihood that your system will be " "compromised by other present and anticipated security exploits." ), i18n( "Security Warning" ), KGuiItem(i18n( "Load External References" )), KStandardGuiItem::cancel(), "OverrideHtmlLoadExtWarning", false); if( result == KMessageBox::Cancel ) { mPreferHtmlLoadExtAction->setChecked( false ); return; } } mFolderHtmlLoadExtPref = !mFolderHtmlLoadExtPref; if (mMsgView) { mMsgView->setHtmlLoadExtOverride(mFolderHtmlLoadExtPref); mMsgView->update( true ); } } //----------------------------------------------------------------------------- void KMMainWidget::slotOverrideThread() { mFolderThreadPref = !mFolderThreadPref; mHeaders->setNestedOverride(mFolderThreadPref); mThreadBySubjectAction->setEnabled(mThreadMessagesAction->isChecked()); } //----------------------------------------------------------------------------- void KMMainWidget::slotToggleSubjectThreading() { mFolderThreadSubjPref = !mFolderThreadSubjPref; mHeaders->setSubjectThreading(mFolderThreadSubjPref); } //----------------------------------------------------------------------------- void KMMainWidget::slotMessageQueuedOrDrafted() { if (!kmkernel->folderIsDraftOrOutbox(mFolder)) return; if (mMsgView) mMsgView->update(true); } //----------------------------------------------------------------------------- void KMMainWidget::slotForwardInlineMsg() { KMMessageList* selected = mHeaders->selectedMsgs(); KMCommand *command = 0L; if(selected && !selected->isEmpty()) { command = new KMForwardCommand( this, *selected, mFolder->identity() ); } else { command = new KMForwardCommand( this, mHeaders->currentMsg(), mFolder->identity() ); } command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotForwardAttachedMsg() { KMMessageList* selected = mHeaders->selectedMsgs(); KMCommand *command = 0L; if(selected && !selected->isEmpty()) { command = new KMForwardAttachedCommand( this, *selected, mFolder->identity() ); } else { command = new KMForwardAttachedCommand( this, mHeaders->currentMsg(), mFolder->identity() ); } command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotUseTemplate() { newFromTemplate( mHeaders->currentMsg() ); } //----------------------------------------------------------------------------- void KMMainWidget::slotResendMsg() { KMCommand *command = new KMResendMessageCommand( this, mHeaders->currentMsg() ); command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotTrashMsg() { mHeaders->deleteMsg(); } //----------------------------------------------------------------------------- void KMMainWidget::slotDeleteMsg( bool confirmDelete ) { mHeaders->moveMsgToFolder( 0, confirmDelete ); } //----------------------------------------------------------------------------- void KMMainWidget::slotTrashThread() { mHeaders->highlightCurrentThread(); mHeaders->deleteMsg(); } //----------------------------------------------------------------------------- void KMMainWidget::slotDeleteThread( bool confirmDelete ) { mHeaders->highlightCurrentThread(); mHeaders->moveMsgToFolder( 0, confirmDelete ); } //----------------------------------------------------------------------------- void KMMainWidget::slotRedirectMsg() { KMCommand *command = new KMRedirectCommand( this, mHeaders->currentMsg() ); command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotCustomReplyToMsg( const QString &tmpl ) { QString text = mMsgView? mMsgView->copyText() : ""; kDebug() << "Reply with template:" << tmpl; KMCommand *command = new KMCustomReplyToCommand( this, mHeaders->currentMsg(), text, tmpl ); command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotCustomReplyAllToMsg( const QString &tmpl ) { QString text = mMsgView? mMsgView->copyText() : ""; kDebug() << "Reply to All with template:" << tmpl; KMCommand *command = new KMCustomReplyAllToCommand( this, mHeaders->currentMsg(), text, tmpl ); command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotCustomForwardMsg( const QString &tmpl ) { kDebug() << "Forward with template:" << tmpl; KMMessageList* selected = mHeaders->selectedMsgs(); KMCommand *command = 0L; if(selected && !selected->isEmpty()) { command = new KMCustomForwardCommand( this, *selected, mFolder->identity(), tmpl ); } else { command = new KMCustomForwardCommand( this, mHeaders->currentMsg(), mFolder->identity(), tmpl ); } command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotNoQuoteReplyToMsg() { KMCommand *command = new KMNoQuoteReplyToCommand( this, mHeaders->currentMsg() ); command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotSubjectFilter() { KMMessage *msg = mHeaders->currentMsg(); if (!msg) return; KMCommand *command = new KMFilterCommand( "Subject", msg->subject() ); command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotMailingListFilter() { KMMessage *msg = mHeaders->currentMsg(); if (!msg) return; KMCommand *command = new KMMailingListFilterCommand( this, msg ); command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotFromFilter() { KMMessage *msg = mHeaders->currentMsg(); if (!msg) return; AddrSpecList al = msg->extractAddrSpecs( "From" ); KMCommand *command; if ( al.empty() ) command = new KMFilterCommand( "From", msg->from() ); else command = new KMFilterCommand( "From", al.front().asString() ); command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotToFilter() { KMMessage *msg = mHeaders->currentMsg(); if (!msg) return; KMCommand *command = new KMFilterCommand( "To", msg->to() ); command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::updateListFilterAction() { //Proxy the mListFilterAction to update the action text QByteArray name; QString value; QString lname = MailingList::name( mHeaders->currentMsg(), name, value ); mListFilterAction->setText( i18n("Filter on Mailing-List...") ); if ( lname.isNull() ) mListFilterAction->setEnabled( false ); else { mListFilterAction->setEnabled( true ); mListFilterAction->setText( i18n( "Filter on Mailing-List %1...", lname ) ); } } //----------------------------------------------------------------------------- void KMMainWidget::slotUndo() { mHeaders->undo(); updateMessageActions(); updateFolderMenu(); } //----------------------------------------------------------------------------- void KMMainWidget::slotToggleUnread() { mFolderTree->toggleColumn(KMFolderTree::unread); } //----------------------------------------------------------------------------- void KMMainWidget::slotToggleTotalColumn() { mFolderTree->toggleColumn(KMFolderTree::total, true); } void KMMainWidget::slotToggleSizeColumn() { mFolderTree->toggleColumn(KMFolderTree::foldersize); } //----------------------------------------------------------------------------- void KMMainWidget::slotJumpToFolder() { KMail::FolderSelectionDialog dlg( this, i18n("Jump to Folder"), false ); KMFolder* dest; if (!dlg.exec()) return; if (!(dest = dlg.folder())) return; slotSelectFolder( dest ); } //----------------------------------------------------------------------------- void KMMainWidget::slotMoveMsg() { KMail::FolderSelectionDialog dlg( this, i18n("Move Message to Folder"), true ); KMFolder* dest; if (!dlg.exec()) return; if (!(dest = dlg.folder())) return; mHeaders->moveMsgToFolder(dest); } //----------------------------------------------------------------------------- void KMMainWidget::slotMoveMsgToFolder( KMFolder *dest) { mHeaders->moveMsgToFolder(dest); } //----------------------------------------------------------------------------- void KMMainWidget::slotCopyMsgToFolder( KMFolder *dest) { mHeaders->copyMsgToFolder(dest); } //----------------------------------------------------------------------------- void KMMainWidget::slotApplyFilters() { mHeaders->applyFiltersOnMsg(); } //----------------------------------------------------------------------------- void KMMainWidget::slotCheckVacation() { updateVactionScriptStatus( false ); if ( !kmkernel->askToGoOnline() ) return; Vacation *vac = new Vacation( this, true /* check only */ ); connect( vac, SIGNAL(scriptActive(bool)), SLOT(updateVactionScriptStatus(bool)) ); } void KMMainWidget::slotEditVacation() { if ( !kmkernel->askToGoOnline() ) { return; } if ( mVacation ) return; mVacation = new Vacation( this ); connect( mVacation, SIGNAL(scriptActive(bool)), SLOT(updateVactionScriptStatus(bool)) ); if ( mVacation->isUsable() ) { connect( mVacation, SIGNAL(result(bool)), mVacation, SLOT(deleteLater()) ); } else { QString msg = i18n("KMail's Out of Office Reply functionality relies on " "server-side filtering. You have not yet configured an " "IMAP server for this.\n" "You can do this on the \"Filtering\" tab of the IMAP " "account configuration."); KMessageBox::sorry( this, msg, i18n("No Server-Side Filtering Configured") ); delete mVacation; // QGuardedPtr sets itself to 0! } } //----------------------------------------------------------------------------- void KMMainWidget::slotDebugSieve() { #if !defined(NDEBUG) if ( mSieveDebugDialog ) return; mSieveDebugDialog = new SieveDebugDialog( this ); mSieveDebugDialog->exec(); delete mSieveDebugDialog; #endif } //----------------------------------------------------------------------------- void KMMainWidget::slotStartCertManager() { if( !QProcess::startDetached("kleopatra" ) ) KMessageBox::error( this, i18n( "Could not start certificate manager; " "please check your installation." ), i18n( "KMail Error" ) ); else kDebug() << "\nslotStartCertManager(): certificate manager started."; } //----------------------------------------------------------------------------- void KMMainWidget::slotStartWatchGnuPG() { if( !QProcess::startDetached("kwatchgnupg") ) KMessageBox::error( this, i18n( "Could not start GnuPG LogViewer (kwatchgnupg); " "please check your installation." ), i18n( "KMail Error" ) ); } //----------------------------------------------------------------------------- void KMMainWidget::slotCopyMsg() { KMail::FolderSelectionDialog dlg( this, i18n("Copy Message to Folder"), true ); KMFolder* dest; if (!dlg.exec()) return; if (!(dest = dlg.folder())) return; mHeaders->copyMsgToFolder(dest); } //----------------------------------------------------------------------------- void KMMainWidget::slotPrintMsg() { bool htmlOverride = mMsgView ? mMsgView->htmlOverride() : false; bool htmlLoadExtOverride = mMsgView ? mMsgView->htmlLoadExtOverride() : false; KConfigGroup reader( KMKernel::config(), "Reader" ); bool useFixedFont = mMsgView ? mMsgView->isFixedFont() : reader.readEntry( "useFixedFont", false ); KMCommand *command = new KMPrintCommand( this, mHeaders->currentMsg(), mMsgView ? mMsgView->headerStyle() : 0, mMsgView ? mMsgView->headerStrategy() : 0, htmlOverride, htmlLoadExtOverride, useFixedFont, overrideEncoding() ); command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotConfigChanged() { readConfig(); mMsgActions->setupForwardActions(); mMsgActions->setupForwardingActionsList( mGUIClient ); } //----------------------------------------------------------------------------- void KMMainWidget::slotSaveMsg() { KMMessage *msg = mHeaders->currentMsg(); if (!msg) return; KMSaveMsgCommand *saveCommand = new KMSaveMsgCommand( this, *mHeaders->selectedMsgs() ); if (saveCommand->url().isEmpty()) delete saveCommand; else saveCommand->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotOpenMsg() { KMOpenMsgCommand *openCommand = new KMOpenMsgCommand( this, KUrl(), overrideEncoding() ); openCommand->start(); } //----------------------------------------------------------------------------- void KMMainWidget::slotSaveAttachments() { KMMessage *msg = mHeaders->currentMsg(); if (!msg) return; KMSaveAttachmentsCommand *saveCommand = new KMSaveAttachmentsCommand( this, *mHeaders->selectedMsgs() ); saveCommand->start(); } void KMMainWidget::slotOnlineStatus() { // KMKernel will emit a signal when we toggle the network state that is caught by // KMMainWidget::slotUpdateOnlineStatus to update our GUI if ( GlobalSettings::self()->networkState() == GlobalSettings::EnumNetworkState::Online ) { // if online; then toggle and set it offline. kmkernel->stopNetworkJobs(); } else { kmkernel->resumeNetworkJobs(); slotCheckVacation(); } } void KMMainWidget::slotUpdateOnlineStatus( GlobalSettings::EnumNetworkState::type ) { QAction *action = actionCollection()->action( "online_status" ); if ( GlobalSettings::self()->networkState() == GlobalSettings::EnumNetworkState::Online ) { action->setText( i18n("Work Offline") ); action->setIcon( KIcon("user-offline") ); } else { action->setText( i18n("Work Online") ); action->setIcon( KIcon("user-online") ); } } //----------------------------------------------------------------------------- void KMMainWidget::slotSendQueued() { if ( !kmkernel->askToGoOnline() ) { return; } kmkernel->msgSender()->sendQueued(); } //----------------------------------------------------------------------------- void KMMainWidget::slotSendQueuedVia( QAction* item ) { if ( !kmkernel->askToGoOnline() ) { return; } QStringList availTransports= MailTransport::TransportManager::self()->transportNames(); if (availTransports.contains(item->text())) kmkernel->msgSender()->sendQueued( item->text() ); } //----------------------------------------------------------------------------- void KMMainWidget::folderSelectedUnread( KMFolder* aFolder ) { folderSelected( aFolder, true ); slotChangeCaption( mFolderTree->currentItem() ); } void KMMainWidget::openFolder() { if ( !mFolder || mFolder->folderType() != KMFolderTypeImap ) { return; } KMFolderImap *imap = static_cast(mFolder->storage()); assert( !mOpenedImapFolder ); imap->open("mainwidget"); // will be closed in the folderSelected slot mOpenedImapFolder = true; // first get new headers before we select the folder imap->setSelected( true ); // If the folder gets closed because of renaming, re-open it again to prevent // an assert in closeFolder() later. disconnect( imap, SIGNAL( closed( KMFolder* ) ), this, SLOT( folderClosed( KMFolder*) ) ); connect( imap, SIGNAL( closed( KMFolder* ) ), this, SLOT( folderClosed( KMFolder*) ) ); } void KMMainWidget::closeFolder() { if ( !mFolder || mFolder->folderType() != KMFolderTypeImap ) { return; } assert( mOpenedImapFolder ); KMFolderImap *imap = static_cast(mFolder->storage()); imap->setSelected( false ); mFolder->close( "mainwidget" ); mOpenedImapFolder = false; } //----------------------------------------------------------------------------- void KMMainWidget::folderClosed( KMFolder *folder ) { Q_UNUSED( folder ); if ( !mFolder || mFolder->folderType() != KMFolderTypeImap ) { return; } mOpenedImapFolder = false; openFolder(); } //----------------------------------------------------------------------------- void KMMainWidget::folderSelected() { folderSelected( mFolder ); updateFolderMenu(); } //----------------------------------------------------------------------------- void KMMainWidget::folderSelected( KMFolder* aFolder, bool forceJumpToUnread ) { KCursorSaver busy(KBusyPtr::busy()); if (mMsgView) mMsgView->clear(true); if ( mFolder && mFolder->folderType() == KMFolderTypeImap && !mFolder->noContent() ) { KMFolderImap *imap = static_cast(mFolder->storage()); if ( mFolder->needsCompacting() && imap->autoExpunge() ) imap->expungeFolder(imap, true); } // Re-enable the msg list and quicksearch if we're showing a splash // screen. This is true either if there's no active folder, or if we // have a timer that is no longer active (i.e. it has already fired) // To make the if() a bit more complicated, we suppress the hiding // when the new folder is also an IMAP folder, because that's an // async operation and we don't want flicker if it results in just // a new splash. bool newFolder = ( (KMFolder*)mFolder != aFolder ); bool isNewImapFolder = aFolder && aFolder->folderType() == KMFolderTypeImap && newFolder; if( !mFolder || ( !isNewImapFolder && mShowBusySplashTimer ) || ( newFolder && mShowingOfflineScreen && !( isNewImapFolder && kmkernel->isOffline() ) ) ) { if ( mMsgView ) { mMsgView->enableMsgDisplay(); mMsgView->clear( true ); } if( mSearchAndHeaders && mHeaders ) mSearchAndHeaders->show(); mShowingOfflineScreen = false; } // Delete any pending timer, if needed it will be recreated below delete mShowBusySplashTimer; mShowBusySplashTimer = 0; if ( newFolder ) { writeFolderConfig(); } if ( mFolder ) { disconnect( mFolder, SIGNAL( changed() ), this, SLOT( updateMarkAsReadAction() ) ); disconnect( mFolder, SIGNAL( msgHeaderChanged( KMFolder*, int ) ), this, SLOT( updateMarkAsReadAction() ) ); disconnect( mFolder, SIGNAL( msgAdded( int ) ), this, SLOT( updateMarkAsReadAction() ) ); disconnect( mFolder, SIGNAL( msgRemoved( KMFolder * ) ), this, SLOT( updateMarkAsReadAction() ) ); } if ( newFolder ) { closeFolder(); } mFolder = aFolder; if ( newFolder ) { openFolder(); } if ( aFolder && aFolder->folderType() == KMFolderTypeImap ) { if ( kmkernel->isOffline() ) { showOfflinePage(); return; } KMFolderImap *imap = static_cast( aFolder->storage() ); if ( newFolder && !mFolder->noContent() ) { connect( imap, SIGNAL( folderComplete( KMFolderImap*, bool ) ), this, SLOT( folderSelected() ) ); imap->getAndCheckFolder(); mHeaders->setFolder( 0 ); updateFolderMenu(); mForceJumpToUnread = forceJumpToUnread; // Set a timer to show a splash screen if fetching folder contents // takes more than the amount of seconds configured in the kmailrc (default 1000 msec) mShowBusySplashTimer = new QTimer( this ); mShowBusySplashTimer->setSingleShot( true ); connect( mShowBusySplashTimer, SIGNAL( timeout() ), this, SLOT( slotShowBusySplash() ) ); mShowBusySplashTimer->start( GlobalSettings::self()->folderLoadingTimeout() ); return; } else { // the folder is complete now - so go ahead disconnect( imap, SIGNAL( folderComplete( KMFolderImap*, bool ) ), this, SLOT( folderSelected() ) ); forceJumpToUnread = mForceJumpToUnread; } } if ( mFolder ) { // == 0 -> pointing to toplevel ("Welcome to KMail") folder connect( mFolder, SIGNAL( changed() ), this, SLOT( updateMarkAsReadAction() ) ); connect( mFolder, SIGNAL( msgHeaderChanged( KMFolder*, int ) ), this, SLOT( updateMarkAsReadAction() ) ); connect( mFolder, SIGNAL( msgAdded( int ) ), this, SLOT( updateMarkAsReadAction() ) ); connect( mFolder, SIGNAL( msgRemoved(KMFolder *) ), this, SLOT( updateMarkAsReadAction() ) ); } readFolderConfig(); if (mMsgView) { mMsgView->setHtmlOverride(mFolderHtmlPref); mMsgView->setHtmlLoadExtOverride(mFolderHtmlLoadExtPref); } mHeaders->setFolder( mFolder, forceJumpToUnread ); updateMessageActions(); updateFolderMenu(); if (!aFolder) slotIntro(); } //----------------------------------------------------------------------------- void KMMainWidget::slotShowBusySplash() { if ( mReaderWindowActive ) { mMsgView->displayBusyPage(); // hide widgets that are in the way: if ( mSearchAndHeaders && mHeaders && mLongFolderList ) mSearchAndHeaders->hide(); } } void KMMainWidget::showOfflinePage() { if ( !mReaderWindowActive ) return; mShowingOfflineScreen = true; mMsgView->displayOfflinePage(); // hide widgets that are in the way: if ( mSearchAndHeaders && mHeaders && mLongFolderList ) mSearchAndHeaders->hide(); } //----------------------------------------------------------------------------- void KMMainWidget::slotMsgSelected(KMMessage *msg) { if ( msg && msg->parent() && !msg->isComplete() ) { if ( msg->transferInProgress() ) return; mMsgView->clear(); mMsgView->setWaitingForSerNum( msg->getMsgSerNum() ); if ( mJob ) { disconnect( mJob, 0, mMsgView, 0 ); delete mJob; } mJob = msg->parent()->createJob( msg, FolderJob::tGetMessage, msg->parent(), "STRUCTURE", mMsgView->attachmentStrategy() ); connect(mJob, SIGNAL(messageRetrieved(KMMessage*)), mMsgView, SLOT(slotMessageArrived(KMMessage*))); mJob->start(); } else { mMsgView->setMsg(msg); } // reset HTML override to the folder setting mMsgView->setHtmlOverride(mFolderHtmlPref); mMsgView->setHtmlLoadExtOverride(mFolderHtmlLoadExtPref); mMsgView->setDecryptMessageOverwrite( false ); mMsgView->setShowSignatureDetails( false ); } //----------------------------------------------------------------------------- void KMMainWidget::slotMsgChanged() { mHeaders->msgChanged(); } //----------------------------------------------------------------------------- void KMMainWidget::slotSelectFolder(KMFolder* folder) { Q3ListViewItem* item = mFolderTree->indexOfFolder(folder); if ( item ) { mFolderTree->ensureItemVisible( item ); mFolderTree->doFolderSelected( item ); } } //----------------------------------------------------------------------------- void KMMainWidget::slotSelectMessage(KMMessage* msg) { int idx = mFolder->find(msg); if (idx != -1) { mHeaders->setCurrentMsg(idx); if (mMsgView) mMsgView->setMsg(msg); else slotMsgActivated(msg); } } //----------------------------------------------------------------------------- void KMMainWidget::slotReplaceMsgByUnencryptedVersion() { kDebug(); KMMessage* oldMsg = mHeaders->currentMsg(); if( oldMsg ) { kDebug() << "Old message found"; if( oldMsg->hasUnencryptedMsg() ) { kDebug() << "Extra unencrypted message found"; KMMessage* newMsg = oldMsg->unencryptedMsg(); // adjust the message id { QString msgId( oldMsg->msgId() ); QString prefix("DecryptedMsg."); int oldIdx = msgId.indexOf(prefix, 0, Qt::CaseInsensitive); if( -1 == oldIdx ) { int leftAngle = msgId.lastIndexOf( '<' ); msgId = msgId.insert( (-1 == leftAngle) ? 0 : ++leftAngle, prefix ); } else { // toggle between "DecryptedMsg." and "DeCryptedMsg." // to avoid same message id QCharRef c = msgId[ oldIdx+2 ]; if( 'C' == c ) c = 'c'; else c = 'C'; } newMsg->setMsgId( msgId ); mMsgView->setIdOfLastViewedMessage( msgId ); } // insert the unencrypted message kDebug() << "Adding unencrypted message to folder"; mFolder->addMsg( newMsg ); /* Figure out its index in the folder for selecting. This must be count()-1, * since we append. Be safe and do find, though, just in case. */ int newMsgIdx = mFolder->find( newMsg ); Q_ASSERT( newMsgIdx != -1 ); /* we need this unget, to have the message displayed correctly initially */ mFolder->unGetMsg( newMsgIdx ); int idx = mFolder->find( oldMsg ); Q_ASSERT( idx != -1 ); /* only select here, so the old one is not un-Gotten before, which would * render the pointer we hold invalid so that find would fail */ mHeaders->setCurrentItemByIndex( newMsgIdx ); // remove the old one if ( idx != -1 ) { kDebug() << "Deleting encrypted message"; mFolder->take( idx ); } kDebug() << "Updating message actions"; updateMessageActions(); kDebug() << "Done."; } else kDebug() << "NO EXTRA UNENCRYPTED MESSAGE FOUND"; } else kDebug() << "PANIC: NO OLD MESSAGE FOUND"; } //----------------------------------------------------------------------------- void KMMainWidget::slotSetThreadStatusNew() { mHeaders->setThreadStatus( MessageStatus::statusNew() ); } //----------------------------------------------------------------------------- void KMMainWidget::slotSetThreadStatusUnread() { mHeaders->setThreadStatus( MessageStatus::statusUnread() ); } //----------------------------------------------------------------------------- void KMMainWidget::slotSetThreadStatusImportant() { mHeaders->setThreadStatus( MessageStatus::statusImportant(), true); } //----------------------------------------------------------------------------- void KMMainWidget::slotSetThreadStatusRead() { mHeaders->setThreadStatus( MessageStatus::statusRead()); } //----------------------------------------------------------------------------- void KMMainWidget::slotSetThreadStatusTodo() { mHeaders->setThreadStatus( MessageStatus::statusTodo(), true); } //----------------------------------------------------------------------------- void KMMainWidget::slotSetThreadStatusWatched() { mHeaders->setThreadStatus( MessageStatus::statusWatched(), true); if (mWatchThreadAction->isChecked()) { mIgnoreThreadAction->setChecked(false); } } //----------------------------------------------------------------------------- void KMMainWidget::slotSetThreadStatusIgnored() { mHeaders->setThreadStatus( MessageStatus::statusIgnored(), true); if (mIgnoreThreadAction->isChecked()) { mWatchThreadAction->setChecked(false); } } //----------------------------------------------------------------------------- void KMMainWidget::slotNextMessage() { mHeaders->nextMessage(); } void KMMainWidget::slotNextUnreadMessage() { if ( !mHeaders->nextUnreadMessage() ) { if ( GlobalSettings::self()->loopOnGotoUnread() == GlobalSettings::EnumLoopOnGotoUnread::LoopInAllFolders ) { mFolderTree->nextUnreadFolder( true ); } } } void KMMainWidget::slotPrevMessage() { mHeaders->prevMessage(); } void KMMainWidget::slotPrevUnreadMessage() { if ( !mHeaders->prevUnreadMessage() ) { if ( GlobalSettings::self()->loopOnGotoUnread() == GlobalSettings::EnumLoopOnGotoUnread::LoopInAllFolders ) { mFolderTree->prevUnreadFolder(); } } } void KMMainWidget::slotDisplayCurrentMessage() { if ( mHeaders->currentMsg() ) slotMsgActivated( mHeaders->currentMsg() ); } //----------------------------------------------------------------------------- void KMMainWidget::slotMsgActivated(KMMessage *msg) { if ( !msg ) return; if (msg->parent() && !msg->isComplete()) { FolderJob *job = msg->parent()->createJob(msg); connect(job, SIGNAL(messageRetrieved(KMMessage*)), SLOT(slotMsgActivated(KMMessage*))); job->start(); return; } if (kmkernel->folderIsDraftOrOutbox(mFolder)) { mMsgActions->editCurrentMessage(); return; } if ( kmkernel->folderIsTemplates( mFolder ) ) { slotUseTemplate(); return; } assert( msg != 0 ); KMReaderMainWin *win = new KMReaderMainWin( mFolderHtmlPref, mFolderHtmlLoadExtPref ); KConfigGroup reader( KMKernel::config(), "Reader" ); bool useFixedFont = mMsgView ? mMsgView->isFixedFont() : reader.readEntry( "useFixedFont", false ); win->setUseFixedFont( useFixedFont ); KMMessage *newMessage = new KMMessage(*msg); newMessage->setParent( msg->parent() ); newMessage->setMsgSerNum( msg->getMsgSerNum() ); newMessage->setReadyToShow( true ); win->showMsg( overrideEncoding(), newMessage ); win->show(); } //----------------------------------------------------------------------------- void KMMainWidget::slotMarkAll() { mHeaders->selectAll( true ); updateMessageActions(); } //----------------------------------------------------------------------------- void KMMainWidget::slotMsgPopup(KMMessage&, const KUrl &aUrl, const QPoint& aPoint) { KMenu *menu = new KMenu; updateMessageMenu(); mUrlCurrent = aUrl; bool urlMenuAdded = false; if ( !aUrl.isEmpty() ) { if ( aUrl.protocol() == "mailto" ) { // popup on a mailto URL menu->addAction( mMsgView->mailToComposeAction() ); menu->addAction( mMsgView->mailToReplyAction() ); menu->addAction( mMsgView->mailToForwardAction() ); menu->addSeparator(); QString email = KPIMUtils::firstEmailAddress( aUrl.path() ); KABC::AddressBook *addressBook = KABC::StdAddressBook::self( true ); KABC::Addressee::List addresseeList = addressBook->findByEmail( email ); if ( addresseeList.count() == 0 ) { menu->addAction( mMsgView->addAddrBookAction() ); } else { menu->addAction( mMsgView->openAddrBookAction() ); } menu->addAction( mMsgView->copyURLAction() ); } else { // popup on a not-mailto URL menu->addAction( mMsgView->urlOpenAction() ); menu->addAction( mMsgView->addBookmarksAction() ); menu->addAction( mMsgView->urlSaveAsAction() ); menu->addAction( mMsgView->copyURLAction() ); } urlMenuAdded = true; kDebug() <<" URL is:" << aUrl; } if ( mMsgView && !mMsgView->copyText().isEmpty() ) { if ( urlMenuAdded ) { menu->addSeparator(); } menu->addAction( mMsgActions->replyMenu() ); menu->addSeparator(); menu->addAction( mMsgView->copyAction() ); menu->addAction( mMsgView->selectAllAction() ); } else if ( !urlMenuAdded ) { // popup somewhere else (i.e., not a URL) on the message if (!mHeaders->currentMsg()) // no messages { delete menu; return; } if ( mFolder->isTemplates() ) { menu->addAction( mUseAction ); } else { menu->addAction( mMsgActions->replyMenu() ); menu->addAction( mMsgActions->forwardMenu() ); } menu->addAction(editAction()); menu->addSeparator(); menu->addAction( mCopyActionMenu ); menu->addAction( mMoveActionMenu ); menu->addSeparator(); menu->addAction( mMsgActions->messageStatusMenu() ); menu->addSeparator(); menu->addAction( viewSourceAction() ); if ( mMsgView ) { menu->addAction( mMsgView->toggleFixFontAction() ); } menu->addSeparator(); menu->addAction( mPrintAction ); menu->addAction( mSaveAsAction ); menu->addAction( mSaveAttachmentsAction ); menu->addSeparator(); if ( mFolder->isTrash() ) { menu->addAction( mDeleteAction ); } else { menu->addAction( mTrashAction ); } menu->addSeparator(); menu->addAction( mMsgActions->createTodoAction() ); } KAcceleratorManager::manage(menu); menu->exec( aPoint, 0 ); delete menu; } //----------------------------------------------------------------------------- void KMMainWidget::getAccountMenu() { QStringList actList; mActMenu->clear(); actList = kmkernel->acctMgr()->getAccounts(); QStringList::Iterator it; foreach ( QString accountName, actList ) { QAction* action = mActMenu->addAction( accountName.replace("&", "&&") ); action->setData( accountName ); } } //----------------------------------------------------------------------------- void KMMainWidget::getTransportMenu() { QStringList availTransports; mSendMenu->clear(); availTransports = MailTransport::TransportManager::self()->transportNames(); QStringList::Iterator it; for(it = availTransports.begin(); it != availTransports.end() ; ++it) mSendMenu->addAction((*it).replace("&", "&&")); } //----------------------------------------------------------------------------- void KMMainWidget::updateCustomTemplateMenus() { if ( !mCustomTemplateMenus ) { mCustomTemplateMenus = new CustomTemplatesMenu( this, actionCollection() ); connect( mCustomTemplateMenus, SIGNAL(replyTemplateSelected( const QString& )), this, SLOT(slotCustomReplyToMsg( const QString& )) ); connect( mCustomTemplateMenus, SIGNAL(replyAllTemplateSelected( const QString& )), this, SLOT(slotCustomReplyAllToMsg( const QString& )) ); connect( mCustomTemplateMenus, SIGNAL(forwardTemplateSelected( const QString& )), this, SLOT(slotCustomForwardMsg( const QString& )) ); } mMsgActions->forwardMenu()->addSeparator(); mMsgActions->forwardMenu()->addAction( mCustomTemplateMenus->forwardActionMenu() ); mMsgActions->replyMenu()->addSeparator(); mMsgActions->replyMenu()->addAction( mCustomTemplateMenus->replyActionMenu() ); mMsgActions->replyMenu()->addAction( mCustomTemplateMenus->replyAllActionMenu() ); } //----------------------------------------------------------------------------- void KMMainWidget::setupActions() { mMsgActions = new KMail::MessageActions( actionCollection(), this ); mMsgActions->setMessageView( mMsgView ); //----- File Menu mSaveAsAction = new KAction(KIcon("document-save"), i18n("Save &As..."), this); actionCollection()->addAction("file_save_as", mSaveAsAction ); connect(mSaveAsAction, SIGNAL(triggered(bool) ), SLOT(slotSaveMsg())); mSaveAsAction->setShortcut(KStandardShortcut::save()); mOpenAction = KStandardAction::open( this, SLOT( slotOpenMsg() ), actionCollection() ); { KAction *action = new KAction(i18n("&Compact All Folders"), this); actionCollection()->addAction("compact_all_folders", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotCompactAll())); } { KAction *action = new KAction(i18n("&Expire All Folders"), this); actionCollection()->addAction("expire_all_folders", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotExpireAll())); } { KAction *action = new KAction(KIcon("view-refresh"), i18n("&Refresh Local IMAP Cache"), this); actionCollection()->addAction("file_invalidate_imap_cache", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotInvalidateIMAPFolders())); } { KAction *action = new KAction(i18n("Empty All &Trash Folders"), this); actionCollection()->addAction("empty_trash", action ); connect(action, SIGNAL(triggered(bool) ), KMKernel::self(), SLOT(slotEmptyTrash())); } { KAction *action = new KAction(KIcon("mail-receive"), i18n("Check &Mail"), this); actionCollection()->addAction("check_mail", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotCheckMail())); action->setShortcut(QKeySequence(Qt::CTRL+Qt::Key_L)); } mFavoritesCheckMailAction = new KAction( KIcon( "mail-receive"), i18n( "Check Mail in Favorite Folders" ), this ); actionCollection()->addAction( "favorite_check_mail", mFavoritesCheckMailAction ); mFavoritesCheckMailAction->setShortcut( QKeySequence( Qt::CTRL+Qt::SHIFT+Qt::Key_L ) ); if ( mFavoriteFolderView ) { connect( mFavoritesCheckMailAction, SIGNAL(triggered(bool)), mFavoriteFolderView, SLOT(checkMail()) ); } KActionMenu *actActionMenu = new KActionMenu(KIcon("mail-receive"), i18n("Check Ma&il"), this); actionCollection()->addAction("check_mail_in", actActionMenu ); actActionMenu->setDelayed(true); //needed for checking "all accounts" connect(actActionMenu, SIGNAL(triggered(bool)), this, SLOT(slotCheckMail())); mActMenu = actActionMenu->menu(); connect(mActMenu, SIGNAL(triggered(QAction*)), SLOT(slotCheckOneAccount(QAction*))); connect(mActMenu, SIGNAL(aboutToShow()), SLOT(getAccountMenu())); { KAction *action = new KAction(KIcon("mail-send"), i18n("&Send Queued Messages"), this); actionCollection()->addAction("send_queued", action ); connect(action, SIGNAL(triggered(bool)), SLOT(slotSendQueued())); } { KAction *action = new KAction( i18n("Onlinestatus (unknown)"), this ); actionCollection()->addAction( "online_status", action ); connect( action, SIGNAL(triggered(bool)), SLOT(slotOnlineStatus()) ); } KActionMenu *sendActionMenu = new KActionMenu(KIcon("mail-send-via"), i18n("Send Queued Messages Via"), this); actionCollection()->addAction("send_queued_via", sendActionMenu ); sendActionMenu->setDelayed(true); mSendMenu = sendActionMenu->menu(); connect(mSendMenu,SIGNAL(triggered(QAction*)), SLOT(slotSendQueuedVia(QAction*))); connect(mSendMenu,SIGNAL(aboutToShow()),SLOT(getTransportMenu())); //----- Tools menu if (parent()->inherits("KMMainWin")) { KAction *action = new KAction(KIcon("help-contents"), i18n("&Address Book"), this); actionCollection()->addAction("addressbook", action ); connect(action, SIGNAL(triggered(bool)), SLOT(slotAddrBook())); if (KStandardDirs::findExe("kaddressbook").isEmpty()) action->setEnabled(false); } { KAction *action = new KAction(KIcon("pgp-keys"), i18n("Certificate Manager"), this); actionCollection()->addAction("tools_start_certman", action ); connect(action, SIGNAL(triggered(bool)), SLOT(slotStartCertManager())); // disable action if no certman binary is around if (KStandardDirs::findExe("kleopatra").isEmpty()) action->setEnabled(false); } { KAction *action = new KAction(KIcon("pgp-keys"), i18n("GnuPG Log Viewer"), this); actionCollection()->addAction("tools_start_kwatchgnupg", action ); connect(action, SIGNAL(triggered(bool)), SLOT(slotStartWatchGnuPG())); // disable action if no kwatchgnupg binary is around bool usableKWatchGnupg = !KStandardDirs::findExe("kwatchgnupg").isEmpty(); #ifdef Q_OS_WIN32 // not ported yet, underlying infrastructure missing on Windows usableKWatchGnupg = false; #endif action->setEnabled(usableKWatchGnupg); } { KAction *action = new KAction(KIcon("document-import"), i18n("&Import Messages"), this); actionCollection()->addAction("import", action ); connect(action, SIGNAL(triggered(bool)), SLOT(slotImport())); if (KStandardDirs::findExe("kmailcvt").isEmpty()) action->setEnabled(false); } #if !defined(NDEBUG) { KAction *action = new KAction(i18n("&Debug Sieve..."), this); actionCollection()->addAction("tools_debug_sieve", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotDebugSieve())); } #endif { KAction *action = new KAction(i18n("Filter &Log Viewer..."), this); actionCollection()->addAction("filter_log_viewer", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotFilterLogViewer())); } { KAction *action = new KAction(i18n("&Anti-Spam Wizard..."), this); actionCollection()->addAction("antiSpamWizard", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotAntiSpamWizard())); } { KAction *action = new KAction(i18n("&Anti-Virus Wizard..."), this); actionCollection()->addAction("antiVirusWizard", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotAntiVirusWizard())); } if ( GlobalSettings::allowOutOfOfficeSettings() ) { KAction *action = new KAction( i18n("Edit \"Out of Office\" Replies..."), this ); actionCollection()->addAction( "tools_edit_vacation", action ); connect( action, SIGNAL(triggered(bool)), SLOT(slotEditVacation()) ); } //----- Edit Menu mTrashAction = new KAction(i18n("&Move to Trash"), this); actionCollection()->addAction("move_to_trash", mTrashAction ); mTrashAction->setIcon(KIcon("user-trash")); mTrashAction->setIconText( i18n( "Trash" ) ); mTrashAction->setShortcut(QKeySequence(Qt::Key_Delete)); mTrashAction->setToolTip(i18n("Move message to trashcan")); connect(mTrashAction, SIGNAL(triggered(bool)), SLOT(slotTrashMsg())); /* The delete action is nowhere in the gui, by default, so we need to make * sure it is plugged into the KAccel now, since that won't happen on * XMLGui construction or manual ->plug(). This is only a problem when run * as a part, though. */ mDeleteAction = new KAction(KIcon("edit-delete"), i18n("&Delete"), this); actionCollection()->addAction("delete", mDeleteAction ); connect(mDeleteAction, SIGNAL(triggered(bool)), SLOT(slotDeleteMsg())); mDeleteAction->setShortcut(QKeySequence(Qt::SHIFT+Qt::Key_Delete)); mTrashThreadAction = new KAction(i18n("M&ove Thread to Trash"), this); actionCollection()->addAction("move_thread_to_trash", mTrashThreadAction ); mTrashThreadAction->setShortcut(QKeySequence(Qt::CTRL+Qt::Key_Delete)); mTrashThreadAction->setIcon(KIcon("user-trash")); mTrashThreadAction->setToolTip(i18n("Move thread to trashcan") ); connect(mTrashThreadAction, SIGNAL(triggered(bool)), SLOT(slotTrashThread())); mDeleteThreadAction = new KAction(KIcon("edit-delete"), i18n("Delete T&hread"), this); actionCollection()->addAction("delete_thread", mDeleteThreadAction ); connect(mDeleteThreadAction, SIGNAL(triggered(bool)), SLOT(slotDeleteThread())); mDeleteThreadAction->setShortcut(QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_Delete)); { KAction *action = new KAction(KIcon("edit-find-mail"), i18n("&Find Messages..."), this); actionCollection()->addAction("search_messages", action ); connect(action, SIGNAL(triggered(bool)), SLOT(slotRequestFullSearchFromQuickSearch())); action->setShortcut(QKeySequence(Qt::Key_S)); } mFindInMessageAction = new KAction(KIcon("edit-find"), i18n("&Find in Message..."), this); actionCollection()->addAction("find_in_messages", mFindInMessageAction ); connect(mFindInMessageAction, SIGNAL(triggered(bool)), SLOT(slotFind())); //FIXME: this causes ambiguous Ctrl-F problems and the shortcut fails //unfortunately, this has the side-effect of having a blank Ctrl-F shortcut //string on the Find menu. //mFindInMessageAction->setShortcut(KStandardShortcut::find()); { KAction *action = new KAction(i18n("Select &All Messages"), this); actionCollection()->addAction("mark_all_messages", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotMarkAll())); action->setShortcut(KStandardShortcut::selectAll()); } //----- Folder Menu mNewFolderAction = new KAction( KIcon("folder-new"), i18n("&New Folder..."), this ); actionCollection()->addAction( "new_folder", mNewFolderAction ); connect( mNewFolderAction, SIGNAL(triggered(bool)), mFolderTree, SLOT(addChildFolder()) ); mModifyFolderAction = new KAction(KIcon("document-properties"), i18n("&Properties"), this); actionCollection()->addAction("modify", mModifyFolderAction ); connect(mModifyFolderAction, SIGNAL(triggered(bool)), SLOT(slotModifyFolder())); mFolderMailingListPropertiesAction = new KAction(i18n("&Mailing List Management..."), this); actionCollection()->addAction("folder_mailinglist_properties", mFolderMailingListPropertiesAction ); connect(mFolderMailingListPropertiesAction, SIGNAL(triggered(bool)), SLOT( slotFolderMailingListProperties())); // mFolderMailingListPropertiesAction->setIcon(KIcon("document-properties-mailing-list")); mFolderShortCutCommandAction = new KAction(KIcon("configure-shortcuts"), i18n("&Assign Shortcut..."), this); actionCollection()->addAction("folder_shortcut_command", mFolderShortCutCommandAction ); connect(mFolderShortCutCommandAction, SIGNAL(triggered(bool) ), SLOT( slotFolderShortcutCommand() )); mMarkAllAsReadAction = new KAction(KIcon("mail-mark-read"), i18n("Mark All Messages as &Read"), this); actionCollection()->addAction("mark_all_as_read", mMarkAllAsReadAction ); connect(mMarkAllAsReadAction, SIGNAL(triggered(bool)), SLOT(slotMarkAllAsRead())); mExpireFolderAction = new KAction(i18n("&Expiration Settings"), this); actionCollection()->addAction("expire", mExpireFolderAction ); connect(mExpireFolderAction, SIGNAL(triggered(bool) ), SLOT(slotExpireFolder())); mCompactFolderAction = new KAction(i18n("&Compact Folder"), this); actionCollection()->addAction("compact", mCompactFolderAction ); connect(mCompactFolderAction, SIGNAL(triggered(bool) ), SLOT(slotCompactFolder())); mRefreshFolderAction = new KAction(KIcon("view-refresh"), i18n("Check Mail &in This Folder"), this); actionCollection()->addAction("refresh_folder", mRefreshFolderAction ); connect(mRefreshFolderAction, SIGNAL(triggered(bool) ), SLOT(slotRefreshFolder())); mRefreshFolderAction->setShortcut(KStandardShortcut::reload()); mTroubleshootFolderAction = 0; // set in initializeIMAPActions mTroubleshootMaildirAction = new KAction( KIcon("tools-wizard"), i18n("Rebuild Index..."), this ); actionCollection()->addAction( "troubleshoot_maildir", mTroubleshootMaildirAction ); connect( mTroubleshootMaildirAction, SIGNAL(triggered()), SLOT(slotTroubleshootMaildir()) ); mEmptyFolderAction = new KAction(KIcon("user-trash"), "foo" /*set in updateFolderMenu*/, this); actionCollection()->addAction("empty", mEmptyFolderAction ); connect(mEmptyFolderAction, SIGNAL(triggered(bool)), SLOT(slotEmptyFolder())); mRemoveFolderAction = new KAction(KIcon("edit-delete"), "foo" /*set in updateFolderMenu*/, this); actionCollection()->addAction("delete_folder", mRemoveFolderAction ); connect(mRemoveFolderAction, SIGNAL(triggered(bool)), SLOT(slotRemoveFolder())); mArchiveFolderAction = new KAction( i18n( "&Archive Folder..." ), this ); actionCollection()->addAction( "archive_folder", mArchiveFolderAction ); connect( mArchiveFolderAction, SIGNAL(triggered(bool)), SLOT(slotArchiveFolder()) ); mPreferHtmlAction = new KToggleAction(i18n("Prefer &HTML to Plain Text"), this); actionCollection()->addAction("prefer_html", mPreferHtmlAction ); connect(mPreferHtmlAction, SIGNAL(triggered(bool) ), SLOT(slotOverrideHtml())); mPreferHtmlLoadExtAction = new KToggleAction(i18n("Load E&xternal References"), this); actionCollection()->addAction("prefer_html_external_refs", mPreferHtmlLoadExtAction ); connect(mPreferHtmlLoadExtAction, SIGNAL(triggered(bool) ), SLOT(slotOverrideHtmlLoadExt())); mThreadMessagesAction = new KToggleAction(i18n("&Thread Messages"), this); actionCollection()->addAction("thread_messages", mThreadMessagesAction ); connect(mThreadMessagesAction, SIGNAL(triggered(bool) ), SLOT(slotOverrideThread())); mThreadBySubjectAction = new KToggleAction(i18n("Thread Messages also by &Subject"), this); actionCollection()->addAction("thread_messages_by_subject", mThreadBySubjectAction ); connect(mThreadBySubjectAction, SIGNAL(triggered(bool) ), SLOT(slotToggleSubjectThreading())); { KAction *action = new KAction(KIcon("edit-copy"), i18n("Copy Folder"), this); action->setShortcut(QKeySequence(Qt::SHIFT+Qt::CTRL+Qt::Key_C)); actionCollection()->addAction("copy_folder", action); connect(action, SIGNAL(triggered(bool)), folderTree(), SLOT(copyFolder())); } { KAction *action = new KAction(KIcon("edit-cut"), i18n("Cut Folder"), this); action->setShortcut(QKeySequence(Qt::SHIFT+Qt::CTRL+Qt::Key_X)); actionCollection()->addAction("cut_folder", action); connect(action, SIGNAL(triggered(bool)), folderTree(), SLOT(cutFolder())); } { KAction *action = new KAction(KIcon("edit-paste"), i18n("Paste Folder"), this); action->setShortcut(QKeySequence(Qt::SHIFT+Qt::CTRL+Qt::Key_V)); actionCollection()->addAction("paste_folder", action); connect(action, SIGNAL(triggered(bool)), folderTree(), SLOT(pasteFolder())); } { KAction *action = new KAction(KIcon("edit-copy"), i18n("Copy Messages"), this); action->setShortcut(QKeySequence(Qt::ALT+Qt::CTRL+Qt::Key_C)); actionCollection()->addAction("copy_messages", action); connect(action, SIGNAL(triggered(bool)), headers(), SLOT(copyMessages())); } { KAction *action = new KAction(KIcon("edit-cut"), i18n("Cut Messages"), this); action->setShortcut(QKeySequence(Qt::ALT+Qt::CTRL+Qt::Key_X)); actionCollection()->addAction("cut_messages", action); connect(action, SIGNAL(triggered(bool)), headers(), SLOT(cutMessages())); } { KAction *action = new KAction(KIcon("edit-paste"), i18n("Paste Messages"), this); action->setShortcut(QKeySequence(Qt::ALT+Qt::CTRL+Qt::Key_V)); actionCollection()->addAction("paste_messages", action); connect(action, SIGNAL(triggered(bool)), headers(), SLOT(pasteMessages())); } //----- Message Menu { KAction *action = new KAction(KIcon("mail-message-new"), i18n("&New Message..."), this); actionCollection()->addAction("new_message", action ); action->setIconText( i18nc("@action:intoolbar New Empty Message", "New" ) ); connect(action, SIGNAL(triggered(bool)), SLOT(slotCompose())); // do not set a New shortcut if kmail is a component if ( !kmkernel->xmlGuiInstance().isValid() ) { action->setShortcut(KStandardShortcut::openNew()); } } mTemplateMenu = new KActionMenu( KIcon( "document-new" ), i18n("Message From &Template"), actionCollection() ); mTemplateMenu->setDelayed( true ); actionCollection()->addAction("new_from_template", mTemplateMenu ); connect( mTemplateMenu->menu(), SIGNAL( aboutToShow() ), this, SLOT( slotShowNewFromTemplate() ) ); connect( mTemplateMenu->menu(), SIGNAL( triggered(QAction*) ), this, SLOT( slotNewFromTemplate(QAction*) ) ); { KAction *action = new KAction(KIcon("mail-message-new-list"), i18n("New Message t&o Mailing-List..."), this); actionCollection()->addAction("post_message", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotPostToML())); action->setShortcut(QKeySequence(Qt::CTRL+Qt::SHIFT+Qt::Key_N)); } mSendAgainAction = new KAction(i18n("Send A&gain..."), this); actionCollection()->addAction("send_again", mSendAgainAction ); connect(mSendAgainAction, SIGNAL(triggered(bool) ), SLOT(slotResendMsg())); //----- Create filter actions mFilterMenu = new KActionMenu(KIcon("view-filter"), i18n("&Create Filter"), this); actionCollection()->addAction("create_filter", mFilterMenu ); connect( mFilterMenu, SIGNAL(triggered(bool)), this, SLOT(slotFilter()) ); mSubjectFilterAction = new KAction(i18n("Filter on &Subject..."), this); actionCollection()->addAction("subject_filter", mSubjectFilterAction ); connect(mSubjectFilterAction, SIGNAL(triggered(bool) ), SLOT(slotSubjectFilter())); mFilterMenu->addAction( mSubjectFilterAction ); mFromFilterAction = new KAction(i18n("Filter on &From..."), this); actionCollection()->addAction("from_filter", mFromFilterAction ); connect(mFromFilterAction, SIGNAL(triggered(bool) ), SLOT(slotFromFilter())); mFilterMenu->addAction( mFromFilterAction ); mToFilterAction = new KAction(i18n("Filter on &To..."), this); actionCollection()->addAction("to_filter", mToFilterAction ); connect(mToFilterAction, SIGNAL(triggered(bool) ), SLOT(slotToFilter())); mFilterMenu->addAction( mToFilterAction ); mListFilterAction = new KAction(i18n("Filter on Mailing-&List..."), this); actionCollection()->addAction("mlist_filter", mListFilterAction ); connect(mListFilterAction, SIGNAL(triggered(bool) ), SLOT(slotMailingListFilter())); mFilterMenu->addAction( mListFilterAction ); mPrintAction = KStandardAction::print (this, SLOT(slotPrintMsg()), actionCollection()); mUseAction = new KAction( KIcon("document-new"), i18n("New Message From &Template"), this ); actionCollection()->addAction("use_template", mUseAction); connect(mUseAction, SIGNAL(triggered(bool) ), SLOT(slotUseTemplate())); mUseAction->setShortcut(QKeySequence(Qt::Key_N)); //----- "Mark Thread" submenu mThreadStatusMenu = new KActionMenu(i18n("Mark &Thread"), this); actionCollection()->addAction("thread_status", mThreadStatusMenu ); mMarkThreadAsReadAction = new KAction(KIcon("mail-mark-read"), i18n("Mark Thread as &Read"), this); actionCollection()->addAction("thread_read", mMarkThreadAsReadAction ); connect(mMarkThreadAsReadAction, SIGNAL(triggered(bool) ), SLOT(slotSetThreadStatusRead())); mMarkThreadAsReadAction->setToolTip(i18n("Mark all messages in the selected thread as read")); mThreadStatusMenu->addAction( mMarkThreadAsReadAction ); mMarkThreadAsNewAction = new KAction(KIcon("mail-mark-unread-new"), i18n("Mark Thread as &New"), this); actionCollection()->addAction("thread_new", mMarkThreadAsNewAction ); connect(mMarkThreadAsNewAction, SIGNAL(triggered(bool) ), SLOT(slotSetThreadStatusNew())); mMarkThreadAsNewAction->setToolTip( i18n("Mark all messages in the selected thread as new")); mThreadStatusMenu->addAction( mMarkThreadAsNewAction ); mMarkThreadAsUnreadAction = new KAction(KIcon("mail-mark-unread"), i18n("Mark Thread as &Unread"), this); actionCollection()->addAction("thread_unread", mMarkThreadAsUnreadAction ); connect(mMarkThreadAsUnreadAction, SIGNAL(triggered(bool) ), SLOT(slotSetThreadStatusUnread())); mMarkThreadAsUnreadAction->setToolTip(i18n("Mark all messages in the selected thread as unread")); mThreadStatusMenu->addAction( mMarkThreadAsUnreadAction ); mThreadStatusMenu->addSeparator(); //----- "Mark Thread" toggle actions mToggleThreadImportantAction = new KToggleAction(KIcon("mail-mark-important"), i18n("Mark Thread as &Important"), this); actionCollection()->addAction("thread_flag", mToggleThreadImportantAction ); connect(mToggleThreadImportantAction, SIGNAL(triggered(bool) ), SLOT(slotSetThreadStatusImportant())); mToggleThreadImportantAction->setCheckedState( KGuiItem(i18n("Remove &Important Thread Mark")) ); mThreadStatusMenu->addAction( mToggleThreadImportantAction ); mToggleThreadTodoAction = new KToggleAction(KIcon("mail-mark-task"), i18n("Mark Thread as &Action Item"), this); actionCollection()->addAction("thread_todo", mToggleThreadTodoAction ); connect(mToggleThreadTodoAction, SIGNAL(triggered(bool) ), SLOT(slotSetThreadStatusTodo())); mToggleThreadTodoAction->setCheckedState( KGuiItem(i18n("Remove &To-do Thread Mark")) ); mThreadStatusMenu->addAction( mToggleThreadTodoAction ); //------- "Watch and ignore thread" actions mWatchThreadAction = new KToggleAction(KIcon("mail-thread-watch"), i18n("&Watch Thread"), this); actionCollection()->addAction("thread_watched", mWatchThreadAction ); connect(mWatchThreadAction, SIGNAL(triggered(bool) ), SLOT(slotSetThreadStatusWatched())); mIgnoreThreadAction = new KToggleAction(KIcon("mail-thread-ignored"), i18n("&Ignore Thread"), this); actionCollection()->addAction("thread_ignored", mIgnoreThreadAction ); connect(mIgnoreThreadAction, SIGNAL(triggered(bool) ), SLOT(slotSetThreadStatusIgnored())); mThreadStatusMenu->addSeparator(); mThreadStatusMenu->addAction( mWatchThreadAction ); mThreadStatusMenu->addAction( mIgnoreThreadAction ); mSaveAttachmentsAction = new KAction(KIcon("mail-attachment"), i18n("Save A&ttachments..."), this); actionCollection()->addAction("file_save_attachments", mSaveAttachmentsAction ); connect(mSaveAttachmentsAction, SIGNAL(triggered(bool) ), SLOT(slotSaveAttachments())); mMoveActionMenu = new KActionMenu( KIcon( "go-jump" ), i18n( "&Move To" ), this ); actionCollection()->addAction( "move_to", mMoveActionMenu ); mCopyActionMenu = new KActionMenu( KIcon( "edit-copy" ), i18n( "&Copy To" ), this ); actionCollection()->addAction( "copy_to", mCopyActionMenu ); mApplyAllFiltersAction = new KAction( KIcon( "view-filter" ), i18n( "Appl&y All Filters" ), this ); actionCollection()->addAction( "apply_filters", mApplyAllFiltersAction ); connect( mApplyAllFiltersAction, SIGNAL(triggered(bool)), SLOT(slotApplyFilters()) ); mApplyAllFiltersAction->setShortcut( QKeySequence( Qt::CTRL + Qt::Key_J ) ); mApplyFilterActionsMenu = new KActionMenu( i18n( "A&pply Filter" ), this ); actionCollection()->addAction( "apply_filter_actions", mApplyFilterActionsMenu ); //----- View Menu // Unread Submenu KActionMenu * unreadMenu = new KActionMenu(i18nc("View->", "&Unread Count"), this); actionCollection()->addAction("view_unread", unreadMenu ); unreadMenu->setToolTip( i18n("Choose how to display the count of unread messages") ); QActionGroup *group = new QActionGroup( this ); mUnreadColumnToggle = new KToggleAction(i18nc("View->Unread Count", "View in &Separate Column"), this); actionCollection()->addAction("view_unread_column", mUnreadColumnToggle ); connect( mUnreadColumnToggle, SIGNAL( triggered(Qt::MouseButtons, Qt::KeyboardModifiers) ), SLOT( slotToggleUnread() ) ); group->addAction( mUnreadColumnToggle ); unreadMenu->addAction( mUnreadColumnToggle ); mUnreadTextToggle = new KToggleAction(i18nc("View->Unread Count", "View After &Folder Name"), this); actionCollection()->addAction("view_unread_text", mUnreadTextToggle ); connect( mUnreadTextToggle, SIGNAL( triggered(Qt::MouseButtons, Qt::KeyboardModifiers) ), SLOT( slotToggleUnread() ) ); group->addAction( mUnreadTextToggle ); unreadMenu->addAction( mUnreadTextToggle ); // toggle for total column mTotalColumnToggle = new KToggleAction(i18nc("View->", "&Total Column"), this); actionCollection()->addAction("view_columns_total", mTotalColumnToggle ); connect(mTotalColumnToggle, SIGNAL(triggered (Qt::MouseButtons, Qt::KeyboardModifiers)), SLOT(slotToggleTotalColumn())); mTotalColumnToggle->setToolTip( i18n("Toggle display of column showing the " "total number of messages in folders.") ); mSizeColumnToggle = new KToggleAction( i18nc("View->", "&Size Column"), this ); actionCollection()->addAction("view_columns_size", mSizeColumnToggle ); connect(mSizeColumnToggle, SIGNAL(triggered (Qt::MouseButtons, Qt::KeyboardModifiers)), SLOT(slotToggleSizeColumn())); mSizeColumnToggle->setToolTip( i18n("Toggle display of column showing the " "total size of messages in folders.") ); { KAction *action = new KAction(i18nc("View->","&Expand Thread"), this); actionCollection()->addAction("expand_thread", action ); action->setShortcut(QKeySequence(Qt::Key_Period)); action->setToolTip(i18n("Expand the current thread")); connect(action, SIGNAL(triggered(bool) ), SLOT(slotExpandThread())); } { KAction *action = new KAction(i18nc("View->","&Collapse Thread"), this); actionCollection()->addAction("collapse_thread", action ); action->setShortcut(QKeySequence(Qt::Key_Comma)); action->setToolTip( i18n("Collapse the current thread")); connect(action, SIGNAL(triggered(bool) ), SLOT(slotCollapseThread())); } { KAction *action = new KAction(i18nc("View->","Ex&pand All Threads"), this); actionCollection()->addAction("expand_all_threads", action ); action->setShortcut(QKeySequence(Qt::CTRL+Qt::Key_Period)); action->setToolTip( i18n("Expand all threads in the current folder")); connect(action, SIGNAL(triggered(bool) ), SLOT(slotExpandAllThreads())); } { KAction *action = new KAction(i18nc("View->","C&ollapse All Threads"), this); actionCollection()->addAction("collapse_all_threads", action ); action->setShortcut(QKeySequence(Qt::CTRL+Qt::Key_Comma)); action->setToolTip( i18n("Collapse all threads in the current folder")); connect(action, SIGNAL(triggered(bool) ), SLOT(slotCollapseAllThreads())); } mViewSourceAction = new KAction(i18n("&View Source"), this); actionCollection()->addAction("view_source", mViewSourceAction ); connect(mViewSourceAction, SIGNAL(triggered(bool) ), SLOT(slotShowMsgSrc())); mViewSourceAction->setShortcut(QKeySequence(Qt::Key_V)); KAction *dukeOfMonmoth = new KAction(i18n("&Display Message"), this); actionCollection()->addAction("display_message", dukeOfMonmoth ); connect(dukeOfMonmoth, SIGNAL(triggered(bool) ), SLOT( slotDisplayCurrentMessage() )); dukeOfMonmoth->setShortcut(QKeySequence(Qt::Key_Return)); //----- Go Menu { KAction *action = new KAction(i18n("&Next Message"), this); actionCollection()->addAction("go_next_message", action ); action->setShortcuts(KShortcut( "N; Right" )); action->setToolTip(i18n("Go to the next message")); connect(action, SIGNAL(triggered(bool) ), SLOT(slotNextMessage())); } { KAction *action = new KAction(i18n("Next &Unread Message"), this); actionCollection()->addAction("go_next_unread_message", action ); action->setShortcut(QKeySequence(Qt::Key_Plus)); if ( QApplication::isRightToLeft() ) { action->setIcon( KIcon( "go-previous" ) ); } else { action->setIcon( KIcon( "go-next" ) ); } action->setIconText( i18n( "Next" ) ); action->setToolTip(i18n("Go to the next unread message")); connect(action, SIGNAL(triggered(bool) ), SLOT(slotNextUnreadMessage())); } { KAction *action = new KAction(i18n("&Previous Message"), this); actionCollection()->addAction("go_prev_message", action ); action->setToolTip(i18n("Go to the previous message")); action->setShortcuts(KShortcut( "P; Left" )); connect(action, SIGNAL(triggered(bool) ), SLOT(slotPrevMessage())); } { KAction *action = new KAction(i18n("Previous Unread &Message"), this); actionCollection()->addAction("go_prev_unread_message", action ); action->setShortcut(QKeySequence(Qt::Key_Minus)); if ( QApplication::isRightToLeft() ) { action->setIcon( KIcon( "go-next" ) ); } else { action->setIcon( KIcon( "go-previous" ) ); } action->setIconText( i18n( "Previous" ) ); action->setToolTip(i18n("Go to the previous unread message")); connect(action, SIGNAL(triggered(bool) ), SLOT(slotPrevUnreadMessage())); } { KAction *action = new KAction(i18n("Next Unread &Folder"), this); actionCollection()->addAction("go_next_unread_folder", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotNextUnreadFolder())); action->setShortcut(QKeySequence(Qt::ALT+Qt::Key_Plus)); action->setToolTip(i18n("Go to the next folder with unread messages")); KShortcut shortcut = KShortcut(action->shortcuts()); shortcut.setAlternate( QKeySequence( Qt::CTRL+Qt::Key_Plus ) ); action->setShortcuts( shortcut ); } { KAction *action = new KAction(i18n("Previous Unread F&older"), this); actionCollection()->addAction("go_prev_unread_folder", action ); action->setShortcut(QKeySequence(Qt::ALT+Qt::Key_Minus)); action->setToolTip(i18n("Go to the previous folder with unread messages")); connect(action, SIGNAL(triggered(bool) ), SLOT(slotPrevUnreadFolder())); KShortcut shortcut = KShortcut(action->shortcuts()); shortcut.setAlternate( QKeySequence( Qt::CTRL+Qt::Key_Minus ) ); action->setShortcuts( shortcut ); } { KAction *action = new KAction(i18nc("Go->","Next Unread &Text"), this); actionCollection()->addAction("go_next_unread_text", action ); action->setShortcut(QKeySequence(Qt::Key_Space)); action->setToolTip(i18n("Go to the next unread text")); action->setWhatsThis( i18n("Scroll down current message. " "If at end of current message, " "go to next unread message.")); connect(action, SIGNAL(triggered(bool) ), SLOT(slotReadOn())); } //----- Settings Menu { KAction *action = new KAction(i18n("Configure &Filters..."), this); action->setMenuRole( QAction::NoRole ); // do not move to application menu on OS X actionCollection()->addAction("filter", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotFilter())); } { KAction *action = new KAction(i18n("Configure &POP Filters..."), this); actionCollection()->addAction("popFilter", action ); action->setMenuRole( QAction::NoRole ); // do not move to application menu on OS X connect(action, SIGNAL(triggered(bool) ), SLOT(slotPopFilter())); } { KAction *action = new KAction(i18n("Manage &Sieve Scripts..."), this); actionCollection()->addAction("sieveFilters", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotManageSieveScripts())); } { KAction *action = new KAction(KIcon("kmail"), i18n("KMail &Introduction"), this); actionCollection()->addAction("help_kmail_welcomepage", action ); action->setToolTip( i18n("Display KMail's Welcome Page") ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotIntro())); } // ----- Standard Actions // KStandardAction::configureNotifications(this, SLOT(slotEditNotifications()), actionCollection()); { KAction *action = new KAction( KIcon("preferences-desktop-notification"), i18n("Configure &Notifications..."), this ); action->setMenuRole( QAction::NoRole ); // do not move to application menu on OS X actionCollection()->addAction( "kmail_configure_notifications", action ); connect(action, SIGNAL(triggered(bool) ), SLOT(slotEditNotifications())); } // KStandardAction::preferences(this, SLOT(slotSettings()), actionCollection()); { KAction *action = new KAction(KIcon("configure"), i18n("&Configure KMail..."), this); actionCollection()->addAction("kmail_configure_kmail", action ); connect(action, SIGNAL(triggered(bool) ), kmkernel, SLOT(slotShowConfigurationDialog())); } actionCollection()->addAction(KStandardAction::Undo, "kmail_undo", this, SLOT(slotUndo())); KStandardAction::tipOfDay( this, SLOT( slotShowTip() ), actionCollection() ); menutimer = new QTimer( this ); menutimer->setObjectName( "menutimer" ); menutimer->setSingleShot( true ); connect( menutimer, SIGNAL( timeout() ), SLOT( updateMessageActions() ) ); connect( kmkernel->undoStack(), SIGNAL( undoStackChanged() ), this, SLOT( slotUpdateUndo() )); initializeIMAPActions( false ); // don't set state, config not read yet updateMessageActions(); updateCustomTemplateMenus(); updateFolderMenu(); } //----------------------------------------------------------------------------- void KMMainWidget::slotEditNotifications() { KComponentData d = kmkernel->xmlGuiInstance(); if ( d.isValid() ) { const KAboutData *a = d.aboutData(); KNotifyConfigWidget::configure( this, a->appName() ); } else { KNotifyConfigWidget::configure( this ); } } void KMMainWidget::slotEditKeys() { KShortcutsDialog::configure( actionCollection(), KShortcutsEditor::LetterShortcutsAllowed ); } //----------------------------------------------------------------------------- void KMMainWidget::slotReadOn() { if ( !mMsgView ) return; if ( !mMsgView->atBottom() ) { mMsgView->slotJumpDown(); return; } slotNextUnreadMessage(); } void KMMainWidget::slotNextUnreadFolder() { if ( !mFolderTree ) return; mFolderTree->nextUnreadFolder(); } void KMMainWidget::slotPrevUnreadFolder() { if ( !mFolderTree ) return; mFolderTree->prevUnreadFolder(); } void KMMainWidget::slotExpandThread() { mHeaders->slotExpandOrCollapseThread( true ); // expand } void KMMainWidget::slotCollapseThread() { mHeaders->slotExpandOrCollapseThread( false ); // collapse } void KMMainWidget::slotExpandAllThreads() { mHeaders->slotExpandOrCollapseAllThreads( true ); // expand } void KMMainWidget::slotCollapseAllThreads() { mHeaders->slotExpandOrCollapseAllThreads( false ); // collapse } //----------------------------------------------------------------------------- void KMMainWidget::slotShowMsgSrc() { if ( mMsgView ) mMsgView->setUpdateAttachment( false ); KMMessage *msg = mHeaders->currentMsg(); if ( !msg ) return; KMCommand *command = new KMShowMsgSrcCommand( this, msg, mMsgView ? mMsgView->isFixedFont() : false ); command->start(); } //----------------------------------------------------------------------------- void KMMainWidget::moveSelectedToFolder( QAction* act ) { if (mMenuToFolder[act]) mHeaders->moveMsgToFolder( mMenuToFolder[act] ); } //----------------------------------------------------------------------------- void KMMainWidget::copySelectedToFolder( QAction* act ) { if (mMenuToFolder[act]) mHeaders->copyMsgToFolder( mMenuToFolder[act] ); } //----------------------------------------------------------------------------- void KMMainWidget::updateMessageMenu() { mMenuToFolder.clear(); folderTree()->folderToPopupMenu( KMFolderTree::MoveMessage, this, &mMenuToFolder, mMoveActionMenu->menu() ); folderTree()->folderToPopupMenu( KMFolderTree::CopyMessage, this, &mMenuToFolder, mCopyActionMenu->menu() ); updateMessageActions(); } void KMMainWidget::startUpdateMessageActionsTimer() { menutimer->stop(); menutimer->start( 20 ); } void KMMainWidget::updateMessageActions() { int count = 0; QList selectedItems; if ( mFolder ) { for (Q3ListViewItem *item = mHeaders->firstChild(); item; item = item->itemBelow()) if (item->isSelected() ) selectedItems.append(item); if ( selectedItems.isEmpty() && mFolder->count() ) // there will always be one in mMsgView count = 1; else count = selectedItems.count(); mMsgActions->setCurrentMessage( mHeaders->currentMsg() ); mMsgActions->setSelectedSernums( mHeaders->selectedSernums() ); mMsgActions->setSelectedVisibleSernums( mHeaders->selectedVisibleSernums() ); } else { mMsgActions->setCurrentMessage( 0 ); } updateListFilterAction(); bool allSelectedInCommonThread = false; if ( mHeaders->isThreaded() && count > 1 ) { allSelectedInCommonThread = true; QList::const_iterator it; for ( it = selectedItems.begin(); it != selectedItems.end(); ++ it ) { Q3ListViewItem * item = *it; if ( item->parent() == 0 && item->childCount() == 0 ) { allSelectedInCommonThread = false; break; } } } else if ( mHeaders->isThreaded() && count == 1 ) { allSelectedInCommonThread = true; } bool mass_actions = count >= 1; bool thread_actions = mass_actions && allSelectedInCommonThread && mHeaders->isThreaded(); bool flags_available = GlobalSettings::self()->allowLocalFlags() || !(mFolder ? mFolder->isReadOnly() : true); mThreadStatusMenu->setEnabled( thread_actions ); // these need to be handled individually, the user might have them // in the toolbar mWatchThreadAction->setEnabled( thread_actions && flags_available ); mIgnoreThreadAction->setEnabled( thread_actions && flags_available ); mMarkThreadAsNewAction->setEnabled( thread_actions ); mMarkThreadAsReadAction->setEnabled( thread_actions ); mMarkThreadAsUnreadAction->setEnabled( thread_actions ); mToggleThreadTodoAction->setEnabled( thread_actions && flags_available ); mToggleThreadImportantAction->setEnabled( thread_actions && flags_available ); mTrashThreadAction->setEnabled( thread_actions && mFolder->canDeleteMessages() ); mDeleteThreadAction->setEnabled( thread_actions && mFolder->canDeleteMessages() ); if (mFolder && mHeaders && mHeaders->currentMsg()) { MessageStatus status = mHeaders->currentMsg()->status(); updateMessageTagActions( count ); if (thread_actions) { mToggleThreadTodoAction->setChecked( status.isTodo() ); mToggleThreadImportantAction->setChecked( status.isImportant() ); mWatchThreadAction->setChecked( status.isWatched() ); mIgnoreThreadAction->setChecked( status.isIgnored() ); } } mMoveActionMenu->setEnabled( mass_actions && mFolder->canDeleteMessages() ); mMoveMsgToFolderAction->setEnabled( mass_actions && mFolder->canDeleteMessages() ); mCopyActionMenu->setEnabled( mass_actions ); mTrashAction->setEnabled( mass_actions && mFolder->canDeleteMessages() ); mDeleteAction->setEnabled( mass_actions && mFolder->canDeleteMessages() ); mFindInMessageAction->setEnabled( mass_actions && !kmkernel->folderIsTemplates( mFolder ) ); mMsgActions->forwardInlineAction()->setEnabled( mass_actions && !kmkernel->folderIsTemplates( mFolder ) ); mMsgActions->forwardAttachedAction()->setEnabled( mass_actions && !kmkernel->folderIsTemplates( mFolder ) ); mMsgActions->forwardMenu()->setEnabled( mass_actions && !kmkernel->folderIsTemplates( mFolder ) ); mMsgActions->forwardInlineAction()->setEnabled( mass_actions ); mMsgActions->forwardAttachedAction()->setEnabled( mass_actions ); mMsgActions->forwardMenu()->setEnabled( mass_actions ); bool single_actions = count == 1; mMsgActions->editAction()->setEnabled( single_actions ); mUseAction->setEnabled( single_actions && kmkernel->folderIsTemplates( mFolder ) ); filterMenu()->setEnabled( single_actions ); mMsgActions->redirectAction()->setEnabled( single_actions && !kmkernel->folderIsTemplates( mFolder ) ); if ( mCustomTemplateMenus ) { mCustomTemplateMenus->forwardActionMenu()->setEnabled( mass_actions ); mCustomTemplateMenus->replyActionMenu()->setEnabled( single_actions ); mCustomTemplateMenus->replyAllActionMenu()->setEnabled( single_actions ); } printAction()->setEnabled( single_actions ); viewSourceAction()->setEnabled( single_actions ); mSendAgainAction->setEnabled( single_actions && ( ( mHeaders->currentMsg() && mHeaders->currentMsg()->status().isSent() ) || ( mFolder && mHeaders->currentMsg() && kmkernel->folderIsSentMailFolder( mFolder ) ) ) ); mSaveAsAction->setEnabled( mass_actions ); bool mails = mFolder && mFolder->count(); bool enable_goto_unread = mails || (GlobalSettings::self()->loopOnGotoUnread() == GlobalSettings::EnumLoopOnGotoUnread::LoopInAllFolders); actionCollection()->action( "go_next_message" )->setEnabled( mails ); actionCollection()->action( "go_next_unread_message" )->setEnabled( enable_goto_unread ); actionCollection()->action( "go_prev_message" )->setEnabled( mails ); actionCollection()->action( "go_prev_unread_message" )->setEnabled( enable_goto_unread ); actionCollection()->action( "send_queued" )->setEnabled( kmkernel->outboxFolder()->count() > 0 ); actionCollection()->action( "send_queued_via" )->setEnabled( kmkernel->outboxFolder()->count() > 0 ); slotUpdateOnlineStatus( static_cast( GlobalSettings::self()->networkState() ) ); if (action( "kmail_undo" )) action( "kmail_undo" )->setEnabled( mHeaders->canUndo() ); if ( count == 1 ) { KMMessage *msg; int aIdx; if((aIdx = mHeaders->currentItemIndex()) <= -1) return; if(!(msg = mFolder->getMsg(aIdx))) return; if ((KMFolder*)mFolder == kmkernel->outboxFolder()) editAction()->setEnabled( !msg->transferInProgress() ); } // Enable / disable all filters. foreach ( QAction *filterAction, mFilterMenuActions ) { filterAction->setEnabled( count > 0 ); } mApplyAllFiltersAction->setEnabled(count); mApplyFilterActionsMenu->setEnabled(count); } // This needs to be updated more often, so it is in its method. void KMMainWidget::updateMarkAsReadAction() { mMarkAllAsReadAction->setEnabled( mFolder && (mFolder->countUnread() > 0) ); } //----------------------------------------------------------------------------- void KMMainWidget::updateFolderMenu() { bool folderWithContent = mFolder && !mFolder->noContent(); bool multiFolder = folderTree()->selectedFolders().count() > 1; mModifyFolderAction->setEnabled( folderWithContent && !multiFolder ); mFolderMailingListPropertiesAction->setEnabled( folderWithContent && !multiFolder ); mCompactFolderAction->setEnabled( folderWithContent && !multiFolder ); // This is the refresh-folder action in the menu. See kmfoldertree for the one in the RMB... bool imap = mFolder && mFolder->folderType() == KMFolderTypeImap; bool cachedImap = mFolder && mFolder->folderType() == KMFolderTypeCachedImap; // For dimap, check that the imap path is known before allowing "check mail in this folder". bool knownImapPath = cachedImap && !static_cast( mFolder->storage() )->imapPath().isEmpty(); mRefreshFolderAction->setEnabled( folderWithContent && ( imap || ( cachedImap && knownImapPath ) ) && !multiFolder ); if ( mTroubleshootFolderAction ) mTroubleshootFolderAction->setEnabled( folderWithContent && ( cachedImap && knownImapPath ) && !multiFolder ); mTroubleshootMaildirAction->setVisible( mFolder && mFolder->folderType() == KMFolderTypeMaildir ); mEmptyFolderAction->setEnabled( folderWithContent && ( mFolder->count() > 0 ) && mFolder->canDeleteMessages() && !multiFolder ); mEmptyFolderAction->setText( (mFolder && kmkernel->folderIsTrash(mFolder)) ? i18n("E&mpty Trash") : i18n("&Move All Messages to Trash") ); mRemoveFolderAction->setEnabled( mFolder && !mFolder->isSystemFolder() && mFolder->canDeleteMessages() && !multiFolder); mRemoveFolderAction->setText( mFolder && mFolder->folderType() == KMFolderTypeSearch ? i18n("&Delete Search") : i18n("&Delete Folder") ); if ( mArchiveFolderAction ) mArchiveFolderAction->setEnabled( mFolder && !multiFolder ); mExpireFolderAction->setEnabled( mFolder && mFolder->isAutoExpire() && !multiFolder && mFolder->canDeleteMessages() ); updateMarkAsReadAction(); // the visual ones only make sense if we are showing a message list mPreferHtmlAction->setEnabled( mHeaders->folder() ? true : false ); mPreferHtmlLoadExtAction->setEnabled( mHeaders->folder() && (mHtmlPref ? !mFolderHtmlPref : mFolderHtmlPref) ? true : false ); mThreadMessagesAction->setEnabled( mHeaders->folder() ? true : false ); mPreferHtmlAction->setChecked( mHtmlPref ? !mFolderHtmlPref : mFolderHtmlPref ); mPreferHtmlLoadExtAction->setChecked( mHtmlLoadExtPref ? !mFolderHtmlLoadExtPref : mFolderHtmlLoadExtPref ); mThreadMessagesAction->setChecked( mThreadPref ? !mFolderThreadPref : mFolderThreadPref ); mThreadBySubjectAction->setEnabled( mHeaders->folder() ? ( mThreadMessagesAction->isChecked()) : false ); mThreadBySubjectAction->setChecked( mFolderThreadSubjPref ); mNewFolderAction->setEnabled( !multiFolder && ( mFolder && mFolder->folderType() != KMFolderTypeSearch )); mRemoveDuplicatesAction->setEnabled( !multiFolder && mFolder && mFolder->canDeleteMessages() ); mFolderShortCutCommandAction->setEnabled( !multiFolder ); } //----------------------------------------------------------------------------- void KMMainWidget::slotIntro() { if ( !mMsgView ) { return; } mMsgView->clear( true ); // hide widgets that are in the way: if ( mSearchAndHeaders && mHeaders && mLongFolderList ) { mSearchAndHeaders->hide(); } mMsgView->displayAboutPage(); closeFolder(); mFolder = 0; } void KMMainWidget::slotShowStartupFolder() { if ( mFolderTree ) { mFolderTree->reload(); mFolderTree->readConfig(); // get rid of old-folders mFolderTree->cleanupConfigFile(); } connect( kmkernel->filterMgr(), SIGNAL( filterListUpdated() ), this, SLOT( initializeFilterActions() ) ); connect( kmkernel->msgTagMgr(), SIGNAL( msgTagListChanged() ), this, SLOT( initializeMessageTagActions() ) ); // Plug various action lists. This can't be done in the constructor, as that is called before // the main window or Kontact calls createGUI(). // This function however is called with a single shot timer. initializeFilterActions(); initializeFolderShortcutActions(); initializeMessageTagActions(); messageActions()->setupForwardingActionsList( mGUIClient ); QString newFeaturesMD5 = KMReaderWin::newFeaturesMD5(); if ( kmkernel->firstStart() || GlobalSettings::self()->previousNewFeaturesMD5() != newFeaturesMD5 ) { GlobalSettings::self()->setPreviousNewFeaturesMD5( newFeaturesMD5 ); slotIntro(); return; } KMFolder* startup = 0; if ( !mStartupFolder.isEmpty() ) { // find the startup-folder startup = kmkernel->findFolderById( mStartupFolder ); } if ( !startup ) startup = kmkernel->inboxFolder(); if ( mFolderTree ) { mFolderTree->showFolder( startup ); } } void KMMainWidget::slotShowTip() { KTipDialog::showTip( this, QString(), true ); } void KMMainWidget::updateMessageTagActions( const int count ) { //TODO: Behaves differently according to number of messages selected KToggleAction *aToggler = 0; if ( 1 == count ) { KMMessageTagList *aTagList = mHeaders->currentMsg()->tagList(); for ( QList::ConstIterator it = mMessageTagMenuActions.constBegin(); it != mMessageTagMenuActions.constEnd(); ++it ) { bool list_present = false; if ( aTagList ) list_present = ( aTagList->indexOf( QString((*it).first->label() ) ) != -1 ); aToggler = static_cast( (*it).second ); aToggler->setChecked( list_present ); } } else if ( count > 1 ) { for ( QList::ConstIterator it = mMessageTagMenuActions.constBegin(); it != mMessageTagMenuActions.constEnd(); ++it ) { aToggler = static_cast( (*it).second ); aToggler->setChecked( false ); aToggler->setText( i18n("Toggle Message Tag %1", (*it).first->name() ) ); } } else { for ( QList::ConstIterator it = mMessageTagMenuActions.constBegin(); it != mMessageTagMenuActions.constEnd(); ++it ) { aToggler = static_cast( (*it).second ); aToggler->setEnabled( false ); } } } QList KMMainWidget::actionCollections() const { return QList() << actionCollection(); } void KMMainWidget::slotUpdateMessageTagList( const QString &name ) { mHeaders->setMessageTagList( name ); } void KMMainWidget::clearMessageTagActions() { //Remove the tag actions from the toolbar if ( !mMessageTagTBarActions.isEmpty() ) { if ( mGUIClient->factory() ) mGUIClient->unplugActionList( "toolbar_messagetag_actions" ); mMessageTagTBarActions.clear(); } //Remove the tag actions from the status menu and the action collection, //then delete them. for ( QList::ConstIterator it = mMessageTagMenuActions.constBegin(); it != mMessageTagMenuActions.constEnd(); ++it ) { mMsgActions->messageStatusMenu()->removeAction( (*it).second ); // This removes and deletes the action at the same time actionCollection()->removeAction( (*it).second ); } mMessageTagMenuActions.clear(); delete mMessageTagToggleMapper; mMessageTagToggleMapper = 0; } void KMMainWidget::initializeMessageTagActions() { clearMessageTagActions(); const QHash *tagDict = kmkernel->msgTagMgr()->msgTagDict(); if ( !tagDict ) return; //Use a mapper to understand which tag button is triggered mMessageTagToggleMapper = new QSignalMapper( this ); connect( mMessageTagToggleMapper, SIGNAL( mapped( const QString& ) ), this, SLOT( slotUpdateMessageTagList( const QString& ) ) ); //TODO: No need to do this anymore, just use the ordered list const int numTags = tagDict->count(); if ( !numTags ) return; for ( int i = 0; i < numTags; ++i ) { mMessageTagMenuActions.append( MessageTagPtrPair( 0, 0 ) ); } KAction *tagAction = 0; QHashIterator it( *tagDict ); while( it.hasNext() ) { it.next(); if ( ! it.value() || it.value()->isEmpty() ) continue; QString cleanName = i18n("Message Tag %1", it.value()->name() ); QString iconText = it.value()->name(); cleanName.replace("&","&&"); tagAction = new KToggleAction( KIcon(it.value()->toolbarIconName()), cleanName, this ); tagAction->setShortcut( it.value()->shortcut() ); tagAction->setIconText( iconText ); actionCollection()->addAction(it.value()->label().toLocal8Bit(), tagAction); connect(tagAction, SIGNAL(triggered(bool)), mMessageTagToggleMapper, SLOT(map())); // The shortcut configuration is done in the config dialog. // The shortcut set in the shortcut dialog would not be saved back to // the tag descriptions correctly. tagAction->setShortcutConfigurable( false ); mMessageTagToggleMapper->setMapping( tagAction, it.value()->label() ); MessageTagPtrPair ptr_pair( it.value(), tagAction ); //Relies on the fact that filters are always numbered from 0 mMessageTagMenuActions[it.value()->priority()] = ptr_pair; } for ( int i=0; i < numTags; ++i ) { mMsgActions->messageStatusMenu()->menu()->addAction( mMessageTagMenuActions[i].second ); if ( ( mMessageTagMenuActions[i].first )->inToolbar() ) mMessageTagTBarActions.append( mMessageTagMenuActions[i].second ); } if ( !mMessageTagTBarActions.isEmpty() && mGUIClient->factory() ) { //Separator doesn't work //mMessageTagTBarActions.prepend( mMessageTagToolbarActionSeparator ); mGUIClient->plugActionList( "toolbar_messagetag_actions", mMessageTagTBarActions ); } } //----------------------------------------------------------------------------- void KMMainWidget::slotChangeCaption( Q3ListViewItem *i ) { if ( !i ) { return; } // set the caption to the current full path QStringList names; for ( Q3ListViewItem * item = i ; item ; item = item->parent() ) { names.prepend( item->text(0) ); } emit captionChangeRequest( names.join( "/" ) ); } //----------------------------------------------------------------------------- void KMMainWidget::removeDuplicates() { if ( !mFolder ) { return; } KMFolder *oFolder = mFolder; mHeaders->setFolder( 0 ); QMap< QString, QList > idMD5s; QList redundantIds; QList::Iterator kt; mFolder->open( "removedups" ); for ( int i = mFolder->count() - 1; i >= 0; --i ) { QString id = (*mFolder)[i]->msgIdMD5(); if ( !id.isEmpty() ) { QString subjMD5 = (*mFolder)[i]->strippedSubjectMD5(); int other = -1; if ( idMD5s.contains(id) ) { other = idMD5s[id].first(); } else { idMD5s[id].append( i ); } if ( other != -1 ) { QString otherSubjMD5 = (*mFolder)[other]->strippedSubjectMD5(); if ( otherSubjMD5 == subjMD5 ) { idMD5s[id].append( i ); } } } } QMap< QString, QList >::Iterator it; for ( it = idMD5s.begin(); it != idMD5s.end() ; ++it ) { QList::Iterator jt; bool finished = false; for ( jt = (*it).begin(); jt != (*it).end() && !finished; ++jt ) if (!((*mFolder)[*jt]->status().isUnread())) { (*it).erase( jt ); (*it).prepend( *jt ); finished = true; } for ( jt = (*it).begin(), ++jt; jt != (*it).end(); ++jt ) redundantIds.append( *jt ); } qSort( redundantIds ); kt = redundantIds.end(); int numDuplicates = 0; if (kt != redundantIds.begin()) do { mFolder->removeMsg( *(--kt) ); ++numDuplicates; } while (kt != redundantIds.begin()); mFolder->close( "removedups" ); mHeaders->setFolder(oFolder); QString msg; if ( numDuplicates ) msg = i18np("Removed %1 duplicate message.", "Removed %1 duplicate messages.", numDuplicates ); else msg = i18n("No duplicate messages found."); BroadcastStatus::instance()->setStatusMsg( msg ); } //----------------------------------------------------------------------------- void KMMainWidget::slotUpdateUndo() { if ( actionCollection()->action( "kmail_undo" ) ) { actionCollection()->action( "kmail_undo" )->setEnabled( mHeaders->canUndo() ); } } //----------------------------------------------------------------------------- void KMMainWidget::clearFilterActions() { if ( !mFilterTBarActions.isEmpty() ) if ( mGUIClient->factory() ) mGUIClient->unplugActionList( "toolbar_filter_actions" ); if ( !mFilterMenuActions.isEmpty() ) if ( mGUIClient->factory() ) mGUIClient->unplugActionList( "menu_filter_actions" ); foreach ( QAction *a, mFilterMenuActions ) actionCollection()->removeAction( a ); mApplyFilterActionsMenu->menu()->clear(); mFilterTBarActions.clear(); mFilterMenuActions.clear(); qDeleteAll( mFilterCommands ); mFilterCommands.clear(); } //----------------------------------------------------------------------------- void KMMainWidget::initializeFolderShortcutActions() { QList< QPointer< KMFolder > > folders = kmkernel->allFolders(); QList< QPointer< KMFolder > >::Iterator it = folders.begin(); while ( it != folders.end() ) { KMFolder *folder = (*it); ++it; slotShortcutChanged( folder ); // load the initial accel } } //----------------------------------------------------------------------------- void KMMainWidget::initializeFilterActions() { clearFilterActions(); mApplyFilterActionsMenu->menu()->addAction( mApplyAllFiltersAction ); bool addedSeparator = false; QList::const_iterator it = kmkernel->filterMgr()->filters().begin(); for ( ;it != kmkernel->filterMgr()->filters().end(); ++it ) { if ( !(*it)->isEmpty() && (*it)->configureShortcut() ) { QString filterName = QString( "Filter %1").arg( (*it)->name() ); QString normalizedName = filterName.replace(' ', '_'); if ( action( normalizedName.toUtf8() ) ) { continue; } KMMetaFilterActionCommand *filterCommand = new KMMetaFilterActionCommand( *it, mHeaders, this ); mFilterCommands.append( filterCommand ); QString displayText = i18n( "Filter %1", (*it)->name() ); QString icon = (*it)->icon(); if ( icon.isEmpty() ) { icon = "system-run"; } KAction *filterAction = new KAction( KIcon( icon ), displayText, actionCollection() ); filterAction->setIconText( (*it)->toolbarName() ); // The shortcut configuration is done in the filter dialog. // The shortcut set in the shortcut dialog would not be saved back to // the filter settings correctly. filterAction->setShortcutConfigurable( false ); actionCollection()->addAction( normalizedName.toLocal8Bit(), filterAction ); connect( filterAction, SIGNAL(triggered(bool) ), filterCommand, SLOT(start()) ); filterAction->setShortcuts( (*it)->shortcut() ); if ( !addedSeparator ) { QAction *a = mApplyFilterActionsMenu->menu()->addSeparator(); mFilterMenuActions.append( a ); addedSeparator = true; } mApplyFilterActionsMenu->menu()->addAction( filterAction ); mFilterMenuActions.append( filterAction ); if ( (*it)->configureToolbar() ) { mFilterTBarActions.append( filterAction ); } } } if ( !mFilterMenuActions.isEmpty() && mGUIClient->factory() ) mGUIClient->plugActionList( "menu_filter_actions", mFilterMenuActions ); if ( !mFilterTBarActions.isEmpty() && mGUIClient->factory() ) { mFilterTBarActions.prepend( mToolbarActionSeparator ); mGUIClient->plugActionList( "toolbar_filter_actions", mFilterTBarActions ); } // Our filters have changed, now enable/disable them updateMessageActions(); } void KMMainWidget::slotFolderRemoved( KMFolder *folder ) { delete mFolderShortcutCommands.take( folder->idString() ); } //----------------------------------------------------------------------------- void KMMainWidget::initializeIMAPActions( bool setState /* false the first time, true later on */ ) { bool hasImapAccount = false; for( KMAccount *a = kmkernel->acctMgr()->first(); a; a = kmkernel->acctMgr()->next() ) { if ( a->type() == KAccount::DImap ) { hasImapAccount = true; break; } } if ( hasImapAccount == ( mTroubleshootFolderAction != 0 ) ) return; // nothing to do KXMLGUIFactory* factory = mGUIClient->factory(); if ( factory ) factory->removeClient( mGUIClient ); if ( !mTroubleshootFolderAction ) { mTroubleshootFolderAction = new KAction(KIcon("tools-wizard"), i18n("&Troubleshoot IMAP Cache..."), this); actionCollection()->addAction("troubleshoot_folder", mTroubleshootFolderAction ); connect(mTroubleshootFolderAction, SIGNAL(triggered(bool)), SLOT(slotTroubleshootFolder())); if ( setState ) updateFolderMenu(); // set initial state of the action } else { delete mTroubleshootFolderAction ; mTroubleshootFolderAction = 0; } if ( factory ) factory->addClient( mGUIClient ); } QList KMMainWidget::actionList() { return actionCollection()->actions(); } void KMMainWidget::slotShortcutChanged( KMFolder *folder ) { // remove the old one, no autodelete in Qt4 slotFolderRemoved( folder ); if ( folder->shortcut().isEmpty() ) return; FolderShortcutCommand *c = new FolderShortcutCommand( this, folder ); mFolderShortcutCommands.insert( folder->idString(), c ); QString actionlabel = i18n( "Folder Shortcut %1", folder->prettyUrl() ); QString actionname = i18n( "Folder Shortcut %1", folder->idString() ); QString normalizedName = actionname.replace(" ", "_"); KAction *action = actionCollection()->addAction( normalizedName ); // The folder shortcut is set in the folder shortcut dialog. // The shortcut set in the shortcut dialog would not be saved back to // the folder settings correctly. action->setShortcutConfigurable( false ); mFolderTree->addAction( action ); action->setText( actionlabel ); connect( action, SIGNAL( triggered(bool) ), c, SLOT( start() ) ); action->setShortcuts( folder->shortcut() ); action->setIcon( folder->useCustomIcons() ? KIcon( folder->unreadIconPath() ) : KIcon( "folder" ) ); c->setAction( action ); // will be deleted along with the command } //----------------------------------------------------------------------------- QString KMMainWidget::findCurrentImapPath() { QString startPath; if ( !mFolder ) { return startPath; } if ( mFolder->folderType() == KMFolderTypeImap ) { startPath = static_cast( mFolder->storage() )->imapPath(); } else if ( mFolder->folderType() == KMFolderTypeCachedImap ) { startPath = static_cast( mFolder->storage() )->imapPath(); } return startPath; } //----------------------------------------------------------------------------- ImapAccountBase *KMMainWidget::findCurrentImapAccountBase() { ImapAccountBase *account = 0; if ( !mFolder ) { return account; } if ( mFolder->folderType() == KMFolderTypeImap ) { account = static_cast( mFolder->storage() )->account(); } else if ( mFolder->folderType() == KMFolderTypeCachedImap ) { account = static_cast( mFolder->storage() )->account(); } return account; } //----------------------------------------------------------------------------- void KMMainWidget::slotSubscriptionDialog() { if ( !kmkernel->askToGoOnline() ) { return; } ImapAccountBase *account = findCurrentImapAccountBase(); if ( !account ) { return; } const QString startPath = findCurrentImapPath(); // KSubscription sets "DestructiveClose" SubscriptionDialog * dialog = new SubscriptionDialog(this, i18n("Subscription"), account, startPath); if ( dialog->exec() ) { // start a new listing if (mFolder->folderType() == KMFolderTypeImap) static_cast(mFolder->storage())->account()->listDirectory(); } } //----------------------------------------------------------------------------- void KMMainWidget::slotLocalSubscriptionDialog() { ImapAccountBase *account = findCurrentImapAccountBase(); if ( !account ) { return; } const QString startPath = findCurrentImapPath(); // KSubscription sets "DestructiveClose" LocalSubscriptionDialog *dialog = new LocalSubscriptionDialog(this, i18n("Local Subscription"), account, startPath); if ( dialog->exec() ) { // start a new listing if (mFolder->folderType() == KMFolderTypeImap) static_cast(mFolder->storage())->account()->listDirectory(); } } //----------------------------------------------------------------------------- void KMMainWidget::slotFolderTreeColumnsChanged() { mTotalColumnToggle->setChecked( mFolderTree->isTotalActive() ); mUnreadColumnToggle->setChecked( mFolderTree->isUnreadActive() ); mUnreadTextToggle->setChecked( !mFolderTree->isUnreadActive() ); mSizeColumnToggle->setChecked( mFolderTree->isSizeActive() ); } void KMMainWidget::toggleSystemTray() { if ( !mSystemTray && GlobalSettings::self()->systemTrayEnabled() ) { mSystemTray = new KMSystemTray(); } else if ( mSystemTray && !GlobalSettings::self()->systemTrayEnabled() ) { // Get rid of system tray on user's request kDebug() << "deleting systray"; delete mSystemTray; mSystemTray = 0; } // Set mode of systemtray. If mode has changed, tray will handle this. if ( mSystemTray ) mSystemTray->setMode( GlobalSettings::self()->systemTrayPolicy() ); } //----------------------------------------------------------------------------- void KMMainWidget::slotAntiSpamWizard() { AntiSpamWizard wiz( AntiSpamWizard::AntiSpam, this, folderTree() ); wiz.exec(); } //----------------------------------------------------------------------------- void KMMainWidget::slotAntiVirusWizard() { AntiSpamWizard wiz( AntiSpamWizard::AntiVirus, this, folderTree() ); wiz.exec(); } //----------------------------------------------------------------------------- void KMMainWidget::slotFilterLogViewer() { FilterLogDialog * dlg = new FilterLogDialog( 0 ); dlg->show(); } //----------------------------------------------------------------------------- void KMMainWidget::updateFileMenu() { QStringList actList = kmkernel->acctMgr()->getAccounts(); actionCollection()->action("check_mail")->setEnabled( actList.size() > 0 ); actionCollection()->action("check_mail_in")->setEnabled( actList.size() > 0 ); actionCollection()->action("favorite_check_mail")->setEnabled( actList.size() > 0 ); } //----------------------------------------------------------------------------- const KMMainWidget::PtrList * KMMainWidget::mainWidgetList() { // better safe than sorry; check whether the global static has already been destroyed if ( theMainWidgetList.isDestroyed() ) { return 0; } return theMainWidgetList; } //----------------------------------------------------------------------------- KMSystemTray *KMMainWidget::systray() const { return mSystemTray; } //----------------------------------------------------------------------------- QString KMMainWidget::overrideEncoding() const { if ( mMsgView ) return mMsgView->overrideEncoding(); else return GlobalSettings::self()->overrideCharacterEncoding(); } void KMMainWidget::slotCreateTodo() { KMMessage *msg = mHeaders->currentMsg(); if ( !msg ) return; KMCommand *command = new CreateTodoCommand( this, msg ); command->start(); } void KMMainWidget::showEvent( QShowEvent *event ) { QWidget::showEvent( event ); mWasEverShown = true; } void KMMainWidget::slotRequestFullSearchFromQuickSearch() { slotSearch(); assert( mSearchWin ); KMSearchPattern pattern; pattern.append( KMSearchRule::createInstance( "", KMSearchRule::FuncContains, mQuickSearchLine->currentSearchTerm() ) ); MessageStatus status = mQuickSearchLine->currentStatus(); if ( !status.isOfUnknownStatus() ) { pattern.append( new KMSearchRuleStatus( status ) ); } mSearchWin->setSearchPattern( pattern ); } void KMMainWidget::updateVactionScriptStatus( bool active ) { mVacationIndicatorActive = active; if ( active ) { mVacationScriptIndicator->setText( i18n("Out of office reply active") ); mVacationScriptIndicator->setBackgroundColor( Qt::yellow ); mVacationScriptIndicator->setCursor( QCursor( Qt::PointingHandCursor ) ); mVacationScriptIndicator->show(); } else { mVacationScriptIndicator->hide(); } } QLabel * KMMainWidget::vacationScriptIndicator() const { return mVacationScriptIndicator; }