forked from Qortal/Brooklyn
489 lines
16 KiB
C++
489 lines
16 KiB
C++
/*
|
|
SPDX-FileCopyrightText: 2006 Aaron Seigo <aseigo@kde.org>
|
|
SPDX-FileCopyrightText: 2014 Vishesh Handa <vhanda@kde.org>
|
|
SPDX-FileCopyrightText: 2016-2020 Harald Sitter <sitter@kde.org>
|
|
|
|
SPDX-License-Identifier: LGPL-2.0-only
|
|
*/
|
|
|
|
#include "servicerunner.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <QMimeData>
|
|
|
|
#include <QDebug>
|
|
#include <QDir>
|
|
#include <QIcon>
|
|
#include <QStandardPaths>
|
|
#include <QUrl>
|
|
#include <QUrlQuery>
|
|
|
|
#include <KActivities/ResourceInstance>
|
|
#include <KApplicationTrader>
|
|
#include <KLocalizedString>
|
|
#include <KNotificationJobUiDelegate>
|
|
#include <KServiceAction>
|
|
#include <KStringHandler>
|
|
#include <KSycoca>
|
|
|
|
#include <KIO/ApplicationLauncherJob>
|
|
#include <KIO/DesktopExecParser>
|
|
|
|
#include "debug.h"
|
|
|
|
namespace
|
|
{
|
|
int weightedLength(const QString &query)
|
|
{
|
|
return KStringHandler::logicalLength(query);
|
|
}
|
|
|
|
inline bool contains(const QString &result, const QStringList &queryList)
|
|
{
|
|
return std::all_of(queryList.cbegin(), queryList.cend(), [&result](const QString &query) {
|
|
return result.contains(query, Qt::CaseInsensitive);
|
|
});
|
|
}
|
|
|
|
inline bool contains(const QStringList &results, const QStringList &queryList)
|
|
{
|
|
return std::all_of(queryList.cbegin(), queryList.cend(), [&results](const QString &query) {
|
|
return std::any_of(results.cbegin(), results.cend(), [&query](const QString &result) {
|
|
return result.contains(query, Qt::CaseInsensitive);
|
|
});
|
|
});
|
|
}
|
|
|
|
} // namespace
|
|
|
|
/**
|
|
* @brief Finds all KServices for a given runner query
|
|
*/
|
|
class ServiceFinder
|
|
{
|
|
public:
|
|
ServiceFinder(ServiceRunner *runner)
|
|
: m_runner(runner)
|
|
{
|
|
}
|
|
|
|
void match(Plasma::RunnerContext &context)
|
|
{
|
|
if (!context.isValid()) {
|
|
return;
|
|
}
|
|
|
|
KSycoca::disableAutoRebuild();
|
|
|
|
term = context.query();
|
|
// Splitting the query term to match using subsequences
|
|
queryList = term.split(QLatin1Char(' '));
|
|
weightedTermLength = weightedLength(term);
|
|
|
|
matchExectuables();
|
|
matchNameKeywordAndGenericName();
|
|
matchCategories();
|
|
matchJumpListActions();
|
|
|
|
context.addMatches(matches);
|
|
}
|
|
|
|
private:
|
|
void seen(const KService::Ptr &service)
|
|
{
|
|
m_seen.insert(service->storageId());
|
|
m_seen.insert(service->exec());
|
|
}
|
|
|
|
void seen(const KServiceAction &action)
|
|
{
|
|
m_seen.insert(action.exec());
|
|
}
|
|
|
|
bool hasSeen(const KService::Ptr &service)
|
|
{
|
|
return m_seen.contains(service->storageId()) && m_seen.contains(service->exec());
|
|
}
|
|
|
|
bool hasSeen(const KServiceAction &action)
|
|
{
|
|
return m_seen.contains(action.exec());
|
|
}
|
|
|
|
bool disqualify(const KService::Ptr &service)
|
|
{
|
|
auto ret = hasSeen(service) || service->noDisplay();
|
|
qCDebug(RUNNER_SERVICES) << service->name() << "disqualified?" << ret;
|
|
seen(service);
|
|
return ret;
|
|
}
|
|
|
|
qreal increaseMatchRelavance(const KService::Ptr &service, const QStringList &strList, const QString &category)
|
|
{
|
|
// Increment the relevance based on all the words (other than the first) of the query list
|
|
qreal relevanceIncrement = 0;
|
|
|
|
for (int i = 1; i < strList.size(); ++i) {
|
|
const auto &str = strList.at(i);
|
|
if (category == QLatin1String("Name")) {
|
|
if (service->name().contains(str, Qt::CaseInsensitive)) {
|
|
relevanceIncrement += 0.01;
|
|
}
|
|
} else if (category == QLatin1String("GenericName")) {
|
|
if (service->genericName().contains(str, Qt::CaseInsensitive)) {
|
|
relevanceIncrement += 0.01;
|
|
}
|
|
} else if (category == QLatin1String("Exec")) {
|
|
if (service->exec().contains(str, Qt::CaseInsensitive)) {
|
|
relevanceIncrement += 0.01;
|
|
}
|
|
} else if (category == QLatin1String("Comment")) {
|
|
if (service->comment().contains(str, Qt::CaseInsensitive)) {
|
|
relevanceIncrement += 0.01;
|
|
}
|
|
}
|
|
}
|
|
|
|
return relevanceIncrement;
|
|
}
|
|
|
|
void setupMatch(const KService::Ptr &service, Plasma::QueryMatch &match)
|
|
{
|
|
const QString name = service->name();
|
|
|
|
match.setText(name);
|
|
|
|
QUrl url(service->storageId());
|
|
url.setScheme(QStringLiteral("applications"));
|
|
match.setData(url);
|
|
QString exec = service->exec();
|
|
// We have a snap, remove the ENV variable
|
|
if (exec.contains(QLatin1String("BAMF_DESKTOP_FILE_HINT"))) {
|
|
const static QRegularExpression snapCleanupRegex(QStringLiteral("env BAMF_DESKTOP_FILE_HINT=.+ "));
|
|
exec.remove(snapCleanupRegex);
|
|
}
|
|
const QStringList resultingArgs = KIO::DesktopExecParser(KService(QString(), exec, QString()), {}).resultingArguments();
|
|
match.setId(QStringLiteral("exec://") + resultingArgs.join(QLatin1Char(' ')));
|
|
if (!service->genericName().isEmpty() && service->genericName() != name) {
|
|
match.setSubtext(service->genericName());
|
|
} else if (!service->comment().isEmpty()) {
|
|
match.setSubtext(service->comment());
|
|
}
|
|
|
|
if (!service->icon().isEmpty()) {
|
|
match.setIconName(service->icon());
|
|
}
|
|
}
|
|
|
|
void matchExectuables()
|
|
{
|
|
if (weightedTermLength < 2) {
|
|
return;
|
|
}
|
|
|
|
const auto executablesFilter = [this](const KService::Ptr &service) {
|
|
return QString::compare(service->name(), term, Qt::CaseInsensitive) == 0;
|
|
};
|
|
const KService::List services = KApplicationTrader::query(executablesFilter);
|
|
|
|
if (services.isEmpty()) {
|
|
return;
|
|
}
|
|
|
|
for (const KService::Ptr &service : services) {
|
|
qCDebug(RUNNER_SERVICES) << service->name() << "is an exact match!" << service->storageId() << service->exec();
|
|
if (disqualify(service)) {
|
|
continue;
|
|
}
|
|
Plasma::QueryMatch match(m_runner);
|
|
match.setType(Plasma::QueryMatch::ExactMatch);
|
|
setupMatch(service, match);
|
|
match.setRelevance(1);
|
|
matches << match;
|
|
}
|
|
}
|
|
|
|
void matchNameKeywordAndGenericName()
|
|
{
|
|
const auto nameKeywordAndGenericNameFilter = [this](const KService::Ptr &service) {
|
|
// Name
|
|
if (contains(service->name(), queryList) || contains(service->exec(), queryList)) {
|
|
return true;
|
|
}
|
|
// If the term length is < 3, no real point searching the Keywords and GenericName
|
|
if (weightedTermLength < 3) {
|
|
return false;
|
|
}
|
|
// Keywords
|
|
if (contains(service->keywords(), queryList)) {
|
|
return true;
|
|
}
|
|
// GenericName
|
|
if (contains(service->genericName(), queryList) || contains(service->untranslatedGenericName(), queryList)) {
|
|
return true;
|
|
}
|
|
// Comment
|
|
if (contains(service->comment(), queryList)) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
const KService::List services = KApplicationTrader::query(nameKeywordAndGenericNameFilter);
|
|
|
|
qCDebug(RUNNER_SERVICES) << "got " << services.count() << " services from " << query;
|
|
for (const KService::Ptr &service : services) {
|
|
if (disqualify(service)) {
|
|
continue;
|
|
}
|
|
|
|
const QString id = service->storageId();
|
|
const QString name = service->name();
|
|
const QString exec = service->exec();
|
|
|
|
Plasma::QueryMatch match(m_runner);
|
|
match.setType(Plasma::QueryMatch::PossibleMatch);
|
|
setupMatch(service, match);
|
|
qreal relevance(0.6);
|
|
|
|
// If the term was < 3 chars and NOT at the beginning of the App's name or Exec, then
|
|
// chances are the user doesn't want that app.
|
|
if (weightedTermLength < 3) {
|
|
if (name.startsWith(term, Qt::CaseInsensitive) || exec.startsWith(term, Qt::CaseInsensitive)) {
|
|
relevance = 0.9;
|
|
} else {
|
|
continue;
|
|
}
|
|
} else if (service->name().contains(queryList[0], Qt::CaseInsensitive)) {
|
|
relevance = 0.8;
|
|
relevance += increaseMatchRelavance(service, queryList, QStringLiteral("Name"));
|
|
|
|
if (service->name().startsWith(queryList[0], Qt::CaseInsensitive)) {
|
|
relevance += 0.1;
|
|
}
|
|
} else if (service->genericName().contains(queryList[0], Qt::CaseInsensitive)) {
|
|
relevance = 0.65;
|
|
relevance += increaseMatchRelavance(service, queryList, QStringLiteral("GenericName"));
|
|
|
|
if (service->genericName().startsWith(queryList[0], Qt::CaseInsensitive)) {
|
|
relevance += 0.05;
|
|
}
|
|
} else if (service->exec().contains(queryList[0], Qt::CaseInsensitive)) {
|
|
relevance = 0.7;
|
|
relevance += increaseMatchRelavance(service, queryList, QStringLiteral("Exec"));
|
|
|
|
if (service->exec().startsWith(queryList[0], Qt::CaseInsensitive)) {
|
|
relevance += 0.05;
|
|
}
|
|
} else if (service->comment().contains(queryList[0], Qt::CaseInsensitive)) {
|
|
relevance = 0.5;
|
|
relevance += increaseMatchRelavance(service, queryList, QStringLiteral("Comment"));
|
|
|
|
if (service->comment().startsWith(queryList[0], Qt::CaseInsensitive)) {
|
|
relevance += 0.05;
|
|
}
|
|
}
|
|
|
|
if (service->categories().contains(QLatin1String("KDE"))) {
|
|
qCDebug(RUNNER_SERVICES) << "found a kde thing" << id << match.subtext() << relevance;
|
|
relevance += .09;
|
|
}
|
|
|
|
qCDebug(RUNNER_SERVICES) << service->name() << "is this relevant:" << relevance;
|
|
match.setRelevance(relevance);
|
|
|
|
matches << match;
|
|
}
|
|
}
|
|
|
|
void matchCategories()
|
|
{
|
|
const auto categoriesFilter = [this](const KService::Ptr &service) {
|
|
return contains(service->categories(), queryList);
|
|
};
|
|
// search for applications whose categories contains the query
|
|
const auto services = KApplicationTrader::query(categoriesFilter);
|
|
|
|
for (const KService::Ptr &service : services) {
|
|
qCDebug(RUNNER_SERVICES) << service->name() << "is an exact match!" << service->storageId() << service->exec();
|
|
if (disqualify(service)) {
|
|
continue;
|
|
}
|
|
|
|
Plasma::QueryMatch match(m_runner);
|
|
match.setType(Plasma::QueryMatch::PossibleMatch);
|
|
setupMatch(service, match);
|
|
|
|
qreal relevance = 0.6;
|
|
if (service->categories().contains(QLatin1String("X-KDE-More")) || !service->showInCurrentDesktop()) {
|
|
relevance = 0.5;
|
|
}
|
|
|
|
if (service->isApplication()) {
|
|
relevance += .04;
|
|
}
|
|
|
|
match.setRelevance(relevance);
|
|
matches << match;
|
|
}
|
|
}
|
|
|
|
void matchJumpListActions()
|
|
{
|
|
if (weightedTermLength < 3) {
|
|
return;
|
|
}
|
|
|
|
const auto hasActionsFilter = [](const KService::Ptr &service) {
|
|
return !service->actions().isEmpty();
|
|
};
|
|
const auto services = KApplicationTrader::query(hasActionsFilter);
|
|
|
|
for (const KService::Ptr &service : services) {
|
|
if (service->noDisplay()) {
|
|
continue;
|
|
}
|
|
|
|
// Skip SystemSettings as we find KCMs already
|
|
if (service->storageId() == QLatin1String("systemsettings.desktop")) {
|
|
continue;
|
|
}
|
|
|
|
const auto actions = service->actions();
|
|
for (const KServiceAction &action : actions) {
|
|
if (action.text().isEmpty() || action.exec().isEmpty() || hasSeen(action)) {
|
|
continue;
|
|
}
|
|
seen(action);
|
|
|
|
const int matchIndex = action.text().indexOf(term, 0, Qt::CaseInsensitive);
|
|
if (matchIndex < 0) {
|
|
continue;
|
|
}
|
|
|
|
Plasma::QueryMatch match(m_runner);
|
|
match.setType(Plasma::QueryMatch::PossibleMatch);
|
|
if (!action.icon().isEmpty()) {
|
|
match.setIconName(action.icon());
|
|
} else {
|
|
match.setIconName(service->icon());
|
|
}
|
|
match.setText(i18nc("Jump list search result, %1 is action (eg. open new tab), %2 is application (eg. browser)",
|
|
"%1 - %2",
|
|
action.text(),
|
|
service->name()));
|
|
|
|
QUrl url(service->storageId());
|
|
url.setScheme(QStringLiteral("applications"));
|
|
|
|
QUrlQuery query;
|
|
query.addQueryItem(QStringLiteral("action"), action.name());
|
|
url.setQuery(query);
|
|
|
|
match.setData(url);
|
|
|
|
qreal relevance = 0.5;
|
|
if (matchIndex == 0) {
|
|
relevance += 0.05;
|
|
}
|
|
|
|
match.setRelevance(relevance);
|
|
|
|
matches << match;
|
|
}
|
|
}
|
|
}
|
|
|
|
ServiceRunner *m_runner;
|
|
QSet<QString> m_seen;
|
|
|
|
QList<Plasma::QueryMatch> matches;
|
|
QString query;
|
|
QString term;
|
|
QStringList queryList;
|
|
int weightedTermLength = -1;
|
|
};
|
|
|
|
ServiceRunner::ServiceRunner(QObject *parent, const KPluginMetaData &metaData, const QVariantList &args)
|
|
: Plasma::AbstractRunner(parent, metaData, args)
|
|
{
|
|
setObjectName(QStringLiteral("Application"));
|
|
setPriority(AbstractRunner::HighestPriority);
|
|
|
|
addSyntax(Plasma::RunnerSyntax(QStringLiteral(":q:"), i18n("Finds applications whose name or description match :q:")));
|
|
}
|
|
|
|
ServiceRunner::~ServiceRunner() = default;
|
|
|
|
void ServiceRunner::match(Plasma::RunnerContext &context)
|
|
{
|
|
// This helper class aids in keeping state across numerous
|
|
// different queries that together form the matches set.
|
|
ServiceFinder finder(this);
|
|
finder.match(context);
|
|
}
|
|
|
|
void ServiceRunner::run(const Plasma::RunnerContext &context, const Plasma::QueryMatch &match)
|
|
{
|
|
Q_UNUSED(context)
|
|
|
|
const QUrl dataUrl = match.data().toUrl();
|
|
|
|
KService::Ptr service = KService::serviceByStorageId(dataUrl.path());
|
|
if (!service) {
|
|
return;
|
|
}
|
|
|
|
KActivities::ResourceInstance::notifyAccessed(QUrl(QStringLiteral("applications:") + service->storageId()), QStringLiteral("org.kde.krunner"));
|
|
|
|
KIO::ApplicationLauncherJob *job = nullptr;
|
|
|
|
const QString actionName = QUrlQuery(dataUrl).queryItemValue(QStringLiteral("action"));
|
|
if (actionName.isEmpty()) {
|
|
job = new KIO::ApplicationLauncherJob(service);
|
|
} else {
|
|
const auto actions = service->actions();
|
|
auto it = std::find_if(actions.begin(), actions.end(), [&actionName](const KServiceAction &action) {
|
|
return action.name() == actionName;
|
|
});
|
|
Q_ASSERT(it != actions.end());
|
|
|
|
job = new KIO::ApplicationLauncherJob(*it);
|
|
}
|
|
|
|
auto *delegate = new KNotificationJobUiDelegate;
|
|
delegate->setAutoErrorHandlingEnabled(true);
|
|
job->setUiDelegate(delegate);
|
|
job->start();
|
|
}
|
|
|
|
QMimeData *ServiceRunner::mimeDataForMatch(const Plasma::QueryMatch &match)
|
|
{
|
|
const QUrl dataUrl = match.data().toUrl();
|
|
|
|
const QString actionName = QUrlQuery(dataUrl).queryItemValue(QStringLiteral("action"));
|
|
if (!actionName.isEmpty()) {
|
|
return nullptr;
|
|
}
|
|
|
|
KService::Ptr service = KService::serviceByStorageId(dataUrl.path());
|
|
if (!service) {
|
|
return nullptr;
|
|
}
|
|
|
|
QString path = service->entryPath();
|
|
if (!QDir::isAbsolutePath(path)) {
|
|
path = QStandardPaths::locate(QStandardPaths::GenericDataLocation, QStringLiteral("kservices5/") + path);
|
|
}
|
|
|
|
if (path.isEmpty()) {
|
|
return nullptr;
|
|
}
|
|
|
|
auto *data = new QMimeData();
|
|
data->setUrls(QList<QUrl>{QUrl::fromLocalFile(path)});
|
|
return data;
|
|
}
|