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.
495 lines
17 KiB
495 lines
17 KiB
/* |
|
* Copyright (C) 2006 Aaron Seigo <aseigo@kde.org> |
|
* Copyright (C) 2014 Vishesh Handa <vhanda@kde.org> |
|
* Copyright (C) 2016-2020 Harald Sitter <sitter@kde.org> |
|
* |
|
* This program is free software; you can redistribute it and/or modify |
|
* it under the terms of the GNU Library General Public License version 2 as |
|
* published by the Free Software Foundation |
|
* |
|
* 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 Library 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 "servicerunner.h" |
|
|
|
#include <algorithm> |
|
|
|
#include <QMimeData> |
|
|
|
#include <QDebug> |
|
#include <QDir> |
|
#include <QIcon> |
|
#include <QStandardPaths> |
|
#include <QUrl> |
|
|
|
#include <KActivities/ResourceInstance> |
|
#include <KLocalizedString> |
|
#include <KNotificationJobUiDelegate> |
|
#include <KServiceAction> |
|
#include <KServiceTypeTrader> |
|
#include <KStringHandler> |
|
#include <KSycoca> |
|
|
|
#include <KIO/ApplicationLauncherJob> |
|
|
|
#include "debug.h" |
|
|
|
namespace |
|
{ |
|
int weightedLength(const QString &query) |
|
{ |
|
return KStringHandler::logicalLength(query); |
|
} |
|
|
|
} // 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(); |
|
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 QVector<QStringRef> &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; |
|
} |
|
|
|
QString generateQuery(const QVector<QStringRef> &strList) |
|
{ |
|
QString keywordTemplate = QStringLiteral("exist Keywords"); |
|
QString genericNameTemplate = QStringLiteral("exist GenericName"); |
|
QString nameTemplate = QStringLiteral("exist Name"); |
|
QString commentTemplate = QStringLiteral("exist Comment"); |
|
|
|
// Search for applications which are executable and the term case-insensitive matches any of |
|
// * a substring of one of the keywords |
|
// * a substring of the GenericName field |
|
// * a substring of the Name field |
|
// Note that before asking for the content of e.g. Keywords and GenericName we need to ask if |
|
// they exist to prevent a tree evaluation error if they are not defined. |
|
for (const QStringRef &str : strList) { |
|
keywordTemplate += QStringLiteral(" and '%1' ~subin Keywords").arg(str.toString()); |
|
genericNameTemplate += QStringLiteral(" and '%1' ~~ GenericName").arg(str.toString()); |
|
nameTemplate += QStringLiteral(" and '%1' ~~ Name").arg(str.toString()); |
|
commentTemplate += QStringLiteral(" and '%1' ~~ Comment").arg(str.toString()); |
|
} |
|
|
|
QString finalQuery = QStringLiteral("exist Exec and ( (%1) or (%2) or (%3) or ('%4' ~~ Exec) or (%5) )") |
|
.arg(keywordTemplate, genericNameTemplate, nameTemplate, strList[0].toString(), commentTemplate); |
|
|
|
qCDebug(RUNNER_SERVICES) << "Final query : " << finalQuery; |
|
return finalQuery; |
|
} |
|
|
|
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); |
|
|
|
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; |
|
} |
|
|
|
// Search for applications which are executable and case-insensitively match the search term |
|
// See https://techbase.kde.org/Development/Tutorials/Services/Traders#The_KTrader_Query_Language |
|
// if the following is unclear to you. |
|
query = QStringLiteral("exist Exec and ('%1' =~ Name)").arg(term); |
|
const KService::List services = KServiceTypeTrader::self()->query(QStringLiteral("Application"), query); |
|
|
|
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() |
|
{ |
|
// Splitting the query term to match using subsequences |
|
QVector<QStringRef> queryList = term.splitRef(QLatin1Char(' ')); |
|
|
|
// If the term length is < 3, no real point searching the Keywords and GenericName |
|
if (weightedTermLength < 3) { |
|
query = QStringLiteral("exist Exec and ( (exist Name and '%1' ~~ Name) or ('%1' ~~ Exec) )").arg(term); |
|
} else { |
|
// Match using subsequences (Bug: 262837) |
|
query = generateQuery(queryList); |
|
} |
|
|
|
KService::List services = KServiceTypeTrader::self()->query(QStringLiteral("Application"), query); |
|
services += KServiceTypeTrader::self()->query(QStringLiteral("KCModule"), query); |
|
|
|
qCDebug(RUNNER_SERVICES) << "got " << services.count() << " services from " << query; |
|
for (const KService::Ptr &service : qAsConst(services)) { |
|
if (disqualify(service)) { |
|
continue; |
|
} |
|
|
|
const QString id = service->storageId(); |
|
const QString name = service->desktopEntryName(); |
|
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; |
|
} |
|
} |
|
|
|
const bool isKCM = service->serviceTypes().contains(QLatin1String("KCModule")); |
|
if (!isKCM && (service->categories().contains(QLatin1String("KDE")) || service->serviceTypes().contains(QLatin1String("KCModule")))) { |
|
qCDebug(RUNNER_SERVICES) << "found a kde thing" << id << match.subtext() << relevance; |
|
relevance += .09; |
|
} |
|
|
|
if (isKCM) { |
|
if (service->parentApp() == QStringLiteral("kinfocenter")) { |
|
match.setMatchCategory(i18n("System Information")); |
|
} else { |
|
match.setMatchCategory(i18n("System Settings")); |
|
} |
|
// KCMs are, on the balance, less relevant. Drop it ever so much. So they may get outscored |
|
// by an otherwise equally applicable match. |
|
relevance -= .001; |
|
} |
|
|
|
qCDebug(RUNNER_SERVICES) << service->name() << "is this relevant:" << relevance; |
|
match.setRelevance(relevance); |
|
|
|
matches << match; |
|
} |
|
} |
|
|
|
void matchCategories() |
|
{ |
|
// search for applications whose categories contains the query |
|
query = QStringLiteral("exist Exec and (exist Categories and '%1' ~subin Categories)").arg(term); |
|
const auto services = KServiceTypeTrader::self()->query(QStringLiteral("Application"), query); |
|
|
|
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; |
|
} |
|
|
|
query = QStringLiteral("exist Actions"); // doesn't work |
|
const auto services = KServiceTypeTrader::self()->query(QStringLiteral("Application")); //, query); |
|
|
|
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; |
|
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; |
|
}
|
|
|