diff --git a/src/identity/identitydialog.cpp b/src/identity/identitydialog.cpp index 3d8b040d6..db5ca4b33 100644 --- a/src/identity/identitydialog.cpp +++ b/src/identity/identitydialog.cpp @@ -1,1125 +1,1125 @@ /* identitydialog.cpp This file is part of KMail, the KDE mail client. Copyright (c) 2002 Marc Mutz Copyright (C) 2014-2018 Laurent Montel KMail is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License, version 2, as published by the Free Software Foundation. KMail is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA In addition, as a special exception, the copyright holders give permission to link the code of this program with any edition of the Qt library by Trolltech AS, Norway (or with modified versions of Qt that use the same license as Qt), and distribute linked combinations including the two. You must obey the GNU General Public License in all respects for all of the code used other than Qt. If you modify this file, you may extend this exception to your version of the file, but you are not obligated to do so. If you do not wish to do so, delete this exception statement from your version. */ #include "identitydialog.h" #include "identityeditvcarddialog.h" #include "identityaddvcarddialog.h" #include "identityinvalidfolder.h" #include "identityfolderrequester.h" #include #include #include "MessageComposer/MessageComposerSettings" #include // other KMail headers: #include "xfaceconfigurator.h" #include #include "mailcommon/folderrequester.h" #ifndef KCM_KPIMIDENTITIES_STANDALONE #include "settings/kmailsettings.h" #include "kmkernel.h" #endif #include "mailcommon/mailkernel.h" #include "job/addressvalidationjob.h" #include #include #include #include "TemplateParser/TemplatesConfiguration" #include "templatesconfiguration_kfg.h" // other kdepim headers: #include #include #include "PimCommon/AutoCorrectionLanguage" #include #include // libkleopatra: #include #include #include #include // gpgme++ #include #include #include #include #include #include using MailTransport::TransportManager; // other KDE headers: #include #include #include #include #include "kmail_debug.h" #include #include #include #include // Qt headers: #include #include #include #include #include #include #include #include #include // other headers: #include #include #include #include #include #include #include using namespace KPIM; using namespace MailTransport; using namespace MailCommon; namespace KMail { class KeySelectionCombo : public Kleo::KeySelectionCombo { Q_OBJECT public: enum KeyType { SigningKey, EncryptionKey }; KeySelectionCombo(KeyType keyType, GpgME::Protocol protocol, QWidget *parent); ~KeySelectionCombo() override; void setIdentity(const QString &name, const QString &email); void init() override; private: void onCustomItemSelected(const QVariant &type); QString mEmail; QString mName; KeyType mKeyType; GpgME::Protocol mProtocol; }; class KeyGenerationJob : public QGpgME::Job { Q_OBJECT public: KeyGenerationJob(const QString &name, const QString &email, KeySelectionCombo *parent); ~KeyGenerationJob() override; void slotCancel() override; void start(); private: void keyGenerated(const GpgME::KeyGenerationResult &result); QString mName; QString mEmail; QGpgME::Job *mJob = nullptr; }; KeyGenerationJob::KeyGenerationJob(const QString &name, const QString &email, KeySelectionCombo *parent) : QGpgME::Job(parent) , mName(name) , mEmail(email) , mJob(nullptr) { } KeyGenerationJob::~KeyGenerationJob() { } void KeyGenerationJob::slotCancel() { if (mJob) { mJob->slotCancel(); } } void KeyGenerationJob::start() { auto job = new Kleo::DefaultKeyGenerationJob(this); connect(job, &Kleo::DefaultKeyGenerationJob::result, this, &KeyGenerationJob::keyGenerated); job->start(mEmail, mName); mJob = job; } void KeyGenerationJob::keyGenerated(const GpgME::KeyGenerationResult &result) { mJob = nullptr; if (result.error()) { KMessageBox::error(qobject_cast(parent()), i18n("Error while generating new key pair: %1", QString::fromUtf8(result.error().asString())), i18n("Key Generation Error")); Q_EMIT done(); return; } KeySelectionCombo *combo = qobject_cast(parent()); combo->setDefaultKey(QLatin1String(result.fingerprint())); connect(combo, &KeySelectionCombo::keyListingFinished, this, &KeyGenerationJob::done); combo->refreshKeys(); } KeySelectionCombo::KeySelectionCombo(KeyType keyType, GpgME::Protocol protocol, QWidget *parent) : Kleo::KeySelectionCombo(parent) , mKeyType(keyType) , mProtocol(protocol) { } KeySelectionCombo::~KeySelectionCombo() { } void KeySelectionCombo::setIdentity(const QString &name, const QString &email) { mName = name; mEmail = email; setIdFilter(email); } void KeySelectionCombo::init() { Kleo::KeySelectionCombo::init(); std::shared_ptr keyFilter(new Kleo::DefaultKeyFilter); keyFilter->setIsOpenPGP(mProtocol == GpgME::OpenPGP ? Kleo::DefaultKeyFilter::Set : Kleo::DefaultKeyFilter::NotSet); if (mKeyType == SigningKey) { keyFilter->setCanSign(Kleo::DefaultKeyFilter::Set); } else { keyFilter->setCanEncrypt(Kleo::DefaultKeyFilter::Set); } keyFilter->setHasSecret(Kleo::DefaultKeyFilter::Set); setKeyFilter(keyFilter); prependCustomItem(QIcon(), i18n("No key"), QStringLiteral("no-key")); if (mProtocol == GpgME::OpenPGP) { appendCustomItem(QIcon::fromTheme(QStringLiteral("password-generate")), i18n("Generate a new key pair"), QStringLiteral("generate-new-key")); } connect(this, &KeySelectionCombo::customItemSelected, this, &KeySelectionCombo::onCustomItemSelected); } void KeySelectionCombo::onCustomItemSelected(const QVariant &type) { if (type == QLatin1String("no-key")) { return; } else if (type == QLatin1String("generate-new-key")) { auto job = new KeyGenerationJob(mName, mEmail, this); auto dlg = new Kleo::ProgressDialog(job, i18n("Generating new key pair..."), parentWidget()); dlg->setModal(true); setEnabled(false); connect(job, &KeyGenerationJob::done, this, [this]() { setEnabled(true); }); job->start(); } } IdentityDialog::IdentityDialog(QWidget *parent) : QDialog(parent) { setWindowTitle(i18n("Edit Identity")); QVBoxLayout *mainLayout = new QVBoxLayout(this); QDialogButtonBox *buttonBox = new QDialogButtonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel | QDialogButtonBox::Help, this); connect(buttonBox->button(QDialogButtonBox::Help), &QPushButton::clicked, this, &IdentityDialog::slotHelp); QPushButton *okButton = buttonBox->button(QDialogButtonBox::Ok); okButton->setDefault(true); okButton->setShortcut(Qt::CTRL | Qt::Key_Return); connect(buttonBox, &QDialogButtonBox::accepted, this, &IdentityDialog::slotAccepted); connect(buttonBox, &QDialogButtonBox::rejected, this, &IdentityDialog::reject); // // Tab Widget: General // int row = -1; QWidget *page = new QWidget(this); mainLayout->addWidget(page); mainLayout->addWidget(buttonBox); QVBoxLayout *vlay = new QVBoxLayout(page); vlay->setMargin(0); mTabWidget = new QTabWidget(page); mTabWidget->setObjectName(QStringLiteral("config-identity-tab")); vlay->addWidget(mTabWidget); QWidget *tab = new QWidget(mTabWidget); mTabWidget->addTab(tab, i18nc("@title:tab General identity settings.", "General")); QGridLayout *glay = new QGridLayout(tab); glay->setRowStretch(3, 1); glay->setColumnStretch(1, 1); // "Name" line edit and label: ++row; mNameEdit = new KLineEdit(tab); glay->addWidget(mNameEdit, row, 1); QLabel *label = new QLabel(i18n("&Your name:"), tab); label->setBuddy(mNameEdit); glay->addWidget(label, row, 0); QString msg = i18n("

Your name

" "

This field should contain your name as you would like " "it to appear in the email header that is sent out;

" "

if you leave this blank your real name will not " "appear, only the email address.

"); label->setWhatsThis(msg); mNameEdit->setWhatsThis(msg); // "Organization" line edit and label: ++row; mOrganizationEdit = new KLineEdit(tab); glay->addWidget(mOrganizationEdit, row, 1); label = new QLabel(i18n("Organi&zation:"), tab); label->setBuddy(mOrganizationEdit); glay->addWidget(label, row, 0); msg = i18n("

Organization

" "

This field should have the name of your organization " "if you would like it to be shown in the email header that " "is sent out.

" "

It is safe (and normal) to leave this blank.

"); label->setWhatsThis(msg); mOrganizationEdit->setWhatsThis(msg); // "Email Address" line edit and label: // (row 3: spacer) ++row; mEmailEdit = new KLineEdit(tab); glay->addWidget(mEmailEdit, row, 1); label = new QLabel(i18n("&Email address:"), tab); label->setBuddy(mEmailEdit); glay->addWidget(label, row, 0); msg = i18n("

Email address

" "

This field should have your full email address.

" "

This address is the primary one, used for all outgoing mail. " "If you have more than one address, either create a new identity, " "or add additional alias addresses in the field below.

" "

If you leave this blank, or get it wrong, people " "will have trouble replying to you.

"); label->setWhatsThis(msg); mEmailEdit->setWhatsThis(msg); KPIM::EmailValidator *emailValidator = new KPIM::EmailValidator(this); mEmailEdit->setValidator(emailValidator); // "Email Aliases" string text edit and label: ++row; mAliasEdit = new KEditListWidget(tab); KPIM::EmailValidator *emailValidator1 = new KPIM::EmailValidator(this); mAliasEdit->lineEdit()->setValidator(emailValidator1); glay->addWidget(mAliasEdit, row, 1); label = new QLabel(i18n("Email a&liases:"), tab); label->setBuddy(mAliasEdit); glay->addWidget(label, row, 0, Qt::AlignTop); msg = i18n("

Email aliases

" "

This field contains alias addresses that should also " "be considered as belonging to this identity (as opposed " "to representing a different identity).

" "

Example:

" "" "" "" "
Primary address:first.last@example.org
Aliases:first@example.org
last@example.org
" "

Type one alias address per line.

"); label->setToolTip(msg); mAliasEdit->setWhatsThis(msg); // // Tab Widget: Cryptography // row = -1; mCryptographyTab = tab = new QWidget(mTabWidget); mTabWidget->addTab(tab, i18n("Cryptography")); glay = new QGridLayout(tab); glay->setColumnStretch(1, 1); // "OpenPGP Signature Key" requester and label: ++row; mPGPSigningKeyRequester = new KeySelectionCombo(KeySelectionCombo::SigningKey, GpgME::OpenPGP, tab); msg = i18n("

The OpenPGP key you choose here will be used " "to digitally sign messages. You can also use GnuPG keys.

" "

You can leave this blank, but KMail will not be able " "to digitally sign emails using OpenPGP; " "normal mail functions will not be affected.

" "

You can find out more about keys at http://www.gnupg.org

"); label = new QLabel(i18n("OpenPGP signing key:"), tab); label->setBuddy(mPGPSigningKeyRequester); mPGPSigningKeyRequester->setWhatsThis(msg); label->setWhatsThis(msg); glay->addWidget(label, row, 0); glay->addWidget(mPGPSigningKeyRequester, row, 1); // "OpenPGP Encryption Key" requester and label: ++row; mPGPEncryptionKeyRequester = new KeySelectionCombo(KeySelectionCombo::EncryptionKey, GpgME::OpenPGP, tab); msg = i18n("

The OpenPGP key you choose here will be used " "to encrypt messages to yourself and for the \"Attach My Public Key\" " "feature in the composer. You can also use GnuPG keys.

" "

You can leave this blank, but KMail will not be able " "to encrypt copies of outgoing messages to you using OpenPGP; " "normal mail functions will not be affected.

" "

You can find out more about keys at http://www.gnupg.org

"); label = new QLabel(i18n("OpenPGP encryption key:"), tab); label->setBuddy(mPGPEncryptionKeyRequester); mPGPEncryptionKeyRequester->setWhatsThis(msg); label->setWhatsThis(msg); glay->addWidget(label, row, 0); glay->addWidget(mPGPEncryptionKeyRequester, row, 1); // "S/MIME Signature Key" requester and label: ++row; mSMIMESigningKeyRequester = new KeySelectionCombo(KeySelectionCombo::SigningKey, GpgME::CMS, tab); msg = i18n("

The S/MIME (X.509) certificate you choose here will be used " "to digitally sign messages.

" "

You can leave this blank, but KMail will not be able " "to digitally sign emails using S/MIME; " "normal mail functions will not be affected.

"); label = new QLabel(i18n("S/MIME signing certificate:"), tab); label->setBuddy(mSMIMESigningKeyRequester); mSMIMESigningKeyRequester->setWhatsThis(msg); label->setWhatsThis(msg); glay->addWidget(label, row, 0); glay->addWidget(mSMIMESigningKeyRequester, row, 1); const QGpgME::Protocol *smimeProtocol = QGpgME::smime(); label->setEnabled(smimeProtocol); mSMIMESigningKeyRequester->setEnabled(smimeProtocol); // "S/MIME Encryption Key" requester and label: ++row; mSMIMEEncryptionKeyRequester = new KeySelectionCombo(KeySelectionCombo::EncryptionKey, GpgME::CMS, tab); msg = i18n("

The S/MIME certificate you choose here will be used " "to encrypt messages to yourself and for the \"Attach My Certificate\" " "feature in the composer.

" "

You can leave this blank, but KMail will not be able " "to encrypt copies of outgoing messages to you using S/MIME; " "normal mail functions will not be affected.

"); label = new QLabel(i18n("S/MIME encryption certificate:"), tab); label->setBuddy(mSMIMEEncryptionKeyRequester); mSMIMEEncryptionKeyRequester->setWhatsThis(msg); label->setWhatsThis(msg); glay->addWidget(label, row, 0); glay->addWidget(mSMIMEEncryptionKeyRequester, row, 1); label->setEnabled(smimeProtocol); mSMIMEEncryptionKeyRequester->setEnabled(smimeProtocol); // "Preferred Crypto Message Format" combobox and label: ++row; mPreferredCryptoMessageFormat = new KComboBox(tab); mPreferredCryptoMessageFormat->setEditable(false); QStringList l; l << Kleo::cryptoMessageFormatToLabel(Kleo::AutoFormat) << Kleo::cryptoMessageFormatToLabel(Kleo::InlineOpenPGPFormat) << Kleo::cryptoMessageFormatToLabel(Kleo::OpenPGPMIMEFormat) << Kleo::cryptoMessageFormatToLabel(Kleo::SMIMEFormat) << Kleo::cryptoMessageFormatToLabel(Kleo::SMIMEOpaqueFormat); mPreferredCryptoMessageFormat->addItems(l); label = new QLabel(i18nc("preferred format of encrypted messages", "Preferred format:"), tab); label->setBuddy(mPreferredCryptoMessageFormat); glay->addWidget(label, row, 0); glay->addWidget(mPreferredCryptoMessageFormat, row, 1); ++row; mAutoSign = new QCheckBox(i18n("Automatically sign messages")); glay->addWidget(mAutoSign, row, 0); ++row; mAutoEncrypt = new QCheckBox(i18n("Automatically encrypt messages when possible")); glay->addWidget(mAutoEncrypt, row, 0); ++row; glay->setRowStretch(row, 1); // // Tab Widget: Advanced // row = -1; tab = new QWidget(mTabWidget); QVBoxLayout *advancedMainLayout = new QVBoxLayout(tab); mIdentityInvalidFolder = new IdentityInvalidFolder(tab); advancedMainLayout->addWidget(mIdentityInvalidFolder); mTabWidget->addTab(tab, i18nc("@title:tab Advanced identity settings.", "Advanced")); glay = new QGridLayout; advancedMainLayout->addLayout(glay); // the last (empty) row takes all the remaining space glay->setColumnStretch(1, 1); // "Reply-To Address" line edit and label: ++row; mReplyToEdit = new KPIM::AddresseeLineEdit(tab, true); - mReplyToEdit->setClearButtonShown(true); + mReplyToEdit->setClearButtonEnabled(true); mReplyToEdit->setObjectName(QStringLiteral("mReplyToEdit")); glay->addWidget(mReplyToEdit, row, 1); label = new QLabel(i18n("&Reply-To address:"), tab); label->setBuddy(mReplyToEdit); glay->addWidget(label, row, 0); msg = i18n("

Reply-To addresses

" "

This sets the Reply-to: header to contain a " "different email address to the normal From: " "address.

" "

This can be useful when you have a group of people " "working together in similar roles. For example, you " "might want any emails sent to have your email in the " "From: field, but any responses to go to " "a group address.

" "

If in doubt, leave this field blank.

"); label->setWhatsThis(msg); mReplyToEdit->setWhatsThis(msg); // "CC addresses" line edit and label: ++row; mCcEdit = new KPIM::AddresseeLineEdit(tab, true); - mCcEdit->setClearButtonShown(true); + mCcEdit->setClearButtonEnabled(true); mCcEdit->setObjectName(QStringLiteral("mCcEdit")); glay->addWidget(mCcEdit, row, 1); label = new QLabel(i18n("&CC addresses:"), tab); label->setBuddy(mCcEdit); glay->addWidget(label, row, 0); msg = i18n("

CC (Carbon Copy) addresses

" "

The addresses that you enter here will be added to each " "outgoing mail that is sent with this identity.

" "

This is commonly used to send a copy of each sent message to " "another account of yours.

" "

To specify more than one address, use commas to separate " "the list of CC recipients.

" "

If in doubt, leave this field blank.

"); label->setWhatsThis(msg); mCcEdit->setWhatsThis(msg); // "BCC addresses" line edit and label: ++row; mBccEdit = new KPIM::AddresseeLineEdit(tab, true); - mBccEdit->setClearButtonShown(true); + mBccEdit->setClearButtonEnabled(true); mBccEdit->setObjectName(QStringLiteral("mBccEdit")); glay->addWidget(mBccEdit, row, 1); label = new QLabel(i18n("&BCC addresses:"), tab); label->setBuddy(mBccEdit); glay->addWidget(label, row, 0); msg = i18n("

BCC (Blind Carbon Copy) addresses

" "

The addresses that you enter here will be added to each " "outgoing mail that is sent with this identity. They will not " "be visible to other recipients.

" "

This is commonly used to send a copy of each sent message to " "another account of yours.

" "

To specify more than one address, use commas to separate " "the list of BCC recipients.

" "

If in doubt, leave this field blank.

"); label->setWhatsThis(msg); mBccEdit->setWhatsThis(msg); // "Dictionary" combo box and label: ++row; mDictionaryCombo = new Sonnet::DictionaryComboBox(tab); glay->addWidget(mDictionaryCombo, row, 1); label = new QLabel(i18n("D&ictionary:"), tab); label->setBuddy(mDictionaryCombo); glay->addWidget(label, row, 0); // "Sent-mail Folder" combo box and label: ++row; mFccFolderRequester = new IdentityFolderRequester(tab); mFccFolderRequester->setShowOutbox(false); glay->addWidget(mFccFolderRequester, row, 1); mSentMailFolderCheck = new QCheckBox(i18n("Sent-mail &folder:"), tab); glay->addWidget(mSentMailFolderCheck, row, 0); connect(mSentMailFolderCheck, &QCheckBox::toggled, mFccFolderRequester, &MailCommon::FolderRequester::setEnabled); // "Drafts Folder" combo box and label: ++row; mDraftsFolderRequester = new IdentityFolderRequester(tab); mDraftsFolderRequester->setShowOutbox(false); glay->addWidget(mDraftsFolderRequester, row, 1); label = new QLabel(i18n("&Drafts folder:"), tab); label->setBuddy(mDraftsFolderRequester); glay->addWidget(label, row, 0); // "Templates Folder" combo box and label: ++row; mTemplatesFolderRequester = new IdentityFolderRequester(tab); mTemplatesFolderRequester->setShowOutbox(false); glay->addWidget(mTemplatesFolderRequester, row, 1); label = new QLabel(i18n("&Templates folder:"), tab); label->setBuddy(mTemplatesFolderRequester); glay->addWidget(label, row, 0); // "Special transport" combobox and label: ++row; mTransportCheck = new QCheckBox(i18n("Outgoing Account:"), tab); glay->addWidget(mTransportCheck, row, 0); mTransportCombo = new TransportComboBox(tab); mTransportCombo->setEnabled(false); // since !mTransportCheck->isChecked() glay->addWidget(mTransportCombo, row, 1); connect(mTransportCheck, &QCheckBox::toggled, mTransportCombo, &MailTransport::TransportComboBox::setEnabled); ++row; mAttachMyVCard = new QCheckBox(i18n("Attach my vCard to message"), tab); glay->addWidget(mAttachMyVCard, row, 0); mEditVCard = new QPushButton(i18n("Create..."), tab); connect(mEditVCard, &QPushButton::clicked, this, &IdentityDialog::slotEditVcard); glay->addWidget(mEditVCard, row, 1); ++row; mAutoCorrectionLanguage = new PimCommon::AutoCorrectionLanguage(tab); glay->addWidget(mAutoCorrectionLanguage, row, 1); label = new QLabel(i18n("Autocorrection language:"), tab); label->setBuddy(mAutoCorrectionLanguage); glay->addWidget(label, row, 0); // "default domain" input field: ++row; QHBoxLayout *hbox = new QHBoxLayout; mDefaultDomainEdit = new KLineEdit(tab); - mDefaultDomainEdit->setClearButtonShown(true); + mDefaultDomainEdit->setClearButtonEnabled(true); hbox->addWidget(mDefaultDomainEdit); QToolButton *restoreDefaultDomainName = new QToolButton; restoreDefaultDomainName->setIcon(QIcon::fromTheme(QStringLiteral("view-refresh"))); restoreDefaultDomainName->setToolTip(i18n("Restore default domain name")); hbox->addWidget(restoreDefaultDomainName); connect(restoreDefaultDomainName, &QToolButton::clicked, this, &IdentityDialog::slotRefreshDefaultDomainName); glay->addLayout(hbox, row, 1); label = new QLabel(i18n("Defaul&t domain:"), tab); label->setBuddy(mDefaultDomainEdit); glay->addWidget(label, row, 0); // and now: add QWhatsThis: msg = i18n("

The default domain is used to complete email " "addresses that only consist of the user's name." "

"); label->setWhatsThis(msg); mDefaultDomainEdit->setWhatsThis(msg); ++row; glay->setRowStretch(row, 1); // the last row is a spacer // // Tab Widget: Templates // tab = new QWidget(mTabWidget); vlay = new QVBoxLayout(tab); QHBoxLayout *tlay = new QHBoxLayout(); vlay->addLayout(tlay); mCustom = new QCheckBox(i18n("&Use custom message templates for this identity"), tab); tlay->addWidget(mCustom, Qt::AlignLeft); mWidget = new TemplateParser::TemplatesConfiguration(tab, QStringLiteral("identity-templates")); mWidget->setEnabled(false); // Move the help label outside of the templates configuration widget, // so that the help can be read even if the widget is not enabled. tlay->addStretch(9); tlay->addWidget(mWidget->helpLabel(), Qt::AlignRight); vlay->addWidget(mWidget); QHBoxLayout *btns = new QHBoxLayout(); mCopyGlobal = new QPushButton(i18n("&Copy Global Templates"), tab); mCopyGlobal->setEnabled(false); btns->addWidget(mCopyGlobal); vlay->addLayout(btns); connect(mCustom, &QCheckBox::toggled, mWidget, &TemplateParser::TemplatesConfiguration::setEnabled); connect(mCustom, &QCheckBox::toggled, mCopyGlobal, &QPushButton::setEnabled); connect(mCopyGlobal, &QPushButton::clicked, this, &IdentityDialog::slotCopyGlobal); mTabWidget->addTab(tab, i18n("Templates")); // // Tab Widget: Signature // mSignatureConfigurator = new KIdentityManagement::SignatureConfigurator(mTabWidget); mTabWidget->addTab(mSignatureConfigurator, i18n("Signature")); // // Tab Widget: Picture // mXFaceConfigurator = new XFaceConfigurator(mTabWidget); mTabWidget->addTab(mXFaceConfigurator, i18n("Picture")); #ifndef KCM_KPIMIDENTITIES_STANDALONE resize(KMailSettings::self()->identityDialogSize()); #endif mNameEdit->setFocus(); connect(mTabWidget, &QTabWidget::currentChanged, this, &IdentityDialog::slotAboutToShow); } IdentityDialog::~IdentityDialog() { #ifndef KCM_KPIMIDENTITIES_STANDALONE KMailSettings::self()->setIdentityDialogSize(size()); #endif } void IdentityDialog::slotHelp() { PimCommon::Util::invokeHelp(QStringLiteral("kmail2/configure-identity.html")); } void IdentityDialog::slotAboutToShow(int index) { QWidget *w = mTabWidget->widget(index); if (w == mCryptographyTab) { // set the configured email address as initial query of the key // requesters: const QString name = mNameEdit->text().trimmed(); const QString email = mEmailEdit->text().trimmed(); mPGPEncryptionKeyRequester->setIdentity(name, email); mPGPSigningKeyRequester->setIdentity(name, email); mSMIMEEncryptionKeyRequester->setIdentity(name, email); mSMIMESigningKeyRequester->setIdentity(name, email); } } void IdentityDialog::slotCopyGlobal() { mWidget->loadFromGlobal(); } void IdentityDialog::slotRefreshDefaultDomainName() { mDefaultDomainEdit->setText(QHostInfo::localHostName()); } void IdentityDialog::slotAccepted() { const QStringList aliases = mAliasEdit->items(); for (const QString &alias : aliases) { if (alias.trimmed().isEmpty()) { continue; } if (!KEmailAddress::isValidSimpleAddress(alias)) { const QString errorMsg(KEmailAddress::simpleEmailAddressErrorMsg()); KMessageBox::sorry(this, errorMsg, i18n("Invalid Email Alias \"%1\"", alias)); return; } } // Validate email addresses const QString email = mEmailEdit->text().trimmed(); if (!KEmailAddress::isValidSimpleAddress(email)) { const QString errorMsg(KEmailAddress::simpleEmailAddressErrorMsg()); KMessageBox::sorry(this, errorMsg, i18n("Invalid Email Address")); return; } // Check if the 'Reply to' and 'BCC' recipients are valid const QString recipients = mReplyToEdit->text().trimmed() + QLatin1String(", ") + mBccEdit->text().trimmed() + QLatin1String(", ") + mCcEdit->text().trimmed(); AddressValidationJob *job = new AddressValidationJob(recipients, this, this); //Use default Value job->setDefaultDomain(mDefaultDomainEdit->text()); job->setProperty("email", email); connect(job, &AddressValidationJob::result, this, &IdentityDialog::slotDelayedButtonClicked); job->start(); } bool IdentityDialog::keyMatchesEmailAddress(const GpgME::Key &key, const QString &email_) { if (key.isNull()) { return true; } const QString email = email_.trimmed().toLower(); const auto uids = key.userIDs(); for (const auto &uid : uids) { QString em = QString::fromUtf8(uid.email() ? uid.email() : uid.id()); if (em.isEmpty()) { continue; } if (em[0] == QLatin1Char('<')) { em = em.mid(1, em.length() - 2); } if (em.toLower() == email) { return true; } } return false; } void IdentityDialog::slotDelayedButtonClicked(KJob *job) { const AddressValidationJob *validationJob = qobject_cast(job); // Abort if one of the recipient addresses is invalid if (!validationJob->isValid()) { return; } const QString email = validationJob->property("email").toString(); const GpgME::Key &pgpSigningKey = mPGPSigningKeyRequester->currentKey(); const GpgME::Key &pgpEncryptionKey = mPGPEncryptionKeyRequester->currentKey(); const GpgME::Key &smimeSigningKey = mSMIMESigningKeyRequester->currentKey(); const GpgME::Key &smimeEncryptionKey = mSMIMEEncryptionKeyRequester->currentKey(); QString msg; bool err = false; if (!keyMatchesEmailAddress(pgpSigningKey, email)) { msg = i18n("One of the configured OpenPGP signing keys does not contain " "any user ID with the configured email address for this " "identity (%1).\n" "This might result in warning messages on the receiving side " "when trying to verify signatures made with this configuration.", email); err = true; } else if (!keyMatchesEmailAddress(pgpEncryptionKey, email)) { msg = i18n("One of the configured OpenPGP encryption keys does not contain " "any user ID with the configured email address for this " "identity (%1).", email); err = true; } else if (!keyMatchesEmailAddress(smimeSigningKey, email)) { msg = i18n("One of the configured S/MIME signing certificates does not contain " "the configured email address for this " "identity (%1).\n" "This might result in warning messages on the receiving side " "when trying to verify signatures made with this configuration.", email); err = true; } else if (!keyMatchesEmailAddress(smimeEncryptionKey, email)) { msg = i18n("One of the configured S/MIME encryption certificates does not contain " "the configured email address for this " "identity (%1).", email); err = true; } if (err) { if (KMessageBox::warningContinueCancel(this, msg, i18n("Email Address Not Found in Key/Certificates"), KStandardGuiItem::cont(), KStandardGuiItem::cancel(), QStringLiteral("warn_email_not_in_certificate")) != KMessageBox::Continue) { return; } } if (mSignatureConfigurator->isSignatureEnabled() && mSignatureConfigurator->signatureType() == Signature::FromFile) { QFileInfo file(mSignatureConfigurator->filePath()); if (!file.isReadable()) { KMessageBox::error(this, i18n("The signature file is not valid")); return; } } accept(); } bool IdentityDialog::checkFolderExists(const QString &folderID) { const Akonadi::Collection folder = CommonKernel->collectionFromId(folderID.toLongLong()); return folder.isValid(); } void IdentityDialog::setIdentity(KIdentityManagement::Identity &ident) { setWindowTitle(i18n("Edit Identity \"%1\"", ident.identityName())); // "General" tab: mNameEdit->setText(ident.fullName()); mOrganizationEdit->setText(ident.organization()); mEmailEdit->setText(ident.primaryEmailAddress()); mAliasEdit->insertStringList(ident.emailAliases()); // "Cryptography" tab: mPGPSigningKeyRequester->setDefaultKey(QLatin1String(ident.pgpSigningKey())); mPGPEncryptionKeyRequester->setDefaultKey(QLatin1String(ident.pgpEncryptionKey())); mSMIMESigningKeyRequester->setDefaultKey(QLatin1String(ident.smimeSigningKey())); mSMIMEEncryptionKeyRequester->setDefaultKey(QLatin1String(ident.smimeEncryptionKey())); mPreferredCryptoMessageFormat->setCurrentIndex(format2cb( Kleo::stringToCryptoMessageFormat(ident.preferredCryptoMessageFormat()))); mAutoSign->setChecked(ident.pgpAutoSign()); mAutoEncrypt->setChecked(ident.pgpAutoEncrypt()); // "Advanced" tab: mReplyToEdit->setText(ident.replyToAddr()); mBccEdit->setText(ident.bcc()); mCcEdit->setText(ident.cc()); const int transportId = ident.transport().isEmpty() ? -1 : ident.transport().toInt(); const Transport *transport = TransportManager::self()->transportById(transportId, true); mTransportCheck->setChecked(transportId != -1); mTransportCombo->setEnabled(transportId != -1); if (transport) { mTransportCombo->setCurrentTransport(transport->id()); } mDictionaryCombo->setCurrentByDictionaryName(ident.dictionary()); mSentMailFolderCheck->setChecked(!ident.disabledFcc()); mFccFolderRequester->setEnabled(mSentMailFolderCheck->isChecked()); bool foundNoExistingFolder = false; if (ident.fcc().isEmpty() || !checkFolderExists(ident.fcc())) { foundNoExistingFolder = true; mFccFolderRequester->setIsInvalidFolder(CommonKernel->sentCollectionFolder()); } else { mFccFolderRequester->setCollection(Akonadi::Collection(ident.fcc().toLongLong())); } if (ident.drafts().isEmpty() || !checkFolderExists(ident.drafts())) { foundNoExistingFolder = true; mDraftsFolderRequester->setIsInvalidFolder(CommonKernel->draftsCollectionFolder()); } else { mDraftsFolderRequester->setCollection(Akonadi::Collection(ident.drafts().toLongLong())); } if (ident.templates().isEmpty() || !checkFolderExists(ident.templates())) { foundNoExistingFolder = true; mTemplatesFolderRequester->setIsInvalidFolder(CommonKernel->templatesCollectionFolder()); } else { mTemplatesFolderRequester->setCollection(Akonadi::Collection(ident.templates().toLongLong())); } if (foundNoExistingFolder) { mIdentityInvalidFolder->setErrorMessage(i18n("Some custom folder for identity does not exist (anymore); therefore, default folders will be used.")); } mVcardFilename = ident.vCardFile(); mAutoCorrectionLanguage->setLanguage(ident.autocorrectionLanguage()); updateVcardButton(); if (mVcardFilename.isEmpty()) { mVcardFilename = QStandardPaths::writableLocation(QStandardPaths::AppLocalDataLocation) + QLatin1Char('/') + ident.identityName() + QLatin1String(".vcf"); QFileInfo fileInfo(mVcardFilename); QDir().mkpath(fileInfo.absolutePath()); } else { //Convert path. const QString path = QStandardPaths::writableLocation(QStandardPaths::AppLocalDataLocation) + QLatin1Char('/') + ident.identityName() + QLatin1String(".vcf"); if (QFileInfo::exists(path) && (mVcardFilename != path)) { mVcardFilename = path; } } mAttachMyVCard->setChecked(ident.attachVcard()); QString defaultDomainName = ident.defaultDomainName(); if (defaultDomainName.isEmpty()) { defaultDomainName = QHostInfo::localHostName(); } mDefaultDomainEdit->setText(defaultDomainName); // "Templates" tab: uint identity = ident.uoid(); QString iid = TemplateParser::TemplatesConfiguration::configIdString(identity); TemplateParser::Templates t(iid); mCustom->setChecked(t.useCustomTemplates()); mWidget->loadFromIdentity(identity); // "Signature" tab: mSignatureConfigurator->setImageLocation(ident); mSignatureConfigurator->setSignature(ident.signature()); mXFaceConfigurator->setXFace(ident.xface()); mXFaceConfigurator->setXFaceEnabled(ident.isXFaceEnabled()); } void IdentityDialog::updateIdentity(KIdentityManagement::Identity &ident) { // "General" tab: ident.setFullName(mNameEdit->text()); ident.setOrganization(mOrganizationEdit->text()); QString email = mEmailEdit->text(); ident.setPrimaryEmailAddress(email); const QStringList aliases = mAliasEdit->items(); QStringList result; for (const QString &alias : aliases) { if (alias.trimmed().isEmpty()) { continue; } if (alias == email) { continue; } result.append(alias); } ident.setEmailAliases(result); // "Cryptography" tab: ident.setPGPSigningKey(mPGPSigningKeyRequester->currentKey().primaryFingerprint()); ident.setPGPEncryptionKey(mPGPEncryptionKeyRequester->currentKey().primaryFingerprint()); ident.setSMIMESigningKey(mSMIMESigningKeyRequester->currentKey().primaryFingerprint()); ident.setSMIMEEncryptionKey(mSMIMEEncryptionKeyRequester->currentKey().primaryFingerprint()); ident.setPreferredCryptoMessageFormat( QLatin1String(Kleo::cryptoMessageFormatToString(cb2format(mPreferredCryptoMessageFormat->currentIndex())))); ident.setPgpAutoSign(mAutoSign->isChecked()); ident.setPgpAutoEncrypt(mAutoEncrypt->isChecked()); // "Advanced" tab: ident.setReplyToAddr(mReplyToEdit->text()); ident.setBcc(mBccEdit->text()); ident.setCc(mCcEdit->text()); ident.setTransport(mTransportCheck->isChecked() ? QString::number(mTransportCombo->currentTransportId()) : QString()); ident.setDictionary(mDictionaryCombo->currentDictionaryName()); ident.setDisabledFcc(!mSentMailFolderCheck->isChecked()); Akonadi::Collection collection = mFccFolderRequester->collection(); if (collection.isValid()) { ident.setFcc(QString::number(collection.id())); Akonadi::EntityDisplayAttribute *attribute = collection.attribute(Akonadi::Collection::AddIfMissing); attribute->setIconName(QStringLiteral("mail-folder-sent")); new Akonadi::CollectionModifyJob(collection); } else { ident.setFcc(QString()); } collection = mDraftsFolderRequester->collection(); if (collection.isValid()) { ident.setDrafts(QString::number(collection.id())); Akonadi::EntityDisplayAttribute *attribute = collection.attribute(Akonadi::Collection::AddIfMissing); attribute->setIconName(QStringLiteral("document-properties")); new Akonadi::CollectionModifyJob(collection); } else { ident.setDrafts(QString()); } collection = mTemplatesFolderRequester->collection(); if (collection.isValid()) { ident.setTemplates(QString::number(collection.id())); Akonadi::EntityDisplayAttribute *attribute = collection.attribute(Akonadi::Collection::AddIfMissing); attribute->setIconName(QStringLiteral("document-new")); new Akonadi::CollectionModifyJob(collection); } else { ident.setTemplates(QString()); } ident.setVCardFile(mVcardFilename); ident.setAutocorrectionLanguage(mAutoCorrectionLanguage->language()); updateVcardButton(); ident.setAttachVcard(mAttachMyVCard->isChecked()); //Add default ? ident.setDefaultDomainName(mDefaultDomainEdit->text()); // "Templates" tab: uint identity = ident.uoid(); QString iid = TemplateParser::TemplatesConfiguration::configIdString(identity); TemplateParser::Templates t(iid); qCDebug(KMAIL_LOG) << "use custom templates for identity" << identity << ":" << mCustom->isChecked(); t.setUseCustomTemplates(mCustom->isChecked()); t.save(); mWidget->saveToIdentity(identity); // "Signature" tab: ident.setSignature(mSignatureConfigurator->signature()); ident.setXFace(mXFaceConfigurator->xface()); ident.setXFaceEnabled(mXFaceConfigurator->isXFaceEnabled()); } void IdentityDialog::slotEditVcard() { if (QFileInfo::exists(mVcardFilename)) { editVcard(mVcardFilename); } else { if (!MailCommon::Kernel::self()->kernelIsRegistered()) { return; } KIdentityManagement::IdentityManager *manager = KernelIf->identityManager(); QPointer dlg = new IdentityAddVcardDialog(manager->shadowIdentities(), this); if (dlg->exec()) { IdentityAddVcardDialog::DuplicateMode mode = dlg->duplicateMode(); switch (mode) { case IdentityAddVcardDialog::Empty: editVcard(mVcardFilename); break; case IdentityAddVcardDialog::ExistingEntry: { KIdentityManagement::Identity ident = manager->modifyIdentityForName(dlg->duplicateVcardFromIdentity()); const QString filename = ident.vCardFile(); if (!filename.isEmpty()) { QFile::copy(filename, mVcardFilename); } editVcard(mVcardFilename); break; } case IdentityAddVcardDialog::FromExistingVCard: { const QString filename = dlg->existingVCard().path(); if (!filename.isEmpty()) { mVcardFilename = filename; } editVcard(mVcardFilename); break; } } } delete dlg; } } void IdentityDialog::editVcard(const QString &filename) { QPointer dlg = new IdentityEditVcardDialog(filename, this); connect(dlg.data(), &IdentityEditVcardDialog::vcardRemoved, this, &IdentityDialog::slotVCardRemoved); if (dlg->exec()) { mVcardFilename = dlg->saveVcard(); } updateVcardButton(); delete dlg; } void IdentityDialog::slotVCardRemoved() { mVcardFilename.clear(); } void IdentityDialog::updateVcardButton() { if (mVcardFilename.isEmpty() || !QFileInfo::exists(mVcardFilename)) { mEditVCard->setText(i18n("Create...")); } else { mEditVCard->setText(i18n("Edit...")); } } } #include "identitydialog.moc" diff --git a/src/identity/newidentitydialog.cpp b/src/identity/newidentitydialog.cpp index 71d9faf63..b8b3f74e3 100644 --- a/src/identity/newidentitydialog.cpp +++ b/src/identity/newidentitydialog.cpp @@ -1,161 +1,161 @@ /* * kmail: KDE mail client * Copyright (C) 2000 Espen Sand, espen@kde.org * Copyright (C) 2001-2003 Marc Mutz, mutz@kde.org * Contains code segments and ideas from earlier kmail dialog code. * Copyright (C) 2010 Volker Krause * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * */ #include "newidentitydialog.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace KMail; NewIdentityDialog::NewIdentityDialog(KIdentityManagement::IdentityManager *manager, QWidget *parent) : QDialog(parent) , mIdentityManager(manager) { setWindowTitle(i18n("New Identity")); QDialogButtonBox *buttonBox = new QDialogButtonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel | QDialogButtonBox::Help, this); QVBoxLayout *mainLayout = new QVBoxLayout(this); mOkButton = buttonBox->button(QDialogButtonBox::Ok); mOkButton->setDefault(true); mOkButton->setShortcut(Qt::CTRL | Qt::Key_Return); connect(buttonBox, &QDialogButtonBox::accepted, this, &NewIdentityDialog::accept); connect(buttonBox, &QDialogButtonBox::rejected, this, &NewIdentityDialog::reject); connect(buttonBox->button(QDialogButtonBox::Help), &QPushButton::clicked, this, &NewIdentityDialog::slotHelp); QWidget *page = new QWidget(this); mainLayout->addWidget(page); mainLayout->addWidget(buttonBox); QVBoxLayout *vlay = new QVBoxLayout(page); vlay->setMargin(0); // row 0: line edit with label QHBoxLayout *hlay = new QHBoxLayout(); // inherits spacing vlay->addLayout(hlay); mLineEdit = new KLineEdit(page); mLineEdit->setFocus(); - mLineEdit->setClearButtonShown(true); + mLineEdit->setClearButtonEnabled(true); QLabel *l = new QLabel(i18n("&New identity:"), page); l->setBuddy(mLineEdit); hlay->addWidget(l); hlay->addWidget(mLineEdit, 1); connect(mLineEdit, &KLineEdit::textChanged, this, &NewIdentityDialog::slotEnableOK); mButtonGroup = new QButtonGroup(page); // row 1: radio button QRadioButton *radio = new QRadioButton(i18n("&With empty fields"), page); radio->setChecked(true); vlay->addWidget(radio); mButtonGroup->addButton(radio, static_cast(Empty)); // row 2: radio button radio = new QRadioButton(i18n("&Use System Settings values"), page); vlay->addWidget(radio); mButtonGroup->addButton(radio, static_cast(ControlCenter)); // row 3: radio button radio = new QRadioButton(i18n("&Duplicate existing identity"), page); vlay->addWidget(radio); mButtonGroup->addButton(radio, static_cast(ExistingEntry)); // row 4: combobox with existing identities and label hlay = new QHBoxLayout(); // inherits spacing vlay->addLayout(hlay); mComboBox = new KComboBox(page); mComboBox->setEditable(false); mComboBox->addItems(manager->shadowIdentities()); mComboBox->setEnabled(false); QLabel *label = new QLabel(i18n("&Existing identities:"), page); label->setBuddy(mComboBox); label->setEnabled(false); hlay->addWidget(label); hlay->addWidget(mComboBox, 1); vlay->addWidget(new KSeparator); vlay->addStretch(1); // spacer // enable/disable combobox and label depending on the third radio // button's state: connect(radio, &QRadioButton::toggled, label, &QLabel::setEnabled); connect(radio, &QRadioButton::toggled, mComboBox, &KComboBox::setEnabled); mOkButton->setEnabled(false); // since line edit is empty resize(400, 180); } void NewIdentityDialog::slotHelp() { PimCommon::Util::invokeHelp(QStringLiteral("kmail2/configure-identity.html"), QStringLiteral("configure-identity-newidentitydialog")); } NewIdentityDialog::DuplicateMode NewIdentityDialog::duplicateMode() const { const int id = mButtonGroup->checkedId(); assert(id == static_cast(Empty) || id == static_cast(ControlCenter) || id == static_cast(ExistingEntry)); return static_cast(id); } void NewIdentityDialog::slotEnableOK(const QString &proposedIdentityName) { // OK button is disabled if const QString name = proposedIdentityName.trimmed(); // name isn't empty if (name.isEmpty()) { mOkButton->setEnabled(false); return; } // or name doesn't yet exist. if (!mIdentityManager->isUnique(name)) { mOkButton->setEnabled(false); return; } mOkButton->setEnabled(true); } QString NewIdentityDialog::identityName() const { return mLineEdit->text(); } QString NewIdentityDialog::duplicateIdentity() const { return mComboBox->currentText(); }