blob: 0c28a4951fa1c707e5cae758e0381d5e829afe8e [file] [log] [blame] [edit]
// Copyright 2022 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/open_tab_provider.h"
#include <algorithm>
#include "base/i18n/case_conversion.h"
#include "base/strings/string_util.h"
#include "base/strings/utf_string_conversions.h"
#include "components/omnibox/browser/suggestion_group_util.h"
#if BUILDFLAG(IS_ANDROID)
#include "components/browser_ui/util/android/url_constants.h"
#endif
#include "components/omnibox/browser/autocomplete_input.h"
#include "components/omnibox/browser/autocomplete_match.h"
#include "components/omnibox/browser/autocomplete_match_classification.h"
#include "components/omnibox/browser/autocomplete_provider_client.h"
#include "components/omnibox/browser/in_memory_url_index_types.h"
#include "components/omnibox/browser/keyword_provider.h"
#include "components/omnibox/browser/match_compare.h"
#include "components/omnibox/browser/omnibox_field_trial.h"
#include "components/omnibox/browser/scoring_functor.h"
#include "components/omnibox/browser/tab_matcher.h"
#include "components/query_parser/query_parser.h"
#include "components/search_engines/template_url.h"
#include "components/strings/grit/components_strings.h"
#include "components/url_formatter/url_formatter.h"
#include "content/public/common/url_constants.h"
#include "third_party/omnibox_proto/groups.pb.h"
#include "ui/base/l10n/l10n_util.h"
#include "url/gurl.h"
namespace {
constexpr bool is_android = !!BUILDFLAG(IS_ANDROID);
int Score(const AutocompleteInput& input,
const query_parser::QueryNodeVector& input_query_nodes,
const TabMatcher::TabWrapper tab) {
// For Hub Search, remove both ZPS and search suggestions that involve open
// chrome prefixed tabs.
#if BUILDFLAG(IS_ANDROID)
if (input.current_page_classification() ==
::metrics::OmniboxEventProto::ANDROID_HUB &&
(tab.url.SchemeIs(browser_ui::kChromeUINativeScheme) ||
tab.url.SchemeIs(content::kChromeUIScheme))) {
return 0;
}
#endif
if ((input.IsZeroSuggest() || input.text().empty()) && is_android) {
return omnibox::kOpenTabMatchZeroSuggestRelevance +
tab.last_shown_time.InSecondsFSinceUnixEpoch();
}
// TODO(crbug.com/40211187): The bookmark provider also uses on `query_parser`
// and
// `ScoringFunctor` to compute its scores. However, it uses normalized match
// titles. (see `Normalize()` in
// components/bookmarks/browser/titled_url_index.cc) IDK its purpose, but we
// should either verify it's unnecessary here, or do likewise here.
// Extract query words from the title.
const std::u16string lower_title = base::i18n::ToLower(tab.title);
query_parser::QueryWordVector title_words;
query_parser::QueryParser::ExtractQueryWords(lower_title, &title_words);
// Extract query words from the URL.
const std::u16string lower_url =
base::i18n::ToLower(base::UTF8ToUTF16(tab.url.spec()));
query_parser::QueryWordVector url_words;
query_parser::QueryParser::ExtractQueryWords(lower_url, &url_words);
// Every input term must be included in either (or both) the title or URL.
query_parser::Snippet::MatchPositions title_matches;
query_parser::Snippet::MatchPositions url_matches;
if (!std::ranges::all_of(input_query_nodes, [&](const auto& query_node) {
// Using local vars so to not short circuit adding URL matches when
// title matches are found.
const bool has_title_match =
query_node->HasMatchIn(title_words, &title_matches);
const bool has_url_match =
query_node->HasMatchIn(url_words, &url_matches);
return has_title_match || has_url_match;
})) {
return 0;
}
// Score the matches following a simplified variation of
// `BookmarkProvider::CalculateBookmarkMatchRelevance()`. `kMaxScore` is
// chosen based on a reasonable upper bound for title & URL length of 100. The
// exact scores don't matter, but using a scale proportional the max title
// length squared will avoid suggestions with different float scores being
// rounded to the same integer score.
const int kMaxScore = 100 * 100;
const double title_factor =
for_each(title_matches.begin(), title_matches.end(),
ScoringFunctor(lower_title.length()))
.ScoringFactor();
const double url_factor = for_each(url_matches.begin(), url_matches.end(),
ScoringFunctor(lower_url.length()))
.ScoringFactor();
const double normalized_factors =
std::min((title_factor + url_factor) / (lower_title.length() + 10), 1.0);
return normalized_factors * kMaxScore;
}
bool ShouldRunProvider(AutocompleteProviderClient* client,
const AutocompleteInput& input,
const AutocompleteInput& adjusted_input) {
bool zps_or_empty = input.IsZeroSuggest() || input.text().empty();
if (is_android) {
return !zps_or_empty || !client->IsIncognitoProfile();
} else {
return !zps_or_empty;
}
}
} // namespace
OpenTabProvider::OpenTabProvider(AutocompleteProviderClient* client)
: AutocompleteProvider(AutocompleteProvider::TYPE_OPEN_TAB),
client_(client) {}
OpenTabProvider::~OpenTabProvider() = default;
void OpenTabProvider::Start(const AutocompleteInput& input,
bool minimal_changes) {
matches_.clear();
// Remove the keyword from input if we're in keyword mode for a starter pack
// engine.
const auto [adjusted_input, template_url] =
AdjustInputForStarterPackKeyword(input, client_->GetTemplateURLService());
if (!ShouldRunProvider(client_, input, adjusted_input)) {
return;
}
// Preprocess the query into query nodes.
const auto adjusted_input_text = std::u16string(
base::TrimWhitespace(base::i18n::ToLower(adjusted_input.text()),
base::TrimPositions::TRIM_ALL));
query_parser::QueryNodeVector input_query_nodes;
query_parser::QueryParser::ParseQueryNodes(
adjusted_input_text,
query_parser::MatchingAlgorithm::ALWAYS_PREFIX_SEARCH,
&input_query_nodes);
// Perform basic substring matching on the query terms.
for (auto& open_tab : client_->GetTabMatcher().GetOpenTabs(&input)) {
const GURL& url = open_tab.url;
if (!url.is_valid()) {
continue;
}
int score = Score(input, input_query_nodes, open_tab);
if (score > 0) {
matches_.push_back(CreateOpenTabMatch(adjusted_input, open_tab.title, url,
score, template_url));
}
}
// If there were no open tab results found, and we're in keyword mode,
// generate a NULL_RESULT_MESSAGE suggestion to keep the user in keyword mode
// and display a no results message.
if (adjusted_input.InKeywordMode() && matches_.empty() && template_url) {
matches_.push_back(
CreateNullResultMessageMatch(adjusted_input, template_url));
}
}
AutocompleteMatch OpenTabProvider::CreateOpenTabMatch(
const AutocompleteInput& input,
const std::u16string& title,
const GURL& url,
int score,
const TemplateURL* template_url) {
DCHECK(url.is_valid());
AutocompleteMatch match(this, score, /*deletable=*/false,
AutocompleteMatchType::OPEN_TAB);
match.destination_url = url;
match.fill_into_edit = base::UTF8ToUTF16(url.spec());
// Setting this ensures that the result deduplication doesn't prioritize
// another default match over an open tab result.
match.allowed_to_be_default_match = true;
// If the input was in the @tabs keyword scope, set the `keyword` and
// `transition` appropriately to avoid popping the user out of keyword mode.
if (template_url) {
match.keyword = template_url->keyword();
match.transition = ui::PAGE_TRANSITION_KEYWORD;
}
// For display in the suggestion UI, elide all optional parts. The user has
// already opened these URLs so there's no need for them to make nuanced
// judgements about the URL (for example, whether the scheme is http:// or
// https:// or whether there's a "www" subdomain).
match.contents = url_formatter::FormatUrl(
url,
AutocompleteMatch::GetFormatTypes(/*preserve_scheme=*/false,
/*preserve_subdomain=*/false),
base::UnescapeRule::SPACES, nullptr, nullptr, nullptr);
auto contents_terms = FindTermMatches(input.text(), match.contents);
match.contents_class = ClassifyTermMatches(
contents_terms, match.contents.size(),
ACMatchClassification::MATCH | ACMatchClassification::URL,
ACMatchClassification::URL);
match.description = title;
auto description_terms = FindTermMatches(input.text(), match.description);
match.description_class = ClassifyTermMatches(
description_terms, match.description.size(), ACMatchClassification::MATCH,
ACMatchClassification::NONE);
if (input.InKeywordMode()) {
match.from_keyword = true;
}
#if BUILDFLAG(IS_ANDROID)
if (input.current_page_classification() ==
::metrics::OmniboxEventProto::ANDROID_HUB) {
match.suggestion_group_id = omnibox::GROUP_MOBILE_OPEN_TABS;
}
#endif
return match;
}
AutocompleteMatch OpenTabProvider::CreateNullResultMessageMatch(
const AutocompleteInput& input,
const TemplateURL* template_url) {
DCHECK(template_url);
// This value doesn't really matter as this suggestion is only displayed when
// no other suggestions were found. Use an arbitrary constant.
constexpr int kRelevanceScore = 1000;
AutocompleteMatch match(this, kRelevanceScore, /*deletable=*/false,
AutocompleteMatchType::NULL_RESULT_MESSAGE);
// These fields are filled in to enable the Keyword UI when only this
// suggestion is available.
match.fill_into_edit = input.text();
match.allowed_to_be_default_match = true;
match.keyword = template_url->keyword();
match.transition = ui::PAGE_TRANSITION_KEYWORD;
// Use this suggestion's contents field to display a message to the user that
// there were no matching results found.
match.contents =
l10n_util::GetStringUTF16(IDS_OMNIBOX_TAB_SEARCH_NO_RESULTS_FOUND);
match.contents_class.push_back(
ACMatchClassification(0, ACMatchClassification::NONE));
match.from_keyword = true;
return match;
}