/* SPDX-FileCopyrightText: 2006 Aaron Seigo SPDX-FileCopyrightText: 2014 Vishesh Handa SPDX-FileCopyrightText: 2016-2020 Harald Sitter SPDX-License-Identifier: LGPL-2.0-only */ #include "servicerunner.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #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 m_seen; QList 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::fromLocalFile(path)}); return data; }