/* SPDX-FileCopyrightText: 2012-2016 Eike Hein SPDX-License-Identifier: GPL-2.0-or-later */ #include "backend.h" #include "log_settings.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace KAStats = KActivities::Stats; using namespace KAStats; using namespace KAStats::Terms; static const QString highlightWindowName = QStringLiteral("org.kde.KWin.HighlightWindow"); static const QString highlightWindowPath = QStringLiteral("/org/kde/KWin/HighlightWindow"); static const QString &highlightWindowInterface = highlightWindowName; static const QString appViewName = QStringLiteral("org.kde.KWin.Effect.WindowView1"); static const QString appViewPath = QStringLiteral("/org/kde/KWin/Effect/WindowView1"); static const QString &appViewInterface = appViewName; static constexpr int NoApplications = 2; // kactivitymanager StatsPlugin WhatToRemember. Backend::Backend(QObject *parent) : QObject(parent) , m_highlightWindows(false) , m_actionGroup(new QActionGroup(this)) , m_activityManagerPluginsSettingsWatcher(KConfigWatcher::create(m_activityManagerPluginsSettings.sharedConfig())) { connect(m_activityManagerPluginsSettingsWatcher.get(), &KConfigWatcher::configChanged, this, [this](const KConfigGroup &group, const QByteArrayList &names) { if (group.name() == QLatin1String("Plugin-org.kde.ActivityManager.Resources.Scoring") && names.contains(QByteArrayLiteral("what-to-remember"))) { m_activityManagerPluginsSettings.load(); } }); } Backend::~Backend() { } bool Backend::highlightWindows() const { return m_highlightWindows; } void Backend::setHighlightWindows(bool highlight) { if (highlight != m_highlightWindows) { m_highlightWindows = highlight; updateWindowHighlight(); Q_EMIT highlightWindowsChanged(); } } QUrl Backend::tryDecodeApplicationsUrl(const QUrl &launcherUrl) { if (launcherUrl.isValid() && launcherUrl.scheme() == QLatin1String("applications")) { const KService::Ptr service = KService::serviceByMenuId(launcherUrl.path()); if (service) { return QUrl::fromLocalFile(service->entryPath()); } } return launcherUrl; } QStringList Backend::applicationCategories(const QUrl &launcherUrl) { const QUrl desktopEntryUrl = tryDecodeApplicationsUrl(launcherUrl); if (!desktopEntryUrl.isValid() || !desktopEntryUrl.isLocalFile() || !KDesktopFile::isDesktopFile(desktopEntryUrl.toLocalFile())) { return QStringList(); } KDesktopFile desktopFile(desktopEntryUrl.toLocalFile()); // Since we can't have dynamic jump list actions, at least add the user's "Places" for file managers. return desktopFile.desktopGroup().readXdgListEntry(QStringLiteral("Categories")); } QVariantList Backend::jumpListActions(const QUrl &launcherUrl, QObject *parent) { QVariantList actions; if (!parent) { return actions; } QUrl desktopEntryUrl = tryDecodeApplicationsUrl(launcherUrl); if (!desktopEntryUrl.isValid() || !desktopEntryUrl.isLocalFile() || !KDesktopFile::isDesktopFile(desktopEntryUrl.toLocalFile())) { return actions; } const KService::Ptr service = KService::serviceByDesktopPath(desktopEntryUrl.toLocalFile()); if (!service) { return actions; } if (service->storageId() == QLatin1String("systemsettings.desktop")) { actions = systemSettingsActions(parent); if (!actions.isEmpty()) { return actions; } } const auto jumpListActions = service->actions(); for (const KServiceAction &serviceAction : jumpListActions) { if (serviceAction.noDisplay()) { continue; } QAction *action = new QAction(parent); action->setText(serviceAction.text()); action->setIcon(QIcon::fromTheme(serviceAction.icon())); if (serviceAction.isSeparator()) { action->setSeparator(true); } connect(action, &QAction::triggered, this, [serviceAction]() { auto *job = new KIO::ApplicationLauncherJob(serviceAction); auto *delegate = new KNotificationJobUiDelegate; delegate->setAutoErrorHandlingEnabled(true); job->setUiDelegate(delegate); job->start(); }); actions << QVariant::fromValue(action); } return actions; } QVariantList Backend::systemSettingsActions(QObject *parent) const { QVariantList actions; if (m_activityManagerPluginsSettings.whatToRemember() == NoApplications) { return actions; } 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 actions; } for (const QString &id : std::as_const(ids)) { KService::Ptr service = KService::serviceByStorageId(id); if (!service || !service->isValid()) { continue; } QAction *action = new QAction(parent); action->setText(service->name()); action->setIcon(QIcon::fromTheme(service->icon())); connect(action, &QAction::triggered, this, [service]() { auto *job = new KIO::ApplicationLauncherJob(service); auto *delegate = new KNotificationJobUiDelegate; delegate->setAutoErrorHandlingEnabled(true); job->setUiDelegate(delegate); job->start(); }); actions << QVariant::fromValue(action); } return actions; } QVariantList Backend::placesActions(const QUrl &launcherUrl, bool showAllPlaces, QObject *parent) { if (!parent) { return QVariantList(); } QUrl desktopEntryUrl = tryDecodeApplicationsUrl(launcherUrl); if (!desktopEntryUrl.isValid() || !desktopEntryUrl.isLocalFile() || !KDesktopFile::isDesktopFile(desktopEntryUrl.toLocalFile())) { return QVariantList(); } QVariantList actions; // Since we can't have dynamic jump list actions, at least add the user's "Places" for file managers. if (!applicationCategories(launcherUrl).contains(QLatin1String("FileManager"))) { return actions; } QString previousGroup; QMenu *subMenu = nullptr; std::unique_ptr placesModel(new KFilePlacesModel()); for (int i = 0; i < placesModel->rowCount(); ++i) { QModelIndex idx = placesModel->index(i, 0); if (placesModel->isHidden(idx)) { continue; } const QString &title = idx.data(Qt::DisplayRole).toString(); const QIcon &icon = idx.data(Qt::DecorationRole).value(); const QUrl &url = idx.data(KFilePlacesModel::UrlRole).toUrl(); QAction *placeAction = new QAction(icon, title, parent); connect(placeAction, &QAction::triggered, this, [url, desktopEntryUrl] { KService::Ptr service = KService::serviceByDesktopPath(desktopEntryUrl.toLocalFile()); if (!service) { return; } auto *job = new KIO::ApplicationLauncherJob(service); auto *delegate = new KNotificationJobUiDelegate; delegate->setAutoErrorHandlingEnabled(true); job->setUiDelegate(delegate); job->setUrls({url}); job->start(); }); const QString &groupName = idx.data(KFilePlacesModel::GroupRole).toString(); if (previousGroup.isEmpty()) { // Skip first group heading. previousGroup = groupName; } // Put all subsequent categories into a submenu. if (previousGroup != groupName) { QAction *subMenuAction = new QAction(groupName, parent); subMenu = new QMenu(); // Cannot parent a QMenu to a QAction, need to delete it manually. connect(parent, &QObject::destroyed, subMenu, &QObject::deleteLater); subMenuAction->setMenu(subMenu); actions << QVariant::fromValue(subMenuAction); previousGroup = groupName; } if (subMenu) { subMenu->addAction(placeAction); } else { actions << QVariant::fromValue(placeAction); } } // There is nothing more frustrating than having a "More" entry that ends up showing just one or two // additional entries. Therefore we truncate to max. 5 entries only if there are more than 7 in total. if (!showAllPlaces && actions.count() > 7) { const int totalActionCount = actions.count(); while (actions.count() > 5) { actions.removeLast(); } QAction *action = new QAction(parent); action->setText(i18ncp("Show all user Places", "%1 more Place", "%1 more Places", totalActionCount - actions.count())); connect(action, &QAction::triggered, this, &Backend::showAllPlaces); actions << QVariant::fromValue(action); } return actions; } QVariantList Backend::recentDocumentActions(const QUrl &launcherUrl, QObject *parent) { QVariantList actions; if (!parent) { return actions; } if (m_activityManagerPluginsSettings.whatToRemember() == NoApplications) { return actions; } QUrl desktopEntryUrl = tryDecodeApplicationsUrl(launcherUrl); if (!desktopEntryUrl.isValid() || !desktopEntryUrl.isLocalFile() || !KDesktopFile::isDesktopFile(desktopEntryUrl.toLocalFile())) { return QVariantList(); } QString desktopName = desktopEntryUrl.fileName(); QString storageId = desktopName; if (storageId.endsWith(QLatin1String(".desktop"))) { storageId = storageId.left(storageId.length() - 8); } auto query = UsedResources | RecentlyUsedFirst | Agent(storageId) | Type::any() | Activity::current(); ResultSet results(query); ResultSet::const_iterator resultIt = results.begin(); int actionCount = 0; bool allFolders = true; bool allDownloads = true; bool allRemoteWithoutFileName = true; const QString downloadsPath = QStandardPaths::writableLocation(QStandardPaths::DownloadLocation); while (actionCount < 5 && resultIt != results.end()) { const QString resource = (*resultIt).resource(); const QString mimetype = (*resultIt).mimetype(); const QUrl url = (*resultIt).url(); ++resultIt; if (!url.isValid()) { continue; } allFolders = allFolders && mimetype == QLatin1String("inode/directory"); allDownloads = allDownloads && url.toLocalFile().startsWith(downloadsPath); allRemoteWithoutFileName = allRemoteWithoutFileName && !url.isLocalFile() && url.fileName().isEmpty(); QString name = url.fileName(); if (name.isEmpty()) { name = url.toDisplayString(); } QString iconName; const QString protocol = url.scheme(); if (!KProtocolInfo::isKnownProtocol(protocol) || KProtocolInfo::isHelperProtocol(protocol)) { const KService::Ptr service = KApplicationTrader::preferredService(QLatin1String("x-scheme-handler/") + protocol); if (service) { iconName = service->icon(); } else if (KProtocolInfo::isKnownProtocol(protocol)) { Q_ASSERT(KProtocolInfo::isHelperProtocol(protocol)); iconName = KProtocolInfo::icon(protocol); } else { // Should not happen? continue; } } else { const KFileItem fileItem(url, mimetype); iconName = fileItem.iconName(); } QAction *action = new QAction(parent); action->setText(name); action->setIcon(QIcon::fromTheme(iconName, QIcon::fromTheme(QStringLiteral("unknown")))); action->setProperty("agent", storageId); action->setProperty("entryPath", desktopEntryUrl); action->setProperty("mimeType", mimetype); action->setData(url); connect(action, &QAction::triggered, this, &Backend::handleRecentDocumentAction); actions << QVariant::fromValue(action); ++actionCount; } if (actionCount > 0) { // Overrides section heading on QML side if (allDownloads) { actions.prepend(i18n("Recent Downloads")); } else if (allRemoteWithoutFileName) { actions.prepend(i18n("Recent Connections")); } else if (allFolders) { actions.prepend(i18n("Recent Places")); } QAction *separatorAction = new QAction(parent); separatorAction->setSeparator(true); actions << QVariant::fromValue(separatorAction); QAction *action = new QAction(parent); if (allDownloads) { action->setText(i18nc("@action:inmenu", "Forget Recent Downloads")); } else if (allRemoteWithoutFileName) { action->setText(i18nc("@action:inmenu", "Forget Recent Connections")); } else if (allFolders) { action->setText(i18nc("@action:inmenu", "Forget Recent Places")); } else { action->setText(i18nc("@action:inmenu", "Forget Recent Files")); } action->setIcon(QIcon::fromTheme(QStringLiteral("edit-clear-history"))); action->setProperty("agent", storageId); connect(action, &QAction::triggered, this, &Backend::handleRecentDocumentAction); actions << QVariant::fromValue(action); } return actions; } void Backend::handleRecentDocumentAction() const { const QAction *action = qobject_cast(sender()); if (!action) { return; } const QString agent = action->property("agent").toString(); if (agent.isEmpty()) { return; } const QString desktopPath = action->property("entryPath").toUrl().toLocalFile(); const QUrl url = action->data().toUrl(); if (desktopPath.isEmpty() || url.isEmpty()) { auto query = UsedResources | Agent(agent) | Type::any() | Activity::current() | Url::file(); KAStats::forgetResources(query); return; } KService::Ptr service = KService::serviceByDesktopPath(desktopPath); if (!service) { return; } // prevents using a service file that does not support opening a mime type for a file it created // for instance spectacle const auto mimetype = action->property("mimeType").toString(); if (!mimetype.isEmpty()) { if (!service->hasMimeType(mimetype)) { // needs to find the application that supports this mimetype service = KApplicationTrader::preferredService(mimetype); if (!service) { // no service found to handle the mimetype return; } else { qCWarning(TASKMANAGER_DEBUG) << "Preventing the file to open with " << service->desktopEntryName() << "no alternative found"; } } } auto *job = new KIO::ApplicationLauncherJob(service); auto *delegate = new KNotificationJobUiDelegate; delegate->setAutoErrorHandlingEnabled(true); job->setUiDelegate(delegate); job->setUrls({url}); job->start(); } void Backend::setActionGroup(QAction *action) const { if (action) { action->setActionGroup(m_actionGroup); } } QRect Backend::globalRect(QQuickItem *item) const { if (!item || !item->window()) { return QRect(); } QRect iconRect(item->x(), item->y(), item->width(), item->height()); iconRect.moveTopLeft(item->parentItem()->mapToScene(iconRect.topLeft()).toPoint()); iconRect.moveTopLeft(item->window()->mapToGlobal(iconRect.topLeft())); return iconRect; } void Backend::activateWindowView(const QVariant &_winIds) { if (!m_windowsToHighlight.isEmpty()) { m_windowsToHighlight.clear(); updateWindowHighlight(); } auto message = QDBusMessage::createMethodCall(appViewName, appViewPath, appViewInterface, QStringLiteral("activate")); message << _winIds.toStringList(); QDBusConnection::sessionBus().asyncCall(message); } bool Backend::isApplication(const QUrl &url) const { if (!url.isValid() || !url.isLocalFile()) { return false; } const QString &localPath = url.toLocalFile(); if (!KDesktopFile::isDesktopFile(localPath)) { return false; } KDesktopFile desktopFile(localPath); return desktopFile.hasApplicationType(); } void Backend::cancelHighlightWindows() { m_windowsToHighlight.clear(); updateWindowHighlight(); } qint64 Backend::parentPid(qint64 pid) const { KSysGuard::Processes procs; procs.updateOrAddProcess(pid); KSysGuard::Process *proc = procs.getProcess(pid); if (!proc) { return -1; } int parentPid = proc->parentPid(); if (parentPid != -1) { procs.updateOrAddProcess(parentPid); KSysGuard::Process *parentProc = procs.getProcess(parentPid); if (!parentProc) { return -1; } if (!proc->cGroup().isEmpty() && parentProc->cGroup() == proc->cGroup()) { return parentProc->pid(); } } return -1; } void Backend::windowsHovered(const QVariant &_winIds, bool hovered) { m_windowsToHighlight.clear(); if (hovered) { m_windowsToHighlight = _winIds.toStringList(); } // Avoid flickering when scrolling in the tooltip QTimer::singleShot(0, this, &Backend::updateWindowHighlight); } void Backend::updateWindowHighlight() { if (!m_highlightWindows) { return; } auto message = QDBusMessage::createMethodCall(highlightWindowName, highlightWindowPath, highlightWindowInterface, QStringLiteral("highlightWindows")); message << m_windowsToHighlight; QDBusConnection::sessionBus().asyncCall(message); } #include "moc_backend.cpp"