mirror of
https://github.com/Qortal/Brooklyn.git
synced 2025-02-12 02:05:54 +00:00
325 lines
14 KiB
C++
325 lines
14 KiB
C++
/*
|
|
SPDX-FileCopyrightText: 2020 David Redondo <david@david-redondo.de>
|
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
*/
|
|
|
|
#include "globalaccelmodel.h"
|
|
|
|
#include <QDBusPendingCallWatcher>
|
|
#include <QIcon>
|
|
|
|
#include <KApplicationTrader>
|
|
#include <KConfigGroup>
|
|
#include <KGlobalAccel>
|
|
#include <KGlobalShortcutInfo>
|
|
#include <KLocalizedString>
|
|
#include <KService>
|
|
#include <kglobalaccel_component_interface.h>
|
|
#include <kglobalaccel_interface.h>
|
|
|
|
#include "kcmkeys_debug.h"
|
|
|
|
static QStringList buildActionId(const QString &componentUnique, const QString &componentFriendly, const QString &actionUnique, const QString &actionFriendly)
|
|
{
|
|
QStringList actionId{"", "", "", ""};
|
|
actionId[KGlobalAccel::ComponentUnique] = componentUnique;
|
|
actionId[KGlobalAccel::ComponentFriendly] = componentFriendly;
|
|
actionId[KGlobalAccel::ActionUnique] = actionUnique;
|
|
actionId[KGlobalAccel::ActionFriendly] = actionFriendly;
|
|
return actionId;
|
|
}
|
|
|
|
GlobalAccelModel::GlobalAccelModel(KGlobalAccelInterface *interface, QObject *parent)
|
|
: BaseModel(parent)
|
|
, m_globalAccelInterface{interface}
|
|
{
|
|
}
|
|
|
|
QVariant GlobalAccelModel::data(const QModelIndex &index, int role) const
|
|
{
|
|
if (role == SupportsMultipleKeysRole) {
|
|
return false;
|
|
}
|
|
return BaseModel::data(index, role);
|
|
}
|
|
|
|
void GlobalAccelModel::load()
|
|
{
|
|
if (!m_globalAccelInterface->isValid()) {
|
|
return;
|
|
}
|
|
beginResetModel();
|
|
m_components.clear();
|
|
auto componentsWatcher = new QDBusPendingCallWatcher(m_globalAccelInterface->allComponents());
|
|
connect(componentsWatcher, &QDBusPendingCallWatcher::finished, this, [this](QDBusPendingCallWatcher *componentsWatcher) {
|
|
QDBusPendingReply<QList<QDBusObjectPath>> componentsReply = *componentsWatcher;
|
|
componentsWatcher->deleteLater();
|
|
if (componentsReply.isError()) {
|
|
genericErrorOccured(QStringLiteral("Error while calling allComponents()"), componentsReply.error());
|
|
endResetModel();
|
|
return;
|
|
}
|
|
const QList<QDBusObjectPath> componentPaths = componentsReply.value();
|
|
int *pendingCalls = new int;
|
|
*pendingCalls = componentPaths.size();
|
|
for (const auto &componentPath : componentPaths) {
|
|
const QString path = componentPath.path();
|
|
KGlobalAccelComponentInterface component(m_globalAccelInterface->service(), path, m_globalAccelInterface->connection());
|
|
auto watcher = new QDBusPendingCallWatcher(component.allShortcutInfos());
|
|
connect(watcher, &QDBusPendingCallWatcher::finished, this, [path, pendingCalls, this](QDBusPendingCallWatcher *watcher) {
|
|
QDBusPendingReply<QList<KGlobalShortcutInfo>> reply = *watcher;
|
|
if (reply.isError()) {
|
|
genericErrorOccured(QStringLiteral("Error while calling allShortCutInfos of") + path, reply.error());
|
|
} else if (!reply.value().isEmpty()) {
|
|
m_components.push_back(loadComponent(reply.value()));
|
|
}
|
|
watcher->deleteLater();
|
|
if (--*pendingCalls == 0) {
|
|
QCollator collator;
|
|
collator.setCaseSensitivity(Qt::CaseInsensitive);
|
|
collator.setNumericMode(true);
|
|
std::sort(m_components.begin(), m_components.end(), [&](const Component &c1, const Component &c2) {
|
|
return c1.type != c2.type ? c1.type < c2.type : collator.compare(c1.displayName, c2.displayName) < 0;
|
|
});
|
|
endResetModel();
|
|
delete pendingCalls;
|
|
}
|
|
});
|
|
}
|
|
});
|
|
}
|
|
|
|
Component GlobalAccelModel::loadComponent(const QList<KGlobalShortcutInfo> &info)
|
|
{
|
|
const QString &componentUnique = info[0].componentUniqueName();
|
|
const QString &componentFriendly = info[0].componentFriendlyName();
|
|
|
|
KService::Ptr service = KService::serviceByStorageId(componentUnique);
|
|
// Not a normal desktop file but maybe specific file in kglobalaccel dir
|
|
if (!service && componentUnique.endsWith(QLatin1String(".desktop"))) {
|
|
service = new KService(QStandardPaths::locate(QStandardPaths::GenericDataLocation, QStringLiteral("kglobalaccel/") + componentUnique));
|
|
}
|
|
|
|
if (!service) {
|
|
// Do we have a service with that name?
|
|
auto filter = [componentUnique, componentFriendly](const KService::Ptr service) {
|
|
return service->name() == componentUnique || service->name() == componentFriendly;
|
|
};
|
|
|
|
const KService::List services = KApplicationTrader::query(filter);
|
|
service = services.value(0, KService::Ptr());
|
|
}
|
|
const QString type = service && service->isApplication() ? i18n("Applications") : i18n("System Services");
|
|
QString icon;
|
|
|
|
static const QHash<QString, QString> hardCodedIcons = {
|
|
{"ActivityManager", "preferences-desktop-activities"},
|
|
{"KDE Keyboard Layout Switcher", "input-keyboard"},
|
|
{"org_kde_powerdevil", "preferences-system-power-management"},
|
|
};
|
|
|
|
if (service && !service->icon().isEmpty()) {
|
|
icon = service->icon();
|
|
} else if (hardCodedIcons.contains(componentUnique)) {
|
|
icon = hardCodedIcons[componentUnique];
|
|
} else {
|
|
icon = componentUnique;
|
|
}
|
|
|
|
Component c{componentUnique, componentFriendly, type, icon, {}, false, false};
|
|
for (const auto &actionInfo : info) {
|
|
const QString &actionUnique = actionInfo.uniqueName();
|
|
const QString &actionFriendly = actionInfo.friendlyName();
|
|
Action action;
|
|
action.id = actionUnique;
|
|
action.displayName = actionFriendly;
|
|
const QList<QKeySequence> defaultShortcuts = actionInfo.defaultKeys();
|
|
for (const auto &keySequence : defaultShortcuts) {
|
|
if (!keySequence.isEmpty()) {
|
|
action.defaultShortcuts.insert(keySequence);
|
|
}
|
|
}
|
|
const QList<QKeySequence> activeShortcuts = actionInfo.keys();
|
|
for (const QKeySequence &keySequence : activeShortcuts) {
|
|
if (!keySequence.isEmpty()) {
|
|
action.activeShortcuts.insert(keySequence);
|
|
}
|
|
}
|
|
action.initialShortcuts = action.activeShortcuts;
|
|
c.actions.push_back(action);
|
|
}
|
|
QCollator collator;
|
|
collator.setCaseSensitivity(Qt::CaseInsensitive);
|
|
collator.setNumericMode(true);
|
|
std::sort(c.actions.begin(), c.actions.end(), [&](const Action &s1, const Action &s2) {
|
|
return collator.compare(s1.displayName, s2.displayName) < 0;
|
|
});
|
|
return c;
|
|
}
|
|
|
|
void GlobalAccelModel::save()
|
|
{
|
|
for (auto it = m_components.rbegin(); it != m_components.rend(); ++it) {
|
|
if (it->pendingDeletion) {
|
|
removeComponent(*it);
|
|
continue;
|
|
}
|
|
for (auto &action : it->actions) {
|
|
if (action.initialShortcuts != action.activeShortcuts) {
|
|
const QStringList actionId = buildActionId(it->id, it->displayName, action.id, action.displayName);
|
|
// TODO: pass action.activeShortcuts to m_globalAccelInterface->setForeignShortcut() as a QSet<QKeySequence>
|
|
// or QList<QKeySequence>?
|
|
QList<QKeySequence> keys;
|
|
keys.reserve(action.activeShortcuts.size());
|
|
for (const QKeySequence &key : qAsConst(action.activeShortcuts)) {
|
|
keys.append(key);
|
|
}
|
|
qCDebug(KCMKEYS) << "Saving" << actionId << action.activeShortcuts << keys;
|
|
auto reply = m_globalAccelInterface->setForeignShortcutKeys(actionId, keys);
|
|
reply.waitForFinished();
|
|
if (!reply.isValid()) {
|
|
qCCritical(KCMKEYS) << "Error while saving";
|
|
if (reply.error().isValid()) {
|
|
qCCritical(KCMKEYS) << reply.error().name() << reply.error().message();
|
|
}
|
|
Q_EMIT errorOccured(i18nc("%1 is the name of the component, %2 is the action for which saving failed",
|
|
"Error while saving shortcut %1: %2",
|
|
it->displayName,
|
|
it->displayName));
|
|
} else {
|
|
action.initialShortcuts = action.activeShortcuts;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GlobalAccelModel::exportToConfig(const KConfigBase &config)
|
|
{
|
|
for (const auto &component : qAsConst(m_components)) {
|
|
if (component.checked) {
|
|
KConfigGroup mainGroup(&config, component.id);
|
|
KConfigGroup group(&mainGroup, "Global Shortcuts");
|
|
for (const auto &action : component.actions) {
|
|
const QList<QKeySequence> shortcutsList(action.activeShortcuts.cbegin(), action.activeShortcuts.cend());
|
|
group.writeEntry(action.id, QKeySequence::listToString(shortcutsList));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GlobalAccelModel::importConfig(const KConfigBase &config)
|
|
{
|
|
const auto groupList = config.groupList();
|
|
for (const auto &componentGroupName : groupList) {
|
|
auto component = std::find_if(m_components.begin(), m_components.end(), [&](const Component &c) {
|
|
return c.id == componentGroupName;
|
|
});
|
|
if (component == m_components.end()) {
|
|
qCWarning(KCMKEYS) << "Ignoring unknown component" << componentGroupName;
|
|
continue;
|
|
}
|
|
KConfigGroup componentGroup(&config, componentGroupName);
|
|
if (!componentGroup.hasGroup("Global Shortcuts")) {
|
|
qCWarning(KCMKEYS) << "Group" << componentGroupName << "has no shortcuts group";
|
|
continue;
|
|
}
|
|
KConfigGroup shortcutsGroup(&componentGroup, "Global Shortcuts");
|
|
const QStringList keys = shortcutsGroup.keyList();
|
|
for (const auto &key : keys) {
|
|
auto action = std::find_if(component->actions.begin(), component->actions.end(), [&](const Action &a) {
|
|
return a.id == key;
|
|
});
|
|
if (action == component->actions.end()) {
|
|
qCWarning(KCMKEYS) << "Ignoring unknown action" << key;
|
|
continue;
|
|
}
|
|
const auto shortcuts = QKeySequence::listFromString(shortcutsGroup.readEntry(key));
|
|
const QSet<QKeySequence> shortcutsSet(shortcuts.cbegin(), shortcuts.cend());
|
|
if (shortcutsSet != action->activeShortcuts) {
|
|
action->activeShortcuts = shortcutsSet;
|
|
const QModelIndex i = index(action - component->actions.begin(), 0, index(component - m_components.begin(), 0));
|
|
Q_EMIT dataChanged(i, i, {CustomShortcutsRole, ActiveShortcutsRole});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GlobalAccelModel::addApplication(const QString &desktopFileName, const QString &displayName)
|
|
{
|
|
// Register a dummy action to trigger kglobalaccel to parse the desktop file
|
|
QStringList actionId = buildActionId(desktopFileName, displayName, QString(), QString());
|
|
m_globalAccelInterface->doRegister(actionId);
|
|
m_globalAccelInterface->unRegister(actionId);
|
|
QCollator collator;
|
|
collator.setCaseSensitivity(Qt::CaseInsensitive);
|
|
collator.setNumericMode(true);
|
|
auto pos = std::lower_bound(m_components.begin(), m_components.end(), displayName, [&](const Component &c, const QString &name) {
|
|
return c.type != i18n("System Services") && collator.compare(c.displayName, name) < 0;
|
|
});
|
|
auto watcher = new QDBusPendingCallWatcher(m_globalAccelInterface->getComponent(desktopFileName));
|
|
connect(watcher, &QDBusPendingCallWatcher::finished, this, [=] {
|
|
QDBusPendingReply<QDBusObjectPath> reply = *watcher;
|
|
watcher->deleteLater();
|
|
if (!reply.isValid()) {
|
|
genericErrorOccured(QStringLiteral("Error while calling objectPath of added application") + desktopFileName, reply.error());
|
|
return;
|
|
}
|
|
KGlobalAccelComponentInterface component(m_globalAccelInterface->service(), reply.value().path(), m_globalAccelInterface->connection());
|
|
auto infoWatcher = new QDBusPendingCallWatcher(component.allShortcutInfos());
|
|
connect(infoWatcher, &QDBusPendingCallWatcher::finished, this, [=] {
|
|
QDBusPendingReply<QList<KGlobalShortcutInfo>> infoReply = *infoWatcher;
|
|
infoWatcher->deleteLater();
|
|
if (!infoReply.isValid()) {
|
|
genericErrorOccured(QStringLiteral("Error while calling allShortCutInfos on new component") + desktopFileName, infoReply.error());
|
|
return;
|
|
}
|
|
if (infoReply.value().isEmpty()) {
|
|
qCWarning(KCMKEYS()) << "New component has no shortcuts:" << reply.value().path();
|
|
Q_EMIT errorOccured(i18nc("%1 is the name of an application", "Error while adding %1, it seems it has no actions."));
|
|
}
|
|
qCDebug(KCMKEYS) << "inserting at " << pos - m_components.begin();
|
|
beginInsertRows(QModelIndex(), pos - m_components.begin(), pos - m_components.begin());
|
|
auto c = loadComponent(infoReply.value());
|
|
m_components.insert(pos, c);
|
|
endInsertRows();
|
|
});
|
|
});
|
|
}
|
|
|
|
void GlobalAccelModel::removeComponent(const Component &component)
|
|
{
|
|
const QString &uniqueName = component.id;
|
|
auto componentReply = m_globalAccelInterface->getComponent(uniqueName);
|
|
componentReply.waitForFinished();
|
|
if (!componentReply.isValid()) {
|
|
genericErrorOccured(QStringLiteral("Error while calling objectPath of component") + uniqueName, componentReply.error());
|
|
return;
|
|
}
|
|
KGlobalAccelComponentInterface componentInterface(m_globalAccelInterface->service(), componentReply.value().path(), m_globalAccelInterface->connection());
|
|
qCDebug(KCMKEYS) << "Cleaning up component at" << componentReply.value();
|
|
auto cleanUpReply = componentInterface.cleanUp();
|
|
cleanUpReply.waitForFinished();
|
|
if (!cleanUpReply.isValid()) {
|
|
genericErrorOccured(QStringLiteral("Error while calling cleanUp of component") + uniqueName, cleanUpReply.error());
|
|
return;
|
|
}
|
|
auto it = std::find_if(m_components.begin(), m_components.end(), [&](const Component &c) {
|
|
return c.id == uniqueName;
|
|
});
|
|
const int row = it - m_components.begin();
|
|
beginRemoveRows(QModelIndex(), row, row);
|
|
m_components.remove(row);
|
|
endRemoveRows();
|
|
}
|
|
|
|
void GlobalAccelModel::genericErrorOccured(const QString &description, const QDBusError &error)
|
|
{
|
|
qCCritical(KCMKEYS) << description;
|
|
if (error.isValid()) {
|
|
qCCritical(KCMKEYS) << error.name() << error.message();
|
|
}
|
|
Q_EMIT this->errorOccured(i18n("Error while communicating with the global shortcuts service"));
|
|
}
|