You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

4219 lines
151 KiB

/* -*- mode: C++; c-file-style: "gnu" -*-
This file is part of KMail, the KDE mail client.
Copyright (c) 2002 Don Sanders <sanders@kde.org>
Based on the work of Stefan Taferner <taferner@kde.org>
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 <config-kmail.h>
#include <assert.h>
#include <QByteArray>
#include <QLabel>
#include <QLayout>
#include <QList>
#include <QSignalMapper>
#include <QSplitter>
#include <QToolBar>
#include <QTextDocument>
#include <QVBoxLayout>
#include <QDBusConnection>
#include <QDBusMessage>
#include <QShortcut>
#include <QProcess>
#include <q3accel.h>
#include <q3stylesheet.h>
#include <kicon.h>
#include <kwindowsystem.h>
#include <krun.h>
#include <kmessagebox.h>
#include <kactionmenu.h>
#include <kmenu.h>
#include <kacceleratormanager.h>
#include <kglobal.h>
#include <kglobalsettings.h>
#include <kstandardshortcut.h>
#include <kshortcutsdialog.h>
#include <kcharsets.h>
#include <kdebug.h>
#include <kfiledialog.h>
#include <ktip.h>
#include <kstandarddirs.h>
#include <kstandardaction.h>
#include <kaddrbookexternal.h>
#include <ktoggleaction.h>
#include <knotification.h>
#include <knotifyconfigwidget.h>
#include <kstringhandler.h>
#include <kconfiggroup.h>
#include <ktoolinvocation.h>
#include <kxmlguifactory.h>
#include <kstatusbar.h>
#include <kaction.h>
#include <kvbox.h>
#include <kpimidentities/identity.h>
#include <kpimidentities/identitymanager.h>
#include <mailtransport/transportmanager.h>
#include <mailtransport/transport.h>
#include <kmime/kmime_mdn.h>
#include <kmime/kmime_header_parsing.h>
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 <headerlistquicksearch.h>
using KMail::HeaderListQuickSearch;
#include "kmheaders.h"
#include "mailinglistpropertiesdialog.h"
#include "templateparser.h"
using KMail::TemplateParser;
#include "statusbarlabel.h"
#if !defined(NDEBUG)
#include "sievedebugdialog.h"
using KMail::SieveDebugDialog;
#endif
#include "managesievescriptsdialog.h"
#include "customtemplatesmenu.h"
#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 ),
mShowBusySplashTimer( 0 ),
mShowingOfflineScreen( false ),
mMsgActions( 0 ),
mAccel( 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;
setUpdatesEnabled( false ); //Enabled again in readConfig()
// 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<QString, int> & ) ),
this, SLOT( slotMailChecked( bool, bool, const QMap<QString, int> & ) ) );
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();
// must be the last line of the constructor:
mStartupDone = true;
KMainWindow *mainWin = dynamic_cast<KMainWindow*>(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()) );
}
//-----------------------------------------------------------------------------
//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( activated() ),
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 mesage 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<int> splitter1Sizes;
QList<int> 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<int> 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( activated() ),
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;
setUpdatesEnabled( false );
if ( mStartupDone )
{
writeConfig();
readPreConfig();
mHeaders->refreshNestedState();
layoutChanged = ( oldLongFolderList != mLongFolderList ) ||
( oldReaderWindowActive != mReaderWindowActive ) ||
( oldReaderWindowBelow != mReaderWindowBelow ) ||
( oldFavoriteFolderView != mEnableFavoriteFolderView ) ||
( oldFolderQuickSearch != mEnableFolderQuickSearch ) ||
( oldQuickSearch != mEnableQuickSearch );
if( layoutChanged ) {
hide();
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->restoreLayout( 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();
}
if ( mStartupDone ) {
// Update systray
toggleSystemTray();
mFolderTree->showFolder( mFolder );
// sanders - New code
mHeaders->setFolder(mFolder);
if (mMsgView) {
int aIdx = mHeaders->currentItemIndex();
if (aIdx != -1)
mMsgView->setMsg( mFolder->getMsg(aIdx), true );
else
mMsgView->clear( true );
}
updateMessageActions();
show();
// sanders - Maybe this fixes a bug?
}
updateMessageMenu();
updateFileMenu();
setUpdatesEnabled( true );
}
//-----------------------------------------------------------------------------
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()
mAccel = new Q3Accel(this, "createWidgets()");
//
// 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() ) );
mAccel->connectItem(mAccel->insertItem(Qt::SHIFT+Qt::Key_Left),
mHeaders, SLOT(selectPrevMessage()));
mAccel->connectItem(mAccel->insertItem(Qt::SHIFT+Qt::Key_Right),
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*) ) );
mAccel->connectItem(mAccel->insertItem(Qt::Key_Up),
mMsgView, SLOT(slotScrollUp()));
mAccel->connectItem(mAccel->insertItem(Qt::Key_Down),
mMsgView, SLOT(slotScrollDown()));
mAccel->connectItem(mAccel->insertItem(Qt::Key_PageUp),
mMsgView, SLOT(slotScrollPrior()));
mAccel->connectItem(mAccel->insertItem(Qt::Key_PageDown),
mMsgView, SLOT(slotScrollNext()));
}
//
// 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 );
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( activated() ),
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));
}
{
QAction *action = new KAction(i18n("Move Message to Folder"), this);
actionCollection()->addAction("move_message_to_folder", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotMoveMsg()));
action->setShortcut(QKeySequence(Qt::Key_M));
}
{
QAction *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));
}
{
QAction *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));
}
{
QAction *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));
mAccel->connectItem(mAccel->insertItem(Qt::Key_Escape),
ProgressManager::instance(), SLOT(slotAbortAll()));
}
{
QAction *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));
mAccel->connectItem(mAccel->insertItem(Qt::CTRL+Qt::Key_Right),
mFolderTree, SLOT(incCurrentFolder()));
}
{
QAction *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));
mAccel->connectItem(mAccel->insertItem(Qt::CTRL+Qt::Key_Left),
mFolderTree, SLOT(decCurrentFolder()));
}
{
QAction *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));
mAccel->connectItem(mAccel->insertItem(Qt::CTRL+Qt::Key_Space),
mFolderTree, SLOT(selectCurrentFolder()));
}
{
QAction *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));
mAccel->connectItem( mAccel->insertItem( Qt::ALT+Qt::Key_Right ),
mHeaders, SLOT( incCurrentMessage() ) );
}
{
QAction *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));
mAccel->connectItem( mAccel->insertItem( Qt::ALT+Qt::Key_Left ),
mHeaders, SLOT( decCurrentMessage() ) );
}
{
QAction *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));
mAccel->connectItem( mAccel->insertItem( Qt::ALT+Qt::Key_Space ),
mHeaders, SLOT( selectCurrentMessage() ) );
}
mAccel->connectItem(mAccel->insertItem(Qt::Key_M),
this, SLOT(slotMoveMsg()) );
mAccel->connectItem(mAccel->insertItem(Qt::Key_C),
this, SLOT(slotCopyMsg()) );
mAccel->connectItem(mAccel->insertItem(Qt::Key_J),
this, SLOT(slotJumpToFolder()) );
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(5006) <<" - account with name '" << item->data().toString() <<"' not found";
}
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotMailChecked( bool newMail, bool sendOnCheck,
const QMap<QString, int> & 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(5006) << 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 += "<br>" + 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",
"<b>New mail arrived</b><br />%1" ,
summary );
}
else {
summary = i18n( "New mail arrived" );
}
if(kmkernel->xmlGuiInstance().isValid()) {
#ifdef __GNUC__
#warning "kde4 : port ?"
#endif
//KNotifyClient::Instance instance(kmkernel->xmlGuiInstance());
KNotification::event( "new-mail-arrived",
summary,QPixmap() ,topLevelWidget() );
}
else
KNotification::event( "new-mail-arrived",
summary,QPixmap() ,topLevelWidget() );
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; idx<mTemplateFolder->count(); ++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<KMFolderTreeItem*>( 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<KMFolderTreeItem*>( mFolderTree->currentItem() );
if ( item )
item->assignShortcut();
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotModifyFolder()
{
if (!mFolderTree) return;
KMFolderTreeItem *item = static_cast<KMFolderTreeItem*>( mFolderTree->currentItem() );
if ( item )
modifyFolder( item );
}
//-----------------------------------------------------------------------------
void KMMainWidget::modifyFolder( KMFolderTreeItem* folderItem )
{
KMFolder* folder = folderItem->folder();
KMFolderTree* folderTree = static_cast<KMFolderTree *>( folderItem->listView() );
KMFolderDialog props( folder, folder->parent(), folderTree,
i18n("Properties of Folder %1", folder->label() ) );
props.exec();
updateFolderMenu();
}
//-----------------------------------------------------------------------------
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("<qt>Are you sure you want to expire the folder <b>%1</b>?</qt>", 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("<qt>Are you sure you want to move all messages from "
"folder <b>%1</b> to the trash?</qt>", 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::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("<qt>Are you sure you want to delete the search <b>%1</b>?<br />"
"Any messages it shows will still be available in their original folder.</qt>",
Qt::escape( mFolder->label() ) );
} else {
title = i18n("Delete Folder");
if ( mFolder->count() == 0 ) {
if ( !mFolder->child() || mFolder->child()->isEmpty() ) {
str = i18n("<qt>Are you sure you want to delete the empty folder "
"<b>%1</b>?</qt>",
Qt::escape( mFolder->label() ) );
}
else {
str = i18n("<qt>Are you sure you want to delete the empty folder "
"<b>%1</b> and all its subfolders? Those subfolders might "
"not be empty and their contents will be discarded as well. "
"<p><b>Beware</b> that discarded messages are not saved "
"into your Trash folder and are permanently deleted.</qt>",
Qt::escape( mFolder->label() ) );
}
} else {
if ( !mFolder->child() || mFolder->child()->isEmpty() ) {
str = i18n("<qt>Are you sure you want to delete the folder "
"<b>%1</b>, discarding its contents? "
"<p><b>Beware</b> that discarded messages are not saved "
"into your Trash folder and are permanently deleted.</qt>",
Qt::escape( mFolder->label() ) );
}
else {
str = i18n("<qt>Are you sure you want to delete the folder <b>%1</b> "
"and all its subfolders, discarding their contents? "
"<p><b>Beware</b> that discarded messages are not saved "
"into your Trash folder and are permanently deleted.</qt>",
Qt::escape( mFolder->label() ) );
}
}
}
if ( KMessageBox::warningContinueCancel( this, str, title,
KGuiItem( i18n("&Delete"), "edit-delete" ),
KStandardGuiItem::cancel(), "",
KMessageBox::Notify | KMessageBox::Dangerous )
== KMessageBox::Continue )
{
if ( mFolder->hasAccounts() ) {
// this folder has an account, so we need to change that to the inbox
for ( AccountList::Iterator it (mFolder->acctList()->begin() ),
end( mFolder->acctList()->end() ); it != end; ++it ) {
(*it)->setFolder( kmkernel->inboxFolder() );
KMessageBox::information(this,
i18n("<qt>The folder you deleted was associated with the account "
"<b>%1</b> which delivered mail into it. The folder the account "
"delivers new mail into was reset to the main Inbox folder.</qt>", (*it)->name()));
}
}
if (mFolder->folderType() == KMFolderTypeImap)
kmkernel->imapFolderMgr()->remove(mFolder);
else if (mFolder->folderType() == KMFolderTypeCachedImap) {
// Deleted by user -> tell the account (see KMFolderCachedImap::listDirectory2)
KMFolderCachedImap* storage = static_cast<KMFolderCachedImap*>( mFolder->storage() );
KMAcctCachedImap* acct = storage->account();
if ( acct )
acct->addDeletedFolder( mFolder );
kmkernel->dimapFolderMgr()->remove(mFolder);
}
else if (mFolder->folderType() == KMFolderTypeSearch)
kmkernel->searchFolderMgr()->remove(mFolder);
else
kmkernel->folderMgr()->remove(mFolder);
}
}
//-----------------------------------------------------------------------------
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<KMFolderImap*>(mFolder->storage());
imap->getAndCheckFolder();
} else if ( mFolder->folderType() == KMFolderTypeCachedImap ) {
KMFolderCachedImap* f = static_cast<KMFolderCachedImap*>( mFolder->storage() );
f->account()->processNewMailSingleFolder( mFolder );
}
}
}
void KMMainWidget::slotTroubleshootFolder()
{
if (mFolder)
{
if ( mFolder->folderType() == KMFolderTypeCachedImap ) {
KMFolderCachedImap* f = static_cast<KMFolderCachedImap*>( mFolder->storage() );
f->slotTroubleshoot();
}
}
}
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::slotForwardMsg()
{
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(5006) <<"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(5006) <<"Reply to All with template:" << tmpl;
KMCommand *command = new KMCustomReplyAllToCommand( this,
mHeaders->currentMsg(),
text,
tmpl );
command->start();
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotCustomForwardMsg( const QString &tmpl )
{
kDebug(5006) <<"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();
}
//-----------------------------------------------------------------------------
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"), true );
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(5006) <<"\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(),
htmlOverride, htmlLoadExtOverride,
useFixedFont, overrideEncoding() );
command->start();
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotConfigChanged()
{
readConfig();
}
//-----------------------------------------------------------------------------
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();
}
}
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<KMFolderImap*>(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 );
}
void KMMainWidget::closeFolder()
{
if ( !mFolder || mFolder->folderType() != KMFolderTypeImap ) {
return;
}
assert( mOpenedImapFolder );
KMFolderImap *imap = static_cast<KMFolderImap*>(mFolder->storage());
imap->setSelected( false );
mFolder->close( "mainwidget" );
mOpenedImapFolder = false;
}
//-----------------------------------------------------------------------------
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<KMFolderImap*>(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<KMFolderImap*>( 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(5006) <<"KMMainWidget::slotReplaceMsgByUnencryptedVersion()";
KMMessage* oldMsg = mHeaders->currentMsg();
if( oldMsg ) {
kDebug(5006) <<"KMMainWidget - old message found";
if( oldMsg->hasUnencryptedMsg() ) {
kDebug(5006) <<"KMMainWidget - 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(5006) <<"KMMainWidget - 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(5006) <<"KMMainWidget - deleting encrypted message";
mFolder->take( idx );
}
kDebug(5006) <<"KMMainWidget - updating message actions";
updateMessageActions();
kDebug(5006) <<"KMMainWidget - done.";
} else
kDebug(5006) <<"KMMainWidget - NO EXTRA UNENCRYPTED MESSAGE FOUND";
} else
kDebug(5006) <<"KMMainWidget - 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 );
}
//-----------------------------------------------------------------------------
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();
menu->addAction( mMsgView->addAddrBookAction() );
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( 0 ) <<" 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 {
if ( !mFolder->isSent() )
menu->addAction( mMsgActions->replyMenu() );
menu->addAction( mForwardActionMenu );
}
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& )) );
}
mForwardActionMenu->addSeparator();
mForwardActionMenu->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::shortcut(KStandardShortcut::Save));
mOpenAction = KStandardAction::open( this, SLOT( slotOpenMsg() ),
actionCollection() );
{
QAction *action = new KAction(i18n("&Compact All Folders"), this);
actionCollection()->addAction("compact_all_folders", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotCompactAll()));
}
{
QAction *action = new KAction(i18n("&Expire All Folders"), this);
actionCollection()->addAction("expire_all_folders", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotExpireAll()));
}
{
QAction *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()));
}
{
QAction *action = new KAction(i18n("Empty All &Trash Folders"), this);
actionCollection()->addAction("empty_trash", action );
connect(action, SIGNAL(triggered(bool) ), KMKernel::self(), SLOT(slotEmptyTrash()));
}
{
QAction *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(activated()),
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(activated()), this, SLOT(slotCheckMail()));
mActMenu = actActionMenu->menu();
connect(mActMenu, SIGNAL(triggered(QAction*)),
SLOT(slotCheckOneAccount(QAction*)));
connect(mActMenu, SIGNAL(aboutToShow()), SLOT(getAccountMenu()));
{
QAction *action = new KAction(KIcon("mail-send"), i18n("&Send Queued Messages"), this);
actionCollection()->addAction("send_queued", action );
connect(action, SIGNAL(triggered(bool)), SLOT(slotSendQueued()));
}
{
QAction *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")) {
QAction *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);
}
{
QAction *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);
}
{
QAction *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
if (KStandardDirs::findExe("kwatchgnupg").isEmpty()) action->setEnabled(false);
}
{
QAction *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)
{
QAction *action = new KAction(i18n("&Debug Sieve..."), this);
actionCollection()->addAction("tools_debug_sieve", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotDebugSieve()));
}
#endif
// @TODO (marc/bo): Test
{
QAction *action = new KAction(KIcon("configure"), i18n("Edit \"Out of Office\" Replies..."), this);
actionCollection()->addAction("tools_edit_vacation", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotEditVacation()));
}
{
QAction *action = new KAction(i18n("Filter &Log Viewer..."), this);
actionCollection()->addAction("filter_log_viewer", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotFilterLogViewer()));
}
{
QAction *action = new KAction(i18n("&Anti-Spam Wizard..."), this);
actionCollection()->addAction("antiSpamWizard", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotAntiSpamWizard()));
}
{
QAction *action = new KAction(i18n("&Anti-Virus Wizard..."), this);
actionCollection()->addAction("antiVirusWizard", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotAntiVirusWizard()));
}
if ( GlobalSettings::allowOutOfOfficeSettings() )
{
QAction *action = new KAction( i18n("Edit \"Out of Office\" Replies..."), this );
actionCollection()->addAction( "tools_edit_vacation", action );
connect( action, SIGNAL(triggered(bool)), SLOT(slotEditVacation()) );
}
{
QAction *action = new KAction( i18n("Filter &Log Viewer..."), this );
actionCollection()->addAction( "filter_log_viewer", action );
connect( action, SIGNAL(triggered(bool)), SLOT(slotFilterLogViewer()) );
}
{
QAction *action = new KAction( i18n("&Anti-Spam Wizard..."), this );
actionCollection()->addAction( "antiSpamWizard", action );
connect( action, SIGNAL(triggered(bool)), SLOT(slotAntiSpamWizard()) );
}
{
QAction *action = new KAction( i18n("&Anti-Virus Wizard..."), this );
actionCollection()->addAction( "antiVirusWizard", action );
connect( action, SIGNAL(triggered(bool)), SLOT(slotAntiVirusWizard()) );
}
//----- 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));
{
QAction *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()));
mFindInMessageAction->setShortcut(KStandardShortcut::shortcut(KStandardShortcut::Find));
{
QAction *action = new KAction(i18n("Select &All Messages"), this);
actionCollection()->addAction("mark_all_messages", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotMarkAll()));
action->setShortcuts(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::shortcut( KStandardShortcut::Reload ));
mTroubleshootFolderAction = 0; // set in initializeIMAPActions
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()));
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()));
{
QAction *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()));
}
{
QAction *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()));
}
{
QAction *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()));
}
{
QAction *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()));
}
{
QAction *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()));
}
{
QAction *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
{
QAction *action = new KAction(KIcon("mail-message-new"), i18n("&New Message..."), this);
actionCollection()->addAction("new_message", action );
action->setIconText( i18n( "New" ) );
connect(action, SIGNAL(triggered(bool)), SLOT(slotCompose()));
action->setShortcuts(KStandardShortcut::shortcut(KStandardShortcut::New));
}
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*) ) );
{
QAction *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));
}
mForwardActionMenu = new KActionMenu(KIcon("mail-forward"), i18nc("Message->","&Forward"), this);
actionCollection()->addAction("message_forward", mForwardActionMenu );
connect( mForwardActionMenu, SIGNAL(activated()), this,
SLOT(slotForwardMsg()) );
mForwardAttachedAction = new KAction(KIcon("mail-forward"), i18nc("Message->Forward->","As &Attachment..."), this);
actionCollection()->addAction("message_forward_as_attachment", mForwardAttachedAction );
mForwardAttachedAction->setShortcut(QKeySequence(Qt::Key_F));
connect(mForwardAttachedAction, SIGNAL(triggered(bool) ), SLOT(slotForwardAttachedMsg()));
mForwardActionMenu->addAction( forwardAttachedAction() );
mForwardAction = new KAction(KIcon("mail-forward"), i18n("&Inline..."), this);
actionCollection()->addAction("message_forward_inline", mForwardAction );
connect(mForwardAction, SIGNAL(triggered(bool) ), SLOT(slotForwardMsg()));
mForwardAction->setShortcut(QKeySequence(Qt::SHIFT+Qt::Key_F));
mForwardActionMenu->addAction( forwardAction() );
mSendAgainAction = new KAction(i18n("Send A&gain..."), this);
actionCollection()->addAction("send_again", mSendAgainAction );
connect(mSendAgainAction, SIGNAL(triggered(bool) ), SLOT(slotResendMsg()));
mRedirectAction = new KAction(KIcon("mail-forward"), i18nc("Message->Forward->","&Redirect..."), this);
actionCollection()->addAction("message_forward_redirect", mRedirectAction );
mRedirectAction->setShortcut(QKeySequence(Qt::Key_E));
connect(mRedirectAction, SIGNAL(triggered(bool) ), SLOT(slotRedirectMsg()));
mForwardActionMenu->addAction( redirectAction() );
//----- Create filter actions
mFilterMenu = new KActionMenu(KIcon("view-filter"), i18n("&Create Filter"), this);
actionCollection()->addAction("create_filter", mFilterMenu );
connect( mFilterMenu, SIGNAL(activated()), 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("file-new"), i18n("New Message From &Template"), this );
actionCollection()->addAction("use_templace", 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("process-stop"), 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(i18n("&Move To"), this);
actionCollection()->addAction("move_to", mMoveActionMenu );
mCopyActionMenu = new KActionMenu(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.") );
{
QAction *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()));
}
{
QAction *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()));
}
{
QAction *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()));
}
{
QAction *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));
QAction *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
{
QAction *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()));
}
{
QAction *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()));
}
{
QAction *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()));
}
{
QAction *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()));
}
{
QAction *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 );
}
{
QAction *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 );
}
{
QAction *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
{
QAction *action = new KAction(i18n("Configure &Filters..."), this);
actionCollection()->addAction("filter", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotFilter()));
}
{
QAction *action = new KAction(i18n("Configure &POP Filters..."), this);
actionCollection()->addAction("popFilter", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotPopFilter()));
}
{
QAction *action = new KAction(i18n("Manage &Sieve Scripts..."), this);
actionCollection()->addAction("sieveFilters", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotManageSieveScripts()));
}
{
QAction *action = new KAction(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());
{
QAction *action = new KAction( KIcon("preferences-desktop-notification"),
i18n("Configure &Notifications..."), this );
actionCollection()->addAction( "kmail_configure_notifications", action );
connect(action, SIGNAL(triggered(bool) ), SLOT(slotEditNotifications()));
}
// KStandardAction::preferences(this, SLOT(slotSettings()), actionCollection());
{
QAction *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()
{
// if(kmkernel->xmlGuiInstance())
// KNotifyDialog::configure(this, 0, kmkernel->xmlGuiInstance()->aboutData());
// 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<Q3ListViewItem*> 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;
Q3ListViewItem * curItemParent = mHeaders->currentItem();
while ( curItemParent->parent() )
curItemParent = curItemParent->parent();
QList<Q3ListViewItem*>::const_iterator it;
for ( it = selectedItems.begin(); it != selectedItems.end(); ++ it ) {
Q3ListViewItem * item = *it;
while ( item->parent() )
item = item->parent();
if ( item != curItemParent ) {
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->isReadOnly() );
mDeleteThreadAction->setEnabled( thread_actions && !mFolder->isReadOnly() );
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->isReadOnly() );
mCopyActionMenu->setEnabled( mass_actions );
mTrashAction->setEnabled( mass_actions && !mFolder->isReadOnly() );
mDeleteAction->setEnabled( mass_actions && !mFolder->isReadOnly() );
mFindInMessageAction->setEnabled( mass_actions );
mForwardAction->setEnabled( mass_actions );
mForwardAttachedAction->setEnabled( mass_actions );
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 );
redirectAction()->setEnabled( single_actions );
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<GlobalSettingsBase::EnumNetworkState::type>( GlobalSettings::self()->networkState() ) );
if (action( "edit_undo" ))
action( "edit_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() );
}
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<KMFolderCachedImap*>( mFolder->storage() )->imapPath().isEmpty();
mRefreshFolderAction->setEnabled( folderWithContent && ( imap
|| ( cachedImap && knownImapPath ) ) && !multiFolder );
if ( mTroubleshootFolderAction )
mTroubleshootFolderAction->setEnabled( folderWithContent && ( cachedImap && knownImapPath ) && !multiFolder );
mEmptyFolderAction->setEnabled( folderWithContent && ( mFolder->count() > 0 ) && !mFolder->isReadOnly() && !multiFolder );
mEmptyFolderAction->setText( (mFolder && kmkernel->folderIsTrash(mFolder))
? i18n("E&mpty Trash") : i18n("&Move All Messages to Trash") );
mRemoveFolderAction->setEnabled( mFolder && !mFolder->isSystemFolder() && !mFolder->isReadOnly() && !multiFolder);
mRemoveFolderAction->setText( mFolder && mFolder->folderType() == KMFolderTypeSearch ? i18n("&Delete Search") : i18n("&Delete Folder") );
mExpireFolderAction->setEnabled( mFolder && mFolder->isAutoExpire() && !multiFolder );
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 );
mRemoveDuplicatesAction->setEnabled( !multiFolder );
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 shortcut filter actions now
initializeFilterActions();
// plug folder shortcut actions
initializeFolderShortcutActions();
// plug tag actions now
initializeMessageTagActions();
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<MessageTagPtrPair>::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<KToggleAction*>( (*it).second );
aToggler->setChecked( list_present );
}
} else if ( count > 1 ) {
for ( QList<MessageTagPtrPair>::ConstIterator it =
mMessageTagMenuActions.constBegin();
it != mMessageTagMenuActions.constEnd(); ++it ) {
aToggler = static_cast<KToggleAction*>( (*it).second );
aToggler->setChecked( false );
aToggler->setText( i18n("Toggle Message Tag %1", (*it).first->name() ) );
}
}
else {
for ( QList<MessageTagPtrPair>::ConstIterator it =
mMessageTagMenuActions.constBegin();
it != mMessageTagMenuActions.constEnd(); ++it ) {
aToggler = static_cast<KToggleAction*>( (*it).second );
aToggler->setEnabled( false );
}
}
}
QList<KActionCollection*> KMMainWidget::actionCollections() const {
return QList<KActionCollection*>() << 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<MessageTagPtrPair>::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<QString,KMMessageTagDescription*> *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<QString,KMMessageTagDescription*> 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.
//Setting the below to true decouples action objects shortcut
//from that of the tag description
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<int> > idMD5s;
QList<int> redundantIds;
QList<int>::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<int> >::Iterator it;
for ( it = idMD5s.begin(); it != idMD5s.end() ; ++it ) {
QList<int>::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( "edit_undo" ) ) {
actionCollection()->action( "edit_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()
{
QString filterName, normalizedName;
KMMetaFilterActionCommand *filterCommand;
QAction *filterAction = 0;
clearFilterActions();
mApplyFilterActionsMenu->menu()->addAction( mApplyAllFiltersAction );
bool addedSeparator = false;
QList<KMFilter*>::const_iterator it = kmkernel->filterMgr()->filters().begin();
for ( ;it != kmkernel->filterMgr()->filters().end(); ++it ) {
if ( !(*it)->isEmpty() && (*it)->configureShortcut() ) {
filterName = QString( "Filter %1").arg( (*it)->name() );
normalizedName = filterName.replace(" ", "_");
if ( action( normalizedName.toUtf8() ) ) {
continue;
}
filterCommand = new KMMetaFilterActionCommand( *it, mHeaders, this );
mFilterCommands.append( filterCommand );
QString as = i18n( "Filter %1", (*it)->name() );
QString icon = (*it)->icon();
if ( icon.isEmpty() ) {
icon = "system-run";
}
filterAction = new KAction( KIcon( icon ), as, actionCollection() );
filterAction->setIconText( (*it)->toolbarName() );
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 );
}
}
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<QAction*> 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(" ", "_");
QAction * action = actionCollection()->addAction( normalizedName );
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<KMFolderImap*>( mFolder->storage() )->imapPath();
} else if ( mFolder->folderType() == KMFolderTypeCachedImap ) {
startPath = static_cast<KMFolderCachedImap*>( mFolder->storage() )->imapPath();
}
return startPath;
}
//-----------------------------------------------------------------------------
ImapAccountBase *KMMainWidget::findCurrentImapAccountBase()
{
ImapAccountBase *account = 0;
if ( !mFolder ) {
return account;
}
if ( mFolder->folderType() == KMFolderTypeImap ) {
account = static_cast<KMFolderImap*>( mFolder->storage() )->account();
} else if ( mFolder->folderType() == KMFolderTypeCachedImap ) {
account = static_cast<KMFolderCachedImap*>( 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<KMFolderImap*>(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<KMFolderImap*>(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(5006) <<"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 );
}
//-----------------------------------------------------------------------------
void KMMainWidget::setAccelsEnabled( bool enabled )
{
if ( mAccel )
mAccel->setEnabled( enabled );
}
//-----------------------------------------------------------------------------
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( "<message>", 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;
}