kwin/plugins/qpa/integration.cpp
Vlad Zahorodnii fb0bcff1c8 plugins/qpa: Improve handling of output addition and removal
With this change, the QPA will try to avoid re-creating all platform
screens if a single output has been connected or disconnected.
2020-12-09 17:44:00 +00:00

180 lines
4.8 KiB
C++

/*
KWin - the KDE window manager
This file is part of the KDE project.
SPDX-FileCopyrightText: 2015 Martin Gräßlin <mgraesslin@kde.org>
SPDX-FileCopyrightText: 2019 Vlad Zahorodnii <vlad.zahorodnii@kde.org>
SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "integration.h"
#include "abstract_output.h"
#include "backingstore.h"
#include "eglplatformcontext.h"
#include "logging.h"
#include "offscreensurface.h"
#include "screen.h"
#include "window.h"
#include "../../main.h"
#include "../../platform.h"
#include "../../screens.h"
#include <QCoreApplication>
#include <QtConcurrentRun>
#include <qpa/qplatformwindow.h>
#include <qpa/qwindowsysteminterface.h>
#include <QtEventDispatcherSupport/private/qunixeventdispatcher_qpa_p.h>
#include <QtFontDatabaseSupport/private/qgenericunixfontdatabase_p.h>
#include <QtThemeSupport/private/qgenericunixthemes_p.h>
namespace KWin
{
namespace QPA
{
Integration::Integration()
: QObject()
, QPlatformIntegration()
, m_fontDb(new QGenericUnixFontDatabase())
{
}
Integration::~Integration()
{
for (QPlatformScreen *platformScreen : m_screens) {
QWindowSystemInterface::handleScreenRemoved(platformScreen);
}
if (m_dummyScreen) {
QWindowSystemInterface::handleScreenRemoved(m_dummyScreen);
}
}
bool Integration::hasCapability(Capability cap) const
{
switch (cap) {
case ThreadedPixmaps:
return true;
case OpenGL:
return true;
case ThreadedOpenGL:
return false;
case BufferQueueingOpenGL:
return false;
case MultipleWindows:
case NonFullScreenWindows:
return true;
case RasterGLSurface:
return false;
default:
return QPlatformIntegration::hasCapability(cap);
}
}
void Integration::initialize()
{
// The QPA is initialized before the platform plugin is loaded.
connect(kwinApp(), &Application::platformCreated, this, &Integration::handlePlatformCreated);
QPlatformIntegration::initialize();
m_dummyScreen = new PlaceholderScreen();
QWindowSystemInterface::handleScreenAdded(m_dummyScreen);
}
QAbstractEventDispatcher *Integration::createEventDispatcher() const
{
return new QUnixEventDispatcherQPA;
}
QPlatformBackingStore *Integration::createPlatformBackingStore(QWindow *window) const
{
return new BackingStore(window);
}
QPlatformWindow *Integration::createPlatformWindow(QWindow *window) const
{
return new Window(window);
}
QPlatformOffscreenSurface *Integration::createPlatformOffscreenSurface(QOffscreenSurface *surface) const
{
return new OffscreenSurface(surface);
}
QPlatformFontDatabase *Integration::fontDatabase() const
{
return m_fontDb.data();
}
QPlatformTheme *Integration::createPlatformTheme(const QString &name) const
{
return QGenericUnixTheme::createUnixTheme(name);
}
QStringList Integration::themeNames() const
{
if (qEnvironmentVariableIsSet("KDE_FULL_SESSION")) {
return QStringList({QStringLiteral("kde")});
}
return QStringList({QLatin1String(QGenericUnixTheme::name)});
}
QPlatformOpenGLContext *Integration::createPlatformOpenGLContext(QOpenGLContext *context) const
{
if (kwinApp()->platform()->sceneEglGlobalShareContext() == EGL_NO_CONTEXT) {
qCWarning(KWIN_QPA) << "Attempting to create a QOpenGLContext before the scene is initialized";
return nullptr;
}
const EGLDisplay eglDisplay = kwinApp()->platform()->sceneEglDisplay();
if (eglDisplay != EGL_NO_DISPLAY) {
EGLPlatformContext *platformContext = new EGLPlatformContext(context, eglDisplay);
return platformContext;
}
return nullptr;
}
void Integration::handlePlatformCreated()
{
connect(kwinApp()->platform(), &Platform::outputEnabled,
this, &Integration::handleOutputEnabled);
connect(kwinApp()->platform(), &Platform::outputDisabled,
this, &Integration::handleOutputDisabled);
const QVector<AbstractOutput *> outputs = kwinApp()->platform()->enabledOutputs();
for (AbstractOutput *output : outputs) {
handleOutputEnabled(output);
}
}
void Integration::handleOutputEnabled(AbstractOutput *output)
{
Screen *platformScreen = new Screen(output);
QWindowSystemInterface::handleScreenAdded(platformScreen);
if (m_dummyScreen) {
QWindowSystemInterface::handleScreenRemoved(m_dummyScreen);
m_dummyScreen = nullptr;
}
}
void Integration::handleOutputDisabled(AbstractOutput *output)
{
Screen *platformScreen = m_screens.take(output);
if (!platformScreen) {
qCWarning(KWIN_QPA) << "Unknown output" << output;
return;
}
if (m_screens.isEmpty()) {
m_dummyScreen = new PlaceholderScreen();
QWindowSystemInterface::handleScreenAdded(m_dummyScreen);
}
QWindowSystemInterface::handleScreenRemoved(platformScreen);
}
}
}