diff options
Diffstat (limited to 'kmail/kmcomposewin.cpp')
-rw-r--r-- | kmail/kmcomposewin.cpp | 5233 |
1 files changed, 5233 insertions, 0 deletions
diff --git a/kmail/kmcomposewin.cpp b/kmail/kmcomposewin.cpp new file mode 100644 index 000000000..8e9eb6b1e --- /dev/null +++ b/kmail/kmcomposewin.cpp @@ -0,0 +1,5233 @@ +// -*- mode: C++; c-file-style: "gnu" -*- +// kmcomposewin.cpp +// Author: Markus Wuebben <markus.wuebben@kde.org> +// This code is published under the GPL. + +#undef GrayScale +#undef Color +#include <config.h> + +#define REALLY_WANT_KMCOMPOSEWIN_H +#include "kmcomposewin.h" +#undef REALLY_WANT_KMCOMPOSEWIN_H + +#include "kmedit.h" +#include "kmlineeditspell.h" +#include "kmatmlistview.h" + +#include "kmmainwin.h" +#include "kmreadermainwin.h" +#include "messagesender.h" +#include "kmmsgpartdlg.h" +#include <kpgpblock.h> +#include <kaddrbook.h> +#include "kmaddrbook.h" +#include "kmmsgdict.h" +#include "kmfolderimap.h" +#include "kmfoldermgr.h" +#include "kmfoldercombobox.h" +#include "kmtransport.h" +#include "kmcommands.h" +#include "kcursorsaver.h" +#include "partNode.h" +#include "encodingdetector.h" +#include "attachmentlistview.h" +#include "transportmanager.h" +using KMail::AttachmentListView; +#include "dictionarycombobox.h" +using KMail::DictionaryComboBox; +#include "addressesdialog.h" +using KPIM::AddressesDialog; +#include "addresseeemailselection.h" +using KPIM::AddresseeEmailSelection; +using KPIM::AddresseeSelectorDialog; +#include <maillistdrag.h> +using KPIM::MailListDrag; +#include "recentaddresses.h" +using KRecentAddress::RecentAddresses; +#include "kleo_util.h" +#include "stl_util.h" +#include "recipientseditor.h" +#include "editorwatcher.h" + +#include "attachmentcollector.h" +#include "objecttreeparser.h" + +#include "kmfoldermaildir.h" + +#include <libkpimidentities/identitymanager.h> +#include <libkpimidentities/identitycombo.h> +#include <libkpimidentities/identity.h> +#include <libkdepim/kfileio.h> +#include <libemailfunctions/email.h> +#include <kleo/cryptobackendfactory.h> +#include <kleo/exportjob.h> +#include <kleo/specialjob.h> +#include <ui/progressdialog.h> +#include <ui/keyselectiondialog.h> + +#include <gpgmepp/context.h> +#include <gpgmepp/key.h> + +#include <kabc/vcardconverter.h> +#include <libkdepim/kvcarddrag.h> +#include <kio/netaccess.h> + +#include "klistboxdialog.h" + +#include "messagecomposer.h" +#include "chiasmuskeyselector.h" + +#include <kcharsets.h> +#include <kcompletionbox.h> +#include <kcursor.h> +#include <kcombobox.h> +#include <kstdaccel.h> +#include <kpopupmenu.h> +#include <kedittoolbar.h> +#include <kkeydialog.h> +#include <kdebug.h> +#include <kfiledialog.h> +#include <kwin.h> +#include <kinputdialog.h> +#include <kmessagebox.h> +#include <kurldrag.h> +#include <kio/scheduler.h> +#include <ktempfile.h> +#include <klocale.h> +#include <kapplication.h> +#include <kstatusbar.h> +#include <kaction.h> +#include <kstdaction.h> +#include <kdirwatch.h> +#include <kstdguiitem.h> +#include <kiconloader.h> +#include <kpushbutton.h> +#include <kuserprofile.h> +#include <krun.h> +#include <ktempdir.h> +#include <kstandarddirs.h> +//#include <keditlistbox.h> +#include "globalsettings.h" +#include "replyphrases.h" + +#include <kspell.h> +#include <kspelldlg.h> +#include <spellingfilter.h> +#include <ksyntaxhighlighter.h> +#include <kcolordialog.h> +#include <kzip.h> +#include <ksavefile.h> + +#include <qtabdialog.h> +#include <qregexp.h> +#include <qbuffer.h> +#include <qtooltip.h> +#include <qtextcodec.h> +#include <qheader.h> +#include <qwhatsthis.h> +#include <qfontdatabase.h> + +#include <mimelib/mimepp.h> + +#include <algorithm> +#include <memory> + +#include <sys/stat.h> +#include <sys/types.h> +#include <stdlib.h> +#include <unistd.h> +#include <errno.h> +#include <fcntl.h> +#include <assert.h> + +#include "kmcomposewin.moc" + +#include "snippetwidget.h" + +KMail::Composer * KMail::makeComposer( KMMessage * msg, uint identitiy ) { + return KMComposeWin::create( msg, identitiy ); +} + +KMail::Composer * KMComposeWin::create( KMMessage * msg, uint identitiy ) { + return new KMComposeWin( msg, identitiy ); +} + +//----------------------------------------------------------------------------- +KMComposeWin::KMComposeWin( KMMessage *aMsg, uint id ) + : MailComposerIface(), KMail::Composer( "kmail-composer#" ), + mSpellCheckInProgress( false ), + mDone( false ), + mAtmModified( false ), + mMsg( 0 ), + mAttachMenu( 0 ), + mSigningAndEncryptionExplicitlyDisabled( false ), + mFolder( 0 ), + mUseHTMLEditor( false ), + mId( id ), + mAttachPK( 0 ), mAttachMPK( 0 ), + mAttachRemoveAction( 0 ), mAttachSaveAction( 0 ), mAttachPropertiesAction( 0 ), + mAppendSignatureAction( 0 ), mPrependSignatureAction( 0 ), mInsertSignatureAction( 0 ), + mSignAction( 0 ), mEncryptAction( 0 ), mRequestMDNAction( 0 ), + mUrgentAction( 0 ), mAllFieldsAction( 0 ), mFromAction( 0 ), + mReplyToAction( 0 ), mToAction( 0 ), mCcAction( 0 ), mBccAction( 0 ), + mSubjectAction( 0 ), + mIdentityAction( 0 ), mTransportAction( 0 ), mFccAction( 0 ), + mWordWrapAction( 0 ), mFixedFontAction( 0 ), mAutoSpellCheckingAction( 0 ), + mDictionaryAction( 0 ), mSnippetAction( 0 ), + mEncodingAction( 0 ), + mCryptoModuleAction( 0 ), + mEncryptChiasmusAction( 0 ), + mEncryptWithChiasmus( false ), + mComposer( 0 ), + mLabelWidth( 0 ), + mAutoSaveTimer( 0 ), mLastAutoSaveErrno( 0 ), + mSignatureStateIndicator( 0 ), mEncryptionStateIndicator( 0 ), + mPreserveUserCursorPosition( false ) +{ + mClassicalRecipients = GlobalSettings::self()->recipientsEditorType() == + GlobalSettings::EnumRecipientsEditorType::Classic; + + mSubjectTextWasSpellChecked = false; + if (kmkernel->xmlGuiInstance()) + setInstance( kmkernel->xmlGuiInstance() ); + mMainWidget = new QWidget(this); + // splitter between the headers area and the actual editor + mHeadersToEditorSplitter = new QSplitter( Qt::Vertical, mMainWidget, "mHeadersToEditorSplitter" ); + mHeadersToEditorSplitter->setChildrenCollapsible( false ); + mHeadersArea = new QWidget( mHeadersToEditorSplitter ); + mHeadersArea->setSizePolicy( mHeadersToEditorSplitter->sizePolicy().horData(), QSizePolicy::Maximum ); + QVBoxLayout *v = new QVBoxLayout( mMainWidget ); + v->addWidget( mHeadersToEditorSplitter ); + mIdentity = new KPIM::IdentityCombo(kmkernel->identityManager(), mHeadersArea); + mDictionaryCombo = new DictionaryComboBox( mHeadersArea ); + mFcc = new KMFolderComboBox(mHeadersArea); + mFcc->showOutboxFolder( false ); + mTransport = new QComboBox(true, mHeadersArea); + mEdtFrom = new KMLineEdit(false,mHeadersArea, "fromLine"); + + mEdtReplyTo = new KMLineEdit(true,mHeadersArea, "replyToLine"); + mLblReplyTo = new QLabel(mHeadersArea); + mBtnReplyTo = new QPushButton("...",mHeadersArea); + mBtnReplyTo->setFocusPolicy(QWidget::NoFocus); + connect(mBtnReplyTo,SIGNAL(clicked()),SLOT(slotAddrBookReplyTo())); + connect(mEdtReplyTo,SIGNAL(completionModeChanged(KGlobalSettings::Completion)), + SLOT(slotCompletionModeChanged(KGlobalSettings::Completion))); + + if ( mClassicalRecipients ) { + mRecipientsEditor = 0; + + mEdtTo = new KMLineEdit(true,mHeadersArea, "toLine"); + mEdtCc = new KMLineEdit(true,mHeadersArea, "ccLine"); + mEdtBcc = new KMLineEdit(true,mHeadersArea, "bccLine"); + + mLblTo = new QLabel(mHeadersArea); + mLblCc = new QLabel(mHeadersArea); + mLblBcc = new QLabel(mHeadersArea); + + mBtnTo = new QPushButton("...",mHeadersArea); + mBtnCc = new QPushButton("...",mHeadersArea); + mBtnBcc = new QPushButton("...",mHeadersArea); + //mBtnFrom = new QPushButton("...",mHeadersArea); + + QString tip = i18n("Select email address(es)"); + QToolTip::add( mBtnTo, tip ); + QToolTip::add( mBtnCc, tip ); + QToolTip::add( mBtnBcc, tip ); + QToolTip::add( mBtnReplyTo, tip ); + + mBtnTo->setFocusPolicy(QWidget::NoFocus); + mBtnCc->setFocusPolicy(QWidget::NoFocus); + mBtnBcc->setFocusPolicy(QWidget::NoFocus); + //mBtnFrom->setFocusPolicy(QWidget::NoFocus); + + connect(mBtnTo,SIGNAL(clicked()),SLOT(slotAddrBookTo())); + connect(mBtnCc,SIGNAL(clicked()),SLOT(slotAddrBookTo())); + connect(mBtnBcc,SIGNAL(clicked()),SLOT(slotAddrBookTo())); + //connect(mBtnFrom,SIGNAL(clicked()),SLOT(slotAddrBookFrom())); + + connect(mEdtTo,SIGNAL(completionModeChanged(KGlobalSettings::Completion)), + SLOT(slotCompletionModeChanged(KGlobalSettings::Completion))); + connect(mEdtCc,SIGNAL(completionModeChanged(KGlobalSettings::Completion)), + SLOT(slotCompletionModeChanged(KGlobalSettings::Completion))); + connect(mEdtBcc,SIGNAL(completionModeChanged(KGlobalSettings::Completion)), + SLOT(slotCompletionModeChanged(KGlobalSettings::Completion))); + + mEdtTo->setFocus(); + } else { + mEdtTo = 0; + mEdtCc = 0; + mEdtBcc = 0; + + mLblTo = 0; + mLblCc = 0; + mLblBcc = 0; + + mBtnTo = 0; + mBtnCc = 0; + mBtnBcc = 0; + //mBtnFrom = 0; + + mRecipientsEditor = new RecipientsEditor( mHeadersArea ); + connect( mRecipientsEditor, + SIGNAL( completionModeChanged( KGlobalSettings::Completion ) ), + SLOT( slotCompletionModeChanged( KGlobalSettings::Completion ) ) ); + connect( mRecipientsEditor, SIGNAL(sizeHintChanged()), SLOT(recipientEditorSizeHintChanged()) ); + + mRecipientsEditor->setFocus(); + } + mEdtSubject = new KMLineEditSpell(false,mHeadersArea, "subjectLine"); + mLblIdentity = new QLabel(mHeadersArea); + mDictionaryLabel = new QLabel( mHeadersArea ); + mLblFcc = new QLabel(mHeadersArea); + mLblTransport = new QLabel(mHeadersArea); + mLblFrom = new QLabel(mHeadersArea); + mLblSubject = new QLabel(mHeadersArea); + QString sticky = i18n("Sticky"); + mBtnIdentity = new QCheckBox(sticky,mHeadersArea); + mBtnFcc = new QCheckBox(sticky,mHeadersArea); + mBtnTransport = new QCheckBox(sticky,mHeadersArea); + + //setWFlags( WType_TopLevel | WStyle_Dialog ); + mHtmlMarkup = GlobalSettings::self()->useHtmlMarkup(); + mShowHeaders = GlobalSettings::self()->headers(); + mDone = false; + mGrid = 0; + mAtmListView = 0; + mAtmList.setAutoDelete(true); + mAtmTempList.setAutoDelete(true); + mAtmModified = false; + mAutoDeleteMsg = false; + mFolder = 0; + mAutoCharset = true; + mFixedFontAction = 0; + mTempDir = 0; + // the attachment view is separated from the editor by a splitter + mSplitter = new QSplitter( Qt::Vertical, mHeadersToEditorSplitter, "mSplitter" ); + mSplitter->setChildrenCollapsible( false ); + mSnippetSplitter = new QSplitter( Qt::Horizontal, mSplitter, "mSnippetSplitter"); + mSnippetSplitter->setChildrenCollapsible( false ); + + QWidget *editorAndCryptoStateIndicators = new QWidget( mSnippetSplitter ); + QVBoxLayout *vbox = new QVBoxLayout( editorAndCryptoStateIndicators ); + QHBoxLayout *hbox = new QHBoxLayout( vbox ); + { + mSignatureStateIndicator = new QLabel( editorAndCryptoStateIndicators ); + mSignatureStateIndicator->setAlignment( Qt::AlignHCenter ); + hbox->addWidget( mSignatureStateIndicator ); + + KConfigGroup reader( KMKernel::config(), "Reader" ); + QPalette p( mSignatureStateIndicator->palette() ); + + QColor defaultSignedColor( 0x40, 0xFF, 0x40 ); // light green // pgp ok, trusted key + QColor defaultEncryptedColor( 0x00, 0x80, 0xFF ); // light blue // pgp encrypted + p.setColor( QColorGroup::Background, reader.readColorEntry( "PGPMessageOkKeyOk", &defaultSignedColor ) ); + mSignatureStateIndicator->setPalette( p ); + + mEncryptionStateIndicator = new QLabel( editorAndCryptoStateIndicators ); + mEncryptionStateIndicator->setAlignment( Qt::AlignHCenter ); + hbox->addWidget( mEncryptionStateIndicator ); + p.setColor( QColorGroup::Background, reader.readColorEntry( "PGPMessageEncr" , &defaultEncryptedColor ) ); + mEncryptionStateIndicator->setPalette( p ); + } + + mEditor = new KMEdit( editorAndCryptoStateIndicators, this, mDictionaryCombo->spellConfig() ); + vbox->addWidget( mEditor ); + + mSnippetWidget = new SnippetWidget( mEditor, actionCollection(), mSnippetSplitter ); + mSnippetWidget->setShown( GlobalSettings::self()->showSnippetManager() ); + + // mSplitter->moveToFirst( editorAndCryptoStateIndicators ); + mSplitter->setOpaqueResize( true ); + + mEditor->initializeAutoSpellChecking(); + mEditor->setTextFormat(Qt::PlainText); + mEditor->setAcceptDrops( true ); + + QWhatsThis::add( mBtnIdentity, + GlobalSettings::self()->stickyIdentityItem()->whatsThis() ); + QWhatsThis::add( mBtnFcc, + GlobalSettings::self()->stickyFccItem()->whatsThis() ); + QWhatsThis::add( mBtnTransport, + GlobalSettings::self()->stickyTransportItem()->whatsThis() ); + + mSpellCheckInProgress=false; + + setCaption( i18n("Composer") ); + setMinimumSize(200,200); + + mBtnIdentity->setFocusPolicy(QWidget::NoFocus); + mBtnFcc->setFocusPolicy(QWidget::NoFocus); + mBtnTransport->setFocusPolicy(QWidget::NoFocus); + + mAtmListView = new AttachmentListView( this, mSplitter, + "attachment list view" ); + mAtmListView->setSelectionMode( QListView::Extended ); + mAtmListView->addColumn( i18n("Name"), 200 ); + mAtmListView->addColumn( i18n("Size"), 80 ); + mAtmListView->addColumn( i18n("Encoding"), 120 ); + int atmColType = mAtmListView->addColumn( i18n("Type"), 120 ); + // Stretch "Type". + mAtmListView->header()->setStretchEnabled( true, atmColType ); + mAtmEncryptColWidth = 80; + mAtmSignColWidth = 80; + mAtmCompressColWidth = 100; + mAtmColCompress = mAtmListView->addColumn( i18n("Compress"), + mAtmCompressColWidth ); + mAtmColEncrypt = mAtmListView->addColumn( i18n("Encrypt"), + mAtmEncryptColWidth ); + mAtmColSign = mAtmListView->addColumn( i18n("Sign"), + mAtmSignColWidth ); + mAtmListView->setColumnWidth( mAtmColEncrypt, 0 ); + mAtmListView->setColumnWidth( mAtmColSign, 0 ); + mAtmListView->setAllColumnsShowFocus( true ); + + connect( mAtmListView, + SIGNAL( doubleClicked( QListViewItem* ) ), + SLOT( slotAttachEdit() ) ); + connect( mAtmListView, + SIGNAL( rightButtonPressed( QListViewItem*, const QPoint&, int ) ), + SLOT( slotAttachPopupMenu( QListViewItem*, const QPoint&, int ) ) ); + connect( mAtmListView, + SIGNAL( selectionChanged() ), + SLOT( slotUpdateAttachActions() ) ); + connect( mAtmListView, + SIGNAL( attachmentDeleted() ), + SLOT( slotAttachRemove() ) ); + connect( mAtmListView, + SIGNAL( dragStarted() ), + SLOT( slotAttachmentDragStarted() ) ); + mAttachMenu = 0; + + readConfig(); + setupStatusBar(); + setupActions(); + setupEditor(); + slotUpdateSignatureAndEncrypionStateIndicators(); + + applyMainWindowSettings(KMKernel::config(), "Composer"); + + connect( mEdtSubject, SIGNAL( subjectTextSpellChecked() ), + SLOT( slotSubjectTextSpellChecked() ) ); + connect(mEdtSubject,SIGNAL(textChanged(const QString&)), + SLOT(slotUpdWinTitle(const QString&))); + connect(mIdentity,SIGNAL(identityChanged(uint)), + SLOT(slotIdentityChanged(uint))); + connect( kmkernel->identityManager(), SIGNAL(changed(uint)), + SLOT(slotIdentityChanged(uint))); + + connect(mEdtFrom,SIGNAL(completionModeChanged(KGlobalSettings::Completion)), + SLOT(slotCompletionModeChanged(KGlobalSettings::Completion))); + connect(kmkernel->folderMgr(),SIGNAL(folderRemoved(KMFolder*)), + SLOT(slotFolderRemoved(KMFolder*))); + connect(kmkernel->imapFolderMgr(),SIGNAL(folderRemoved(KMFolder*)), + SLOT(slotFolderRemoved(KMFolder*))); + connect(kmkernel->dimapFolderMgr(),SIGNAL(folderRemoved(KMFolder*)), + SLOT(slotFolderRemoved(KMFolder*))); + connect( kmkernel, SIGNAL( configChanged() ), + this, SLOT( slotConfigChanged() ) ); + + connect (mEditor, SIGNAL (spellcheck_done(int)), + this, SLOT (slotSpellcheckDone (int))); + connect (mEditor, SIGNAL( attachPNGImageData(const QByteArray &) ), + this, SLOT ( slotAttachPNGImageData(const QByteArray &) ) ); + connect (mEditor, SIGNAL( focusChanged(bool) ), + this, SLOT (editorFocusChanged(bool)) ); + + mMainWidget->resize(480,510); + setCentralWidget(mMainWidget); + rethinkFields(); + + if ( !mClassicalRecipients ) { + // This is ugly, but if it isn't called the line edits in the recipients + // editor aren't wide enough until the first resize event comes. + rethinkFields(); + } + + if ( GlobalSettings::self()->useExternalEditor() ) { + mEditor->setUseExternalEditor(true); + mEditor->setExternalEditorPath( GlobalSettings::self()->externalEditor() ); + } + + initAutoSave(); + slotUpdateSignatureActions(); + mMsg = 0; + if (aMsg) + setMsg(aMsg); + fontChanged( mEditor->currentFont() ); // set toolbar buttons to correct values + + mDone = true; +} + +//----------------------------------------------------------------------------- +KMComposeWin::~KMComposeWin() +{ + writeConfig(); + if (mFolder && mMsg) + { + mAutoDeleteMsg = false; + mFolder->addMsg(mMsg); + // Ensure that the message is correctly and fully parsed + mFolder->unGetMsg( mFolder->count() - 1 ); + } + if (mAutoDeleteMsg) { + delete mMsg; + mMsg = 0; + } + QMap<KIO::Job*, atmLoadData>::Iterator it = mMapAtmLoadData.begin(); + while ( it != mMapAtmLoadData.end() ) + { + KIO::Job *job = it.key(); + mMapAtmLoadData.remove( it ); + job->kill(); + it = mMapAtmLoadData.begin(); + } + deleteAll( mComposedMessages ); + + for ( std::set<KTempDir*>::iterator it = mTempDirs.begin() ; it != mTempDirs.end() ; ++it ) { + delete *it; + } +} + +void KMComposeWin::setAutoDeleteWindow( bool f ) +{ + if ( f ) + setWFlags( getWFlags() | WDestructiveClose ); + else + setWFlags( getWFlags() & ~WDestructiveClose ); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::send(int how) +{ + switch (how) { + case 1: + slotSendNow(); + break; + default: + case 0: + // TODO: find out, what the default send method is and send it this way + case 2: + slotSendLater(); + break; + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::addAttachmentsAndSend(const KURL::List &urls, const QString &/*comment*/, int how) +{ + if (urls.isEmpty()) + { + send(how); + return; + } + mAttachFilesSend = how; + mAttachFilesPending = urls; + connect(this, SIGNAL(attachmentAdded(const KURL&, bool)), SLOT(slotAttachedFile(const KURL&))); + for( KURL::List::ConstIterator itr = urls.begin(); itr != urls.end(); ++itr ) { + if (!addAttach( *itr )) + mAttachFilesPending.remove(mAttachFilesPending.find(*itr)); // only remove one copy of the url + } + + if (mAttachFilesPending.isEmpty() && mAttachFilesSend == how) + { + send(mAttachFilesSend); + mAttachFilesSend = -1; + } +} + +void KMComposeWin::slotAttachedFile(const KURL &url) +{ + if (mAttachFilesPending.isEmpty()) + return; + mAttachFilesPending.remove(mAttachFilesPending.find(url)); // only remove one copy of url + if (mAttachFilesPending.isEmpty()) + { + send(mAttachFilesSend); + mAttachFilesSend = -1; + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::addAttachment(KURL url,QString /*comment*/) +{ + addAttach(url); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::addAttachment(const QString &name, + const QCString &/*cte*/, + const QByteArray &data, + const QCString &type, + const QCString &subType, + const QCString ¶mAttr, + const QString ¶mValue, + const QCString &contDisp) +{ + if (!data.isEmpty()) { + KMMessagePart *msgPart = new KMMessagePart; + msgPart->setName(name); + if( type == "message" && subType == "rfc822" ) { + msgPart->setMessageBody( data ); + } else { + QValueList<int> dummy; + msgPart->setBodyAndGuessCte(data, dummy, + kmkernel->msgSender()->sendQuotedPrintable()); + } + msgPart->setTypeStr(type); + msgPart->setSubtypeStr(subType); + msgPart->setParameter(paramAttr,paramValue); + msgPart->setContentDisposition(contDisp); + addAttach(msgPart); + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAttachPNGImageData(const QByteArray &image) +{ + bool ok; + + QString attName = KInputDialog::getText( "KMail", i18n("Name of the attachment:"), QString::null, &ok, this ); + if ( !ok ) + return; + + if ( !attName.lower().endsWith(".png") ) attName += ".png"; + + addAttachment( attName, "base64", image, "image", "png", QCString(), QString(), QCString() ); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::setBody(QString body) +{ + mEditor->setText(body); +} + +//----------------------------------------------------------------------------- +bool KMComposeWin::event(QEvent *e) +{ + if (e->type() == QEvent::ApplicationPaletteChange) + { + readColorConfig(); + } + return KMail::Composer::event(e); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::readColorConfig(void) +{ + if ( GlobalSettings::self()->useDefaultColors() ) { + mForeColor = QColor(kapp->palette().active().text()); + mBackColor = QColor(kapp->palette().active().base()); + } else { + mForeColor = GlobalSettings::self()->foregroundColor(); + mBackColor = GlobalSettings::self()->backgroundColor(); + } + + // Color setup + mPalette = kapp->palette(); + QColorGroup cgrp = mPalette.active(); + cgrp.setColor( QColorGroup::Base, mBackColor); + cgrp.setColor( QColorGroup::Text, mForeColor); + mPalette.setDisabled(cgrp); + mPalette.setActive(cgrp); + mPalette.setInactive(cgrp); + + mEdtFrom->setPalette(mPalette); + mEdtReplyTo->setPalette(mPalette); + if ( mClassicalRecipients ) { + mEdtTo->setPalette(mPalette); + mEdtCc->setPalette(mPalette); + mEdtBcc->setPalette(mPalette); + } + mEdtSubject->setPalette(mPalette); + mTransport->setPalette(mPalette); + mEditor->setPalette(mPalette); + mFcc->setPalette(mPalette); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::readConfig( bool reload /* = false */ ) +{ + mDefCharset = KMMessage::defaultCharset(); + mBtnIdentity->setChecked( GlobalSettings::self()->stickyIdentity() ); + if (mBtnIdentity->isChecked()) { + mId = (GlobalSettings::self()->previousIdentity()!=0) ? + GlobalSettings::self()->previousIdentity() : mId; + } + mBtnFcc->setChecked( GlobalSettings::self()->stickyFcc() ); + mBtnTransport->setChecked( GlobalSettings::self()->stickyTransport() ); + QStringList transportHistory = GlobalSettings::self()->transportHistory(); + QString currentTransport = GlobalSettings::self()->currentTransport(); + + mEdtFrom->setCompletionMode( (KGlobalSettings::Completion)GlobalSettings::self()->completionMode() ); + mEdtReplyTo->setCompletionMode( (KGlobalSettings::Completion)GlobalSettings::self()->completionMode() ); + if ( mClassicalRecipients ) { + mEdtTo->setCompletionMode( (KGlobalSettings::Completion)GlobalSettings::self()->completionMode() ); + mEdtCc->setCompletionMode( (KGlobalSettings::Completion)GlobalSettings::self()->completionMode() ); + mEdtBcc->setCompletionMode( (KGlobalSettings::Completion)GlobalSettings::self()->completionMode() ); + } + else + mRecipientsEditor->setCompletionMode( (KGlobalSettings::Completion)GlobalSettings::self()->completionMode() ); + + readColorConfig(); + + if ( GlobalSettings::self()->useDefaultFonts() ) { + mBodyFont = KGlobalSettings::generalFont(); + mFixedFont = KGlobalSettings::fixedFont(); + } else { + mBodyFont = GlobalSettings::self()->composerFont(); + mFixedFont = GlobalSettings::self()->fixedFont(); + } + + slotUpdateFont(); + mEdtFrom->setFont(mBodyFont); + mEdtReplyTo->setFont(mBodyFont); + if ( mClassicalRecipients ) { + mEdtTo->setFont(mBodyFont); + mEdtCc->setFont(mBodyFont); + mEdtBcc->setFont(mBodyFont); + } + mEdtSubject->setFont(mBodyFont); + + if ( !reload ) { + QSize siz = GlobalSettings::self()->composerSize(); + if (siz.width() < 200) siz.setWidth(200); + if (siz.height() < 200) siz.setHeight(200); + resize(siz); + + if ( !GlobalSettings::self()->snippetSplitterPosition().isEmpty() ) { + mSnippetSplitter->setSizes( GlobalSettings::self()->snippetSplitterPosition() ); + } else { + QValueList<int> defaults; + defaults << (int)(width() * 0.8) << (int)(width() * 0.2); + mSnippetSplitter->setSizes( defaults ); + } + } + + mIdentity->setCurrentIdentity( mId ); + + kdDebug(5006) << "KMComposeWin::readConfig. " << mIdentity->currentIdentityName() << endl; + const KPIM::Identity & ident = + kmkernel->identityManager()->identityForUoid( mIdentity->currentIdentity() ); + + mDictionaryCombo->setCurrentByDictionary( ident.dictionary() ); + + mTransport->clear(); + mTransport->insertStringList( KMTransportInfo::availableTransports() ); + while ( transportHistory.count() > (uint)GlobalSettings::self()->maxTransportEntries() ) + transportHistory.remove( transportHistory.last() ); + mTransport->insertStringList( transportHistory ); + mTransport->setCurrentText( GlobalSettings::self()->defaultTransport() ); + if ( mBtnTransport->isChecked() ) { + setTransport( currentTransport ); + } + + QString fccName = ""; + if ( mBtnFcc->isChecked() ) { + fccName = GlobalSettings::self()->previousFcc(); + } else if ( !ident.fcc().isEmpty() ) { + fccName = ident.fcc(); + } + + setFcc( fccName ); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::writeConfig(void) +{ + GlobalSettings::self()->setHeaders( mShowHeaders ); + GlobalSettings::self()->setStickyTransport( mBtnTransport->isChecked() ); + GlobalSettings::self()->setStickyIdentity( mBtnIdentity->isChecked() ); + GlobalSettings::self()->setStickyFcc( mBtnFcc->isChecked() ); + GlobalSettings::self()->setPreviousIdentity( mIdentity->currentIdentity() ); + GlobalSettings::self()->setCurrentTransport( mTransport->currentText() ); + GlobalSettings::self()->setPreviousFcc( mFcc->getFolder()->idString() ); + GlobalSettings::self()->setAutoSpellChecking( + mAutoSpellCheckingAction->isChecked() ); + QStringList transportHistory = GlobalSettings::self()->transportHistory(); + transportHistory.remove(mTransport->currentText()); + if (KMTransportInfo::availableTransports().findIndex(mTransport + ->currentText()) == -1) { + transportHistory.prepend(mTransport->currentText()); + } + GlobalSettings::self()->setTransportHistory( transportHistory ); + GlobalSettings::self()->setUseFixedFont( mFixedFontAction->isChecked() ); + GlobalSettings::self()->setUseHtmlMarkup( mHtmlMarkup ); + GlobalSettings::self()->setComposerSize( size() ); + GlobalSettings::self()->setShowSnippetManager( mSnippetAction->isChecked() ); + + KConfigGroupSaver saver( KMKernel::config(), "Geometry" ); + saveMainWindowSettings( KMKernel::config(), "Composer" ); + GlobalSettings::setSnippetSplitterPosition( mSnippetSplitter->sizes() ); + + // make sure config changes are written to disk, cf. bug 127538 + GlobalSettings::self()->writeConfig(); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::autoSaveMessage() +{ + kdDebug(5006) << k_funcinfo << endl; + if ( !mMsg || mComposer || mAutoSaveFilename.isEmpty() ) + return; + kdDebug(5006) << k_funcinfo << "autosaving message" << endl; + + if ( mAutoSaveTimer ) + mAutoSaveTimer->stop(); + + connect( this, SIGNAL( applyChangesDone( bool ) ), + this, SLOT( slotContinueAutoSave() ) ); + // This method is called when KMail crashed, so don't try signing/encryption + // and don't disable controls because it is also called from a timer and + // then the disabling is distracting. + applyChanges( true, true ); + + // Don't continue before the applyChanges is done! +} + +void KMComposeWin::slotContinueAutoSave() +{ + disconnect( this, SIGNAL( applyChangesDone( bool ) ), + this, SLOT( slotContinueAutoSave() ) ); + + // Ok, it's done now - continue dead letter saving + if ( mComposedMessages.isEmpty() ) { + kdDebug(5006) << "Composing the message failed." << endl; + return; + } + KMMessage *msg = mComposedMessages.first(); + if ( !msg ) // a bit of extra defensiveness + return; + + kdDebug(5006) << k_funcinfo << "opening autoSaveFile " << mAutoSaveFilename + << endl; + const QString filename = + KMKernel::localDataPath() + "autosave/cur/" + mAutoSaveFilename; + KSaveFile autoSaveFile( filename, 0600 ); + int status = autoSaveFile.status(); + kdDebug(5006) << k_funcinfo << "autoSaveFile.status() = " << status << endl; + if ( status == 0 ) { // no error + kdDebug(5006) << "autosaving message in " << filename << endl; + int fd = autoSaveFile.handle(); + const DwString& msgStr = msg->asDwString(); + if ( ::write( fd, msgStr.data(), msgStr.length() ) == -1 ) + status = errno; + } + if ( status == 0 ) { + kdDebug(5006) << k_funcinfo << "closing autoSaveFile" << endl; + autoSaveFile.close(); + mLastAutoSaveErrno = 0; + } + else { + kdDebug(5006) << k_funcinfo << "autosaving failed" << endl; + autoSaveFile.abort(); + if ( status != mLastAutoSaveErrno ) { + // don't show the same error message twice + KMessageBox::queuedMessageBox( 0, KMessageBox::Sorry, + i18n("Autosaving the message as %1 " + "failed.\n" + "Reason: %2" ) + .arg( filename, strerror( status ) ), + i18n("Autosaving Failed") ); + mLastAutoSaveErrno = status; + } + } + + if ( autoSaveInterval() > 0 ) + updateAutoSave(); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotView(void) +{ + if (!mDone) + return; // otherwise called from rethinkFields during the construction + // which is not the intended behavior + int id; + + //This sucks awfully, but no, I cannot get an activated(int id) from + // actionContainer() + if (!sender()->isA("KToggleAction")) + return; + KToggleAction *act = (KToggleAction *) sender(); + + if (act == mAllFieldsAction) + id = 0; + else if (act == mIdentityAction) + id = HDR_IDENTITY; + else if (act == mTransportAction) + id = HDR_TRANSPORT; + else if (act == mFromAction) + id = HDR_FROM; + else if (act == mReplyToAction) + id = HDR_REPLY_TO; + else if (act == mToAction) + id = HDR_TO; + else if (act == mCcAction) + id = HDR_CC; + else if (act == mBccAction) + id = HDR_BCC; + else if (act == mSubjectAction) + id = HDR_SUBJECT; + else if (act == mFccAction) + id = HDR_FCC; + else if ( act == mDictionaryAction ) + id = HDR_DICTIONARY; + else + { + id = 0; + kdDebug(5006) << "Something is wrong (Oh, yeah?)" << endl; + return; + } + + // sanders There's a bug here this logic doesn't work if no + // fields are shown and then show all fields is selected. + // Instead of all fields being shown none are. + if (!act->isChecked()) + { + // hide header + if (id > 0) mShowHeaders = mShowHeaders & ~id; + else mShowHeaders = abs(mShowHeaders); + } + else + { + // show header + if (id > 0) mShowHeaders |= id; + else mShowHeaders = -abs(mShowHeaders); + } + rethinkFields(true); +} + +int KMComposeWin::calcColumnWidth(int which, long allShowing, int width) +{ + if ( (allShowing & which) == 0 ) + return width; + + QLabel *w; + if ( which == HDR_IDENTITY ) + w = mLblIdentity; + else if ( which == HDR_DICTIONARY ) + w = mDictionaryLabel; + else if ( which == HDR_FCC ) + w = mLblFcc; + else if ( which == HDR_TRANSPORT ) + w = mLblTransport; + else if ( which == HDR_FROM ) + w = mLblFrom; + else if ( which == HDR_REPLY_TO ) + w = mLblReplyTo; + else if ( which == HDR_SUBJECT ) + w = mLblSubject; + else + return width; + + w->setBuddy( mEditor ); // set dummy so we don't calculate width of '&' for this label. + w->adjustSize(); + w->show(); + return QMAX( width, w->sizeHint().width() ); +} + +void KMComposeWin::rethinkFields(bool fromSlot) +{ + //This sucks even more but again no ids. sorry (sven) + int mask, row, numRows; + long showHeaders; + + if (mShowHeaders < 0) + showHeaders = HDR_ALL; + else + showHeaders = mShowHeaders; + + for (mask=1,mNumHeaders=0; mask<=showHeaders; mask<<=1) + if ((showHeaders&mask) != 0) mNumHeaders++; + + numRows = mNumHeaders + 1; + + delete mGrid; + + mGrid = new QGridLayout( mHeadersArea, numRows, 3, KDialogBase::marginHint()/2, KDialogBase::spacingHint()); + mGrid->setColStretch(0, 1); + mGrid->setColStretch(1, 100); + mGrid->setColStretch(2, 1); + mGrid->setRowStretch(mNumHeaders, 100); + + row = 0; + kdDebug(5006) << "KMComposeWin::rethinkFields" << endl; + if (mRecipientsEditor) + mLabelWidth = mRecipientsEditor->setFirstColumnWidth( 0 ); + mLabelWidth = calcColumnWidth( HDR_IDENTITY, showHeaders, mLabelWidth ); + mLabelWidth = calcColumnWidth( HDR_DICTIONARY, showHeaders, mLabelWidth ); + mLabelWidth = calcColumnWidth( HDR_FCC, showHeaders, mLabelWidth ); + mLabelWidth = calcColumnWidth( HDR_TRANSPORT, showHeaders, mLabelWidth ); + mLabelWidth = calcColumnWidth( HDR_FROM, showHeaders, mLabelWidth ); + mLabelWidth = calcColumnWidth( HDR_REPLY_TO, showHeaders, mLabelWidth ); + mLabelWidth = calcColumnWidth( HDR_SUBJECT, showHeaders, mLabelWidth ); + + if (!fromSlot) mAllFieldsAction->setChecked(showHeaders==HDR_ALL); + + if (!fromSlot) mIdentityAction->setChecked(abs(mShowHeaders)&HDR_IDENTITY); + rethinkHeaderLine(showHeaders,HDR_IDENTITY, row, i18n("&Identity:"), + mLblIdentity, mIdentity, mBtnIdentity); + + if (!fromSlot) mDictionaryAction->setChecked(abs(mShowHeaders)&HDR_DICTIONARY); + rethinkHeaderLine(showHeaders,HDR_DICTIONARY, row, i18n("&Dictionary:"), + mDictionaryLabel, mDictionaryCombo, 0 ); + + if (!fromSlot) mFccAction->setChecked(abs(mShowHeaders)&HDR_FCC); + rethinkHeaderLine(showHeaders,HDR_FCC, row, i18n("&Sent-Mail folder:"), + mLblFcc, mFcc, mBtnFcc); + + if (!fromSlot) mTransportAction->setChecked(abs(mShowHeaders)&HDR_TRANSPORT); + rethinkHeaderLine(showHeaders,HDR_TRANSPORT, row, i18n("&Mail transport:"), + mLblTransport, mTransport, mBtnTransport); + + if (!fromSlot) mFromAction->setChecked(abs(mShowHeaders)&HDR_FROM); + rethinkHeaderLine(showHeaders,HDR_FROM, row, i18n("sender address field", "&From:"), + mLblFrom, mEdtFrom /*, mBtnFrom */ ); + + QWidget *prevFocus = mEdtFrom; + + if (!fromSlot) mReplyToAction->setChecked(abs(mShowHeaders)&HDR_REPLY_TO); + rethinkHeaderLine(showHeaders,HDR_REPLY_TO,row,i18n("&Reply to:"), + mLblReplyTo, mEdtReplyTo, mBtnReplyTo); + if ( showHeaders & HDR_REPLY_TO ) { + prevFocus = connectFocusMoving( prevFocus, mEdtReplyTo ); + } + + if ( mClassicalRecipients ) { + if (!fromSlot) mToAction->setChecked(abs(mShowHeaders)&HDR_TO); + rethinkHeaderLine(showHeaders, HDR_TO, row, i18n("recipient address field", "&To:"), + mLblTo, mEdtTo, mBtnTo, + i18n("Primary Recipients"), + i18n("<qt>The email addresses you put " + "in this field receive a copy of the email.</qt>")); + if ( showHeaders & HDR_TO ) { + prevFocus = connectFocusMoving( prevFocus, mEdtTo ); + } + + if (!fromSlot) mCcAction->setChecked(abs(mShowHeaders)&HDR_CC); + rethinkHeaderLine(showHeaders, HDR_CC, row, i18n("&Copy to (CC):"), + mLblCc, mEdtCc, mBtnCc, + i18n("Additional Recipients"), + i18n("<qt>The email addresses you put " + "in this field receive a copy of the email. " + "Technically it is the same thing as putting all the " + "addresses in the <b>To:</b> field but differs in " + "that it usually symbolises the receiver of the " + "Carbon Copy (CC) is a listener, not the main " + "recipient.</qt>")); + if ( showHeaders & HDR_CC ) { + prevFocus = connectFocusMoving( prevFocus, mEdtCc ); + } + + if (!fromSlot) mBccAction->setChecked(abs(mShowHeaders)&HDR_BCC); + rethinkHeaderLine(showHeaders,HDR_BCC, row, i18n("&Blind copy to (BCC):"), + mLblBcc, mEdtBcc, mBtnBcc, + i18n("Hidden Recipients"), + i18n("<qt>Essentially the same thing " + "as the <b>Copy To:</b> field but differs in that " + "all other recipients do not see who receives a " + "blind copy.</qt>")); + if ( showHeaders & HDR_BCC ) { + prevFocus = connectFocusMoving( prevFocus, mEdtBcc ); + } + } else { + mGrid->addMultiCellWidget( mRecipientsEditor, row, row, 0, 2 ); + ++row; + + if ( showHeaders & HDR_REPLY_TO ) { + connect( mEdtReplyTo, SIGNAL( focusDown() ), mRecipientsEditor, + SLOT( setFocusTop() ) ); + } else { + connect( mEdtFrom, SIGNAL( focusDown() ), mRecipientsEditor, + SLOT( setFocusTop() ) ); + } + if ( showHeaders & HDR_REPLY_TO ) { + connect( mRecipientsEditor, SIGNAL( focusUp() ), mEdtReplyTo, SLOT( setFocus() ) ); + } else { + connect( mRecipientsEditor, SIGNAL( focusUp() ), mEdtFrom, SLOT( setFocus() ) ); + } + + connect( mRecipientsEditor, SIGNAL( focusDown() ), mEdtSubject, + SLOT( setFocus() ) ); + connect( mEdtSubject, SIGNAL( focusUp() ), mRecipientsEditor, + SLOT( setFocusBottom() ) ); + + prevFocus = mRecipientsEditor; + } + if (!fromSlot) mSubjectAction->setChecked(abs(mShowHeaders)&HDR_SUBJECT); + rethinkHeaderLine(showHeaders,HDR_SUBJECT, row, i18n("S&ubject:"), + mLblSubject, mEdtSubject); + connectFocusMoving( mEdtSubject, mEditor ); + + assert(row<=mNumHeaders); + + + if( !mAtmList.isEmpty() ) + mAtmListView->show(); + else + mAtmListView->hide(); + resize(this->size()); + repaint(); + + mHeadersArea->setMaximumHeight( mHeadersArea->sizeHint().height() ); + mGrid->activate(); + mHeadersArea->show(); + + slotUpdateAttachActions(); + mIdentityAction->setEnabled(!mAllFieldsAction->isChecked()); + mDictionaryAction->setEnabled( !mAllFieldsAction->isChecked() ); + mTransportAction->setEnabled(!mAllFieldsAction->isChecked()); + mFromAction->setEnabled(!mAllFieldsAction->isChecked()); + if ( mReplyToAction ) mReplyToAction->setEnabled(!mAllFieldsAction->isChecked()); + if ( mToAction ) mToAction->setEnabled(!mAllFieldsAction->isChecked()); + if ( mCcAction ) mCcAction->setEnabled(!mAllFieldsAction->isChecked()); + if ( mBccAction ) mBccAction->setEnabled(!mAllFieldsAction->isChecked()); + mFccAction->setEnabled(!mAllFieldsAction->isChecked()); + mSubjectAction->setEnabled(!mAllFieldsAction->isChecked()); + if (mRecipientsEditor) + mRecipientsEditor->setFirstColumnWidth( mLabelWidth ); +} + +QWidget *KMComposeWin::connectFocusMoving( QWidget *prev, QWidget *next ) +{ + connect( prev, SIGNAL( focusDown() ), next, SLOT( setFocus() ) ); + connect( next, SIGNAL( focusUp() ), prev, SLOT( setFocus() ) ); + + return next; +} + +//----------------------------------------------------------------------------- +void KMComposeWin::rethinkHeaderLine(int aValue, int aMask, int& aRow, + const QString &aLabelStr, QLabel* aLbl, + QLineEdit* aEdt, QPushButton* aBtn, + const QString &toolTip, const QString &whatsThis ) +{ + if (aValue & aMask) + { + aLbl->setText(aLabelStr); + if ( !toolTip.isEmpty() ) + QToolTip::add( aLbl, toolTip ); + if ( !whatsThis.isEmpty() ) + QWhatsThis::add( aLbl, whatsThis ); + aLbl->setFixedWidth( mLabelWidth ); + aLbl->setBuddy(aEdt); + mGrid->addWidget(aLbl, aRow, 0); + aEdt->setBackgroundColor( mBackColor ); + aEdt->show(); + + if (aBtn) { + mGrid->addWidget(aEdt, aRow, 1); + + mGrid->addWidget(aBtn, aRow, 2); + aBtn->show(); + } else { + mGrid->addMultiCellWidget(aEdt, aRow, aRow, 1, 2 ); + } + aRow++; + } + else + { + aLbl->hide(); + aEdt->hide(); + if (aBtn) aBtn->hide(); + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::rethinkHeaderLine(int aValue, int aMask, int& aRow, + const QString &aLabelStr, QLabel* aLbl, + QComboBox* aCbx, QCheckBox* aChk) +{ + if (aValue & aMask) + { + aLbl->setText(aLabelStr); + aLbl->adjustSize(); + aLbl->resize((int)aLbl->sizeHint().width(),aLbl->sizeHint().height() + 6); + aLbl->setMinimumSize(aLbl->size()); + aLbl->show(); + aLbl->setBuddy(aCbx); + mGrid->addWidget(aLbl, aRow, 0); + aCbx->show(); + aCbx->setMinimumSize(100, aLbl->height()+2); + + mGrid->addWidget(aCbx, aRow, 1); + if ( aChk ) { + mGrid->addWidget(aChk, aRow, 2); + aChk->setFixedSize(aChk->sizeHint().width(), aLbl->height()); + aChk->show(); + } + aRow++; + } + else + { + aLbl->hide(); + aCbx->hide(); + if ( aChk ) + aChk->hide(); + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::getTransportMenu() +{ + QStringList availTransports; + + mActNowMenu->clear(); + mActLaterMenu->clear(); + availTransports = KMail::TransportManager::transportNames(); + QStringList::Iterator it; + int id = 0; + for(it = availTransports.begin(); it != availTransports.end() ; ++it, id++) + { + mActNowMenu->insertItem((*it).replace("&", "&&"), id); + mActLaterMenu->insertItem((*it).replace("&", "&&"), id); + } +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::setupActions(void) +{ + KActionMenu *actActionNowMenu, *actActionLaterMenu; + + if (kmkernel->msgSender()->sendImmediate()) //default == send now? + { + //default = send now, alternative = queue + ( void ) new KAction( i18n("&Send Mail"), "mail_send", CTRL+Key_Return, + this, SLOT(slotSendNow()), actionCollection(),"send_default"); + + // FIXME: change to mail_send_via icon when this exits. + actActionNowMenu = new KActionMenu (i18n("&Send Mail Via"), "mail_send", + actionCollection(), "send_default_via" ); + + (void) new KAction (i18n("Send &Later"), "queue", 0, this, + SLOT(slotSendLater()), actionCollection(),"send_alternative"); + actActionLaterMenu = new KActionMenu (i18n("Send &Later Via"), "queue", + actionCollection(), "send_alternative_via" ); + + } + else //no, default = send later + { + //default = queue, alternative = send now + (void) new KAction (i18n("Send &Later"), "queue", + CTRL+Key_Return, + this, SLOT(slotSendLater()), actionCollection(),"send_default"); + actActionLaterMenu = new KActionMenu (i18n("Send &Later Via"), "queue", + actionCollection(), "send_default_via" ); + + ( void ) new KAction( i18n("&Send Mail"), "mail_send", 0, + this, SLOT(slotSendNow()), actionCollection(),"send_alternative"); + + // FIXME: change to mail_send_via icon when this exits. + actActionNowMenu = new KActionMenu (i18n("&Send Mail Via"), "mail_send", + actionCollection(), "send_alternative_via" ); + + } + + // needed for sending "default transport" + actActionNowMenu->setDelayed(true); + actActionLaterMenu->setDelayed(true); + + connect( actActionNowMenu, SIGNAL( activated() ), this, + SLOT( slotSendNow() ) ); + connect( actActionLaterMenu, SIGNAL( activated() ), this, + SLOT( slotSendLater() ) ); + + + mActNowMenu = actActionNowMenu->popupMenu(); + mActLaterMenu = actActionLaterMenu->popupMenu(); + + connect( mActNowMenu, SIGNAL( activated( int ) ), this, + SLOT( slotSendNowVia( int ) ) ); + connect( mActNowMenu, SIGNAL( aboutToShow() ), this, + SLOT( getTransportMenu() ) ); + + connect( mActLaterMenu, SIGNAL( activated( int ) ), this, + SLOT( slotSendLaterVia( int ) ) ); + connect( mActLaterMenu, SIGNAL( aboutToShow() ), this, + SLOT( getTransportMenu() ) ); + + + + + (void) new KAction (i18n("Save as &Draft"), "filesave", 0, + this, SLOT(slotSaveDraft()), + actionCollection(), "save_in_drafts"); + (void) new KAction (i18n("Save as &Template"), "filesave", 0, + this, SLOT(slotSaveTemplate()), + actionCollection(), "save_in_templates"); + (void) new KAction (i18n("&Insert File..."), "fileopen", 0, + this, SLOT(slotInsertFile()), + actionCollection(), "insert_file"); + mRecentAction = new KRecentFilesAction (i18n("&Insert File Recent"), + "fileopen", 0, + this, SLOT(slotInsertRecentFile(const KURL&)), + actionCollection(), "insert_file_recent"); + + mRecentAction->loadEntries( KMKernel::config() ); + + (void) new KAction (i18n("&Address Book"), "contents",0, + this, SLOT(slotAddrBook()), + actionCollection(), "addressbook"); + (void) new KAction (i18n("&New Composer"), "mail_new", + KStdAccel::shortcut(KStdAccel::New), + this, SLOT(slotNewComposer()), + actionCollection(), "new_composer"); + (void) new KAction (i18n("New Main &Window"), "window_new", 0, + this, SLOT(slotNewMailReader()), + actionCollection(), "open_mailreader"); + + if ( !mClassicalRecipients ) { + new KAction( i18n("Select &Recipients..."), CTRL + Key_L, mRecipientsEditor, + SLOT( selectRecipients() ), actionCollection(), "select_recipients" ); + new KAction( i18n("Save &Distribution List..."), 0, mRecipientsEditor, + SLOT( saveDistributionList() ), actionCollection(), + "save_distribution_list" ); + } + + //KStdAction::save(this, SLOT(), actionCollection(), "save_message"); + KStdAction::print (this, SLOT(slotPrint()), actionCollection()); + KStdAction::close (this, SLOT(slotClose()), actionCollection()); + + KStdAction::undo (this, SLOT(slotUndo()), actionCollection()); + KStdAction::redo (this, SLOT(slotRedo()), actionCollection()); + KStdAction::cut (this, SLOT(slotCut()), actionCollection()); + KStdAction::copy (this, SLOT(slotCopy()), actionCollection()); + KStdAction::pasteText (this, SLOT(slotPasteClipboard()), actionCollection()); + KStdAction::selectAll (this, SLOT(slotMarkAll()), actionCollection()); + + KStdAction::find (this, SLOT(slotFind()), actionCollection()); + KStdAction::findNext(this, SLOT(slotSearchAgain()), actionCollection()); + + KStdAction::replace (this, SLOT(slotReplace()), actionCollection()); + KStdAction::spelling (this, SLOT(slotSpellcheck()), actionCollection(), "spellcheck"); + + mPasteQuotation = new KAction (i18n("Pa&ste as Quotation"),0,this,SLOT( slotPasteClipboardAsQuotation()), + actionCollection(), "paste_quoted"); + + (void) new KAction (i18n("Paste as Attac&hment"),0,this,SLOT( slotPasteClipboardAsAttachment()), + actionCollection(), "paste_att"); + + mAddQuoteChars = new KAction(i18n("Add &Quote Characters"), 0, this, + SLOT(slotAddQuotes()), actionCollection(), "tools_quote"); + + mRemQuoteChars = new KAction(i18n("Re&move Quote Characters"), 0, this, + SLOT(slotRemoveQuotes()), actionCollection(), "tools_unquote"); + + + (void) new KAction (i18n("Cl&ean Spaces"), 0, this, SLOT(slotCleanSpace()), + actionCollection(), "clean_spaces"); + + mFixedFontAction = new KToggleAction( i18n("Use Fi&xed Font"), 0, this, + SLOT(slotUpdateFont()), actionCollection(), "toggle_fixedfont" ); + mFixedFontAction->setChecked( GlobalSettings::self()->useFixedFont() ); + + //these are checkable!!! + mUrgentAction = new KToggleAction (i18n("&Urgent"), 0, + actionCollection(), + "urgent"); + mRequestMDNAction = new KToggleAction ( i18n("&Request Disposition Notification"), 0, + actionCollection(), + "options_request_mdn"); + mRequestMDNAction->setChecked(GlobalSettings::self()->requestMDN()); + //----- Message-Encoding Submenu + mEncodingAction = new KSelectAction( i18n( "Se&t Encoding" ), "charset", + 0, this, SLOT(slotSetCharset() ), + actionCollection(), "charsets" ); + mWordWrapAction = new KToggleAction (i18n("&Wordwrap"), 0, + actionCollection(), "wordwrap"); + mWordWrapAction->setChecked(GlobalSettings::self()->wordWrap()); + connect(mWordWrapAction, SIGNAL(toggled(bool)), SLOT(slotWordWrapToggled(bool))); + + mSnippetAction = new KToggleAction ( i18n("&Snippets"), 0, + actionCollection(), "snippets"); + connect(mSnippetAction, SIGNAL(toggled(bool)), mSnippetWidget, SLOT(setShown(bool)) ); + mSnippetAction->setChecked( GlobalSettings::self()->showSnippetManager() ); + + mAutoSpellCheckingAction = + new KToggleAction( i18n( "&Automatic Spellchecking" ), "spellcheck", 0, + actionCollection(), "options_auto_spellchecking" ); + const bool spellChecking = GlobalSettings::self()->autoSpellChecking(); + mAutoSpellCheckingAction->setEnabled( !GlobalSettings::self()->useExternalEditor() ); + mAutoSpellCheckingAction->setChecked( !GlobalSettings::self()->useExternalEditor() && spellChecking ); + slotAutoSpellCheckingToggled( !GlobalSettings::self()->useExternalEditor() && spellChecking ); + connect( mAutoSpellCheckingAction, SIGNAL( toggled( bool ) ), + this, SLOT( slotAutoSpellCheckingToggled( bool ) ) ); + + QStringList encodings = KMMsgBase::supportedEncodings(true); + encodings.prepend( i18n("Auto-Detect")); + mEncodingAction->setItems( encodings ); + mEncodingAction->setCurrentItem( -1 ); + + //these are checkable!!! + markupAction = new KToggleAction (i18n("Formatting (HTML)"), 0, this, + SLOT(slotToggleMarkup()), + actionCollection(), "html"); + + mAllFieldsAction = new KToggleAction (i18n("&All Fields"), 0, this, + SLOT(slotView()), + actionCollection(), "show_all_fields"); + mIdentityAction = new KToggleAction (i18n("&Identity"), 0, this, + SLOT(slotView()), + actionCollection(), "show_identity"); + mDictionaryAction = new KToggleAction (i18n("&Dictionary"), 0, this, + SLOT(slotView()), + actionCollection(), "show_dictionary"); + mFccAction = new KToggleAction (i18n("&Sent-Mail Folder"), 0, this, + SLOT(slotView()), + actionCollection(), "show_fcc"); + mTransportAction = new KToggleAction (i18n("&Mail Transport"), 0, this, + SLOT(slotView()), + actionCollection(), "show_transport"); + mFromAction = new KToggleAction (i18n("&From"), 0, this, + SLOT(slotView()), + actionCollection(), "show_from"); + mReplyToAction = new KToggleAction (i18n("&Reply To"), 0, this, + SLOT(slotView()), + actionCollection(), "show_reply_to"); + if ( mClassicalRecipients ) { + mToAction = new KToggleAction (i18n("&To"), 0, this, + SLOT(slotView()), + actionCollection(), "show_to"); + mCcAction = new KToggleAction (i18n("&CC"), 0, this, + SLOT(slotView()), + actionCollection(), "show_cc"); + mBccAction = new KToggleAction (i18n("&BCC"), 0, this, + SLOT(slotView()), + actionCollection(), "show_bcc"); + } + mSubjectAction = new KToggleAction (i18n("S&ubject"), 0, this, + SLOT(slotView()), + actionCollection(), "show_subject"); + //end of checkable + + mAppendSignatureAction = new KAction (i18n("Append S&ignature"), 0, this, + SLOT(slotAppendSignature()), + actionCollection(), "append_signature"); + mPrependSignatureAction = new KAction (i18n("Prepend S&ignature"), 0, this, + SLOT(slotPrependSignature()), + actionCollection(), "prepend_signature"); + + mInsertSignatureAction = new KAction (i18n("Insert Signature At C&ursor Position"), "edit", 0, this, + SLOT(slotInsertSignatureAtCursor()), + actionCollection(), "insert_signature_at_cursor_position"); + + mAttachPK = new KAction (i18n("Attach &Public Key..."), 0, this, + SLOT(slotInsertPublicKey()), + actionCollection(), "attach_public_key"); + mAttachMPK = new KAction (i18n("Attach &My Public Key"), 0, this, + SLOT(slotInsertMyPublicKey()), + actionCollection(), "attach_my_public_key"); + (void) new KAction (i18n("&Attach File..."), "attach", + 0, this, SLOT(slotAttachFile()), + actionCollection(), "attach"); + mAttachRemoveAction = new KAction (i18n("&Remove Attachment"), 0, this, + SLOT(slotAttachRemove()), + actionCollection(), "remove"); + mAttachSaveAction = new KAction (i18n("&Save Attachment As..."), "filesave",0, + this, SLOT(slotAttachSave()), + actionCollection(), "attach_save"); + mAttachPropertiesAction = new KAction (i18n("Attachment Pr&operties"), 0, this, + SLOT(slotAttachProperties()), + actionCollection(), "attach_properties"); + + setStandardToolBarMenuEnabled(true); + + KStdAction::keyBindings(this, SLOT(slotEditKeys()), actionCollection()); + KStdAction::configureToolbars(this, SLOT(slotEditToolbars()), actionCollection()); + KStdAction::preferences(kmkernel, SLOT(slotShowConfigurationDialog()), actionCollection()); + + (void) new KAction (i18n("&Spellchecker..."), 0, this, SLOT(slotSpellcheckConfig()), + actionCollection(), "setup_spellchecker"); + + if ( Kleo::CryptoBackendFactory::instance()->protocol( "Chiasmus" ) ) { + KToggleAction * a = new KToggleAction( i18n( "Encrypt Message with Chiasmus..." ), + "chidecrypted", 0, actionCollection(), + "encrypt_message_chiasmus" ); + a->setCheckedState( KGuiItem( i18n( "Encrypt Message with Chiasmus..." ), "chiencrypted" ) ); + mEncryptChiasmusAction = a; + connect( mEncryptChiasmusAction, SIGNAL(toggled(bool)), + this, SLOT(slotEncryptChiasmusToggled(bool)) ); + } else { + mEncryptChiasmusAction = 0; + } + + mEncryptAction = new KToggleAction (i18n("&Encrypt Message"), + "decrypted", 0, + actionCollection(), "encrypt_message"); + mSignAction = new KToggleAction (i18n("&Sign Message"), + "signature", 0, + actionCollection(), "sign_message"); + // get PGP user id for the chosen identity + const KPIM::Identity & ident = + kmkernel->identityManager()->identityForUoidOrDefault( mIdentity->currentIdentity() ); + // PENDING(marc): check the uses of this member and split it into + // smime/openpgp and or enc/sign, if necessary: + mLastIdentityHasSigningKey = !ident.pgpSigningKey().isEmpty() || !ident.smimeSigningKey().isEmpty(); + mLastIdentityHasEncryptionKey = !ident.pgpEncryptionKey().isEmpty() || !ident.smimeEncryptionKey().isEmpty(); + + mLastEncryptActionState = false; + mLastSignActionState = GlobalSettings::self()->pgpAutoSign(); + + // "Attach public key" is only possible if OpenPGP support is available: + mAttachPK->setEnabled( Kleo::CryptoBackendFactory::instance()->openpgp() ); + + // "Attach my public key" is only possible if OpenPGP support is + // available and the user specified his key for the current identity: + mAttachMPK->setEnabled( Kleo::CryptoBackendFactory::instance()->openpgp() && + !ident.pgpEncryptionKey().isEmpty() ); + + if ( !Kleo::CryptoBackendFactory::instance()->openpgp() && !Kleo::CryptoBackendFactory::instance()->smime() ) { + // no crypto whatsoever + mEncryptAction->setEnabled( false ); + setEncryption( false ); + mSignAction->setEnabled( false ); + setSigning( false ); + } else { + const bool canOpenPGPSign = Kleo::CryptoBackendFactory::instance()->openpgp() + && !ident.pgpSigningKey().isEmpty(); + const bool canSMIMESign = Kleo::CryptoBackendFactory::instance()->smime() + && !ident.smimeSigningKey().isEmpty(); + + setEncryption( false ); + setSigning( ( canOpenPGPSign || canSMIMESign ) && GlobalSettings::self()->pgpAutoSign() ); + } + + connect(mEncryptAction, SIGNAL(toggled(bool)), + SLOT(slotEncryptToggled( bool ))); + connect(mSignAction, SIGNAL(toggled(bool)), + SLOT(slotSignToggled( bool ))); + + QStringList l; + for ( int i = 0 ; i < numCryptoMessageFormats ; ++i ) + l.push_back( Kleo::cryptoMessageFormatToLabel( cryptoMessageFormats[i] ) ); + + mCryptoModuleAction = new KSelectAction( i18n( "&Cryptographic Message Format" ), 0, + this, SLOT(slotSelectCryptoModule()), + actionCollection(), "options_select_crypto" ); + mCryptoModuleAction->setItems( l ); + mCryptoModuleAction->setCurrentItem( format2cb( ident.preferredCryptoMessageFormat() ) ); + slotSelectCryptoModule( true /* initialize */ ); + + QStringList styleItems; + styleItems << i18n( "Standard" ); + styleItems << i18n( "Bulleted List (Disc)" ); + styleItems << i18n( "Bulleted List (Circle)" ); + styleItems << i18n( "Bulleted List (Square)" ); + styleItems << i18n( "Ordered List (Decimal)" ); + styleItems << i18n( "Ordered List (Alpha lower)" ); + styleItems << i18n( "Ordered List (Alpha upper)" ); + + listAction = new KSelectAction( i18n( "Select Style" ), 0, actionCollection(), + "text_list" ); + listAction->setItems( styleItems ); + connect( listAction, SIGNAL( activated( const QString& ) ), + SLOT( slotListAction( const QString& ) ) ); + fontAction = new KFontAction( "Select Font", 0, actionCollection(), + "text_font" ); + connect( fontAction, SIGNAL( activated( const QString& ) ), + SLOT( slotFontAction( const QString& ) ) ); + fontSizeAction = new KFontSizeAction( "Select Size", 0, actionCollection(), + "text_size" ); + connect( fontSizeAction, SIGNAL( fontSizeChanged( int ) ), + SLOT( slotSizeAction( int ) ) ); + + alignLeftAction = new KToggleAction (i18n("Align Left"), "text_left", 0, + this, SLOT(slotAlignLeft()), actionCollection(), + "align_left"); + alignLeftAction->setChecked( true ); + alignRightAction = new KToggleAction (i18n("Align Right"), "text_right", 0, + this, SLOT(slotAlignRight()), actionCollection(), + "align_right"); + alignCenterAction = new KToggleAction (i18n("Align Center"), "text_center", 0, + this, SLOT(slotAlignCenter()), actionCollection(), + "align_center"); + textBoldAction = new KToggleAction( i18n("&Bold"), "text_bold", CTRL+Key_B, + this, SLOT(slotTextBold()), + actionCollection(), "text_bold"); + textItalicAction = new KToggleAction( i18n("&Italic"), "text_italic", CTRL+Key_I, + this, SLOT(slotTextItalic()), + actionCollection(), "text_italic"); + textUnderAction = new KToggleAction( i18n("&Underline"), "text_under", CTRL+Key_U, + this, SLOT(slotTextUnder()), + actionCollection(), "text_under"); + actionFormatReset = new KAction( i18n( "Reset Font Settings" ), "eraser", 0, + this, SLOT( slotFormatReset() ), + actionCollection(), "format_reset"); + actionFormatColor = new KAction( i18n( "Text Color..." ), "colorize", 0, + this, SLOT( slotTextColor() ), + actionCollection(), "format_color"); + + // editorFocusChanged(false); + createGUI("kmcomposerui.rc"); + + connect( toolBar("htmlToolBar"), SIGNAL( visibilityChanged(bool) ), + this, SLOT( htmlToolBarVisibilityChanged(bool) ) ); + + // In Kontact, this entry would read "Configure Kontact", but bring + // up KMail's config dialog. That's sensible, though, so fix the label. + KAction* configureAction = actionCollection()->action("options_configure" ); + if ( configureAction ) + configureAction->setText( i18n("Configure KMail..." ) ); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::setupStatusBar(void) +{ + statusBar()->insertItem("", 0, 1); + statusBar()->setItemAlignment(0, AlignLeft | AlignVCenter); + + statusBar()->insertItem(i18n( " Spellcheck: %1 ").arg( " " ), 3, 0, true ); + statusBar()->insertItem(i18n( " Column: %1 ").arg(" "), 2, 0, true); + statusBar()->insertItem(i18n( " Line: %1 ").arg(" "), 1, 0, true); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::updateCursorPosition() +{ + int col,line; + QString temp; + line = mEditor->currentLine(); + col = mEditor->currentColumn(); + temp = i18n(" Line: %1 ").arg(line+1); + statusBar()->changeItem(temp,1); + temp = i18n(" Column: %1 ").arg(col+1); + statusBar()->changeItem(temp,2); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::setupEditor(void) +{ + //QPopupMenu* menu; + mEditor->setModified(false); + QFontMetrics fm(mBodyFont); + mEditor->setTabStopWidth(fm.width(QChar(' ')) * 8); + //mEditor->setFocusPolicy(QWidget::ClickFocus); + + slotWordWrapToggled( GlobalSettings::self()->wordWrap() ); + + // Font setup + slotUpdateFont(); + + /* installRBPopup() is broken in kdelibs, we should wait for + the new klibtextedit (dnaber, 2002-01-01) + menu = new QPopupMenu(this); + //#ifdef BROKEN + menu->insertItem(i18n("Undo"),mEditor, + SLOT(undo()), KStdAccel::shortcut(KStdAccel::Undo)); + menu->insertItem(i18n("Redo"),mEditor, + SLOT(redo()), KStdAccel::shortcut(KStdAccel::Redo)); + menu->insertSeparator(); + //#endif //BROKEN + menu->insertItem(i18n("Cut"), this, SLOT(slotCut())); + menu->insertItem(i18n("Copy"), this, SLOT(slotCopy())); + menu->insertItem(i18n("Paste"), this, SLOT(slotPasteClipboard())); + menu->insertItem(i18n("Mark All"),this, SLOT(slotMarkAll())); + menu->insertSeparator(); + menu->insertItem(i18n("Find..."), this, SLOT(slotFind())); + menu->insertItem(i18n("Replace..."), this, SLOT(slotReplace())); + menu->insertSeparator(); + menu->insertItem(i18n("Fixed Font Widths"), this, SLOT(slotUpdateFont())); + mEditor->installRBPopup(menu); + */ + updateCursorPosition(); + connect(mEditor,SIGNAL(CursorPositionChanged()),SLOT(updateCursorPosition())); + connect( mEditor, SIGNAL( currentFontChanged( const QFont & ) ), + this, SLOT( fontChanged( const QFont & ) ) ); + connect( mEditor, SIGNAL( currentAlignmentChanged( int ) ), + this, SLOT( alignmentChanged( int ) ) ); + +} + + +//----------------------------------------------------------------------------- +static QString cleanedUpHeaderString( const QString & s ) +{ + // remove invalid characters from the header strings + QString res( s ); + res.replace( '\r', "" ); + res.replace( '\n', " " ); + return res.stripWhiteSpace(); +} + +//----------------------------------------------------------------------------- +QString KMComposeWin::subject() const +{ + return cleanedUpHeaderString( mEdtSubject->text() ); +} + +//----------------------------------------------------------------------------- +QString KMComposeWin::to() const +{ + if ( mEdtTo ) { + return cleanedUpHeaderString( mEdtTo->text() ); + } else if ( mRecipientsEditor ) { + return mRecipientsEditor->recipientString( Recipient::To ); + } else { + return QString::null; + } +} + +//----------------------------------------------------------------------------- +QString KMComposeWin::cc() const +{ + if ( mEdtCc && !mEdtCc->isHidden() ) { + return cleanedUpHeaderString( mEdtCc->text() ); + } else if ( mRecipientsEditor ) { + return mRecipientsEditor->recipientString( Recipient::Cc ); + } else { + return QString::null; + } +} + +//----------------------------------------------------------------------------- +QString KMComposeWin::bcc() const +{ + if ( mEdtBcc && !mEdtBcc->isHidden() ) { + return cleanedUpHeaderString( mEdtBcc->text() ); + } else if ( mRecipientsEditor ) { + return mRecipientsEditor->recipientString( Recipient::Bcc ); + } else { + return QString::null; + } +} + +//----------------------------------------------------------------------------- +QString KMComposeWin::from() const +{ + return cleanedUpHeaderString( mEdtFrom->text() ); +} + +//----------------------------------------------------------------------------- +QString KMComposeWin::replyTo() const +{ + if ( mEdtReplyTo ) { + return cleanedUpHeaderString( mEdtReplyTo->text() ); + } else { + return QString::null; + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::verifyWordWrapLengthIsAdequate(const QString &body) +{ + int maxLineLength = 0; + int curPos; + int oldPos = 0; + if (mEditor->QTextEdit::wordWrap() == QTextEdit::FixedColumnWidth) { + for (curPos = 0; curPos < (int)body.length(); ++curPos) + if (body[curPos] == '\n') { + if ((curPos - oldPos) > maxLineLength) + maxLineLength = curPos - oldPos; + oldPos = curPos; + } + if ((curPos - oldPos) > maxLineLength) + maxLineLength = curPos - oldPos; + if (mEditor->wrapColumnOrWidth() < maxLineLength) // column + mEditor->setWrapColumnOrWidth(maxLineLength); + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::decryptOrStripOffCleartextSignature( QCString& body ) +{ + QPtrList<Kpgp::Block> pgpBlocks; + QStrList nonPgpBlocks; + if( Kpgp::Module::prepareMessageForDecryption( body, + pgpBlocks, nonPgpBlocks ) ) + { + // Only decrypt/strip off the signature if there is only one OpenPGP + // block in the message + if( pgpBlocks.count() == 1 ) + { + Kpgp::Block* block = pgpBlocks.first(); + if( ( block->type() == Kpgp::PgpMessageBlock ) || + ( block->type() == Kpgp::ClearsignedBlock ) ) + { + if( block->type() == Kpgp::PgpMessageBlock ) + // try to decrypt this OpenPGP block + block->decrypt(); + else + // strip off the signature + block->verify(); + + body = nonPgpBlocks.first() + + block->text() + + nonPgpBlocks.last(); + } + } + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::setTransport( const QString & transport ) +{ + kdDebug(5006) << "KMComposeWin::setTransport( \"" << transport << "\" )" << endl; + // Don't change the transport combobox if transport is empty + if ( transport.isEmpty() ) + return; + + bool transportFound = false; + for ( int i = 0; i < mTransport->count(); ++i ) { + if ( mTransport->text(i) == transport ) { + transportFound = true; + mTransport->setCurrentItem(i); + kdDebug(5006) << "transport found, it's no. " << i << " in the list" << endl; + break; + } + } + if ( !transportFound ) { // unknown transport + kdDebug(5006) << "unknown transport \"" << transport << "\"" << endl; + if ( transport.startsWith("smtp://") || transport.startsWith("smtps://") || + transport.startsWith("file://") ) { + // set custom transport + mTransport->setEditText( transport ); + } + else { + // neither known nor custom transport -> use default transport + mTransport->setCurrentText( GlobalSettings::self()->defaultTransport() ); + } + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::setMsg(KMMessage* newMsg, bool mayAutoSign, + bool allowDecryption, bool isModified) +{ + //assert(newMsg!=0); + if(!newMsg) + { + kdDebug(5006) << "KMComposeWin::setMsg() : newMsg == 0!" << endl; + return; + } + mMsg = newMsg; + KPIM::IdentityManager * im = kmkernel->identityManager(); + + mEdtFrom->setText(mMsg->from()); + mEdtReplyTo->setText(mMsg->replyTo()); + if ( mClassicalRecipients ) { + mEdtTo->setText(mMsg->to()); + mEdtCc->setText(mMsg->cc()); + mEdtBcc->setText(mMsg->bcc()); + } else { + mRecipientsEditor->setRecipientString( mMsg->to(), Recipient::To ); + mRecipientsEditor->setRecipientString( mMsg->cc(), Recipient::Cc ); + mRecipientsEditor->setRecipientString( mMsg->bcc(), Recipient::Bcc ); + mRecipientsEditor->setFocusBottom(); + } + mEdtSubject->setText(mMsg->subject()); + + const bool stickyIdentity = mBtnIdentity->isChecked(); + const bool messageHasIdentity = !newMsg->headerField("X-KMail-Identity").isEmpty(); + if (!stickyIdentity && messageHasIdentity) + mId = newMsg->headerField("X-KMail-Identity").stripWhiteSpace().toUInt(); + + // don't overwrite the header values with identity specific values + // unless the identity is sticky + if ( !stickyIdentity ) { + disconnect(mIdentity,SIGNAL(identityChanged(uint)), + this, SLOT(slotIdentityChanged(uint))); + } + // load the mId into the gui, sticky or not, without emitting + mIdentity->setCurrentIdentity( mId ); + const uint idToApply = mId; + if ( !stickyIdentity ) { + connect(mIdentity,SIGNAL(identityChanged(uint)), + this, SLOT(slotIdentityChanged(uint))); + } else { + // load the message's state into the mId, without applying it to the gui + // that's so we can detect that the id changed (because a sticky was set) + // on apply() + if ( messageHasIdentity ) + mId = newMsg->headerField("X-KMail-Identity").stripWhiteSpace().toUInt(); + else + mId = im->defaultIdentity().uoid(); + } + // manually load the identity's value into the fields; either the one from the + // messge, where appropriate, or the one from the sticky identity. What's in + // mId might have changed meanwhile, thus the save value + slotIdentityChanged( idToApply ); + + const KPIM::Identity & ident = im->identityForUoid( mIdentity->currentIdentity() ); + + // check for the presence of a DNT header, indicating that MDN's were + // requested + QString mdnAddr = newMsg->headerField("Disposition-Notification-To"); + mRequestMDNAction->setChecked( ( !mdnAddr.isEmpty() && + im->thatIsMe( mdnAddr ) ) || + GlobalSettings::self()->requestMDN() ); + + // check for presence of a priority header, indicating urgent mail: + mUrgentAction->setChecked( newMsg->isUrgent() ); + + if (!ident.isXFaceEnabled() || ident.xface().isEmpty()) + mMsg->removeHeaderField("X-Face"); + else + { + QString xface = ident.xface(); + if (!xface.isEmpty()) + { + int numNL = ( xface.length() - 1 ) / 70; + for ( int i = numNL; i > 0; --i ) + xface.insert( i*70, "\n\t" ); + mMsg->setHeaderField("X-Face", xface); + } + } + + // enable/disable encryption if the message was/wasn't encrypted + switch ( mMsg->encryptionState() ) { + case KMMsgFullyEncrypted: // fall through + case KMMsgPartiallyEncrypted: + mLastEncryptActionState = true; + break; + case KMMsgNotEncrypted: + mLastEncryptActionState = false; + break; + default: // nothing + break; + } + + // enable/disable signing if the message was/wasn't signed + switch ( mMsg->signatureState() ) { + case KMMsgFullySigned: // fall through + case KMMsgPartiallySigned: + mLastSignActionState = true; + break; + case KMMsgNotSigned: + mLastSignActionState = false; + break; + default: // nothing + break; + } + + // if these headers are present, the state of the message should be overruled + if ( mMsg->headers().FindField( "X-KMail-SignatureActionEnabled" ) ) + mLastSignActionState = (mMsg->headerField( "X-KMail-SignatureActionEnabled" ) == "true"); + if ( mMsg->headers().FindField( "X-KMail-EncryptActionEnabled" ) ) + mLastEncryptActionState = (mMsg->headerField( "X-KMail-EncryptActionEnabled" ) == "true"); + if ( mMsg->headers().FindField( "X-KMail-CryptoMessageFormat" ) ) + mCryptoModuleAction->setCurrentItem( format2cb( static_cast<Kleo::CryptoMessageFormat>( + mMsg->headerField( "X-KMail-CryptoMessageFormat" ).toInt() ) ) ); + + mLastIdentityHasSigningKey = !ident.pgpSigningKey().isEmpty() || !ident.smimeSigningKey().isEmpty(); + mLastIdentityHasEncryptionKey = !ident.pgpEncryptionKey().isEmpty() || !ident.smimeEncryptionKey().isEmpty(); + + if ( Kleo::CryptoBackendFactory::instance()->openpgp() || Kleo::CryptoBackendFactory::instance()->smime() ) { + const bool canOpenPGPSign = Kleo::CryptoBackendFactory::instance()->openpgp() + && !ident.pgpSigningKey().isEmpty(); + const bool canSMIMESign = Kleo::CryptoBackendFactory::instance()->smime() + && !ident.smimeSigningKey().isEmpty(); + + setEncryption( mLastEncryptActionState ); + setSigning( ( canOpenPGPSign || canSMIMESign ) && mLastSignActionState ); + } + slotUpdateSignatureAndEncrypionStateIndicators(); + + // "Attach my public key" is only possible if the user uses OpenPGP + // support and he specified his key: + mAttachMPK->setEnabled( Kleo::CryptoBackendFactory::instance()->openpgp() && + !ident.pgpEncryptionKey().isEmpty() ); + + QString transport = newMsg->headerField("X-KMail-Transport"); + if (!mBtnTransport->isChecked() && !transport.isEmpty()) + setTransport( transport ); + + if (!mBtnFcc->isChecked()) + { + if (!mMsg->fcc().isEmpty()) + setFcc(mMsg->fcc()); + else + setFcc(ident.fcc()); + } + + mDictionaryCombo->setCurrentByDictionary( ident.dictionary() ); + + partNode * root = partNode::fromMessage( mMsg ); + + KMail::ObjectTreeParser otp; // all defaults are ok + otp.parseObjectTree( root ); + + KMail::AttachmentCollector ac; + ac.setDiveIntoEncryptions( true ); + ac.setDiveIntoSignatures( true ); + ac.setDiveIntoMessages( false ); + + ac.collectAttachmentsFrom( root ); + + for ( std::vector<partNode*>::const_iterator it = ac.attachments().begin() ; it != ac.attachments().end() ; ++it ) + addAttach( new KMMessagePart( (*it)->msgPart() ) ); + + mEditor->setText( otp.textualContent() ); + mCharset = otp.textualContentCharset(); + if ( partNode * n = root->findType( DwMime::kTypeText, DwMime::kSubtypeHtml ) ) + if ( partNode * p = n->parentNode() ) + if ( p->hasType( DwMime::kTypeMultipart ) && + p->hasSubType( DwMime::kSubtypeAlternative ) ) + if ( mMsg->headerField( "X-KMail-Markup" ) == "true" ) { + toggleMarkup( true ); + + // get cte decoded body part + mCharset = n->msgPart().charset(); + QCString bodyDecoded = n->msgPart().bodyDecoded(); + + // respect html part charset + const QTextCodec *codec = KMMsgBase::codecForName( mCharset ); + if ( codec ) { + mEditor->setText( codec->toUnicode( bodyDecoded ) ); + } else { + mEditor->setText( QString::fromLocal8Bit( bodyDecoded ) ); + } + } + + if ( mCharset.isEmpty() ) + mCharset = mMsg->charset(); + if ( mCharset.isEmpty() ) + mCharset = mDefCharset; + setCharset( mCharset ); + + /* Handle the special case of non-mime mails */ + if ( mMsg->numBodyParts() == 0 && otp.textualContent().isEmpty() ) { + mCharset=mMsg->charset(); + if ( mCharset.isEmpty() || mCharset == "default" ) + mCharset = mDefCharset; + + QCString bodyDecoded = mMsg->bodyDecoded(); + + if( allowDecryption ) + decryptOrStripOffCleartextSignature( bodyDecoded ); + + const QTextCodec *codec = KMMsgBase::codecForName(mCharset); + if (codec) { + mEditor->setText(codec->toUnicode(bodyDecoded)); + } else + mEditor->setText(QString::fromLocal8Bit(bodyDecoded)); + } +#ifdef BROKEN_FOR_OPAQUE_SIGNED_OR_ENCRYPTED_MAILS + const int num = mMsg->numBodyParts(); + kdDebug(5006) << "KMComposeWin::setMsg() mMsg->numBodyParts=" + << mMsg->numBodyParts() << endl; + + if ( num > 0 ) { + KMMessagePart bodyPart; + int firstAttachment = 0; + + mMsg->bodyPart(1, &bodyPart); + if ( bodyPart.typeStr().lower() == "text" && + bodyPart.subtypeStr().lower() == "html" ) { + // check whether we are inside a mp/al body part + partNode *root = partNode::fromMessage( mMsg ); + partNode *node = root->findType( DwMime::kTypeText, + DwMime::kSubtypeHtml ); + if ( node && node->parentNode() && + node->parentNode()->hasType( DwMime::kTypeMultipart ) && + node->parentNode()->hasSubType( DwMime::kSubtypeAlternative ) ) { + // we have a mp/al body part with a text and an html body + kdDebug(5006) << "KMComposeWin::setMsg() : text/html found" << endl; + firstAttachment = 2; + if ( mMsg->headerField( "X-KMail-Markup" ) == "true" ) + toggleMarkup( true ); + } + delete root; root = 0; + } + if ( firstAttachment == 0 ) { + mMsg->bodyPart(0, &bodyPart); + if ( bodyPart.typeStr().lower() == "text" ) { + // we have a mp/mx body with a text body + kdDebug(5006) << "KMComposeWin::setMsg() : text/* found" << endl; + firstAttachment = 1; + } + } + + if ( firstAttachment != 0 ) // there's text to show + { + mCharset = bodyPart.charset(); + if ( mCharset.isEmpty() || mCharset == "default" ) + mCharset = mDefCharset; + + QCString bodyDecoded = bodyPart.bodyDecoded(); + + if( allowDecryption ) + decryptOrStripOffCleartextSignature( bodyDecoded ); + + // As nobody seems to know the purpose of the following line and + // as it breaks word wrapping of long lines if drafts with attachments + // are opened for editting in the composer (cf. Bug#41102) I comment it + // out. Ingo, 2002-04-21 + //verifyWordWrapLengthIsAdequate(bodyDecoded); + + const QTextCodec *codec = KMMsgBase::codecForName(mCharset); + if (codec) + mEditor->setText(codec->toUnicode(bodyDecoded)); + else + mEditor->setText(QString::fromLocal8Bit(bodyDecoded)); + //mEditor->insertLine("\n", -1); <-- why ? + } else mEditor->setText(""); + for( int i = firstAttachment; i < num; ++i ) + { + KMMessagePart *msgPart = new KMMessagePart; + mMsg->bodyPart(i, msgPart); + QCString mimeType = msgPart->typeStr().lower() + '/' + + msgPart->subtypeStr().lower(); + // don't add the detached signature as attachment when editting a + // PGP/MIME signed message + if( mimeType != "application/pgp-signature" ) { + addAttach(msgPart); + } + } + } else{ + mCharset=mMsg->charset(); + if ( mCharset.isEmpty() || mCharset == "default" ) + mCharset = mDefCharset; + + QCString bodyDecoded = mMsg->bodyDecoded(); + + if( allowDecryption ) + decryptOrStripOffCleartextSignature( bodyDecoded ); + + const QTextCodec *codec = KMMsgBase::codecForName(mCharset); + if (codec) { + mEditor->setText(codec->toUnicode(bodyDecoded)); + } else + mEditor->setText(QString::fromLocal8Bit(bodyDecoded)); + } + + setCharset(mCharset); +#endif // BROKEN_FOR_OPAQUE_SIGNED_OR_ENCRYPTED_MAILS + + if( (GlobalSettings::self()->autoTextSignature()=="auto") && mayAutoSign ) { + // + // Espen 2000-05-16 + // Delay the signature appending. It may start a fileseletor. + // Not user friendy if this modal fileseletor opens before the + // composer. + // + //QTimer::singleShot( 200, this, SLOT(slotAppendSignature()) ); + if ( GlobalSettings::self()->prependSignature() ) { + QTimer::singleShot( 0, this, SLOT(slotPrependSignature()) ); + } else { + QTimer::singleShot( 0, this, SLOT(slotAppendSignature()) ); + } + } + + if ( mMsg->getCursorPos() > 0 ) { + // The message has a cursor position explicitly set, so avoid + // changing it when appending the signature. + mPreserveUserCursorPosition = true; + } + setModified( isModified ); + + // do this even for new messages + mEditor->setCursorPositionFromStart( (unsigned int) mMsg->getCursorPos() ); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::setFcc( const QString &idString ) +{ + // check if the sent-mail folder still exists + if ( ! idString.isEmpty() && kmkernel->findFolderById( idString ) ) { + mFcc->setFolder( idString ); + } else { + mFcc->setFolder( kmkernel->sentFolder() ); + } +} + + +//----------------------------------------------------------------------------- +bool KMComposeWin::isModified() const +{ + return ( mEditor->isModified() || + mEdtFrom->edited() || + ( mEdtReplyTo && mEdtReplyTo->edited() ) || + ( mEdtTo && mEdtTo->edited() ) || + ( mEdtCc && mEdtCc->edited() ) || + ( mEdtBcc && mEdtBcc->edited() ) || + ( mRecipientsEditor && mRecipientsEditor->isModified() ) || + mEdtSubject->edited() || + mAtmModified || + ( mTransport->lineEdit() && mTransport->lineEdit()->edited() ) ); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::setModified( bool modified ) +{ + mEditor->setModified( modified ); + if ( !modified ) { + mEdtFrom->setEdited( false ); + if ( mEdtReplyTo ) mEdtReplyTo->setEdited( false ); + if ( mEdtTo ) mEdtTo->setEdited( false ); + if ( mEdtCc ) mEdtCc->setEdited( false ); + if ( mEdtBcc ) mEdtBcc->setEdited( false ); + if ( mRecipientsEditor ) mRecipientsEditor->clearModified(); + mEdtSubject->setEdited( false ); + mAtmModified = false ; + if ( mTransport->lineEdit() ) + mTransport->lineEdit()->setEdited( false ); + } +} + + +//----------------------------------------------------------------------------- +bool KMComposeWin::queryClose () +{ + if ( !mEditor->checkExternalEditorFinished() ) + return false; + if ( kmkernel->shuttingDown() || kapp->sessionSaving() ) + return true; + if ( mComposer && mComposer->isPerformingSignOperation() ) // since the non-gpg-agent gpg plugin gets a passphrase using QDialog::exec() + return false; // the user can try to close the window, which destroys mComposer mid-call. + + if ( isModified() ) { + bool istemplate = ( mFolder!=0 && mFolder->isTemplates() ); + const QString savebut = ( istemplate ? + i18n("Re&save as Template") : + i18n("&Save as Draft") ); + const QString savetext = ( istemplate ? + i18n("Resave this message in the Templates folder. " + "It can then be used at a later time.") : + i18n("Save this message in the Drafts folder. " + "It can then be edited and sent at a later time.") ); + + const int rc = KMessageBox::warningYesNoCancel( this, + i18n("Do you want to save the message for later or discard it?"), + i18n("Close Composer"), + KGuiItem(savebut, "filesave", QString::null, savetext), + KStdGuiItem::discard() ); + if ( rc == KMessageBox::Cancel ) + return false; + else if ( rc == KMessageBox::Yes ) { + // doSend will close the window. Just return false from this method + if ( istemplate ) { + slotSaveTemplate(); + } else { + slotSaveDraft(); + } + return false; + } + } + cleanupAutoSave(); + return true; +} + +//----------------------------------------------------------------------------- +bool KMComposeWin::userForgotAttachment() +{ + bool checkForForgottenAttachments = GlobalSettings::self()->showForgottenAttachmentWarning(); + + if ( !checkForForgottenAttachments || ( mAtmList.count() > 0 ) ) + return false; + + + QStringList attachWordsList = GlobalSettings::self()->attachmentKeywords(); + + if ( attachWordsList.isEmpty() ) { + // default value (FIXME: this is duplicated in configuredialog.cpp) + attachWordsList << QString::fromLatin1("attachment") + << QString::fromLatin1("attached"); + if ( QString::fromLatin1("attachment") != i18n("attachment") ) + attachWordsList << i18n("attachment"); + if ( QString::fromLatin1("attached") != i18n("attached") ) + attachWordsList << i18n("attached"); + } + + QRegExp rx ( QString::fromLatin1("\\b") + + attachWordsList.join("\\b|\\b") + + QString::fromLatin1("\\b") ); + rx.setCaseSensitive( false ); + + bool gotMatch = false; + + // check whether the subject contains one of the attachment key words + // unless the message is a reply or a forwarded message + QString subj = subject(); + gotMatch = ( KMMessage::stripOffPrefixes( subj ) == subj ) + && ( rx.search( subj ) >= 0 ); + + if ( !gotMatch ) { + // check whether the non-quoted text contains one of the attachment key + // words + QRegExp quotationRx ("^([ \\t]*([|>:}#]|[A-Za-z]+>))+"); + for ( int i = 0; i < mEditor->numLines(); ++i ) { + QString line = mEditor->textLine( i ); + gotMatch = ( quotationRx.search( line ) < 0 ) + && ( rx.search( line ) >= 0 ); + if ( gotMatch ) + break; + } + } + + if ( !gotMatch ) + return false; + + int rc = KMessageBox::warningYesNoCancel( this, + i18n("The message you have composed seems to refer to an " + "attached file but you have not attached anything.\n" + "Do you want to attach a file to your message?"), + i18n("File Attachment Reminder"), + i18n("&Attach File..."), + i18n("&Send as Is") ); + if ( rc == KMessageBox::Cancel ) + return true; + if ( rc == KMessageBox::Yes ) { + slotAttachFile(); + //preceed with editing + return true; + } + return false; +} + +//----------------------------------------------------------------------------- +void KMComposeWin::applyChanges( bool dontSignNorEncrypt, bool dontDisable ) +{ + kdDebug(5006) << "entering KMComposeWin::applyChanges" << endl; + + if(!mMsg || mComposer) { + kdDebug(5006) << "KMComposeWin::applyChanges() : mMsg == 0!\n" << endl; + emit applyChangesDone( false ); + return; + } + + // Make new job and execute it + mComposer = new MessageComposer( this ); + connect( mComposer, SIGNAL( done( bool ) ), + this, SLOT( slotComposerDone( bool ) ) ); + + // TODO: Add a cancel button for the following operations? + // Disable any input to the window, so that we have a snapshot of the + // composed stuff + if ( !dontDisable ) setEnabled( false ); + // apply the current state to the composer and let it do it's thing + mComposer->setDisableBreaking( mDisableBreaking ); // FIXME + mComposer->applyChanges( dontSignNorEncrypt ); +} + +void KMComposeWin::slotComposerDone( bool rc ) +{ + deleteAll( mComposedMessages ); + mComposedMessages = mComposer->composedMessageList(); + emit applyChangesDone( rc ); + delete mComposer; + mComposer = 0; + + // re-enable the composewin, the messsage composition is now done + setEnabled( true ); +} + +const KPIM::Identity & KMComposeWin::identity() const { + return kmkernel->identityManager()->identityForUoidOrDefault( mIdentity->currentIdentity() ); +} + +uint KMComposeWin::identityUid() const { + return mIdentity->currentIdentity(); +} + +Kleo::CryptoMessageFormat KMComposeWin::cryptoMessageFormat() const { + if ( !mCryptoModuleAction ) + return Kleo::AutoFormat; + return cb2format( mCryptoModuleAction->currentItem() ); +} + +bool KMComposeWin::encryptToSelf() const { +// return !Kpgp::Module::getKpgp() || Kpgp::Module::getKpgp()->encryptToSelf(); + KConfigGroup group( KMKernel::config(), "Composer" ); + return group.readBoolEntry( "crypto-encrypt-to-self", true ); +} + +bool KMComposeWin::queryExit () +{ + return true; +} + +//----------------------------------------------------------------------------- +bool KMComposeWin::addAttach(const KURL aUrl) +{ + if ( !aUrl.isValid() ) { + KMessageBox::sorry( this, i18n( "<qt><p>KMail could not recognize the location of the attachment (%1);</p>" + "<p>you have to specify the full path if you wish to attach a file.</p></qt>" ) + .arg( aUrl.prettyURL() ) ); + return false; + } + + const int maxAttachmentSize = GlobalSettings::maximumAttachmentSize(); + const uint maximumAttachmentSizeInByte = maxAttachmentSize*1024*1024; + if ( aUrl.isLocalFile() && QFileInfo( aUrl.pathOrURL() ).size() > maximumAttachmentSizeInByte ) { + KMessageBox::sorry( this, i18n( "<qt><p>Your administrator has disallowed attaching files bigger than %1 MB.</p>" ).arg( maxAttachmentSize ) ); + return false; + } + + KIO::TransferJob *job = KIO::get(aUrl); + KIO::Scheduler::scheduleJob( job ); + atmLoadData ld; + ld.url = aUrl; + ld.data = QByteArray(); + ld.insert = false; + if( !aUrl.fileEncoding().isEmpty() ) + ld.encoding = aUrl.fileEncoding().latin1(); + + mMapAtmLoadData.insert(job, ld); + mAttachJobs[job] = aUrl; + connect(job, SIGNAL(result(KIO::Job *)), + this, SLOT(slotAttachFileResult(KIO::Job *))); + connect(job, SIGNAL(data(KIO::Job *, const QByteArray &)), + this, SLOT(slotAttachFileData(KIO::Job *, const QByteArray &))); + return true; +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::addAttach(const KMMessagePart* msgPart) +{ + mAtmList.append(msgPart); + + // show the attachment listbox if it does not up to now + if (mAtmList.count()==1) + { + mAtmListView->resize(mAtmListView->width(), 50); + mAtmListView->show(); + resize(size()); + } + + // add a line in the attachment listbox + KMAtmListViewItem *lvi = new KMAtmListViewItem( mAtmListView ); + msgPartToItem(msgPart, lvi); + mAtmItemList.append(lvi); + + // the Attach file job has finished, so the possibly present tmp dir can be deleted now. + if ( mTempDir != 0 ) { + delete mTempDir; + mTempDir = 0; + } + + connect( lvi, SIGNAL( compress( int ) ), + this, SLOT( compressAttach( int ) ) ); + connect( lvi, SIGNAL( uncompress( int ) ), + this, SLOT( uncompressAttach( int ) ) ); + + slotUpdateAttachActions(); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotUpdateAttachActions() +{ + int selectedCount = 0; + for ( QPtrListIterator<QListViewItem> it(mAtmItemList); *it; ++it ) { + if ( (*it)->isSelected() ) { + ++selectedCount; + } + } + + mAttachRemoveAction->setEnabled( selectedCount >= 1 ); + mAttachSaveAction->setEnabled( selectedCount == 1 ); + mAttachPropertiesAction->setEnabled( selectedCount == 1 ); +} + + +//----------------------------------------------------------------------------- + +QString KMComposeWin::prettyMimeType( const QString& type ) +{ + QString t = type.lower(); + KServiceType::Ptr st = KServiceType::serviceType( t ); + return st ? st->comment() : t; +} + +void KMComposeWin::msgPartToItem(const KMMessagePart* msgPart, + KMAtmListViewItem *lvi, bool loadDefaults) +{ + assert(msgPart != 0); + + if (!msgPart->fileName().isEmpty()) + lvi->setText(0, msgPart->fileName()); + else + lvi->setText(0, msgPart->name()); + lvi->setText(1, KIO::convertSize( msgPart->decodedSize())); + lvi->setText(2, msgPart->contentTransferEncodingStr()); + lvi->setText(3, prettyMimeType(msgPart->typeStr() + "/" + msgPart->subtypeStr())); + lvi->setAttachmentSize(msgPart->decodedSize()); + + if ( loadDefaults ) { + if( canSignEncryptAttachments() ) { + lvi->enableCryptoCBs( true ); + lvi->setEncrypt( mEncryptAction->isChecked() ); + lvi->setSign( mSignAction->isChecked() ); + } else { + lvi->enableCryptoCBs( false ); + } + } +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::removeAttach(const QString &aUrl) +{ + int idx; + KMMessagePart* msgPart; + for(idx=0,msgPart=mAtmList.first(); msgPart; + msgPart=mAtmList.next(),idx++) { + if (msgPart->name() == aUrl) { + removeAttach(idx); + return; + } + } +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::removeAttach(int idx) +{ + mAtmModified = true; + mAtmList.remove(idx); + delete mAtmItemList.take(idx); + + if( mAtmList.isEmpty() ) + { + mAtmListView->hide(); + mAtmListView->setMinimumSize(0, 0); + resize(size()); + } +} + + +//----------------------------------------------------------------------------- +bool KMComposeWin::encryptFlagOfAttachment(int idx) +{ + return (int)(mAtmItemList.count()) > idx + ? static_cast<KMAtmListViewItem*>( mAtmItemList.at( idx ) )->isEncrypt() + : false; +} + + +//----------------------------------------------------------------------------- +bool KMComposeWin::signFlagOfAttachment(int idx) +{ + return (int)(mAtmItemList.count()) > idx + ? ((KMAtmListViewItem*)(mAtmItemList.at( idx )))->isSign() + : false; +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::addrBookSelInto() +{ + if ( mClassicalRecipients ) { + if ( GlobalSettings::self()->addresseeSelectorType() == + GlobalSettings::EnumAddresseeSelectorType::New ) { + addrBookSelIntoNew(); + } else { + addrBookSelIntoOld(); + } + } else { + kdWarning() << "To be implemented: call recipients picker." << endl; + } +} + +void KMComposeWin::addrBookSelIntoOld() +{ + AddressesDialog dlg( this ); + QString txt; + QStringList lst; + + txt = to(); + if ( !txt.isEmpty() ) { + lst = KPIM::splitEmailAddrList( txt ); + dlg.setSelectedTo( lst ); + } + + txt = mEdtCc->text(); + if ( !txt.isEmpty() ) { + lst = KPIM::splitEmailAddrList( txt ); + dlg.setSelectedCC( lst ); + } + + txt = mEdtBcc->text(); + if ( !txt.isEmpty() ) { + lst = KPIM::splitEmailAddrList( txt ); + dlg.setSelectedBCC( lst ); + } + + dlg.setRecentAddresses( RecentAddresses::self( KMKernel::config() )->kabcAddresses() ); + + if (dlg.exec()==QDialog::Rejected) return; + + mEdtTo->setText( dlg.to().join(", ") ); + mEdtTo->setEdited( true ); + + mEdtCc->setText( dlg.cc().join(", ") ); + mEdtCc->setEdited( true ); + + mEdtBcc->setText( dlg.bcc().join(", ") ); + mEdtBcc->setEdited( true ); + + //Make sure BCC field is shown if needed + if ( !mEdtBcc->text().isEmpty() ) { + mShowHeaders |= HDR_BCC; + rethinkFields( false ); + } +} + +void KMComposeWin::addrBookSelIntoNew() +{ + AddresseeEmailSelection selection; + + AddresseeSelectorDialog dlg( &selection ); + + QString txt; + QStringList lst; + + txt = to(); + if ( !txt.isEmpty() ) { + lst = KPIM::splitEmailAddrList( txt ); + selection.setSelectedTo( lst ); + } + + txt = mEdtCc->text(); + if ( !txt.isEmpty() ) { + lst = KPIM::splitEmailAddrList( txt ); + selection.setSelectedCC( lst ); + } + + txt = mEdtBcc->text(); + if ( !txt.isEmpty() ) { + lst = KPIM::splitEmailAddrList( txt ); + selection.setSelectedBCC( lst ); + } + + if (dlg.exec()==QDialog::Rejected) return; + + QStringList list = selection.to() + selection.toDistributionLists(); + mEdtTo->setText( list.join(", ") ); + mEdtTo->setEdited( true ); + + list = selection.cc() + selection.ccDistributionLists(); + mEdtCc->setText( list.join(", ") ); + mEdtCc->setEdited( true ); + + list = selection.bcc() + selection.bccDistributionLists(); + mEdtBcc->setText( list.join(", ") ); + mEdtBcc->setEdited( true ); + + //Make sure BCC field is shown if needed + if ( !mEdtBcc->text().isEmpty() ) { + mShowHeaders |= HDR_BCC; + rethinkFields( false ); + } +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::setCharset(const QCString& aCharset, bool forceDefault) +{ + if ((forceDefault && GlobalSettings::self()->forceReplyCharset()) || aCharset.isEmpty()) + mCharset = mDefCharset; + else + mCharset = aCharset.lower(); + + if ( mCharset.isEmpty() || mCharset == "default" ) + mCharset = mDefCharset; + + if (mAutoCharset) + { + mEncodingAction->setCurrentItem( 0 ); + return; + } + + QStringList encodings = mEncodingAction->items(); + int i = 0; + bool charsetFound = false; + for ( QStringList::Iterator it = encodings.begin(); it != encodings.end(); + ++it, i++ ) + { + if (i > 0 && ((mCharset == "us-ascii" && i == 1) || + (i != 1 && KGlobal::charsets()->codecForName( + KGlobal::charsets()->encodingForName(*it)) + == KGlobal::charsets()->codecForName(mCharset)))) + { + mEncodingAction->setCurrentItem( i ); + slotSetCharset(); + charsetFound = true; + break; + } + } + if (!aCharset.isEmpty() && !charsetFound) setCharset("", true); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAddrBook() +{ + KAddrBookExternal::openAddressBook(this); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAddrBookFrom() +{ + addrBookSelInto(); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAddrBookReplyTo() +{ + addrBookSelInto(); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAddrBookTo() +{ + addrBookSelInto(); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAttachFile() +{ + // Create File Dialog and return selected file(s) + // We will not care about any permissions, existence or whatsoever in + // this function. + + KFileDialog fdlg(QString::null, QString::null, this, 0, true); + fdlg.setOperationMode( KFileDialog::Other ); + fdlg.setCaption(i18n("Attach File")); + fdlg.okButton()->setGuiItem(KGuiItem(i18n("&Attach"),"fileopen")); + fdlg.setMode(KFile::Files); + fdlg.exec(); + KURL::List files = fdlg.selectedURLs(); + + for (KURL::List::Iterator it = files.begin(); it != files.end(); ++it) + addAttach(*it); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAttachFileData(KIO::Job *job, const QByteArray &data) +{ + QMap<KIO::Job*, atmLoadData>::Iterator it = mMapAtmLoadData.find(job); + assert(it != mMapAtmLoadData.end()); + QBuffer buff((*it).data); + buff.open(IO_WriteOnly | IO_Append); + buff.writeBlock(data.data(), data.size()); + buff.close(); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAttachFileResult(KIO::Job *job) +{ + QMap<KIO::Job*, atmLoadData>::Iterator it = mMapAtmLoadData.find(job); + assert(it != mMapAtmLoadData.end()); + KURL attachURL; + QMap<KIO::Job*, KURL>::iterator jit = mAttachJobs.find(job); + bool attachURLfound = (jit != mAttachJobs.end()); + if (attachURLfound) + { + attachURL = jit.data(); + mAttachJobs.remove(jit); + } + if (job->error()) + { + mMapAtmLoadData.remove(it); + job->showErrorDialog(); + if (attachURLfound) + emit attachmentAdded(attachURL, false); + return; + } + if ((*it).insert) + { + (*it).data.resize((*it).data.size() + 1); + (*it).data[(*it).data.size() - 1] = '\0'; + if ( const QTextCodec * codec = KGlobal::charsets()->codecForName((*it).encoding) ) + mEditor->insert( codec->toUnicode( (*it).data ) ); + else + mEditor->insert( QString::fromLocal8Bit( (*it).data ) ); + mMapAtmLoadData.remove(it); + if (attachURLfound) + emit attachmentAdded(attachURL, true); + return; + } + QCString partCharset; + if ( !( *it ).url.fileEncoding().isEmpty() ) { + partCharset = QCString( ( *it ).url.fileEncoding().latin1() ); + } else { + EncodingDetector ed; + KLocale *loc = KGlobal::locale(); + ed.setAutoDetectLanguage( EncodingDetector::scriptForLanguageCode ( loc->language() ) ); + ed.analyze( (*it).data ); + partCharset = ed.encoding(); + if ( partCharset.isEmpty() ) //shouldn't happen + partCharset = mCharset; + } + + KMMessagePart* msgPart; + + KCursorSaver busy(KBusyPtr::busy()); + QString name( (*it).url.fileName() ); + // ask the job for the mime type of the file + QString mimeType = static_cast<KIO::MimetypeJob*>(job)->mimetype(); + + if ( name.isEmpty() ) { + // URL ends with '/' (e.g. http://www.kde.org/) + // guess a reasonable filename + if( mimeType == "text/html" ) + name = "index.html"; + else { + // try to determine a reasonable extension + QStringList patterns( KMimeType::mimeType( mimeType )->patterns() ); + QString ext; + if( !patterns.isEmpty() ) { + ext = patterns[0]; + int i = ext.findRev( '.' ); + if( i == -1 ) + ext.prepend( '.' ); + else if( i > 0 ) + ext = ext.mid( i ); + } + name = QString("unknown") += ext; + } + } + + name.truncate( 256 ); // is this needed? + + QCString encoding = KMMsgBase::autoDetectCharset(partCharset, + KMMessage::preferredCharsets(), name); + if ( encoding.isEmpty() ) + encoding = "utf-8"; + + QCString encName; + if ( GlobalSettings::self()->outlookCompatibleAttachments() ) + encName = KMMsgBase::encodeRFC2047String( name, encoding ); + else + encName = KMMsgBase::encodeRFC2231String( name, encoding ); + bool RFC2231encoded = false; + if ( !GlobalSettings::self()->outlookCompatibleAttachments() ) + RFC2231encoded = name != QString( encName ); + + // create message part + msgPart = new KMMessagePart; + msgPart->setName(name); + QValueList<int> allowedCTEs; + if ( mimeType == "message/rfc822" ) { + msgPart->setMessageBody( (*it).data ); + allowedCTEs << DwMime::kCte7bit; + allowedCTEs << DwMime::kCte8bit; + } else { + msgPart->setBodyAndGuessCte((*it).data, allowedCTEs, + !kmkernel->msgSender()->sendQuotedPrintable()); + kdDebug(5006) << "autodetected cte: " << msgPart->cteStr() << endl; + } + int slash = mimeType.find( '/' ); + if( slash == -1 ) + slash = mimeType.length(); + msgPart->setTypeStr( mimeType.left( slash ).latin1() ); + msgPart->setSubtypeStr( mimeType.mid( slash + 1 ).latin1() ); + msgPart->setContentDisposition(QCString("attachment;\n\tfilename") + + ( RFC2231encoded ? "*=" + encName : "=\"" + encName + '"' ) ); + + mMapAtmLoadData.remove(it); + + msgPart->setCharset(partCharset); + + // show message part dialog, if not configured away (default): + KConfigGroup composer(KMKernel::config(), "Composer"); + if ( GlobalSettings::self()->showMessagePartDialogOnAttach() ) { + const KCursorSaver saver( QCursor::ArrowCursor ); + KMMsgPartDialogCompat dlg(mMainWidget); + int encodings = 0; + for ( QValueListConstIterator<int> it = allowedCTEs.begin() ; + it != allowedCTEs.end() ; ++it ) + switch ( *it ) { + case DwMime::kCteBase64: encodings |= KMMsgPartDialog::Base64; break; + case DwMime::kCteQp: encodings |= KMMsgPartDialog::QuotedPrintable; break; + case DwMime::kCte7bit: encodings |= KMMsgPartDialog::SevenBit; break; + case DwMime::kCte8bit: encodings |= KMMsgPartDialog::EightBit; break; + default: ; + } + dlg.setShownEncodings( encodings ); + dlg.setMsgPart(msgPart); + if (!dlg.exec()) { + delete msgPart; + msgPart = 0; + if (attachURLfound) + emit attachmentAdded(attachURL, false); + return; + } + } + mAtmModified = true; + if (msgPart->typeStr().lower() != "text") msgPart->setCharset(QCString()); + + // add the new attachment to the list + addAttach(msgPart); + + if (attachURLfound) + emit attachmentAdded(attachURL, true); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotInsertFile() +{ + KFileDialog fdlg(QString::null, QString::null, this, 0, true); + fdlg.setOperationMode( KFileDialog::Opening ); + fdlg.okButton()->setText(i18n("&Insert")); + fdlg.setCaption(i18n("Insert File")); + fdlg.toolBar()->insertCombo(KMMsgBase::supportedEncodings(false), 4711, + false, 0, 0, 0); + KComboBox *combo = fdlg.toolBar()->getCombo(4711); + for (int i = 0; i < combo->count(); i++) + if (KGlobal::charsets()->codecForName(KGlobal::charsets()-> + encodingForName(combo->text(i))) + == QTextCodec::codecForLocale()) combo->setCurrentItem(i); + if (!fdlg.exec()) return; + + KURL u = fdlg.selectedURL(); + mRecentAction->addURL(u); + // Prevent race condition updating list when multiple composers are open + { + KConfig *config = KMKernel::config(); + KConfigGroupSaver saver( config, "Composer" ); + QString encoding = KGlobal::charsets()->encodingForName(combo->currentText()).latin1(); + QStringList urls = config->readListEntry( "recent-urls" ); + QStringList encodings = config->readListEntry( "recent-encodings" ); + // Prevent config file from growing without bound + // Would be nicer to get this constant from KRecentFilesAction + uint mMaxRecentFiles = 30; + while (urls.count() > mMaxRecentFiles) + urls.erase( urls.fromLast() ); + while (encodings.count() > mMaxRecentFiles) + encodings.erase( encodings.fromLast() ); + // sanity check + if (urls.count() != encodings.count()) { + urls.clear(); + encodings.clear(); + } + urls.prepend( u.prettyURL() ); + encodings.prepend( encoding ); + config->writeEntry( "recent-urls", urls ); + config->writeEntry( "recent-encodings", encodings ); + mRecentAction->saveEntries( config ); + } + slotInsertRecentFile(u); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotInsertRecentFile(const KURL& u) +{ + if (u.fileName().isEmpty()) return; + + KIO::Job *job = KIO::get(u); + atmLoadData ld; + ld.url = u; + ld.data = QByteArray(); + ld.insert = true; + // Get the encoding previously used when inserting this file + { + KConfig *config = KMKernel::config(); + KConfigGroupSaver saver( config, "Composer" ); + QStringList urls = config->readListEntry( "recent-urls" ); + QStringList encodings = config->readListEntry( "recent-encodings" ); + int index = urls.findIndex( u.prettyURL() ); + if (index != -1) { + QString encoding = encodings[ index ]; + ld.encoding = encoding.latin1(); + } + } + mMapAtmLoadData.insert(job, ld); + connect(job, SIGNAL(result(KIO::Job *)), + this, SLOT(slotAttachFileResult(KIO::Job *))); + connect(job, SIGNAL(data(KIO::Job *, const QByteArray &)), + this, SLOT(slotAttachFileData(KIO::Job *, const QByteArray &))); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotSetCharset() +{ + if (mEncodingAction->currentItem() == 0) + { + mAutoCharset = true; + return; + } + mAutoCharset = false; + + mCharset = KGlobal::charsets()->encodingForName( mEncodingAction-> + currentText() ).latin1(); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotSelectCryptoModule( bool init ) +{ + if ( !init ) { + setModified( true ); + } + if( canSignEncryptAttachments() ) { + // if the encrypt/sign columns are hidden then show them + if( 0 == mAtmListView->columnWidth( mAtmColEncrypt ) ) { + // set/unset signing/encryption for all attachments according to the + // state of the global sign/encrypt action + if( !mAtmList.isEmpty() ) { + for( KMAtmListViewItem* lvi = static_cast<KMAtmListViewItem*>( mAtmItemList.first() ); + lvi; + lvi = static_cast<KMAtmListViewItem*>( mAtmItemList.next() ) ) { + lvi->setSign( mSignAction->isChecked() ); + lvi->setEncrypt( mEncryptAction->isChecked() ); + } + } + int totalWidth = 0; + // determine the total width of the columns + for( int col=0; col < mAtmColEncrypt; col++ ) + totalWidth += mAtmListView->columnWidth( col ); + int reducedTotalWidth = totalWidth - mAtmEncryptColWidth + - mAtmSignColWidth; + // reduce the width of all columns so that the encrypt and sign column + // fit + int usedWidth = 0; + for( int col=0; col < mAtmColEncrypt-1; col++ ) { + int newWidth = mAtmListView->columnWidth( col ) * reducedTotalWidth + / totalWidth; + mAtmListView->setColumnWidth( col, newWidth ); + usedWidth += newWidth; + } + // the last column before the encrypt column gets the remaining space + // (because of rounding errors the width of this column isn't calculated + // the same way as the width of the other columns) + mAtmListView->setColumnWidth( mAtmColEncrypt-1, + reducedTotalWidth - usedWidth ); + mAtmListView->setColumnWidth( mAtmColEncrypt, mAtmEncryptColWidth ); + mAtmListView->setColumnWidth( mAtmColSign, mAtmSignColWidth ); + for( KMAtmListViewItem* lvi = static_cast<KMAtmListViewItem*>( mAtmItemList.first() ); + lvi; + lvi = static_cast<KMAtmListViewItem*>( mAtmItemList.next() ) ) { + lvi->enableCryptoCBs( true ); + } + } + } else { + // if the encrypt/sign columns are visible then hide them + if( 0 != mAtmListView->columnWidth( mAtmColEncrypt ) ) { + mAtmEncryptColWidth = mAtmListView->columnWidth( mAtmColEncrypt ); + mAtmSignColWidth = mAtmListView->columnWidth( mAtmColSign ); + int totalWidth = 0; + // determine the total width of the columns + for( int col=0; col < mAtmListView->columns(); col++ ) + totalWidth += mAtmListView->columnWidth( col ); + int reducedTotalWidth = totalWidth - mAtmEncryptColWidth + - mAtmSignColWidth; + // increase the width of all columns so that the visible columns take + // up the whole space + int usedWidth = 0; + for( int col=0; col < mAtmColEncrypt-1; col++ ) { + int newWidth = mAtmListView->columnWidth( col ) * totalWidth + / reducedTotalWidth; + mAtmListView->setColumnWidth( col, newWidth ); + usedWidth += newWidth; + } + // the last column before the encrypt column gets the remaining space + // (because of rounding errors the width of this column isn't calculated + // the same way as the width of the other columns) + mAtmListView->setColumnWidth( mAtmColEncrypt-1, totalWidth - usedWidth ); + mAtmListView->setColumnWidth( mAtmColEncrypt, 0 ); + mAtmListView->setColumnWidth( mAtmColSign, 0 ); + for( KMAtmListViewItem* lvi = static_cast<KMAtmListViewItem*>( mAtmItemList.first() ); + lvi; + lvi = static_cast<KMAtmListViewItem*>( mAtmItemList.next() ) ) { + lvi->enableCryptoCBs( false ); + } + } + } +} + +static void showExportError( QWidget * w, const GpgME::Error & err ) { + assert( err ); + const QString msg = i18n("<qt><p>An error occurred while trying to export " + "the key from the backend:</p>" + "<p><b>%1</b></p></qt>") + .arg( QString::fromLocal8Bit( err.asString() ) ); + KMessageBox::error( w, msg, i18n("Key Export Failed") ); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotInsertMyPublicKey() +{ + // get PGP user id for the chosen identity + mFingerprint = + kmkernel->identityManager()->identityForUoidOrDefault( mIdentity->currentIdentity() ).pgpEncryptionKey(); + if ( !mFingerprint.isEmpty() ) + startPublicKeyExport(); +} + +void KMComposeWin::startPublicKeyExport() { + if ( mFingerprint.isEmpty() || !Kleo::CryptoBackendFactory::instance()->openpgp() ) + return; + Kleo::ExportJob * job = Kleo::CryptoBackendFactory::instance()->openpgp()->publicKeyExportJob( true ); + assert( job ); + + connect( job, SIGNAL(result(const GpgME::Error&,const QByteArray&)), + this, SLOT(slotPublicKeyExportResult(const GpgME::Error&,const QByteArray&)) ); + + const GpgME::Error err = job->start( mFingerprint ); + if ( err ) + showExportError( this, err ); + else + (void)new Kleo::ProgressDialog( job, i18n("Exporting key..."), this ); +} + +void KMComposeWin::slotPublicKeyExportResult( const GpgME::Error & err, const QByteArray & keydata ) { + if ( err ) { + showExportError( this, err ); + return; + } + + // create message part + KMMessagePart * msgPart = new KMMessagePart(); + msgPart->setName( i18n("OpenPGP key 0x%1").arg( mFingerprint ) ); + msgPart->setTypeStr("application"); + msgPart->setSubtypeStr("pgp-keys"); + QValueList<int> dummy; + msgPart->setBodyAndGuessCte(keydata, dummy, false); + msgPart->setContentDisposition( "attachment;\n\tfilename=0x" + QCString( mFingerprint.latin1() ) + ".asc" ); + + // add the new attachment to the list + addAttach(msgPart); + rethinkFields(); //work around initial-size bug in Qt-1.32 +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotInsertPublicKey() +{ + Kleo::KeySelectionDialog dlg( i18n("Attach Public OpenPGP Key"), + i18n("Select the public key which should " + "be attached."), + std::vector<GpgME::Key>(), + Kleo::KeySelectionDialog::PublicKeys|Kleo::KeySelectionDialog::OpenPGPKeys, + false /* no multi selection */, + false /* no remember choice box */, + this, "attach public key selection dialog" ); + if ( dlg.exec() != QDialog::Accepted ) + return; + + mFingerprint = dlg.fingerprint(); + startPublicKeyExport(); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAttachPopupMenu(QListViewItem *, const QPoint &, int) +{ + if (!mAttachMenu) + { + mAttachMenu = new QPopupMenu(this); + + mOpenId = mAttachMenu->insertItem(i18n("to open", "Open"), this, + SLOT(slotAttachOpen())); + mOpenWithId = mAttachMenu->insertItem(i18n("Open With..."), this, + SLOT(slotAttachOpenWith())); + mViewId = mAttachMenu->insertItem(i18n("to view", "View"), this, + SLOT(slotAttachView())); + mEditId = mAttachMenu->insertItem( i18n("Edit"), this, SLOT(slotAttachEdit()) ); + mEditWithId = mAttachMenu->insertItem( i18n("Edit With..."), this, + SLOT(slotAttachEditWith()) ); + mRemoveId = mAttachMenu->insertItem(i18n("Remove"), this, SLOT(slotAttachRemove())); + mSaveAsId = mAttachMenu->insertItem( SmallIconSet("filesaveas"), i18n("Save As..."), this, + SLOT( slotAttachSave() ) ); + mPropertiesId = mAttachMenu->insertItem( i18n("Properties"), this, + SLOT( slotAttachProperties() ) ); + mAttachMenu->insertSeparator(); + mAttachMenu->insertItem(i18n("Add Attachment..."), this, SLOT(slotAttachFile())); + } + + int selectedCount = 0; + for ( QPtrListIterator<QListViewItem> it(mAtmItemList); *it; ++it ) { + if ( (*it)->isSelected() ) { + ++selectedCount; + } + } + + mAttachMenu->setItemEnabled( mOpenId, selectedCount > 0 ); + mAttachMenu->setItemEnabled( mOpenWithId, selectedCount > 0 ); + mAttachMenu->setItemEnabled( mViewId, selectedCount > 0 ); + mAttachMenu->setItemEnabled( mEditId, selectedCount == 1 ); + mAttachMenu->setItemEnabled( mEditWithId, selectedCount == 1 ); + mAttachMenu->setItemEnabled( mRemoveId, selectedCount > 0 ); + mAttachMenu->setItemEnabled( mSaveAsId, selectedCount == 1 ); + mAttachMenu->setItemEnabled( mPropertiesId, selectedCount == 1 ); + + mAttachMenu->popup(QCursor::pos()); +} + +//----------------------------------------------------------------------------- +int KMComposeWin::currentAttachmentNum() +{ + int i = 0; + for ( QPtrListIterator<QListViewItem> it(mAtmItemList); *it; ++it, ++i ) + if ( *it == mAtmListView->currentItem() ) + return i; + return -1; +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAttachProperties() +{ + int idx = currentAttachmentNum(); + + if (idx < 0) return; + + KMMessagePart* msgPart = mAtmList.at(idx); + msgPart->setCharset(mCharset); + + KMMsgPartDialogCompat dlg(mMainWidget); + dlg.setMsgPart(msgPart); + KMAtmListViewItem* listItem = (KMAtmListViewItem*)(mAtmItemList.at(idx)); + if( canSignEncryptAttachments() && listItem ) { + dlg.setCanSign( true ); + dlg.setCanEncrypt( true ); + dlg.setSigned( listItem->isSign() ); + dlg.setEncrypted( listItem->isEncrypt() ); + } else { + dlg.setCanSign( false ); + dlg.setCanEncrypt( false ); + } + if (dlg.exec()) + { + mAtmModified = true; + // values may have changed, so recreate the listbox line + if( listItem ) { + msgPartToItem(msgPart, listItem); + if( canSignEncryptAttachments() ) { + listItem->setSign( dlg.isSigned() ); + listItem->setEncrypt( dlg.isEncrypted() ); + } + } + } + if (msgPart->typeStr().lower() != "text") msgPart->setCharset(QCString()); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::compressAttach( int idx ) +{ + if (idx < 0) return; + + unsigned int i; + for ( i = 0; i < mAtmItemList.count(); ++i ) + if ( mAtmItemList.at( i )->itemPos() == idx ) + break; + + if ( i > mAtmItemList.count() ) + return; + + KMMessagePart* msgPart; + msgPart = mAtmList.at( i ); + QByteArray array; + QBuffer dev( array ); + KZip zip( &dev ); + QByteArray decoded = msgPart->bodyDecodedBinary(); + if ( ! zip.open( IO_WriteOnly ) ) { + KMessageBox::sorry(0, i18n("KMail could not compress the file.") ); + static_cast<KMAtmListViewItem*>( mAtmItemList.at( i ) )->setCompress( false ); + return; + } + + zip.setCompression( KZip::DeflateCompression ); + if ( ! zip.writeFile( msgPart->name(), "", "", decoded.size(), + decoded.data() ) ) { + KMessageBox::sorry(0, i18n("KMail could not compress the file.") ); + static_cast<KMAtmListViewItem*>( mAtmItemList.at( i ) )->setCompress( false ); + return; + } + zip.close(); + if ( array.size() >= decoded.size() ) { + if ( KMessageBox::questionYesNo( this, i18n("The compressed file is larger " + "than the original. Do you want to keep the original one?" ), QString::null, i18n("Keep"), i18n("Compress") ) + == KMessageBox::Yes ) { + static_cast<KMAtmListViewItem*>( mAtmItemList.at( i ) )->setCompress( false ); + return; + } + } + static_cast<KMAtmListViewItem*>( mAtmItemList.at( i ) )->setUncompressedCodec( + msgPart->cteStr() ); + + msgPart->setCteStr( "base64" ); + msgPart->setBodyEncodedBinary( array ); + QString name = msgPart->name() + ".zip"; + + msgPart->setName( name ); + + QCString cDisp = "attachment;"; + QCString encoding = KMMsgBase::autoDetectCharset( msgPart->charset(), + KMMessage::preferredCharsets(), name ); + kdDebug(5006) << "encoding: " << encoding << endl; + if ( encoding.isEmpty() ) encoding = "utf-8"; + kdDebug(5006) << "encoding after: " << encoding << endl; + QCString encName; + if ( GlobalSettings::self()->outlookCompatibleAttachments() ) + encName = KMMsgBase::encodeRFC2047String( name, encoding ); + else + encName = KMMsgBase::encodeRFC2231String( name, encoding ); + + cDisp += "\n\tfilename"; + if ( name != QString( encName ) ) + cDisp += "*=" + encName; + else + cDisp += "=\"" + encName + '"'; + msgPart->setContentDisposition( cDisp ); + + static_cast<KMAtmListViewItem*>( mAtmItemList.at( i ) )->setUncompressedMimeType( + msgPart->typeStr(), msgPart->subtypeStr() ); + msgPart->setTypeStr( "application" ); + msgPart->setSubtypeStr( "x-zip" ); + + KMAtmListViewItem* listItem = static_cast<KMAtmListViewItem*>( mAtmItemList.at( i ) ); + msgPartToItem( msgPart, listItem, false ); +} + +//----------------------------------------------------------------------------- + +void KMComposeWin::uncompressAttach( int idx ) +{ + if (idx < 0) return; + + unsigned int i; + for ( i = 0; i < mAtmItemList.count(); ++i ) + if ( mAtmItemList.at( i )->itemPos() == idx ) + break; + + if ( i > mAtmItemList.count() ) + return; + + KMMessagePart* msgPart; + msgPart = mAtmList.at( i ); + + QBuffer dev( msgPart->bodyDecodedBinary() ); + KZip zip( &dev ); + QByteArray decoded; + + decoded = msgPart->bodyDecodedBinary(); + if ( ! zip.open( IO_ReadOnly ) ) { + KMessageBox::sorry(0, i18n("KMail could not uncompress the file.") ); + static_cast<KMAtmListViewItem *>( mAtmItemList.at( i ) )->setCompress( true ); + return; + } + const KArchiveDirectory *dir = zip.directory(); + + KZipFileEntry *entry; + if ( dir->entries().count() != 1 ) { + KMessageBox::sorry(0, i18n("KMail could not uncompress the file.") ); + static_cast<KMAtmListViewItem *>( mAtmItemList.at( i ) )->setCompress( true ); + return; + } + entry = (KZipFileEntry*)dir->entry( dir->entries()[0] ); + + msgPart->setCteStr( + static_cast<KMAtmListViewItem*>( mAtmItemList.at(i) )->uncompressedCodec() ); + + msgPart->setBodyEncodedBinary( entry->data() ); + QString name = entry->name(); + msgPart->setName( name ); + + zip.close(); + + QCString cDisp = "attachment;"; + QCString encoding = KMMsgBase::autoDetectCharset( msgPart->charset(), + KMMessage::preferredCharsets(), name ); + if ( encoding.isEmpty() ) encoding = "utf-8"; + + QCString encName; + if ( GlobalSettings::self()->outlookCompatibleAttachments() ) + encName = KMMsgBase::encodeRFC2047String( name, encoding ); + else + encName = KMMsgBase::encodeRFC2231String( name, encoding ); + + cDisp += "\n\tfilename"; + if ( name != QString( encName ) ) + cDisp += "*=" + encName; + else + cDisp += "=\"" + encName + '"'; + msgPart->setContentDisposition( cDisp ); + + QCString type, subtype; + static_cast<KMAtmListViewItem*>( mAtmItemList.at( i ) )->uncompressedMimeType( type, + subtype ); + + msgPart->setTypeStr( type ); + msgPart->setSubtypeStr( subtype ); + + KMAtmListViewItem* listItem = static_cast<KMAtmListViewItem*>(mAtmItemList.at( i )); + msgPartToItem( msgPart, listItem, false ); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAttachView() +{ + int i = 0; + for ( QPtrListIterator<QListViewItem> it(mAtmItemList); *it; ++it, ++i ) { + if ( (*it)->isSelected() ) { + viewAttach( i ); + } + } +} +//----------------------------------------------------------------------------- +void KMComposeWin::slotAttachOpen() +{ + int i = 0; + for ( QPtrListIterator<QListViewItem> it(mAtmItemList); *it; ++it, ++i ) { + if ( (*it)->isSelected() ) { + openAttach( i, false ); + } + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAttachOpenWith() +{ + int i = 0; + for ( QPtrListIterator<QListViewItem> it(mAtmItemList); *it; ++it, ++i ) { + if ( (*it)->isSelected() ) { + openAttach( i, true ); + } + } +} + +void KMComposeWin::slotAttachEdit() +{ + int i = 0; + for ( QPtrListIterator<QListViewItem> it(mAtmItemList); *it; ++it, ++i ) { + if ( (*it)->isSelected() ) { + editAttach( i, false ); + } + } +} + +void KMComposeWin::slotAttachEditWith() +{ + int i = 0; + for ( QPtrListIterator<QListViewItem> it(mAtmItemList); *it; ++it, ++i ) { + if ( (*it)->isSelected() ) { + editAttach( i, true ); + } + } +} + +//----------------------------------------------------------------------------- +bool KMComposeWin::inlineSigningEncryptionSelected() { + if ( !mSignAction->isChecked() && !mEncryptAction->isChecked() ) + return false; + return cryptoMessageFormat() == Kleo::InlineOpenPGPFormat; +} + +//----------------------------------------------------------------------------- +void KMComposeWin::viewAttach( int index ) +{ + QString pname; + KMMessagePart* msgPart; + msgPart = mAtmList.at(index); + pname = msgPart->name().stripWhiteSpace(); + if (pname.isEmpty()) pname=msgPart->contentDescription(); + if (pname.isEmpty()) pname="unnamed"; + + KTempFile* atmTempFile = new KTempFile(); + mAtmTempList.append( atmTempFile ); + atmTempFile->setAutoDelete( true ); + KPIM::kByteArrayToFile(msgPart->bodyDecodedBinary(), atmTempFile->name(), false, false, + false); + KMReaderMainWin *win = new KMReaderMainWin(msgPart, false, + atmTempFile->name(), pname, mCharset ); + win->show(); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::openAttach( int index, bool with ) +{ + KMMessagePart* msgPart = mAtmList.at(index); + const QString contentTypeStr = + ( msgPart->typeStr() + '/' + msgPart->subtypeStr() ).lower(); + + KMimeType::Ptr mimetype; + mimetype = KMimeType::mimeType( contentTypeStr ); + + KTempFile* atmTempFile = new KTempFile(); + mAtmTempList.append( atmTempFile ); + const bool autoDelete = true; + atmTempFile->setAutoDelete( autoDelete ); + + KURL url; + url.setPath( atmTempFile->name() ); + + KPIM::kByteArrayToFile( msgPart->bodyDecodedBinary(), atmTempFile->name(), false, false, + false ); + if ( ::chmod( QFile::encodeName( atmTempFile->name() ), S_IRUSR ) != 0) { + QFile::remove(url.path()); + return; + } + + KService::Ptr offer = + KServiceTypeProfile::preferredService( mimetype->name(), "Application" ); + + if ( with || !offer || mimetype->name() == "application/octet-stream" ) { + if ( ( !KRun::displayOpenWithDialog( url, autoDelete ) ) && autoDelete ) { + QFile::remove(url.path()); + } + } + else { + if ( ( !KRun::run( *offer, url, autoDelete ) ) && autoDelete ) { + QFile::remove( url.path() ); + } + } +} + +void KMComposeWin::editAttach(int index, bool openWith) +{ + KMMessagePart* msgPart = mAtmList.at(index); + const QString contentTypeStr = + ( msgPart->typeStr() + '/' + msgPart->subtypeStr() ).lower(); + + KTempFile* atmTempFile = new KTempFile(); + mAtmTempList.append( atmTempFile ); + atmTempFile->setAutoDelete( true ); + atmTempFile->file()->writeBlock( msgPart->bodyDecodedBinary() ); + atmTempFile->file()->flush(); + + + KMail::EditorWatcher *watcher = new KMail::EditorWatcher( KURL( atmTempFile->name() ), contentTypeStr, openWith, this ); + connect( watcher, SIGNAL(editDone(KMail::EditorWatcher*)), SLOT(slotEditDone(KMail::EditorWatcher*)) ); + if ( watcher->start() ) { + mEditorMap.insert( watcher, msgPart ); + mEditorTempFiles.insert( watcher, atmTempFile ); + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAttachSave() +{ + KMMessagePart* msgPart; + QString fileName, pname; + int idx = currentAttachmentNum(); + + if (idx < 0) return; + + msgPart = mAtmList.at(idx); + pname = msgPart->name(); + if (pname.isEmpty()) pname="unnamed"; + + KURL url = KFileDialog::getSaveURL(QString::null, QString::null, 0, i18n("Save Attachment As")); + + if( url.isEmpty() ) + return; + + kmkernel->byteArrayToRemoteFile(msgPart->bodyDecodedBinary(), url); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAttachRemove() +{ + bool attachmentRemoved = false; + int i = 0; + for ( QPtrListIterator<QListViewItem> it(mAtmItemList); *it; ) { + if ( (*it)->isSelected() ) { + removeAttach( i ); + attachmentRemoved = true; + } + else { + ++it; + ++i; + } + } + + if ( attachmentRemoved ) { + setModified( true ); + slotUpdateAttachActions(); + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotFind() +{ + mEditor->search(); +} + +void KMComposeWin::slotSearchAgain() +{ + mEditor->repeatSearch(); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotReplace() +{ + mEditor->replace(); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotUpdateFont() +{ + kdDebug() << "KMComposeWin::slotUpdateFont " << endl; + if ( ! mFixedFontAction ) { + return; + } + mEditor->setFont( mFixedFontAction->isChecked() ? mFixedFont : mBodyFont ); +} + +QString KMComposeWin::quotePrefixName() const +{ + if ( !msg() ) + return QString::null; + + int languageNr = GlobalSettings::self()->replyCurrentLanguage(); + ReplyPhrases replyPhrases( QString::number(languageNr) ); + replyPhrases.readConfig(); + QString quotePrefix = msg()->formatString( + replyPhrases.indentPrefix() ); + + quotePrefix = msg()->formatString(quotePrefix); + return quotePrefix; +} + +void KMComposeWin::slotPasteClipboardAsQuotation() +{ + if( mEditor->hasFocus() && msg() ) + { + QString s = QApplication::clipboard()->text(); + if (!s.isEmpty()) + mEditor->insert(addQuotesToText(s)); + } +} + +void KMComposeWin::slotPasteClipboardAsAttachment() +{ + KURL url( QApplication::clipboard()->text( QClipboard::Clipboard ) ); + if ( url.isValid() ) { + addAttach(QApplication::clipboard()->text( QClipboard::Clipboard ) ); + return; + } + + QMimeSource *mimeSource = QApplication::clipboard()->data(); + if ( QImageDrag::canDecode(mimeSource) ) { + slotAttachPNGImageData(mimeSource->encodedData("image/png")); + } + else { + bool ok; + QString attName = KInputDialog::getText( "KMail", i18n("Name of the attachment:"), QString::null, &ok, this ); + if ( !ok ) + return; + KMMessagePart *msgPart = new KMMessagePart; + msgPart->setName(attName); + QValueList<int> dummy; + msgPart->setBodyAndGuessCte(QCString(QApplication::clipboard()->text().latin1()), dummy, + kmkernel->msgSender()->sendQuotedPrintable()); + addAttach(msgPart); + } +} + +void KMComposeWin::slotAddQuotes() +{ + if( mEditor->hasFocus() && msg() ) + { + // TODO: I think this is backwards. + // i.e, if no region is marked then add quotes to every line + // else add quotes only on the lines that are marked. + + if ( mEditor->hasMarkedText() ) { + QString s = mEditor->markedText(); + if(!s.isEmpty()) + mEditor->insert(addQuotesToText(s)); + } else { + int l = mEditor->currentLine(); + int c = mEditor->currentColumn(); + QString s = mEditor->textLine(l); + s.prepend(quotePrefixName()); + mEditor->insertLine(s,l); + mEditor->removeLine(l+1); + mEditor->setCursorPosition(l,c+2); + } + } +} + +QString KMComposeWin::addQuotesToText(const QString &inputText) +{ + QString answer = QString( inputText ); + QString indentStr = quotePrefixName(); + answer.replace( '\n', '\n' + indentStr); + answer.prepend( indentStr ); + answer += '\n'; + return KMMessage::smartQuote( answer, GlobalSettings::self()->lineWrapWidth() ); +} + +QString KMComposeWin::removeQuotesFromText(const QString &inputText) +{ + QString s = inputText; + + // remove first leading quote + QString quotePrefix = '^' + quotePrefixName(); + QRegExp rx(quotePrefix); + s.remove(rx); + + // now remove all remaining leading quotes + quotePrefix = '\n' + quotePrefixName(); + rx = quotePrefix; + s.replace(rx, "\n"); + + return s; +} + +void KMComposeWin::slotRemoveQuotes() +{ + if( mEditor->hasFocus() && msg() ) + { + // TODO: I think this is backwards. + // i.e, if no region is marked then remove quotes from every line + // else remove quotes only on the lines that are marked. + + if ( mEditor->hasMarkedText() ) { + QString s = mEditor->markedText(); + mEditor->insert(removeQuotesFromText(s)); + } else { + int l = mEditor->currentLine(); + int c = mEditor->currentColumn(); + QString s = mEditor->textLine(l); + mEditor->insertLine(removeQuotesFromText(s),l); + mEditor->removeLine(l+1); + mEditor->setCursorPosition(l,c-2); + } + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotUndo() +{ + QWidget* fw = focusWidget(); + if (!fw) return; + + if ( ::qt_cast<KEdit*>(fw) ) + static_cast<QTextEdit*>(fw)->undo(); + else if (::qt_cast<QLineEdit*>(fw)) + static_cast<QLineEdit*>(fw)->undo(); +} + +void KMComposeWin::slotRedo() +{ + QWidget* fw = focusWidget(); + if (!fw) return; + + if (::qt_cast<KEdit*>(fw)) + static_cast<KEdit*>(fw)->redo(); + else if (::qt_cast<QLineEdit*>(fw)) + static_cast<QLineEdit*>(fw)->redo(); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotCut() +{ + QWidget* fw = focusWidget(); + if (!fw) return; + + if (::qt_cast<KEdit*>(fw)) + static_cast<KEdit*>(fw)->cut(); + else if (::qt_cast<QLineEdit*>(fw)) + static_cast<QLineEdit*>(fw)->cut(); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotCopy() +{ + QWidget* fw = focusWidget(); + if (!fw) return; + +#ifdef KeyPress +#undef KeyPress +#endif + + QKeyEvent k(QEvent::KeyPress, Key_C, 0, ControlButton); + kapp->notify(fw, &k); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotPasteClipboard() +{ + paste( QClipboard::Clipboard ); +} + +void KMComposeWin::paste( QClipboard::Mode mode ) +{ + QWidget* fw = focusWidget(); + if (!fw) return; + + QMimeSource *mimeSource = QApplication::clipboard()->data( mode ); + if ( mimeSource->provides("image/png") ) { + slotAttachPNGImageData(mimeSource->encodedData("image/png")); + } else if ( KURLDrag::canDecode( mimeSource ) ) { + KURL::List urlList; + if( KURLDrag::decode( mimeSource, urlList ) ) { + const QString asText = i18n("Add as Text"); + const QString asAttachment = i18n("Add as Attachment"); + const QString text = i18n("Please select whether you want to insert the content as text into the editor, " + "or append the referenced file as an attachment."); + const QString caption = i18n("Paste as text or attachment?"); + + int id = KMessageBox::questionYesNoCancel( this, text, caption, + KGuiItem( asText ), KGuiItem( asAttachment) ); + switch ( id) { + case KMessageBox::Yes: + for ( KURL::List::Iterator it = urlList.begin(); + it != urlList.end(); ++it ) { + mEditor->insert( (*it).url() ); + } + break; + case KMessageBox::No: + for ( KURL::List::Iterator it = urlList.begin(); + it != urlList.end(); ++it ) { + addAttach( *it ); + } + break; + } + } + } else if ( QTextDrag::canDecode( mimeSource ) ) { + QString s; + if ( QTextDrag::decode( mimeSource, s ) ) + mEditor->insert( s ); + } +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotMarkAll() +{ + QWidget* fw = focusWidget(); + if (!fw) return; + + if (::qt_cast<QLineEdit*>(fw)) + static_cast<QLineEdit*>(fw)->selectAll(); + else if (::qt_cast<KEdit*>(fw)) + static_cast<KEdit*>(fw)->selectAll(); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotClose() +{ + close(false); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotNewComposer() +{ + KMComposeWin* win; + KMMessage* msg = new KMMessage; + + msg->initHeader(); + win = new KMComposeWin(msg); + win->show(); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotNewMailReader() +{ + KMMainWin *kmmwin = new KMMainWin(0); + kmmwin->show(); + //d->resize(d->size()); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotUpdWinTitle(const QString& text) +{ + QString s( text ); + // Remove characters that show badly in most window decorations: + // newlines tend to become boxes. + if (text.isEmpty()) + setCaption("("+i18n("unnamed")+")"); + else setCaption( s.replace( QChar('\n'), ' ' ) ); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotEncryptToggled(bool on) +{ + setEncryption( on, true /* set by the user */ ); + slotUpdateSignatureAndEncrypionStateIndicators(); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::setEncryption( bool encrypt, bool setByUser ) +{ + if ( setByUser ) + setModified( true ); + if ( !mEncryptAction->isEnabled() ) + encrypt = false; + // check if the user wants to encrypt messages to himself and if he defined + // an encryption key for the current identity + else if ( encrypt && encryptToSelf() && !mLastIdentityHasEncryptionKey ) { + if ( setByUser ) + KMessageBox::sorry( this, + i18n("<qt><p>You have requested that messages be " + "encrypted to yourself, but the currently selected " + "identity does not define an (OpenPGP or S/MIME) " + "encryption key to use for this.</p>" + "<p>Please select the key(s) to use " + "in the identity configuration.</p>" + "</qt>"), + i18n("Undefined Encryption Key") ); + encrypt = false; + } + + // make sure the mEncryptAction is in the right state + mEncryptAction->setChecked( encrypt ); + + // show the appropriate icon + if ( encrypt ) + mEncryptAction->setIcon("encrypted"); + else + mEncryptAction->setIcon("decrypted"); + + // mark the attachments for (no) encryption + if ( canSignEncryptAttachments() ) { + for ( KMAtmListViewItem* entry = + static_cast<KMAtmListViewItem*>( mAtmItemList.first() ); + entry; + entry = static_cast<KMAtmListViewItem*>( mAtmItemList.next() ) ) + entry->setEncrypt( encrypt ); + } +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotSignToggled(bool on) +{ + setSigning( on, true /* set by the user */ ); + slotUpdateSignatureAndEncrypionStateIndicators(); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::setSigning( bool sign, bool setByUser ) +{ + if ( setByUser ) + setModified( true ); + if ( !mSignAction->isEnabled() ) + sign = false; + + // check if the user defined a signing key for the current identity + if ( sign && !mLastIdentityHasSigningKey ) { + if ( setByUser ) + KMessageBox::sorry( this, + i18n("<qt><p>In order to be able to sign " + "this message you first have to " + "define the (OpenPGP or S/MIME) signing key " + "to use.</p>" + "<p>Please select the key to use " + "in the identity configuration.</p>" + "</qt>"), + i18n("Undefined Signing Key") ); + sign = false; + } + + // make sure the mSignAction is in the right state + mSignAction->setChecked( sign ); + + // mark the attachments for (no) signing + if ( canSignEncryptAttachments() ) { + for ( KMAtmListViewItem* entry = + static_cast<KMAtmListViewItem*>( mAtmItemList.first() ); + entry; + entry = static_cast<KMAtmListViewItem*>( mAtmItemList.next() ) ) + entry->setSign( sign ); + } +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotWordWrapToggled(bool on) +{ + if (on) + { + mEditor->setWordWrap( QTextEdit::FixedColumnWidth ); + mEditor->setWrapColumnOrWidth( GlobalSettings::self()->lineWrapWidth() ); + } + else + { + mEditor->setWordWrap( QTextEdit::WidgetWidth ); + } +} + + +void KMComposeWin::disableWordWrap() +{ + mEditor->setWordWrap( QTextEdit::NoWrap ); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotPrint() +{ + mMessageWasModified = isModified(); + connect( this, SIGNAL( applyChangesDone( bool ) ), + this, SLOT( slotContinuePrint( bool ) ) ); + applyChanges( true ); +} + +void KMComposeWin::slotContinuePrint( bool rc ) +{ + disconnect( this, SIGNAL( applyChangesDone( bool ) ), + this, SLOT( slotContinuePrint( bool ) ) ); + + if( rc ) { + if ( mComposedMessages.isEmpty() ) { + kdDebug(5006) << "Composing the message failed." << endl; + return; + } + KMCommand *command = new KMPrintCommand( this, mComposedMessages.first() ); + command->start(); + setModified( mMessageWasModified ); + } +} + +//---------------------------------------------------------------------------- +bool KMComposeWin::validateAddresses( QWidget * parent, const QString & addresses ) +{ + QString brokenAddress; + KPIM::EmailParseResult errorCode = KMMessage::isValidEmailAddressList( KMMessage::expandAliases( addresses ), brokenAddress ); + if ( !( errorCode == KPIM::AddressOk || errorCode == KPIM::AddressEmpty ) ) { + QString errorMsg( "<qt><p><b>" + brokenAddress + + "</b></p><p>" + KPIM::emailParseResultToString( errorCode ) + + "</p></qt>" ); + KMessageBox::sorry( parent, errorMsg, i18n("Invalid Email Address") ); + return false; + } + return true; +} + +//---------------------------------------------------------------------------- +void KMComposeWin::doSend( KMail::MessageSender::SendMethod method, + KMComposeWin::SaveIn saveIn ) +{ + if ( method != KMail::MessageSender::SendLater && kmkernel->isOffline() ) { + KMessageBox::information( this, + i18n("KMail is currently in offline mode," + "your messages will be kept in the outbox until you go online."), + i18n("Online/Offline"), "kmailIsOffline" ); + mSendMethod = KMail::MessageSender::SendLater; + } else { + mSendMethod = method; + } + mSaveIn = saveIn; + + if ( saveIn == KMComposeWin::None ) { + if ( KPIM::getFirstEmailAddress( from() ).isEmpty() ) { + if ( !( mShowHeaders & HDR_FROM ) ) { + mShowHeaders |= HDR_FROM; + rethinkFields( false ); + } + mEdtFrom->setFocus(); + KMessageBox::sorry( this, + i18n("You must enter your email address in the " + "From: field. You should also set your email " + "address for all identities, so that you do " + "not have to enter it for each message.") ); + return; + } + if ( to().isEmpty() ) + { + if ( cc().isEmpty() && bcc().isEmpty()) { + if ( mEdtTo ) mEdtTo->setFocus(); + KMessageBox::information( this, + i18n("You must specify at least one receiver," + "either in the To: field or as CC or as BCC.") ); + return; + } + else { + if ( mEdtTo ) mEdtTo->setFocus(); + int rc = + KMessageBox::questionYesNo( this, + i18n("To field is missing." + "Send message anyway?"), + i18n("No To: specified") ); + if ( rc == KMessageBox::No ){ + return; + } + } + } + + // Validate the To:, CC: and BCC fields + if ( !validateAddresses( this, to().stripWhiteSpace() ) ) { + return; + } + + if ( !validateAddresses( this, cc().stripWhiteSpace() ) ) { + return; + } + + if ( !validateAddresses( this, bcc().stripWhiteSpace() ) ) { + return; + } + + if (subject().isEmpty()) + { + mEdtSubject->setFocus(); + int rc = + KMessageBox::questionYesNo( this, + i18n("You did not specify a subject. " + "Send message anyway?"), + i18n("No Subject Specified"), + i18n("S&end as Is"), + i18n("&Specify the Subject"), + "no_subject_specified" ); + if( rc == KMessageBox::No ) + { + return; + } + } + + if ( userForgotAttachment() ) + return; + } + + KCursorSaver busy(KBusyPtr::busy()); + mMsg->setDateToday(); + + // If a user sets up their outgoing messages preferences wrong and then + // sends mail that gets 'stuck' in their outbox, they should be able to + // rectify the problem by editing their outgoing preferences and + // resending. + // Hence this following conditional + QString hf = mMsg->headerField("X-KMail-Transport"); + if ((mTransport->currentText() != mTransport->text(0)) || + (!hf.isEmpty() && (hf != mTransport->text(0)))) + mMsg->setHeaderField("X-KMail-Transport", mTransport->currentText()); + + mDisableBreaking = ( saveIn != KMComposeWin::None ); + + const bool neverEncrypt = ( mDisableBreaking && GlobalSettings::self()->neverEncryptDrafts() ) + || mSigningAndEncryptionExplicitlyDisabled; + connect( this, SIGNAL( applyChangesDone( bool ) ), + SLOT( slotContinueDoSend( bool ) ) ); + + if ( mEditor->textFormat() == Qt::RichText ) + mMsg->setHeaderField( "X-KMail-Markup", "true" ); + else + mMsg->removeHeaderField( "X-KMail-Markup" ); + if ( mEditor->textFormat() == Qt::RichText && inlineSigningEncryptionSelected() ) { + QString keepBtnText = mEncryptAction->isChecked() ? + mSignAction->isChecked() ? i18n( "&Keep markup, do not sign/encrypt" ) + : i18n( "&Keep markup, do not encrypt" ) + : i18n( "&Keep markup, do not sign" ); + QString yesBtnText = mEncryptAction->isChecked() ? + mSignAction->isChecked() ? i18n("Sign/Encrypt (delete markup)") + : i18n( "Encrypt (delete markup)" ) + : i18n( "Sign (delete markup)" ); + int ret = KMessageBox::warningYesNoCancel(this, + i18n("<qt><p>Inline signing/encrypting of HTML messages is not possible;</p>" + "<p>do you want to delete your markup?</p></qt>"), + i18n("Sign/Encrypt Message?"), + KGuiItem( yesBtnText ), + KGuiItem( keepBtnText ) ); + if ( KMessageBox::Cancel == ret ) + return; + if ( KMessageBox::No == ret ) { + mEncryptAction->setChecked(false); + mSignAction->setChecked(false); + } + else { + toggleMarkup(false); + } + } + + if (neverEncrypt && saveIn != KMComposeWin::None ) { + // we can't use the state of the mail itself, to remember the + // signing and encryption state, so let's add a header instead + mMsg->setHeaderField( "X-KMail-SignatureActionEnabled", mSignAction->isChecked()? "true":"false" ); + mMsg->setHeaderField( "X-KMail-EncryptActionEnabled", mEncryptAction->isChecked()? "true":"false" ); + mMsg->setHeaderField( "X-KMail-CryptoMessageFormat", QString::number( cryptoMessageFormat() ) ); + } else { + mMsg->removeHeaderField( "X-KMail-SignatureActionEnabled" ); + mMsg->removeHeaderField( "X-KMail-EncryptActionEnabled" ); + mMsg->removeHeaderField( "X-KMail-CryptoMessageFormat" ); + } + + + kdDebug(5006) << "KMComposeWin::doSend() - calling applyChanges()" + << endl; + applyChanges( neverEncrypt ); +} + +bool KMComposeWin::saveDraftOrTemplate( const QString &folderName, + KMMessage *msg ) +{ + KMFolder *theFolder = 0, *imapTheFolder = 0; + // get the draftsFolder + if ( !folderName.isEmpty() ) { + theFolder = kmkernel->folderMgr()->findIdString( folderName ); + if ( theFolder == 0 ) + // This is *NOT* supposed to be "imapDraftsFolder", because a + // dIMAP folder works like a normal folder + theFolder = kmkernel->dimapFolderMgr()->findIdString( folderName ); + if ( theFolder == 0 ) + imapTheFolder = kmkernel->imapFolderMgr()->findIdString( folderName ); + if ( !theFolder && !imapTheFolder ) { + const KPIM::Identity & id = kmkernel->identityManager() + ->identityForUoidOrDefault( msg->headerField( "X-KMail-Identity" ).stripWhiteSpace().toUInt() ); + KMessageBox::information( 0, + i18n("The custom drafts or templates folder for " + "identify \"%1\" does not exist (anymore); " + "therefore, the default drafts or templates " + "folder will be used.") + .arg( id.identityName() ) ); + } + } + if ( imapTheFolder && imapTheFolder->noContent() ) + imapTheFolder = 0; + + bool didOpen = false; + if ( theFolder == 0 ) { + theFolder = ( mSaveIn==KMComposeWin::Drafts ? + kmkernel->draftsFolder() : kmkernel->templatesFolder() ); + } else { + //XXX this looks really, really fishy + theFolder->open( "composer" ); + didOpen = true; + } + kdDebug(5006) << k_funcinfo << "theFolder=" << theFolder->name() << endl; + if ( imapTheFolder ) + kdDebug(5006) << k_funcinfo << "imapTheFolder=" << imapTheFolder->name() << endl; + + bool sentOk = !( theFolder->addMsg( msg ) ); + + // Ensure the message is correctly and fully parsed + theFolder->unGetMsg( theFolder->count() - 1 ); + msg = theFolder->getMsg( theFolder->count() - 1 ); + // Does that assignment needs to be propagated out to the caller? + // Assuming the send is OK, the iterator is set to 0 immediately afterwards. + if ( imapTheFolder ) { + // move the message to the imap-folder and highlight it + imapTheFolder->moveMsg( msg ); + (static_cast<KMFolderImap*>( imapTheFolder->storage() ))->getFolder(); + } + + if ( didOpen ) + theFolder->close( "composer" ); + return sentOk; +} + +void KMComposeWin::slotContinueDoSend( bool sentOk ) +{ + kdDebug(5006) << "KMComposeWin::slotContinueDoSend( " << sentOk << " )" + << endl; + disconnect( this, SIGNAL( applyChangesDone( bool ) ), + this, SLOT( slotContinueDoSend( bool ) ) ); + + if ( !sentOk ) { + mDisableBreaking = false; + return; + } + + for ( QValueVector<KMMessage*>::iterator it = mComposedMessages.begin() ; it != mComposedMessages.end() ; ++it ) { + + // remove fields that contain no data (e.g. an empty Cc: or Bcc:) + (*it)->cleanupHeader(); + + // needed for imap + (*it)->setComplete( true ); + + if ( mSaveIn==KMComposeWin::Drafts ) { + sentOk = saveDraftOrTemplate( (*it)->drafts(), (*it) ); + } else if ( mSaveIn==KMComposeWin::Templates ) { + sentOk = saveDraftOrTemplate( (*it)->templates(), (*it) ); + } else { + (*it)->setTo( KMMessage::expandAliases( to() )); + (*it)->setCc( KMMessage::expandAliases( cc() )); + if( !mComposer->originalBCC().isEmpty() ) + (*it)->setBcc( KMMessage::expandAliases( mComposer->originalBCC() )); + QString recips = (*it)->headerField( "X-KMail-Recipients" ); + if( !recips.isEmpty() ) { + (*it)->setHeaderField( "X-KMail-Recipients", KMMessage::expandAliases( recips ), KMMessage::Address ); + } + (*it)->cleanupHeader(); + sentOk = kmkernel->msgSender()->send((*it), mSendMethod); + } + + if (!sentOk) + return; + + *it = 0; // don't kill it later... + } + + RecentAddresses::self( KMKernel::config() )->add( bcc() ); + RecentAddresses::self( KMKernel::config() )->add( cc() ); + RecentAddresses::self( KMKernel::config() )->add( to() ); + + setModified( false ); + mAutoDeleteMsg = false; + mFolder = 0; + cleanupAutoSave(); + close(); + return; +} + + + +//---------------------------------------------------------------------------- +void KMComposeWin::slotSendLater() +{ + if ( mEditor->checkExternalEditorFinished() ) + doSend( KMail::MessageSender::SendLater ); +} + + +//---------------------------------------------------------------------------- +void KMComposeWin::slotSaveDraft() { + if ( mEditor->checkExternalEditorFinished() ) + doSend( KMail::MessageSender::SendLater, KMComposeWin::Drafts ); +} + +//---------------------------------------------------------------------------- +void KMComposeWin::slotSaveTemplate() { + if ( mEditor->checkExternalEditorFinished() ) + doSend( KMail::MessageSender::SendLater, KMComposeWin::Templates ); +} + +//---------------------------------------------------------------------------- +void KMComposeWin::slotSendNowVia( int item ) +{ + QStringList availTransports= KMail::TransportManager::transportNames(); + QString customTransport = availTransports[ item ]; + + mTransport->setCurrentText( customTransport ); + slotSendNow(); +} + +//---------------------------------------------------------------------------- +void KMComposeWin::slotSendLaterVia( int item ) +{ + QStringList availTransports= KMail::TransportManager::transportNames(); + QString customTransport = availTransports[ item ]; + + mTransport->setCurrentText( customTransport ); + slotSendLater(); +} + + +//---------------------------------------------------------------------------- +void KMComposeWin::slotSendNow() { + if ( !mEditor->checkExternalEditorFinished() ) + return; + if ( GlobalSettings::self()->confirmBeforeSend() ) + { + int rc = KMessageBox::warningYesNoCancel( mMainWidget, + i18n("About to send email..."), + i18n("Send Confirmation"), + i18n("&Send Now"), + i18n("Send &Later") ); + + if ( rc == KMessageBox::Yes ) + doSend( KMail::MessageSender::SendImmediate ); + else if ( rc == KMessageBox::No ) + doSend( KMail::MessageSender::SendLater ); + } + else + doSend( KMail::MessageSender::SendImmediate ); +} + +//---------------------------------------------------------------------------- +void KMComposeWin::slotAppendSignature() +{ + insertSignature(); +} + +//---------------------------------------------------------------------------- +void KMComposeWin::slotPrependSignature() +{ + insertSignature( false ); +} + +//---------------------------------------------------------------------------- +void KMComposeWin::slotInsertSignatureAtCursor() +{ + insertSignature( false, mEditor->currentLine() ); +} + +//---------------------------------------------------------------------------- +void KMComposeWin::insertSignature( bool append, int pos ) +{ + bool mod = mEditor->isModified(); + + const KPIM::Identity &ident = + kmkernel->identityManager()-> + identityForUoidOrDefault( mIdentity->currentIdentity() ); + + mOldSigText = GlobalSettings::self()->prependSignature()? ident.signature().rawText() : ident.signatureText(); + + if( !mOldSigText.isEmpty() ) + { + mEditor->sync(); + if ( append ) { + mEditor->setText( mEditor->text() + mOldSigText ); + } else { + mOldSigText = "\n\n"+mOldSigText+"\n"; + mEditor->insertAt(mOldSigText, pos, 0); + } + mEditor->update(); + mEditor->setModified(mod); + + if ( mPreserveUserCursorPosition ) { + mEditor->setCursorPositionFromStart( (unsigned int) mMsg->getCursorPos() ); + // Only keep the cursor from the mMsg *once* based on the + // preserve-cursor-position setting; this handles the case where + // the message comes from a template with a specific cursor + // position set and the signature is appended automatically. + mPreserveUserCursorPosition = false; + } else { + // for append and prepend, move the cursor to 0,0, for insertAt, + // keep it in the same row, but move to first column + mEditor->setCursorPosition( pos, 0 ); + if ( !append && pos == 0 ) + mEditor->setContentsPos( 0, 0 ); + } + mEditor->sync(); + } +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotHelp() +{ + kapp->invokeHelp(); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotCleanSpace() +{ + // Originally we simply used the KEdit::cleanWhiteSpace() method, + // but that code doesn't handle quoted-lines or signatures, so instead + // we now simply use regexp's to squeeze sequences of tabs and spaces + // into a single space, and make sure all our lines are single-spaced. + // + // Yes, extra space in a quote string is squeezed. + // Signatures are respected (i.e. not cleaned). + + QString s; + if ( mEditor->hasMarkedText() ) { + s = mEditor->markedText(); + if( s.isEmpty() ) + return; + } else { + s = mEditor->text(); + } + + // Remove the signature for now. + QString sig; + bool restore = false; + const KPIM::Identity & ident = + kmkernel->identityManager()->identityForUoid( mId ); + if ( !ident.isNull() ) { + sig = ident.signatureText(); + if( !sig.isEmpty() ) { + if( s.endsWith( sig ) ) { + s.truncate( s.length() - sig.length() ); + restore = true; + } + } + } + + // Squeeze tabs and spaces + QRegExp squeeze( "[\t ]+" ); + s.replace( squeeze, QChar( ' ' ) ); + + // Remove trailing whitespace + QRegExp trailing( "\\s+$" ); + s.replace( trailing, QChar( '\n' ) ); + + // Single space lines + QRegExp singleSpace( "[\n]{2,}" ); + s.replace( singleSpace, QChar( '\n' ) ); + + // Restore the signature + if ( restore ) + s.append( sig ); + + // Put the new text in place. + // The lines below do not clear the undo history, but unfortuately cause + // the side-effect that you need to press Ctrl-Z twice (first Ctrl-Z will + // show cleared text area) to get back the original, pre-cleaned text. + // If you use mEditor->setText( s ) then the undo history is cleared so + // that isn't a good solution either. + // TODO: is Qt4 better at handling the undo history?? + if ( !mEditor->hasMarkedText() ) + mEditor->clear(); + mEditor->insert( s ); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotToggleMarkup() +{ + if ( markupAction->isChecked() ) { + mHtmlMarkup = true; + toolBar("htmlToolBar")->show(); + // markup will be toggled as soon as markup is actually used + fontChanged( mEditor->currentFont() ); // set buttons in correct position + mSaveFont = mEditor->currentFont(); + } + else + toggleMarkup(false); + +} +//----------------------------------------------------------------------------- +void KMComposeWin::toggleMarkup(bool markup) +{ + if ( markup ) { + if ( !mUseHTMLEditor ) { + kdDebug(5006) << "setting RichText editor" << endl; + mUseHTMLEditor = true; // set it directly to true. setColor hits another toggleMarkup + mHtmlMarkup = true; + + // set all highlighted text caused by spelling back to black + int paraFrom, indexFrom, paraTo, indexTo; + mEditor->getSelection ( ¶From, &indexFrom, ¶To, &indexTo); + mEditor->selectAll(); + // save the buttonstates because setColor calls fontChanged + bool _bold = textBoldAction->isChecked(); + bool _italic = textItalicAction->isChecked(); + mEditor->setColor(QColor(0,0,0)); + textBoldAction->setChecked(_bold); + textItalicAction->setChecked(_italic); + mEditor->setSelection ( paraFrom, indexFrom, paraTo, indexTo); + + mEditor->setTextFormat(Qt::RichText); + mEditor->setModified(true); + markupAction->setChecked(true); + toolBar( "htmlToolBar" )->show(); + mEditor->deleteAutoSpellChecking(); + mAutoSpellCheckingAction->setChecked(false); + slotAutoSpellCheckingToggled(false); + } + } else { // markup is to be turned off + kdDebug(5006) << "setting PlainText editor" << endl; + mHtmlMarkup = false; + toolBar("htmlToolBar")->hide(); + if ( mUseHTMLEditor ) { // it was turned on + mUseHTMLEditor = false; + mEditor->setTextFormat(Qt::PlainText); + QString text = mEditor->text(); + mEditor->setText(text); // otherwise the text still looks formatted + mEditor->setModified(true); + slotAutoSpellCheckingToggled(true); + } + } +} + +void KMComposeWin::htmlToolBarVisibilityChanged( bool visible ) +{ + // disable markup if the user hides the HTML toolbar + if ( !visible ) { + markupAction->setChecked( false ); + toggleMarkup( false ); + } +} + +void KMComposeWin::slotSubjectTextSpellChecked() +{ + mSubjectTextWasSpellChecked = true; +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotAutoSpellCheckingToggled( bool on ) +{ + if ( mEditor->autoSpellChecking(on) == -1 ) { + mAutoSpellCheckingAction->setChecked(false); // set it to false again + } + + QString temp; + if ( on ) + temp = i18n( "Spellcheck: on" ); + else + temp = i18n( "Spellcheck: off" ); + statusBar()->changeItem( temp, 3 ); +} +//----------------------------------------------------------------------------- +void KMComposeWin::slotSpellcheck() +{ + if (mSpellCheckInProgress) return; + mSubjectTextWasSpellChecked = false; + mSpellCheckInProgress=true; + /* + connect (mEditor, SIGNAL (spellcheck_progress (unsigned)), + this, SLOT (spell_progress (unsigned))); + */ + + mEditor->spellcheck(); +} +//----------------------------------------------------------------------------- +void KMComposeWin::slotUpdateSignatureActions() +{ + //Check if an identity has signature or not and turn on/off actions in the + //edit menu accordingly. + const KPIM::Identity & ident = + kmkernel->identityManager()->identityForUoidOrDefault( mIdentity->currentIdentity() ); + QString sig = ident.signatureText(); + + if ( sig.isEmpty() ) { + mAppendSignatureAction->setEnabled( false ); + mPrependSignatureAction->setEnabled( false ); + mInsertSignatureAction->setEnabled( false ); + } + else { + mAppendSignatureAction->setEnabled( true ); + mPrependSignatureAction->setEnabled( true ); + mInsertSignatureAction->setEnabled( true ); + } +} + +void KMComposeWin::polish() +{ + // Ensure the html toolbar is appropriately shown/hidden + markupAction->setChecked(mHtmlMarkup); + if (mHtmlMarkup) + toolBar("htmlToolBar")->show(); + else + toolBar("htmlToolBar")->hide(); + KMail::Composer::polish(); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotSpellcheckDone(int result) +{ + kdDebug(5006) << "spell check complete: result = " << result << endl; + mSpellCheckInProgress=false; + + switch( result ) + { + case KS_CANCEL: + statusBar()->changeItem(i18n(" Spell check canceled."),0); + break; + case KS_STOP: + statusBar()->changeItem(i18n(" Spell check stopped."),0); + break; + default: + statusBar()->changeItem(i18n(" Spell check complete."),0); + break; + } + QTimer::singleShot( 2000, this, SLOT(slotSpellcheckDoneClearStatus()) ); +} + +void KMComposeWin::slotSpellcheckDoneClearStatus() +{ + statusBar()->changeItem("", 0); +} + + +//----------------------------------------------------------------------------- +void KMComposeWin::slotIdentityChanged( uint uoid ) +{ + const KPIM::Identity & ident = + kmkernel->identityManager()->identityForUoid( uoid ); + if( ident.isNull() ) return; + + //Turn on/off signature actions if identity has no signature. + slotUpdateSignatureActions(); + + if( !ident.fullEmailAddr().isNull() ) + mEdtFrom->setText(ident.fullEmailAddr()); + // make sure the From field is shown if it does not contain a valid email address + if ( KPIM::getFirstEmailAddress( from() ).isEmpty() ) + mShowHeaders |= HDR_FROM; + if ( mEdtReplyTo ) mEdtReplyTo->setText(ident.replyToAddr()); + + if ( mRecipientsEditor ) { + // remove BCC of old identity and add BCC of new identity (if they differ) + const KPIM::Identity & oldIdentity = + kmkernel->identityManager()->identityForUoidOrDefault( mId ); + if ( oldIdentity.bcc() != ident.bcc() ) { + mRecipientsEditor->removeRecipient( oldIdentity.bcc(), Recipient::Bcc ); + mRecipientsEditor->addRecipient( ident.bcc(), Recipient::Bcc ); + mRecipientsEditor->setFocusBottom(); + } + } + + // don't overwrite the BCC field under certain circomstances + // NOT edited and preset BCC from the identity + if( mEdtBcc && !mEdtBcc->edited() && !ident.bcc().isEmpty() ) { + // BCC NOT empty AND contains a diff adress then the preset BCC + // of the new identity + if( !mEdtBcc->text().isEmpty() && mEdtBcc->text() != ident.bcc() && !mEdtBcc->edited() ) { + mEdtBcc->setText( ident.bcc() ); + } else { + // user type into the editbox an address that != to the preset bcc + // of the identity, we assume that since the user typed it + // they want to keep it + if ( mEdtBcc->text() != ident.bcc() && !mEdtBcc->text().isEmpty() ) { + QString temp_string( mEdtBcc->text() + QString::fromLatin1(",") + ident.bcc() ); + mEdtBcc->setText( temp_string ); + } else { + // if the user typed the same address as the preset BCC + // from the identity we will overwrite it to avoid duplicates. + mEdtBcc->setText( ident.bcc() ); + } + } + } + // user edited the bcc box and has a preset bcc in the identity + // we will append whatever the user typed to the preset address + // allowing the user to keep all addresses + if( mEdtBcc && mEdtBcc->edited() && !ident.bcc().isEmpty() ) { + if( !mEdtBcc->text().isEmpty() ) { + QString temp_string ( mEdtBcc->text() + QString::fromLatin1(",") + ident.bcc() ); + mEdtBcc->setText( temp_string ); + } else { + mEdtBcc->setText( ident.bcc() ); + } + } + // user typed nothing and the identity does not have a preset bcc + // we then reset the value to get rid of any previous + // values if the user changed identity mid way through. + if( mEdtBcc && !mEdtBcc->edited() && ident.bcc().isEmpty() ) { + mEdtBcc->setText( ident.bcc() ); + } + // make sure the BCC field is shown because else it's ignored + if ( !ident.bcc().isEmpty() ) { + mShowHeaders |= HDR_BCC; + } + + if ( ident.organization().isEmpty() ) + mMsg->removeHeaderField("Organization"); + else + mMsg->setHeaderField("Organization", ident.organization()); + + if (!ident.isXFaceEnabled() || ident.xface().isEmpty()) + mMsg->removeHeaderField("X-Face"); + else + { + QString xface = ident.xface(); + if (!xface.isEmpty()) + { + int numNL = ( xface.length() - 1 ) / 70; + for ( int i = numNL; i > 0; --i ) + xface.insert( i*70, "\n\t" ); + mMsg->setHeaderField("X-Face", xface); + } + } + + if ( !mBtnTransport->isChecked() ) { + QString transp = ident.transport(); + if ( transp.isEmpty() ) + { + mMsg->removeHeaderField("X-KMail-Transport"); + transp = GlobalSettings::self()->defaultTransport(); + } + else + mMsg->setHeaderField("X-KMail-Transport", transp); + setTransport( transp ); + } + + mDictionaryCombo->setCurrentByDictionary( ident.dictionary() ); + + if ( !mBtnFcc->isChecked() ) { + setFcc( ident.fcc() ); + } + + QString edtText = mEditor->text(); + + if ( mOldSigText.isEmpty() ) { + const KPIM::Identity &id = + kmkernel-> + identityManager()-> + identityForUoidOrDefault( mMsg->headerField( "X-KMail-Identity" ). + stripWhiteSpace().toUInt() ); + mOldSigText = id.signatureText(); + } + + // try to truncate the old sig + // First remove any trailing whitespace + while ( !edtText.isEmpty() && edtText[edtText.length()-1].isSpace() ) + edtText.truncate( edtText.length() - 1 ); + // From the sig too, just in case + while ( !mOldSigText.isEmpty() && mOldSigText[mOldSigText.length()-1].isSpace() ) + mOldSigText.truncate( mOldSigText.length() - 1 ); + + if( edtText.endsWith( mOldSigText ) ) + edtText.truncate( edtText.length() - mOldSigText.length() ); + + // now append the new sig + mOldSigText = ident.signatureText(); + if( ( !mOldSigText.isEmpty() ) && + ( GlobalSettings::self()->autoTextSignature() == "auto" ) ) { + edtText.append( mOldSigText ); + } + mEditor->setText( edtText ); + + // disable certain actions if there is no PGP user identity set + // for this profile + bool bNewIdentityHasSigningKey = !ident.pgpSigningKey().isEmpty() || !ident.smimeSigningKey().isEmpty(); + bool bNewIdentityHasEncryptionKey = !ident.pgpSigningKey().isEmpty() || !ident.smimeSigningKey().isEmpty(); + mAttachMPK->setEnabled( Kleo::CryptoBackendFactory::instance()->openpgp() && + !ident.pgpEncryptionKey().isEmpty() ); + // save the state of the sign and encrypt button + if ( !bNewIdentityHasEncryptionKey && mLastIdentityHasEncryptionKey ) { + mLastEncryptActionState = mEncryptAction->isChecked(); + setEncryption( false ); + } + if ( !bNewIdentityHasSigningKey && mLastIdentityHasSigningKey ) { + mLastSignActionState = mSignAction->isChecked(); + setSigning( false ); + } + // restore the last state of the sign and encrypt button + if ( bNewIdentityHasEncryptionKey && !mLastIdentityHasEncryptionKey ) + setEncryption( mLastEncryptActionState ); + if ( bNewIdentityHasSigningKey && !mLastIdentityHasSigningKey ) + setSigning( mLastSignActionState ); + + mLastIdentityHasSigningKey = bNewIdentityHasSigningKey; + mLastIdentityHasEncryptionKey = bNewIdentityHasEncryptionKey; + + setModified( true ); + mId = uoid; + + // make sure the From and BCC fields are shown if necessary + rethinkFields( false ); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotSpellcheckConfig() +{ + KDialogBase dlg(KDialogBase::Plain, i18n("Spellchecker"), + KDialogBase::Ok|KDialogBase::Cancel, KDialogBase::Ok, + this, 0, true, true ); + KWin kwin; + QTabDialog qtd (this, "tabdialog", true); + KSpellConfig mKSpellConfig (&qtd); + mKSpellConfig.layout()->setMargin( KDialog::marginHint() ); + + qtd.addTab (&mKSpellConfig, i18n("Spellchecker")); + qtd.setCancelButton (); + + kwin.setIcons (qtd.winId(), kapp->icon(), kapp->miniIcon()); + qtd.setCancelButton(KStdGuiItem::cancel().text()); + qtd.setOkButton(KStdGuiItem::ok().text()); + + if (qtd.exec()) + mKSpellConfig.writeGlobalSettings(); +} + +//----------------------------------------------------------------------------- +void KMComposeWin::slotStatusMessage(const QString &message) +{ + statusBar()->changeItem( message, 0 ); +} + +void KMComposeWin::slotEditToolbars() +{ + saveMainWindowSettings(KMKernel::config(), "Composer"); + KEditToolbar dlg(guiFactory(), this); + + connect( &dlg, SIGNAL(newToolbarConfig()), + SLOT(slotUpdateToolbars()) ); + + dlg.exec(); +} + +void KMComposeWin::slotUpdateToolbars() +{ + createGUI("kmcomposerui.rc"); + applyMainWindowSettings(KMKernel::config(), "Composer"); +} + +void KMComposeWin::slotEditKeys() +{ + KKeyDialog::configure( actionCollection(), + false /*don't allow one-letter shortcuts*/ + ); +} + +void KMComposeWin::setReplyFocus( bool hasMessage ) +{ + mEditor->setFocus(); + if ( hasMessage ) { + if( mMsg->getCursorPos() ) { + mEditor->setCursorPositionFromStart( (unsigned int) mMsg->getCursorPos() ); + } else { + mEditor->setCursorPosition( 1, 0 ); + } + } +} + +void KMComposeWin::setFocusToSubject() +{ + mEdtSubject->setFocus(); +} + +int KMComposeWin::autoSaveInterval() const +{ + return GlobalSettings::self()->autosaveInterval() * 1000 * 60; +} + +void KMComposeWin::initAutoSave() +{ + kdDebug(5006) << k_funcinfo << endl; + // make sure the autosave folder exists + KMFolderMaildir::createMaildirFolders( KMKernel::localDataPath() + "autosave" ); + if ( mAutoSaveFilename.isEmpty() ) { + mAutoSaveFilename = KMFolderMaildir::constructValidFileName(); + } + + updateAutoSave(); +} + +void KMComposeWin::updateAutoSave() +{ + if ( autoSaveInterval() == 0 ) { + delete mAutoSaveTimer; mAutoSaveTimer = 0; + } + else { + if ( !mAutoSaveTimer ) { + mAutoSaveTimer = new QTimer( this, "mAutoSaveTimer" ); + connect( mAutoSaveTimer, SIGNAL( timeout() ), + this, SLOT( autoSaveMessage() ) ); + } + mAutoSaveTimer->start( autoSaveInterval() ); + } +} + +void KMComposeWin::setAutoSaveFilename( const QString & filename ) +{ + if ( !mAutoSaveFilename.isEmpty() ) + KMFolderMaildir::removeFile( KMKernel::localDataPath() + "autosave", + mAutoSaveFilename ); + mAutoSaveFilename = filename; +} + +void KMComposeWin::cleanupAutoSave() +{ + delete mAutoSaveTimer; mAutoSaveTimer = 0; + if ( !mAutoSaveFilename.isEmpty() ) { + kdDebug(5006) << k_funcinfo << "deleting autosave file " + << mAutoSaveFilename << endl; + KMFolderMaildir::removeFile( KMKernel::localDataPath() + "autosave", + mAutoSaveFilename ); + mAutoSaveFilename = QString(); + } +} + +void KMComposeWin::slotCompletionModeChanged( KGlobalSettings::Completion mode) +{ + GlobalSettings::self()->setCompletionMode( (int) mode ); + + // sync all the lineedits to the same completion mode + mEdtFrom->setCompletionMode( mode ); + mEdtReplyTo->setCompletionMode( mode ); + if ( mClassicalRecipients ) { + mEdtTo->setCompletionMode( mode ); + mEdtCc->setCompletionMode( mode ); + mEdtBcc->setCompletionMode( mode ); + }else + mRecipientsEditor->setCompletionMode( mode ); +} + +void KMComposeWin::slotConfigChanged() +{ + readConfig( true /*reload*/); + updateAutoSave(); + rethinkFields(); + slotWordWrapToggled( mWordWrapAction->isChecked() ); +} + +/* +* checks if the drafts-folder has been deleted +* that is not nice so we set the system-drafts-folder +*/ +void KMComposeWin::slotFolderRemoved(KMFolder* folder) +{ + // TODO: need to handle templates here? + if ( (mFolder) && (folder->idString() == mFolder->idString()) ) + { + mFolder = kmkernel->draftsFolder(); + kdDebug(5006) << "restoring drafts to " << mFolder->idString() << endl; + } + if (mMsg) mMsg->setParent(0); +} + + +void KMComposeWin::editorFocusChanged(bool gained) +{ + mPasteQuotation->setEnabled(gained); + mAddQuoteChars->setEnabled(gained); + mRemQuoteChars->setEnabled(gained); +} + +void KMComposeWin::slotSetAlwaysSend( bool bAlways ) +{ + mAlwaysSend = bAlways; +} + +void KMComposeWin::slotListAction( const QString& style ) +{ + toggleMarkup(true); + if ( style == i18n( "Standard" ) ) + mEditor->setParagType( QStyleSheetItem::DisplayBlock, QStyleSheetItem::ListDisc ); + else if ( style == i18n( "Bulleted List (Disc)" ) ) + mEditor->setParagType( QStyleSheetItem::DisplayListItem, QStyleSheetItem::ListDisc ); + else if ( style == i18n( "Bulleted List (Circle)" ) ) + mEditor->setParagType( QStyleSheetItem::DisplayListItem, QStyleSheetItem::ListCircle ); + else if ( style == i18n( "Bulleted List (Square)" ) ) + mEditor->setParagType( QStyleSheetItem::DisplayListItem, QStyleSheetItem::ListSquare ); + else if ( style == i18n( "Ordered List (Decimal)" )) + mEditor->setParagType( QStyleSheetItem::DisplayListItem, QStyleSheetItem::ListDecimal ); + else if ( style == i18n( "Ordered List (Alpha lower)" ) ) + mEditor->setParagType( QStyleSheetItem::DisplayListItem, QStyleSheetItem::ListLowerAlpha ); + else if ( style == i18n( "Ordered List (Alpha upper)" ) ) + mEditor->setParagType( QStyleSheetItem::DisplayListItem, QStyleSheetItem::ListUpperAlpha ); + mEditor->viewport()->setFocus(); +} + +void KMComposeWin::slotFontAction( const QString& font) +{ + toggleMarkup(true); + mEditor->QTextEdit::setFamily( font ); + mEditor->viewport()->setFocus(); +} + +void KMComposeWin::slotSizeAction( int size ) +{ + toggleMarkup(true); + mEditor->setPointSize( size ); + mEditor->viewport()->setFocus(); +} + +void KMComposeWin::slotAlignLeft() +{ + toggleMarkup(true); + mEditor->QTextEdit::setAlignment( AlignLeft ); +} + +void KMComposeWin::slotAlignCenter() +{ + toggleMarkup(true); + mEditor->QTextEdit::setAlignment( AlignHCenter ); +} + +void KMComposeWin::slotAlignRight() +{ + toggleMarkup(true); + mEditor->QTextEdit::setAlignment( AlignRight ); +} + +void KMComposeWin::slotTextBold() +{ + toggleMarkup(true); + mEditor->QTextEdit::setBold( textBoldAction->isChecked() ); +} + +void KMComposeWin::slotTextItalic() +{ + toggleMarkup(true); + mEditor->QTextEdit::setItalic( textItalicAction->isChecked() ); +} + +void KMComposeWin::slotTextUnder() +{ + toggleMarkup(true); + mEditor->QTextEdit::setUnderline( textUnderAction->isChecked() ); +} + +void KMComposeWin::slotFormatReset() +{ + mEditor->setColor(mForeColor); + mEditor->setCurrentFont( mSaveFont ); // fontChanged is called now +} +void KMComposeWin::slotTextColor() +{ + QColor color = mEditor->color(); + + if ( KColorDialog::getColor( color, this ) ) { + toggleMarkup(true); + mEditor->setColor( color ); + } +} + +void KMComposeWin::fontChanged( const QFont &f ) +{ + QFont fontTemp = f; + fontTemp.setBold( true ); + fontTemp.setItalic( true ); + QFontInfo fontInfo( fontTemp ); + + if ( fontInfo.bold() ) { + textBoldAction->setChecked( f.bold() ); + textBoldAction->setEnabled( true ) ; + } else { + textBoldAction->setEnabled( false ); + } + + if ( fontInfo.italic() ) { + textItalicAction->setChecked( f.italic() ); + textItalicAction->setEnabled( true ) ; + } else { + textItalicAction->setEnabled( false ); + } + + textUnderAction->setChecked( f.underline() ); + + fontAction->setFont( f.family() ); + fontSizeAction->setFontSize( f.pointSize() ); +} + +void KMComposeWin::alignmentChanged( int a ) +{ + //toggleMarkup(); + alignLeftAction->setChecked( ( a == AlignAuto ) || ( a & AlignLeft ) ); + alignCenterAction->setChecked( ( a & AlignHCenter ) ); + alignRightAction->setChecked( ( a & AlignRight ) ); +} + +namespace { + class KToggleActionResetter { + KToggleAction * mAction; + bool mOn; + public: + KToggleActionResetter( KToggleAction * action, bool on ) + : mAction( action ), mOn( on ) {} + ~KToggleActionResetter() { + if ( mAction ) + mAction->setChecked( mOn ); + } + void disable() { mAction = 0; } + }; +} + +void KMComposeWin::slotEncryptChiasmusToggled( bool on ) { + mEncryptWithChiasmus = false; + + if ( !on ) + return; + + KToggleActionResetter resetter( mEncryptChiasmusAction, false ); + + const Kleo::CryptoBackend::Protocol * chiasmus = + Kleo::CryptoBackendFactory::instance()->protocol( "Chiasmus" ); + + if ( !chiasmus ) { + const QString msg = Kleo::CryptoBackendFactory::instance()->knowsAboutProtocol( "Chiasmus" ) + ? i18n( "Please configure a Crypto Backend to use for " + "Chiasmus encryption first.\n" + "You can do this in the Crypto Backends tab of " + "the configure dialog's Security page." ) + : i18n( "It looks as though libkleopatra was compiled without " + "Chiasmus support. You might want to recompile " + "libkleopatra with --enable-chiasmus."); + KMessageBox::information( this, msg, i18n("No Chiasmus Backend Configured" ) ); + return; + } + + STD_NAMESPACE_PREFIX auto_ptr<Kleo::SpecialJob> job( chiasmus->specialJob( "x-obtain-keys", QMap<QString,QVariant>() ) ); + if ( !job.get() ) { + const QString msg = i18n( "Chiasmus backend does not offer the " + "\"x-obtain-keys\" function. Please report this bug." ); + KMessageBox::error( this, msg, i18n( "Chiasmus Backend Error" ) ); + return; + } + + if ( job->exec() ) { + job->showErrorDialog( this, i18n( "Chiasmus Backend Error" ) ); + return; + } + + const QVariant result = job->property( "result" ); + if ( result.type() != QVariant::StringList ) { + const QString msg = i18n( "Unexpected return value from Chiasmus backend: " + "The \"x-obtain-keys\" function did not return a " + "string list. Please report this bug." ); + KMessageBox::error( this, msg, i18n( "Chiasmus Backend Error" ) ); + return; + } + + const QStringList keys = result.toStringList(); + if ( keys.empty() ) { + const QString msg = i18n( "No keys have been found. Please check that a " + "valid key path has been set in the Chiasmus " + "configuration." ); + KMessageBox::information( this, msg, i18n( "No Chiasmus Keys Found" ) ); + return; + } + + ChiasmusKeySelector selectorDlg( this, i18n( "Chiasmus Encryption Key Selection" ), + keys, GlobalSettings::chiasmusKey(), + GlobalSettings::chiasmusOptions() ); + if ( selectorDlg.exec() != QDialog::Accepted ) + return; + + GlobalSettings::setChiasmusOptions( selectorDlg.options() ); + GlobalSettings::setChiasmusKey( selectorDlg.key() ); + assert( !GlobalSettings::chiasmusKey().isEmpty() ); + mEncryptWithChiasmus = true; + resetter.disable(); +} + +void KMComposeWin::slotEditDone(KMail::EditorWatcher * watcher) +{ + kdDebug(5006) << k_funcinfo << endl; + KMMessagePart *part = mEditorMap[ watcher ]; + KTempFile *tf = mEditorTempFiles[ watcher ]; + mEditorMap.remove( watcher ); + mEditorTempFiles.remove( watcher ); + if ( !watcher->fileChanged() ) + return; + + tf->file()->reset(); + QByteArray data = tf->file()->readAll(); + part->setBodyEncodedBinary( data ); +} + + +void KMComposeWin::slotUpdateSignatureAndEncrypionStateIndicators() +{ + const bool showIndicatorsAlways = false; // FIXME config option? + mSignatureStateIndicator->setText( mSignAction->isChecked()? i18n("Message will be signed") : i18n("Message will not be signed") ); + mEncryptionStateIndicator->setText( mEncryptAction->isChecked()? i18n("Message will be encrypted") : i18n("Message will not be encrypted") ); + if ( !showIndicatorsAlways ) { + mSignatureStateIndicator->setShown( mSignAction->isChecked() ); + mEncryptionStateIndicator->setShown( mEncryptAction->isChecked() ); + } +} + +void KMComposeWin::slotAttachmentDragStarted() +{ + kdDebug(5006) << k_funcinfo << endl; + int idx = 0; + QStringList filenames; + for ( QPtrListIterator<QListViewItem> it(mAtmItemList); *it; ++it, ++idx ) { + if ( (*it)->isSelected() ) { + KMMessagePart* msgPart = mAtmList.at(idx); + KTempDir * tempDir = new KTempDir(); // will be deleted on composer close + tempDir->setAutoDelete( true ); + mTempDirs.insert( tempDir ); + const QString fileName = tempDir->name() + "/" + msgPart->name(); + KPIM::kByteArrayToFile(msgPart->bodyDecodedBinary(), + fileName, + false, false, false); + KURL url; + url.setPath( fileName ); + filenames << url.path(); + } + } + if ( filenames.isEmpty() ) return; + + QUriDrag *drag = new QUriDrag( mAtmListView ); + drag->setFileNames( filenames ); + drag->dragCopy(); +} + +void KMComposeWin::recipientEditorSizeHintChanged() +{ + QTimer::singleShot( 1, this, SLOT(setMaximumHeaderSize()) ); +} + +void KMComposeWin::setMaximumHeaderSize() +{ + mHeadersArea->setMaximumHeight( mHeadersArea->sizeHint().height() ); +} + |