mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-01-04 01:05:58 +03:00
157 lines
5.3 KiB
C++
157 lines
5.3 KiB
C++
/*
|
|
* 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 <AK/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<void> AutoComplete::parse_google_autocomplete(Vector<JsonValue> const& json)
|
|
{
|
|
if (json.size() != 5)
|
|
return Error::from_string_view("Invalid JSON, expected 5 elements in array"sv);
|
|
|
|
if (!json[0].is_string())
|
|
return Error::from_string_view("Invalid JSON, expected first element to be a string"sv);
|
|
auto query = TRY(String::from_deprecated_string(json[0].as_string()));
|
|
|
|
if (!json[1].is_array())
|
|
return Error::from_string_view("Invalid JSON, expected second element to be an array"sv);
|
|
auto suggestions_array = json[1].as_array().values();
|
|
|
|
if (query != m_query)
|
|
return Error::from_string_view("Invalid JSON, query does not match"sv);
|
|
|
|
for (auto& suggestion : suggestions_array) {
|
|
m_auto_complete_model->add(TRY(String::from_deprecated_string(suggestion.as_string())));
|
|
}
|
|
|
|
return {};
|
|
}
|
|
|
|
ErrorOr<void> AutoComplete::parse_duckduckgo_autocomplete(Vector<JsonValue> const& json)
|
|
{
|
|
for (auto const& suggestion : json) {
|
|
auto maybe_value = suggestion.as_object().get("phrase"sv);
|
|
if (!maybe_value.has_value())
|
|
continue;
|
|
m_auto_complete_model->add(TRY(String::from_deprecated_string(maybe_value->as_string())));
|
|
}
|
|
|
|
return {};
|
|
}
|
|
|
|
ErrorOr<void> 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_deprecated_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_view("Invalid JSON, query does not match"sv);
|
|
|
|
for (auto& suggestion_object : suggestions_object) {
|
|
if (!suggestion_object.is_object())
|
|
return Error::from_string_view("Invalid JSON, expected value to be an object"sv);
|
|
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);
|
|
|
|
m_auto_complete_model->add(TRY(String::from_deprecated_string(suggestion.get("k"sv)->as_string())));
|
|
};
|
|
|
|
return {};
|
|
}
|
|
|
|
ErrorOr<void> AutoComplete::got_network_response(QNetworkReply* reply)
|
|
{
|
|
if (reply->error() == QNetworkReply::NetworkError::OperationCanceledError)
|
|
return {};
|
|
|
|
AK::JsonParser parser(ak_deprecated_string_from_qstring(reply->readAll()));
|
|
auto json = TRY(parser.parse());
|
|
|
|
auto engine_name = Settings::the()->autocomplete_engine().name;
|
|
if (engine_name == "Google")
|
|
return parse_google_autocomplete(json.as_array().values());
|
|
|
|
if (engine_name == "DuckDuckGo")
|
|
return parse_duckduckgo_autocomplete(json.as_array().values());
|
|
|
|
if (engine_name == "Yahoo")
|
|
return parse_yahoo_autocomplete(json.as_object());
|
|
|
|
return Error::from_string_view("Invalid engine name"sv);
|
|
}
|
|
|
|
ErrorOr<String> AutoComplete::search_url_from_query(StringView query)
|
|
{
|
|
auto search_engine = TRY(ak_string_from_qstring(Settings::the()->search_engine().url));
|
|
return search_engine.replace("{}"sv, AK::URL::percent_encode(query), ReplaceMode::FirstOnly);
|
|
}
|
|
|
|
ErrorOr<String> AutoComplete::auto_complete_url_from_query(StringView query)
|
|
{
|
|
auto autocomplete_engine = TRY(ak_string_from_qstring(Settings::the()->autocomplete_engine().url));
|
|
return autocomplete_engine.replace("{}"sv, AK::URL::percent_encode(query), ReplaceMode::FirstOnly);
|
|
}
|
|
|
|
void AutoComplete::clear_suggestions()
|
|
{
|
|
m_auto_complete_model->clear();
|
|
}
|
|
|
|
ErrorOr<void> AutoComplete::get_search_suggestions(StringView search_string)
|
|
{
|
|
m_query = TRY(String::from_utf8(search_string));
|
|
if (m_reply)
|
|
m_reply->abort();
|
|
|
|
m_auto_complete_model->clear();
|
|
m_auto_complete_model->add(m_query);
|
|
|
|
QNetworkRequest request { QUrl(qstring_from_ak_string(TRY(auto_complete_url_from_query(m_query)))) };
|
|
m_reply = m_manager->get(request);
|
|
|
|
return {};
|
|
}
|
|
|
|
}
|