/** * SPDX-FileCopyrightText: 2015 David Edmundson <davidedmundson@kde.org> * * SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL */ #include "runcommand_config.h" #include <QDebug> #include <QFileDialog> #include <QHBoxLayout> #include <QHeaderView> #include <QJsonArray> #include <QJsonDocument> #include <QMenu> #include <QPushButton> #include <QStandardItemModel> #include <QStandardPaths> #include <QTableView> #include <QUuid> #include <KLocalizedString> #include <KPluginFactory> #include <dbushelper.h> K_PLUGIN_FACTORY(ShareConfigFactory, registerPlugin<RunCommandConfig>();) RunCommandConfig::RunCommandConfig(QObject *parent, const QVariantList &args) : KdeConnectPluginKcm(parent, args, QStringLiteral("kdeconnect_runcommand")) { // The qdbus executable name is different on some systems QString qdbusExe = QStringLiteral("qdbus-qt5"); if (QStandardPaths::findExecutable(qdbusExe).isEmpty()) { qdbusExe = QStringLiteral("qdbus"); } QMenu *defaultMenu = new QMenu(widget()); #ifdef Q_OS_WIN addSuggestedCommand(defaultMenu, i18n("Schedule a shutdown"), QStringLiteral("shutdown /s /t 60")); addSuggestedCommand(defaultMenu, i18n("Shutdown now"), QStringLiteral("shutdown /s /t 0")); addSuggestedCommand(defaultMenu, i18n("Cancel last shutdown"), QStringLiteral("shutdown /a")); addSuggestedCommand(defaultMenu, i18n("Schedule a reboot"), QStringLiteral("shutdown /r /t 60")); addSuggestedCommand(defaultMenu, i18n("Suspend"), QStringLiteral("rundll32.exe powrprof.dll,SetSuspendState 0,1,0")); addSuggestedCommand(defaultMenu, i18n("Lock Screen"), QStringLiteral("rundll32.exe user32.dll,LockWorkStation")); addSuggestedCommand( defaultMenu, i18n("Say Hello"), QStringLiteral("PowerShell -Command \"Add-Type –AssemblyName System.Speech; (New-Object System.Speech.Synthesis.SpeechSynthesizer).Speak('hello');\"")); #else addSuggestedCommand(defaultMenu, i18n("Shutdown"), QStringLiteral("systemctl poweroff")); addSuggestedCommand(defaultMenu, i18n("Reboot"), QStringLiteral("systemctl reboot")); addSuggestedCommand(defaultMenu, i18n("Suspend"), QStringLiteral("systemctl suspend")); addSuggestedCommand( defaultMenu, i18n("Maximum Brightness"), QStringLiteral("%0 org.kde.Solid.PowerManagement /org/kde/Solid/PowerManagement/Actions/BrightnessControl " "org.kde.Solid.PowerManagement.Actions.BrightnessControl.setBrightness `%0 org.kde.Solid.PowerManagement " "/org/kde/Solid/PowerManagement/Actions/BrightnessControl org.kde.Solid.PowerManagement.Actions.BrightnessControl.brightnessMax`") .arg(qdbusExe)); addSuggestedCommand(defaultMenu, i18n("Lock Screen"), QStringLiteral("loginctl lock-session")); addSuggestedCommand(defaultMenu, i18n("Unlock Screen"), QStringLiteral("loginctl unlock-session")); addSuggestedCommand(defaultMenu, i18n("Close All Vaults"), QStringLiteral("%0 org.kde.kded5 /modules/plasmavault closeAllVaults").arg(qdbusExe)); addSuggestedCommand(defaultMenu, i18n("Forcefully Close All Vaults"), QStringLiteral("%0 org.kde.kded5 /modules/plasmavault forceCloseAllVaults").arg(qdbusExe)); #endif QTableView *table = new QTableView(widget()); table->horizontalHeader()->setStretchLastSection(true); table->verticalHeader()->setVisible(false); QPushButton *button = new QPushButton(QIcon::fromTheme(QStringLiteral("list-add")), i18n("Sample commands"), widget()); button->setMenu(defaultMenu); QHBoxLayout *importExportLayout = new QHBoxLayout(); QPushButton *exportButton = new QPushButton(i18n("Export"), widget()); importExportLayout->addWidget(exportButton); connect(exportButton, &QPushButton::clicked, this, &RunCommandConfig::exportCommands); QPushButton *importButton = new QPushButton(i18n("Import"), widget()); importExportLayout->addWidget(importButton); connect(importButton, &QPushButton::clicked, this, &RunCommandConfig::importCommands); QVBoxLayout *layout = new QVBoxLayout(); layout->addWidget(table); layout->addLayout(importExportLayout); layout->addWidget(button); widget()->setLayout(layout); m_entriesModel = new QStandardItemModel(this); table->setModel(m_entriesModel); m_entriesModel->setHorizontalHeaderLabels(QStringList() << i18n("Name") << i18n("Command")); } RunCommandConfig::~RunCommandConfig() { } void RunCommandConfig::exportCommands() { QString filePath = QFileDialog::getSaveFileName(widget(), i18n("Export Commands"), QDir::homePath(), QStringLiteral("JSON (*.json)")); if (filePath.isEmpty()) return; QFile file(filePath); if (!file.open(QFile::WriteOnly | QFile::Text)) { qWarning() << "Could not write to file:" << filePath; return; } QJsonArray jsonArray; for (int i = 0; i < m_entriesModel->rowCount(); i++) { QJsonObject jsonObj; jsonObj[QStringLiteral("name")] = m_entriesModel->index(i, 0).data().toString(); jsonObj[QStringLiteral("command")] = m_entriesModel->index(i, 1).data().toString(); jsonArray.append(jsonObj); } QJsonDocument jsonDocument(jsonArray); file.write(jsonDocument.toJson()); file.close(); } void RunCommandConfig::importCommands() { QString filePath = QFileDialog::getOpenFileName(widget(), i18n("Import Commands"), QDir::homePath(), QStringLiteral("JSON (*.json)")); if (filePath.isEmpty()) return; QFile file(filePath); if (!file.open(QFile::ReadOnly | QFile::Text)) { qWarning() << "Could not read file:" << filePath; return; } QByteArray jsonData = file.readAll(); file.close(); QJsonDocument jsonDoc = QJsonDocument::fromJson(jsonData); if (jsonDoc.isNull() || !jsonDoc.isArray()) { qWarning() << "Invalid JSON format."; return; } // Clear the current command list m_entriesModel->removeRows(0, m_entriesModel->rowCount()); // Populate the model with the imported commands QJsonArray jsonArray = jsonDoc.array(); for (const QJsonValue &jsonValue : jsonArray) { QJsonObject jsonObj = jsonValue.toObject(); QString name = jsonObj.value(QStringLiteral("name")).toString(); QString command = jsonObj.value(QStringLiteral("command")).toString(); insertRow(m_entriesModel->rowCount(), name, command); } markAsChanged(); } void RunCommandConfig::addSuggestedCommand(QMenu *menu, const QString &name, const QString &command) { auto action = new QAction(name); connect(action, &QAction::triggered, action, [this, name, command]() { insertRow(0, name, command); markAsChanged(); }); menu->addAction(action); } void RunCommandConfig::defaults() { KCModule::defaults(); m_entriesModel->removeRows(0, m_entriesModel->rowCount()); markAsChanged(); } void RunCommandConfig::load() { KCModule::load(); QJsonDocument jsonDocument = QJsonDocument::fromJson(config()->getByteArray(QStringLiteral("commands"), "{}")); QJsonObject jsonConfig = jsonDocument.object(); const QStringList keys = jsonConfig.keys(); for (const QString &key : keys) { const QJsonObject entry = jsonConfig[key].toObject(); const QString name = entry[QStringLiteral("name")].toString(); const QString command = entry[QStringLiteral("command")].toString(); QStandardItem *newName = new QStandardItem(name); newName->setEditable(true); newName->setData(key); QStandardItem *newCommand = new QStandardItem(command); newName->setEditable(true); m_entriesModel->appendRow(QList<QStandardItem *>() << newName << newCommand); } m_entriesModel->sort(0); insertEmptyRow(); connect(m_entriesModel, &QAbstractItemModel::dataChanged, this, &RunCommandConfig::onDataChanged); } void RunCommandConfig::save() { KCModule::save(); QJsonObject jsonConfig; for (int i = 0; i < m_entriesModel->rowCount(); i++) { QString key = m_entriesModel->item(i, 0)->data().toString(); const QString name = m_entriesModel->item(i, 0)->text(); const QString command = m_entriesModel->item(i, 1)->text(); if (name.isEmpty() || command.isEmpty()) { continue; } if (key.isEmpty()) { key = QUuid::createUuid().toString(); DBusHelper::filterNonExportableCharacters(key); } QJsonObject entry; entry[QStringLiteral("name")] = name; entry[QStringLiteral("command")] = command; jsonConfig[key] = entry; } QJsonDocument document; document.setObject(jsonConfig); config()->set(QStringLiteral("commands"), document.toJson(QJsonDocument::Compact)); } void RunCommandConfig::insertEmptyRow() { insertRow(m_entriesModel->rowCount(), {}, {}); } void RunCommandConfig::insertRow(int i, const QString &name, const QString &command) { QStandardItem *newName = new QStandardItem(name); newName->setEditable(true); QStandardItem *newCommand = new QStandardItem(command); newName->setEditable(true); m_entriesModel->insertRow(i, QList<QStandardItem *>() << newName << newCommand); } void RunCommandConfig::onDataChanged(const QModelIndex &topLeft, const QModelIndex &bottomRight) { markAsChanged(); Q_UNUSED(topLeft); if (bottomRight.row() == m_entriesModel->rowCount() - 1) { // TODO check both entries are still empty insertEmptyRow(); } } #include "runcommand_config.moc"