3
0
mirror of https://github.com/Qortal/Brooklyn.git synced 2025-02-12 02:05:54 +00:00
Scare Crowe d2ebfd0519 QortalOS Titan 5.60.12
Screw the description like that inbred T3Q
2022-03-05 21:17:59 +05:00

620 lines
18 KiB
C++

/*
SPDX-FileCopyrightText: 2012-2016 Eike Hein <hein@kde.org>
SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "backend.h"
#include "log_settings.h"
#include <KConfigGroup>
#include <KDesktopFile>
#include <KFileItem>
#include <KFilePlacesModel>
#include <KLocalizedString>
#include <KNotificationJobUiDelegate>
#include <KService>
#include <KServiceAction>
#include <KWindowEffects>
#include <KWindowSystem>
#include <KIO/ApplicationLauncherJob>
#include <KService/KApplicationTrader>
#include <QAction>
#include <QActionGroup>
#include <QApplication>
#include <QDBusConnection>
#include <QDBusConnectionInterface>
#include <QDBusMessage>
#include <QDBusMetaType>
#include <QDBusPendingCall>
#include <QDBusReply>
#include <QDBusServiceWatcher>
#include <QJsonArray>
#include <QMenu>
#include <QQuickItem>
#include <QQuickWindow>
#include <QScopedPointer>
#include <QTimer>
#include <QVersionNumber>
#include <KActivities/Consumer>
#include <KActivities/Stats/Cleaning>
#include <KActivities/Stats/ResultSet>
#include <KActivities/Stats/Terms>
#include <processcore/process.h>
#include <processcore/processes.h>
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<QAction *>(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<QAction *>(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<KFilePlacesModel> 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<QIcon>();
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<QAction *>(action);
++actionCount;
}
if (actionCount > 0) {
QAction *separatorAction = new QAction(parent);
separatorAction->setSeparator(true);
actions << QVariant::fromValue<QAction *>(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<QAction *>(action);
}
return actions;
}
void Backend::handleRecentDocumentAction() const
{
const QAction *action = qobject_cast<QAction *>(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<QUrl> Backend::jsonArrayToUrlList(const QJsonArray &array) const
{
QList<QUrl> 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);
}