Messagelib

widgetbase.cpp
1/******************************************************************************
2 *
3 * SPDX-FileCopyrightText: 2008 Szymon Tomasz Stefanek <pragma@kvirc.net>
4 *
5 * SPDX-License-Identifier: GPL-2.0-or-later
6 *
7 *******************************************************************************/
8
9#include "core/widgetbase.h"
10#include "config-messagelist.h"
11#include "core/aggregation.h"
12#include "core/filter.h"
13#include "core/filtersavedmanager.h"
14#include "core/manager.h"
15#include "core/messageitem.h"
16#include "core/model.h"
17#include "core/optionset.h"
18#include "core/storagemodelbase.h"
19#include "core/theme.h"
20#include "core/view.h"
21#include "core/widgets/quicksearchwarning.h"
22#if !FORCE_DISABLE_AKONADI_SEARCH
23#include "core/widgets/searchcollectionindexingwarning.h"
24#endif
25#include "core/widgets/tablockedwarning.h"
26#include "messagelistsettings.h"
27#include "widgets/searchlinestatus.h"
28
29#include "utils/configureaggregationsdialog.h"
30#include "utils/configurethemesdialog.h"
31
32#include <QActionGroup>
33#include <QHeaderView>
34#include <QPointer>
35#include <QTimer>
36#include <QVBoxLayout>
37#include <QVariant>
38
39#include "messagelist_debug.h"
40#include <KLocalizedString>
41#include <KMessageBox>
42#include <QAction>
43#include <QComboBox>
44#include <QMenu>
45
46#include "core/widgets/filternamedialog.h"
47#include <Akonadi/Collection>
48#include <Akonadi/MessageStatus>
49#include <chrono>
50
51#include "core/widgets/searchlinecommand.h"
52using namespace std::chrono_literals;
53
54using namespace MessageList::Core;
55
56class Widget::WidgetPrivate
57{
58public:
59 WidgetPrivate(Widget *owner)
60 : q(owner)
61 {
62 }
63
64 /**
65 * Small helper for switching SortOrder::MessageSorting and SortOrder::SortDirection
66 * on the fly.
67 * After doing this, the sort indicator in the header is updated.
68 */
69 void switchMessageSorting(SortOrder::MessageSorting messageSorting, SortOrder::SortDirection sortDirection, int logicalHeaderColumnIndex);
70
71 /**
72 * Check if our sort order can still be used with this aggregation.
73 * This can happen if the global aggregation changed, for example we can now
74 * have "most recent in subtree" sorting with an aggregation without threading.
75 * If this happens, reset to the default sort order and don't use the global sort
76 * order.
77 */
78 void checkSortOrder(const StorageModel *storageModel);
79
80 void setDefaultAggregationForStorageModel(const StorageModel *storageModel);
81 void setDefaultThemeForStorageModel(const StorageModel *storageModel);
82 void setDefaultSortOrderForStorageModel(const StorageModel *storageModel);
83 void applyFilter();
84
85 Widget *const q;
86
87 QuickSearchWarning *quickSearchWarning = nullptr;
88#if !FORCE_DISABLE_AKONADI_SEARCH
89 SearchCollectionIndexingWarning *searchCollectionIndexingWarning = nullptr;
90#endif
91 TabLockedWarning *tabLockedWarning = nullptr;
92 QuickSearchLine *quickSearchLine = nullptr;
93 View *mView = nullptr;
94 QString mLastAggregationId;
95 QString mLastThemeId;
96 QTimer *mSearchTimer = nullptr;
97 StorageModel *mStorageModel = nullptr; ///< The currently displayed storage. The storage itself
98 /// is owned by MessageList::Widget.
99 Aggregation *mAggregation = nullptr; ///< The currently set aggregation mode, a deep copy
100 Theme *mTheme = nullptr; ///< The currently set theme, a deep copy
101 SortOrder mSortOrder; ///< The currently set sort order
102 Filter *mFilter = nullptr; ///< The currently applied filter, owned by us.
103 bool mStorageUsesPrivateTheme = false; ///< true if the current folder does not use the global theme
104 bool mStorageUsesPrivateAggregation = false; ///< true if the current folder does not use the global aggregation
105 bool mStorageUsesPrivateSortOrder = false; ///< true if the current folder does not use the global sort order
106 Akonadi::Collection mCurrentFolder; ///< The current folder
107 int mCurrentStatusFilterIndex = 0;
108 bool mStatusFilterComboPopulationInProgress = false;
109 bool mLockTab = false;
110};
111
112Widget::Widget(QWidget *pParent)
113 : QWidget(pParent)
114 , d(new WidgetPrivate(this))
115{
116 Manager::registerWidget(this);
117 connect(Manager::instance(), &Manager::aggregationsChanged, this, &Widget::aggregationsChanged);
118 connect(Manager::instance(), &Manager::themesChanged, this, &Widget::themesChanged);
119
120 setAutoFillBackground(true);
121 setObjectName(QLatin1StringView("messagelistwidget"));
122
123 auto g = new QVBoxLayout(this);
124 g->setContentsMargins({});
125 g->setSpacing(0);
126
127 d->quickSearchLine = new QuickSearchLine(this);
128 d->quickSearchLine->setObjectName(QLatin1StringView("quicksearchline"));
129 connect(d->quickSearchLine, &QuickSearchLine::clearButtonClicked, this, &Widget::searchEditClearButtonClicked);
130
131 connect(d->quickSearchLine, &QuickSearchLine::searchEditTextEdited, this, &Widget::searchEditTextEdited);
132 connect(d->quickSearchLine, &QuickSearchLine::searchOptionChanged, this, &Widget::searchEditTextEdited);
133 connect(d->quickSearchLine, &QuickSearchLine::statusButtonsClicked, this, &Widget::slotStatusButtonsClicked);
134 connect(d->quickSearchLine, &QuickSearchLine::forceLostFocus, this, &Widget::forceLostFocus);
135 connect(d->quickSearchLine, &QuickSearchLine::saveFilter, this, &Widget::slotSaveFilter);
136 connect(d->quickSearchLine, &QuickSearchLine::activateFilter, this, &Widget::slotActivateFilter);
137 g->addWidget(d->quickSearchLine, 0);
138 d->quickSearchWarning = new QuickSearchWarning(this);
139 g->addWidget(d->quickSearchWarning, 0);
140#if !FORCE_DISABLE_AKONADI_SEARCH
141 d->searchCollectionIndexingWarning = new SearchCollectionIndexingWarning(this);
142 g->addWidget(d->searchCollectionIndexingWarning, 0);
143#endif
144
145 d->tabLockedWarning = new TabLockedWarning(this);
146 g->addWidget(d->tabLockedWarning, 0);
147 connect(d->tabLockedWarning, &TabLockedWarning::unlockTabRequested, this, [this]() {
148 setLockTab(false);
149 Q_EMIT unlockTabRequested();
150 // Fix icon!
151 });
152
153 d->mView = new View(this);
154 d->mView->setFrameStyle(QFrame::NoFrame);
155 d->mView->setSortOrder(&d->mSortOrder);
156 d->mView->setObjectName(QLatin1StringView("messagealistview"));
157 g->addWidget(d->mView, 1);
158
160 d->mSearchTimer = nullptr;
161}
162
163Widget::~Widget()
164{
165 d->mView->setStorageModel(nullptr);
166
167 Manager::unregisterWidget(this);
168
169 delete d->mSearchTimer;
170 delete d->mTheme;
171 delete d->mAggregation;
172 delete d->mFilter;
173 delete d->mStorageModel;
174}
175
176void Widget::slotActivateFilter(Filter *f)
177{
178 // setFilter reset filter => get info before to call setFilter
179 const auto status = f->status();
180 const auto options = f->currentOptions();
181 const auto str = f->searchString();
182 setFilter(f);
183 d->quickSearchLine->searchEdit()->setText(str);
184 d->quickSearchLine->setSearchOptions(options);
185 d->quickSearchLine->setFilterMessageStatus(status);
186}
187
188void Widget::slotSaveFilter()
189{
190 if (d->mFilter) {
191 QPointer<FilterNameDialog> dlg = new FilterNameDialog(this);
192 dlg->setExistingFilterNames(FilterSavedManager::self()->existingFilterNames());
193 if (dlg->exec()) {
194 FilterSavedManager::self()->saveFilter(d->mFilter, dlg->filterName(), dlg->iconName());
195 }
196 delete dlg;
197 } else {
198 KMessageBox::information(this, i18n("No filter defined."), i18nc("@title:window", "Create Filter"));
199 }
200}
201
203{
204 QLineEdit *const lineEdit = d->quickSearchLine->searchEdit();
205 if (!show) {
206 // if we hide it we do not want to apply the filter,
207 // otherwise someone is maybe stuck with x new emails
208 // and cannot read it because of filter
209 lineEdit->clear();
210
211 // we focus the message list if we hide the searchbar
212 d->mView->setFocus(Qt::OtherFocusReason);
213 } else {
214 // on show: we focus the lineedit for fast filtering
216 if (d->mFilter) {
217 resetFilter();
218 }
219 }
220 d->quickSearchLine->changeQuicksearchVisibility(show);
221 MessageListSettings::self()->setShowQuickSearch(show);
222}
223
225{
226 if (d->mStatusFilterComboPopulationInProgress) {
227 return;
228 }
229 d->mStatusFilterComboPopulationInProgress = true;
230 QComboBox *tagFilterComboBox = d->quickSearchLine->tagFilterComboBox();
231 d->mCurrentStatusFilterIndex = (tagFilterComboBox->currentIndex() != -1) ? tagFilterComboBox->currentIndex() : 0;
232 disconnect(tagFilterComboBox, &QComboBox::currentIndexChanged, this, &Widget::statusSelected);
233
234 tagFilterComboBox->clear();
235
237}
238
239void Widget::addMessageTagItem(const QPixmap &icon, const QString &text, const QVariant &data)
240{
241 d->quickSearchLine->tagFilterComboBox()->addItem(icon, text, data);
242}
243
245{
246 d->quickSearchLine->updateComboboxVisibility();
247 connect(d->quickSearchLine->tagFilterComboBox(), &QComboBox::currentIndexChanged, this, &Widget::statusSelected);
248 d->quickSearchLine->tagFilterComboBox()->setCurrentIndex(
249 d->mCurrentStatusFilterIndex >= d->quickSearchLine->tagFilterComboBox()->count() ? 0 : d->mCurrentStatusFilterIndex);
250 d->mStatusFilterComboPopulationInProgress = false;
251}
252
257
258MessageList::Core::SearchMessageByButtons::SearchOptions Widget::currentOptions() const
259{
260 return d->quickSearchLine->searchOptions();
261}
262
264{
265 if (d->mFilter) {
266 return d->mFilter->status();
267 }
268 return {};
269}
270
271QList<SearchLineCommand::SearchLineInfo> Widget::searchLineCommands() const
272{
273 if (d->mFilter) {
274 return d->mFilter->searchLineCommands();
275 }
276 return {};
277}
278
280{
281 if (d->mFilter) {
282 return d->mFilter->searchString();
283 }
284 return {};
285}
286
287void Widget::WidgetPrivate::setDefaultAggregationForStorageModel(const StorageModel *storageModel)
288{
289 const Aggregation *opt = Manager::instance()->aggregationForStorageModel(storageModel, &mStorageUsesPrivateAggregation);
290
291 Q_ASSERT(opt);
292
293 delete mAggregation;
294 mAggregation = new Aggregation(*opt);
295
296 mView->setAggregation(mAggregation);
297
298 mLastAggregationId = opt->id();
299}
300
301void Widget::WidgetPrivate::setDefaultThemeForStorageModel(const StorageModel *storageModel)
302{
303 const Theme *opt = Manager::instance()->themeForStorageModel(storageModel, &mStorageUsesPrivateTheme);
304
305 Q_ASSERT(opt);
306
307 delete mTheme;
308 mTheme = new Theme(*opt);
309
310 mView->setTheme(mTheme);
311
312 mLastThemeId = opt->id();
313}
314
315void Widget::WidgetPrivate::checkSortOrder(const StorageModel *storageModel)
316{
317 if (storageModel && mAggregation && !mSortOrder.validForAggregation(mAggregation)) {
318 qCDebug(MESSAGELIST_LOG) << "Could not restore sort order for folder" << storageModel->id();
319 mSortOrder = SortOrder::defaultForAggregation(mAggregation, mSortOrder);
320
321 // Change the global sort order if the sort order didn't fit the global aggregation.
322 // Otherwise, if it is a per-folder aggregation, make the sort order per-folder too.
323 if (mStorageUsesPrivateAggregation) {
324 mStorageUsesPrivateSortOrder = true;
325 }
326 if (mStorageModel) {
327 Manager::instance()->saveSortOrderForStorageModel(storageModel, mSortOrder, mStorageUsesPrivateSortOrder);
328 }
329 switchMessageSorting(mSortOrder.messageSorting(), mSortOrder.messageSortDirection(), -1);
330 }
331}
332
333void Widget::WidgetPrivate::setDefaultSortOrderForStorageModel(const StorageModel *storageModel)
334{
335 // Load the sort order from config and update column headers
336 mSortOrder = Manager::instance()->sortOrderForStorageModel(storageModel, &mStorageUsesPrivateSortOrder);
337 switchMessageSorting(mSortOrder.messageSorting(), mSortOrder.messageSortDirection(), -1);
338 checkSortOrder(storageModel);
339}
340
341void Widget::saveCurrentSelection()
342{
343 if (d->mStorageModel) {
344 // Save the current selection
345 MessageItem *lastSelectedMessageItem = d->mView->currentMessageItem(false);
346 if (lastSelectedMessageItem) {
347 d->mStorageModel->savePreSelectedMessage(lastSelectedMessageItem->uniqueId());
348 }
349 }
350}
351
353{
354 if (storageModel == d->mStorageModel) {
355 return; // nothing to do here
356 }
357
358 d->setDefaultAggregationForStorageModel(storageModel);
359 d->setDefaultThemeForStorageModel(storageModel);
360 d->setDefaultSortOrderForStorageModel(storageModel);
361
362 const bool isLocked = d->quickSearchLine->searchEdit()->locked();
363 // qDebug() << " isLocked " << isLocked;
364 if (!isLocked) {
365 if (d->mSearchTimer) {
366 d->mSearchTimer->stop();
367 delete d->mSearchTimer;
368 d->mSearchTimer = nullptr;
369 }
370
371 d->quickSearchLine->searchEdit()->clear();
372
373 if (d->mFilter) {
374 resetFilter();
375 }
376 } else {
377 searchTimerFired();
378 }
379 StorageModel *oldModel = d->mStorageModel;
380
381 d->mStorageModel = storageModel;
382 d->mView->setStorageModel(d->mStorageModel, preSelectionMode);
383
384 delete oldModel;
385
386 d->quickSearchLine->tagFilterComboBox()->setEnabled(d->mStorageModel);
387 d->quickSearchLine->searchEdit()->setEnabled(d->mStorageModel);
388 d->quickSearchLine->setContainsOutboundMessages(d->mStorageModel->containsOutboundMessages());
389}
390
392{
393 return d->mStorageModel;
394}
395
397{
398 return d->quickSearchLine->searchEdit();
399}
400
401View *Widget::view() const
402{
403 return d->mView;
404}
405
406void Widget::themeMenuAboutToShow()
407{
408 if (!d->mStorageModel) {
409 return;
410 }
411
412 auto menu = qobject_cast<QMenu *>(sender());
413 if (!menu) {
414 return;
415 }
416 themeMenuAboutToShow(menu);
417}
418
419void Widget::themeMenuAboutToShow(QMenu *menu)
420{
421 menu->clear();
422
423 menu->addSection(i18n("Theme"));
424
425 auto grp = new QActionGroup(menu);
426
427 QList<Theme *> sortedThemes = Manager::instance()->themes().values();
428
429 QAction *act;
430
431 std::sort(sortedThemes.begin(), sortedThemes.end(), MessageList::Core::Theme::compareName);
432
433 for (const auto theme : std::as_const(sortedThemes)) {
434 act = menu->addAction(theme->name());
435 act->setCheckable(true);
436 grp->addAction(act);
437 act->setChecked(d->mLastThemeId == theme->id());
438 act->setData(QVariant(theme->id()));
439 connect(act, &QAction::triggered, this, &Widget::themeSelected);
440 }
441
442 menu->addSeparator();
443
444 act = menu->addAction(i18n("Configure..."));
445 connect(act, &QAction::triggered, this, &Widget::configureThemes);
446}
447
448void Widget::setPrivateSortOrderForStorage()
449{
450 if (!d->mStorageModel) {
451 return;
452 }
453
454 d->mStorageUsesPrivateSortOrder = !d->mStorageUsesPrivateSortOrder;
455
456 Manager::instance()->saveSortOrderForStorageModel(d->mStorageModel, d->mSortOrder, d->mStorageUsesPrivateSortOrder);
457}
458
459void Widget::configureThemes()
460{
461 auto dialog = new Utils::ConfigureThemesDialog(window());
462 dialog->selectTheme(d->mLastThemeId);
463 dialog->show();
464}
465
466void Widget::themeSelected(bool)
467{
468 if (!d->mStorageModel) {
469 return; // nuthin to do
470 }
471
472 auto act = qobject_cast<QAction *>(sender());
473 if (!act) {
474 return;
475 }
476
477 QVariant v = act->data();
478 const QString id = v.toString();
479
480 if (id.isEmpty()) {
481 return;
482 }
483
484 const Theme *opt = Manager::instance()->theme(id);
485
486 delete d->mTheme;
487 d->mTheme = new Theme(*opt);
488
489 d->mView->setTheme(d->mTheme);
490
491 d->mLastThemeId = opt->id();
492
493 // mStorageUsesPrivateTheme = false;
494
495 Manager::instance()->saveThemeForStorageModel(d->mStorageModel, opt->id(), d->mStorageUsesPrivateTheme);
496
497 d->mView->reload();
498}
499
500void Widget::aggregationMenuAboutToShow()
501{
502 auto menu = qobject_cast<QMenu *>(sender());
503 if (!menu) {
504 return;
505 }
506 aggregationMenuAboutToShow(menu);
507}
508
509void Widget::aggregationMenuAboutToShow(QMenu *menu)
510{
511 menu->clear();
512
513 menu->addSection(i18n("Aggregation"));
514
515 auto grp = new QActionGroup(menu);
516
517 QList<Aggregation *> sortedAggregations = Manager::instance()->aggregations().values();
518
519 QAction *act;
520
521 std::sort(sortedAggregations.begin(), sortedAggregations.end(), MessageList::Core::Aggregation::compareName);
522
523 for (const auto agg : std::as_const(sortedAggregations)) {
524 act = menu->addAction(agg->name());
525 act->setCheckable(true);
526 grp->addAction(act);
527 act->setChecked(d->mLastAggregationId == agg->id());
528 act->setData(QVariant(agg->id()));
529 connect(act, &QAction::triggered, this, &Widget::aggregationSelected);
530 }
531
532 menu->addSeparator();
533
534 act = menu->addAction(i18n("Configure..."));
535 act->setData(QVariant(QString()));
536 connect(act, &QAction::triggered, this, &Widget::aggregationSelected);
537}
538
539void Widget::aggregationSelected(bool)
540{
541 auto act = qobject_cast<QAction *>(sender());
542 if (!act) {
543 return;
544 }
545
546 QVariant v = act->data();
547 QString id = v.toString();
548
549 if (id.isEmpty()) {
550 auto dialog = new Utils::ConfigureAggregationsDialog(window());
551 dialog->selectAggregation(d->mLastAggregationId);
552 dialog->show();
553 return;
554 }
555
556 if (!d->mStorageModel) {
557 return; // nuthin to do
558 }
559
560 const Aggregation *opt = Manager::instance()->aggregation(id);
561
562 delete d->mAggregation;
563 d->mAggregation = new Aggregation(*opt);
564
565 d->mView->setAggregation(d->mAggregation);
566
567 d->mLastAggregationId = opt->id();
568
569 // mStorageUsesPrivateAggregation = false;
570
571 Manager::instance()->saveAggregationForStorageModel(d->mStorageModel, opt->id(), d->mStorageUsesPrivateAggregation);
572
573 // The sort order might not be valid anymore for this aggregation
574 d->checkSortOrder(d->mStorageModel);
575
576 d->mView->reload();
577}
578
579void Widget::sortOrderMenuAboutToShow()
580{
581 if (!d->mAggregation) {
582 return;
583 }
584
585 auto menu = qobject_cast<QMenu *>(sender());
586 if (!menu) {
587 return;
588 }
589 sortOrderMenuAboutToShow(menu);
590}
591
592void Widget::sortOrderMenuAboutToShow(QMenu *menu)
593{
594 menu->clear();
595
596 menu->addSection(i18n("Message Sort Order"));
597
598 QActionGroup *grp;
599 QAction *act;
600 QList<QPair<QString, int>> options;
601
602 grp = new QActionGroup(menu);
603
604 options = SortOrder::enumerateMessageSortingOptions(d->mAggregation->threading());
605 for (const auto &opt : std::as_const(options)) {
606 act = menu->addAction(opt.first);
607 act->setCheckable(true);
608 grp->addAction(act);
609 act->setChecked(d->mSortOrder.messageSorting() == opt.second);
610 act->setData(QVariant(opt.second));
611 }
612
613 connect(grp, &QActionGroup::triggered, this, &Widget::messageSortingSelected);
614
615 options = SortOrder::enumerateMessageSortDirectionOptions(d->mSortOrder.messageSorting());
616
617 if (options.size() >= 2) {
618 menu->addSection(i18n("Message Sort Direction"));
619
620 grp = new QActionGroup(menu);
621 for (const auto &opt : std::as_const(options)) {
622 act = menu->addAction(opt.first);
623 act->setCheckable(true);
624 grp->addAction(act);
625 act->setChecked(d->mSortOrder.messageSortDirection() == opt.second);
626 act->setData(QVariant(opt.second));
627 }
628
629 connect(grp, &QActionGroup::triggered, this, &Widget::messageSortDirectionSelected);
630 }
631
632 options = SortOrder::enumerateGroupSortingOptions(d->mAggregation->grouping());
633
634 if (options.size() >= 2) {
635 menu->addSection(i18n("Group Sort Order"));
636
637 grp = new QActionGroup(menu);
638 for (const auto &opt : std::as_const(options)) {
639 act = menu->addAction(opt.first);
640 act->setCheckable(true);
641 grp->addAction(act);
642 act->setChecked(d->mSortOrder.groupSorting() == opt.second);
643 act->setData(QVariant(opt.second));
644 }
645
646 connect(grp, &QActionGroup::triggered, this, &Widget::groupSortingSelected);
647 }
648
649 options = SortOrder::enumerateGroupSortDirectionOptions(d->mAggregation->grouping(), d->mSortOrder.groupSorting());
650
651 if (options.size() >= 2) {
652 menu->addSection(i18n("Group Sort Direction"));
653
654 grp = new QActionGroup(menu);
655 for (const auto &opt : std::as_const(options)) {
656 act = menu->addAction(opt.first);
657 act->setCheckable(true);
658 grp->addAction(act);
659 act->setChecked(d->mSortOrder.groupSortDirection() == opt.second);
660 act->setData(QVariant(opt.second));
661 }
662
663 connect(grp, &QActionGroup::triggered, this, &Widget::groupSortDirectionSelected);
664 }
665
666 menu->addSeparator();
667 act = menu->addAction(i18n("Folder Always Uses This Sort Order"));
668 act->setCheckable(true);
669 act->setChecked(d->mStorageUsesPrivateSortOrder);
670 connect(act, &QAction::triggered, this, &Widget::setPrivateSortOrderForStorage);
671}
672
673void Widget::WidgetPrivate::switchMessageSorting(SortOrder::MessageSorting messageSorting, SortOrder::SortDirection sortDirection, int logicalHeaderColumnIndex)
674{
675 mSortOrder.setMessageSorting(messageSorting);
676 mSortOrder.setMessageSortDirection(sortDirection);
677
678 // If the logicalHeaderColumnIndex was specified then we already know which
679 // column we should set the sort indicator to. If it wasn't specified (it's -1)
680 // then we need to find it out in the theme.
681
682 if (logicalHeaderColumnIndex == -1) {
683 // try to find the specified message sorting in the theme columns
684 const auto columns = mTheme->columns();
685 int idx = 0;
686
687 // First try with a well defined message sorting.
688
689 for (const auto column : std::as_const(columns)) {
690 if (!mView->header()->isSectionHidden(idx)) {
691 if (column->messageSorting() == messageSorting) {
692 // found a visible column with this message sorting
693 logicalHeaderColumnIndex = idx;
694 break;
695 }
696 }
697 ++idx;
698 }
699
700 // if still not found, try again with a wider range
701 if (logicalHeaderColumnIndex == -1) {
702 idx = 0;
703 for (const auto column : std::as_const(columns)) {
704 if (!mView->header()->isSectionHidden(idx)) {
705 if (((column->messageSorting() == SortOrder::SortMessagesBySenderOrReceiver)
706 || (column->messageSorting() == SortOrder::SortMessagesByReceiver) || (column->messageSorting() == SortOrder::SortMessagesBySender))
707 && ((messageSorting == SortOrder::SortMessagesBySenderOrReceiver) || (messageSorting == SortOrder::SortMessagesByReceiver)
708 || (messageSorting == SortOrder::SortMessagesBySender))) {
709 // found a visible column with this message sorting
710 logicalHeaderColumnIndex = idx;
711 break;
712 }
713 }
714 ++idx;
715 }
716 }
717 }
718
719 if (logicalHeaderColumnIndex == -1) {
720 // not found: either not a column-based sorting or the related column is hidden
721 mView->header()->setSortIndicatorShown(false);
722 return;
723 }
724
725 mView->header()->setSortIndicatorShown(true);
726
727 if (sortDirection == SortOrder::Ascending) {
728 mView->header()->setSortIndicator(logicalHeaderColumnIndex, Qt::AscendingOrder);
729 } else {
730 mView->header()->setSortIndicator(logicalHeaderColumnIndex, Qt::DescendingOrder);
731 }
732}
733
734void Widget::messageSortingSelected(QAction *action)
735{
736 if (!d->mAggregation) {
737 return;
738 }
739 if (!action) {
740 return;
741 }
742
743 if (!d->mStorageModel) {
744 return;
745 }
746
747 bool ok;
748 auto ord = static_cast<SortOrder::MessageSorting>(action->data().toInt(&ok));
749
750 if (!ok) {
751 return;
752 }
753
754 d->switchMessageSorting(ord, d->mSortOrder.messageSortDirection(), -1);
755 Manager::instance()->saveSortOrderForStorageModel(d->mStorageModel, d->mSortOrder, d->mStorageUsesPrivateSortOrder);
756
757 d->mView->reload();
758}
759
760void Widget::messageSortDirectionSelected(QAction *action)
761{
762 if (!d->mAggregation) {
763 return;
764 }
765 if (!action) {
766 return;
767 }
768 if (!d->mStorageModel) {
769 return;
770 }
771
772 bool ok;
773 auto ord = static_cast<SortOrder::SortDirection>(action->data().toInt(&ok));
774
775 if (!ok) {
776 return;
777 }
778
779 d->switchMessageSorting(d->mSortOrder.messageSorting(), ord, -1);
780 Manager::instance()->saveSortOrderForStorageModel(d->mStorageModel, d->mSortOrder, d->mStorageUsesPrivateSortOrder);
781
782 d->mView->reload();
783}
784
785void Widget::groupSortingSelected(QAction *action)
786{
787 if (!d->mAggregation) {
788 return;
789 }
790 if (!action) {
791 return;
792 }
793
794 if (!d->mStorageModel) {
795 return;
796 }
797
798 bool ok;
799 auto ord = static_cast<SortOrder::GroupSorting>(action->data().toInt(&ok));
800
801 if (!ok) {
802 return;
803 }
804
805 d->mSortOrder.setGroupSorting(ord);
806 Manager::instance()->saveSortOrderForStorageModel(d->mStorageModel, d->mSortOrder, d->mStorageUsesPrivateSortOrder);
807
808 d->mView->reload();
809}
810
811void Widget::groupSortDirectionSelected(QAction *action)
812{
813 if (!d->mAggregation) {
814 return;
815 }
816 if (!action) {
817 return;
818 }
819 if (!d->mStorageModel) {
820 return;
821 }
822
823 bool ok;
824 auto ord = static_cast<SortOrder::SortDirection>(action->data().toInt(&ok));
825
826 if (!ok) {
827 return;
828 }
829
830 d->mSortOrder.setGroupSortDirection(ord);
831 Manager::instance()->saveSortOrderForStorageModel(d->mStorageModel, d->mSortOrder, d->mStorageUsesPrivateSortOrder);
832
833 d->mView->reload();
834}
835
836void Widget::setFilter(Filter *filter)
837{
838 resetFilter();
839 d->mFilter = filter;
840 d->mView->model()->setFilter(d->mFilter);
841}
842
843void Widget::resetFilter()
844{
845 delete d->mFilter;
846 d->mFilter = nullptr;
847 d->mView->model()->setFilter(nullptr);
848 d->quickSearchLine->resetFilter();
849 d->quickSearchWarning->animatedHide();
850}
851
853{
854 if (!d->mTheme) {
855 return;
856 }
857
858 if (!d->mAggregation) {
859 return;
860 }
861
862 if (logicalIndex >= d->mTheme->columns().count()) {
863 return;
864 }
865
866 if (!d->mStorageModel) {
867 return;
868 }
869
870 auto column = d->mTheme->column(logicalIndex);
871 if (!column) {
872 return; // should never happen...
873 }
874
875 if (column->messageSorting() == SortOrder::NoMessageSorting) {
876 return; // this is a null op.
877 }
878
879 if (d->mSortOrder.messageSorting() == column->messageSorting()) {
880 // switch sort direction
881 if (d->mSortOrder.messageSortDirection() == SortOrder::Ascending) {
882 d->switchMessageSorting(d->mSortOrder.messageSorting(), SortOrder::Descending, logicalIndex);
883 } else {
884 d->switchMessageSorting(d->mSortOrder.messageSorting(), SortOrder::Ascending, logicalIndex);
885 }
886 } else {
887 // keep sort direction but switch sort order
888 d->switchMessageSorting(column->messageSorting(), d->mSortOrder.messageSortDirection(), logicalIndex);
889 }
890 Manager::instance()->saveSortOrderForStorageModel(d->mStorageModel, d->mSortOrder, d->mStorageUsesPrivateSortOrder);
891
892 d->mView->reload();
893}
894
896{
897 d->setDefaultThemeForStorageModel(d->mStorageModel);
898
899 d->mView->reload();
900}
901
903{
904 d->setDefaultAggregationForStorageModel(d->mStorageModel);
905 d->checkSortOrder(d->mStorageModel);
906
907 d->mView->reload();
908}
909
911{
912 // nothing here: must be overridden in derived classes
914}
915
916void Widget::tagIdSelected(const QVariant &data)
917{
918 const QString tagId = data.toString();
919
920 if (tagId.isEmpty()) {
921 if (d->mFilter) {
922 if (d->mFilter->isEmpty()) {
923 resetFilter();
924 return;
925 }
926 }
927 } else {
928 if (!d->mFilter) {
929 d->mFilter = new Filter();
930 }
931 d->mFilter->setTagId({tagId});
932 }
933
934 d->mView->model()->setFilter(d->mFilter);
935}
936
937void Widget::setLockTab(bool lock)
938{
939 d->mLockTab = lock;
940 if (lock) {
941 d->tabLockedWarning->animatedShow();
942 } else {
943 d->tabLockedWarning->animatedHide();
944 }
945}
946
947bool Widget::isLocked() const
948{
949 return d->mLockTab;
950}
951
952void Widget::statusSelected(int index)
953{
954 if (index == 0) {
955 resetFilter();
956 return;
957 }
958 tagIdSelected(d->quickSearchLine->tagFilterComboBox()->itemData(index));
959 d->mView->model()->setFilter(d->mFilter);
960}
961
962void Widget::searchEditTextEdited()
963{
964 // This slot is called whenever the user edits the search QLineEdit.
965 // Since the user is likely to type more than one character
966 // so we start the real search after a short delay in order to catch
967 // multiple textEdited() signals.
968
969 if (!d->mSearchTimer) {
970 d->mSearchTimer = new QTimer(this);
971 connect(d->mSearchTimer, &QTimer::timeout, this, &Widget::searchTimerFired);
972 } else {
973 d->mSearchTimer->stop(); // eventually
974 }
975
976 d->mSearchTimer->setSingleShot(true);
977 d->mSearchTimer->start(1s);
978}
979
980void Widget::slotStatusButtonsClicked()
981{
982 // We also arbitrarily set tagId to an empty string, though we *could* allow filtering
983 // by status AND tag...
984 if (d->mFilter) {
985 d->mFilter->setTagId(QStringList());
986 }
987
988 auto lst = d->quickSearchLine->status();
989 if (lst.isEmpty()) {
990 if (d->mFilter) {
991 d->mFilter->setStatus(lst);
992 if (d->mFilter->isEmpty()) {
993 qCDebug(MESSAGELIST_LOG) << " RESET FILTER";
994 resetFilter();
995 return;
996 }
997 }
998 } else {
999 // don't have this status bit
1000 if (!d->mFilter) {
1001 d->mFilter = new Filter();
1002 }
1003 d->mFilter->setStatus(lst);
1004 }
1005
1006 d->mView->model()->setFilter(d->mFilter);
1007}
1008
1009void Widget::searchTimerFired()
1010{
1011 // A search is pending.
1012
1013 if (d->mSearchTimer) {
1014 d->mSearchTimer->stop();
1015 }
1016
1017 if (!d->mFilter) {
1018 d->mFilter = new Filter();
1019 }
1020
1021 const QString text = d->quickSearchLine->searchEdit()->text();
1022 if (!text.isEmpty()) {
1023 d->quickSearchLine->addCompletionItem(text);
1024 }
1025 d->mFilter->setCurrentFolder(d->mCurrentFolder);
1026 SearchLineCommand command;
1027 command.parseSearchLineCommand(text);
1028 if (command.hasOnlyOneLiteralCommand()) {
1029 SearchLineCommand commandChanged;
1030 const SearchMessageByButtons::SearchOptions options = d->quickSearchLine->searchOptions();
1031 d->mFilter->setOptions(options);
1032 SearchLineCommand::SearchLineInfo info;
1033 info.argument = command.searchLineInfo().at(0).argument;
1034 if (options & SearchMessageByButtons::SearchEveryWhere) {
1035 info.type = SearchLineCommand::SearchLineType::Literal;
1036 } else if (options & SearchMessageByButtons::SearchAgainstSubject) {
1037 info.type = SearchLineCommand::SearchLineType::Subject;
1038 } else if (options & SearchMessageByButtons::SearchAgainstBody) {
1039 info.type = SearchLineCommand::SearchLineType::Body;
1040 } else if (options & SearchMessageByButtons::SearchAgainstFrom) {
1041 info.type = SearchLineCommand::SearchLineType::From;
1042 } else if (options & SearchMessageByButtons::SearchAgainstBcc) {
1043 info.type = SearchLineCommand::SearchLineType::Bcc;
1044 } else if (options & SearchMessageByButtons::SearchAgainstCc) {
1045 info.type = SearchLineCommand::SearchLineType::Cc;
1046 } else if (options & SearchMessageByButtons::SearchAgainstTo) {
1047 info.type = SearchLineCommand::SearchLineType::To;
1048 }
1049 commandChanged.setSearchLineInfo({info});
1050 command = commandChanged;
1051 }
1052 // qDebug() << " text " << text << " command " << command.searchLineInfo();
1053
1054 d->mFilter->setSearchString(command);
1055 if (d->mFilter->isEmpty()) {
1056 resetFilter();
1057 return;
1058 }
1059
1060 d->mView->model()->setFilter(d->mFilter);
1061}
1062
1063void Widget::searchEditClearButtonClicked()
1064{
1065 if (!d->mFilter) {
1066 return;
1067 }
1068
1069 resetFilter();
1070
1071 d->mView->scrollTo(d->mView->currentIndex(), QAbstractItemView::PositionAtCenter);
1072}
1073
1077
1081
1085
1089
1091{
1092}
1093
1097
1101
1105
1109
1111{
1112 Q_UNUSED(msg)
1113 Q_UNUSED(set)
1114 Q_UNUSED(clear)
1115}
1116
1117void Widget::focusQuickSearch(const QString &selectedText)
1118{
1119 d->quickSearchLine->focusQuickSearch(selectedText);
1120}
1121
1123{
1124 return d->mView->isThreaded();
1125}
1126
1128{
1129 return d->mView->selectionEmpty();
1130}
1131
1132Akonadi::Collection Widget::currentFolder() const
1133{
1134 return d->mCurrentFolder;
1135}
1136
1138{
1139 if (!d->mLockTab) {
1140 d->mCurrentFolder = collection;
1141#if !FORCE_DISABLE_AKONADI_SEARCH
1142 d->searchCollectionIndexingWarning->setCollection(collection);
1143#endif
1144 }
1145}
1146
1147bool Widget::searchEditHasFocus() const
1148{
1149 return d->quickSearchLine->searchEdit()->hasFocus();
1150}
1151
1152#include "moc_widgetbase.cpp"
constexpr bool isEmpty() const
A set of aggregation options that can be applied to the MessageList::Model in a single shot.
Definition aggregation.h:29
This class is responsible of matching messages that should be displayed in the View.
Definition filter.h:34
The MessageItem class.
Definition messageitem.h:36
const QString & id() const
Returns the unique id of this OptionSet.
Definition optionset.h:51
The QuickSearchLine class.
GroupSorting
How to sort the groups If you add values here please look at the implementations of the enumerate* fu...
Definition sortorder.h:35
SortDirection
The "generic" sort direction: used for groups and for messages If you add values here please look at ...
Definition sortorder.h:50
static QList< QPair< QString, int > > enumerateGroupSortingOptions(Aggregation::Grouping g)
Enumerates the group sorting options compatible with the specified Grouping.
Definition sortorder.cpp:98
static QList< QPair< QString, int > > enumerateMessageSortDirectionOptions(MessageSorting ms)
Enumerates the available message sorting directions for the specified MessageSorting option.
Definition sortorder.cpp:80
MessageSorting
The available message sorting options.
Definition sortorder.h:60
@ NoMessageSorting
Don't sort the messages at all.
Definition sortorder.h:61
@ SortMessagesBySenderOrReceiver
Sort the messages by sender or receiver.
Definition sortorder.h:64
@ SortMessagesBySender
Sort the messages by sender.
Definition sortorder.h:65
@ SortMessagesByReceiver
Sort the messages by receiver.
Definition sortorder.h:66
static QList< QPair< QString, int > > enumerateMessageSortingOptions(Aggregation::Threading t)
Enumerates the message sorting options compatible with the specified Threading setting.
Definition sortorder.cpp:60
static SortOrder defaultForAggregation(const Aggregation *aggregation, SortOrder oldSortOrder)
Returns the default sort order for the given aggregation.
static QList< QPair< QString, int > > enumerateGroupSortDirectionOptions(Aggregation::Grouping g, GroupSorting groupSorting)
Enumerates the group sort direction options compatible with the specified Grouping and GroupSorting.
The QAbstractItemModel based interface that you need to provide for your storage to work with Message...
virtual QString id() const =0
Returns an unique id for this Storage collection.
The Theme class defines the visual appearance of the MessageList.
Definition theme.h:48
MessageItem * currentMessageItem(bool selectIfNeeded=true) const
Returns the current MessageItem (that is bound to current StorageModel).
Definition view.cpp:849
void setAggregation(const Aggregation *aggregation)
Sets the aggregation for this view.
Definition view.cpp:249
virtual void viewDragMoveEvent(QDragMoveEvent *e)
This is called by View when a drag move event is received.
bool isThreaded() const
Returns true if the current Aggregation is threaded, false otherwise (or if there is no current Aggre...
virtual void fillMessageTagCombo()
Called when the "Message Status/Tag" filter menu is opened by the user.
virtual void viewMessageStatusChangeRequest(MessageItem *msg, Akonadi::MessageStatus set, Akonadi::MessageStatus clear)
This is called by View when a message item is manipulated by the user in a way that it's status shoul...
View * view() const
Returns the View attached to this Widget.
bool selectionEmpty() const
Fast function that determines if the selection is empty.
void changeQuicksearchVisibility(bool)
Shows or hides the quicksearch field, the filter combobox and the toolbutton for advanced search.
StorageModel * storageModel() const
Returns the StorageModel currently set.
virtual void viewDropEvent(QDropEvent *e)
This is called by View when a drop event is received.
void slotViewHeaderSectionClicked(int logicalIndex)
Handles header section clicks switching the Aggregation MessageSorting on-the-fly.
virtual void viewMessageListContextPopupRequest(const QList< MessageItem * > &selectedItems, const QPoint &globalPos)
This is called by View when a message is right clicked.
void setCurrentFolder(const Akonadi::Collection &collection)
Sets the current folder.
virtual void viewSelectionChanged()
This is called by View when selection changes.
virtual void viewMessageActivated(MessageItem *msg)
This is called by View when a message is double-clicked or activated by other input means.
virtual void viewStartDragRequest()
This is called by View when a drag can possibly be started.
virtual void viewGroupHeaderContextPopupRequest(GroupHeaderItem *group, const QPoint &globalPos)
This is called by View when a group header is right clicked.
void setCurrentStatusFilterItem()
Must be called by fillMessageTagCombo()
void populateStatusFilterCombo()
This is called to setup the status filter's QComboBox.
QLineEdit * quickSearch() const
Returns the search line of this widget.
void aggregationsChanged()
This is called by Manager when the option sets stored within have changed.
void themesChanged()
This is called by Manager when the option sets stored within have changed.
Core::MessageItem * currentMessageItem() const
Returns the current MessageItem in the current folder.
void setStorageModel(StorageModel *storageModel, PreSelectionMode preSelectionMode=PreSelectLastSelected)
Sets the storage model for this Widget.
QList< Akonadi::MessageStatus > currentFilterStatus() const
Returns the Akonadi::MessageStatus in the current quicksearch field.
virtual void viewMessageSelected(MessageItem *msg)
This is called by View when a message is single-clicked (thus selected and made current)
virtual void viewDragEnterEvent(QDragEnterEvent *e)
This is called by View when a drag enter event is received.
void focusQuickSearch(const QString &selectedText)
Sets the focus on the quick search line of the currently active tab.
QString currentFilterSearchString() const
Returns the search term in the current quicksearch field.
The Akonadi specific implementation of the Core::StorageModel.
Q_SCRIPTABLE CaptureState status()
QString i18nc(const char *context, const char *text, const TYPE &arg...)
QString i18n(const char *text, const TYPE &arg...)
void information(QWidget *parent, const QString &text, const QString &title=QString(), const QString &dontShowAgainName=QString(), Options options=Notify)
The implementation independent part of the MessageList library.
Definition aggregation.h:22
PreSelectionMode
Pre-selection is the action of automatically selecting a message just after the folder has finished l...
void setCheckable(bool)
void setChecked(bool)
QVariant data() const const
void setData(const QVariant &data)
void triggered(bool checked)
QAction * addAction(QAction *action)
void clear()
void currentIndexChanged(int index)
void sectionClicked(int logicalIndex)
void clear()
const_reference at(qsizetype i) const const
iterator begin()
iterator end()
qsizetype size() const const
QAction * addAction(const QIcon &icon, const QString &text, Functor functor, const QKeySequence &shortcut)
QAction * addSection(const QIcon &icon, const QString &text)
QAction * addSeparator()
void clear()
QMetaObject::Connection connect(const QObject *sender, PointerToMemberFunction signal, Functor functor)
bool disconnect(const QMetaObject::Connection &connection)
T qobject_cast(QObject *object)
QObject * sender() const const
bool isEmpty() const const
OtherFocusReason
AscendingOrder
QFuture< void > filter(QThreadPool *pool, Sequence &sequence, KeepFunctor &&filterFunction)
QFuture< ArgsType< Signal > > connect(Sender *sender, Signal signal)
void timeout()
QString toString() const const
void setFocus()
void show()
QWidget * window() const const
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Mar 28 2025 11:49:15 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.