You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
464 lines
14 KiB
464 lines
14 KiB
/*************************************************************************** |
|
* Copyright (C) 2013 by Aurélien Gâteau <agateau@kde.org> * |
|
* Copyright (C) 2014 by Eike Hein <hein@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, write to the * |
|
* Free Software Foundation, Inc., * |
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA . * |
|
***************************************************************************/ |
|
|
|
#include "actionlist.h" |
|
#include "menuentryeditor.h" |
|
|
|
#include <config-appstream.h> |
|
|
|
#include <QApplication> |
|
#include <QDesktopServices> |
|
#include <QDir> |
|
#include <QStandardPaths> |
|
|
|
#include <KApplicationTrader> |
|
#include <KIO/ApplicationLauncherJob> |
|
#include <KLocalizedString> |
|
#include <KNotificationJobUiDelegate> |
|
#include <KPropertiesDialog> |
|
#include <KProtocolInfo> |
|
|
|
#include <KActivities/Stats/Cleaning> |
|
#include <KActivities/Stats/ResultSet> |
|
#include <KActivities/Stats/Terms> |
|
|
|
#include "containmentinterface.h" |
|
|
|
#ifdef HAVE_APPSTREAMQT |
|
#include <AppStreamQt/pool.h> |
|
#endif |
|
|
|
namespace KAStats = KActivities::Stats; |
|
|
|
using namespace KAStats; |
|
using namespace KAStats::Terms; |
|
|
|
namespace Kicker |
|
{ |
|
QVariantMap createActionItem(const QString &label, const QString &icon, const QString &actionId, const QVariant &argument) |
|
{ |
|
QVariantMap map; |
|
|
|
map[QStringLiteral("text")] = label; |
|
map[QStringLiteral("icon")] = icon; |
|
map[QStringLiteral("actionId")] = actionId; |
|
|
|
if (argument.isValid()) { |
|
map[QStringLiteral("actionArgument")] = argument; |
|
} |
|
|
|
return map; |
|
} |
|
|
|
QVariantMap createTitleActionItem(const QString &label) |
|
{ |
|
QVariantMap map; |
|
|
|
map[QStringLiteral("text")] = label; |
|
map[QStringLiteral("type")] = QStringLiteral("title"); |
|
|
|
return map; |
|
} |
|
|
|
QVariantMap createSeparatorActionItem() |
|
{ |
|
QVariantMap map; |
|
|
|
map[QStringLiteral("type")] = QStringLiteral("separator"); |
|
|
|
return map; |
|
} |
|
|
|
QVariantList createActionListForFileItem(const KFileItem &fileItem) |
|
{ |
|
QVariantList list; |
|
|
|
const KService::List services = KApplicationTrader::queryByMimeType(fileItem.mimetype()); |
|
|
|
if (!services.isEmpty()) { |
|
list << createTitleActionItem(i18n("Open with:")); |
|
|
|
for (const KService::Ptr service : services) { |
|
const QString text = service->name().replace(QLatin1Char('&'), QStringLiteral("&&")); |
|
QVariantMap item = createActionItem(text, service->icon(), QStringLiteral("_kicker_fileItem_openWith"), service->entryPath()); |
|
|
|
list << item; |
|
} |
|
|
|
list << createSeparatorActionItem(); |
|
} |
|
|
|
const QVariantMap &propertiesItem = |
|
createActionItem(i18n("Properties"), QStringLiteral("document-properties"), QStringLiteral("_kicker_fileItem_properties")); |
|
list << propertiesItem; |
|
|
|
return list; |
|
} |
|
|
|
bool handleFileItemAction(const KFileItem &fileItem, const QString &actionId, const QVariant &argument, bool *close) |
|
{ |
|
if (actionId == QLatin1String("_kicker_fileItem_properties")) { |
|
KPropertiesDialog *dlg = new KPropertiesDialog(fileItem, QApplication::activeWindow()); |
|
dlg->setAttribute(Qt::WA_DeleteOnClose); |
|
dlg->show(); |
|
|
|
*close = false; |
|
|
|
return true; |
|
} |
|
|
|
if (actionId == QLatin1String("_kicker_fileItem_openWith")) { |
|
const QString path = argument.toString(); |
|
const KService::Ptr service = KService::serviceByDesktopPath(path); |
|
|
|
if (!service) { |
|
return false; |
|
} |
|
|
|
auto *job = new KIO::ApplicationLauncherJob(service); |
|
job->setUrls({fileItem.url()}); |
|
job->setUiDelegate(new KNotificationJobUiDelegate(KJobUiDelegate::AutoHandlingEnabled)); |
|
job->start(); |
|
|
|
*close = true; |
|
|
|
return true; |
|
} |
|
|
|
return false; |
|
} |
|
|
|
QVariantList createAddLauncherActionList(QObject *appletInterface, const KService::Ptr &service) |
|
{ |
|
QVariantList actionList; |
|
if (!service) { |
|
return actionList; |
|
} |
|
|
|
if (ContainmentInterface::mayAddLauncher(appletInterface, ContainmentInterface::Desktop)) { |
|
QVariantMap addToDesktopAction = Kicker::createActionItem(i18n("Add to Desktop"), QStringLiteral("list-add"), QStringLiteral("addToDesktop")); |
|
actionList << addToDesktopAction; |
|
} |
|
|
|
if (ContainmentInterface::mayAddLauncher(appletInterface, ContainmentInterface::Panel)) { |
|
QVariantMap addToPanelAction = Kicker::createActionItem(i18n("Add to Panel (Widget)"), QStringLiteral("list-add"), QStringLiteral("addToPanel")); |
|
actionList << addToPanelAction; |
|
} |
|
|
|
if (service && ContainmentInterface::mayAddLauncher(appletInterface, ContainmentInterface::TaskManager, Kicker::resolvedServiceEntryPath(service))) { |
|
QVariantMap addToTaskManagerAction = Kicker::createActionItem(i18n("Pin to Task Manager"), QStringLiteral("pin"), QStringLiteral("addToTaskManager")); |
|
actionList << addToTaskManagerAction; |
|
} |
|
|
|
return actionList; |
|
} |
|
|
|
bool handleAddLauncherAction(const QString &actionId, QObject *appletInterface, const KService::Ptr &service) |
|
{ |
|
if (!service) { |
|
return false; |
|
} |
|
|
|
if (actionId == QLatin1String("addToDesktop")) { |
|
if (ContainmentInterface::mayAddLauncher(appletInterface, ContainmentInterface::Desktop)) { |
|
ContainmentInterface::addLauncher(appletInterface, ContainmentInterface::Desktop, Kicker::resolvedServiceEntryPath(service)); |
|
} |
|
return true; |
|
} else if (actionId == QLatin1String("addToPanel")) { |
|
if (ContainmentInterface::mayAddLauncher(appletInterface, ContainmentInterface::Panel)) { |
|
ContainmentInterface::addLauncher(appletInterface, ContainmentInterface::Panel, Kicker::resolvedServiceEntryPath(service)); |
|
} |
|
return true; |
|
} else if (actionId == QLatin1String("addToTaskManager")) { |
|
if (ContainmentInterface::mayAddLauncher(appletInterface, ContainmentInterface::TaskManager, Kicker::resolvedServiceEntryPath(service))) { |
|
ContainmentInterface::addLauncher(appletInterface, ContainmentInterface::TaskManager, Kicker::resolvedServiceEntryPath(service)); |
|
} |
|
return true; |
|
} |
|
|
|
return false; |
|
} |
|
|
|
QString storageIdFromService(KService::Ptr service) |
|
{ |
|
QString storageId = service->storageId(); |
|
|
|
if (storageId.endsWith(QLatin1String(".desktop"))) { |
|
storageId = storageId.left(storageId.length() - 8); |
|
} |
|
|
|
return storageId; |
|
} |
|
|
|
QVariantList jumpListActions(KService::Ptr service) |
|
{ |
|
QVariantList list; |
|
|
|
if (!service) { |
|
return list; |
|
} |
|
|
|
// Add frequently used settings modules similar to SystemSetting's overview page. |
|
if (service->storageId() == QLatin1String("systemsettings.desktop")) { |
|
list = systemSettingsActions(); |
|
|
|
if (!list.isEmpty()) { |
|
return list; |
|
} |
|
} |
|
|
|
const auto &actions = service->actions(); |
|
for (const KServiceAction &action : actions) { |
|
if (action.text().isEmpty() || action.exec().isEmpty()) { |
|
continue; |
|
} |
|
|
|
QVariantMap item = createActionItem(action.text(), action.icon(), QStringLiteral("_kicker_jumpListAction"), action.exec()); |
|
|
|
list << item; |
|
} |
|
|
|
return list; |
|
} |
|
|
|
QVariantList systemSettingsActions() |
|
{ |
|
QVariantList list; |
|
|
|
auto query = AllResources | Agent(QStringLiteral("org.kde.systemsettings")) | HighScoredFirst | Limit(5); |
|
|
|
ResultSet results(query); |
|
|
|
QStringList ids; |
|
for (const ResultSet::Result &result : results) { |
|
ids << QUrl(result.resource()).path(); |
|
} |
|
|
|
if (ids.count() < 5) { |
|
// We'll load the default set of settings from its jump list actions. |
|
return list; |
|
} |
|
|
|
for (const QString &id : qAsConst(ids)) { |
|
KService::Ptr service = KService::serviceByStorageId(id); |
|
if (!service || !service->isValid()) { |
|
continue; |
|
} |
|
|
|
list << createActionItem(service->name(), service->icon(), QStringLiteral("_kicker_jumpListAction"), service->exec()); |
|
} |
|
|
|
return list; |
|
} |
|
|
|
QVariantList recentDocumentActions(KService::Ptr service) |
|
{ |
|
QVariantList list; |
|
|
|
if (!service) { |
|
return list; |
|
} |
|
|
|
const QString storageId = storageIdFromService(service); |
|
|
|
if (storageId.isEmpty()) { |
|
return list; |
|
} |
|
|
|
// clang-format off |
|
auto query = UsedResources |
|
| RecentlyUsedFirst |
|
| Agent(storageId) |
|
| Type::any() |
|
| Activity::current() |
|
| Url::file(); |
|
// clang-format on |
|
|
|
ResultSet results(query); |
|
|
|
ResultSet::const_iterator resultIt; |
|
resultIt = results.begin(); |
|
|
|
while (list.count() < 6 && resultIt != results.end()) { |
|
const QString resource = (*resultIt).resource(); |
|
++resultIt; |
|
|
|
const QUrl url(resource); |
|
|
|
if (!url.isValid()) { |
|
continue; |
|
} |
|
|
|
const KFileItem fileItem(url); |
|
|
|
if (!fileItem.isFile()) { |
|
continue; |
|
} |
|
|
|
if (list.isEmpty()) { |
|
list << createTitleActionItem(i18n("Recent Files")); |
|
} |
|
|
|
QVariantMap item = createActionItem(url.fileName(), fileItem.iconName(), QStringLiteral("_kicker_recentDocument"), resource); |
|
|
|
list << item; |
|
} |
|
|
|
if (!list.isEmpty()) { |
|
QVariantMap forgetAction = |
|
createActionItem(i18n("Forget Recent Files"), QStringLiteral("edit-clear-history"), QStringLiteral("_kicker_forgetRecentDocuments")); |
|
list << forgetAction; |
|
} |
|
|
|
return list; |
|
} |
|
|
|
bool handleRecentDocumentAction(KService::Ptr service, const QString &actionId, const QVariant &_argument) |
|
{ |
|
if (!service) { |
|
return false; |
|
} |
|
|
|
if (actionId == QLatin1String("_kicker_forgetRecentDocuments")) { |
|
const QString storageId = storageIdFromService(service); |
|
|
|
if (storageId.isEmpty()) { |
|
return false; |
|
} |
|
|
|
// clang-format off |
|
auto query = UsedResources |
|
| Agent(storageId) |
|
| Type::any() |
|
| Activity::current() |
|
| Url::file(); |
|
// clang-format on |
|
|
|
KAStats::forgetResources(query); |
|
|
|
return false; |
|
} |
|
|
|
QString argument = _argument.toString(); |
|
|
|
if (argument.isEmpty()) { |
|
return false; |
|
} |
|
|
|
auto *job = new KIO::ApplicationLauncherJob(service); |
|
job->setUrls({QUrl(argument)}); |
|
job->setUiDelegate(new KNotificationJobUiDelegate(KJobUiDelegate::AutoHandlingEnabled)); |
|
return job->exec(); |
|
} |
|
|
|
Q_GLOBAL_STATIC(MenuEntryEditor, menuEntryEditor) |
|
|
|
bool canEditApplication(const KService::Ptr &service) |
|
{ |
|
return (service->isApplication() && menuEntryEditor->canEdit(service->entryPath())); |
|
} |
|
|
|
void editApplication(const QString &entryPath, const QString &menuId) |
|
{ |
|
menuEntryEditor->edit(entryPath, menuId); |
|
} |
|
|
|
QVariantList editApplicationAction(const KService::Ptr &service) |
|
{ |
|
QVariantList actionList; |
|
|
|
if (canEditApplication(service)) { |
|
// TODO: Using the KMenuEdit icon might be misleading. |
|
QVariantMap editAction = Kicker::createActionItem(i18n("Edit Application..."), QStringLiteral("kmenuedit"), QStringLiteral("editApplication")); |
|
actionList << editAction; |
|
} |
|
|
|
return actionList; |
|
} |
|
|
|
bool handleEditApplicationAction(const QString &actionId, const KService::Ptr &service) |
|
{ |
|
if (service && actionId == QLatin1String("editApplication") && canEditApplication(service)) { |
|
Kicker::editApplication(service->entryPath(), service->menuId()); |
|
|
|
return true; |
|
} |
|
|
|
return false; |
|
} |
|
|
|
#ifdef HAVE_APPSTREAMQT |
|
Q_GLOBAL_STATIC(AppStream::Pool, appstreamPool) |
|
#endif |
|
|
|
QVariantList appstreamActions(const KService::Ptr &service) |
|
{ |
|
QVariantList ret; |
|
|
|
#ifdef HAVE_APPSTREAMQT |
|
const KService::Ptr appStreamHandler = KApplicationTrader::preferredService(QStringLiteral("x-scheme-handler/appstream")); |
|
|
|
// Don't show action if we can't find any app to handle appstream:// URLs. |
|
if (!appStreamHandler) { |
|
if (!KProtocolInfo::isHelperProtocol(QStringLiteral("appstream")) || KProtocolInfo::exec(QStringLiteral("appstream")).isEmpty()) { |
|
return ret; |
|
} |
|
} |
|
|
|
if (!appstreamPool.exists()) { |
|
appstreamPool->load(); |
|
} |
|
|
|
const auto components = appstreamPool->componentsById(service->desktopEntryName() + QLatin1String(".desktop")); |
|
for (const auto &component : components) { |
|
const QString componentId = component.id(); |
|
|
|
QVariantMap appstreamAction = Kicker::createActionItem(i18nc("@action opens a software center with the application", "Uninstall or Manage Add-Ons..."), |
|
appStreamHandler->icon(), |
|
"manageApplication", |
|
QVariant(QLatin1String("appstream://") + componentId)); |
|
ret << appstreamAction; |
|
} |
|
#else |
|
Q_UNUSED(service) |
|
#endif |
|
|
|
return ret; |
|
} |
|
|
|
bool handleAppstreamActions(const QString &actionId, const QVariant &argument) |
|
{ |
|
if (actionId == QLatin1String("manageApplication")) { |
|
return QDesktopServices::openUrl(QUrl(argument.toString())); |
|
} |
|
|
|
return false; |
|
} |
|
|
|
QString resolvedServiceEntryPath(const KService::Ptr &service) |
|
{ |
|
QString path = service->entryPath(); |
|
if (!QDir::isAbsolutePath(path)) { |
|
path = QStandardPaths::locate(QStandardPaths::GenericDataLocation, QLatin1String("kservices5/") + path); |
|
} |
|
return path; |
|
} |
|
|
|
}
|
|
|