AutoComplete.cpp 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. /*
  2. * Copyright (c) 2023, Cameron Youell <cameronyouell@gmail.com>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include "AutoComplete.h"
  7. #include "Settings.h"
  8. #include <AK/JsonArray.h>
  9. #include <AK/JsonObject.h>
  10. #include <AK/JsonParser.h>
  11. #include <AK/URL.h>
  12. namespace Ladybird {
  13. AutoComplete::AutoComplete(QWidget* parent)
  14. : QCompleter(parent)
  15. {
  16. m_tree_view = new QTreeView(parent);
  17. m_manager = new QNetworkAccessManager(this);
  18. m_auto_complete_model = new AutoCompleteModel(this);
  19. setCompletionMode(QCompleter::UnfilteredPopupCompletion);
  20. setModel(m_auto_complete_model);
  21. setPopup(m_tree_view);
  22. m_tree_view->setRootIsDecorated(false);
  23. m_tree_view->setHeaderHidden(true);
  24. connect(this, QOverload<QModelIndex const&>::of(&QCompleter::activated), this, [&](QModelIndex const& index) {
  25. emit activated(index);
  26. });
  27. connect(m_manager, &QNetworkAccessManager::finished, this, [&](QNetworkReply* reply) {
  28. auto result = got_network_response(reply);
  29. if (result.is_error())
  30. dbgln("AutoComplete::got_network_response: Error {}", result.error());
  31. });
  32. }
  33. ErrorOr<void> AutoComplete::parse_google_autocomplete(Vector<JsonValue> const& json)
  34. {
  35. if (json.size() != 5)
  36. return Error::from_string_view("Invalid JSON, expected 5 elements in array"sv);
  37. if (!json[0].is_string())
  38. return Error::from_string_view("Invalid JSON, expected first element to be a string"sv);
  39. auto query = TRY(String::from_deprecated_string(json[0].as_string()));
  40. if (!json[1].is_array())
  41. return Error::from_string_view("Invalid JSON, expected second element to be an array"sv);
  42. auto suggestions_array = json[1].as_array().values();
  43. if (query != m_query)
  44. return Error::from_string_view("Invalid JSON, query does not match"sv);
  45. for (auto& suggestion : suggestions_array) {
  46. m_auto_complete_model->add(TRY(String::from_deprecated_string(suggestion.as_string())));
  47. }
  48. return {};
  49. }
  50. ErrorOr<void> AutoComplete::parse_duckduckgo_autocomplete(Vector<JsonValue> const& json)
  51. {
  52. for (auto const& suggestion : json) {
  53. auto maybe_value = suggestion.as_object().get("phrase"sv);
  54. if (!maybe_value.has_value())
  55. continue;
  56. m_auto_complete_model->add(TRY(String::from_deprecated_string(maybe_value->as_string())));
  57. }
  58. return {};
  59. }
  60. ErrorOr<void> AutoComplete::parse_yahoo_autocomplete(JsonObject const& json)
  61. {
  62. if (!json.get("q"sv).has_value() || !json.get("q"sv)->is_string())
  63. return Error::from_string_view("Invalid JSON, expected \"q\" to be a string"sv);
  64. auto query = TRY(String::from_deprecated_string(json.get("q"sv)->as_string()));
  65. if (!json.get("r"sv).has_value() || !json.get("r"sv)->is_array())
  66. return Error::from_string_view("Invalid JSON, expected \"r\" to be an object"sv);
  67. auto suggestions_object = json.get("r"sv)->as_array().values();
  68. if (query != m_query)
  69. return Error::from_string_view("Invalid JSON, query does not match"sv);
  70. for (auto& suggestion_object : suggestions_object) {
  71. if (!suggestion_object.is_object())
  72. return Error::from_string_view("Invalid JSON, expected value to be an object"sv);
  73. auto suggestion = suggestion_object.as_object();
  74. if (!suggestion.get("k"sv).has_value() || !suggestion.get("k"sv)->is_string())
  75. return Error::from_string_view("Invalid JSON, expected \"k\" to be a string"sv);
  76. m_auto_complete_model->add(TRY(String::from_deprecated_string(suggestion.get("k"sv)->as_string())));
  77. };
  78. return {};
  79. }
  80. ErrorOr<void> AutoComplete::got_network_response(QNetworkReply* reply)
  81. {
  82. if (reply->error() == QNetworkReply::NetworkError::OperationCanceledError)
  83. return {};
  84. AK::JsonParser parser(ak_deprecated_string_from_qstring(reply->readAll()));
  85. auto json = TRY(parser.parse());
  86. auto engine_name = Settings::the()->autocomplete_engine().name;
  87. if (engine_name == "Google")
  88. return parse_google_autocomplete(json.as_array().values());
  89. if (engine_name == "DuckDuckGo")
  90. return parse_duckduckgo_autocomplete(json.as_array().values());
  91. if (engine_name == "Yahoo")
  92. return parse_yahoo_autocomplete(json.as_object());
  93. return Error::from_string_view("Invalid engine name"sv);
  94. }
  95. ErrorOr<String> AutoComplete::auto_complete_url_from_query(StringView query)
  96. {
  97. auto autocomplete_engine = TRY(ak_string_from_qstring(Settings::the()->autocomplete_engine().url));
  98. return autocomplete_engine.replace("{}"sv, AK::URL::percent_encode(query), ReplaceMode::FirstOnly);
  99. }
  100. void AutoComplete::clear_suggestions()
  101. {
  102. m_auto_complete_model->clear();
  103. }
  104. ErrorOr<void> AutoComplete::get_search_suggestions(StringView search_string)
  105. {
  106. m_query = TRY(String::from_utf8(search_string));
  107. if (m_reply)
  108. m_reply->abort();
  109. m_auto_complete_model->clear();
  110. m_auto_complete_model->add(m_query);
  111. QNetworkRequest request { QUrl(qstring_from_ak_string(TRY(auto_complete_url_from_query(m_query)))) };
  112. m_reply = m_manager->get(request);
  113. return {};
  114. }
  115. }