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.
tdepim/kmail/kmmainwidget.cpp

3980 lines
141 KiB

// -*- mode: C++; c-file-style: "gnu" -*-
// kmmainwidget.cpp
//#define MALLOC_DEBUG 1
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <kwin.h>
#ifdef MALLOC_DEBUG
#include <malloc.h>
#endif
#undef Unsorted // X headers...
#include <qaccel.h>
#include <qlayout.h>
#include <qhbox.h>
#include <qvbox.h>
#include <qpopupmenu.h>
#include <qpushbutton.h>
#include <qptrlist.h>
#include <kopenwith.h>
#include <kmessagebox.h>
#include <klistviewsearchline.h>
#include <kiconloader.h>
#include <kpopupmenu.h>
#include <kaccelmanager.h>
#include <kglobalsettings.h>
#include <kstdaccel.h>
#include <kkeydialog.h>
#include <kcharsets.h>
#include <knotifyclient.h>
#include <kdebug.h>
#include <kapplication.h>
#include <kfiledialog.h>
#include <ktip.h>
#include <knotifydialog.h>
#include <kstandarddirs.h>
#include <dcopclient.h>
#include <kaddrbook.h>
#include <kaccel.h>
#include <kstringhandler.h>
#include <qvaluevector.h>
#include "globalsettings.h"
#include "kcursorsaver.h"
#include "broadcaststatus.h"
using KPIM::BroadcastStatus;
#include "kmfoldermgr.h"
#include "kmfolderdia.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 "kmfolderseldlg.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 "imapaccountbase.h"
#include "transportmanager.h"
using KMail::ImapAccountBase;
#include "vacation.h"
using KMail::Vacation;
#include "favoritefolderview.h"
#include <qsignalmapper.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>
#include "klistviewindexedsearchline.h"
using KMail::HeaderListQuickSearch;
#include "kmheaders.h"
#include "mailinglistpropertiesdialog.h"
#include "templateparser.h"
#if !defined(NDEBUG)
#include "sievedebugdialog.h"
using KMail::SieveDebugDialog;
#endif
#include <libkpimidentities/identity.h>
#include <libkpimidentities/identitymanager.h>
#include <assert.h>
#include <kstatusbar.h>
#include <kstaticdeleter.h>
#include <kaction.h>
#include <kmime_mdn.h>
#include <kmime_header_parsing.h>
using namespace KMime;
using KMime::Types::AddrSpecList;
#include "progressmanager.h"
using KPIM::ProgressManager;
#include "managesievescriptsdialog.h"
#include <qstylesheet.h>
#include "customtemplates.h"
#include "customtemplates_kfg.h"
#include "kmmainwidget.moc"
QValueList<KMMainWidget*>* KMMainWidget::s_mainWidgetList = 0;
static KStaticDeleter<QValueList<KMMainWidget*> > mwlsd;
//-----------------------------------------------------------------------------
KMMainWidget::KMMainWidget(QWidget *parent, const char *name,
KXMLGUIClient *aGUIClient,
KActionCollection *actionCollection, KConfig* config ) :
QWidget(parent, name),
mFavoritesCheckMailAction( 0 ),
mFavoriteFolderView( 0 ),
mFolderView( 0 ),
mFolderViewParent( 0 ),
mFolderViewSplitter( 0 ),
mQuickSearchLine( 0 ),
mShowBusySplashTimer( 0 ),
mShowingOfflineScreen( false ),
mMsgActions( 0 ),
mVacationIndicatorActive( false )
{
// must be the first line of the constructor:
mStartupDone = 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;
mActionCollection = actionCollection;
mTopLayout = new QVBoxLayout(this);
mFilterMenuActions.setAutoDelete(true);
mFilterTBarActions.setAutoDelete(false);
mFilterCommands.setAutoDelete(true);
mFolderShortcutCommands.setAutoDelete(true);
mJob = 0;
mConfig = config;
mGUIClient = aGUIClient;
mCustomReplyActionMenu = 0;
mCustomReplyAllActionMenu = 0;
mCustomForwardActionMenu = 0;
mCustomReplyMapper = 0;
mCustomReplyAllMapper = 0;
mCustomForwardMapper = 0;
// FIXME This should become a line separator as soon as the API
// is extended in kdelibs.
mToolbarActionSeparator = new KActionSeparator( actionCollection );
if( !s_mainWidgetList )
mwlsd.setObject( s_mainWidgetList, new QValueList<KMMainWidget*>() );
s_mainWidgetList->append( this );
mPanner1Sep << 1 << 1;
mPanner2Sep << 1 << 1;
setMinimumSize(400, 300);
readPreConfig();
createWidgets();
setupActions();
readConfig();
activatePanners();
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(QListViewItem*)),
this, SLOT(slotChangeCaption(QListViewItem*)));
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 KStatusBarLabel( QString(), 0, sb );
mVacationScriptIndicator->hide();
connect( mVacationScriptIndicator, SIGNAL(itemReleased(int)), 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()
{
s_mainWidgetList->remove( this );
destruct();
}
//-----------------------------------------------------------------------------
//This method performs all cleanup that requires the kernel to exist.
void KMMainWidget::destruct()
{
if (mDestructed)
return;
if (mSearchWin)
mSearchWin->close();
writeConfig();
writeFolderConfig();
delete mHeaders;
delete mFolderTree;
delete mSystemTray;
delete mMsgView;
mDestructed = true;
}
//-----------------------------------------------------------------------------
void KMMainWidget::readPreConfig(void)
{
const KConfigGroup geometry( KMKernel::config(), "Geometry" );
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.readBoolEntry( "nestedMessages", false );
mHtmlPref = reader.readBoolEntry( "htmlMail", false );
mHtmlLoadExtPref = reader.readBoolEntry( "htmlLoadExternal", false );
mEnableFavoriteFolderView = GlobalSettings::self()->enableFavoriteFolderView();
mEnableFolderQuickSearch = GlobalSettings::self()->enableFolderQuickSearch();
mEnableQuickSearch = GlobalSettings::self()->quickSearchActive();
}
//-----------------------------------------------------------------------------
void KMMainWidget::readFolderConfig(void)
{
if (!mFolder)
return;
KConfig *config = KMKernel::config();
KConfigGroupSaver saver(config, "Folder-" + mFolder->idString());
mFolderThreadPref = config->readBoolEntry( "threadMessagesOverride", false );
mFolderThreadSubjPref = config->readBoolEntry( "threadMessagesBySubject", true );
mFolderHtmlPref = config->readBoolEntry( "htmlMailOverride", false );
mFolderHtmlLoadExtPref = config->readBoolEntry( "htmlLoadExternalOverride", false );
}
//-----------------------------------------------------------------------------
void KMMainWidget::writeFolderConfig(void)
{
if (!mFolder)
return;
KConfig *config = KMKernel::config();
KConfigGroupSaver saver(config, "Folder-" + mFolder->idString());
config->writeEntry( "threadMessagesOverride", mFolderThreadPref );
config->writeEntry( "threadMessagesBySubject", mFolderThreadSubjPref );
config->writeEntry( "htmlMailOverride", mFolderHtmlPref );
config->writeEntry( "htmlLoadExternalOverride", mFolderHtmlLoadExtPref );
}
//-----------------------------------------------------------------------------
void KMMainWidget::readConfig(void)
{
KConfig *config = KMKernel::config();
bool oldLongFolderList = mLongFolderList;
bool oldReaderWindowActive = mReaderWindowActive;
bool oldReaderWindowBelow = mReaderWindowBelow;
bool oldFavoriteFolderView = mEnableFavoriteFolderView;
bool oldFolderQuickSearch = mEnableFolderQuickSearch;
bool oldQuickSearch = mEnableQuickSearch;
QString str;
QSize siz;
if (mStartupDone)
{
writeConfig();
readPreConfig();
mHeaders->refreshNestedState();
bool layoutChanged = ( oldLongFolderList != mLongFolderList )
|| ( oldReaderWindowActive != mReaderWindowActive )
|| ( oldReaderWindowBelow != mReaderWindowBelow )
|| ( oldFavoriteFolderView != mEnableFavoriteFolderView )
|| ( oldFolderQuickSearch != mEnableFolderQuickSearch )
|| ( oldQuickSearch != mEnableQuickSearch );
if( layoutChanged ) {
hide();
// delete all panners
delete mPanner1; // will always delete the others
createWidgets();
}
}
{ // area for config group "Geometry"
KConfigGroupSaver saver(config, "Geometry");
// size of the mainwin
QSize defaultSize(750,560);
siz = config->readSizeEntry("MainWin", &defaultSize);
if (!siz.isEmpty())
resize(siz);
// default width of the foldertree
static const int folderpanewidth = 250;
const int folderW = config->readNumEntry( "FolderPaneWidth", folderpanewidth );
const int headerW = config->readNumEntry( "HeaderPaneWidth", width()-folderpanewidth );
const int headerH = config->readNumEntry( "HeaderPaneHeight", 180 );
const int readerH = config->readNumEntry( "ReaderPaneHeight", 280 );
mPanner1Sep.clear();
mPanner2Sep.clear();
QValueList<int> & widths = mLongFolderList ? mPanner1Sep : mPanner2Sep ;
QValueList<int> & heights = mLongFolderList ? mPanner2Sep : mPanner1Sep ;
widths << folderW << headerW;
heights << headerH << readerH;
bool layoutChanged = ( oldLongFolderList != mLongFolderList )
|| ( oldReaderWindowActive != mReaderWindowActive )
|| ( oldReaderWindowBelow != mReaderWindowBelow );
if (!mStartupDone || 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 = config->readNumEntry("UnreadColumn", 1);
const int totalColumn = config->readNumEntry("TotalColumn", 2);
const int sizeColumn = config->readNumEntry("SizeColumn", 3);
/* 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 );
mUnreadColumnToggle->setChecked( mFolderTree->isUnreadActive() );
mUnreadTextToggle->setChecked( !mFolderTree->isUnreadActive() );
mTotalColumnToggle->setChecked( mFolderTree->isTotalActive() );
mSizeColumnToggle->setChecked( mFolderTree->isSizeActive() );
mFolderTree->updatePopup();
}
}
if (mMsgView)
mMsgView->readConfig();
mHeaders->readConfig();
mHeaders->restoreLayout(KMKernel::config(), "Header-Geometry");
if ( mFolderViewSplitter && !GlobalSettings::self()->folderViewSplitterPosition().isEmpty() ) {
mFolderViewSplitter->setSizes( GlobalSettings::self()->folderViewSplitterPosition() );
} else {
QValueList<int> defaults;
defaults << (int)(height() * 0.2) << (int)(height() * 0.8);
mFolderViewSplitter->setSizes( defaults );
}
mFolderTree->readConfig();
if ( mFavoriteFolderView )
mFavoriteFolderView->readConfig();
mFavoritesCheckMailAction->setEnabled( GlobalSettings::self()->enableFavoriteFolderView() );
{ // area for config group "General"
KConfigGroupSaver saver(config, "General");
mBeepOnNew = config->readBoolEntry("beep-on-mail", false);
mConfirmEmpty = config->readBoolEntry("confirm-before-empty", true);
// startup-Folder, defaults to system-inbox
mStartupFolder = config->readEntry("startupFolder", kmkernel->inboxFolder()->idString());
if (!mStartupDone)
{
// check mail on startup
bool check = config->readBoolEntry("checkmail-startup", false);
if (check)
// do it after building the kmmainwin, so that the progressdialog is available
QTimer::singleShot( 0, this, SLOT( slotCheckMail() ) );
}
}
// reload foldertree
mFolderTree->reload();
// Re-activate panners
if (mStartupDone)
{
// Update systray
toggleSystemTray();
bool layoutChanged = ( oldLongFolderList != mLongFolderList )
|| ( oldReaderWindowActive != mReaderWindowActive )
|| ( oldReaderWindowBelow != mReaderWindowBelow )
|| ( oldFavoriteFolderView != mEnableFavoriteFolderView )
|| ( oldFolderQuickSearch != mEnableFolderQuickSearch )
|| ( oldQuickSearch != mEnableQuickSearch );
if ( layoutChanged ) {
activatePanners();
}
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();
}
//-----------------------------------------------------------------------------
void KMMainWidget::writeConfig(void)
{
QString s;
KConfig *config = KMKernel::config();
KConfigGroup geometry( config, "Geometry" );
if (mMsgView)
mMsgView->writeConfig();
if ( mFolderViewSplitter )
GlobalSettings::setFolderViewSplitterPosition( mFolderViewSplitter->sizes() );
mFolderTree->writeConfig();
if ( mFavoriteFolderView )
mFavoriteFolderView->writeConfig();
geometry.writeEntry( "MainWin", this->geometry().size() );
const QValueList<int> widths = ( mLongFolderList ? mPanner1 : mPanner2 )->sizes();
const QValueList<int> heights = ( mLongFolderList ? mPanner2 : mPanner1 )->sizes();
geometry.writeEntry( "FolderPaneWidth", widths[0] );
geometry.writeEntry( "HeaderPaneWidth", widths[1] );
// Only save when the widget is shown (to avoid saving a wrong value)
if ( mSearchAndHeaders && mSearchAndHeaders->isShown() ) {
geometry.writeEntry( "HeaderPaneHeight", heights[0] );
geometry.writeEntry( "ReaderPaneHeight", heights[1] );
}
// 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::createWidgets(void)
{
// Create the splitters according to the layout settings
QWidget *headerParent = 0,
*mimeParent = 0, *messageParent = 0;
const bool opaqueResize = KGlobalSettings::opaqueResize();
if ( mLongFolderList ) {
// superior splitter: folder tree vs. rest
// inferior splitter: headers vs. message vs. mime tree
mPanner1 = new QSplitter( Qt::Horizontal, this, "panner 1" );
mPanner1->setOpaqueResize( opaqueResize );
Qt::Orientation orientation = mReaderWindowBelow ? Qt::Vertical : Qt::Horizontal;
mPanner2 = new QSplitter( orientation, mPanner1, "panner 2" );
mPanner2->setOpaqueResize( opaqueResize );
mPanner2->setChildrenCollapsible( false );
mFolderViewParent = mPanner1;
headerParent = mimeParent = messageParent = mPanner2;
} else /* !mLongFolderList */ {
// superior splitter: ( folder tree + headers ) vs. message vs. mime
// inferior splitter: folder tree vs. headers
mPanner1 = new QSplitter( Qt::Vertical, this, "panner 1" );
mPanner1->setOpaqueResize( opaqueResize );
mPanner1->setChildrenCollapsible( false );
mPanner2 = new QSplitter( Qt::Horizontal, mPanner1, "panner 2" );
mPanner2->setOpaqueResize( opaqueResize );
headerParent = mFolderViewParent = mPanner2;
mimeParent = messageParent = mPanner1;
}
#ifndef NDEBUG
if( mPanner1 ) mPanner1->dumpObjectTree();
if( mPanner2 ) mPanner2->dumpObjectTree();
#endif
mTopLayout->add( mPanner1 );
// BUG -sanders these accelerators stop working after switching
// between long/short folder layout
// Probably need to disconnect them first.
// create list of messages
#ifndef NDEBUG
headerParent->dumpObjectTree();
#endif
mSearchAndHeaders = new QVBox( headerParent );
mSearchToolBar = new KToolBar( mSearchAndHeaders, "search toolbar");
mSearchToolBar->setMovingEnabled(false);
mSearchToolBar->boxLayout()->setSpacing( KDialog::spacingHint() );
QLabel *label = new QLabel( i18n("S&earch:"), mSearchToolBar, "kde toolbar widget" );
mHeaders = new KMHeaders(this, mSearchAndHeaders, "headers");
#ifdef HAVE_INDEXLIB
mQuickSearchLine = new KListViewIndexedSearchLine( mSearchToolBar, mHeaders,
actionCollection(), "headers quick search line" );
#else
mQuickSearchLine = new HeaderListQuickSearch( mSearchToolBar, mHeaders,
actionCollection(), "headers quick search line" );
#endif
label->setBuddy( mQuickSearchLine );
connect( mQuickSearchLine, SIGNAL( requestFullSearch() ),
this, SLOT( slotRequestFullSearchFromQuickSearch() ) );
mSearchToolBar->setStretchableWidget( mQuickSearchLine );
connect( mHeaders, SIGNAL( messageListUpdated() ),
mQuickSearchLine, SLOT( updateSearch() ) );
if ( !GlobalSettings::self()->quickSearchActive() ) mSearchToolBar->hide();
if (mReaderWindowActive) {
connect(mHeaders, SIGNAL(selected(KMMessage*)),
this, SLOT(slotMsgSelected(KMMessage*)));
}
connect(mHeaders, SIGNAL(activated(KMMessage*)),
this, SLOT(slotMsgActivated(KMMessage*)));
connect( mHeaders, SIGNAL( selectionChanged() ),
SLOT( startUpdateMessageActionsTimer() ) );
QAccel *accel = actionCollection()->kaccel();
accel->connectItem(accel->insertItem(SHIFT+Key_Left),
mHeaders, SLOT(selectPrevMessage()));
accel->connectItem(accel->insertItem(SHIFT+Key_Right),
mHeaders, SLOT(selectNextMessage()));
if (mReaderWindowActive) {
mMsgView = new KMReaderWin(messageParent, 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()));
accel->connectItem(accel->insertItem(Key_Up),
mMsgView, SLOT(slotScrollUp()));
accel->connectItem(accel->insertItem(Key_Down),
mMsgView, SLOT(slotScrollDown()));
accel->connectItem(accel->insertItem(Key_Prior),
mMsgView, SLOT(slotScrollPrior()));
accel->connectItem(accel->insertItem(Key_Next),
mMsgView, SLOT(slotScrollNext()));
} else {
mMsgView = NULL;
}
KAction *action;
action = new KAction( i18n("Move Message to Folder"), Key_M, this,
SLOT(slotMoveMsg()), actionCollection(),
"move_message_to_folder" );
action->plugAccel( actionCollection()->kaccel() );
action = new KAction( i18n("Copy Message to Folder"), Key_C, this,
SLOT(slotCopyMsg()), actionCollection(),
"copy_message_to_folder" );
action->plugAccel( actionCollection()->kaccel() );
action = new KAction( i18n("Jump to Folder"), Key_J, this,
SLOT(slotJumpToFolder()), actionCollection(),
"jump_to_folder" );
action->plugAccel( actionCollection()->kaccel() );
// create list of folders
mFolderViewSplitter = new QSplitter( Qt::Vertical, mFolderViewParent );
mFolderViewSplitter->setOpaqueResize( KGlobalSettings::opaqueResize() );
mFavoriteFolderView = new KMail::FavoriteFolderView( this, mFolderViewSplitter );
if ( mFavoritesCheckMailAction )
connect( mFavoritesCheckMailAction, SIGNAL(activated()), mFavoriteFolderView, SLOT(checkMail()) );
QWidget *folderTreeParent = mFolderViewParent;
if ( GlobalSettings::enableFavoriteFolderView() ) {
folderTreeParent = mFolderViewSplitter;
mFolderView = mFolderViewSplitter;
}
// the "folder tree" consists of a quicksearch input field and the tree itself
mSearchAndTree = new QVBox(folderTreeParent);
mFolderQuickSearch = new QHBox(mSearchAndTree);
QPushButton *clear = new QPushButton(QApplication::reverseLayout()
? SmallIcon("clear_left")
: SmallIcon("locationbar_erase"), "", mFolderQuickSearch);
clear->setFlat(true);
KListViewSearchLine *search = new KListViewSearchLine(mFolderQuickSearch);
mFolderTree = new KMFolderTree(this, mSearchAndTree, "folderTree");
search->setListView(mFolderTree);
connect(clear, SIGNAL(clicked()), search, SLOT(clear()));
if ( !GlobalSettings::enableFolderQuickSearch() ) {
mFolderQuickSearch->hide();
}
if ( !GlobalSettings::enableFavoriteFolderView() ) {
mFolderView = mSearchAndTree;
}
connect( mFolderTree, SIGNAL(folderSelected(KMFolder*)),
mFavoriteFolderView, SLOT(folderTreeSelectionChanged(KMFolder*)) );
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()));
if ( mFavoriteFolderView ) {
connect( mFavoriteFolderView, SIGNAL(folderDrop(KMFolder*)), SLOT(slotMoveMsgToFolder(KMFolder*)) );
connect( mFavoriteFolderView, SIGNAL(folderDropCopy(KMFolder*)), SLOT(slotCopyMsgToFolder(KMFolder*)) );
}
//Commands not worthy of menu items, but that deserve configurable keybindings
mRemoveDuplicatesAction = new KAction(
i18n("Remove Duplicate Messages"), CTRL+Key_Asterisk, this,
SLOT(removeDuplicates()), actionCollection(), "remove_duplicate_messages");
action->plugAccel( actionCollection()->kaccel() );
action = new KAction(
i18n("Abort Current Operation"), Key_Escape, ProgressManager::instance(),
SLOT(slotAbortAll()), actionCollection(), "cancel" );
action->plugAccel( actionCollection()->kaccel() );
action = new KAction(
i18n("Focus on Next Folder"), CTRL+Key_Right, mFolderTree,
SLOT(incCurrentFolder()), actionCollection(), "inc_current_folder");
action->plugAccel( actionCollection()->kaccel() );
action = new KAction(
i18n("Focus on Previous Folder"), CTRL+Key_Left, mFolderTree,
SLOT(decCurrentFolder()), actionCollection(), "dec_current_folder");
action->plugAccel( actionCollection()->kaccel() );
action = new KAction(
i18n("Select Folder with Focus"), CTRL+Key_Space, mFolderTree,
SLOT(selectCurrentFolder()), actionCollection(), "select_current_folder");
action->plugAccel( actionCollection()->kaccel() );
action = new KAction(
i18n("Focus on Next Message"), ALT+Key_Right, mHeaders,
SLOT(incCurrentMessage()), actionCollection(), "inc_current_message");
action->plugAccel( actionCollection()->kaccel() );
action = new KAction(
i18n("Focus on Previous Message"), ALT+Key_Left, mHeaders,
SLOT(decCurrentMessage()), actionCollection(), "dec_current_message");
action->plugAccel( actionCollection()->kaccel() );
action = new KAction(
i18n("Select Message with Focus"), ALT+Key_Space, mHeaders,
SLOT( selectCurrentMessage() ), actionCollection(), "select_current_message");
action->plugAccel( actionCollection()->kaccel() );
connect( kmkernel->outboxFolder(), SIGNAL( msgRemoved(int, QString) ),
SLOT( startUpdateMessageActionsTimer() ) );
connect( kmkernel->outboxFolder(), SIGNAL( msgAdded(int) ),
SLOT( startUpdateMessageActionsTimer() ) );
}
//-----------------------------------------------------------------------------
void KMMainWidget::activatePanners(void)
{
if (mMsgView) {
QObject::disconnect( mMsgView->copyAction(),
SIGNAL( activated() ),
mMsgView, SLOT( slotCopySelectedText() ));
}
setupFolderView();
if ( mLongFolderList ) {
mSearchAndHeaders->reparent( mPanner2, 0, QPoint( 0, 0 ) );
if (mMsgView) {
mMsgView->reparent( mPanner2, 0, QPoint( 0, 0 ) );
mPanner2->moveToLast( mMsgView );
}
mFolderViewParent = mPanner1;
mFolderView->reparent( mFolderViewParent, 0, QPoint( 0, 0 ) );
mPanner1->moveToLast( mPanner2 );
mPanner1->setSizes( mPanner1Sep );
mPanner1->setResizeMode( mFolderView, QSplitter::KeepSize );
mPanner2->setSizes( mPanner2Sep );
mPanner2->setResizeMode( mSearchAndHeaders, QSplitter::KeepSize );
} else /* !mLongFolderList */ {
mFolderViewParent = mPanner2;
mFolderView->reparent( mFolderViewParent, 0, QPoint( 0, 0 ) );
mSearchAndHeaders->reparent( mPanner2, 0, QPoint( 0, 0 ) );
mPanner2->moveToLast( mSearchAndHeaders );
mPanner1->moveToFirst( mPanner2 );
if (mMsgView) {
mMsgView->reparent( mPanner1, 0, QPoint( 0, 0 ) );
mPanner1->moveToLast( mMsgView );
}
mPanner1->setSizes( mPanner1Sep );
mPanner2->setSizes( mPanner2Sep );
mPanner1->setResizeMode( mPanner2, QSplitter::KeepSize );
mPanner2->setResizeMode( mFolderView, QSplitter::KeepSize );
}
if (mMsgView) {
QObject::connect( mMsgView->copyAction(),
SIGNAL( activated() ),
mMsgView, SLOT( slotCopySelectedText() ));
}
}
//-----------------------------------------------------------------------------
void KMMainWidget::hide()
{
QWidget::hide();
}
//-----------------------------------------------------------------------------
void KMMainWidget::show()
{
QWidget::show();
}
//-------------------------------------------------------------------------
void KMMainWidget::slotSearch()
{
if(!mSearchWin)
{
mSearchWin = new SearchWindow(this, "Search", mFolder, false);
connect(mSearchWin, SIGNAL(destroyed()),
this, SLOT(slotSearchClosed()));
}
else
{
mSearchWin->activateFolder(mFolder);
}
mSearchWin->show();
KWin::activateWindow( mSearchWin->winId() );
}
//-------------------------------------------------------------------------
void KMMainWidget::slotSearchClosed()
{
mSearchWin = 0;
}
//-------------------------------------------------------------------------
void KMMainWidget::slotFind()
{
if( mMsgView )
mMsgView->slotFind();
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotHelp()
{
kapp->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()
{
KAddrBookExternal::openAddressBook(this);
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotImport()
{
KRun::runCommand("kmailcvt");
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotCheckMail()
{
if ( !kmkernel->askToGoOnline() ) {
return;
}
kmkernel->acctMgr()->checkMail(true);
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotCheckOneAccount(int item)
{
if ( !kmkernel->askToGoOnline() ) {
return;
}
kmkernel->acctMgr()->intCheckMail(item);
}
//-----------------------------------------------------------------------------
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;
kapp->dcopClient()->emitDCOPSignal( "unreadCountChanged()", QByteArray() );
// 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 ) {
kdDebug(5006) << newInFolder.find( *it ).data() << " new message(s) in "
<< *it << endl;
KMFolder *folder = kmkernel->findFolderById( *it );
if ( folder && !folder->ignoreNewMail() ) {
showNotification = true;
if ( GlobalSettings::self()->verboseNewMailNotification() ) {
summary += "<br>" + i18n( "1 new message in %1",
"%n new messages in %1",
newInFolder.find( *it ).data() )
.arg( 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 = i18n( "%1 is a list of the number of new messages per folder",
"<b>New mail arrived</b><br>%1" )
.arg( summary );
}
else {
summary = i18n( "New mail arrived" );
}
if(kmkernel->xmlGuiInstance()) {
KNotifyClient::Instance instance(kmkernel->xmlGuiInstance());
KNotifyClient::event( topLevelWidget()->winId(), "new-mail-arrived",
summary );
}
else
KNotifyClient::event( topLevelWidget()->winId(), "new-mail-arrived",
summary );
if (mBeepOnNew) {
KNotifyClient::beep();
}
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotCompose()
{
KMail::Composer * win;
KMMessage* msg = new KMMessage;
if ( mFolder ) {
msg->initHeader( mFolder->identity() );
TemplateParser parser( msg, TemplateParser::NewMessage,
"", false, false, false, false );
parser.process( NULL, mFolder );
win = KMail::makeComposer( msg, mFolder->identity() );
} else {
msg->initHeader();
TemplateParser parser( msg, TemplateParser::NewMessage,
"", false, 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 KPIM::Identity & ident =
kmkernel->identityManager()->identityForUoidOrDefault( mFolder->identity() );
mTemplateFolder = kmkernel->folderMgr()->findIdString( ident.templates() );
}
else mTemplateFolder = kmkernel->templatesFolder();
if ( !mTemplateFolder )
return;
mTemplateMenu->popupMenu()->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");
mTemplateMenu->popupMenu()->insertItem(
KStringHandler::rsqueeze( subj.replace( "&", "&&" ) ), idx );
}
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotNewFromTemplate( int id )
{
if ( !mTemplateFolder )
return;
newFromTemplate(mTemplateFolder->getMsg( id ) );
}
//-----------------------------------------------------------------------------
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").arg( 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();
KConfigGroupSaver saver(config, "General");
if (config->readBoolEntry("warn-before-expire", true)) {
str = i18n("<qt>Are you sure you want to expire the folder <b>%1</b>?</qt>").arg(QStyleSheet::escape( mFolder->label() ));
if (KMessageBox::warningContinueCancel(this, str, i18n("Expire Folder"),
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>").arg( QStyleSheet::escape( mFolder->label() ) );
if (KMessageBox::warningContinueCancel(this, text, title, KGuiItem( title, "edittrash"))
!= 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>")
.arg( QStyleSheet::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>")
.arg( QStyleSheet::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>")
.arg( QStyleSheet::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>")
.arg( QStyleSheet::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>")
.arg( QStyleSheet::escape( mFolder->label() ) );
}
}
}
if (KMessageBox::warningContinueCancel(this, str, title,
KGuiItem( i18n("&Delete"), "editdelete"))
== 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>").arg( (*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"), i18n("&Refresh") ) == KMessageBox::Continue )
kmkernel->acctMgr()->invalidateIMAPFolders();
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotExpireAll() {
KConfig *config = KMKernel::config();
int ret = 0;
KConfigGroupSaver saver(config, "General");
if (config->readBoolEntry("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?"), 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" ),
i18n( "Use HTML" ),
"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" ),
i18n( "Load External References" ),
"OverrideHtmlLoadExtWarning", false);
if( result == KMessageBox::Cancel ) {
mPreferHtmlLoadExtAction->setChecked( false );
return;
}
}
mFolderHtmlLoadExtPref = !mFolderHtmlLoadExtPref;
if (mMsgView) {
mMsgView->setHtmlLoadExtOverride(mFolderHtmlLoadExtPref);
mMsgView->update( true );
}
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotOverrideThread()
{
mFolderThreadPref = !mFolderThreadPref;
mHeaders->setNestedOverride(mFolderThreadPref);
mThreadBySubjectAction->setEnabled(mThreadMessagesAction->isChecked());
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotToggleSubjectThreading()
{
mFolderThreadSubjPref = !mFolderThreadSubjPref;
mHeaders->setSubjectThreading(mFolderThreadSubjPref);
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotMessageQueuedOrDrafted()
{
if (!kmkernel->folderIsDraftOrOutbox(mFolder))
return;
if (mMsgView)
mMsgView->update(true);
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotForwardInlineMsg()
{
KMMessageList* selected = mHeaders->selectedMsgs();
KMCommand *command = 0L;
if(selected && !selected->isEmpty()) {
command = new KMForwardInlineCommand( this, *selected,
mFolder->identity() );
} else {
command = new KMForwardInlineCommand( 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::slotForwardDigestMsg()
{
KMMessageList* selected = mHeaders->selectedMsgs();
KMCommand *command = 0L;
if(selected && !selected->isEmpty()) {
command = new KMForwardDigestCommand( this, *selected, mFolder->identity() );
} else {
command = new KMForwardDigestCommand( 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( int tid )
{
QString text = mMsgView? mMsgView->copyText() : "";
QString tmpl = mCustomTemplates[ tid ];
kdDebug() << "Reply with template: " << tmpl << " (" << tid << ")" << endl;
KMCommand *command = new KMCustomReplyToCommand( this,
mHeaders->currentMsg(),
text,
tmpl );
command->start();
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotCustomReplyAllToMsg( int tid )
{
QString text = mMsgView? mMsgView->copyText() : "";
QString tmpl = mCustomTemplates[ tid ];
kdDebug() << "Reply to All with template: " << tmpl << " (" << tid << ")" << endl;
KMCommand *command = new KMCustomReplyAllToCommand( this,
mHeaders->currentMsg(),
text,
tmpl );
command->start();
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotCustomForwardMsg( int tid )
{
QString tmpl = mCustomTemplates[ tid ];
kdDebug() << "Forward with template: " << tmpl << " (" << tid << ")" << endl;
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
QCString 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..." ).arg( 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::KMFolderSelDlg dlg( this, i18n("Jump to Folder"), true );
KMFolder* dest;
if (!dlg.exec()) return;
if (!(dest = dlg.folder())) return;
slotSelectFolder( dest );
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotMoveMsg()
{
KMail::KMFolderSelDlg 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()
{
KProcess certManagerProc; // save to create on the heap, since
// there is no parent
certManagerProc << "kleopatra";
if( !certManagerProc.start( KProcess::DontCare ) )
KMessageBox::error( this, i18n( "Could not start certificate manager; "
"please check your installation." ),
i18n( "KMail Error" ) );
else
kdDebug(5006) << "\nslotStartCertManager(): certificate manager started.\n" << endl;
// process continues to run even after the KProcess object goes
// out of scope here, since it is started in DontCare run mode.
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotStartWatchGnuPG()
{
KProcess certManagerProc;
certManagerProc << "kwatchgnupg";
if( !certManagerProc.start( KProcess::DontCare ) )
KMessageBox::error( this, i18n( "Could not start GnuPG LogViewer (kwatchgnupg); "
"please check your installation." ),
i18n( "KMail Error" ) );
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotCopyMsg()
{
KMail::KMFolderSelDlg 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.readBoolEntry( "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, 0, 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 )
{
if ( GlobalSettings::self()->networkState() == GlobalSettings::EnumNetworkState::Online )
actionCollection()->action( "online_status" )->setText( i18n("Work Offline") );
else
actionCollection()->action( "online_status" )->setText( i18n("Work Online") );
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotSendQueued()
{
if ( !kmkernel->askToGoOnline() ) {
return;
}
kmkernel->msgSender()->sendQueued();
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotSendQueuedVia( int item )
{
if ( !kmkernel->askToGoOnline() ) {
return;
}
QStringList availTransports= KMail::TransportManager::transportNames();
QString customTransport = availTransports[ item ];
kmkernel->msgSender()->sendQueued( customTransport );
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotViewChange()
{
if(mBodyPartsMenu->isItemChecked(mBodyPartsMenu->idAt(0)))
{
mBodyPartsMenu->setItemChecked(mBodyPartsMenu->idAt(0),false);
mBodyPartsMenu->setItemChecked(mBodyPartsMenu->idAt(1),true);
}
else if(mBodyPartsMenu->isItemChecked(mBodyPartsMenu->idAt(1)))
{
mBodyPartsMenu->setItemChecked(mBodyPartsMenu->idAt(1),false);
mBodyPartsMenu->setItemChecked(mBodyPartsMenu->idAt(0),true);
}
//mMsgView->setInline(!mMsgView->isInline());
}
//-----------------------------------------------------------------------------
void KMMainWidget::folderSelectedUnread( KMFolder* aFolder )
{
folderSelected( aFolder, true );
slotChangeCaption( mFolderTree->currentItem() );
}
//-----------------------------------------------------------------------------
void KMMainWidget::folderSelected()
{
folderSelected( mFolder );
updateFolderMenu();
// opened() before the getAndCheckFolder() in folderSelected
if ( mFolder && mFolder->folderType() == KMFolderTypeImap )
mFolder->close("mainwidget");
}
//-----------------------------------------------------------------------------
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() ) );
}
mFolder = aFolder;
if ( aFolder && aFolder->folderType() == KMFolderTypeImap )
{
if ( kmkernel->isOffline() ) {
showOfflinePage();
return;
}
KMFolderImap *imap = static_cast<KMFolderImap*>(aFolder->storage());
if ( newFolder && !mFolder->noContent() )
{
imap->open("mainwidget"); // will be closed in the folderSelected slot
// first get new headers before we select the folder
imap->setSelected( true );
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 );
connect( mShowBusySplashTimer, SIGNAL( timeout() ), this, SLOT( slotShowBusySplash() ) );
mShowBusySplashTimer->start( GlobalSettings::self()->folderLoadingTimeout(), true );
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, 0,
"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)
{
QListViewItem* 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()
{
kdDebug(5006) << "KMMainWidget::slotReplaceMsgByUnencryptedVersion()" << endl;
KMMessage* oldMsg = mHeaders->currentMsg();
if( oldMsg ) {
kdDebug(5006) << "KMMainWidget - old message found" << endl;
if( oldMsg->hasUnencryptedMsg() ) {
kdDebug(5006) << "KMMainWidget - extra unencrypted message found" << endl;
KMMessage* newMsg = oldMsg->unencryptedMsg();
// adjust the message id
{
QString msgId( oldMsg->msgId() );
QString prefix("DecryptedMsg.");
int oldIdx = msgId.find(prefix, 0, false);
if( -1 == oldIdx ) {
int leftAngle = msgId.findRev( '<' );
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
kdDebug(5006) << "KMMainWidget - adding unencrypted message to folder" << endl;
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 ) {
kdDebug(5006) << "KMMainWidget - deleting encrypted message" << endl;
mFolder->take( idx );
}
kdDebug(5006) << "KMMainWidget - updating message actions" << endl;
updateMessageActions();
kdDebug(5006) << "KMMainWidget - done." << endl;
} else
kdDebug(5006) << "KMMainWidget - NO EXTRA UNENCRYPTED MESSAGE FOUND" << endl;
} else
kdDebug(5006) << "KMMainWidget - PANIC: NO OLD MESSAGE FOUND" << endl;
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotSetThreadStatusNew()
{
mHeaders->setThreadStatus(KMMsgStatusNew);
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotSetThreadStatusUnread()
{
mHeaders->setThreadStatus(KMMsgStatusUnread);
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotSetThreadStatusFlag()
{
mHeaders->setThreadStatus(KMMsgStatusFlag, true);
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotSetThreadStatusRead()
{
mHeaders->setThreadStatus(KMMsgStatusRead);
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotSetThreadStatusTodo()
{
mHeaders->setThreadStatus(KMMsgStatusTodo, true);
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotSetThreadStatusWatched()
{
mHeaders->setThreadStatus(KMMsgStatusWatched, true);
if (mWatchThreadAction->isChecked()) {
mIgnoreThreadAction->setChecked(false);
}
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotSetThreadStatusIgnored()
{
mHeaders->setThreadStatus(KMMsgStatusIgnored, 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::slotNextImportantMessage() {
//mHeaders->nextImportantMessage();
}
void KMMainWidget::slotPrevMessage() { mHeaders->prevMessage(); }
void KMMainWidget::slotPrevUnreadMessage()
{
if ( !mHeaders->prevUnreadMessage() )
if ( GlobalSettings::self()->loopOnGotoUnread() == GlobalSettings::EnumLoopOnGotoUnread::LoopInAllFolders )
mFolderTree->prevUnreadFolder();
}
void KMMainWidget::slotPrevImportantMessage() {
//mHeaders->prevImportantMessage();
}
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.readBoolEntry( "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)
{
KPopupMenu * menu = new KPopupMenu;
updateMessageMenu();
mUrlCurrent = aUrl;
bool urlMenuAdded = false;
if (!aUrl.isEmpty())
{
if (aUrl.protocol() == "mailto")
{
// popup on a mailto URL
mMsgView->mailToComposeAction()->plug( menu );
mMsgView->mailToReplyAction()->plug( menu );
mMsgView->mailToForwardAction()->plug( menu );
menu->insertSeparator();
mMsgView->addAddrBookAction()->plug( menu );
mMsgView->openAddrBookAction()->plug( menu );
mMsgView->copyURLAction()->plug( menu );
mMsgView->startImChatAction()->plug( menu );
// only enable if our KIMProxy is functional
mMsgView->startImChatAction()->setEnabled( kmkernel->imProxy()->initialize() );
} else {
// popup on a not-mailto URL
mMsgView->urlOpenAction()->plug( menu );
mMsgView->addBookmarksAction()->plug( menu );
mMsgView->urlSaveAsAction()->plug( menu );
mMsgView->copyURLAction()->plug( menu );
}
if ( aUrl.protocol() == "im" )
{
// popup on an IM address
// no need to check the KIMProxy is initialized, as these protocols will
// only be present if it is.
mMsgView->startImChatAction()->plug( menu );
}
urlMenuAdded=true;
kdDebug( 0 ) << k_funcinfo << " URL is: " << aUrl << endl;
}
if(mMsgView && !mMsgView->copyText().isEmpty()) {
if ( urlMenuAdded )
menu->insertSeparator();
mMsgActions->replyMenu()->plug(menu);
menu->insertSeparator();
mMsgView->copyAction()->plug( menu );
mMsgView->selectAllAction()->plug( menu );
} 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() ) {
mUseAction->plug( menu );
} else {
if ( !mFolder->isSent() )
mMsgActions->replyMenu()->plug( menu );
mForwardActionMenu->plug( menu );
}
editAction()->plug(menu);
menu->insertSeparator();
mCopyActionMenu->plug( menu );
mMoveActionMenu->plug( menu );
menu->insertSeparator();
mMsgActions->messageStatusMenu()->plug( menu );
menu->insertSeparator();
viewSourceAction()->plug(menu);
if(mMsgView) {
mMsgView->toggleFixFontAction()->plug(menu);
}
menu->insertSeparator();
mPrintAction->plug( menu );
mSaveAsAction->plug( menu );
mSaveAttachmentsAction->plug( menu );
menu->insertSeparator();
if( mFolder->isTrash() )
mDeleteAction->plug( menu );
else
mTrashAction->plug( menu );
menu->insertSeparator();
mMsgActions->createTodoAction()->plug( menu );
}
KAcceleratorManager::manage(menu);
menu->exec(aPoint, 0);
delete menu;
}
//-----------------------------------------------------------------------------
void KMMainWidget::getAccountMenu()
{
QStringList actList;
mActMenu->clear();
actList = kmkernel->acctMgr()->getAccounts();
QStringList::Iterator it;
int id = 0;
for(it = actList.begin(); it != actList.end() ; ++it, id++)
mActMenu->insertItem((*it).replace("&", "&&"), id);
}
//-----------------------------------------------------------------------------
void KMMainWidget::getTransportMenu()
{
QStringList availTransports;
mSendMenu->clear();
availTransports = KMail::TransportManager::transportNames();
QStringList::Iterator it;
int id = 0;
for(it = availTransports.begin(); it != availTransports.end() ; ++it, id++)
mSendMenu->insertItem((*it).replace("&", "&&"), id);
}
//-----------------------------------------------------------------------------
void KMMainWidget::updateCustomTemplateMenus()
{
if ( !mCustomTemplateActions.isEmpty() ) {
QPtrList<KAction>::iterator ait = mCustomTemplateActions.begin();
for ( ; ait != mCustomTemplateActions.end() ; ++ait ) {
(*ait)->unplugAll();
delete (*ait);
}
mCustomTemplateActions.clear();
}
delete mCustomReplyActionMenu;
delete mCustomReplyAllActionMenu;
delete mCustomForwardActionMenu;
delete mCustomReplyMapper;
delete mCustomReplyAllMapper;
delete mCustomForwardMapper;
mCustomForwardActionMenu =
new KActionMenu( i18n("Forward With Custom Template"),
"mail_custom_forward",
actionCollection(), "custom_forward" );
QSignalMapper *mCustomForwardMapper = new QSignalMapper( this );
connect( mCustomForwardMapper, SIGNAL( mapped( int ) ),
this, SLOT( slotCustomForwardMsg( int ) ) );
mForwardActionMenu->insert( mCustomForwardActionMenu );
mCustomReplyActionMenu =
new KActionMenu( i18n("Reply With Custom Template"), "mail_custom_reply",
actionCollection(), "custom_reply" );
QSignalMapper *mCustomReplyMapper = new QSignalMapper( this );
connect( mCustomReplyMapper, SIGNAL( mapped( int ) ),
this, SLOT( slotCustomReplyToMsg( int ) ) );
mMsgActions->replyMenu()->insert( mCustomReplyActionMenu );
mCustomReplyAllActionMenu =
new KActionMenu( i18n("Reply to All With Custom Template"),
"mail_custom_reply_all",
actionCollection(), "custom_reply_all" );
QSignalMapper *mCustomReplyAllMapper = new QSignalMapper( this );
connect( mCustomReplyAllMapper, SIGNAL( mapped( int ) ),
this, SLOT( slotCustomReplyAllToMsg( int ) ) );
mMsgActions->replyMenu()->insert( mCustomReplyAllActionMenu );
mCustomTemplates.clear();
QStringList list = GlobalSettingsBase::self()->customTemplates();
QStringList::iterator it = list.begin();
int idx = 0;
int replyc = 0;
int replyallc = 0;
int forwardc = 0;
for ( ; it != list.end(); ++it ) {
CTemplates t( *it );
mCustomTemplates.append( *it );
KAction *action;
switch ( t.type() ) {
case CustomTemplates::TReply:
action = new KAction( (*it).replace( "&", "&&" ),
KShortcut( t.shortcut() ),
mCustomReplyMapper,
SLOT( map() ),
actionCollection(),
(*it).utf8() );
mCustomReplyMapper->setMapping( action, idx );
mCustomReplyActionMenu->insert( action, idx );
mCustomTemplateActions.append( action );
++replyc;
break;
case CustomTemplates::TReplyAll:
action = new KAction( (*it).replace( "&", "&&" ),
KShortcut( t.shortcut() ),
mCustomReplyAllMapper,
SLOT( map() ),
actionCollection(),
(*it).utf8() );
mCustomReplyAllMapper->setMapping( action, idx );
mCustomReplyAllActionMenu->insert( action, idx );
mCustomTemplateActions.append( action );
++replyallc;
break;
case CustomTemplates::TForward:
action = new KAction( (*it).replace( "&", "&&" ),
KShortcut( t.shortcut() ),
mCustomForwardMapper,
SLOT( map() ),
actionCollection(),
(*it).utf8() );
mCustomForwardMapper->setMapping( action, idx );
mCustomForwardActionMenu->insert( action, idx );
mCustomTemplateActions.append( action );
++forwardc;
break;
case CustomTemplates::TUniversal:
action = new KAction( (*it).replace( "&", "&&" ),
KShortcut::null(),
mCustomReplyMapper,
SLOT( map() ),
actionCollection(),
(*it).utf8() );
mCustomReplyMapper->setMapping( action, idx );
mCustomReplyActionMenu->insert( action, idx );
mCustomTemplateActions.append( action );
++replyc;
action = new KAction( (*it).replace( "&", "&&" ),
KShortcut::null(),
mCustomReplyAllMapper,
SLOT( map() ),
actionCollection(),
(*it).utf8() );
mCustomReplyAllMapper->setMapping( action, idx );
mCustomReplyAllActionMenu->insert( action, idx );
mCustomTemplateActions.append( action );
++replyallc;
action = new KAction( (*it).replace( "&", "&&" ),
KShortcut::null(),
mCustomForwardMapper,
SLOT( map() ),
actionCollection(),
(*it).utf8() );
mCustomForwardMapper->setMapping( action, idx );
mCustomForwardActionMenu->insert( action, idx );
mCustomTemplateActions.append( action );
++forwardc;
break;
}
++idx;
}
if ( !replyc ) {
mCustomReplyActionMenu->popupMenu()->insertItem( i18n( "(no custom templates)" ), 0 );
mCustomReplyActionMenu->popupMenu()->setItemEnabled( 0, false );
mCustomReplyActionMenu->setEnabled(false);
}
if ( !replyallc ) {
mCustomReplyAllActionMenu->popupMenu()->insertItem( i18n( "(no custom templates)" ), 0 );
mCustomReplyAllActionMenu->popupMenu()->setItemEnabled( 0, false );
mCustomReplyAllActionMenu->setEnabled(false);
}
if ( !forwardc ) {
mCustomForwardActionMenu->popupMenu()->insertItem( i18n( "(no custom templates)" ), 0 );
mCustomForwardActionMenu->popupMenu()->setItemEnabled( 0, false );
mCustomForwardActionMenu->setEnabled(false);
}
}
//-----------------------------------------------------------------------------
void KMMainWidget::setupActions()
{
mMsgActions = new KMail::MessageActions( actionCollection(), this );
mMsgActions->setMessageView( mMsgView );
//----- File Menu
mSaveAsAction = new KAction( i18n("Save &As..."), "filesave",
KStdAccel::shortcut(KStdAccel::Save),
this, SLOT(slotSaveMsg()), actionCollection(), "file_save_as" );
mOpenAction = KStdAction::open( this, SLOT( slotOpenMsg() ),
actionCollection() );
(void) new KAction( i18n("&Compact All Folders"), 0,
this, SLOT(slotCompactAll()),
actionCollection(), "compact_all_folders" );
(void) new KAction( i18n("&Expire All Folders"), 0,
this, SLOT(slotExpireAll()),
actionCollection(), "expire_all_folders" );
(void) new KAction( i18n("&Refresh Local IMAP Cache"), "refresh",
this, SLOT(slotInvalidateIMAPFolders()),
actionCollection(), "file_invalidate_imap_cache" );
(void) new KAction( i18n("Empty All &Trash Folders"), 0,
KMKernel::self(), SLOT(slotEmptyTrash()),
actionCollection(), "empty_trash" );
(void) new KAction( i18n("Check &Mail"), "mail_get", CTRL+Key_L,
this, SLOT(slotCheckMail()),
actionCollection(), "check_mail" );
mFavoritesCheckMailAction = new KAction( i18n("Check Mail in Favorite Folders"),
"mail_get", CTRL+SHIFT+Key_L, 0, 0,
actionCollection(), "favorite_check_mail" );
if ( mFavoriteFolderView )
connect( mFavoritesCheckMailAction, SIGNAL(activated()), mFavoriteFolderView, SLOT(checkMail()) );
KActionMenu *actActionMenu = new
KActionMenu( i18n("Check Mail &In"), "mail_get", actionCollection(),
"check_mail_in" );
actActionMenu->setDelayed(true); //needed for checking "all accounts"
connect(actActionMenu,SIGNAL(activated()),this,SLOT(slotCheckMail()));
mActMenu = actActionMenu->popupMenu();
connect(mActMenu,SIGNAL(activated(int)),this,SLOT(slotCheckOneAccount(int)));
connect(mActMenu,SIGNAL(aboutToShow()),this,SLOT(getAccountMenu()));
(void) new KAction( i18n("&Send Queued Messages"), "mail_send", 0, this,
SLOT(slotSendQueued()), actionCollection(), "send_queued");
(void) new KAction( i18n("Online Status (unknown)"), "online_status", 0, this,
SLOT(slotOnlineStatus()), actionCollection(), "online_status");
KActionMenu *sendActionMenu = new
KActionMenu( i18n("Send Queued Messages Via"), "mail_send_via", actionCollection(),
"send_queued_via" );
sendActionMenu->setDelayed(true);
mSendMenu = sendActionMenu->popupMenu();
connect(mSendMenu,SIGNAL(activated(int)), this, SLOT(slotSendQueuedVia(int)));
connect(mSendMenu,SIGNAL(aboutToShow()),this,SLOT(getTransportMenu()));
KAction *act;
//----- Tools menu
if (parent()->inherits("KMMainWin")) {
act = new KAction( i18n("&Address Book..."), "contents", 0, this,
SLOT(slotAddrBook()), actionCollection(), "addressbook" );
if (KStandardDirs::findExe("kaddressbook").isEmpty()) act->setEnabled(false);
}
act = new KAction( i18n("Certificate Manager..."), "pgp-keys", 0, this,
SLOT(slotStartCertManager()), actionCollection(), "tools_start_certman");
// disable action if no certman binary is around
if (KStandardDirs::findExe("kleopatra").isEmpty()) act->setEnabled(false);
act = new KAction( i18n("GnuPG Log Viewer..."), "pgp-keys", 0, this,
SLOT(slotStartWatchGnuPG()), actionCollection(), "tools_start_kwatchgnupg");
// disable action if no kwatchgnupg binary is around
if (KStandardDirs::findExe("kwatchgnupg").isEmpty()) act->setEnabled(false);
act = new KAction( i18n("&Import Messages..."), "fileopen", 0, this,
SLOT(slotImport()), actionCollection(), "import" );
if (KStandardDirs::findExe("kmailcvt").isEmpty()) act->setEnabled(false);
#if !defined(NDEBUG)
(void) new KAction( i18n("&Debug Sieve..."),
"idea", 0, this, SLOT(slotDebugSieve()),
actionCollection(), "tools_debug_sieve" );
#endif
if ( GlobalSettings::allowOutOfOfficeSettings() ) {
(void) new KAction( i18n("Edit \"Out of Office\" Replies..."),
"configure", 0, this, SLOT(slotEditVacation()),
actionCollection(), "tools_edit_vacation" );
}
(void) new KAction( i18n("Filter &Log Viewer..."), 0, this,
SLOT(slotFilterLogViewer()), actionCollection(), "filter_log_viewer" );
(void) new KAction( i18n("&Anti-Spam Wizard..."), 0, this,
SLOT(slotAntiSpamWizard()), actionCollection(), "antiSpamWizard" );
(void) new KAction( i18n("&Anti-Virus Wizard..."), 0, this,
SLOT(slotAntiVirusWizard()), actionCollection(), "antiVirusWizard" );
//----- Edit Menu
mTrashAction = new KAction( KGuiItem( i18n("&Move to Trash"), "edittrash",
i18n("Move message to trashcan") ),
Key_Delete, this, SLOT(slotTrashMsg()),
actionCollection(), "move_to_trash" );
/* 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( i18n("&Delete"), "editdelete", SHIFT+Key_Delete, this,
SLOT(slotDeleteMsg()), actionCollection(), "delete" );
mDeleteAction->plugAccel( actionCollection()->kaccel() );
mTrashThreadAction = new KAction( KGuiItem( i18n("M&ove Thread to Trash"), "edittrash",
i18n("Move thread to trashcan") ),
CTRL+Key_Delete, this, SLOT(slotTrashThread()),
actionCollection(), "move_thread_to_trash" );
mDeleteThreadAction = new KAction( i18n("Delete T&hread"), "editdelete", CTRL+SHIFT+Key_Delete, this,
SLOT(slotDeleteThread()), actionCollection(), "delete_thread" );
(void) new KAction( i18n("&Find Messages..."), "mail_find", Key_S, this,
SLOT(slotRequestFullSearchFromQuickSearch()), actionCollection(), "search_messages" );
mFindInMessageAction = new KAction( i18n("&Find in Message..."), "find", KStdAccel::shortcut(KStdAccel::Find), this,
SLOT(slotFind()), actionCollection(), "find_in_messages" );
(void) new KAction( i18n("Select &All Messages"), KStdAccel::selectAll(), this,
SLOT(slotMarkAll()), actionCollection(), "mark_all_messages" );
//----- Folder Menu
mNewFolderAction = new KAction( i18n("&New Folder..."), "folder_new", 0, mFolderTree,
SLOT(addChildFolder()), actionCollection(), "new_folder" );
mModifyFolderAction = new KAction( i18n("&Properties"), "configure", 0, this,
SLOT(slotModifyFolder()), actionCollection(), "modify" );
mFolderMailingListPropertiesAction = new KAction( i18n("&Mailing List Management..."),
/*"folder_mailinglist_properties",*/ 0, this, SLOT( slotFolderMailingListProperties() ),
actionCollection(), "folder_mailinglist_properties" );
mFolderShortCutCommandAction = new KAction( i18n("&Assign Shortcut..."), "configure_shortcuts",
0, this, SLOT( slotFolderShortcutCommand() ), actionCollection(),
"folder_shortcut_command" );
mMarkAllAsReadAction = new KAction( i18n("Mark All Messages as &Read"), "goto", 0, this,
SLOT(slotMarkAllAsRead()), actionCollection(), "mark_all_as_read" );
mExpireFolderAction = new KAction(i18n("&Expiration Settings"), 0, this, SLOT(slotExpireFolder()),
actionCollection(), "expire");
mCompactFolderAction = new KAction( i18n("&Compact Folder"), 0, this,
SLOT(slotCompactFolder()), actionCollection(), "compact" );
mRefreshFolderAction = new KAction( i18n("Check Mail &in This Folder"), "reload",
KStdAccel::shortcut( KStdAccel::Reload ), this,
SLOT(slotRefreshFolder()),
actionCollection(), "refresh_folder" );
mTroubleshootFolderAction = 0; // set in initializeIMAPActions
mEmptyFolderAction = new KAction( "foo" /*set in updateFolderMenu*/, "edittrash", 0, this,
SLOT(slotEmptyFolder()), actionCollection(), "empty" );
mRemoveFolderAction = new KAction( "foo" /*set in updateFolderMenu*/, "editdelete", 0, this,
SLOT(slotRemoveFolder()), actionCollection(), "delete_folder" );
mPreferHtmlAction = new KToggleAction( i18n("Prefer &HTML to Plain Text"), 0, this,
SLOT(slotOverrideHtml()), actionCollection(), "prefer_html" );
mPreferHtmlLoadExtAction = new KToggleAction( i18n("Load E&xternal References"), 0, this,
SLOT(slotOverrideHtmlLoadExt()), actionCollection(), "prefer_html_external_refs" );
mThreadMessagesAction = new KToggleAction( i18n("&Thread Messages"), 0, this,
SLOT(slotOverrideThread()), actionCollection(), "thread_messages" );
mThreadBySubjectAction = new KToggleAction( i18n("Thread Messages also by &Subject"), 0, this,
SLOT(slotToggleSubjectThreading()), actionCollection(), "thread_messages_by_subject" );
new KAction( i18n("Copy Folder"), "editcopy", SHIFT+CTRL+Key_C, folderTree(),
SLOT(copyFolder()), actionCollection(), "copy_folder" );
new KAction( i18n("Cut Folder"), "editcut", SHIFT+CTRL+Key_X, folderTree(),
SLOT(cutFolder()), actionCollection(), "cut_folder" );
new KAction( i18n("Paste Folder"), "editpaste", SHIFT+CTRL+Key_V, folderTree(),
SLOT(pasteFolder()), actionCollection(), "paste_folder" );
new KAction( i18n("Copy Messages"), "editcopy", ALT+CTRL+Key_C, headers(),
SLOT(copyMessages()), actionCollection(), "copy_messages" );
new KAction( i18n("Cut Messages"), "editcut", ALT+CTRL+Key_X, headers(),
SLOT(cutMessages()), actionCollection(), "cut_messages" );
new KAction( i18n("Paste Messages"), "editpaste", ALT+CTRL+Key_V, headers(),
SLOT(pasteMessages()), actionCollection(), "paste_messages" );
//----- Message Menu
(void) new KAction( i18n("&New Message..."), "mail_new", KStdAccel::shortcut(KStdAccel::New), this,
SLOT(slotCompose()), actionCollection(), "new_message" );
mTemplateMenu =
new KActionMenu( i18n("New Message From &Template"), "filenew",
actionCollection(), "new_from_template" );
mTemplateMenu->setDelayed( true );
connect( mTemplateMenu->popupMenu(), SIGNAL( aboutToShow() ), this,
SLOT( slotShowNewFromTemplate() ) );
connect( mTemplateMenu->popupMenu(), SIGNAL( activated(int) ), this,
SLOT( slotNewFromTemplate(int) ) );
KAction* newToML = new KAction( i18n("New Message t&o Mailing-List..."), "mail_post_to",
CTRL+SHIFT+Key_N, this,
SLOT(slotPostToML()), actionCollection(), "post_message" );
newToML->plugAccel( actionCollection()->kaccel() );
mForwardActionMenu = new KActionMenu( i18n("Message->","&Forward"),
"mail_forward", actionCollection(),
"message_forward" );
mForwardInlineAction = new KAction( i18n("&Inline..."),
"mail_forward", 0, this,
SLOT(slotForwardInlineMsg()),
actionCollection(),
"message_forward_inline" );
mForwardAttachedAction = new KAction( i18n("Message->Forward->","As &Attachment..."),
"mail_forward", 0, this,
SLOT(slotForwardAttachedMsg()),
actionCollection(),
"message_forward_as_attachment" );
mForwardDigestAction = new KAction( i18n("Message->Forward->","As Di&gest..."),
"mail_forward", 0, this,
SLOT(slotForwardDigestMsg()),
actionCollection(),
"message_forward_as_digest" );
mRedirectAction = new KAction( i18n("Message->Forward->","&Redirect..."),
"mail_forward", Key_E, this,
SLOT(slotRedirectMsg()),
actionCollection(),
"message_forward_redirect" );
if ( GlobalSettings::self()->forwardingInlineByDefault() ) {
mForwardActionMenu->insert( mForwardInlineAction );
mForwardActionMenu->insert( mForwardAttachedAction );
mForwardInlineAction->setShortcut( Key_F );
mForwardAttachedAction->setShortcut( SHIFT+Key_F );
connect( mForwardActionMenu, SIGNAL(activated()), this,
SLOT(slotForwardInlineMsg()) );
} else {
mForwardActionMenu->insert( mForwardAttachedAction );
mForwardActionMenu->insert( mForwardInlineAction );
mForwardInlineAction->setShortcut( SHIFT+Key_F );
mForwardAttachedAction->setShortcut( Key_F );
connect( mForwardActionMenu, SIGNAL(activated()), this,
SLOT(slotForwardAttachedMsg()) );
}
mForwardActionMenu->insert( mForwardDigestAction );
mForwardActionMenu->insert( mRedirectAction );
mSendAgainAction = new KAction( i18n("Send A&gain..."), 0, this,
SLOT(slotResendMsg()), actionCollection(), "send_again" );
//----- Create filter actions
mFilterMenu = new KActionMenu( i18n("&Create Filter"), "filter", actionCollection(), "create_filter" );
connect( mFilterMenu, SIGNAL(activated()), this,
SLOT(slotFilter()) );
mSubjectFilterAction = new KAction( i18n("Filter on &Subject..."), 0, this,
SLOT(slotSubjectFilter()),
actionCollection(), "subject_filter");
mFilterMenu->insert( mSubjectFilterAction );
mFromFilterAction = new KAction( i18n("Filter on &From..."), 0, this,
SLOT(slotFromFilter()),
actionCollection(), "from_filter");
mFilterMenu->insert( mFromFilterAction );
mToFilterAction = new KAction( i18n("Filter on &To..."), 0, this,
SLOT(slotToFilter()),
actionCollection(), "to_filter");
mFilterMenu->insert( mToFilterAction );
mListFilterAction = new KAction( i18n("Filter on Mailing-&List..."), 0, this,
SLOT(slotMailingListFilter()), actionCollection(),
"mlist_filter");
mFilterMenu->insert( mListFilterAction );
mPrintAction = KStdAction::print (this, SLOT(slotPrintMsg()), actionCollection());
mUseAction = new KAction( i18n("New Message From &Template"), "filenew",
Key_N, this, SLOT( slotUseTemplate() ),
actionCollection(), "use_template" );
mUseAction->plugAccel( actionCollection()->kaccel() );
//----- "Mark Thread" submenu
mThreadStatusMenu = new KActionMenu ( i18n( "Mark &Thread" ),
actionCollection(), "thread_status" );
mMarkThreadAsReadAction = new KAction(KGuiItem(i18n("Mark Thread as &Read"), "kmmsgread",
i18n("Mark all messages in the selected thread as read")),
0, this, SLOT(slotSetThreadStatusRead()),
actionCollection(), "thread_read");
mThreadStatusMenu->insert( mMarkThreadAsReadAction );
mMarkThreadAsNewAction = new KAction(KGuiItem(i18n("Mark Thread as &New"), "kmmsgnew",
i18n("Mark all messages in the selected thread as new")),
0, this, SLOT(slotSetThreadStatusNew()),
actionCollection(), "thread_new");
mThreadStatusMenu->insert( mMarkThreadAsNewAction );
mMarkThreadAsUnreadAction = new KAction(KGuiItem(i18n("Mark Thread as &Unread"), "kmmsgunseen",
i18n("Mark all messages in the selected thread as unread")),
0, this, SLOT(slotSetThreadStatusUnread()),
actionCollection(), "thread_unread");
mThreadStatusMenu->insert( mMarkThreadAsUnreadAction );
mThreadStatusMenu->insert( new KActionSeparator( this ) );
//----- "Mark Thread" toggle actions
mToggleThreadFlagAction = new KToggleAction(i18n("Mark Thread as &Important"), "mail_flag",
0, this, SLOT(slotSetThreadStatusFlag()),
actionCollection(), "thread_flag");
mToggleThreadFlagAction->setCheckedState( i18n("Remove &Important Thread Mark") );
mThreadStatusMenu->insert( mToggleThreadFlagAction );
mToggleThreadTodoAction = new KToggleAction(i18n("Mark Thread as &Action Item"), "mail_todo",
0, this, SLOT(slotSetThreadStatusTodo()),
actionCollection(), "thread_todo");
mToggleThreadTodoAction->setCheckedState( i18n("Remove &Action Item Thread Mark") );
mThreadStatusMenu->insert( mToggleThreadTodoAction );
//------- "Watch and ignore thread" actions
mWatchThreadAction = new KToggleAction(i18n("&Watch Thread"), "kmmsgwatched",
0, this, SLOT(slotSetThreadStatusWatched()),
actionCollection(), "thread_watched");
mIgnoreThreadAction = new KToggleAction(i18n("&Ignore Thread"), "mail_ignore",
0, this, SLOT(slotSetThreadStatusIgnored()),
actionCollection(), "thread_ignored");
mThreadStatusMenu->insert( new KActionSeparator( this ) );
mThreadStatusMenu->insert( mWatchThreadAction );
mThreadStatusMenu->insert( mIgnoreThreadAction );
mSaveAttachmentsAction = new KAction( i18n("Save A&ttachments..."), "attach",
0, this, SLOT(slotSaveAttachments()),
actionCollection(), "file_save_attachments" );
mMoveActionMenu = new KActionMenu( i18n("&Move To" ),
actionCollection(), "move_to" );
mCopyActionMenu = new KActionMenu( i18n("&Copy To" ),
actionCollection(), "copy_to" );
mApplyAllFiltersAction = new KAction( i18n("Appl&y All Filters"), "filter",
CTRL+Key_J, this,
SLOT(slotApplyFilters()),
actionCollection(), "apply_filters" );
mApplyFilterActionsMenu = new KActionMenu( i18n("A&pply Filter" ),
actionCollection(),
"apply_filter_actions" );
//----- View Menu
// Unread Submenu
KActionMenu * unreadMenu =
new KActionMenu( i18n("View->", "&Unread Count"),
actionCollection(), "view_unread" );
unreadMenu->setToolTip( i18n("Choose how to display the count of unread messages") );
mUnreadColumnToggle = new KRadioAction( i18n("View->Unread Count", "View in &Separate Column"), 0, this,
SLOT(slotToggleUnread()),
actionCollection(), "view_unread_column" );
mUnreadColumnToggle->setExclusiveGroup( "view_unread_group" );
unreadMenu->insert( mUnreadColumnToggle );
mUnreadTextToggle = new KRadioAction( i18n("View->Unread Count", "View After &Folder Name"), 0, this,
SLOT(slotToggleUnread()),
actionCollection(), "view_unread_text" );
mUnreadTextToggle->setExclusiveGroup( "view_unread_group" );
unreadMenu->insert( mUnreadTextToggle );
// toggle for total column
mTotalColumnToggle = new KToggleAction( i18n("View->", "&Total Column"), 0, this,
SLOT(slotToggleTotalColumn()),
actionCollection(), "view_columns_total" );
mTotalColumnToggle->setToolTip( i18n("Toggle display of column showing the "
"total number of messages in folders.") );
mSizeColumnToggle = new KToggleAction( i18n("View->", "&Size Column"), 0, this,
SLOT(slotToggleSizeColumn()),
actionCollection(), "view_columns_size" );
mSizeColumnToggle->setToolTip( i18n("Toggle display of column showing the "
"total size of messages in folders.") );
(void)new KAction( KGuiItem( i18n("View->","&Expand Thread"), QString::null,
i18n("Expand the current thread") ),
Key_Period, this,
SLOT(slotExpandThread()),
actionCollection(), "expand_thread" );
(void)new KAction( KGuiItem( i18n("View->","&Collapse Thread"), QString::null,
i18n("Collapse the current thread") ),
Key_Comma, this,
SLOT(slotCollapseThread()),
actionCollection(), "collapse_thread" );
(void)new KAction( KGuiItem( i18n("View->","Ex&pand All Threads"), QString::null,
i18n("Expand all threads in the current folder") ),
CTRL+Key_Period, this,
SLOT(slotExpandAllThreads()),
actionCollection(), "expand_all_threads" );
(void)new KAction( KGuiItem( i18n("View->","C&ollapse All Threads"), QString::null,
i18n("Collapse all threads in the current folder") ),
CTRL+Key_Comma, this,
SLOT(slotCollapseAllThreads()),
actionCollection(), "collapse_all_threads" );
mViewSourceAction = new KAction( i18n("&View Source"), Key_V, this,
SLOT(slotShowMsgSrc()), actionCollection(),
"view_source" );
KAction* dukeOfMonmoth = new KAction( i18n("&Display Message"), Key_Return, this,
SLOT( slotDisplayCurrentMessage() ), actionCollection(),
"display_message" );
dukeOfMonmoth->plugAccel( actionCollection()->kaccel() );
//----- Go Menu
new KAction( KGuiItem( i18n("&Next Message"), QString::null,
i18n("Go to the next message") ),
"N;Right", this, SLOT(slotNextMessage()),
actionCollection(), "go_next_message" );
new KAction( KGuiItem( i18n("Next &Unread Message"),
QApplication::reverseLayout() ? "previous" : "next",
i18n("Go to the next unread message") ),
Key_Plus, this, SLOT(slotNextUnreadMessage()),
actionCollection(), "go_next_unread_message" );
/* ### needs better support from folders:
new KAction( KGuiItem( i18n("Next &Important Message"), QString::null,
i18n("Go to the next important message") ),
0, this, SLOT(slotNextImportantMessage()),
actionCollection(), "go_next_important_message" );
*/
new KAction( KGuiItem( i18n("&Previous Message"), QString::null,
i18n("Go to the previous message") ),
"P;Left", this, SLOT(slotPrevMessage()),
actionCollection(), "go_prev_message" );
new KAction( KGuiItem( i18n("Previous Unread &Message"),
QApplication::reverseLayout() ? "next" : "previous",
i18n("Go to the previous unread message") ),
Key_Minus, this, SLOT(slotPrevUnreadMessage()),
actionCollection(), "go_prev_unread_message" );
/* needs better support from folders:
new KAction( KGuiItem( i18n("Previous I&mportant Message"), QString::null,
i18n("Go to the previous important message") ),
0, this, SLOT(slotPrevImportantMessage()),
actionCollection(), "go_prev_important_message" );
*/
KAction *action =
new KAction( KGuiItem( i18n("Next Unread &Folder"), QString::null,
i18n("Go to the next folder with unread messages") ),
ALT+Key_Plus, this, SLOT(slotNextUnreadFolder()),
actionCollection(), "go_next_unread_folder" );
KShortcut shortcut = action->shortcut();
shortcut.append( KKey( CTRL+Key_Plus ) );
action->setShortcut( shortcut );
action =
new KAction( KGuiItem( i18n("Previous Unread F&older"), QString::null,
i18n("Go to the previous folder with unread messages") ),
ALT+Key_Minus, this, SLOT(slotPrevUnreadFolder()),
actionCollection(), "go_prev_unread_folder" );
shortcut = action->shortcut();
shortcut.append( KKey( CTRL+Key_Minus ) );
action->setShortcut( shortcut );
new KAction( KGuiItem( i18n("Go->","Next Unread &Text"), QString::null,
i18n("Go to the next unread text"),
i18n("Scroll down current message. "
"If at end of current message, "
"go to next unread message.") ),
Key_Space, this, SLOT(slotReadOn()),
actionCollection(), "go_next_unread_text" );
//----- Settings Menu
(void) new KAction( i18n("Configure &Filters..."), 0, this,
SLOT(slotFilter()), actionCollection(), "filter" );
(void) new KAction( i18n("Configure &POP Filters..."), 0, this,
SLOT(slotPopFilter()), actionCollection(), "popFilter" );
(void) new KAction( i18n("Manage &Sieve Scripts..."), 0, this,
SLOT(slotManageSieveScripts()), actionCollection(), "sieveFilters" );
(void) new KAction( KGuiItem( i18n("KMail &Introduction"), 0,
i18n("Display KMail's Welcome Page") ),
0, this, SLOT(slotIntro()),
actionCollection(), "help_kmail_welcomepage" );
// ----- Standard Actions
// KStdAction::configureNotifications(this, SLOT(slotEditNotifications()), actionCollection());
(void) new KAction( i18n("Configure &Notifications..."),
"knotify", 0, this,
SLOT(slotEditNotifications()), actionCollection(),
"kmail_configure_notifications" );
// KStdAction::preferences(this, SLOT(slotSettings()), actionCollection());
(void) new KAction( i18n("&Configure KMail..."),
"configure", 0, kmkernel,
SLOT(slotShowConfigurationDialog()), actionCollection(),
"kmail_configure_kmail" );
KStdAction::undo(this, SLOT(slotUndo()), actionCollection(), "kmail_undo");
KStdAction::tipOfDay( this, SLOT( slotShowTip() ), actionCollection() );
menutimer = new QTimer( this, "menutimer" );
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::setupForwardingActionsList()
{
QPtrList<KAction> mForwardActionList;
if ( GlobalSettings::self()->forwardingInlineByDefault() ) {
mGUIClient->unplugActionList( "forward_action_list" );
mForwardActionList.append( mForwardInlineAction );
mForwardActionList.append( mForwardAttachedAction );
mForwardActionList.append( mForwardDigestAction );
mForwardActionList.append( mRedirectAction );
mGUIClient->plugActionList( "forward_action_list", mForwardActionList );
} else {
mGUIClient->unplugActionList( "forward_action_list" );
mForwardActionList.append( mForwardAttachedAction );
mForwardActionList.append( mForwardInlineAction );
mForwardActionList.append( mForwardDigestAction );
mForwardActionList.append( mRedirectAction );
mGUIClient->plugActionList( "forward_action_list", mForwardActionList );
}
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotEditNotifications()
{
if(kmkernel->xmlGuiInstance())
KNotifyDialog::configure(this, 0, kmkernel->xmlGuiInstance()->aboutData());
else
KNotifyDialog::configure(this);
}
void KMMainWidget::slotEditKeys()
{
KKeyDialog::configure( actionCollection(),
true /*allow one-letter shortcuts*/
);
}
//-----------------------------------------------------------------------------
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( int menuId )
{
if (mMenuToFolder[menuId])
mHeaders->moveMsgToFolder( mMenuToFolder[menuId] );
}
//-----------------------------------------------------------------------------
void KMMainWidget::copySelectedToFolder(int menuId )
{
if (mMenuToFolder[menuId])
mHeaders->copyMsgToFolder( mMenuToFolder[menuId] );
}
//-----------------------------------------------------------------------------
void KMMainWidget::updateMessageMenu()
{
mMenuToFolder.clear();
folderTree()->folderToPopupMenu( KMFolderTree::MoveMessage, this,
&mMenuToFolder, mMoveActionMenu->popupMenu() );
folderTree()->folderToPopupMenu( KMFolderTree::CopyMessage, this,
&mMenuToFolder, mCopyActionMenu->popupMenu() );
updateMessageActions();
}
void KMMainWidget::startUpdateMessageActionsTimer()
{
menutimer->stop();
menutimer->start( 20, true );
}
void KMMainWidget::updateMessageActions()
{
int count = 0;
QPtrList<QListViewItem> selectedItems;
if ( mFolder ) {
for (QListViewItem *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;
for ( QPtrListIterator<QListViewItem> it( selectedItems ) ;
it.current() ; ++ it ) {
QListViewItem * item = *it;
if ( item->parent()==0 && item->childCount()==0 ) {
allSelectedInCommonThread = false;
break;
}
}
}
else if ( mHeaders->isThreaded() && count == 1 ) {
allSelectedInCommonThread = true;
}
QListViewItem *curItemParent = mHeaders->currentItem();
bool parent_thread = 0;
if ( curItemParent && curItemParent->firstChild() != 0 ) parent_thread = 1;
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 );
mToggleThreadFlagAction->setEnabled( thread_actions && flags_available );
mTrashThreadAction->setEnabled( thread_actions && !mFolder->isReadOnly() );
mDeleteThreadAction->setEnabled( thread_actions && !mFolder->isReadOnly() );
if (mFolder && mHeaders && mHeaders->currentMsg()) {
if (thread_actions) {
mToggleThreadTodoAction->setChecked(mHeaders->currentMsg()->isTodo());
mToggleThreadFlagAction->setChecked(mHeaders->currentMsg()->isImportant());
mWatchThreadAction->setChecked( mHeaders->currentMsg()->isWatched());
mIgnoreThreadAction->setChecked( mHeaders->currentMsg()->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 );
mForwardInlineAction->setEnabled( mass_actions );
mForwardAttachedAction->setEnabled( mass_actions );
mForwardDigestAction->setEnabled( count > 1 || parent_thread );
forwardMenu()->setEnabled( mass_actions );
bool single_actions = count == 1;
mUseAction->setEnabled( single_actions &&
kmkernel->folderIsTemplates( mFolder ) );
filterMenu()->setEnabled( single_actions );
redirectAction()->setEnabled( single_actions );
printAction()->setEnabled( single_actions );
viewSourceAction()->setEnabled( single_actions );
mSendAgainAction->setEnabled( single_actions
&& ( mHeaders->currentMsg() && mHeaders->currentMsg()->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 );
}
#ifdef MALLOC_DEBUG
static QString fmt(long n) {
char buf[32];
if(n > 1024*1024*1024)
sprintf(buf, "%0.2f GB", ((double)n)/1024.0/1024.0/1024.0);
else if(n > 1024*1024)
sprintf(buf, "%0.2f MB", ((double)n)/1024.0/1024.0);
else if(n > 1024)
sprintf(buf, "%0.2f KB", ((double)n)/1024.0);
else
sprintf(buf, "%ld Byte", n);
return QString(buf);
}
#endif
void KMMainWidget::slotMemInfo() {
#ifdef MALLOC_DEBUG
struct mallinfo mi;
mi = mallinfo();
QString s = QString("\nMALLOC - Info\n\n"
"Number of mmapped regions : %1\n"
"Memory allocated in use : %2\n"
"Memory allocated, not used: %3\n"
"Memory total allocated : %4\n"
"Max. freeable memory : %5\n")
.arg(mi.hblks).arg(fmt(mi.uordblks)).arg(fmt(mi.fordblks))
.arg(fmt(mi.arena)).arg(fmt(mi.keepcost));
KMessageBox::information(0, s, "Malloc information", s);
#endif
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotIntro()
{
if ( !mMsgView ) return;
mMsgView->clear( true );
// hide widgets that are in the way:
if ( mSearchAndHeaders && mHeaders && mLongFolderList )
mSearchAndHeaders->hide();
mMsgView->displayAboutPage();
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() ));
// plug shortcut filter actions now
initializeFilterActions();
// plug folder shortcut actions
initializeFolderShortcutActions();
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::null, true );
}
//-----------------------------------------------------------------------------
void KMMainWidget::slotChangeCaption(QListViewItem * i)
{
if ( !i ) return;
// set the caption to the current full path
QStringList names;
for ( QListViewItem * 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, QValueList<int> > idMD5s;
QValueList<int> redundantIds;
QValueList<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, QValueList<int> >::Iterator it;
for ( it = idMD5s.begin(); it != idMD5s.end() ; ++it ) {
QValueList<int>::Iterator jt;
bool finished = false;
for ( jt = (*it).begin(); jt != (*it).end() && !finished; ++jt )
if (!((*mFolder)[*jt]->isUnread())) {
(*it).remove( jt );
(*it).prepend( *jt );
finished = true;
}
for ( jt = (*it).begin(), ++jt; jt != (*it).end(); ++jt )
redundantIds.append( *jt );
}
qHeapSort( 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 = i18n("Removed %n duplicate message.",
"Removed %n 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" );
mFilterTBarActions.clear();
}
mApplyFilterActionsMenu->popupMenu()->clear();
if ( !mFilterMenuActions.isEmpty() ) {
if ( mGUIClient->factory() )
mGUIClient->unplugActionList( "menu_filter_actions" );
mFilterMenuActions.clear();
}
mFilterCommands.clear();
}
//-----------------------------------------------------------------------------
void KMMainWidget::initializeFolderShortcutActions()
{
// If we are loaded as a part, this will be set to fals, since the part
// does xml loading. Temporarily set to true, in that case, so the
// accels are added to the collection as expected.
bool old = actionCollection()->isAutoConnectShortcuts();
actionCollection()->setAutoConnectShortcuts( true );
QValueList< QGuardedPtr< KMFolder > > folders = kmkernel->allFolders();
QValueList< QGuardedPtr< KMFolder > >::Iterator it = folders.begin();
while ( it != folders.end() ) {
KMFolder *folder = (*it);
++it;
slotShortcutChanged( folder ); // load the initial accel
}
actionCollection()->setAutoConnectShortcuts( old );
}
//-----------------------------------------------------------------------------
void KMMainWidget::initializeFilterActions()
{
QString filterName, normalizedName;
KMMetaFilterActionCommand *filterCommand;
KAction *filterAction = 0;
clearFilterActions();
mApplyAllFiltersAction->plug(mApplyFilterActionsMenu->popupMenu());
bool addedSeparator = false;
QValueListConstIterator<KMFilter*> it = kmkernel->filterMgr()->filters().constBegin();
for ( ;it != kmkernel->filterMgr()->filters().constEnd(); ++it ) {
if (!(*it)->isEmpty() && (*it)->configureShortcut()) {
filterName = QString("Filter %1").arg((*it)->name());
normalizedName = filterName.replace(" ", "_");
if (action(normalizedName.utf8()))
continue;
filterCommand = new KMMetaFilterActionCommand(*it, mHeaders, this);
mFilterCommands.append(filterCommand);
QString as = i18n("Filter %1").arg((*it)->name());
QString icon = (*it)->icon();
if ( icon.isEmpty() )
icon = "gear";
filterAction = new KAction(as, icon, (*it)->shortcut(), filterCommand,
SLOT(start()), actionCollection(),
normalizedName.local8Bit());
if(!addedSeparator) {
mApplyFilterActionsMenu->popupMenu()->insertSeparator();
addedSeparator = !addedSeparator;
mFilterMenuActions.append( new KActionSeparator());
}
filterAction->plug( mApplyFilterActionsMenu->popupMenu() );
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 )
{
mFolderShortcutCommands.remove( 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() == "cachedimap" ) {
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( i18n("&Troubleshoot IMAP Cache..."), "wizard", 0,
this, SLOT(slotTroubleshootFolder()), actionCollection(), "troubleshoot_folder" );
if ( setState )
updateFolderMenu(); // set initial state of the action
} else {
delete mTroubleshootFolderAction ;
mTroubleshootFolderAction = 0;
}
if ( factory )
factory->addClient( mGUIClient );
}
bool KMMainWidget::shortcutIsValid( const KShortcut &sc ) const
{
KActionPtrList actions = actionCollection()->actions();
KActionPtrList::Iterator it( actions.begin() );
for ( ; it != actions.end(); it++ ) {
if ( (*it)->shortcut() == sc ) return false;
}
return true;
}
void KMMainWidget::slotShortcutChanged( KMFolder *folder )
{
// remove the old one, autodelete
mFolderShortcutCommands.remove( folder->idString() );
if ( folder->shortcut().isNull() )
return;
FolderShortcutCommand *c = new FolderShortcutCommand( this, folder );
mFolderShortcutCommands.insert( folder->idString(), c );
QString actionlabel = QString( "FolderShortcut %1").arg( folder->prettyURL() );
QString actionname = QString( "FolderShortcut %1").arg( folder->idString() );
QString normalizedName = actionname.replace(" ", "_");
KAction* action =
new KAction(actionlabel, folder->shortcut(), c, SLOT(start()),
actionCollection(), normalizedName.local8Bit());
action->setIcon( folder->unreadIconPath() );
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 "DestruciveClose"
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 "DestruciveClose"
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() );
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
kdDebug(5006) << "deleting systray" << endl;
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 )
{
actionCollection()->kaccel()->setEnabled( enabled );
}
//-----------------------------------------------------------------------------
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::setupFolderView()
{
if ( GlobalSettings::self()->enableFavoriteFolderView() ) {
mFolderView = mFolderViewSplitter;
mSearchAndTree->reparent( mFolderViewSplitter, 0, QPoint( 0, 0 ) );
mFolderViewSplitter->show();
mFavoriteFolderView->show();
} else {
mFolderView = mSearchAndTree;
mFolderViewSplitter->hide();
mFavoriteFolderView->hide();
}
mFolderView->reparent( mFolderViewParent, 0, QPoint( 0, 0 ) );
mFolderViewParent->moveToFirst( mFolderView );
mSearchAndTree->show();
}
void KMMainWidget::slotRequestFullSearchFromQuickSearch()
{
slotSearch();
#ifdef HAVE_INDEXLIB
return;
#endif
assert( mSearchWin );
KMSearchPattern pattern;
pattern.append( KMSearchRule::createInstance( "<message>", KMSearchRule::FuncContains, mQuickSearchLine->currentSearchTerm() ) );
int status = mQuickSearchLine->currentStatus();
if ( status != 0 ) {
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->setPaletteBackgroundColor( Qt::yellow );
mVacationScriptIndicator->setCursor( QCursor( Qt::PointingHandCursor ) );
mVacationScriptIndicator->show();
} else {
mVacationScriptIndicator->hide();
}
}