Mailcommon

mailkernel.cpp
1/*
2 SPDX-FileCopyrightText: 2010 Klarälvdalens Datakonsult AB, a KDAB Group company <info@kdab.com>
3 SPDX-FileCopyrightText: 2010 Andras Mantia <andras@kdab.com>
4
5 SPDX-License-Identifier: GPL-2.0-or-later
6*/
7
8#include "mailkernel.h"
9#include "mailcommon_debug.h"
10#include "util/mailutil.h"
11#include <PimCommon/PimUtil>
12#include <PimCommonAkonadi/ImapResourceCapabilitiesManager>
13
14#include "util/resourcereadconfigfile.h"
15#include <Akonadi/AgentInstance>
16#include <Akonadi/AgentManager>
17#include <Akonadi/EntityMimeTypeFilterModel>
18#include <Akonadi/SpecialMailCollectionsDiscoveryJob>
19#include <Akonadi/SpecialMailCollectionsRequestJob>
20#include <QApplication>
21
22#include <KIdentityManagementCore/Identity>
23#include <KIdentityManagementCore/IdentityManager>
24
25#include <KLocalizedString>
26#include <KMessageBox>
27
28namespace MailCommon
29{
30class KernelPrivate
31{
32public:
33 KernelPrivate()
34 : kernel(new Kernel)
35 {
36 }
37
38 ~KernelPrivate()
39 {
40 qCDebug(MAILCOMMON_LOG);
41 delete kernel;
42 }
43
44 Kernel *const kernel;
45};
46
47Q_GLOBAL_STATIC(KernelPrivate, sInstance)
48
49Kernel::Kernel(QObject *parent)
50 : QObject(parent)
51{
52 mKernelIf = nullptr;
53 mSettingsIf = nullptr;
54 mFilterIf = nullptr;
55 mImapResourceManager = new PimCommon::ImapResourceCapabilitiesManager(this);
56}
57
58Kernel::~Kernel()
59{
60 qCDebug(MAILCOMMON_LOG);
61}
62
63Kernel *Kernel::self()
64{
65 return sInstance->kernel; // will create it
66}
67
69{
70 mKernelIf = kernelIf;
71}
72
73bool Kernel::kernelIsRegistered() const
74{
75 return mKernelIf != nullptr;
76}
77
78IKernel *Kernel::kernelIf() const
79{
80 Q_ASSERT(mKernelIf);
81 return mKernelIf;
82}
83
85{
86 mSettingsIf = settingsIf;
87}
88
89ISettings *Kernel::settingsIf() const
90{
91 Q_ASSERT(mSettingsIf);
92 return mSettingsIf;
93}
94
96{
97 mFilterIf = filterIf;
98}
99
100IFilter *Kernel::filterIf() const
101{
102 Q_ASSERT(mFilterIf);
103 return mFilterIf;
104}
105
106PimCommon::ImapResourceCapabilitiesManager *Kernel::imapResourceManager() const
107{
108 return mImapResourceManager;
109}
110
115
116Akonadi::Collection Kernel::trashCollectionFolder()
117{
119}
120
121Akonadi::Collection Kernel::inboxCollectionFolder()
122{
124}
125
126Akonadi::Collection Kernel::outboxCollectionFolder()
127{
129}
130
131Akonadi::Collection Kernel::sentCollectionFolder()
132{
134}
135
136Akonadi::Collection Kernel::draftsCollectionFolder()
137{
139}
140
141Akonadi::Collection Kernel::templatesCollectionFolder()
142{
144}
145
146Akonadi::Collection Kernel::spamsCollectionFolder()
147{
149}
150
151bool Kernel::isSystemFolderCollection(const Akonadi::Collection &col)
152{
153 return col == inboxCollectionFolder() || col == outboxCollectionFolder() || col == sentCollectionFolder() || col == trashCollectionFolder()
154 || col == draftsCollectionFolder() || col == templatesCollectionFolder() || col == spamsCollectionFolder();
155}
156
158{
159 return col == inboxCollectionFolder();
160}
161
162//-----------------------------------------------------------------------------
163void Kernel::initFolders()
164{
165 qCDebug(MAILCOMMON_LOG) << "Initialized and looking for specialcollection folders.";
166 findCreateDefaultCollection(Akonadi::SpecialMailCollections::Inbox);
167 findCreateDefaultCollection(Akonadi::SpecialMailCollections::Outbox);
168 findCreateDefaultCollection(Akonadi::SpecialMailCollections::SentMail);
169 findCreateDefaultCollection(Akonadi::SpecialMailCollections::Drafts);
170 findCreateDefaultCollection(Akonadi::SpecialMailCollections::Trash);
171 findCreateDefaultCollection(Akonadi::SpecialMailCollections::Templates);
172
173 auto job = new Akonadi::SpecialMailCollectionsDiscoveryJob(this);
174 job->start();
175 // we don't connect to the job directly: it will register stuff into SpecialMailCollections, which will Q_EMIT signals.
176}
177
178void Kernel::findCreateDefaultCollection(Akonadi::SpecialMailCollections::Type type)
179{
180 if (Akonadi::SpecialMailCollections::self()->hasDefaultCollection(type)) {
182
183 if (!(col.rights() & Akonadi::Collection::AllRights)) {
184 emergencyExit(i18n("You do not have read/write permission to your inbox folder."));
185 }
186 } else {
188
189 connect(job, &Akonadi::SpecialMailCollectionsRequestJob::result, this, &Kernel::createDefaultCollectionDone);
190
191 job->requestDefaultCollection(type);
192 }
193}
194
195void Kernel::createDefaultCollectionDone(KJob *job)
196{
197 if (job->error()) {
198 emergencyExit(job->errorText());
199 return;
200 }
201
202 auto requestJob = qobject_cast<Akonadi::SpecialMailCollectionsRequestJob *>(job);
203
204 const Akonadi::Collection col = requestJob->collection();
205 if (!(col.rights() & Akonadi::Collection::AllRights)) {
206 emergencyExit(i18n("You do not have read/write permission to your inbox folder."));
207 }
214
217 this,
218 &Kernel::slotDefaultCollectionsChanged,
220}
221
222void Kernel::slotDefaultCollectionsChanged()
223{
226 this,
227 &Kernel::slotDefaultCollectionsChanged);
228 initFolders();
229}
230
231void Kernel::emergencyExit(const QString &reason)
232{
233 QString mesg;
234 if (reason.isEmpty()) {
235 mesg = i18n("The Email program encountered a fatal error and will terminate now");
236 } else {
237 mesg = i18n(
238 "The Email program encountered a fatal error and will terminate now.\n"
239 "The error was:\n%1",
240 reason);
241 }
242
243 qCWarning(MAILCOMMON_LOG) << mesg;
244
245 // Show error box for the first error that caused emergencyExit.
246 static bool s_showingErrorBox = false;
247 if (!s_showingErrorBox) {
248 s_showingErrorBox = true;
249 if (qApp) { // see bug 313104
250 KMessageBox::error(nullptr, mesg);
251 }
252 ::exit(1);
253 }
254}
255
257{
259 return true;
260 }
261
262 return folderIsDrafts(col);
263}
264
265bool Kernel::folderIsDrafts(const Akonadi::Collection &col)
266{
268 return true;
269 }
270
271 const QString idString = QString::number(col.id());
272 if (idString.isEmpty()) {
273 return false;
274 }
275
276 // search the identities if the folder matches the drafts-folder
277 const KIdentityManagementCore::IdentityManager *im = KernelIf->identityManager();
278 KIdentityManagementCore::IdentityManager::ConstIterator end(im->end());
279 for (KIdentityManagementCore::IdentityManager::ConstIterator it = im->begin(); it != end; ++it) {
280 if ((*it).drafts() == idString) {
281 return true;
282 }
283 }
284 return false;
285}
286
287bool Kernel::folderIsTemplates(const Akonadi::Collection &col)
288{
290 return true;
291 }
292 const QString idString = QString::number(col.id());
293 if (idString.isEmpty()) {
294 return false;
295 }
296
297 // search the identities if the folder matches the templates-folder
298 const KIdentityManagementCore::IdentityManager *im = KernelIf->identityManager();
299 KIdentityManagementCore::IdentityManager::ConstIterator end(im->end());
300 for (KIdentityManagementCore::IdentityManager::ConstIterator it = im->begin(); it != end; ++it) {
301 if ((*it).templates() == idString) {
302 return true;
303 }
304 }
305 return false;
306}
307
308bool Kernel::folderIsSpams(const Akonadi::Collection &col)
309{
311 return true;
312 }
313
314 const QString idString = QString::number(col.id());
315 if (idString.isEmpty()) {
316 return false;
317 }
318
319 // search the identities if the folder matches the drafts-folder
320 const KIdentityManagementCore::IdentityManager *im = KernelIf->identityManager();
321 KIdentityManagementCore::IdentityManager::ConstIterator end(im->end());
322 for (KIdentityManagementCore::IdentityManager::ConstIterator it = im->begin(); it != end; ++it) {
323 if (!(*it).disabledSpam() && (*it).spam() == idString) {
324 return true;
325 }
326 }
327 return false;
328}
329
339
341{
343 return true;
344 }
345
346 const Akonadi::AgentInstance::List lst = MailCommon::Util::agentInstances();
347 for (const Akonadi::AgentInstance &agent : lst) {
349 if (col == trash) {
350 return true;
351 }
352 }
353 return false;
354}
355
357{
359 return true;
360 }
361 const QString idString = QString::number(col.id());
362 if (idString.isEmpty()) {
363 return false;
364 }
365
366 // search the identities if the folder matches the sent-folder
367 const KIdentityManagementCore::IdentityManager *im = KernelIf->identityManager();
368 KIdentityManagementCore::IdentityManager::ConstIterator end(im->end());
369 for (KIdentityManagementCore::IdentityManager::ConstIterator it = im->begin(); it != end; ++it) {
370 if ((*it).fcc() == idString) {
371 return true;
372 }
373 }
374 return false;
375}
376
377bool Kernel::folderIsInbox(const Akonadi::Collection &collection)
378{
379 const QString collectionRemoteIdLower = collection.remoteId().toLower();
380 if (collectionRemoteIdLower == QLatin1StringView("inbox") || collectionRemoteIdLower == QLatin1StringView("/inbox")
381 || collectionRemoteIdLower == QLatin1StringView(".inbox") || collectionRemoteIdLower == QLatin1StringView("|inbox")) {
382 return true;
383 }
384 // Fix order. Remoteid is not "inbox" when translated
385 if (Akonadi::SpecialMailCollections::self()->specialCollectionType(collection) == Akonadi::SpecialMailCollections::Inbox) {
386 return true;
387 }
388
389 // MBOX is one folder only, treat as inbox
390 if (collection.resource().contains(MBOX_RESOURCE_IDENTIFIER)) {
391 return true;
392 }
393 return false;
394}
395
396QMap<QString, Akonadi::Collection::Id> Kernel::pop3ResourceTargetCollection()
397{
398 QMap<QString, Akonadi::Collection::Id> mapIdentifierCollectionId;
399 const Akonadi::AgentInstance::List lst = MailCommon::Util::agentInstances();
400 for (const Akonadi::AgentInstance &type : lst) {
401 if (type.status() == Akonadi::AgentInstance::Broken) {
402 continue;
403 }
404 const QString typeIdentifier = type.identifier();
405 if (typeIdentifier.contains(POP3_RESOURCE_IDENTIFIER)) {
406 MailCommon::ResourceReadConfigFile resourceFile(typeIdentifier);
407 const KConfigGroup grp = resourceFile.group(QStringLiteral("General"));
408 if (grp.isValid()) {
409 const Akonadi::Collection::Id targetCollection = grp.readEntry(QStringLiteral("targetCollection"), -1);
410 mapIdentifierCollectionId.insert(typeIdentifier, targetCollection);
411 }
412 }
413 }
414 return mapIdentifierCollectionId;
415}
416
417#if MAILCOMMON_HAVE_ACTIVITY_SUPPORT
418PimCommonActivities::ActivitiesBaseManager *Kernel::activitiesBaseManager() const
419{
420 return mActivitiesBaseManager;
421}
422
423void Kernel::registerActivitiesBaseManager(PimCommonActivities::ActivitiesBaseManager *manager)
424{
425 mActivitiesBaseManager = manager;
426}
427#endif
428}
429
430#include "moc_mailkernel.cpp"
QList< AgentInstance > List
static AgentManager * self()
AgentInstance instance(const QString &identifier) const
QString resource() const
bool isValid() const
Rights rights() const
QString remoteId() const
static Collection updatedCollection(const QAbstractItemModel *model, qint64 collectionId)
static SpecialMailCollections * self()
Akonadi::Collection collection(Type type, const AgentInstance &instance) const
Akonadi::Collection defaultCollection(Type type) const
KConfigGroup group(const QString &group)
bool isValid() const
QString readEntry(const char *key, const char *aDefault=nullptr) const
int error() const
void result(KJob *job)
QString errorText() const
Filter related interface.
Generic interface for mail kernels.
Interface to access some settings.
void registerSettingsIf(ISettings *settingsIf)
Registers the interface dealing with mail settings.
Akonadi::Collection trashCollectionFromResource(const Akonadi::Collection &col)
Returns the trash folder for the resource which col belongs to.
bool folderIsDraftOrOutbox(const Akonadi::Collection &collection)
Returns true if the folder is either the outbox or one of the drafts-folders.
Akonadi::Collection collectionFromId(Akonadi::Collection::Id id) const
Returns the collection associated with the given id, or an invalid collection if not found.
void registerKernelIf(IKernel *kernelIf)
Registers the interface dealing with main mail functionality.
bool folderIsTrash(const Akonadi::Collection &collection)
Returns true if the folder is a trash folder.
bool isMainFolderCollection(const Akonadi::Collection &col)
Returns true if this folder is the inbox on the local disk.
bool folderIsSentMailFolder(const Akonadi::Collection &)
Returns true if the folder is one of the sent-mail folders.
void registerFilterIf(IFilter *filterIf)
Registers the interface dealing with mail settings.
QString i18n(const char *text, const TYPE &arg...)
Type type(const QSqlDatabase &db)
void error(QWidget *parent, const QString &text, const QString &title, const KGuiItem &buttonOk, Options options=Notify)
const QList< QKeySequence > & end()
The filter dialog.
iterator insert(const Key &key, const T &value)
bool contains(QChar ch, Qt::CaseSensitivity cs) const const
bool isEmpty() const const
QString number(double n, char format, int precision)
QString toLower() const const
UniqueConnection
QFuture< ArgsType< Signal > > connect(Sender *sender, Signal signal)
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Apr 4 2025 12:00:26 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.