/* This file is part of the KDE project SPDX-FileCopyrightText: 2014 Marco Martin SPDX-FileCopyrightText: 2014 Vishesh Handa SPDX-FileCopyrightText: 2019 Cyril Rossi SPDX-License-Identifier: LGPL-2.0-only */ #include "kcm.h" #include #include #include #include #include #include #include #include #include #include #include #include "splashscreendata.h" #include "splashscreensettings.h" K_PLUGIN_FACTORY_WITH_JSON(KCMSplashScreenFactory, "kcm_splashscreen.json", registerPlugin(); registerPlugin();) KCMSplashScreen::KCMSplashScreen(QObject *parent, const QVariantList &args) : KQuickAddons::ManagedConfigModule(parent, args) , m_data(new SplashScreenData(this)) , m_model(new QStandardItemModel(this)) { qmlRegisterType(); qmlRegisterType(); KAboutData *about = new KAboutData(QStringLiteral("kcm_splashscreen"), i18n("Splash Screen"), QStringLiteral("0.1"), QString(), KAboutLicense::LGPL); about->addAuthor(i18n("Marco Martin"), QString(), QStringLiteral("mart@kde.org")); setAboutData(about); setButtons(Help | Apply | Default); QHash roles = m_model->roleNames(); roles[PluginNameRole] = "pluginName"; roles[ScreenshotRole] = "screenshot"; roles[DescriptionRole] = "description"; roles[UninstallableRole] = "uninstallable"; roles[PendingDeletionRole] = "pendingDeletion"; m_model->setItemRoleNames(roles); m_sortModel = new QSortFilterProxyModel(this); m_sortModel->setSourceModel(m_model); m_sortModel->setSortLocaleAware(true); m_sortModel->setSortRole(Qt::DisplayRole); m_sortModel->setSortCaseSensitivity(Qt::CaseInsensitive); m_sortModel->setDynamicSortFilter(true); connect(m_model, &QAbstractItemModel::dataChanged, this, [this] { bool hasPendingDeletions = !pendingDeletions().isEmpty(); setNeedsSave(m_data->settings()->isSaveNeeded() || hasPendingDeletions); setRepresentsDefaults(m_data->settings()->isDefaults() && !hasPendingDeletions); }); } QList KCMSplashScreen::availablePackages(const QString &component) { QList packages; QStringList paths; const QStringList dataPaths = QStandardPaths::standardLocations(QStandardPaths::GenericDataLocation); for (const QString &path : dataPaths) { QDir dir(path + QStringLiteral("/plasma/look-and-feel")); paths << dir.entryList(QDir::AllDirs | QDir::NoDotAndDotDot); } for (const QString &path : qAsConst(paths)) { KPackage::Package pkg = KPackage::PackageLoader::self()->loadPackage(QStringLiteral("Plasma/LookAndFeel")); pkg.setPath(path); pkg.setFallbackPackage(KPackage::Package()); if (component.isEmpty() || !pkg.filePath(component.toUtf8()).isEmpty()) { packages << pkg; } } return packages; } SplashScreenSettings *KCMSplashScreen::splashScreenSettings() const { return m_data->settings(); } QAbstractProxyModel *KCMSplashScreen::splashSortedModel() const { return m_sortModel; } void KCMSplashScreen::ghnsEntryChanged(KNSCore::EntryWrapper *wrapper) { auto removeItemFromModel = [this](const QStringList &files) { if (!files.isEmpty()) { const QString guessedPluginId = QFileInfo(files.constFirst()).fileName(); const int index = pluginIndex(guessedPluginId); if (index != -1) { m_model->removeRows(index, 1); } } }; const KNSCore::EntryInternal entry = wrapper->entry(); if (entry.status() == KNS3::Entry::Deleted) { removeItemFromModel(entry.uninstalledFiles()); } else if (entry.status() == KNS3::Entry::Installed) { removeItemFromModel(entry.installedFiles()); KPackage::Package pkg = KPackage::PackageLoader::self()->loadPackage(QStringLiteral("Plasma/LookAndFeel")); pkg.setPath(entry.installedFiles().constFirst()); addKPackageToModel(pkg); m_sortModel->sort(Qt::DisplayRole); } } void KCMSplashScreen::addKPackageToModel(const KPackage::Package &pkg) { const static QString writableLocation = QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation); QStandardItem *row = new QStandardItem(pkg.metadata().name()); row->setData(pkg.metadata().pluginId(), PluginNameRole); row->setData(pkg.filePath("previews", QStringLiteral("splash.png")), ScreenshotRole); row->setData(pkg.metadata().description(), DescriptionRole); row->setData(pkg.path().startsWith(writableLocation), UninstallableRole); row->setData(false, PendingDeletionRole); m_packageRoot = writableLocation + QLatin1Char('/') + pkg.defaultPackageRoot(); m_model->appendRow(row); } void KCMSplashScreen::load() { m_data->settings()->load(); m_model->clear(); const QList pkgs = availablePackages(QStringLiteral("splashmainscript")); for (const KPackage::Package &pkg : pkgs) { addKPackageToModel(pkg); } m_sortModel->sort(Qt::DisplayRole); QStandardItem *row = new QStandardItem(i18n("None")); row->setData("None", PluginNameRole); row->setData(i18n("No splash screen will be shown"), DescriptionRole); row->setData(false, UninstallableRole); m_model->insertRow(0, row); if (-1 == pluginIndex(m_data->settings()->theme())) { defaults(); } Q_EMIT m_data->settings()->themeChanged(); } void KCMSplashScreen::save() { using namespace KPackage; PackageStructure *structure = PackageLoader::self()->loadPackageStructure(QStringLiteral("Plasma/LookAndFeel")); const QStringList pendingDeletionPlugins = pendingDeletions(); for (const QString &plugin : pendingDeletionPlugins) { if (plugin == m_data->settings()->theme()) { Q_EMIT error(i18n("You cannot delete the currently selected splash screen")); m_model->setData(m_model->index(pluginIndex(plugin), 0), false, Roles::PendingDeletionRole); continue; } KJob *uninstallJob = Package(structure).uninstall(plugin, m_packageRoot); connect(uninstallJob, &KJob::result, this, [this, uninstallJob, plugin]() { if (uninstallJob->error()) { Q_EMIT error(uninstallJob->errorString()); } else { m_model->removeRows(pluginIndex(plugin), 1); } }); } m_data->settings()->setEngine(m_data->settings()->theme() == QStringLiteral("None") ? QStringLiteral("none") : QStringLiteral("KSplashQML")); ManagedConfigModule::save(); } int KCMSplashScreen::sortModelPluginIndex(const QString &pluginName) const { return m_sortModel->mapFromSource(m_model->index(pluginIndex(pluginName), 0)).row(); } int KCMSplashScreen::pluginIndex(const QString &pluginName) const { const auto results = m_model->match(m_model->index(0, 0), PluginNameRole, pluginName, 1, Qt::MatchExactly); if (results.count() == 1) { return results.first().row(); } return -1; } bool KCMSplashScreen::testing() const { return m_testProcess; } void KCMSplashScreen::test(const QString &plugin) { if (plugin.isEmpty() || plugin == QLatin1String("None") || m_testProcess) { return; } m_testProcess = new QProcess(this); connect(m_testProcess, &QProcess::errorOccurred, this, [this](QProcess::ProcessError error) { Q_UNUSED(error) Q_EMIT testingFailed(QString::fromLocal8Bit(m_testProcess->readAllStandardError())); }); connect(m_testProcess, QOverload::of(&QProcess::finished), this, [this](int exitCode, QProcess::ExitStatus exitStatus) { Q_UNUSED(exitCode) Q_UNUSED(exitStatus) m_testProcess->deleteLater(); m_testProcess = nullptr; Q_EMIT testingChanged(); }); Q_EMIT testingChanged(); m_testProcess->start(QStringLiteral("ksplashqml"), {plugin, QStringLiteral("--test")}); } QStringList KCMSplashScreen::pendingDeletions() { QStringList pendingDeletions; for (int i = 0, count = m_model->rowCount(); i < count; ++i) { if (m_model->item(i)->data(Roles::PendingDeletionRole).toBool()) { pendingDeletions << m_model->item(i)->data(Roles::PluginNameRole).toString(); } } return pendingDeletions; } void KCMSplashScreen::defaults() { ManagedConfigModule::defaults(); // Make sure we clear all pending deletions for (int i = 0, count = m_model->rowCount(); i < count; ++i) { m_model->item(i)->setData(false, Roles::PendingDeletionRole); } } #include "kcm.moc"