/* SPDX-FileCopyrightText: 2007 Teemu Rytilahti SPDX-License-Identifier: LGPL-2.0-only */ #include "webshortcutrunner.h" #include #include #include #include #include #include #include #include #include #include WebshortcutRunner::WebshortcutRunner(QObject *parent, const KPluginMetaData &metaData, const QVariantList &args) : Plasma::AbstractRunner(parent, metaData, args) , m_match(this) , m_filterBeforeRun(false) { setObjectName(QStringLiteral("Web Shortcut")); m_match.setType(Plasma::QueryMatch::ExactMatch); m_match.setRelevance(0.9); // Listen for KUriFilter plugin config changes and update state... QDBusConnection sessionDbus = QDBusConnection::sessionBus(); sessionDbus.connect(QString(), QStringLiteral("/"), QStringLiteral("org.kde.KUriFilterPlugin"), QStringLiteral("configure"), this, SLOT(loadSyntaxes())); loadSyntaxes(); configurePrivateBrowsingActions(); connect(KSycoca::self(), &KSycoca::databaseChanged, this, &WebshortcutRunner::configurePrivateBrowsingActions); setMinLetterCount(3); } WebshortcutRunner::~WebshortcutRunner() { } void WebshortcutRunner::loadSyntaxes() { KUriFilterData filterData(QStringLiteral(":q")); filterData.setSearchFilteringOptions(KUriFilterData::RetrieveAvailableSearchProvidersOnly); if (KUriFilter::self()->filterSearchUri(filterData, KUriFilter::NormalTextFilter)) { m_delimiter = filterData.searchTermSeparator(); } QList syns; const QStringList providers = filterData.preferredSearchProviders(); const QRegularExpression replaceRegex(QStringLiteral(":q$")); const QString placeholder = QStringLiteral(":q:"); for (const QString &provider : providers) { Plasma::RunnerSyntax s(filterData.queryForPreferredSearchProvider(provider).replace(replaceRegex, placeholder), i18n("Opens \"%1\" in a web browser with the query :q:.", provider)); syns << s; } if (!providers.isEmpty()) { QString defaultKey = filterData.queryForSearchProvider(providers.constFirst()).defaultKey(); Plasma::RunnerSyntax s(QStringLiteral("!%1 :q:").arg(defaultKey), i18n("Search using the DuckDuckGo bang syntax")); syns << s; } setSyntaxes(syns); m_lastFailedKey.clear(); m_lastProvider.clear(); m_lastKey.clear(); } void WebshortcutRunner::configurePrivateBrowsingActions() { qDeleteAll(m_match.actions()); m_match.setActions({}); const QString browserFile = KSharedConfig::openConfig(QStringLiteral("kdeglobals"))->group("General").readEntry("BrowserApplication"); KService::Ptr service; if (!browserFile.isEmpty()) { service = KService::serviceByStorageId(browserFile); } if (!service) { service = KApplicationTrader::preferredService(QStringLiteral("text/html")); } if (!service) { return; } const auto actions = service->actions(); for (const auto &action : actions) { bool containsPrivate = action.text().contains(QLatin1String("private"), Qt::CaseInsensitive); bool containsIncognito = action.text().contains(QLatin1String("incognito"), Qt::CaseInsensitive); if (containsPrivate || containsIncognito) { m_privateAction = action; const QString actionText = containsPrivate ? i18n("Search in private window") : i18n("Search in incognito window"); const QIcon icon = QIcon::fromTheme(QStringLiteral("view-private"), QIcon::fromTheme(QStringLiteral("view-hidden"))); m_match.setActions({new QAction(icon, actionText, this)}); return; } } } void WebshortcutRunner::match(Plasma::RunnerContext &context) { const QString term = context.query(); const static QRegularExpression bangRegex(QStringLiteral("!([^ ]+).*")); const static QRegularExpression normalRegex(QStringLiteral("^([^ ]+)%1").arg(QRegularExpression::escape(m_delimiter))); const auto bangMatch = bangRegex.match(term); QString key; QString rawQuery = term; if (bangMatch.hasMatch()) { key = bangMatch.captured(1); rawQuery = rawQuery.remove(rawQuery.indexOf(key) - 1, key.size() + 1); } else { const auto normalMatch = normalRegex.match(term); if (normalMatch.hasMatch()) { key = normalMatch.captured(0); rawQuery = rawQuery.mid(key.length()); } } if (key.isEmpty() || key == m_lastFailedKey) { return; // we already know it's going to suck ;) } // Do a fake user feedback text update if the keyword has not changed. // There is no point filtering the request on every key stroke. // filtering if (m_lastKey == key) { m_filterBeforeRun = true; m_match.setText(i18n("Search %1 for %2", m_lastProvider, rawQuery)); context.addMatch(m_match); return; } KUriFilterData filterData(term); if (!KUriFilter::self()->filterSearchUri(filterData, KUriFilter::WebShortcutFilter)) { m_lastFailedKey = key; return; } // Reuse key/provider for next matches. Other variables ca be reused, because the same match object is used m_lastKey = key; m_lastProvider = filterData.searchProvider(); m_match.setIconName(filterData.iconName()); m_match.setId(QStringLiteral("WebShortcut:") + key); m_match.setText(i18n("Search %1 for %2", m_lastProvider, filterData.searchTerm())); m_match.setData(filterData.uri()); context.addMatch(m_match); } void WebshortcutRunner::run(const Plasma::RunnerContext &context, const Plasma::QueryMatch &match) { QUrl location; if (m_filterBeforeRun) { m_filterBeforeRun = false; KUriFilterData filterData(context.query()); if (KUriFilter::self()->filterSearchUri(filterData, KUriFilter::WebShortcutFilter)) location = filterData.uri(); } else { location = match.data().toUrl(); } if (!location.isEmpty()) { if (match.selectedAction()) { QString command; // Chrome's exec line does not have a URL placeholder // Firefox's does, but only sometimes, depending on the distro // Replace placeholders if found, otherwise append at the end if (m_privateAction.exec().contains("%u")) { command = m_privateAction.exec().replace("%u", KShell::quoteArg(location.toString())); } else if (m_privateAction.exec().contains("%U")) { command = m_privateAction.exec().replace("%U", KShell::quoteArg(location.toString())); } else { command = m_privateAction.exec() + QLatin1Char(' ') + KShell::quoteArg(location.toString()); } auto *job = new KIO::CommandLauncherJob(command); job->start(); } else { auto job = new KIO::OpenUrlJob(location); job->start(); } } } K_PLUGIN_CLASS_WITH_JSON(WebshortcutRunner, "plasma-runner-webshortcuts.json") #include "webshortcutrunner.moc"