blob: 6e67615c6e5e1f6bb8ce01e34503fa8ddd3a3900 [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/omnibox/browser/autocomplete_classifier.h"
#include <utility>
#include "base/auto_reset.h"
#include "base/feature_list.h"
#include "base/strings/utf_string_conversions.h"
#include "base/trace_event/trace_event.h"
#include "build/build_config.h"
#include "components/history_embeddings/history_embeddings_features.h"
#include "components/omnibox/browser/autocomplete_controller.h"
#include "components/omnibox/browser/autocomplete_input.h"
#include "components/omnibox/browser/autocomplete_match.h"
#include "components/omnibox/browser/autocomplete_provider.h"
#include "components/omnibox/browser/omnibox_field_trial.h"
#include "components/omnibox/common/omnibox_features.h"
#include "third_party/metrics_proto/omnibox_event.pb.h"
#include "url/gurl.h"
#if !BUILDFLAG(IS_IOS)
#include "components/history_clusters/core/config.h" // nogncheck
#endif
AutocompleteClassifier::AutocompleteClassifier(
std::unique_ptr<AutocompleteController> controller,
std::unique_ptr<AutocompleteSchemeClassifier> scheme_classifier)
: controller_(std::move(controller)),
scheme_classifier_(std::move(scheme_classifier)),
inside_classify_(false) {}
AutocompleteClassifier::~AutocompleteClassifier() {
// We should only reach here after Shutdown() has been called.
DCHECK(!controller_);
}
void AutocompleteClassifier::Shutdown() {
controller_.reset();
}
// static
int AutocompleteClassifier::DefaultOmniboxProviders(bool is_low_memory_device) {
return
#if !BUILDFLAG(IS_ANDROID) && !BUILDFLAG(IS_IOS)
// Custom search engines cannot be used on mobile.
AutocompleteProvider::TYPE_KEYWORD | AutocompleteProvider::TYPE_OPEN_TAB |
AutocompleteProvider::TYPE_FEATURED_SEARCH |
#else
AutocompleteProvider::TYPE_CLIPBOARD |
AutocompleteProvider::TYPE_MOST_VISITED_SITES |
AutocompleteProvider::TYPE_VERBATIM_MATCH |
#endif
#if BUILDFLAG(IS_ANDROID)
AutocompleteProvider::TYPE_VOICE_SUGGEST |
// Only enabled for hub search.
AutocompleteProvider::TYPE_OPEN_TAB |
#endif
#if !BUILDFLAG(IS_IOS)
(history_clusters::GetConfig().is_journeys_enabled_no_locale_check &&
history_clusters::GetConfig().omnibox_history_cluster_provider
? AutocompleteProvider::TYPE_HISTORY_CLUSTER_PROVIDER
: 0) |
#endif
AutocompleteProvider::TYPE_ZERO_SUGGEST |
AutocompleteProvider::TYPE_ZERO_SUGGEST_LOCAL_HISTORY |
(base::FeatureList::IsEnabled(omnibox::kDocumentProvider)
? AutocompleteProvider::TYPE_DOCUMENT
: 0) |
(OmniboxFieldTrial::IsOnDeviceHeadSuggestEnabledForAnyMode()
? AutocompleteProvider::TYPE_ON_DEVICE_HEAD
: 0) |
AutocompleteProvider::TYPE_BOOKMARK | AutocompleteProvider::TYPE_BUILTIN |
AutocompleteProvider::TYPE_HISTORY_QUICK |
AutocompleteProvider::TYPE_HISTORY_URL |
AutocompleteProvider::TYPE_SEARCH | AutocompleteProvider::TYPE_SHORTCUTS |
AutocompleteProvider::TYPE_HISTORY_FUZZY |
AutocompleteProvider::TYPE_CALCULATOR |
#if !BUILDFLAG(IS_ANDROID) && !BUILDFLAG(IS_IOS)
(history_embeddings::GetFeatureParameters().omnibox_scoped ||
history_embeddings::GetFeatureParameters().omnibox_unscoped
? AutocompleteProvider::TYPE_HISTORY_EMBEDDINGS
: 0)
#else
0
#endif
;
}
void AutocompleteClassifier::Classify(
const std::u16string& text,
bool prefer_keyword,
bool allow_exact_keyword_match,
metrics::OmniboxEventProto::PageClassification page_classification,
AutocompleteMatch* match,
GURL* alternate_nav_url) {
TRACE_EVENT1("omnibox", "AutocompleteClassifier::Classify", "text",
base::UTF16ToUTF8(text));
DCHECK(!inside_classify_);
base::AutoReset<bool> reset(&inside_classify_, true);
AutocompleteInput input(text, page_classification, *scheme_classifier_);
input.set_prevent_inline_autocomplete(true);
// If the user in keyword mode (which is often the case when |prefer_keyword|
// is true), ideally we'd set |input|'s keyword_mode_entry_method field.
// However, in the context of this code, we don't know how the keyword mode
// was entered. Moreover, we cannot add that as a parameter to Classify()
// because many callers do not know how keyword mode was entered. Luckily,
// Classify()'s purpose is to determine the default match, and at this time
// |keyword_mode_entry_method| only ends up affecting the ranking of
// lower-down suggestions.
input.set_prefer_keyword(prefer_keyword);
input.set_allow_exact_keyword_match(allow_exact_keyword_match);
input.set_omit_asynchronous_matches(true);
controller_->Start(input);
DCHECK(controller_->done());
auto* default_match = controller_->result().default_match();
if (!default_match) {
if (alternate_nav_url)
*alternate_nav_url = GURL();
return;
}
*match = *default_match;
if (alternate_nav_url) {
*alternate_nav_url = AutocompleteResult::ComputeAlternateNavUrl(
input, *match, controller_->autocomplete_provider_client());
}
}