Plasma-workspace

tasksmodel.cpp
1/*
2 SPDX-FileCopyrightText: 2016 Eike Hein <hein@kde.org>
3
4 SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL
5*/
6
7#include "tasksmodel.h"
8#include "activityinfo.h"
9#include "concatenatetasksproxymodel.h"
10#include "flattentaskgroupsproxymodel.h"
11#include "taskfilterproxymodel.h"
12#include "taskgroupingproxymodel.h"
13#include "tasktools.h"
14#include "virtualdesktopinfo.h"
15
16#include "launchertasksmodel.h"
17#include "startuptasksmodel.h"
18#include "windowtasksmodel.h"
19
20#include "launchertasksmodel_p.h"
21
22#include <QDateTime>
23#include <QGuiApplication>
24#include <QList>
25#include <QTimer>
26#include <QUrl>
27
28#include <numeric>
29#include <optional>
30
31namespace TaskManager
32{
33class Q_DECL_HIDDEN TasksModel::Private
34{
35public:
36 Private(TasksModel *q);
37 ~Private();
38
39 static int instanceCount;
40
41 static WindowTasksModel *windowTasksModel;
42 static StartupTasksModel *startupTasksModel;
43 LauncherTasksModel *launcherTasksModel = nullptr;
44 ConcatenateTasksProxyModel *concatProxyModel = nullptr;
45 TaskFilterProxyModel *filterProxyModel = nullptr;
46 TaskGroupingProxyModel *groupingProxyModel = nullptr;
47 FlattenTaskGroupsProxyModel *flattenGroupsProxyModel = nullptr;
48 AbstractTasksModelIface *abstractTasksSourceModel = nullptr;
49
50 bool anyTaskDemandsAttention = false;
51
52 int launcherCount = 0;
53
54 SortMode sortMode = SortAlpha;
55 bool separateLaunchers = true;
56 bool launchInPlace = false;
57 bool hideActivatedLaunchers = true;
58 bool launchersEverSet = false;
59 bool launcherSortingDirty = false;
60 bool launcherCheckNeeded = false;
61 QList<int> sortedPreFilterRows;
62 QList<int> sortRowInsertQueue;
63 bool sortRowInsertQueueStale = false;
64 std::shared_ptr<VirtualDesktopInfo> virtualDesktopInfo;
65 QHash<QString, int> activityTaskCounts;
66 std::shared_ptr<ActivityInfo> activityInfo;
67
68 bool groupInline = false;
69 int groupingWindowTasksThreshold = -1;
70
71 bool usedByQml = false;
72 bool componentComplete = false;
73
74 void initModels();
75 void initLauncherTasksModel();
76 void updateAnyTaskDemandsAttention();
77 void updateManualSortMap();
78 void consolidateManualSortMapForGroup(const QModelIndex &groupingProxyIndex);
79 void updateGroupInline();
80 QModelIndex preFilterIndex(const QModelIndex &sourceIndex) const;
81 void updateActivityTaskCounts();
82 void forceResort();
83 bool lessThan(const QModelIndex &left, const QModelIndex &right, bool sortOnlyLaunchers = false) const;
84 std::optional<bool> lessThanByVirtualDesktop(const QModelIndex &left, const QModelIndex &right) const;
85
86private:
87 TasksModel *const q;
88};
89
90class TasksModel::TasksModelLessThan
91{
92public:
93 inline TasksModelLessThan(const QAbstractItemModel *s, TasksModel *p, bool sortOnlyLaunchers)
94 : sourceModel(s)
95 , tasksModel(p)
96 , sortOnlyLaunchers(sortOnlyLaunchers)
97 {
98 }
99
100 inline bool operator()(int r1, int r2) const
101 {
102 QModelIndex i1 = sourceModel->index(r1, 0);
103 QModelIndex i2 = sourceModel->index(r2, 0);
104 return tasksModel->d->lessThan(i1, i2, sortOnlyLaunchers);
105 }
106
107private:
108 const QAbstractItemModel *sourceModel;
109 const TasksModel *tasksModel;
110 bool sortOnlyLaunchers;
111};
112
113int TasksModel::Private::instanceCount = 0;
114WindowTasksModel *TasksModel::Private::windowTasksModel = nullptr;
115StartupTasksModel *TasksModel::Private::startupTasksModel = nullptr;
116
117TasksModel::Private::Private(TasksModel *q)
118 : q(q)
119{
120 ++instanceCount;
121}
122
123TasksModel::Private::~Private()
124{
125 --instanceCount;
126
127 if (!instanceCount) {
128 delete windowTasksModel;
129 windowTasksModel = nullptr;
130 delete startupTasksModel;
131 startupTasksModel = nullptr;
132 }
133}
134
135void TasksModel::Private::initModels()
136{
137 // NOTE: Overview over the entire model chain assembled here:
138 // WindowTasksModel, StartupTasksModel, LauncherTasksModel
139 // -> concatProxyModel concatenates them into a single list.
140 // -> filterProxyModel filters by state (e.g. virtual desktop).
141 // -> groupingProxyModel groups by application (we go from flat list to tree).
142 // -> flattenGroupsProxyModel (optionally, if groupInline == true) flattens groups out.
143 // -> TasksModel collapses (top-level) items into task lifecycle abstraction; sorts.
144
145 concatProxyModel = new ConcatenateTasksProxyModel(q);
146
147 if (!windowTasksModel) {
148 windowTasksModel = new WindowTasksModel();
149 }
150
151 concatProxyModel->addSourceModel(windowTasksModel);
152
153 QObject::connect(windowTasksModel, &QAbstractItemModel::rowsInserted, q, [this]() {
154 if (sortMode == SortActivity) {
155 updateActivityTaskCounts();
156 }
157 });
158
159 QObject::connect(windowTasksModel, &QAbstractItemModel::rowsRemoved, q, [this]() {
160 if (sortMode == SortActivity) {
161 updateActivityTaskCounts();
162 forceResort();
163 }
164 // the active task may have potentially changed, so signal that so that users
165 // will recompute it
166 Q_EMIT q->activeTaskChanged();
167 });
168
169 QObject::connect(windowTasksModel,
171 q,
172 [this](const QModelIndex &topLeft, const QModelIndex &bottomRight, const QList<int> &roles) {
173 Q_UNUSED(topLeft)
174 Q_UNUSED(bottomRight)
175
176 if (sortMode == SortActivity && roles.contains(AbstractTasksModel::Activities)) {
177 updateActivityTaskCounts();
178 }
179
180 if (roles.contains(AbstractTasksModel::IsActive)) {
181 Q_EMIT q->activeTaskChanged();
182 }
183
184 // In manual sort mode, updateManualSortMap() may consult the sortRowInsertQueue
185 // for new tasks to sort in. Hidden tasks remain in the queue to potentially sort
186 // them later, when they are are actually revealed to the user.
187 // This is particularly useful in concert with taskmanagerrulesrc's SkipTaskbar
188 // key, which is used to hide window tasks which update from bogus to useful
189 // window metadata early in startup. The role change then coincides with positive
190 // app identification, which is when updateManualSortMap() becomes able to sort the
191 // task adjacent to its launcher when required to do so.
192 if (sortMode == SortManual && roles.contains(AbstractTasksModel::SkipTaskbar)) {
193 updateManualSortMap();
194 }
195 });
196
197 if (!startupTasksModel) {
198 startupTasksModel = new StartupTasksModel();
199 }
200
201 concatProxyModel->addSourceModel(startupTasksModel);
202
203 // If we're in manual sort mode, we need to seed the sort map on pending row
204 // insertions.
205 QObject::connect(concatProxyModel, &QAbstractItemModel::rowsAboutToBeInserted, q, [this](const QModelIndex &parent, int start, int end) {
206 Q_UNUSED(parent)
207
208 if (sortMode != SortManual) {
209 return;
210 }
211
212 const int delta = (end - start) + 1;
213 for (auto it = sortedPreFilterRows.begin(); it != sortedPreFilterRows.end(); ++it) {
214 if ((*it) >= start) {
215 *it += delta;
216 }
217 }
218
219 for (int i = start; i <= end; ++i) {
220 sortedPreFilterRows.append(i);
221
222 if (!separateLaunchers) {
223 if (sortRowInsertQueueStale) {
224 sortRowInsertQueue.clear();
225 sortRowInsertQueueStale = false;
226 }
227
228 sortRowInsertQueue.append(sortedPreFilterRows.count() - 1);
229 }
230 }
231 });
232
233 // If we're in manual sort mode, we need to update the sort map on row insertions.
234 QObject::connect(concatProxyModel, &QAbstractItemModel::rowsInserted, q, [this](const QModelIndex &parent, int start, int end) {
235 Q_UNUSED(parent)
236 Q_UNUSED(start)
237 Q_UNUSED(end)
238
239 if (sortMode == SortManual) {
240 updateManualSortMap();
241 }
242 });
243
244 // If we're in manual sort mode, we need to update the sort map after row removals.
245 QObject::connect(concatProxyModel, &QAbstractItemModel::rowsRemoved, q, [this](const QModelIndex &parent, int first, int last) {
246 Q_UNUSED(parent)
247
248 if (sortMode != SortManual) {
249 return;
250 }
251
252 if (sortRowInsertQueueStale) {
253 sortRowInsertQueue.clear();
254 sortRowInsertQueueStale = false;
255 }
256
257 for (int i = first; i <= last; ++i) {
258 sortedPreFilterRows.removeOne(i);
259 }
260
261 const int delta = (last - first) + 1;
262 for (auto it = sortedPreFilterRows.begin(); it != sortedPreFilterRows.end(); ++it) {
263 if ((*it) > last) {
264 *it -= delta;
265 }
266 }
267 });
268
269 filterProxyModel = new TaskFilterProxyModel(q);
270 filterProxyModel->setSourceModel(concatProxyModel);
271 QObject::connect(filterProxyModel, &TaskFilterProxyModel::virtualDesktopChanged, q, &TasksModel::virtualDesktopChanged);
272 QObject::connect(filterProxyModel, &TaskFilterProxyModel::screenGeometryChanged, q, &TasksModel::screenGeometryChanged);
273 QObject::connect(filterProxyModel, &TaskFilterProxyModel::regionGeometryChanged, q, &TasksModel::regionGeometryChanged);
274 QObject::connect(filterProxyModel, &TaskFilterProxyModel::activityChanged, q, &TasksModel::activityChanged);
275 QObject::connect(filterProxyModel, &TaskFilterProxyModel::filterByVirtualDesktopChanged, q, &TasksModel::filterByVirtualDesktopChanged);
276 QObject::connect(filterProxyModel, &TaskFilterProxyModel::filterByScreenChanged, q, &TasksModel::filterByScreenChanged);
277 QObject::connect(filterProxyModel, &TaskFilterProxyModel::filterByActivityChanged, q, &TasksModel::filterByActivityChanged);
278 QObject::connect(filterProxyModel, &TaskFilterProxyModel::filterByRegionChanged, q, &TasksModel::filterByRegionChanged);
279 QObject::connect(filterProxyModel, &TaskFilterProxyModel::filterMinimizedChanged, q, &TasksModel::filterMinimizedChanged);
280 QObject::connect(filterProxyModel, &TaskFilterProxyModel::filterNotMinimizedChanged, q, &TasksModel::filterNotMinimizedChanged);
281 QObject::connect(filterProxyModel, &TaskFilterProxyModel::filterNotMaximizedChanged, q, &TasksModel::filterNotMaximizedChanged);
282 QObject::connect(filterProxyModel, &TaskFilterProxyModel::filterHiddenChanged, q, &TasksModel::filterHiddenChanged);
283
284 groupingProxyModel = new TaskGroupingProxyModel(q);
285 groupingProxyModel->setSourceModel(filterProxyModel);
286 QObject::connect(groupingProxyModel, &TaskGroupingProxyModel::groupModeChanged, q, &TasksModel::groupModeChanged);
287 QObject::connect(groupingProxyModel, &TaskGroupingProxyModel::blacklistedAppIdsChanged, q, &TasksModel::groupingAppIdBlacklistChanged);
288 QObject::connect(groupingProxyModel, &TaskGroupingProxyModel::blacklistedLauncherUrlsChanged, q, &TasksModel::groupingLauncherUrlBlacklistChanged);
289
290 QObject::connect(groupingProxyModel, &QAbstractItemModel::rowsInserted, q, [this](const QModelIndex &parent, int first, int last) {
291 if (parent.isValid()) {
292 if (sortMode == SortManual) {
293 consolidateManualSortMapForGroup(parent);
294 }
295
296 // Existence of a group means everything below this has already been done.
297 return;
298 }
299
300 bool demandsAttentionUpdateNeeded = false;
301
302 for (int i = first; i <= last; ++i) {
303 const QModelIndex &sourceIndex = groupingProxyModel->index(i, 0);
304 const QString &appId = sourceIndex.data(AbstractTasksModel::AppId).toString();
305
306 if (sourceIndex.data(AbstractTasksModel::IsDemandingAttention).toBool()) {
307 demandsAttentionUpdateNeeded = true;
308 }
309
310 // When we get a window we have a startup for, cause the startup to be re-filtered.
311 if (sourceIndex.data(AbstractTasksModel::IsWindow).toBool()) {
312 const QString &appName = sourceIndex.data(AbstractTasksModel::AppName).toString();
313
314 for (int j = 0; j < filterProxyModel->rowCount(); ++j) {
315 QModelIndex filterIndex = filterProxyModel->index(j, 0);
316
317 if (!filterIndex.data(AbstractTasksModel::IsStartup).toBool()) {
318 continue;
319 }
320
321 if ((!appId.isEmpty() && appId == filterIndex.data(AbstractTasksModel::AppId).toString())
322 || (!appName.isEmpty() && appName == filterIndex.data(AbstractTasksModel::AppName).toString())) {
323 Q_EMIT filterProxyModel->dataChanged(filterIndex, filterIndex);
324 }
325 }
326 }
327
328 // When we get a window or startup we have a launcher for, cause the launcher to be re-filtered.
329 if (sourceIndex.data(AbstractTasksModel::IsWindow).toBool() || sourceIndex.data(AbstractTasksModel::IsStartup).toBool()) {
330 for (int j = 0; j < filterProxyModel->rowCount(); ++j) {
331 const QModelIndex &filterIndex = filterProxyModel->index(j, 0);
332
333 if (!filterIndex.data(AbstractTasksModel::IsLauncher).toBool()) {
334 continue;
335 }
336
337 if (appsMatch(sourceIndex, filterIndex)) {
338 Q_EMIT filterProxyModel->dataChanged(filterIndex, filterIndex);
339 }
340 }
341 }
342 }
343
344 if (!anyTaskDemandsAttention && demandsAttentionUpdateNeeded) {
345 updateAnyTaskDemandsAttention();
346 }
347 });
348
349 QObject::connect(groupingProxyModel, &QAbstractItemModel::rowsAboutToBeRemoved, q, [this](const QModelIndex &parent, int first, int last) {
350 // We can ignore group members.
351 if (parent.isValid()) {
352 return;
353 }
354
355 for (int i = first; i <= last; ++i) {
356 const QModelIndex &sourceIndex = groupingProxyModel->index(i, 0);
357
358 // When a window or startup task is removed, we have to trigger a re-filter of
359 // our launchers to (possibly) pop them back in.
360 // NOTE: An older revision of this code compared the window and startup tasks
361 // to the launchers to figure out which launchers should be re-filtered. This
362 // was fine until we discovered that certain applications (e.g. Google Chrome)
363 // change their window metadata specifically during tear-down, sometimes
364 // breaking TaskTools::appsMatch (it's a race) and causing the associated
365 // launcher to remain hidden. Therefore we now consider any top-level window or
366 // startup task removal a trigger to re-filter all launchers. We don't do this
367 // in response to the window metadata changes (even though it would be strictly
368 // more correct, as then-ending identity match-up was what caused the launcher
369 // to be hidden) because we don't want the launcher and window/startup task to
370 // briefly co-exist in the model.
371 if (!launcherCheckNeeded && launcherTasksModel
372 && (sourceIndex.data(AbstractTasksModel::IsWindow).toBool() || sourceIndex.data(AbstractTasksModel::IsStartup).toBool())) {
373 launcherCheckNeeded = true;
374 }
375 }
376 });
377
378 QObject::connect(filterProxyModel, &QAbstractItemModel::rowsRemoved, q, [this](const QModelIndex &parent, int first, int last) {
379 Q_UNUSED(parent)
380 Q_UNUSED(first)
381 Q_UNUSED(last)
382
383 if (launcherCheckNeeded) {
384 for (int i = 0; i < filterProxyModel->rowCount(); ++i) {
385 const QModelIndex &idx = filterProxyModel->index(i, 0);
386
387 if (idx.data(AbstractTasksModel::IsLauncher).toBool()) {
388 Q_EMIT filterProxyModel->dataChanged(idx, idx);
389 }
390 }
391
392 launcherCheckNeeded = false;
393 }
394
395 // One of the removed tasks might have been demanding attention, but
396 // we can't check the state after the window has been closed already,
397 // so we always have to do a full update.
398 if (anyTaskDemandsAttention) {
399 updateAnyTaskDemandsAttention();
400 }
401 });
402
403 // Update anyTaskDemandsAttention on source data changes.
404 QObject::connect(groupingProxyModel,
406 q,
407 [this](const QModelIndex &topLeft, const QModelIndex &bottomRight, const QList<int> &roles) {
408 Q_UNUSED(bottomRight)
409
410 // We can ignore group members.
411 if (topLeft.parent().isValid()) {
412 return;
413 }
414
415 if (roles.isEmpty() || roles.contains(AbstractTasksModel::IsDemandingAttention)) {
416 updateAnyTaskDemandsAttention();
417 }
418
419 if (roles.isEmpty() || roles.contains(AbstractTasksModel::AppId)) {
420 for (int i = topLeft.row(); i <= bottomRight.row(); ++i) {
421 const QModelIndex &sourceIndex = groupingProxyModel->index(i, 0);
422
423 // When a window task changes identity to one we have a launcher for, cause
424 // the launcher to be re-filtered.
425 if (sourceIndex.data(AbstractTasksModel::IsWindow).toBool()) {
426 for (int i = 0; i < filterProxyModel->rowCount(); ++i) {
427 const QModelIndex &filterIndex = filterProxyModel->index(i, 0);
428
429 if (!filterIndex.data(AbstractTasksModel::IsLauncher).toBool()) {
430 continue;
431 }
432
433 if (appsMatch(sourceIndex, filterIndex)) {
434 Q_EMIT filterProxyModel->dataChanged(filterIndex, filterIndex);
435 }
436 }
437 }
438 }
439 }
440 });
441
442 // Update anyTaskDemandsAttention on source model resets.
443 QObject::connect(groupingProxyModel, &QAbstractItemModel::modelReset, q, [this]() {
444 updateAnyTaskDemandsAttention();
445 });
446}
447
448void TasksModel::Private::updateAnyTaskDemandsAttention()
449{
450 bool taskFound = false;
451
452 for (int i = 0; i < groupingProxyModel->rowCount(); ++i) {
453 if (groupingProxyModel->index(i, 0).data(AbstractTasksModel::IsDemandingAttention).toBool()) {
454 taskFound = true;
455 break;
456 }
457 }
458
459 if (taskFound != anyTaskDemandsAttention) {
460 anyTaskDemandsAttention = taskFound;
461 Q_EMIT q->anyTaskDemandsAttentionChanged();
462 }
463}
464
465void TasksModel::Private::initLauncherTasksModel()
466{
467 if (launcherTasksModel) {
468 return;
469 }
470
471 launcherTasksModel = new LauncherTasksModel(q);
472 QObject::connect(launcherTasksModel, &LauncherTasksModel::launcherListChanged, q, &TasksModel::launcherListChanged);
473 QObject::connect(launcherTasksModel, &LauncherTasksModel::launcherListChanged, q, &TasksModel::updateLauncherCount);
474
475 // TODO: On the assumptions that adding/removing launchers is a rare event and
476 // the HasLaunchers data role is rarely used, this refreshes it for all rows in
477 // the model. If those assumptions are proven wrong later, this could be
478 // optimized to only refresh non-launcher rows matching the inserted or about-
479 // to-be-removed launcherTasksModel rows using TaskTools::appsMatch().
480 QObject::connect(launcherTasksModel, &LauncherTasksModel::launcherListChanged, q, [this]() {
481 Q_EMIT q->dataChanged(q->index(0, 0), q->index(q->rowCount() - 1, 0), QList<int>{AbstractTasksModel::HasLauncher});
482 });
483
484 // data() implements AbstractTasksModel::HasLauncher by checking with
485 // TaskTools::appsMatch, which evaluates ::AppId and ::LauncherUrlWithoutIcon.
486 QObject::connect(q, &QAbstractItemModel::dataChanged, q, [this](const QModelIndex &topLeft, const QModelIndex &bottomRight, const QList<int> &roles) {
487 if (roles.contains(AbstractTasksModel::AppId) || roles.contains(AbstractTasksModel::LauncherUrlWithoutIcon)) {
488 for (int i = topLeft.row(); i <= bottomRight.row(); ++i) {
489 const QModelIndex &index = q->index(i, 0);
490
491 if (!index.data(AbstractTasksModel::IsLauncher).toBool()) {
492 Q_EMIT q->dataChanged(index, index, QList<int>{AbstractTasksModel::HasLauncher});
493 }
494 }
495 }
496 });
497
498 concatProxyModel->addSourceModel(launcherTasksModel);
499}
500
501void TasksModel::Private::updateManualSortMap()
502{
503 // Empty map; full sort.
504 if (sortedPreFilterRows.isEmpty()) {
505 sortedPreFilterRows.reserve(concatProxyModel->rowCount());
506
507 for (int i = 0; i < concatProxyModel->rowCount(); ++i) {
508 sortedPreFilterRows.append(i);
509 }
510
511 // Full sort.
512 TasksModelLessThan lt(concatProxyModel, q, false);
513 std::stable_sort(sortedPreFilterRows.begin(), sortedPreFilterRows.end(), lt);
514
515 // Consolidate sort map entries for groups.
516 if (q->groupMode() != GroupDisabled) {
517 for (int i = 0; i < groupingProxyModel->rowCount(); ++i) {
518 const QModelIndex &groupingIndex = groupingProxyModel->index(i, 0);
519
520 if (groupingIndex.data(AbstractTasksModel::IsGroupParent).toBool()) {
521 consolidateManualSortMapForGroup(groupingIndex);
522 }
523 }
524 }
525
526 return;
527 }
528
529 // Existing map; check whether launchers need sorting by launcher list position.
530 if (separateLaunchers) {
531 // Sort only launchers.
532 TasksModelLessThan lt(concatProxyModel, q, true);
533 std::stable_sort(sortedPreFilterRows.begin(), sortedPreFilterRows.end(), lt);
534 // Otherwise process any entries in the insert queue and move them intelligently
535 // in the sort map.
536 } else {
537 QMutableListIterator<int> i(sortRowInsertQueue);
538
539 while (i.hasNext()) {
540 i.next();
541
542 const int row = i.value();
543 const QModelIndex &idx = concatProxyModel->index(sortedPreFilterRows.at(row), 0);
544
545 // If a window task is currently hidden, we may want to keep it in the queue
546 // to sort it in later once it gets revealed.
547 // This is important in concert with taskmanagerrulesrc's SkipTaskbar key, which
548 // is used to hide window tasks which update from bogus to useful window metadata
549 // early in startup. Once the task no longer uses bogus metadata listed in the
550 // config key, its SkipTaskbar role changes to false, and then is it possible to
551 // sort the task adjacent to its launcher in the code below.
552 if (idx.data(AbstractTasksModel::IsWindow).toBool() && idx.data(AbstractTasksModel::SkipTaskbar).toBool()) {
553 // Since we're going to keep a row in the queue for now, make sure to
554 // mark the queue as stale so it's cleared on appends or row removals
555 // when they follow this sorting attempt. This frees us from having to
556 // update the indices in the queue to keep them valid.
557 // This means windowing system changes such as the opening or closing
558 // of a window task which happen during the time period that a window
559 // task has known bogus metadata, can upset what we're trying to
560 // achieve with this exception. However, due to the briefness of the
561 // time period and usage patterns, this is improbable, making this
562 // likely good enough. If it turns out not to be, this decision may be
563 // revisited later.
564 sortRowInsertQueueStale = true;
565
566 break;
567 } else {
568 i.remove();
569 }
570
571 bool moved = false;
572
573 // Try to move the task up to its right-most app sibling, unless this
574 // is us sorting in a launcher list for the first time.
575 if (launchersEverSet && !idx.data(AbstractTasksModel::IsLauncher).toBool()) {
576 for (int j = (row - 1); j >= 0; --j) {
577 const QModelIndex &concatProxyIndex = concatProxyModel->index(sortedPreFilterRows.at(j), 0);
578
579 // Once we got a match, check if the filter model accepts the potential
580 // sibling. We don't want to sort new tasks in next to tasks it will
581 // filter out once it sees it anyway.
582 if (appsMatch(concatProxyIndex, idx) && filterProxyModel->acceptsRow(concatProxyIndex.row())) {
583 sortedPreFilterRows.move(row, j + 1);
584 moved = true;
585
586 break;
587 }
588 }
589 }
590
591 int insertPos = 0;
592
593 // If unsuccessful or skipped, and the new task is a launcher, put after
594 // the rightmost launcher or launcher-backed task in the map, or failing
595 // that at the start of the map.
596 if (!moved && idx.data(AbstractTasksModel::IsLauncher).toBool()) {
597 for (int j = 0; j < row; ++j) {
598 const QModelIndex &concatProxyIndex = concatProxyModel->index(sortedPreFilterRows.at(j), 0);
599
600 if (concatProxyIndex.data(AbstractTasksModel::IsLauncher).toBool()
601 || launcherTasksModel->launcherPosition(concatProxyIndex.data(AbstractTasksModel::LauncherUrlWithoutIcon).toUrl()) != -1) {
602 insertPos = j + 1;
603 } else {
604 break;
605 }
606 }
607
608 sortedPreFilterRows.move(row, insertPos);
609 moved = true;
610 }
611
612 // If we sorted in a launcher and it's the first time we're sorting in a
613 // launcher list, move existing windows to the launcher position now.
614 if (moved && !launchersEverSet) {
615 for (int j = (sortedPreFilterRows.count() - 1); j >= 0; --j) {
616 const QModelIndex &concatProxyIndex = concatProxyModel->index(sortedPreFilterRows.at(j), 0);
617
618 if (!concatProxyIndex.data(AbstractTasksModel::IsLauncher).toBool()
619 && idx.data(AbstractTasksModel::LauncherUrlWithoutIcon) == concatProxyIndex.data(AbstractTasksModel::LauncherUrlWithoutIcon)) {
620 sortedPreFilterRows.move(j, insertPos);
621
622 if (insertPos > j) {
623 --insertPos;
624 }
625 }
626 }
627 }
628 }
629 }
630}
631
632void TasksModel::Private::consolidateManualSortMapForGroup(const QModelIndex &groupingProxyIndex)
633{
634 // Consolidates sort map entries for a group's items to be contiguous
635 // after the group's first item and the same order as in groupingProxyModel.
636
637 const int childCount = groupingProxyModel->rowCount(groupingProxyIndex);
638
639 if (!childCount) {
640 return;
641 }
642
643 const QModelIndex &leader = groupingProxyModel->index(0, 0, groupingProxyIndex);
644 const QModelIndex &preFilterLeader = filterProxyModel->mapToSource(groupingProxyModel->mapToSource(leader));
645
646 // We're moving the trailing children to the sort map position of
647 // the first child, so we're skipping the first child.
648 for (int i = 1; i < childCount; ++i) {
649 const QModelIndex &child = groupingProxyModel->index(i, 0, groupingProxyIndex);
650 const QModelIndex &preFilterChild = filterProxyModel->mapToSource(groupingProxyModel->mapToSource(child));
651 const int leaderPos = sortedPreFilterRows.indexOf(preFilterLeader.row());
652 const int childPos = sortedPreFilterRows.indexOf(preFilterChild.row());
653 const int insertPos = (leaderPos + i) + ((leaderPos + i) > childPos ? -1 : 0);
654 sortedPreFilterRows.move(childPos, insertPos);
655 }
656}
657
658void TasksModel::Private::updateGroupInline()
659{
660 if (usedByQml && !componentComplete) {
661 return;
662 }
663
664 bool hadSourceModel = (q->sourceModel() != nullptr);
665
666 if (q->groupMode() != GroupDisabled && groupInline) {
667 if (flattenGroupsProxyModel) {
668 return;
669 }
670
671 // Exempting tasks which demand attention from grouping is not
672 // necessary when all group children are shown inline anyway
673 // and would interfere with our sort-tasks-together goals.
674 groupingProxyModel->setGroupDemandingAttention(true);
675
676 // Likewise, ignore the window tasks threshold when making
677 // grouping decisions.
678 groupingProxyModel->setWindowTasksThreshold(-1);
679
680 flattenGroupsProxyModel = new FlattenTaskGroupsProxyModel(q);
681 flattenGroupsProxyModel->setSourceModel(groupingProxyModel);
682
683 abstractTasksSourceModel = flattenGroupsProxyModel;
684 q->setSourceModel(flattenGroupsProxyModel);
685
686 if (sortMode == SortManual) {
687 forceResort();
688 }
689 } else {
690 if (hadSourceModel && !flattenGroupsProxyModel) {
691 return;
692 }
693
694 groupingProxyModel->setGroupDemandingAttention(false);
695 groupingProxyModel->setWindowTasksThreshold(groupingWindowTasksThreshold);
696
697 abstractTasksSourceModel = groupingProxyModel;
698 q->setSourceModel(groupingProxyModel);
699
700 delete flattenGroupsProxyModel;
701 flattenGroupsProxyModel = nullptr;
702
703 if (hadSourceModel && sortMode == SortManual) {
704 forceResort();
705 }
706 }
707
708 // Minor optimization: We only make these connections after we populate for
709 // the first time to avoid some churn.
710 if (!hadSourceModel) {
711 QObject::connect(q, &QAbstractItemModel::rowsInserted, q, &TasksModel::updateLauncherCount, Qt::UniqueConnection);
712 QObject::connect(q, &QAbstractItemModel::rowsRemoved, q, &TasksModel::updateLauncherCount, Qt::UniqueConnection);
713 QObject::connect(q, &QAbstractItemModel::modelReset, q, &TasksModel::updateLauncherCount, Qt::UniqueConnection);
714
717 QObject::connect(q, &QAbstractItemModel::modelReset, q, &TasksModel::countChanged, Qt::UniqueConnection);
718 }
719}
720
721QModelIndex TasksModel::Private::preFilterIndex(const QModelIndex &sourceIndex) const
722{
723 // Only in inline grouping mode, we have an additional proxy layer.
724 if (flattenGroupsProxyModel) {
725 return filterProxyModel->mapToSource(groupingProxyModel->mapToSource(flattenGroupsProxyModel->mapToSource(sourceIndex)));
726 } else {
727 return filterProxyModel->mapToSource(groupingProxyModel->mapToSource(sourceIndex));
728 }
729}
730
731void TasksModel::Private::updateActivityTaskCounts()
732{
733 // Collects the number of window tasks on each activity.
734
735 activityTaskCounts.clear();
736
737 if (!windowTasksModel || !activityInfo) {
738 return;
739 }
740
741 for (const auto activities = activityInfo->runningActivities(); const QString &activity : activities) {
742 activityTaskCounts.insert(activity, 0);
743 }
744
745 for (int i = 0; i < windowTasksModel->rowCount(); ++i) {
746 const QModelIndex &windowIndex = windowTasksModel->index(i, 0);
747 const QStringList &activities = windowIndex.data(AbstractTasksModel::Activities).toStringList();
748
749 if (activities.isEmpty()) {
750 QMutableHashIterator<QString, int> it(activityTaskCounts);
751
752 while (it.hasNext()) {
753 it.next();
754 it.setValue(it.value() + 1);
755 }
756 } else {
757 for (const QString &activity : activities) {
758 ++activityTaskCounts[activity];
759 }
760 }
761 }
762}
763
764void TasksModel::Private::forceResort()
765{
766 // HACK: This causes QSortFilterProxyModel to run all rows through
767 // our lessThan() implementation again.
768 q->setDynamicSortFilter(false);
769 q->setDynamicSortFilter(true);
770}
771
772std::optional<bool> TasksModel::Private::lessThanByVirtualDesktop(const QModelIndex &left, const QModelIndex &right) const
773{
774 const bool leftAll = left.data(AbstractTasksModel::IsOnAllVirtualDesktops).toBool();
775 const bool rightAll = right.data(AbstractTasksModel::IsOnAllVirtualDesktops).toBool();
776
777 if (leftAll && !rightAll) {
778 return true;
779 }
780
781 if (!leftAll && rightAll) {
782 return false;
783 }
784
785 if (!leftAll && !rightAll) {
786 const auto getDesktop = [this](const QModelIndex &model) {
787 const QVariantList modelDesktops = model.data(AbstractTasksModel::VirtualDesktops).toList();
788 QVariant modelDesktop;
789 int modelDesktopPos = virtualDesktopInfo->numberOfDesktops();
790 for (const QVariant &desktop : modelDesktops) {
791 const int desktopPos = virtualDesktopInfo->position(desktop);
792
793 if (desktopPos <= modelDesktopPos) {
794 modelDesktop = desktop;
795 modelDesktopPos = desktopPos;
796 }
797 }
798 return modelDesktop;
799 };
800
801 const QVariant leftDesktop = getDesktop(left);
802 const QVariant rightDesktop = getDesktop(right);
803
804 if (!leftDesktop.isNull() && !rightDesktop.isNull() && (leftDesktop != rightDesktop)) {
805 return (virtualDesktopInfo->position(leftDesktop) < virtualDesktopInfo->position(rightDesktop));
806 } else if (!leftDesktop.isNull() && rightDesktop.isNull()) {
807 return false;
808 } else if (leftDesktop.isNull() && !rightDesktop.isNull()) {
809 return true;
810 }
811 }
812
813 // We couldn't determine the order (e.g., both on all desktops or the same desktop)
814 return std::nullopt;
815}
816
817bool TasksModel::Private::lessThan(const QModelIndex &left, const QModelIndex &right, bool sortOnlyLaunchers) const
818{
819 // Launcher tasks go first.
820 // When launchInPlace is enabled, startup and window tasks are sorted
821 // as the launchers they replace (see also move()).
822
823 if (separateLaunchers) {
824 const bool leftIsLauncher = left.data(AbstractTasksModel::IsLauncher).toBool();
825 const bool rightIsLauncher = right.data(AbstractTasksModel::IsLauncher).toBool();
826
827 if (leftIsLauncher && rightIsLauncher) {
828 return (left.row() < right.row());
829 }
830
831 const int leftPos = q->launcherPosition(left.data(AbstractTasksModel::LauncherUrlWithoutIcon).toUrl());
832 const int rightPos = q->launcherPosition(right.data(AbstractTasksModel::LauncherUrlWithoutIcon).toUrl());
833
834 if (leftIsLauncher && !rightIsLauncher) {
835 if (launchInPlace) {
836 if (rightPos != -1) {
837 return (leftPos < rightPos);
838 }
839 }
840
841 return true;
842 }
843
844 if (!leftIsLauncher && rightIsLauncher) {
845 if (launchInPlace) {
846 if (leftPos != -1) {
847 return (leftPos < rightPos);
848 }
849 }
850
851 return false;
852 }
853
854 // neither is a launcher
855 if (launchInPlace) {
856 if (leftPos != -1 && rightPos != -1) {
857 return (leftPos < rightPos);
858 } else if (leftPos != -1 && rightPos == -1) {
859 return true;
860 } else if (leftPos == -1 && rightPos != -1) {
861 return false;
862 }
863 }
864 }
865
866 // If told to stop after launchers we fall through to the existing map if it exists.
867 if (sortOnlyLaunchers && !sortedPreFilterRows.isEmpty()) {
868 return (sortedPreFilterRows.indexOf(left.row()) < sortedPreFilterRows.indexOf(right.row()));
869 }
870
871 // Sort other cases by sort mode.
872 switch (sortMode) {
873 case SortDisabled: {
874 return (left.row() < right.row());
875 }
876
877 case SortWindowPositionHorizontal: {
878
879 if (auto result = lessThanByVirtualDesktop(left, right)) {
880 return *result;
881 }
882
883 const QRect leftGeom = left.data(AbstractTasksModel::Geometry).value<QRect>();
884 const QRect rightGeom = right.data(AbstractTasksModel::Geometry).value<QRect>();
885
886 if (leftGeom.x() != rightGeom.x()) {
888 return leftGeom.right() > rightGeom.right();
889 } else {
890 return leftGeom.x() < rightGeom.x();
891 }
892 }
893 if (leftGeom.y() != rightGeom.y()) {
894 return leftGeom.y() < rightGeom.y();
895 }
896
897 Q_FALLTHROUGH();
898 }
899
900 case SortLastActivated: {
901 const auto getSortDateTime = [](const QModelIndex &model) {
902 // Check if the task is in a group
903 const QModelIndex topMost = model.parent().isValid() ? model.parent() : model;
904 QDateTime sortDateTime = topMost.data(AbstractTasksModel::LastActivated).toDateTime();
905 if (!sortDateTime.isValid()) {
906 sortDateTime = model.data(Qt::DisplayRole).toDateTime();
907 }
908 return sortDateTime;
909 };
910
911 const QDateTime leftSortDateTime = getSortDateTime(left);
912 const QDateTime rightSortDateTime = getSortDateTime(right);
913
914 if (leftSortDateTime != rightSortDateTime) {
915 // Move latest to leftmost
916 return leftSortDateTime > rightSortDateTime;
917 }
918
919 Q_FALLTHROUGH();
920 }
921 // fall through
922 case SortVirtualDesktop: {
923
924 if (auto result = lessThanByVirtualDesktop(left, right)) {
925 return *result;
926 }
927
928 Q_FALLTHROUGH();
929 }
930 // fall through
931 case SortActivity: {
932 // updateActivityTaskCounts() counts the number of window tasks on each
933 // activity. This will sort tasks by comparing a cumulative score made
934 // up of the task counts for each activity a task is assigned to, and
935 // otherwise fall through to alphabetical sorting.
936 const auto getScore = [this](const QModelIndex &model) {
937 const QStringList activities = model.data(AbstractTasksModel::Activities).toStringList();
938 const int score = std::accumulate(activities.cbegin(), activities.cend(), -1, [this](int a, const QString &activity) {
939 return a + activityTaskCounts[activity];
940 });
941 return score;
942 };
943
944 int leftScore = getScore(left);
945 int rightScore = getScore(right);
946
947 if (leftScore == -1 || rightScore == -1) {
948 const int sumScore = std::accumulate(activityTaskCounts.constBegin(), activityTaskCounts.constEnd(), 0);
949
950 if (leftScore == -1) {
951 leftScore = sumScore;
952 }
953
954 if (rightScore == -1) {
955 rightScore = sumScore;
956 }
957 }
958
959 if (leftScore != rightScore) {
960 return (leftScore > rightScore);
961 }
962 }
963 // fall through
964 case SortAlpha:
965 // fall through
966 default: {
967 // The overall goal of alphabetic sorting is to sort tasks belonging to the
968 // same app together, while sorting the resulting sets alphabetically among
969 // themselves by the app name. The following code tries to achieve this by
970 // going for AppName first, and falling back to DisplayRole - which for
971 // window-type tasks generally contains the window title - if AppName is
972 // not available. When comparing tasks with identical resulting sort strings,
973 // we sort them by the source model order (i.e. insertion/creation). Older
974 // versions of this code compared tasks by a concatenation of AppName and
975 // DisplayRole at all times, but always sorting by the window title does more
976 // than our goal description - and can cause tasks within an app's set to move
977 // around when window titles change, which is a nuisance for users (especially
978 // in case of tabbed apps that have the window title reflect the active tab,
979 // e.g. web browsers). To recap, the common case is "sort by AppName, then
980 // insertion order", only swapping out AppName for DisplayRole (i.e. window
981 // title) when necessary.
982
983 QString leftSortString = left.data(AbstractTasksModel::AppName).toString();
984
985 if (leftSortString.isEmpty()) {
986 leftSortString = left.data(Qt::DisplayRole).toString();
987 }
988
989 QString rightSortString = right.data(AbstractTasksModel::AppName).toString();
990
991 if (rightSortString.isEmpty()) {
992 rightSortString = right.data(Qt::DisplayRole).toString();
993 }
994
995 const int sortResult = leftSortString.localeAwareCompare(rightSortString);
996
997 // If the string are identical fall back to source model (creation/append) order.
998 if (sortResult == 0) {
999 return (left.row() < right.row());
1000 }
1001
1002 return (sortResult < 0);
1003 }
1004 }
1005}
1006
1007TasksModel::TasksModel(QObject *parent)
1008 : QSortFilterProxyModel(parent)
1009 , d(new Private(this))
1010{
1011 d->initModels();
1012
1013 // Start sorting.
1014 sort(0);
1015
1016 connect(this, &TasksModel::sourceModelChanged, this, &TasksModel::countChanged);
1017
1018 // Private::updateGroupInline() sets our source model, populating the model. We
1019 // delay running this until the QML runtime had a chance to call our implementation
1020 // of QQmlParserStatus::classBegin(), setting Private::usedByQml to true. If used
1021 // by QML, Private::updateGroupInline() will abort if the component is not yet
1022 // complete, instead getting called through QQmlParserStatus::componentComplete()
1023 // only after all properties have been set. This avoids delegate churn in Qt Quick
1024 // views using the model. If not used by QML, Private::updateGroupInline() will run
1025 // directly.
1026 QTimer::singleShot(0, this, [this]() {
1027 d->updateGroupInline();
1028 });
1029}
1030
1031TasksModel::~TasksModel()
1032{
1033}
1034
1035QHash<int, QByteArray> TasksModel::roleNames() const
1036{
1037 if (d->windowTasksModel) {
1038 return d->windowTasksModel->roleNames();
1039 }
1040
1041 return QHash<int, QByteArray>();
1042}
1043
1044int TasksModel::rowCount(const QModelIndex &parent) const
1045{
1046 return QSortFilterProxyModel::rowCount(parent);
1047}
1048
1049QVariant TasksModel::data(const QModelIndex &proxyIndex, int role) const
1050{
1051 if (role == AbstractTasksModel::HasLauncher && proxyIndex.isValid() && proxyIndex.row() < rowCount()) {
1052 if (proxyIndex.data(AbstractTasksModel::IsLauncher).toBool()) {
1053 return true;
1054 } else {
1055 if (!d->launcherTasksModel) {
1056 return false;
1057 }
1058 for (int i = 0; i < d->launcherTasksModel->rowCount(); ++i) {
1059 const QModelIndex &launcherIndex = d->launcherTasksModel->index(i, 0);
1060
1061 if (appsMatch(proxyIndex, launcherIndex)) {
1062 return true;
1063 }
1064 }
1065
1066 return false;
1067 }
1068 } else if (rowCount(proxyIndex) && role == AbstractTasksModel::WinIdList) {
1069 QVariantList winIds;
1070
1071 for (int i = 0; i < rowCount(proxyIndex); ++i) {
1072 winIds.append(index(i, 0, proxyIndex).data(AbstractTasksModel::WinIdList).toList());
1073 }
1074
1075 return winIds;
1076 }
1077
1078 return QSortFilterProxyModel::data(proxyIndex, role);
1079}
1080
1081void TasksModel::updateLauncherCount()
1082{
1083 if (!d->launcherTasksModel) {
1084 return;
1085 }
1086
1087 int count = 0;
1088
1089 for (int i = 0; i < rowCount(); ++i) {
1090 if (index(i, 0).data(AbstractTasksModel::IsLauncher).toBool()) {
1091 ++count;
1092 }
1093 }
1094
1095 if (d->launcherCount != count) {
1096 d->launcherCount = count;
1097 Q_EMIT launcherCountChanged();
1098 }
1099}
1100
1101int TasksModel::launcherCount() const
1102{
1103 return d->launcherCount;
1104}
1105
1106bool TasksModel::anyTaskDemandsAttention() const
1107{
1108 return d->anyTaskDemandsAttention;
1109}
1110
1111QVariant TasksModel::virtualDesktop() const
1112{
1113 return d->filterProxyModel->virtualDesktop();
1114}
1115
1117{
1118 d->filterProxyModel->setVirtualDesktop(desktop);
1119}
1120
1121QRect TasksModel::screenGeometry() const
1122{
1123 return d->filterProxyModel->screenGeometry();
1124}
1125
1127{
1128 d->filterProxyModel->setScreenGeometry(geometry);
1129}
1130
1131QRect TasksModel::regionGeometry() const
1132{
1133 return d->filterProxyModel->regionGeometry();
1134}
1135
1137{
1138 d->filterProxyModel->setRegionGeometry(geometry);
1139}
1140
1141QString TasksModel::activity() const
1142{
1143 return d->filterProxyModel->activity();
1144}
1145
1147{
1148 d->filterProxyModel->setActivity(activity);
1149}
1150
1151bool TasksModel::filterByVirtualDesktop() const
1152{
1153 return d->filterProxyModel->filterByVirtualDesktop();
1154}
1155
1157{
1158 d->filterProxyModel->setFilterByVirtualDesktop(filter);
1159}
1160
1161bool TasksModel::filterByScreen() const
1162{
1163 return d->filterProxyModel->filterByScreen();
1164}
1165
1167{
1168 d->filterProxyModel->setFilterByScreen(filter);
1169}
1170
1171bool TasksModel::filterByActivity() const
1172{
1173 return d->filterProxyModel->filterByActivity();
1174}
1175
1177{
1178 d->filterProxyModel->setFilterByActivity(filter);
1179}
1180
1181RegionFilterMode::Mode TasksModel::filterByRegion() const
1182{
1183 return d->filterProxyModel->filterByRegion();
1184}
1185
1186void TasksModel::setFilterByRegion(RegionFilterMode::Mode mode)
1187{
1188 d->filterProxyModel->setFilterByRegion(mode);
1189}
1190
1191bool TasksModel::filterMinimized() const
1192{
1193 return d->filterProxyModel->filterMinimized();
1194}
1195
1197{
1198 d->filterProxyModel->setFilterMinimized(filter);
1199}
1200
1201bool TasksModel::filterNotMinimized() const
1202{
1203 return d->filterProxyModel->filterNotMinimized();
1204}
1205
1207{
1208 d->filterProxyModel->setFilterNotMinimized(filter);
1209}
1210
1211bool TasksModel::filterNotMaximized() const
1212{
1213 return d->filterProxyModel->filterNotMaximized();
1214}
1215
1217{
1218 d->filterProxyModel->setFilterNotMaximized(filter);
1219}
1220
1221bool TasksModel::filterHidden() const
1222{
1223 return d->filterProxyModel->filterHidden();
1224}
1225
1227{
1228 d->filterProxyModel->setFilterHidden(filter);
1229}
1230
1231TasksModel::SortMode TasksModel::sortMode() const
1232{
1233 return d->sortMode;
1234}
1235
1237{
1238 if (d->sortMode != mode) {
1239 if (mode == SortManual) {
1240 d->updateManualSortMap();
1241 } else if (d->sortMode == SortManual) {
1242 d->sortedPreFilterRows.clear();
1243 }
1244
1245 if (mode == SortVirtualDesktop || mode == SortWindowPositionHorizontal) {
1246 d->virtualDesktopInfo = virtualDesktopInfo();
1248 } else if (d->sortMode == SortVirtualDesktop || d->sortMode == SortWindowPositionHorizontal) {
1249 d->virtualDesktopInfo = nullptr;
1251 }
1252
1253 if (mode == SortActivity) {
1254 d->activityInfo = activityInfo();
1255
1256 d->updateActivityTaskCounts();
1258 } else if (d->sortMode == SortActivity) {
1259 d->activityInfo = nullptr;
1260
1261 d->activityTaskCounts.clear();
1263 }
1264
1265 if (mode == SortLastActivated) {
1267 }
1268
1269 d->sortMode = mode;
1270
1271 d->forceResort();
1272
1273 Q_EMIT sortModeChanged();
1274 }
1275}
1276
1277bool TasksModel::separateLaunchers() const
1278{
1279 return d->separateLaunchers;
1280}
1281
1283{
1284 if (d->separateLaunchers != separate) {
1285 d->separateLaunchers = separate;
1286
1287 d->updateManualSortMap();
1288 d->forceResort();
1289
1290 Q_EMIT separateLaunchersChanged();
1291 }
1292}
1293
1294bool TasksModel::launchInPlace() const
1295{
1296 return d->launchInPlace;
1297}
1298
1299void TasksModel::setLaunchInPlace(bool launchInPlace)
1300{
1301 if (d->launchInPlace != launchInPlace) {
1302 d->launchInPlace = launchInPlace;
1303
1304 d->forceResort();
1305
1306 Q_EMIT launchInPlaceChanged();
1307 }
1308}
1309
1310TasksModel::GroupMode TasksModel::groupMode() const
1311{
1312 if (!d->groupingProxyModel) {
1313 return GroupDisabled;
1314 }
1315
1316 return d->groupingProxyModel->groupMode();
1317}
1318
1319bool TasksModel::hideActivatedLaunchers() const
1320{
1321 return d->hideActivatedLaunchers;
1322}
1323
1324void TasksModel::setHideActivatedLaunchers(bool hideActivatedLaunchers)
1325{
1326 if (d->hideActivatedLaunchers != hideActivatedLaunchers) {
1327 d->hideActivatedLaunchers = hideActivatedLaunchers;
1328
1329 d->updateManualSortMap();
1331 d->forceResort();
1332
1333 Q_EMIT hideActivatedLaunchersChanged();
1334 }
1335}
1336
1338{
1339 if (d->groupingProxyModel) {
1340 if (mode == GroupDisabled && d->flattenGroupsProxyModel) {
1341 d->flattenGroupsProxyModel->setSourceModel(nullptr);
1342 }
1343
1344 d->groupingProxyModel->setGroupMode(mode);
1345 d->updateGroupInline();
1346 }
1347}
1348
1349bool TasksModel::groupInline() const
1350{
1351 return d->groupInline;
1352}
1353
1354void TasksModel::setGroupInline(bool groupInline)
1355{
1356 if (d->groupInline != groupInline) {
1357 d->groupInline = groupInline;
1358
1359 d->updateGroupInline();
1360
1361 Q_EMIT groupInlineChanged();
1362 }
1363}
1364
1365int TasksModel::groupingWindowTasksThreshold() const
1366{
1367 return d->groupingWindowTasksThreshold;
1368}
1369
1371{
1372 if (d->groupingWindowTasksThreshold != threshold) {
1373 d->groupingWindowTasksThreshold = threshold;
1374
1375 if (!d->groupInline && d->groupingProxyModel) {
1376 d->groupingProxyModel->setWindowTasksThreshold(threshold);
1377 }
1378
1379 Q_EMIT groupingWindowTasksThresholdChanged();
1380 }
1381}
1382
1383QStringList TasksModel::groupingAppIdBlacklist() const
1384{
1385 if (!d->groupingProxyModel) {
1386 return QStringList();
1387 }
1388
1389 return d->groupingProxyModel->blacklistedAppIds();
1390}
1391
1393{
1394 if (d->groupingProxyModel) {
1395 d->groupingProxyModel->setBlacklistedAppIds(list);
1396 }
1397}
1398
1399QStringList TasksModel::groupingLauncherUrlBlacklist() const
1400{
1401 if (!d->groupingProxyModel) {
1402 return QStringList();
1403 }
1404
1405 return d->groupingProxyModel->blacklistedLauncherUrls();
1406}
1407
1409{
1410 if (d->groupingProxyModel) {
1411 d->groupingProxyModel->setBlacklistedLauncherUrls(list);
1412 }
1413}
1414
1415bool TasksModel::taskReorderingEnabled() const
1416{
1417 return dynamicSortFilter();
1418}
1419
1421{
1422 enabled ? setDynamicSortFilter(true) : setDynamicSortFilter(false);
1423
1424 Q_EMIT taskReorderingEnabledChanged();
1425}
1426
1427QStringList TasksModel::launcherList() const
1428{
1429 if (d->launcherTasksModel) {
1430 return d->launcherTasksModel->launcherList();
1431 }
1432
1433 return QStringList();
1434}
1435
1437{
1438 d->initLauncherTasksModel();
1439 d->launcherTasksModel->setLauncherList(launchers);
1440 d->launchersEverSet = true;
1441}
1442
1444{
1445 d->initLauncherTasksModel();
1446
1447 bool added = d->launcherTasksModel->requestAddLauncher(url);
1448
1449 // If using manual and launch-in-place sorting with separate launchers,
1450 // we need to trigger a sort map update to move any window tasks to
1451 // their launcher position now.
1452 if (added && d->sortMode == SortManual && (d->launchInPlace || !d->separateLaunchers)) {
1453 d->updateManualSortMap();
1454 d->forceResort();
1455 }
1456
1457 return added;
1458}
1459
1461{
1462 if (d->launcherTasksModel) {
1463 bool removed = d->launcherTasksModel->requestRemoveLauncher(url);
1464
1465 // If using manual and launch-in-place sorting with separate launchers,
1466 // we need to trigger a sort map update to move any window tasks no
1467 // longer backed by a launcher out of the launcher area.
1468 if (removed && d->sortMode == SortManual && (d->launchInPlace || !d->separateLaunchers)) {
1469 d->updateManualSortMap();
1470 d->forceResort();
1471 }
1472
1473 return removed;
1474 }
1475
1476 return false;
1477}
1478
1480{
1481 d->initLauncherTasksModel();
1482
1483 bool added = d->launcherTasksModel->requestAddLauncherToActivity(url, activity);
1484
1485 // If using manual and launch-in-place sorting with separate launchers,
1486 // we need to trigger a sort map update to move any window tasks to
1487 // their launcher position now.
1488 if (added && d->sortMode == SortManual && (d->launchInPlace || !d->separateLaunchers)) {
1489 d->updateManualSortMap();
1490 d->forceResort();
1491 }
1492
1493 return added;
1494}
1495
1497{
1498 if (d->launcherTasksModel) {
1499 bool removed = d->launcherTasksModel->requestRemoveLauncherFromActivity(url, activity);
1500
1501 // If using manual and launch-in-place sorting with separate launchers,
1502 // we need to trigger a sort map update to move any window tasks no
1503 // longer backed by a launcher out of the launcher area.
1504 if (removed && d->sortMode == SortManual && (d->launchInPlace || !d->separateLaunchers)) {
1505 d->updateManualSortMap();
1506 d->forceResort();
1507 }
1508
1509 return removed;
1510 }
1511
1512 return false;
1513}
1514
1516{
1517 if (d->launcherTasksModel) {
1518 return d->launcherTasksModel->launcherActivities(url);
1519 }
1520
1521 return {};
1522}
1523
1525{
1526 if (d->launcherTasksModel) {
1527 return d->launcherTasksModel->launcherPosition(url);
1528 }
1529
1530 return -1;
1531}
1532
1534{
1535 if (index.isValid() && index.model() == this) {
1536 d->abstractTasksSourceModel->requestActivate(mapToSource(index));
1537 }
1538}
1539
1541{
1542 if (index.isValid() && index.model() == this) {
1543 d->abstractTasksSourceModel->requestNewInstance(mapToSource(index));
1544 }
1545}
1546
1548{
1549 if (index.isValid() && index.model() == this) {
1550 d->abstractTasksSourceModel->requestOpenUrls(mapToSource(index), urls);
1551 }
1552}
1553
1555{
1556 if (index.isValid() && index.model() == this) {
1557 d->abstractTasksSourceModel->requestClose(mapToSource(index));
1558 }
1559}
1560
1562{
1563 if (index.isValid() && index.model() == this) {
1564 d->abstractTasksSourceModel->requestMove(mapToSource(index));
1565 }
1566}
1567
1569{
1570 if (index.isValid() && index.model() == this) {
1571 d->abstractTasksSourceModel->requestResize(mapToSource(index));
1572 }
1573}
1574
1576{
1577 if (index.isValid() && index.model() == this) {
1578 d->abstractTasksSourceModel->requestToggleMinimized(mapToSource(index));
1579 }
1580}
1581
1583{
1584 if (index.isValid() && index.model() == this) {
1585 d->abstractTasksSourceModel->requestToggleMaximized(mapToSource(index));
1586 }
1587}
1588
1590{
1591 if (index.isValid() && index.model() == this) {
1592 d->abstractTasksSourceModel->requestToggleKeepAbove(mapToSource(index));
1593 }
1594}
1595
1597{
1598 if (index.isValid() && index.model() == this) {
1599 d->abstractTasksSourceModel->requestToggleKeepBelow(mapToSource(index));
1600 }
1601}
1602
1604{
1605 if (index.isValid() && index.model() == this) {
1606 d->abstractTasksSourceModel->requestToggleFullScreen(mapToSource(index));
1607 }
1608}
1609
1611{
1612 if (index.isValid() && index.model() == this) {
1613 d->abstractTasksSourceModel->requestToggleShaded(mapToSource(index));
1614 }
1615}
1616
1618{
1619 if (index.isValid() && index.model() == this) {
1620 d->abstractTasksSourceModel->requestToggleNoBorder(mapToSource(index));
1621 }
1622}
1623
1624void TasksModel::requestVirtualDesktops(const QModelIndex &index, const QVariantList &desktops)
1625{
1626 if (index.isValid() && index.model() == this) {
1627 d->abstractTasksSourceModel->requestVirtualDesktops(mapToSource(index), desktops);
1628 }
1629}
1630
1632{
1633 if (index.isValid() && index.model() == this) {
1634 d->abstractTasksSourceModel->requestNewVirtualDesktop(mapToSource(index));
1635 }
1636}
1637
1639{
1640 if (index.isValid() && index.model() == this) {
1641 d->groupingProxyModel->requestActivities(mapToSource(index), activities);
1642 }
1643}
1644
1646{
1647 if (!index.isValid() || index.model() != this || !index.data(AbstractTasksModel::IsWindow).toBool()) {
1648 return;
1649 }
1650 d->abstractTasksSourceModel->requestPublishDelegateGeometry(mapToSource(index), geometry, delegate);
1651}
1652
1654{
1655 if (index.isValid() && index.model() == this) {
1656 const QModelIndex &target = (d->flattenGroupsProxyModel ? d->flattenGroupsProxyModel->mapToSource(mapToSource(index)) : mapToSource(index));
1657 d->groupingProxyModel->requestToggleGrouping(target);
1658 }
1659}
1660
1661bool TasksModel::move(int row, int newPos, const QModelIndex &parent)
1662{
1663 /*
1664 * NOTE After doing any modification in TasksModel::move, make sure fixes listed below are not regressed.
1665 * - https://bugs.kde.org/444816
1666 * - https://bugs.kde.org/448912
1667 * - https://invent.kde.org/plasma/plasma-workspace/-/commit/ea51795e8c571513e1ff583350ab8649bc857fc2
1668 */
1669
1670 if (d->sortMode != SortManual || row == newPos || newPos < 0 || newPos >= rowCount(parent)) {
1671 return false;
1672 }
1673
1674 const QModelIndex &idx = index(row, 0, parent);
1675 bool isLauncherMove = false;
1676
1677 // Figure out if we're moving a launcher so we can run barrier checks.
1678 if (idx.isValid()) {
1680 isLauncherMove = true;
1681 // When using launch-in-place sorting, launcher-backed window tasks act as launchers.
1682 } else if ((d->launchInPlace || !d->separateLaunchers) && idx.data(AbstractTasksModel::IsWindow).toBool()) {
1683 const QUrl &launcherUrl = idx.data(AbstractTasksModel::LauncherUrlWithoutIcon).toUrl();
1684 const int launcherPos = launcherPosition(launcherUrl);
1685
1686 if (launcherPos != -1) {
1687 isLauncherMove = true;
1688 }
1689 }
1690 } else {
1691 return false;
1692 }
1693
1694 if (d->separateLaunchers && !parent.isValid() /* Exclude tasks in a group */) {
1695 int firstTask = 0;
1696 if (d->launcherTasksModel) {
1697 if (d->launchInPlace) {
1698 firstTask = d->launcherTasksModel->rowCountForActivity(activity());
1699 } else {
1700 firstTask = launcherCount();
1701 }
1702 }
1703
1704 // Don't allow launchers to be moved past the last launcher.
1705 if (isLauncherMove && newPos >= firstTask) {
1706 return false;
1707 }
1708
1709 // Don't allow tasks to be moved into the launchers.
1710 if (!isLauncherMove && newPos < firstTask) {
1711 return false;
1712 }
1713 }
1714
1715 // Treat flattened-out groups as single items.
1716 if (d->flattenGroupsProxyModel) {
1717 QModelIndex groupingRowIndex = d->flattenGroupsProxyModel->mapToSource(mapToSource(index(row, 0)));
1718 const QModelIndex &groupingRowIndexParent = groupingRowIndex.parent();
1719 QModelIndex groupingNewPosIndex = d->flattenGroupsProxyModel->mapToSource(mapToSource(index(newPos, 0)));
1720 const QModelIndex &groupingNewPosIndexParent = groupingNewPosIndex.parent();
1721
1722 // Disallow moves within a flattened-out group (TODO: for now, anyway).
1723 if (groupingRowIndexParent.isValid() && (groupingRowIndexParent == groupingNewPosIndex || groupingRowIndexParent == groupingNewPosIndexParent)) {
1724 return false;
1725 }
1726
1727 int offset = 0;
1728 int extraChildCount = 0;
1729
1730 if (groupingRowIndexParent.isValid()) {
1731 offset = groupingRowIndex.row();
1732 extraChildCount = d->groupingProxyModel->rowCount(groupingRowIndexParent) - 1;
1733 groupingRowIndex = groupingRowIndexParent;
1734 }
1735
1736 if (groupingNewPosIndexParent.isValid()) {
1737 int extra = d->groupingProxyModel->rowCount(groupingNewPosIndexParent) - 1;
1738
1739 if (newPos > row) {
1740 newPos += extra;
1741 newPos -= groupingNewPosIndex.row();
1742 groupingNewPosIndex = groupingNewPosIndexParent.model()->index(extra, 0, groupingNewPosIndexParent);
1743 } else {
1744 newPos -= groupingNewPosIndex.row();
1745 groupingNewPosIndex = groupingNewPosIndexParent;
1746 }
1747 }
1748
1749 beginMoveRows(QModelIndex(), (row - offset), (row - offset) + extraChildCount, QModelIndex(), (newPos > row) ? newPos + 1 : newPos);
1750
1751 row = d->sortedPreFilterRows.indexOf(d->filterProxyModel->mapToSource(d->groupingProxyModel->mapToSource(groupingRowIndex)).row());
1752 newPos = d->sortedPreFilterRows.indexOf(d->filterProxyModel->mapToSource(d->groupingProxyModel->mapToSource(groupingNewPosIndex)).row());
1753
1754 // Update sort mappings.
1755 d->sortedPreFilterRows.move(row, newPos);
1756
1757 endMoveRows();
1758
1759 if (groupingRowIndexParent.isValid()) {
1760 d->consolidateManualSortMapForGroup(groupingRowIndexParent);
1761 }
1762
1763 } else {
1764 beginMoveRows(parent, row, row, parent, (newPos > row) ? newPos + 1 : newPos);
1765
1766 // Translate to sort map indices.
1767 const QModelIndex &groupingRowIndex = mapToSource(index(row, 0, parent));
1768 const QModelIndex &preFilterRowIndex = d->preFilterIndex(groupingRowIndex);
1769
1770 const bool groupNotDisabled = !parent.isValid() && groupMode() != GroupDisabled;
1771 QModelIndex adjacentGroupingRowIndex; // Also consolidate the adjacent group parent
1772 if (groupNotDisabled) {
1773 if (newPos > row && row + 1 < rowCount(parent)) {
1774 adjacentGroupingRowIndex = mapToSource(index(row + 1, 0, parent) /* task on the right */);
1775 } else if (newPos < row && row - 1 >= 0) {
1776 adjacentGroupingRowIndex = mapToSource(index(row - 1, 0, parent) /* task on the left */);
1777 }
1778 }
1779
1780 row = d->sortedPreFilterRows.indexOf(preFilterRowIndex.row());
1781 newPos = d->sortedPreFilterRows.indexOf(d->preFilterIndex(mapToSource(index(newPos, 0, parent))).row());
1782
1783 // Update sort mapping.
1784 d->sortedPreFilterRows.move(row, newPos);
1785
1786 endMoveRows();
1787
1788 // If we moved a group parent, consolidate sort map for children.
1789 if (groupNotDisabled) {
1790 if (d->groupingProxyModel->rowCount(groupingRowIndex)) {
1791 d->consolidateManualSortMapForGroup(groupingRowIndex);
1792 }
1793 // Special case: Before moving, the task at newPos is a group parent
1794 // Before moving: [Task] [Group parent] [Other task in group]
1795 // After moving: [Group parent (not consolidated yet)] [Task, newPos] [Other task in group]
1796 if (int childCount = d->groupingProxyModel->rowCount(adjacentGroupingRowIndex); childCount && adjacentGroupingRowIndex.isValid()) {
1797 d->consolidateManualSortMapForGroup(adjacentGroupingRowIndex);
1798 if (newPos > row) {
1799 newPos += childCount - 1;
1800 // After consolidation: [Group parent (not consolidated yet)] [Other task in group] [Task, newPos]
1801 }
1802 // No need to consider newPos < row
1803 // Before moving: [Group parent, newPos] [Other task in group] [Task]
1804 // After moving: [Task, newPos] [Group parent] [Other task in group]
1805 }
1806 }
1807 }
1808
1809 // Resort.
1810 d->forceResort();
1811
1812 if (!d->separateLaunchers) {
1813 if (isLauncherMove) {
1814 const QModelIndex &idx = d->concatProxyModel->index(d->sortedPreFilterRows.at(newPos), 0);
1815 const QUrl &launcherUrl = idx.data(AbstractTasksModel::LauncherUrlWithoutIcon).toUrl();
1816
1817 // Move launcher for launcher-backed task along with task if launchers
1818 // are not being kept separate.
1819 // We don't need to resort again because the launcher is implicitly hidden
1820 // at this time.
1822 const int launcherPos = d->launcherTasksModel->launcherPosition(launcherUrl);
1823 const QModelIndex &launcherIndex = d->launcherTasksModel->index(launcherPos, 0);
1824 const int sortIndex = d->sortedPreFilterRows.indexOf(d->concatProxyModel->mapFromSource(launcherIndex).row());
1825 d->sortedPreFilterRows.move(sortIndex, newPos);
1826
1827 if (row > newPos && newPos >= 1) {
1828 const QModelIndex beforeIdx = d->concatProxyModel->index(d->sortedPreFilterRows.at(newPos - 1), 0);
1829 if (beforeIdx.data(AbstractTasksModel::IsLauncher).toBool()) {
1830 // Search forward to skip grouped tasks
1831 int afterPos = newPos + 1;
1832 for (; afterPos < d->sortedPreFilterRows.size(); ++afterPos) {
1833 const QModelIndex tempIdx = d->concatProxyModel->index(d->sortedPreFilterRows.at(afterPos), 0);
1834 if (!appsMatch(idx, tempIdx)) {
1835 break;
1836 }
1837 }
1838
1839 const QModelIndex afterIdx = d->concatProxyModel->index(d->sortedPreFilterRows.at(afterPos), 0);
1840 if (appsMatch(beforeIdx, afterIdx)) {
1841 d->sortedPreFilterRows.move(newPos - 1, afterPos - 1);
1842 }
1843 }
1844 }
1845 // Otherwise move matching windows to after the launcher task (they are
1846 // currently hidden but might be on another virtual desktop).
1847 } else {
1848 for (int i = (d->sortedPreFilterRows.count() - 1); i >= 0; --i) {
1849 const QModelIndex &concatProxyIndex = d->concatProxyModel->index(d->sortedPreFilterRows.at(i), 0);
1850
1851 if (launcherUrl == concatProxyIndex.data(AbstractTasksModel::LauncherUrlWithoutIcon).toUrl()) {
1852 d->sortedPreFilterRows.move(i, newPos);
1853
1854 if (newPos > i) {
1855 --newPos;
1856 }
1857 }
1858 }
1859 }
1860 } else if (newPos > 0 && newPos < d->sortedPreFilterRows.size() - 1) {
1861 /*
1862 * When dragging an unpinned task, a pinned task can also be moved.
1863 * In this case, sortedPreFilterRows is like:
1864 * - before moving: [pinned 1 (launcher item)] [pinned 1 (window)] [unpinned]
1865 * - after moving: [pinned 1 (launcher item)] [unpinned] [pinned 1 (window)]
1866 * So also check the indexes before and after the unpinned task.
1867 */
1868 const QModelIndex beforeIdx = d->concatProxyModel->index(d->sortedPreFilterRows.at(newPos - 1), 0, parent);
1869 const QModelIndex afterIdx = d->concatProxyModel->index(d->sortedPreFilterRows.at(newPos + 1), 0, parent);
1870 // BUG 462508: check if any item is a launcher
1871 const bool hasLauncher = beforeIdx.data(AbstractTasksModel::IsLauncher).toBool() || afterIdx.data(AbstractTasksModel::IsLauncher).toBool();
1872
1873 if (hasLauncher && appsMatch(beforeIdx, afterIdx)) {
1874 // after adjusting: [unpinned] [pinned 1 (launcher item)] [pinned 1]
1875 d->sortedPreFilterRows.move(newPos, newPos + (row < newPos ? 1 : -1));
1876 }
1877 }
1878 }
1879
1880 // Setup for syncLaunchers().
1881 d->launcherSortingDirty = isLauncherMove;
1882
1883 return true;
1884}
1885
1887{
1888 // Writes the launcher order exposed through the model back to the launcher
1889 // tasks model, committing any move() operations to persistent state.
1890
1891 if (!d->launcherTasksModel || !d->launcherSortingDirty) {
1892 return;
1893 }
1894
1895 QMap<int, QString> sortedShownLaunchers;
1896 QStringList sortedHiddenLaunchers;
1897
1898 for (const auto launchers = launcherList(); const QString &launcherUrlStr : launchers) {
1899 int row = -1;
1900 QList<QStringView> activities;
1901 QUrl launcherUrl;
1902
1903 std::tie(launcherUrl, activities) = deserializeLauncher(launcherUrlStr);
1904
1905 for (int i = 0; i < rowCount(); ++i) {
1906 const QUrl &rowLauncherUrl = index(i, 0).data(AbstractTasksModel::LauncherUrlWithoutIcon).toUrl();
1907
1908 // `LauncherTasksModel::launcherList()` returns data in a format suitable for writing
1909 // to persistent configuration storage, e.g. `preferred://browser`. We mean to compare
1910 // this last "save state" to a higher, resolved URL representation to compute the delta
1911 // so we need to move the unresolved URLs through `TaskTools::appDataFromUrl()` first.
1912 // TODO: This bypasses an existing lookup cache for the resolved app data that exists
1913 // in LauncherTasksModel. It's likely a good idea to eventually move these caches out
1914 // of the various models and share them among users of `TaskTools::appDataFromUrl()`,
1915 // and then also do resolution implicitly in `TaskTools::launcherUrlsMatch`, to speed
1916 // things up slightly and make the models simpler (central cache eviction, ...).
1917 if (launcherUrlsMatch(appDataFromUrl(launcherUrl).url, rowLauncherUrl, IgnoreQueryItems)) {
1918 row = i;
1919 break;
1920 }
1921 }
1922
1923 if (row != -1) {
1924 sortedShownLaunchers.insert(row, launcherUrlStr);
1925 } else {
1926 sortedHiddenLaunchers << launcherUrlStr;
1927 }
1928 }
1929
1930 // Prep sort map for source model data changes.
1931 if (d->sortMode == SortManual) {
1932 QList<int> sortMapIndices;
1933 QList<int> preFilterRows;
1934
1935 for (int i = 0; i < d->launcherTasksModel->rowCount(); ++i) {
1936 const QModelIndex &launcherIndex = d->launcherTasksModel->index(i, 0);
1937 const QModelIndex &concatIndex = d->concatProxyModel->mapFromSource(launcherIndex);
1938 sortMapIndices << d->sortedPreFilterRows.indexOf(concatIndex.row());
1939 preFilterRows << concatIndex.row();
1940 }
1941
1942 // We're going to write back launcher model entries in the sort
1943 // map in concat model order, matching the reordered launcher list
1944 // we're about to pass down.
1945 std::sort(sortMapIndices.begin(), sortMapIndices.end());
1946
1947 for (int i = 0; i < sortMapIndices.count(); ++i) {
1948 d->sortedPreFilterRows.replace(sortMapIndices.at(i), preFilterRows.at(i));
1949 }
1950 }
1951
1952 setLauncherList(sortedShownLaunchers.values() + sortedHiddenLaunchers);
1953
1954 // The accepted rows are outdated after the item order is changed
1956 d->forceResort();
1957
1958 d->launcherSortingDirty = false;
1959}
1960
1961QModelIndex TasksModel::activeTask() const
1962{
1963 for (int i = 0; i < rowCount(); ++i) {
1964 const QModelIndex &idx = index(i, 0);
1965
1967 if (groupMode() != GroupDisabled && rowCount(idx)) {
1968 for (int j = 0; j < rowCount(idx); ++j) {
1969 const QModelIndex &child = index(j, 0, idx);
1970
1972 return child;
1973 }
1974 }
1975 } else {
1976 return idx;
1977 }
1978 }
1979 }
1980
1981 return QModelIndex();
1982}
1983
1984QModelIndex TasksModel::makeModelIndex(int row, int childRow) const
1985{
1986 if (row < 0 || row >= rowCount()) {
1987 return QModelIndex();
1988 }
1989
1990 if (childRow == -1) {
1991 return index(row, 0);
1992 } else {
1993 const QModelIndex &parent = index(row, 0);
1994
1995 if (childRow < rowCount(parent)) {
1996 return index(childRow, 0, parent);
1997 }
1998 }
1999
2000 return QModelIndex();
2001}
2002
2004{
2005 return QPersistentModelIndex(makeModelIndex(row, childCount));
2006}
2007
2008void TasksModel::classBegin()
2009{
2010 d->usedByQml = true;
2011}
2012
2013void TasksModel::componentComplete()
2014{
2015 d->componentComplete = true;
2016
2017 // Sets our source model, populating the model.
2018 d->updateGroupInline();
2019}
2020
2021bool TasksModel::filterAcceptsRow(int sourceRow, const QModelIndex &sourceParent) const
2022{
2023 // All our filtering occurs at the top-level; anything below always
2024 // goes through.
2025 if (sourceParent.isValid()) {
2026 return true;
2027 }
2028
2029 const QModelIndex &sourceIndex = sourceModel()->index(sourceRow, 0);
2030
2031 // In inline grouping mode, filter out group parents.
2032 if (d->groupInline && d->flattenGroupsProxyModel && sourceIndex.data(AbstractTasksModel::IsGroupParent).toBool()) {
2033 return false;
2034 }
2035
2036 const QString &appId = sourceIndex.data(AbstractTasksModel::AppId).toString();
2037 const QString &appName = sourceIndex.data(AbstractTasksModel::AppName).toString();
2038
2039 // Filter startup tasks we already have a window task for.
2040 if (sourceIndex.data(AbstractTasksModel::IsStartup).toBool()) {
2041 for (int i = 0; i < d->filterProxyModel->rowCount(); ++i) {
2042 const QModelIndex &filterIndex = d->filterProxyModel->index(i, 0);
2043
2044 if (!filterIndex.data(AbstractTasksModel::IsWindow).toBool()) {
2045 continue;
2046 }
2047
2048 if ((!appId.isEmpty() && appId == filterIndex.data(AbstractTasksModel::AppId).toString())
2049 || (!appName.isEmpty() && appName == filterIndex.data(AbstractTasksModel::AppName).toString())) {
2050 return false;
2051 }
2052 }
2053 }
2054
2055 // Filter launcher tasks we already have a startup or window task for (that
2056 // got through filtering).
2057 if (d->hideActivatedLaunchers && sourceIndex.data(AbstractTasksModel::IsLauncher).toBool()) {
2058 for (int i = 0; i < d->filterProxyModel->rowCount(); ++i) {
2059 const QModelIndex &filteredIndex = d->filterProxyModel->index(i, 0);
2060
2061 if (!filteredIndex.data(AbstractTasksModel::IsWindow).toBool() && !filteredIndex.data(AbstractTasksModel::IsStartup).toBool()) {
2062 continue;
2063 }
2064
2065 if (appsMatch(sourceIndex, filteredIndex)) {
2066 return false;
2067 }
2068 }
2069 }
2070
2071 return true;
2072}
2073
2074bool TasksModel::lessThan(const QModelIndex &left, const QModelIndex &right) const
2075{
2076 // In manual sort mode, sort by map.
2077 if (d->sortMode == SortManual) {
2078 return (d->sortedPreFilterRows.indexOf(d->preFilterIndex(left).row()) < d->sortedPreFilterRows.indexOf(d->preFilterIndex(right).row()));
2079 }
2080
2081 return d->lessThan(left, right);
2082}
2083
2084std::shared_ptr<VirtualDesktopInfo> TasksModel::virtualDesktopInfo() const
2085{
2086 static std::weak_ptr<VirtualDesktopInfo> s_virtualDesktopInfo;
2087 if (s_virtualDesktopInfo.expired()) {
2088 auto ptr = std::make_shared<VirtualDesktopInfo>();
2089 s_virtualDesktopInfo = ptr;
2090 return ptr;
2091 }
2092 return s_virtualDesktopInfo.lock();
2093}
2094
2095std::shared_ptr<ActivityInfo> TasksModel::activityInfo() const
2096{
2097 static std::weak_ptr<ActivityInfo> s_activityInfo;
2098 if (s_activityInfo.expired()) {
2099 auto ptr = std::make_shared<ActivityInfo>();
2100 s_activityInfo = ptr;
2101 return ptr;
2102 }
2103 return s_activityInfo.lock();
2104}
2105}
2106
2107#include "moc_tasksmodel.cpp"
@ IsLauncher
This is a launcher task.
@ Activities
Activities for the task (i.e.
@ IsActive
This is the currently active task.
@ IsWindow
This is a window task.
@ VirtualDesktops
Virtual desktops for the task (i.e.
@ LauncherUrlWithoutIcon
Special path to get a launcher URL while skipping fallback icon encoding.
@ LastActivated
The timestamp of the last time a task was the active task.
A tasks model for startup notifications.
A unified tasks model.
Definition tasksmodel.h:48
void setFilterNotMinimized(bool filter)
Set whether non-minimized tasks should be filtered.
void setLaunchInPlace(bool launchInPlace)
Sets whether window tasks should be sorted as their associated launcher tasks or separately.
Q_INVOKABLE int launcherPosition(const QUrl &url) const
Return the position of the launcher with the given URL.
Q_INVOKABLE void requestToggleMinimized(const QModelIndex &index) override
Request toggling the minimized state of the task at the given index.
Q_INVOKABLE void requestMove(const QModelIndex &index) override
Request starting an interactive move for the task at the given index.
void setFilterByVirtualDesktop(bool filter)
Set whether tasks should be filtered by virtual desktop.
void setFilterByActivity(bool filter)
Set whether tasks should be filtered by activity.
void setFilterByRegion(RegionFilterMode::Mode mode)
Set whether tasks should be filtered by region.
void setRegionGeometry(const QRect &geometry)
Set the geometry of the screen to use in filtering by region.
@ GroupDisabled
No grouping is done.
Definition tasksmodel.h:102
Q_INVOKABLE bool requestAddLauncher(const QUrl &url)
Request adding a launcher with the given URL.
void setGroupingWindowTasksThreshold(int threshold)
Sets the number of window tasks (AbstractTasksModel::IsWindow) above which groups will be formed,...
Q_INVOKABLE void requestToggleGrouping(const QModelIndex &index)
Request toggling whether the task at the given index, along with any tasks matching its kind,...
Q_INVOKABLE bool move(int row, int newPos, const QModelIndex &parent=QModelIndex())
Moves a task to a new position in the list.
Q_INVOKABLE void requestToggleKeepBelow(const QModelIndex &index) override
Request toggling the keep-below state of the task at the given index.
Q_INVOKABLE bool requestRemoveLauncher(const QUrl &url)
Request removing the launcher with the given URL.
Q_INVOKABLE QPersistentModelIndex makePersistentModelIndex(int row, int childRow=-1) const
Given a row in the model, returns a QPersistentModelIndex for it.
void setGroupingAppIdBlacklist(const QStringList &list)
Sets the blacklist of app ids (AbstractTasksModel::AppId) that is consulted before grouping a task.
void setGroupingLauncherUrlBlacklist(const QStringList &list)
Sets the blacklist of launcher URLs (AbstractTasksModel::LauncherUrl) that is consulted before groupi...
Q_INVOKABLE bool requestAddLauncherToActivity(const QUrl &url, const QString &activity)
Request adding a launcher with the given URL to current activity.
std::shared_ptr< ActivityInfo > activityInfo() const
Q_INVOKABLE void requestActivities(const QModelIndex &index, const QStringList &activities) override
Request moving the task at the given index to the specified activities.
Q_INVOKABLE bool requestRemoveLauncherFromActivity(const QUrl &url, const QString &activity)
Request removing the launcher with the given URL from the current activity.
void setLauncherList(const QStringList &launchers)
Replace the list of launcher URL strings.
Q_INVOKABLE void requestActivate(const QModelIndex &index) override
Request activation of the task at the given index.
void setFilterNotMaximized(bool filter)
Set whether non-maximized tasks should be filtered.
std::shared_ptr< VirtualDesktopInfo > virtualDesktopInfo() const
Q_INVOKABLE void requestToggleFullScreen(const QModelIndex &index) override
Request toggling the fullscreen state of the task at the given index.
Q_INVOKABLE void requestToggleKeepAbove(const QModelIndex &index) override
Request toggling the keep-above state of the task at the given index.
void setFilterMinimized(bool filter)
Sets whether non-minimized tasks should be filtered out.
void setTaskReorderingEnabled(bool enabled)
Enables or disables tasks reordering.
Q_INVOKABLE void requestResize(const QModelIndex &index) override
Request starting an interactive resize for the task at the given index.
void setGroupInline(bool groupInline)
Sets whether grouping is done "inline" or not, i.e.
void setFilterHidden(bool filter)
Set whether hidden tasks should be filtered.
Q_INVOKABLE void requestNewInstance(const QModelIndex &index) override
Request an additional instance of the application backing the task at the given index.
Q_INVOKABLE void requestToggleNoBorder(const QModelIndex &index) override
Request toggling the no border state of the task at given index.
Q_INVOKABLE void requestNewVirtualDesktop(const QModelIndex &index) override
Request entering the window at the given index on a new virtual desktop, which is created in response...
@ SortLastActivated
Tasks are sorted by the last time they were active.
Definition tasksmodel.h:96
@ SortManual
Tasks can be moved with move() and syncLaunchers().
Definition tasksmodel.h:92
@ SortActivity
Tasks are sorted by the number of tasks on the activities they're on.
Definition tasksmodel.h:95
@ SortWindowPositionHorizontal
Tasks are sorted by the virtual desktop they are on, then by window coordinates.
Definition tasksmodel.h:97
@ SortVirtualDesktop
Tasks are sorted by the virtual desktop they are on.
Definition tasksmodel.h:94
void setVirtualDesktop(const QVariant &desktop=QVariant())
Set the id of the virtual desktop to use in filtering by virtual desktop.
Q_INVOKABLE void requestVirtualDesktops(const QModelIndex &index, const QVariantList &desktops) override
Request entering the window at the given index on the specified virtual desktops.
void setSeparateLaunchers(bool separate)
Sets whether launchers are kept separate from other kinds of tasks.
Q_INVOKABLE void requestToggleMaximized(const QModelIndex &index) override
Request toggling the maximized state of the task at the given index.
void setHideActivatedLaunchers(bool hideActivatedLaunchers)
Sets whether launchers should be hidden after they have been activated.
Q_INVOKABLE void requestPublishDelegateGeometry(const QModelIndex &index, const QRect &geometry, QObject *delegate=nullptr) override
Request informing the window manager of new geometry for a visual delegate for the task at the given ...
void setActivity(const QString &activity)
Set the id of the activity to use in filtering by activity.
void setGroupMode(TasksModel::GroupMode mode)
Sets the group mode, i.e.
Q_INVOKABLE void syncLaunchers()
Updates the launcher list to reflect the new order after calls to move(), if needed.
void setSortMode(SortMode mode)
Sets the sort mode used in sorting tasks.
void setFilterByScreen(bool filter)
Set whether tasks should be filtered by screen.
Q_INVOKABLE void requestClose(const QModelIndex &index) override
Request the task at the given index be closed.
Q_INVOKABLE void requestToggleShaded(const QModelIndex &index) override
Request toggling the shaded state of the task at the given index.
Q_INVOKABLE QStringList launcherActivities(const QUrl &url)
Return the list of activities the launcher belongs to.
Q_INVOKABLE void requestOpenUrls(const QModelIndex &index, const QList< QUrl > &urls) override
Requests to open the given URLs with the application backing the task at the given index.
Q_INVOKABLE QModelIndex makeModelIndex(int row, int childRow=-1) const
Given a row in the model, returns a QModelIndex for it.
void setScreenGeometry(const QRect &geometry)
Set the geometry of the screen to use in filtering by screen.
Q_SCRIPTABLE QString start(QString train="")
QAction * end(const QObject *recvr, const char *slot, QObject *parent)
QCA_EXPORT QString appName()
QAbstractItemModel(QObject *parent)
bool beginMoveRows(const QModelIndex &sourceParent, int sourceFirst, int sourceLast, const QModelIndex &destinationParent, int destinationChild)
void dataChanged(const QModelIndex &topLeft, const QModelIndex &bottomRight, const QList< int > &roles)
virtual QModelIndex index(int row, int column, const QModelIndex &parent) const const=0
void rowsAboutToBeInserted(const QModelIndex &parent, int start, int end)
void rowsAboutToBeRemoved(const QModelIndex &parent, int first, int last)
void rowsInserted(const QModelIndex &parent, int first, int last)
void rowsRemoved(const QModelIndex &parent, int first, int last)
bool isValid() const const
bool isRightToLeft()
const_reference at(qsizetype i) const const
iterator begin()
const_iterator cbegin() const const
const_iterator cend() const const
bool contains(const AT &value) const const
qsizetype count() const const
pointer data()
iterator end()
qsizetype indexOf(const AT &value, qsizetype from) const const
bool isEmpty() const const
iterator insert(const Key &key, const T &value)
QList< T > values() const const
QVariant data(int role) const const
bool isValid() const const
const QAbstractItemModel * model() const const
QModelIndex parent() const const
int row() const const
Q_EMITQ_EMIT
QMetaObject::Connection connect(const QObject *sender, PointerToMemberFunction signal, Functor functor)
int right() const const
int x() const const
int y() const const
virtual QVariant data(const QModelIndex &index, int role) const const override
virtual QModelIndex index(int row, int column, const QModelIndex &parent) const const override
virtual QModelIndex mapToSource(const QModelIndex &proxyIndex) const const override
virtual QModelIndex parent(const QModelIndex &child) const const override
virtual int rowCount(const QModelIndex &parent) const const override
void setSortRole(int role)
bool isEmpty() const const
int localeAwareCompare(QStringView s1, QStringView s2)
UniqueConnection
DisplayRole
QTextStream & left(QTextStream &stream)
QTextStream & right(QTextStream &stream)
QFuture< ArgsType< Signal > > connect(Sender *sender, Signal signal)
bool isNull() const const
bool toBool() const const
QDateTime toDateTime() const const
QString toString() const const
QStringList toStringList() const const
QUrl toUrl() const const
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Mar 28 2025 11:53:53 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.