6a580d2b90
A decoration plugin can indicate that it supports themes. For this the decoration plugin needs to specify in the org.kde.kdecoration2 group of the JSON meta data the following: themes: true, defaultTheme: "nameofDefaultTheme" If the themes key/value pair is present and set to true KWin will read the entry "theme" in the [org.kde.kdecoration2] group with the value of defaultTheme as the default value. The read value (if there is one) is passed to the created decoration in the QVariantList. The QVariantList contains a QVariantMap as first entry and the theme name is passed for key "theme".
158 lines
5 KiB
C++
158 lines
5 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");
|
|
|
|
DecorationBridge::DecorationBridge(QObject *parent)
|
|
: QObject(parent)
|
|
, KDecoration2::DecorationBridge()
|
|
, m_factory(nullptr)
|
|
, m_blur(false)
|
|
{
|
|
}
|
|
|
|
DecorationBridge::~DecorationBridge() = default;
|
|
|
|
DecorationBridge *DecorationBridge::self()
|
|
{
|
|
return static_cast<KWin::Decoration::DecorationBridge*>(KDecoration2::DecorationBridge::self());
|
|
}
|
|
|
|
void DecorationBridge::init()
|
|
{
|
|
KConfigGroup config = KSharedConfig::openConfig(KWIN_CONFIG)->group(s_pluginName);
|
|
KDecoration2::DecorationSettings::self(this);
|
|
|
|
const auto offers = KPluginTrader::self()->query(s_pluginName,
|
|
s_pluginName,
|
|
QStringLiteral("[X-KDE-PluginInfo-Name] == '%1'").arg(config.readEntry("library", "org.kde.breeze")));
|
|
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());
|
|
}
|
|
}
|
|
|
|
void DecorationBridge::loadMetaData(const QJsonObject &object)
|
|
{
|
|
// reset all settings
|
|
m_blur = false;
|
|
m_theme = 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"));
|
|
const KConfigGroup config = KSharedConfig::openConfig(KWIN_CONFIG)->group(s_pluginName);
|
|
m_theme = config.readEntry("theme", it != map.end() ? it.value().toString() : QString());
|
|
}
|
|
|
|
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;
|
|
}
|
|
QVariantList args;
|
|
if (!m_theme.isEmpty()) {
|
|
args << QVariantMap({ {QStringLiteral("theme"), m_theme} });
|
|
}
|
|
auto deco = m_factory->create<KDecoration2::Decoration>(client, args);
|
|
deco->init();
|
|
return deco;
|
|
}
|
|
|
|
|
|
} // Decoration
|
|
} // KWin
|