123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159 |
- /*
- * Copyright (c) 2023, Cameron Youell <cameronyouell@gmail.com>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #include "AutoComplete.h"
- #include "Settings.h"
- #include <AK/JsonArray.h>
- #include <AK/JsonObject.h>
- #include <AK/JsonParser.h>
- #include <LibURL/URL.h>
- namespace Ladybird {
- AutoComplete::AutoComplete(QWidget* parent)
- : QCompleter(parent)
- {
- m_tree_view = new QTreeView(parent);
- m_manager = new QNetworkAccessManager(this);
- m_auto_complete_model = new AutoCompleteModel(this);
- setCompletionMode(QCompleter::UnfilteredPopupCompletion);
- setModel(m_auto_complete_model);
- setPopup(m_tree_view);
- m_tree_view->setRootIsDecorated(false);
- m_tree_view->setHeaderHidden(true);
- connect(this, QOverload<QModelIndex const&>::of(&QCompleter::activated), this, [&](QModelIndex const& index) {
- emit activated(index);
- });
- connect(m_manager, &QNetworkAccessManager::finished, this, [&](QNetworkReply* reply) {
- auto result = got_network_response(reply);
- if (result.is_error())
- dbgln("AutoComplete::got_network_response: Error {}", result.error());
- });
- }
- ErrorOr<Vector<String>> AutoComplete::parse_google_autocomplete(Vector<JsonValue> const& json)
- {
- if (json.size() != 5)
- return Error::from_string_literal("Invalid JSON, expected 5 elements in array");
- if (!json[0].is_string())
- return Error::from_string_literal("Invalid JSON, expected first element to be a string");
- auto query = TRY(String::from_byte_string(json[0].as_string()));
- if (!json[1].is_array())
- return Error::from_string_literal("Invalid JSON, expected second element to be an array");
- auto suggestions_array = json[1].as_array().values();
- if (query != m_query)
- return Error::from_string_literal("Invalid JSON, query does not match");
- Vector<String> results;
- results.ensure_capacity(suggestions_array.size());
- for (auto& suggestion : suggestions_array)
- results.unchecked_append(MUST(String::from_byte_string(suggestion.as_string())));
- return results;
- }
- ErrorOr<Vector<String>> AutoComplete::parse_duckduckgo_autocomplete(Vector<JsonValue> const& json)
- {
- Vector<String> results;
- for (auto const& suggestion : json) {
- auto maybe_value = suggestion.as_object().get("phrase"sv);
- if (!maybe_value.has_value())
- continue;
- results.append(MUST(String::from_byte_string(maybe_value->as_string())));
- }
- return results;
- }
- ErrorOr<Vector<String>> AutoComplete::parse_yahoo_autocomplete(JsonObject const& json)
- {
- if (!json.get("q"sv).has_value() || !json.get("q"sv)->is_string())
- return Error::from_string_view("Invalid JSON, expected \"q\" to be a string"sv);
- auto query = TRY(String::from_byte_string(json.get("q"sv)->as_string()));
- if (!json.get("r"sv).has_value() || !json.get("r"sv)->is_array())
- return Error::from_string_view("Invalid JSON, expected \"r\" to be an object"sv);
- auto suggestions_object = json.get("r"sv)->as_array().values();
- if (query != m_query)
- return Error::from_string_literal("Invalid JSON, query does not match");
- Vector<String> results;
- results.ensure_capacity(suggestions_object.size());
- for (auto& suggestion_object : suggestions_object) {
- if (!suggestion_object.is_object())
- return Error::from_string_literal("Invalid JSON, expected value to be an object");
- auto suggestion = suggestion_object.as_object();
- if (!suggestion.get("k"sv).has_value() || !suggestion.get("k"sv)->is_string())
- return Error::from_string_view("Invalid JSON, expected \"k\" to be a string"sv);
- results.unchecked_append(MUST(String::from_byte_string(suggestion.get("k"sv)->as_string())));
- }
- return results;
- }
- ErrorOr<void> AutoComplete::got_network_response(QNetworkReply* reply)
- {
- if (reply->error() == QNetworkReply::NetworkError::OperationCanceledError)
- return {};
- AK::JsonParser parser(ak_byte_string_from_qstring(reply->readAll()));
- auto json = TRY(parser.parse());
- auto engine_name = Settings::the()->autocomplete_engine().name;
- Vector<String> results;
- if (engine_name == "Google") {
- results = TRY(parse_google_autocomplete(json.as_array().values()));
- } else if (engine_name == "DuckDuckGo") {
- results = TRY(parse_duckduckgo_autocomplete(json.as_array().values()));
- } else if (engine_name == "Yahoo")
- results = TRY(parse_yahoo_autocomplete(json.as_object()));
- else {
- return Error::from_string_literal("Invalid engine name");
- }
- constexpr size_t MAX_AUTOCOMPLETE_RESULTS = 6;
- if (results.is_empty()) {
- results.append(m_query);
- } else if (results.size() > MAX_AUTOCOMPLETE_RESULTS) {
- results.shrink(MAX_AUTOCOMPLETE_RESULTS);
- }
- m_auto_complete_model->replace_suggestions(move(results));
- return {};
- }
- String AutoComplete::auto_complete_url_from_query(StringView query)
- {
- auto autocomplete_engine = ak_string_from_qstring(Settings::the()->autocomplete_engine().url);
- return MUST(autocomplete_engine.replace("{}"sv, URL::percent_encode(query), ReplaceMode::FirstOnly));
- }
- void AutoComplete::clear_suggestions()
- {
- m_auto_complete_model->clear();
- }
- void AutoComplete::get_search_suggestions(String search_string)
- {
- m_query = move(search_string);
- if (m_reply)
- m_reply->abort();
- QNetworkRequest request { QUrl(qstring_from_ak_string(auto_complete_url_from_query(m_query))) };
- m_reply = m_manager->get(request);
- }
- }
|