Add kwin full source tree, greeter login, zsh, pcid service, and build system improvements
This commit is contained in:
@@ -0,0 +1,523 @@
|
||||
/*
|
||||
KWin - the KDE window manager
|
||||
This file is part of the KDE project.
|
||||
|
||||
SPDX-FileCopyrightText: 2012 Martin Gräßlin <mgraesslin@kde.org>
|
||||
|
||||
SPDX-License-Identifier: GPL-2.0-or-later
|
||||
*/
|
||||
|
||||
// own
|
||||
#include "dbusinterface.h"
|
||||
#include "compositingadaptor.h"
|
||||
#include "pluginsadaptor.h"
|
||||
#include "virtualdesktopmanageradaptor.h"
|
||||
|
||||
// kwin
|
||||
#include "compositor.h"
|
||||
#include "core/output.h"
|
||||
#include "core/renderbackend.h"
|
||||
#include "debug_console.h"
|
||||
#include "kwinadaptor.h"
|
||||
#include "main.h"
|
||||
#include "placement.h"
|
||||
#include "pluginmanager.h"
|
||||
#include "virtualdesktops.h"
|
||||
#include "window.h"
|
||||
#include "workspace.h"
|
||||
#if KWIN_BUILD_ACTIVITIES
|
||||
#include "activities.h"
|
||||
#endif
|
||||
|
||||
// Qt
|
||||
#include <QDBusConnection>
|
||||
#include <QOpenGLContext>
|
||||
|
||||
namespace KWin
|
||||
{
|
||||
|
||||
DBusInterface::DBusInterface(QObject *parent)
|
||||
: QObject(parent)
|
||||
, m_serviceName(QStringLiteral("org.kde.KWin"))
|
||||
{
|
||||
(void)new KWinAdaptor(this);
|
||||
|
||||
QDBusConnection dbus = QDBusConnection::sessionBus();
|
||||
dbus.registerObject(QStringLiteral("/KWin"), this);
|
||||
dbus.registerService(m_serviceName);
|
||||
dbus.connect(QString(), QStringLiteral("/KWin"), QStringLiteral("org.kde.KWin"), QStringLiteral("reloadConfig"),
|
||||
Workspace::self(), SLOT(slotReloadConfig()));
|
||||
|
||||
connect(Workspace::self(), &Workspace::showingDesktopChanged, this, &DBusInterface::onShowingDesktopChanged);
|
||||
}
|
||||
|
||||
DBusInterface::~DBusInterface()
|
||||
{
|
||||
QDBusConnection::sessionBus().unregisterService(m_serviceName);
|
||||
}
|
||||
|
||||
bool DBusInterface::showingDesktop() const
|
||||
{
|
||||
return workspace()->showingDesktop();
|
||||
}
|
||||
|
||||
void DBusInterface::reconfigure()
|
||||
{
|
||||
Workspace::self()->reconfigure();
|
||||
}
|
||||
|
||||
void DBusInterface::killWindow()
|
||||
{
|
||||
Workspace::self()->slotKillWindow();
|
||||
}
|
||||
|
||||
void DBusInterface::cascadeDesktop()
|
||||
{
|
||||
workspace()->placement()->cascadeDesktop();
|
||||
}
|
||||
|
||||
void DBusInterface::unclutterDesktop()
|
||||
{
|
||||
workspace()->placement()->unclutterDesktop();
|
||||
}
|
||||
|
||||
QString DBusInterface::supportInformation()
|
||||
{
|
||||
return Workspace::self()->supportInformation();
|
||||
}
|
||||
|
||||
QString DBusInterface::activeOutputName()
|
||||
{
|
||||
return Workspace::self()->activeOutput()->name();
|
||||
}
|
||||
|
||||
bool DBusInterface::startActivity(const QString &in0)
|
||||
{
|
||||
#if KWIN_BUILD_ACTIVITIES
|
||||
if (!Workspace::self()->activities()) {
|
||||
return false;
|
||||
}
|
||||
return Workspace::self()->activities()->start(in0);
|
||||
#else
|
||||
return false;
|
||||
#endif
|
||||
}
|
||||
|
||||
bool DBusInterface::stopActivity(const QString &in0)
|
||||
{
|
||||
#if KWIN_BUILD_ACTIVITIES
|
||||
if (!Workspace::self()->activities()) {
|
||||
return false;
|
||||
}
|
||||
return Workspace::self()->activities()->stop(in0);
|
||||
#else
|
||||
return false;
|
||||
#endif
|
||||
}
|
||||
|
||||
int DBusInterface::currentDesktop()
|
||||
{
|
||||
return VirtualDesktopManager::self()->current();
|
||||
}
|
||||
|
||||
bool DBusInterface::setCurrentDesktop(int desktop)
|
||||
{
|
||||
return VirtualDesktopManager::self()->setCurrent(desktop);
|
||||
}
|
||||
|
||||
void DBusInterface::nextDesktop()
|
||||
{
|
||||
VirtualDesktopManager::self()->moveTo(VirtualDesktopManager::Direction::Next);
|
||||
}
|
||||
|
||||
void DBusInterface::previousDesktop()
|
||||
{
|
||||
VirtualDesktopManager::self()->moveTo(VirtualDesktopManager::Direction::Previous);
|
||||
}
|
||||
|
||||
void DBusInterface::showDebugConsole()
|
||||
{
|
||||
DebugConsole *console = new DebugConsole;
|
||||
console->show();
|
||||
}
|
||||
|
||||
void DBusInterface::replace()
|
||||
{
|
||||
QCoreApplication::exit(133);
|
||||
}
|
||||
|
||||
namespace
|
||||
{
|
||||
QVariantMap clientToVariantMap(const Window *c)
|
||||
{
|
||||
return
|
||||
{
|
||||
{QStringLiteral("resourceClass"), c->resourceClass()},
|
||||
{QStringLiteral("resourceName"), c->resourceName()},
|
||||
{QStringLiteral("desktopFile"), c->desktopFileName()},
|
||||
{QStringLiteral("role"), c->windowRole()},
|
||||
{QStringLiteral("caption"), c->captionNormal()},
|
||||
{QStringLiteral("clientMachine"), c->wmClientMachine(true)},
|
||||
{QStringLiteral("localhost"), c->isLocalhost()},
|
||||
{QStringLiteral("type"), int(c->windowType())},
|
||||
{QStringLiteral("x"), c->x()},
|
||||
{QStringLiteral("y"), c->y()},
|
||||
{QStringLiteral("width"), c->width()},
|
||||
{QStringLiteral("height"), c->height()},
|
||||
{QStringLiteral("desktops"), c->desktopIds()},
|
||||
{QStringLiteral("minimized"), c->isMinimized()},
|
||||
{QStringLiteral("shaded"), c->isShade()},
|
||||
{QStringLiteral("fullscreen"), c->isFullScreen()},
|
||||
{QStringLiteral("keepAbove"), c->keepAbove()},
|
||||
{QStringLiteral("keepBelow"), c->keepBelow()},
|
||||
{QStringLiteral("noBorder"), c->noBorder()},
|
||||
{QStringLiteral("skipTaskbar"), c->skipTaskbar()},
|
||||
{QStringLiteral("skipPager"), c->skipPager()},
|
||||
{QStringLiteral("skipSwitcher"), c->skipSwitcher()},
|
||||
{QStringLiteral("maximizeHorizontal"), c->maximizeMode() & MaximizeHorizontal},
|
||||
{QStringLiteral("maximizeVertical"), c->maximizeMode() & MaximizeVertical},
|
||||
{QStringLiteral("uuid"), c->internalId().toString()},
|
||||
#if KWIN_BUILD_ACTIVITIES
|
||||
{QStringLiteral("activities"), c->activities()},
|
||||
#endif
|
||||
{QStringLiteral("layer"), c->layer()},
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
QVariantMap DBusInterface::queryWindowInfo()
|
||||
{
|
||||
m_replyQueryWindowInfo = message();
|
||||
setDelayedReply(true);
|
||||
kwinApp()->startInteractiveWindowSelection(
|
||||
[this](Window *t) {
|
||||
if (!t) {
|
||||
QDBusConnection::sessionBus().send(m_replyQueryWindowInfo.createErrorReply(
|
||||
QStringLiteral("org.kde.KWin.Error.UserCancel"),
|
||||
QStringLiteral("User cancelled the query")));
|
||||
return;
|
||||
}
|
||||
if (t->isClient()) {
|
||||
QDBusConnection::sessionBus().send(m_replyQueryWindowInfo.createReply(clientToVariantMap(t)));
|
||||
} else {
|
||||
QDBusConnection::sessionBus().send(m_replyQueryWindowInfo.createErrorReply(
|
||||
QStringLiteral("org.kde.KWin.Error.InvalidWindow"),
|
||||
QStringLiteral("Tried to query information about an unmanaged window")));
|
||||
}
|
||||
});
|
||||
return QVariantMap{};
|
||||
}
|
||||
|
||||
QVariantMap DBusInterface::getWindowInfo(const QString &uuid)
|
||||
{
|
||||
const auto window = workspace()->findWindow(QUuid::fromString(uuid));
|
||||
if (window) {
|
||||
return clientToVariantMap(window);
|
||||
} else {
|
||||
return {};
|
||||
}
|
||||
}
|
||||
|
||||
void DBusInterface::showDesktop(bool show)
|
||||
{
|
||||
workspace()->setShowingDesktop(show, true);
|
||||
|
||||
auto m = message();
|
||||
if (m.service().isEmpty()) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Keep track of whatever D-Bus client asked to show the desktop. If
|
||||
// they disappear from the bus, cancel the show desktop state so we do
|
||||
// not end up in a state where we are stuck showing the desktop.
|
||||
static QPointer<QDBusServiceWatcher> watcher;
|
||||
|
||||
if (show) {
|
||||
if (watcher) {
|
||||
// If we get a second call to `showDesktop(true)`, drop the previous
|
||||
// watcher and watch the new client. That way, we simply always
|
||||
// track the last state.
|
||||
watcher->deleteLater();
|
||||
}
|
||||
|
||||
watcher = new QDBusServiceWatcher(m.service(), QDBusConnection::sessionBus(), QDBusServiceWatcher::WatchForUnregistration, this);
|
||||
connect(watcher, &QDBusServiceWatcher::serviceUnregistered, []() {
|
||||
workspace()->setShowingDesktop(false, true);
|
||||
watcher->deleteLater();
|
||||
});
|
||||
} else if (watcher) {
|
||||
// Someone cancelled showing the desktop, so there's no more need to
|
||||
// watch to cancel the show desktop state.
|
||||
watcher->deleteLater();
|
||||
}
|
||||
}
|
||||
|
||||
void DBusInterface::onShowingDesktopChanged(bool show, bool /*animated*/)
|
||||
{
|
||||
Q_EMIT showingDesktopChanged(show);
|
||||
}
|
||||
|
||||
CompositorDBusInterface::CompositorDBusInterface(Compositor *parent)
|
||||
: QObject(parent)
|
||||
, m_compositor(parent)
|
||||
{
|
||||
connect(m_compositor, &Compositor::compositingToggled, this, &CompositorDBusInterface::compositingToggled);
|
||||
new CompositingAdaptor(this);
|
||||
QDBusConnection dbus = QDBusConnection::sessionBus();
|
||||
dbus.registerObject(QStringLiteral("/Compositor"), this);
|
||||
dbus.connect(QString(), QStringLiteral("/Compositor"), QStringLiteral("org.kde.kwin.Compositing"),
|
||||
QStringLiteral("reinit"), this, SLOT(reinitialize()));
|
||||
}
|
||||
|
||||
QString CompositorDBusInterface::compositingType() const
|
||||
{
|
||||
if (!m_compositor->compositing()) {
|
||||
return QStringLiteral("none");
|
||||
}
|
||||
switch (m_compositor->backend()->compositingType()) {
|
||||
case OpenGLCompositing:
|
||||
if (QOpenGLContext::openGLModuleType() == QOpenGLContext::LibGLES) {
|
||||
return QStringLiteral("gles");
|
||||
} else {
|
||||
return QStringLiteral("gl2");
|
||||
}
|
||||
case QPainterCompositing:
|
||||
return QStringLiteral("qpainter");
|
||||
case NoCompositing:
|
||||
default:
|
||||
return QStringLiteral("none");
|
||||
}
|
||||
}
|
||||
|
||||
bool CompositorDBusInterface::isActive() const
|
||||
{
|
||||
return m_compositor->isActive();
|
||||
}
|
||||
|
||||
bool CompositorDBusInterface::isCompositingPossible() const
|
||||
{
|
||||
return m_compositor->compositingPossible();
|
||||
}
|
||||
|
||||
QString CompositorDBusInterface::compositingNotPossibleReason() const
|
||||
{
|
||||
return m_compositor->compositingNotPossibleReason();
|
||||
}
|
||||
|
||||
bool CompositorDBusInterface::isOpenGLBroken() const
|
||||
{
|
||||
return m_compositor->openGLCompositingIsBroken();
|
||||
}
|
||||
|
||||
bool CompositorDBusInterface::platformRequiresCompositing() const
|
||||
{
|
||||
return kwinApp()->operationMode() != Application::OperationModeX11; // TODO: Remove this property?
|
||||
}
|
||||
|
||||
void CompositorDBusInterface::reinitialize()
|
||||
{
|
||||
m_compositor->reinitialize();
|
||||
}
|
||||
|
||||
QStringList CompositorDBusInterface::supportedOpenGLPlatformInterfaces() const
|
||||
{
|
||||
QStringList interfaces;
|
||||
bool supportsGlx = false;
|
||||
#if HAVE_GLX
|
||||
supportsGlx = (kwinApp()->operationMode() == Application::OperationModeX11);
|
||||
#endif
|
||||
if (QOpenGLContext::openGLModuleType() == QOpenGLContext::LibGLES) {
|
||||
supportsGlx = false;
|
||||
}
|
||||
if (supportsGlx) {
|
||||
interfaces << QStringLiteral("glx");
|
||||
}
|
||||
interfaces << QStringLiteral("egl");
|
||||
return interfaces;
|
||||
}
|
||||
|
||||
VirtualDesktopManagerDBusInterface::VirtualDesktopManagerDBusInterface(VirtualDesktopManager *parent)
|
||||
: QObject(parent)
|
||||
, m_manager(parent)
|
||||
{
|
||||
qDBusRegisterMetaType<KWin::DBusDesktopDataStruct>();
|
||||
qDBusRegisterMetaType<KWin::DBusDesktopDataVector>();
|
||||
|
||||
new VirtualDesktopManagerAdaptor(this);
|
||||
QDBusConnection::sessionBus().registerObject(QStringLiteral("/VirtualDesktopManager"),
|
||||
QStringLiteral("org.kde.KWin.VirtualDesktopManager"),
|
||||
this);
|
||||
|
||||
connect(m_manager, &VirtualDesktopManager::currentChanged, this, [this]() {
|
||||
Q_EMIT currentChanged(m_manager->currentDesktop()->id());
|
||||
});
|
||||
|
||||
connect(m_manager, &VirtualDesktopManager::countChanged, this, [this](uint previousCount, uint newCount) {
|
||||
Q_EMIT countChanged(newCount);
|
||||
Q_EMIT desktopsChanged(desktops());
|
||||
});
|
||||
|
||||
connect(m_manager, &VirtualDesktopManager::navigationWrappingAroundChanged, this, [this]() {
|
||||
Q_EMIT navigationWrappingAroundChanged(isNavigationWrappingAround());
|
||||
});
|
||||
|
||||
connect(m_manager, &VirtualDesktopManager::rowsChanged, this, &VirtualDesktopManagerDBusInterface::rowsChanged);
|
||||
|
||||
const QList<VirtualDesktop *> allDesks = m_manager->desktops();
|
||||
for (auto *vd : allDesks) {
|
||||
connect(vd, &VirtualDesktop::x11DesktopNumberChanged, this, [this, vd]() {
|
||||
DBusDesktopDataStruct data{.position = vd->x11DesktopNumber() - 1, .id = vd->id(), .name = vd->name()};
|
||||
Q_EMIT desktopDataChanged(vd->id(), data);
|
||||
Q_EMIT desktopsChanged(desktops());
|
||||
});
|
||||
connect(vd, &VirtualDesktop::nameChanged, this, [this, vd]() {
|
||||
DBusDesktopDataStruct data{.position = vd->x11DesktopNumber() - 1, .id = vd->id(), .name = vd->name()};
|
||||
Q_EMIT desktopDataChanged(vd->id(), data);
|
||||
Q_EMIT desktopsChanged(desktops());
|
||||
});
|
||||
}
|
||||
connect(m_manager, &VirtualDesktopManager::desktopAdded, this, [this](VirtualDesktop *vd) {
|
||||
connect(vd, &VirtualDesktop::x11DesktopNumberChanged, this, [this, vd]() {
|
||||
DBusDesktopDataStruct data{.position = vd->x11DesktopNumber() - 1, .id = vd->id(), .name = vd->name()};
|
||||
Q_EMIT desktopDataChanged(vd->id(), data);
|
||||
Q_EMIT desktopsChanged(desktops());
|
||||
});
|
||||
connect(vd, &VirtualDesktop::nameChanged, this, [this, vd]() {
|
||||
DBusDesktopDataStruct data{.position = vd->x11DesktopNumber() - 1, .id = vd->id(), .name = vd->name()};
|
||||
Q_EMIT desktopDataChanged(vd->id(), data);
|
||||
Q_EMIT desktopsChanged(desktops());
|
||||
});
|
||||
DBusDesktopDataStruct data{.position = vd->x11DesktopNumber() - 1, .id = vd->id(), .name = vd->name()};
|
||||
Q_EMIT desktopCreated(vd->id(), data);
|
||||
Q_EMIT desktopsChanged(desktops());
|
||||
});
|
||||
connect(m_manager, &VirtualDesktopManager::desktopRemoved, this, [this](VirtualDesktop *vd) {
|
||||
Q_EMIT desktopRemoved(vd->id());
|
||||
Q_EMIT desktopsChanged(desktops());
|
||||
});
|
||||
}
|
||||
|
||||
uint VirtualDesktopManagerDBusInterface::count() const
|
||||
{
|
||||
return m_manager->count();
|
||||
}
|
||||
|
||||
void VirtualDesktopManagerDBusInterface::setRows(uint rows)
|
||||
{
|
||||
if (static_cast<uint>(m_manager->grid().height()) == rows) {
|
||||
return;
|
||||
}
|
||||
|
||||
m_manager->setRows(rows);
|
||||
m_manager->save();
|
||||
}
|
||||
|
||||
uint VirtualDesktopManagerDBusInterface::rows() const
|
||||
{
|
||||
return m_manager->rows();
|
||||
}
|
||||
|
||||
void VirtualDesktopManagerDBusInterface::setCurrent(const QString &id)
|
||||
{
|
||||
if (m_manager->currentDesktop()->id() == id) {
|
||||
return;
|
||||
}
|
||||
|
||||
auto *vd = m_manager->desktopForId(id);
|
||||
if (vd) {
|
||||
m_manager->setCurrent(vd);
|
||||
}
|
||||
}
|
||||
|
||||
QString VirtualDesktopManagerDBusInterface::current() const
|
||||
{
|
||||
return m_manager->currentDesktop()->id();
|
||||
}
|
||||
|
||||
void VirtualDesktopManagerDBusInterface::setNavigationWrappingAround(bool wraps)
|
||||
{
|
||||
if (m_manager->isNavigationWrappingAround() == wraps) {
|
||||
return;
|
||||
}
|
||||
|
||||
m_manager->setNavigationWrappingAround(wraps);
|
||||
}
|
||||
|
||||
bool VirtualDesktopManagerDBusInterface::isNavigationWrappingAround() const
|
||||
{
|
||||
return m_manager->isNavigationWrappingAround();
|
||||
}
|
||||
|
||||
DBusDesktopDataVector VirtualDesktopManagerDBusInterface::desktops() const
|
||||
{
|
||||
const auto desks = m_manager->desktops();
|
||||
DBusDesktopDataVector desktopVect;
|
||||
desktopVect.reserve(m_manager->count());
|
||||
|
||||
std::transform(desks.constBegin(), desks.constEnd(),
|
||||
std::back_inserter(desktopVect),
|
||||
[](const VirtualDesktop *vd) {
|
||||
return DBusDesktopDataStruct{.position = vd->x11DesktopNumber() - 1, .id = vd->id(), .name = vd->name()};
|
||||
});
|
||||
|
||||
return desktopVect;
|
||||
}
|
||||
|
||||
void VirtualDesktopManagerDBusInterface::createDesktop(uint position, const QString &name)
|
||||
{
|
||||
m_manager->createVirtualDesktop(position, name);
|
||||
}
|
||||
|
||||
void VirtualDesktopManagerDBusInterface::setDesktopName(const QString &id, const QString &name)
|
||||
{
|
||||
VirtualDesktop *vd = m_manager->desktopForId(id);
|
||||
if (!vd) {
|
||||
return;
|
||||
}
|
||||
if (vd->name() == name) {
|
||||
return;
|
||||
}
|
||||
|
||||
vd->setName(name);
|
||||
m_manager->save();
|
||||
}
|
||||
|
||||
void VirtualDesktopManagerDBusInterface::removeDesktop(const QString &id)
|
||||
{
|
||||
m_manager->removeVirtualDesktop(id);
|
||||
}
|
||||
|
||||
PluginManagerDBusInterface::PluginManagerDBusInterface(PluginManager *manager)
|
||||
: QObject(manager)
|
||||
, m_manager(manager)
|
||||
{
|
||||
new PluginsAdaptor(this);
|
||||
|
||||
QDBusConnection::sessionBus().registerObject(QStringLiteral("/Plugins"),
|
||||
QStringLiteral("org.kde.KWin.Plugins"),
|
||||
this);
|
||||
}
|
||||
|
||||
QStringList PluginManagerDBusInterface::loadedPlugins() const
|
||||
{
|
||||
return m_manager->loadedPlugins();
|
||||
}
|
||||
|
||||
QStringList PluginManagerDBusInterface::availablePlugins() const
|
||||
{
|
||||
return m_manager->availablePlugins();
|
||||
}
|
||||
|
||||
bool PluginManagerDBusInterface::LoadPlugin(const QString &name)
|
||||
{
|
||||
return m_manager->loadPlugin(name);
|
||||
}
|
||||
|
||||
void PluginManagerDBusInterface::UnloadPlugin(const QString &name)
|
||||
{
|
||||
m_manager->unloadPlugin(name);
|
||||
}
|
||||
|
||||
} // namespace
|
||||
|
||||
#include "moc_dbusinterface.cpp"
|
||||
Reference in New Issue
Block a user