/* 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 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 presentWindowsName = QStringLiteral("org.kde.KWin.PresentWindows"); static const QString presentWindowsPath = QStringLiteral("/org/kde/KWin/PresentWindows"); static const QString &presentWindowsInterface = presentWindowsName; Backend::Backend(QObject *parent) : QObject(parent) , m_highlightWindows(false) , m_actionGroup(new QActionGroup(this)) { m_canPresentWindows = QDBusConnection::sessionBus().interface()->isServiceRegistered(presentWindowsName); auto watcher = new QDBusServiceWatcher(presentWindowsName, QDBusConnection::sessionBus(), QDBusServiceWatcher::WatchForRegistration | QDBusServiceWatcher::WatchForUnregistration, this); connect(watcher, &QDBusServiceWatcher::serviceRegistered, this, [this] { m_canPresentWindows = true; Q_EMIT canPresentWindowsChanged(); }); connect(watcher, &QDBusServiceWatcher::serviceUnregistered, this, [this] { m_canPresentWindows = false; Q_EMIT canPresentWindowsChanged(); }); } Backend::~Backend() { } QQuickItem *Backend::taskManagerItem() const { return m_taskManagerItem; } void Backend::setTaskManagerItem(QQuickItem *item) { if (item != m_taskManagerItem) { m_taskManagerItem = item; Q_EMIT taskManagerItemChanged(); } } QQuickWindow *Backend::groupDialog() const { return m_groupDialog; } void Backend::setGroupDialog(QQuickWindow *dialog) { if (dialog != m_groupDialog) { m_groupDialog = dialog; Q_EMIT groupDialogChanged(); } } 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; 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 : qAsConst(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; QScopedPointer 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) { if (!parent) { return QVariantList(); } QUrl desktopEntryUrl = tryDecodeApplicationsUrl(launcherUrl); if (!desktopEntryUrl.isValid() || !desktopEntryUrl.isLocalFile() || !KDesktopFile::isDesktopFile(desktopEntryUrl.toLocalFile())) { return QVariantList(); } QVariantList actions; 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::files() | Activity::current() | Url::file(); ResultSet results(query); ResultSet::const_iterator resultIt = results.begin(); int actionCount = 0; while (actionCount < 5 && resultIt != results.end()) { const QString resource = (*resultIt).resource(); const QString mimetype = (*resultIt).mimetype(); ++resultIt; const QUrl url = QUrl::fromLocalFile(resource); if (!url.isValid()) { continue; } const KFileItem fileItem(url, KFileItem::SkipMimeTypeFromContent); QAction *action = new QAction(parent); action->setText(url.fileName()); action->setIcon(QIcon::fromTheme(fileItem.iconName(), QIcon::fromTheme(QStringLiteral("unknown")))); action->setProperty("agent", storageId); action->setProperty("entryPath", desktopEntryUrl); action->setProperty("mimeType", mimetype); action->setData(resource); connect(action, &QAction::triggered, this, &Backend::handleRecentDocumentAction); actions << QVariant::fromValue(action); ++actionCount; } if (actionCount > 0) { QAction *separatorAction = new QAction(parent); separatorAction->setSeparator(true); actions << QVariant::fromValue(separatorAction); QAction *action = new QAction(parent); action->setText(i18n("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 QString resource = action->data().toString(); if (desktopPath.isEmpty() || resource.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({QUrl(resource)}); 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::ungrabMouse(QQuickItem *item) const { // this is a workaround where Qt will fail to realize a mouse has been released // this happens if a window which does not accept focus spawns a new window that takes focus and X grab // whilst the mouse is depressed // https://bugreports.qt.io/browse/QTBUG-59044 // this causes the next click to go missing // by releasing manually we avoid that situation auto ungrabMouseHack = [item]() { if (item && item->window() && item->window()->mouseGrabberItem()) { item->window()->mouseGrabberItem()->ungrabMouse(); } }; // pre 5.8.0 QQuickWindow code is "item->grabMouse(); sendEvent(item, mouseEvent)" // post 5.8.0 QQuickWindow code is sendEvent(item, mouseEvent); item->grabMouse() if (QVersionNumber::fromString(QString::fromLatin1(qVersion())) > QVersionNumber(5, 8, 0)) { QTimer::singleShot(0, item, ungrabMouseHack); } else { ungrabMouseHack(); } // end workaround } bool Backend::canPresentWindows() const { return m_canPresentWindows; } void Backend::presentWindows(const QVariant &_winIds) { if (m_windowsToHighlight.count()) { m_windowsToHighlight.clear(); updateWindowHighlight(); } auto message = QDBusMessage::createMethodCall(presentWindowsName, presentWindowsPath, presentWindowsInterface, QStringLiteral("presentWindows")); 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(); } QList Backend::jsonArrayToUrlList(const QJsonArray &array) const { QList urls; urls.reserve(array.count()); for (auto it = array.constBegin(), end = array.constEnd(); it != end; ++it) { urls << QUrl(it->toString()); } return urls; } 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); }