2023-09-28 14:37:38 +03:00
|
|
|
/* This file is part of the KDE project
|
|
|
|
Copyright (C) 2023 Ivailo Monev <xakepa10@gmail.com>
|
|
|
|
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
|
|
modify it under the terms of the GNU Library General Public
|
|
|
|
License version 2, as published by the Free Software Foundation.
|
|
|
|
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
Library General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU Library General Public License
|
|
|
|
along with this library; see the file COPYING.LIB. If not, write to
|
|
|
|
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
|
|
Boston, MA 02110-1301, USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "ktaskmanager.h"
|
|
|
|
|
|
|
|
#include <QMutex>
|
2023-09-29 04:18:39 +03:00
|
|
|
#include <QX11Info>
|
2023-09-28 14:37:38 +03:00
|
|
|
#include <KGlobal>
|
|
|
|
#include <KWindowSystem>
|
|
|
|
#include <KIconLoader>
|
2023-09-29 04:18:39 +03:00
|
|
|
#include <KIcon>
|
|
|
|
#include <KLocale>
|
2023-09-28 14:37:38 +03:00
|
|
|
#include <KDebug>
|
2023-09-29 04:18:39 +03:00
|
|
|
#include <netwm.h>
|
2023-09-28 14:37:38 +03:00
|
|
|
|
|
|
|
static const int s_nodesktop = -1;
|
|
|
|
|
|
|
|
static QByteArray kGetTaskID()
|
|
|
|
{
|
|
|
|
return qRandomUuid();
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool kIsTaskWindow(const WId window)
|
|
|
|
{
|
|
|
|
const KWindowInfo kwindowinfo = KWindowSystem::windowInfo(window, NET::WMWindowType | NET::WMState);
|
|
|
|
NET::WindowType kwindowtype = kwindowinfo.windowType(
|
|
|
|
NET::NormalMask | NET::DesktopMask | NET::DockMask |
|
|
|
|
NET::ToolbarMask | NET::MenuMask | NET::DialogMask |
|
|
|
|
NET::UtilityMask | NET::SplashMask
|
|
|
|
);
|
|
|
|
if (kwindowtype != NET::Normal && kwindowtype != NET::Unknown &&
|
|
|
|
kwindowtype != NET::Dialog && kwindowtype != NET::Utility) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (kwindowinfo.state() & NET::SkipTaskbar) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void kUpdateTask(KTaskManager::Task &task, const bool force = false)
|
|
|
|
{
|
|
|
|
const KWindowInfo kwindowinfo = KWindowSystem::windowInfo(
|
|
|
|
task.window,
|
|
|
|
NET::WMVisibleName | NET::WMDesktop
|
|
|
|
);
|
|
|
|
if (task.icon.isNull() || force) {
|
|
|
|
task.icon = KWindowSystem::icon(task.window);
|
|
|
|
}
|
|
|
|
if (task.name.isEmpty() || force) {
|
|
|
|
task.name = kwindowinfo.visibleName();
|
|
|
|
}
|
|
|
|
task.desktop = kwindowinfo.desktop();
|
|
|
|
}
|
|
|
|
|
2023-09-29 04:18:39 +03:00
|
|
|
class KTaskAction : public QAction
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
public:
|
|
|
|
enum KActionType {
|
|
|
|
ActionMinimize = 0,
|
|
|
|
ActionUnminimize = 1,
|
|
|
|
ActionMaximize = 2,
|
|
|
|
ActionUnmaximize = 3,
|
|
|
|
ActionClose = 4,
|
|
|
|
};
|
|
|
|
KTaskAction(const WId window, const KActionType type,
|
|
|
|
QObject *parent);
|
|
|
|
|
|
|
|
private Q_SLOTS:
|
|
|
|
void slotMinimize();
|
|
|
|
void slotUnminimize();
|
|
|
|
void slotMaximize();
|
|
|
|
void slotUnmaximize();
|
|
|
|
void slotClose();
|
|
|
|
|
|
|
|
private:
|
|
|
|
WId m_window;
|
|
|
|
};
|
|
|
|
|
|
|
|
KTaskAction::KTaskAction(const WId window, const KActionType type,
|
|
|
|
QObject *parent)
|
|
|
|
: QAction(parent),
|
|
|
|
m_window(window)
|
|
|
|
{
|
|
|
|
switch (type) {
|
2023-09-29 04:39:31 +03:00
|
|
|
// unlike the kwin decoration menu the minimize action is checked based on the window state
|
2023-09-29 04:18:39 +03:00
|
|
|
// just like maximize is
|
|
|
|
case KTaskAction::ActionMinimize: {
|
|
|
|
connect(this, SIGNAL(triggered()), this, SLOT(slotMinimize()));
|
|
|
|
setText(i18n("Mi&nimize"));
|
|
|
|
setCheckable(true);
|
|
|
|
setChecked(false);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case KTaskAction::ActionUnminimize: {
|
|
|
|
connect(this, SIGNAL(triggered()), this, SLOT(slotUnminimize()));
|
|
|
|
setText(i18n("Mi&nimize"));
|
|
|
|
setCheckable(true);
|
|
|
|
setChecked(true);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case KTaskAction::ActionMaximize: {
|
|
|
|
connect(this, SIGNAL(triggered()), this, SLOT(slotMaximize()));
|
|
|
|
setText(i18n("Ma&ximize"));
|
|
|
|
setCheckable(true);
|
|
|
|
setChecked(false);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case KTaskAction::ActionUnmaximize: {
|
|
|
|
connect(this, SIGNAL(triggered()), this, SLOT(slotUnmaximize()));
|
|
|
|
setText(i18n("Ma&ximize"));
|
|
|
|
setCheckable(true);
|
|
|
|
setChecked(true);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case KTaskAction::ActionClose: {
|
|
|
|
connect(this, SIGNAL(triggered()), this, SLOT(slotClose()));
|
|
|
|
setText(i18n("&Close"));
|
|
|
|
setIcon(KIcon("window-close"));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void KTaskAction::slotMinimize()
|
|
|
|
{
|
|
|
|
// using KWindowSystem for the animation
|
|
|
|
KWindowSystem::minimizeWindow(m_window);
|
|
|
|
}
|
|
|
|
|
|
|
|
void KTaskAction::slotUnminimize()
|
|
|
|
{
|
|
|
|
// see above
|
|
|
|
KWindowSystem::unminimizeWindow(m_window);
|
|
|
|
}
|
|
|
|
|
|
|
|
void KTaskAction::slotMaximize()
|
|
|
|
{
|
|
|
|
// this does not change the window visibility
|
|
|
|
KWindowSystem::setState(m_window, NET::Max);
|
|
|
|
}
|
|
|
|
|
|
|
|
void KTaskAction::slotUnmaximize()
|
|
|
|
{
|
|
|
|
KWindowSystem::clearState(m_window, NET::Max);
|
|
|
|
}
|
|
|
|
|
|
|
|
void KTaskAction::slotClose()
|
|
|
|
{
|
|
|
|
NETRootInfo netrootinfo(QX11Info::display(), NET::CloseWindow);
|
|
|
|
netrootinfo.closeWindowRequest(m_window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-09-28 14:37:38 +03:00
|
|
|
class KTaskManagerPrivate : QObject
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
public:
|
|
|
|
KTaskManagerPrivate(QObject *parent);
|
|
|
|
|
|
|
|
QList<KTaskManager::Task> tasks;
|
|
|
|
|
|
|
|
private Q_SLOTS:
|
|
|
|
void slotNewWindow(const WId window);
|
|
|
|
void slotChangedWindow(const WId window);
|
|
|
|
void slotRemovedWindow(const WId window);
|
|
|
|
void slotActiveWindow(const WId window);
|
|
|
|
|
|
|
|
private:
|
|
|
|
QMutex m_mutex;
|
|
|
|
};
|
|
|
|
|
|
|
|
KTaskManagerPrivate::KTaskManagerPrivate(QObject *parent)
|
2023-09-29 14:59:24 +03:00
|
|
|
: QObject(parent)
|
2023-09-28 14:37:38 +03:00
|
|
|
{
|
|
|
|
foreach (const WId window, KWindowSystem::stackingOrder()) {
|
|
|
|
if (!kIsTaskWindow(window)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
kDebug() << "adding task for" << window;
|
|
|
|
KTaskManager::Task task;
|
|
|
|
task.id = kGetTaskID();
|
|
|
|
task.window = window;
|
|
|
|
task.desktop = s_nodesktop;
|
|
|
|
kUpdateTask(task);
|
|
|
|
tasks.append(task);
|
|
|
|
KTaskManager* ktaskmanager = qobject_cast<KTaskManager*>(parent);
|
|
|
|
emit ktaskmanager->taskAdded(task);
|
|
|
|
}
|
|
|
|
connect(
|
|
|
|
KWindowSystem::self(), SIGNAL(windowAdded(WId)),
|
|
|
|
this, SLOT(slotNewWindow(WId))
|
|
|
|
);
|
|
|
|
connect(
|
|
|
|
KWindowSystem::self(), SIGNAL(windowChanged(WId)),
|
|
|
|
this, SLOT(slotChangedWindow(WId))
|
|
|
|
);
|
|
|
|
connect(
|
|
|
|
KWindowSystem::self(), SIGNAL(windowRemoved(WId)),
|
|
|
|
this, SLOT(slotRemovedWindow(WId))
|
|
|
|
);
|
|
|
|
connect(
|
|
|
|
KWindowSystem::self(), SIGNAL(activeWindowChanged(WId)),
|
|
|
|
this, SLOT(slotActiveWindow(WId))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
void KTaskManagerPrivate::slotNewWindow(const WId window)
|
|
|
|
{
|
|
|
|
if (!kIsTaskWindow(window)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
QMutexLocker locker(&m_mutex);
|
|
|
|
kDebug() << "new window task for" << window;
|
|
|
|
KTaskManager::Task task;
|
|
|
|
task.id = kGetTaskID();
|
|
|
|
task.window = window;
|
|
|
|
task.desktop = s_nodesktop;
|
|
|
|
kUpdateTask(task);
|
|
|
|
tasks.append(task);
|
2023-09-29 14:59:24 +03:00
|
|
|
KTaskManager* ktaskmanager = qobject_cast<KTaskManager*>(parent());
|
2023-09-28 14:37:38 +03:00
|
|
|
emit ktaskmanager->taskAdded(task);
|
|
|
|
}
|
|
|
|
|
|
|
|
void KTaskManagerPrivate::slotChangedWindow(const WId window)
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&m_mutex);
|
|
|
|
QMutableListIterator<KTaskManager::Task> iter(tasks);
|
|
|
|
while (iter.hasNext()) {
|
|
|
|
KTaskManager::Task &task = iter.next();
|
|
|
|
if (task.window == window) {
|
|
|
|
kDebug() << "changed window task for" << window;
|
|
|
|
kUpdateTask(task, true);
|
|
|
|
KTaskManager* ktaskmanager = qobject_cast<KTaskManager*>(parent());
|
|
|
|
emit ktaskmanager->taskChanged(task);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void KTaskManagerPrivate::slotRemovedWindow(const WId window)
|
|
|
|
{
|
|
|
|
QMutexLocker locker(&m_mutex);
|
|
|
|
QMutableListIterator<KTaskManager::Task> iter(tasks);
|
|
|
|
while (iter.hasNext()) {
|
|
|
|
const KTaskManager::Task task = iter.next();
|
|
|
|
if (task.window == window) {
|
|
|
|
kDebug() << "removed window task for" << window;
|
|
|
|
iter.remove();
|
|
|
|
KTaskManager* ktaskmanager = qobject_cast<KTaskManager*>(parent());
|
|
|
|
emit ktaskmanager->taskRemoved(task);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void KTaskManagerPrivate::slotActiveWindow(const WId window)
|
|
|
|
{
|
|
|
|
// the active state is not tracked so it is unknown which window became inactive
|
|
|
|
KTaskManager* ktaskmanager = qobject_cast<KTaskManager*>(parent());
|
|
|
|
QMutexLocker locker(&m_mutex);
|
|
|
|
QMutableListIterator<KTaskManager::Task> iter(tasks);
|
|
|
|
while (iter.hasNext()) {
|
|
|
|
KTaskManager::Task &task = iter.next();
|
|
|
|
kUpdateTask(task);
|
|
|
|
emit ktaskmanager->taskChanged(task);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
K_GLOBAL_STATIC(KTaskManager, globalktaskmanager)
|
|
|
|
|
|
|
|
KTaskManager::KTaskManager(QObject *parent)
|
|
|
|
: QObject(parent),
|
|
|
|
d(nullptr)
|
|
|
|
{
|
|
|
|
d = new KTaskManagerPrivate(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
KTaskManager::~KTaskManager()
|
|
|
|
{
|
|
|
|
delete d;
|
|
|
|
}
|
|
|
|
|
|
|
|
QList<KTaskManager::Task> KTaskManager::tasks() const
|
|
|
|
{
|
|
|
|
return d->tasks;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool KTaskManager::isActive(const KTaskManager::Task &task) const
|
|
|
|
{
|
|
|
|
// TODO: transients
|
|
|
|
return (task.window == KWindowSystem::activeWindow());
|
|
|
|
}
|
|
|
|
|
|
|
|
bool KTaskManager::demandsAttention(const KTaskManager::Task &task) const
|
|
|
|
{
|
|
|
|
// TODO: including transients
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-09-29 14:59:24 +03:00
|
|
|
void KTaskManager::activateRaiseOrIconify(const KTaskManager::Task &task)
|
2023-09-28 14:37:38 +03:00
|
|
|
{
|
|
|
|
if (isActive(task)) {
|
|
|
|
KWindowSystem::minimizeWindow(task.window);
|
2023-09-29 14:59:24 +03:00
|
|
|
return;
|
2023-09-28 14:37:38 +03:00
|
|
|
}
|
|
|
|
KWindowSystem::activateWindow(task.window);
|
|
|
|
KWindowSystem::raiseWindow(task.window);
|
|
|
|
}
|
|
|
|
|
|
|
|
KTaskManager* KTaskManager::self()
|
|
|
|
{
|
|
|
|
return globalktaskmanager;
|
|
|
|
}
|
|
|
|
|
2023-09-29 04:18:39 +03:00
|
|
|
QMenu* KTaskManager::menuForWindow(WId windowid, QWidget *parent)
|
2023-09-28 14:37:38 +03:00
|
|
|
{
|
2023-09-29 04:18:39 +03:00
|
|
|
foreach (const KTaskManager::Task &task, KTaskManager::self()->tasks()) {
|
|
|
|
if (task.window == windowid) {
|
|
|
|
return menuForTask(task, parent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
QMenu* KTaskManager::menuForTask(const KTaskManager::Task &task, QWidget *parent)
|
|
|
|
{
|
|
|
|
QMenu* result = new QMenu(parent);
|
|
|
|
result->setTitle(task.name);
|
|
|
|
result->setIcon(KIcon(task.icon));
|
|
|
|
const KWindowInfo kwindowinfo = KWindowSystem::windowInfo(
|
|
|
|
task.window,
|
|
|
|
NET::WMState | NET::XAWMState,
|
|
|
|
NET::WM2AllowedActions
|
|
|
|
);
|
|
|
|
KTaskAction* ktaskaction = nullptr;
|
|
|
|
if (kwindowinfo.actionSupported(NET::ActionMinimize)) {
|
|
|
|
ktaskaction = new KTaskAction(
|
|
|
|
task.window,
|
|
|
|
kwindowinfo.isMinimized() ? KTaskAction::ActionUnminimize : KTaskAction::ActionMinimize,
|
|
|
|
result
|
|
|
|
);
|
|
|
|
result->addAction(ktaskaction);
|
|
|
|
}
|
|
|
|
if (kwindowinfo.actionSupported(NET::ActionMax)) {
|
|
|
|
ktaskaction = new KTaskAction(
|
|
|
|
task.window,
|
|
|
|
kwindowinfo.hasState(NET::Max) ? KTaskAction::ActionUnmaximize : KTaskAction::ActionMaximize,
|
|
|
|
result
|
|
|
|
);
|
|
|
|
result->addAction(ktaskaction);
|
|
|
|
}
|
|
|
|
if (kwindowinfo.actionSupported(NET::ActionClose)) {
|
|
|
|
result->addSeparator();
|
|
|
|
ktaskaction = new KTaskAction(
|
|
|
|
task.window,
|
|
|
|
KTaskAction::ActionClose, result
|
|
|
|
);
|
|
|
|
result->addAction(ktaskaction);
|
|
|
|
}
|
|
|
|
return result;
|
2023-09-28 14:37:38 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#include "ktaskmanager.moc"
|