255 lines
8.5 KiB
C++
Raw Normal View History

2022-04-02 18:24:21 +05:00
/*
SPDX-FileCopyrightText: 2014 Marco Martin <mart@kde.org>
SPDX-FileCopyrightText: 2014 Vishesh Handa <me@vhanda.in>
SPDX-FileCopyrightText: 2016 David Rosca <nowrep@gmail.com>
SPDX-FileCopyrightText: 2018 Kai Uwe Broulik <kde@privat.broulik.de>
SPDX-FileCopyrightText: 2019 Kevin Ottens <kevin.ottens@enioka.com>
SPDX-License-Identifier: LGPL-2.0-only
*/
#include "kcm.h"
#include <KLocalizedString>
#include <KPluginFactory>
#include <KIO/FileCopyJob>
#include <KIO/JobUiDelegate>
#include <Plasma/Svg>
#include <Plasma/Theme>
#include <QDebug>
#include <QProcess>
#include <QQuickItem>
#include <QQuickWindow>
#include <QStandardItemModel>
#include <QStandardPaths>
#include <QTemporaryFile>
#include "desktopthemedata.h"
#include "desktopthemesettings.h"
#include "filterproxymodel.h"
#include "themesmodel.h"
Q_LOGGING_CATEGORY(KCM_DESKTOP_THEME, "kcm_desktoptheme")
K_PLUGIN_FACTORY_WITH_JSON(KCMDesktopThemeFactory, "kcm_desktoptheme.json", registerPlugin<KCMDesktopTheme>(); registerPlugin<DesktopThemeData>();)
KCMDesktopTheme::KCMDesktopTheme(QObject *parent, const KPluginMetaData &data, const QVariantList &args)
: KQuickAddons::ManagedConfigModule(parent, data, args)
, m_data(new DesktopThemeData(this))
, m_model(new ThemesModel(this))
, m_filteredModel(new FilterProxyModel(this))
, m_haveThemeExplorerInstalled(false)
{
qmlRegisterAnonymousType<DesktopThemeSettings>("org.kde.private.kcms.desktoptheme", 1);
qmlRegisterUncreatableType<ThemesModel>("org.kde.private.kcms.desktoptheme", 1, 0, "ThemesModel", "Cannot create ThemesModel");
qmlRegisterUncreatableType<FilterProxyModel>("org.kde.private.kcms.desktoptheme", 1, 0, "FilterProxyModel", "Cannot create FilterProxyModel");
setButtons(Apply | Default | Help);
m_haveThemeExplorerInstalled = !QStandardPaths::findExecutable(QStringLiteral("plasmathemeexplorer")).isEmpty();
connect(m_model, &ThemesModel::pendingDeletionsChanged, this, &KCMDesktopTheme::settingsChanged);
connect(m_model, &ThemesModel::selectedThemeChanged, this, [this](const QString &pluginName) {
desktopThemeSettings()->setName(pluginName);
});
connect(desktopThemeSettings(), &DesktopThemeSettings::nameChanged, this, [this] {
m_model->setSelectedTheme(desktopThemeSettings()->name());
});
connect(m_model, &ThemesModel::selectedThemeChanged, m_filteredModel, &FilterProxyModel::setSelectedTheme);
m_filteredModel->setSourceModel(m_model);
}
KCMDesktopTheme::~KCMDesktopTheme()
{
}
DesktopThemeSettings *KCMDesktopTheme::desktopThemeSettings() const
{
return m_data->settings();
}
ThemesModel *KCMDesktopTheme::desktopThemeModel() const
{
return m_model;
}
FilterProxyModel *KCMDesktopTheme::filteredModel() const
{
return m_filteredModel;
}
bool KCMDesktopTheme::downloadingFile() const
{
return m_tempCopyJob;
}
void KCMDesktopTheme::installThemeFromFile(const QUrl &url)
{
if (url.isLocalFile()) {
installTheme(url.toLocalFile());
return;
}
if (m_tempCopyJob) {
return;
}
m_tempInstallFile.reset(new QTemporaryFile());
if (!m_tempInstallFile->open()) {
Q_EMIT showErrorMessage(i18n("Unable to create a temporary file."));
m_tempInstallFile.reset();
return;
}
m_tempCopyJob = KIO::file_copy(url, QUrl::fromLocalFile(m_tempInstallFile->fileName()), -1, KIO::Overwrite);
m_tempCopyJob->uiDelegate()->setAutoErrorHandlingEnabled(true);
Q_EMIT downloadingFileChanged();
connect(m_tempCopyJob, &KIO::FileCopyJob::result, this, [this, url](KJob *job) {
if (job->error() != KJob::NoError) {
Q_EMIT showErrorMessage(i18n("Unable to download the theme: %1", job->errorText()));
return;
}
installTheme(m_tempInstallFile->fileName());
m_tempInstallFile.reset();
});
connect(m_tempCopyJob, &QObject::destroyed, this, &KCMDesktopTheme::downloadingFileChanged);
}
void KCMDesktopTheme::installTheme(const QString &path)
{
qCDebug(KCM_DESKTOP_THEME) << "Installing ... " << path;
const QString program = QStringLiteral("kpackagetool5");
const QStringList arguments = {QStringLiteral("--type"), QStringLiteral("Plasma/Theme"), QStringLiteral("--install"), path};
qCDebug(KCM_DESKTOP_THEME) << program << arguments.join(QLatin1Char(' '));
QProcess *myProcess = new QProcess(this);
connect(myProcess,
static_cast<void (QProcess::*)(int, QProcess::ExitStatus)>(&QProcess::finished),
this,
[this](int exitCode, QProcess::ExitStatus exitStatus) {
Q_UNUSED(exitStatus)
if (exitCode == 0) {
Q_EMIT showSuccessMessage(i18n("Theme installed successfully."));
load();
} else {
Q_EMIT showErrorMessage(i18n("Theme installation failed."));
}
});
connect(myProcess, &QProcess::errorOccurred, this, [this](QProcess::ProcessError e) {
qCWarning(KCM_DESKTOP_THEME) << "Theme installation failed: " << e;
Q_EMIT showErrorMessage(i18n("Theme installation failed."));
});
myProcess->start(program, arguments);
}
void KCMDesktopTheme::applyPlasmaTheme(QQuickItem *item, const QString &themeName)
{
if (!item) {
return;
}
Plasma::Theme *theme = m_themes[themeName];
if (!theme) {
theme = new Plasma::Theme(themeName, this);
m_themes[themeName] = theme;
}
Q_FOREACH (Plasma::Svg *svg, item->findChildren<Plasma::Svg *>()) {
svg->setTheme(theme);
svg->setUsingRenderingCache(false);
}
}
void KCMDesktopTheme::load()
{
ManagedConfigModule::load();
m_model->load();
m_model->setSelectedTheme(desktopThemeSettings()->name());
}
void KCMDesktopTheme::save()
{
ManagedConfigModule::save();
Plasma::Theme().setThemeName(desktopThemeSettings()->name());
processPendingDeletions();
}
void KCMDesktopTheme::defaults()
{
ManagedConfigModule::defaults();
// can this be done more elegantly?
const auto pendingDeletions = m_model->match(m_model->index(0, 0), ThemesModel::PendingDeletionRole, true);
for (const QModelIndex &idx : pendingDeletions) {
m_model->setData(idx, false, ThemesModel::PendingDeletionRole);
}
}
bool KCMDesktopTheme::canEditThemes() const
{
return m_haveThemeExplorerInstalled;
}
void KCMDesktopTheme::editTheme(const QString &theme)
{
QProcess::startDetached(QStringLiteral("plasmathemeexplorer"), {QStringLiteral("-t"), theme});
}
bool KCMDesktopTheme::isSaveNeeded() const
{
return !m_model->match(m_model->index(0, 0), ThemesModel::PendingDeletionRole, true).isEmpty();
}
void KCMDesktopTheme::processPendingDeletions()
{
const QString program = QStringLiteral("plasmapkg2");
const auto pendingDeletions = m_model->match(m_model->index(0, 0), ThemesModel::PendingDeletionRole, true, -1 /*all*/);
QVector<QPersistentModelIndex> persistentPendingDeletions;
// turn into persistent model index so we can delete as we go
std::transform(pendingDeletions.begin(), pendingDeletions.end(), std::back_inserter(persistentPendingDeletions), [](const QModelIndex &idx) {
return QPersistentModelIndex(idx);
});
for (const QPersistentModelIndex &idx : persistentPendingDeletions) {
const QString pluginName = idx.data(ThemesModel::PluginNameRole).toString();
const QString displayName = idx.data(Qt::DisplayRole).toString();
Q_ASSERT(pluginName != desktopThemeSettings()->name());
const QStringList arguments = {QStringLiteral("-t"), QStringLiteral("theme"), QStringLiteral("-r"), pluginName};
QProcess *process = new QProcess(this);
connect(process,
static_cast<void (QProcess::*)(int, QProcess::ExitStatus)>(&QProcess::finished),
this,
[this, process, idx, pluginName, displayName](int exitCode, QProcess::ExitStatus exitStatus) {
Q_UNUSED(exitStatus)
if (exitCode == 0) {
m_model->removeRow(idx.row());
} else {
Q_EMIT showErrorMessage(i18n("Removing theme failed: %1", QString::fromLocal8Bit(process->readAllStandardOutput().trimmed())));
m_model->setData(idx, false, ThemesModel::PendingDeletionRole);
}
process->deleteLater();
});
process->start(program, arguments);
process->waitForFinished(); // needed so it deletes fine when "OK" is clicked and the dialog destroyed
}
}
#include "kcm.moc"