kwin/decorations/decorationbridge.cpp
Martin Gräßlin fcec334b3c [kdecoration] Adjust to change that DecorationBridge is no longer a singleton
* Aurorae needs to pass QVariantList args to parent Decoration
* DecorationBridge implementation needs to be a KWIN_SINGLETON
* DecorationBridge needs to be passed with args to created Decoration
2014-10-30 09:01:06 +01:00

207 lines
6.2 KiB
C++

/********************************************************************
KWin - the KDE window manager
This file is part of the KDE project.
Copyright (C) 2014 Martin Gräßlin <mgraesslin@kde.org>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*********************************************************************/
#include "decorationbridge.h"
#include "decoratedclient.h"
#include "decorationrenderer.h"
#include "settings.h"
// KWin core
#include "client.h"
#include "composite.h"
#include "scene.h"
#include "workspace.h"
// KDecoration
#include <KDecoration2/Decoration>
#include <KDecoration2/DecoratedClient>
#include <KDecoration2/DecorationSettings>
// Frameworks
#include <KPluginTrader>
#include <KPluginLoader>
// Qt
#include <QDebug>
#include <QPainter>
namespace KWin
{
namespace Decoration
{
static const QString s_pluginName = QStringLiteral("org.kde.kdecoration2");
KWIN_SINGLETON_FACTORY(DecorationBridge)
DecorationBridge::DecorationBridge(QObject *parent)
: KDecoration2::DecorationBridge(parent)
, m_factory(nullptr)
, m_blur(false)
, m_settings()
{
}
DecorationBridge::~DecorationBridge()
{
s_self = nullptr;
}
static QString readPlugin()
{
return KSharedConfig::openConfig(KWIN_CONFIG)->group(s_pluginName).readEntry("library", QStringLiteral("org.kde.breeze"));
}
QString DecorationBridge::readTheme() const
{
return KSharedConfig::openConfig(KWIN_CONFIG)->group(s_pluginName).readEntry("theme", m_defaultTheme);
}
void DecorationBridge::init()
{
m_plugin = readPlugin();
m_settings = QSharedPointer<KDecoration2::DecorationSettings>::create(this);
initPlugin();
}
void DecorationBridge::initPlugin()
{
const auto offers = KPluginTrader::self()->query(s_pluginName,
s_pluginName,
QStringLiteral("[X-KDE-PluginInfo-Name] == '%1'").arg(m_plugin));
if (offers.isEmpty()) {
qWarning() << "Could not locate decoration plugin";
return;
}
qDebug() << "Trying to load decoration plugin: " << offers.first().libraryPath();
KPluginLoader loader(offers.first().libraryPath());
KPluginFactory *factory = loader.factory();
if (!factory) {
qWarning() << "Error loading plugin:" << loader.errorString();
} else {
m_factory = factory;
loadMetaData(loader.metaData().value(QStringLiteral("MetaData")).toObject());
}
}
static void recreateDecorations()
{
Workspace::self()->forEachClient([](Client *c) { c->updateDecoration(true, true); });
}
void DecorationBridge::reconfigure()
{
const QString newPlugin = readPlugin();
if (newPlugin != m_plugin) {
// plugin changed, recreate everything
auto oldFactory = m_factory;
const auto oldPluginName = m_plugin;
m_plugin = newPlugin;
initPlugin();
if (m_factory == oldFactory) {
// loading new plugin failed
m_factory = oldFactory;
m_plugin = oldPluginName;
} else {
recreateDecorations();
// TODO: unload and destroy old plugin
}
} else {
// same plugin, but theme might have changed
const QString oldTheme = m_theme;
m_theme = readTheme();
if (m_theme != oldTheme) {
recreateDecorations();
}
}
}
void DecorationBridge::loadMetaData(const QJsonObject &object)
{
// reset all settings
m_blur = false;
m_theme = QString();
m_defaultTheme = QString();
// load the settings
const QJsonValue decoSettings = object.value(s_pluginName);
if (decoSettings.isUndefined()) {
// no settings
return;
}
const QVariantMap decoSettingsMap = decoSettings.toObject().toVariantMap();
auto blurIt = decoSettingsMap.find(QStringLiteral("blur"));
if (blurIt != decoSettingsMap.end()) {
m_blur = blurIt.value().toBool();
}
findTheme(decoSettingsMap);
}
void DecorationBridge::findTheme(const QVariantMap &map)
{
auto it = map.find(QStringLiteral("themes"));
if (it == map.end()) {
return;
}
if (!it.value().toBool()) {
return;
}
it = map.find(QStringLiteral("defaultTheme"));
m_defaultTheme = it != map.end() ? it.value().toString() : QString();
m_theme = readTheme();
}
std::unique_ptr<KDecoration2::DecoratedClientPrivate> DecorationBridge::createClient(KDecoration2::DecoratedClient *client, KDecoration2::Decoration *decoration)
{
return std::unique_ptr<DecoratedClientImpl>(new DecoratedClientImpl(static_cast<Client*>(decoration->parent()), client, decoration));
}
std::unique_ptr<KDecoration2::DecorationSettingsPrivate> DecorationBridge::settings(KDecoration2::DecorationSettings *parent)
{
return std::unique_ptr<SettingsImpl>(new SettingsImpl(parent));
}
void DecorationBridge::update(KDecoration2::Decoration *decoration, const QRect &geometry)
{
// TODO: remove check once all compositors implement it
if (Client *c = Workspace::self()->findClient([decoration] (const Client *client) { return client->decoration() == decoration; })) {
if (Renderer *renderer = c->decoratedClient()->renderer()) {
renderer->schedule(geometry);
}
}
}
KDecoration2::Decoration *DecorationBridge::createDecoration(Client *client)
{
if (!m_factory) {
return nullptr;
}
QVariantMap args({ {QStringLiteral("bridge"), QVariant::fromValue(this)} });
if (!m_theme.isEmpty()) {
args.insert(QStringLiteral("theme"), m_theme);
}
auto deco = m_factory->create<KDecoration2::Decoration>(client, QVariantList({args}));
deco->setSettings(m_settings);
deco->init();
return deco;
}
} // Decoration
} // KWin