/* * This file is part of the KDE Baloo Project * Copyright (C) 2014 Vishesh Handa * Copyright (C) 2017 David Edmundson * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA * */ #include "baloosearchrunner.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "krunner1adaptor.h" static const QString s_openParentDirId = QStringLiteral("openParentDir"); int main(int argc, char **argv) { Baloo::IndexerConfig config; if (!config.fileIndexingEnabled()) { return -1; } QCoreApplication::setAttribute(Qt::AA_DisableSessionManager); QApplication::setQuitOnLastWindowClosed(false); QApplication app(argc, argv); // KRun needs widgets for error message boxes SearchRunner r; return app.exec(); } SearchRunner::SearchRunner(QObject *parent) : QObject(parent) , m_timer(new QTimer(this)) { m_timer->setSingleShot(true); connect(m_timer, &QTimer::timeout, this, &SearchRunner::performMatch); new Krunner1Adaptor(this); qDBusRegisterMetaType(); qDBusRegisterMetaType(); qDBusRegisterMetaType(); qDBusRegisterMetaType(); QDBusConnection::sessionBus().registerObject(QStringLiteral("/runner"), this); QDBusConnection::sessionBus().registerService(QStringLiteral("org.kde.runners.baloo")); } SearchRunner::~SearchRunner() { } RemoteActions SearchRunner::Actions() { return RemoteActions({RemoteAction{s_openParentDirId, i18n("Open Containing Folder"), QStringLiteral("document-open-folder")}}); } RemoteMatches SearchRunner::Match(const QString &searchTerm) { // Do not try to show results for queries starting with = // this should trigger the calculator, but the AdvancedQueryParser::parse method // in baloo interpreted it as an operator, BUG 345134 if (searchTerm.startsWith(QLatin1Char('='))) { return RemoteMatches(); } setDelayedReply(true); if (m_lastRequest.type() != QDBusMessage::InvalidMessage) { QDBusConnection::sessionBus().send(m_lastRequest.createReply(QVariantList())); } m_lastRequest = message(); m_searchTerm = searchTerm; // Baloo (as of 2014-11-20) is single threaded. It has an internal mutex which results in // queries being queued one after another. Also, Baloo is really really slow for small queries // For example - on my SSD, it takes about 1.4 seconds for 'f' with an SSD. // When searching for "fire", it results in "f", "fi", "fir" and then "fire" being searched // We're therefore hacking around this by having a small delay for very short queries so that // they do not get queued internally in Baloo // int waitTimeMs = 0; if (searchTerm.length() <= 3) { waitTimeMs = 100; } // we're still using the event delayed call even when the length is < 3 so that if we have multiple Match() calls in our DBus queue, we only process the // last one m_timer->start(waitTimeMs); return RemoteMatches(); } void SearchRunner::performMatch() { // Filter out duplicates QSet foundUrls; // The location runner handles file paths, otherwise we would end up with duplicate entries QFileInfo fileInfo(KShell::tildeExpand(m_searchTerm)); if (fileInfo.exists()) { foundUrls << QUrl::fromLocalFile(fileInfo.absoluteFilePath()); } RemoteMatches matches; matches << matchInternal(m_searchTerm, QStringLiteral("Audio"), i18n("Audio"), foundUrls); matches << matchInternal(m_searchTerm, QStringLiteral("Image"), i18n("Image"), foundUrls); matches << matchInternal(m_searchTerm, QStringLiteral("Video"), i18n("Video"), foundUrls); matches << matchInternal(m_searchTerm, QStringLiteral("Spreadsheet"), i18n("Spreadsheet"), foundUrls); matches << matchInternal(m_searchTerm, QStringLiteral("Presentation"), i18n("Presentation"), foundUrls); matches << matchInternal(m_searchTerm, QStringLiteral("Folder"), i18n("Folder"), foundUrls); matches << matchInternal(m_searchTerm, QStringLiteral("Document"), i18n("Document"), foundUrls); matches << matchInternal(m_searchTerm, QStringLiteral("Archive"), i18n("Archive"), foundUrls); QDBusConnection::sessionBus().send(m_lastRequest.createReply(QVariant::fromValue(matches))); m_lastRequest = QDBusMessage(); } RemoteMatches SearchRunner::matchInternal(const QString &searchTerm, const QString &type, const QString &category, QSet &foundUrls) { Baloo::Query query; query.setSearchString(searchTerm); query.setType(type); query.setLimit(10); Baloo::ResultIterator it = query.exec(); RemoteMatches matches; QMimeDatabase mimeDb; // KRunner is absolutely daft and allows plugins to set the global // relevance levels. so Baloo should not set the relevance of results too // high because then Applications will often appear after if the application // runner has not a higher relevance. So stupid. // Each runner plugin should not have to know about the others. // Anyway, that's why we're starting with .75 float relevance = .75; while (it.next()) { RemoteMatch match; QString localUrl = it.filePath(); const QUrl url = QUrl::fromLocalFile(localUrl); if (foundUrls.contains(url)) { continue; } foundUrls.insert(url); match.id = url.toString(); match.text = url.fileName(); match.iconName = mimeDb.mimeTypeForFile(localUrl).iconName(); match.relevance = relevance; match.type = Plasma::QueryMatch::PossibleMatch; QVariantMap properties; QString folderPath = url.adjusted(QUrl::RemoveFilename | QUrl::StripTrailingSlash).toLocalFile(); folderPath = KShell::tildeCollapse(folderPath); properties[QStringLiteral("urls")] = QStringList({QString::fromLocal8Bit(url.toEncoded())}); properties[QStringLiteral("subtext")] = folderPath; properties[QStringLiteral("category")] = category; match.properties = properties; relevance -= 0.05; matches << match; } return matches; } void SearchRunner::Run(const QString &id, const QString &actionId) { const QUrl url(id); if (actionId == s_openParentDirId) { KIO::highlightInFileManager({url}); return; } auto *job = new KIO::OpenUrlJob(url); job->setUiDelegate(new KNotificationJobUiDelegate(KJobUiDelegate::AutoErrorHandlingEnabled)); job->setRunExecutables(false); job->start(); }