blob: cef61a45348494187071c7c4f729d5f30492ab6b [file] [log] [blame]
// 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 "searchbox_handler.h"
#include "base/memory/raw_ptr.h"
#include "base/test/metrics/histogram_tester.h"
#include "base/test/scoped_feature_list.h"
#include "base/test/test_future.h"
#include "chrome/browser/ui/browser_window/test/mock_browser_window_interface.h"
#include "chrome/browser/ui/omnibox/omnibox_controller.h"
#include "chrome/browser/ui/tab_ui_helper.h"
#include "chrome/browser/ui/tabs/alert/tab_alert_controller.h"
#include "chrome/browser/ui/tabs/tab_strip_model.h"
#include "chrome/browser/ui/tabs/test_tab_strip_model_delegate.h"
#include "chrome/browser/ui/webui/new_tab_page/composebox/variations/composebox_fieldtrial.h"
#include "chrome/browser/ui/webui/omnibox_popup/omnibox_popup_ui.h"
#include "chrome/browser/ui/webui/searchbox/lens_searchbox_client.h"
#include "chrome/browser/ui/webui/searchbox/searchbox_test_utils.h"
#include "chrome/browser/ui/webui/searchbox/webui_omnibox_handler.h"
#include "chrome/browser/ui/webui/webui_embedding_context.h"
#include "chrome/test/base/testing_profile.h"
#include "components/lens/tab_contextualization_controller.h"
#include "components/omnibox/browser/autocomplete_controller.h"
#include "components/omnibox/browser/mock_autocomplete_provider_client.h"
#include "components/omnibox/browser/omnibox_prefs.h"
#include "components/omnibox/browser/test_omnibox_client.h"
#include "components/omnibox/browser/vector_icons.h"
#include "components/search/ntp_features.h"
#include "components/variations/scoped_variations_ids_provider.h"
#include "components/variations/variations_ids_provider.h"
#include "content/public/test/browser_task_environment.h"
#include "content/public/test/test_renderer_host.h"
#include "content/public/test/test_web_ui.h"
#include "content/public/test/test_web_ui_data_source.h"
#include "content/public/test/web_contents_tester.h"
#include "lens_searchbox_handler.h"
#include "realbox_handler.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/metrics_proto/omnibox_event.pb.h"
#include "third_party/metrics_proto/omnibox_focus_type.pb.h"
#include "ui/base/webui/web_ui_util.h"
class SearchboxHandlerTest : public ::testing::Test {
public:
SearchboxHandlerTest() = default;
SearchboxHandlerTest(const SearchboxHandlerTest&) = delete;
SearchboxHandlerTest& operator=(const SearchboxHandlerTest&) = delete;
~SearchboxHandlerTest() override = default;
content::TestWebUIDataSource* source() { return source_.get(); }
TestingProfile* profile() { return profile_.get(); }
protected:
testing::NiceMock<MockSearchboxPage> page_;
raw_ptr<testing::NiceMock<MockAutocompleteController>>
autocomplete_controller_;
raw_ptr<testing::NiceMock<MockOmniboxEditModel>> omnibox_edit_model_;
content::BrowserTaskEnvironment task_environment_;
std::unique_ptr<content::TestWebUIDataSource> source_;
std::unique_ptr<TestingProfile> profile_;
variations::test::ScopedVariationsIdsProvider scoped_variations_ids_provider_{
variations::VariationsIdsProvider::Mode::kUseSignedInState};
void SetUp() override {
source_ = content::TestWebUIDataSource::Create("test-data-source");
TestingProfile::Builder profile_builder;
profile_ = profile_builder.Build();
ASSERT_EQ(
variations::VariationsIdsProvider::ForceIdsResult::SUCCESS,
variations::VariationsIdsProvider::GetInstance()
->ForceVariationIdsForTesting(
/*variation_ids=*/{"100"}, /*command_line_variation_ids=*/""));
}
void TearDown() override {
omnibox_edit_model_ = nullptr;
autocomplete_controller_ = nullptr;
}
};
class RealboxHandlerTest : public SearchboxHandlerTest {
public:
RealboxHandlerTest() = default;
RealboxHandlerTest(const RealboxHandlerTest&) = delete;
RealboxHandlerTest& operator=(const RealboxHandlerTest&) = delete;
~RealboxHandlerTest() override = default;
protected:
content::RenderViewHostTestEnabler test_render_host_factories_;
std::unique_ptr<content::WebContents> web_contents_;
std::unique_ptr<RealboxHandler> handler_;
void SetUp() override {
SearchboxHandlerTest::SetUp();
web_contents_ =
content::WebContentsTester::CreateTestWebContents(profile(), nullptr);
handler_ = std::make_unique<RealboxHandler>(
mojo::PendingReceiver<searchbox::mojom::PageHandler>(), profile(),
web_contents_.get());
handler_->SetPage(page_.BindAndGetRemote());
}
void TearDown() override {
SearchboxHandlerTest::TearDown();
handler_.reset();
}
};
TEST_F(RealboxHandlerTest, RealboxLensVariationsContainsVariations) {
SearchboxHandler::SetupWebUIDataSource(source()->GetWebUIDataSource(),
profile());
EXPECT_EQ("CGQ", *source()->GetLocalizedStrings()->FindString(
"searchboxLensVariations"));
}
TEST_F(RealboxHandlerTest, AutocompleteController_Start) {
// Stop observing the AutocompleteController instance which will be destroyed.
handler_->autocomplete_controller_observation_.Reset();
// Set a mock AutocompleteController.
auto autocomplete_controller =
std::make_unique<testing::NiceMock<MockAutocompleteController>>(
std::make_unique<MockAutocompleteProviderClient>(), 0);
autocomplete_controller_ = autocomplete_controller.get();
handler_->omnibox_controller()->SetAutocompleteControllerForTesting(
std::move(autocomplete_controller));
// Set a mock OmniboxEditModel.
auto omnibox_edit_model =
std::make_unique<testing::NiceMock<MockOmniboxEditModel>>(
handler_->omnibox_controller());
omnibox_edit_model_ = omnibox_edit_model.get();
handler_->omnibox_controller()->SetEditModelForTesting(
std::move(omnibox_edit_model));
{
SCOPED_TRACE("Empty input");
std::u16string input_text;
EXPECT_CALL(*omnibox_edit_model_, SetUserText(_))
.Times(1)
.WillOnce(SaveArg<0>(&input_text));
AutocompleteInput input;
EXPECT_CALL(*autocomplete_controller_, Start(_))
.Times(1)
.WillOnce(SaveArg<0>(&input));
handler_->QueryAutocomplete(u"", /*prevent_inline_autocomplete=*/false);
EXPECT_EQ(input_text, u"");
EXPECT_EQ(input.text(), u"");
EXPECT_EQ(input.focus_type(), metrics::OmniboxFocusType::INTERACTION_FOCUS);
EXPECT_EQ(input.current_url().spec(), "");
EXPECT_EQ(input.current_page_classification(),
metrics::OmniboxEventProto::NTP_REALBOX);
EXPECT_FALSE(input.lens_overlay_suggest_inputs().has_value());
testing::Mock::VerifyAndClearExpectations(omnibox_edit_model_);
testing::Mock::VerifyAndClearExpectations(autocomplete_controller_);
}
{
SCOPED_TRACE("Non-empty input");
std::u16string input_text;
EXPECT_CALL(*omnibox_edit_model_, SetUserText(_))
.Times(1)
.WillOnce(SaveArg<0>(&input_text));
AutocompleteInput input;
EXPECT_CALL(*autocomplete_controller_, Start(_))
.Times(1)
.WillOnce(SaveArg<0>(&input));
handler_->QueryAutocomplete(u"a", /*prevent_inline_autocomplete=*/false);
EXPECT_EQ(input_text, u"a");
EXPECT_EQ(input.text(), u"a");
EXPECT_EQ(input.focus_type(),
metrics::OmniboxFocusType::INTERACTION_DEFAULT);
EXPECT_EQ(input.current_url().spec(), "");
EXPECT_EQ(input.current_page_classification(),
metrics::OmniboxEventProto::NTP_REALBOX);
EXPECT_FALSE(input.lens_overlay_suggest_inputs().has_value());
testing::Mock::VerifyAndClearExpectations(omnibox_edit_model_);
testing::Mock::VerifyAndClearExpectations(autocomplete_controller_);
}
}
TEST_F(RealboxHandlerTest, GetPlaceholderConfig) {
base::test::TestFuture<searchbox::mojom::PlaceholderConfigPtr> future;
handler_->GetPlaceholderConfig(future.GetCallback());
auto config = future.Take();
ASSERT_GT(config->texts.size(), 0u);
ASSERT_EQ(config->change_text_animation_interval.InMilliseconds(), 4000u);
ASSERT_EQ(config->fade_text_animation_duration.InMilliseconds(), 250u);
}
TEST_F(RealboxHandlerTest, AddFileContext) {
const auto token = base::UnguessableToken::Create();
const std::string image_data_url = "";
const bool is_deletable = true;
// SelectedFileInfoPtr is a move-only type, so capture it in the lambda.
searchbox::mojom::SelectedFileInfoPtr captured_file_info;
EXPECT_CALL(page_, AddFileContext(token, testing::_))
.Times(1)
.WillOnce([&](const base::UnguessableToken&,
searchbox::mojom::SelectedFileInfoPtr info) {
captured_file_info = std::move(info);
});
searchbox::mojom::SelectedFileInfoPtr file_info =
searchbox::mojom::SelectedFileInfo::New();
file_info->file_name = "Visual Selection";
file_info->mime_type = "image/png";
file_info->image_data_url = image_data_url;
file_info->is_deletable = is_deletable;
handler_->AddFileContextFromBrowser(token, file_info.Clone());
page_.FlushForTesting();
ASSERT_TRUE(captured_file_info);
ASSERT_EQ(captured_file_info->file_name, file_info->file_name);
ASSERT_EQ(captured_file_info->mime_type, file_info->mime_type);
ASSERT_EQ(captured_file_info->image_data_url, file_info->image_data_url);
ASSERT_EQ(captured_file_info->is_deletable, file_info->is_deletable);
}
class LensSearchboxHandlerTest : public SearchboxHandlerTest {
public:
LensSearchboxHandlerTest() = default;
LensSearchboxHandlerTest(const LensSearchboxHandlerTest&) = delete;
LensSearchboxHandlerTest& operator=(const LensSearchboxHandlerTest&) = delete;
~LensSearchboxHandlerTest() override = default;
protected:
std::unique_ptr<testing::NiceMock<MockLensSearchboxClient>>
lens_searchbox_client_;
std::unique_ptr<LensSearchboxHandler> handler_;
private:
void SetUp() override {
SearchboxHandlerTest::SetUp();
// Set a mock LensSearchboxClient.
lens_searchbox_client_ =
std::make_unique<testing::NiceMock<MockLensSearchboxClient>>();
handler_ = std::make_unique<LensSearchboxHandler>(
mojo::PendingReceiver<searchbox::mojom::PageHandler>(), profile(),
/*web_contents=*/nullptr, lens_searchbox_client_.get());
handler_->SetPage(page_.BindAndGetRemote());
}
};
TEST_F(LensSearchboxHandlerTest, Lens_AutocompleteController_Start) {
// Stop observing the AutocompleteController instance which will be destroyed.
handler_->autocomplete_controller_observation_.Reset();
// Set a mock AutocompleteController.
auto autocomplete_controller =
std::make_unique<testing::NiceMock<MockAutocompleteController>>(
std::make_unique<MockAutocompleteProviderClient>(), 0);
autocomplete_controller_ = autocomplete_controller.get();
handler_->omnibox_controller()->SetAutocompleteControllerForTesting(
std::move(autocomplete_controller));
// Set a mock OmniboxEditModel.
auto omnibox_edit_model =
std::make_unique<testing::NiceMock<MockOmniboxEditModel>>(
handler_->omnibox_controller());
omnibox_edit_model_ = omnibox_edit_model.get();
handler_->omnibox_controller()->SetEditModelForTesting(
std::move(omnibox_edit_model));
{
SCOPED_TRACE("Empty input");
std::u16string input_text;
EXPECT_CALL(*omnibox_edit_model_, SetUserText(_))
.Times(1)
.WillOnce(SaveArg<0>(&input_text));
AutocompleteInput input;
EXPECT_CALL(*autocomplete_controller_, Start(_))
.Times(1)
.WillOnce(SaveArg<0>(&input));
EXPECT_CALL(*lens_searchbox_client_, GetPageClassification())
.Times(1)
.WillOnce(Return(metrics::OmniboxEventProto::CONTEXTUAL_SEARCHBOX));
GURL page_url("https://example.com");
EXPECT_CALL(*lens_searchbox_client_, GetPageURL())
.Times(1)
.WillOnce(ReturnRef(page_url));
lens::proto::LensOverlaySuggestInputs suggest_inputs;
suggest_inputs.set_encoded_image_signals("xyz");
suggest_inputs.set_encoded_request_id("abc");
suggest_inputs.set_search_session_id("123");
suggest_inputs.set_encoded_visual_search_interaction_log_data("321");
EXPECT_CALL(*lens_searchbox_client_, GetLensSuggestInputs())
.WillRepeatedly(ReturnRef(suggest_inputs));
handler_->QueryAutocomplete(u"", /*prevent_inline_autocomplete=*/false);
EXPECT_EQ(input_text, u"");
EXPECT_EQ(input.text(), u"");
EXPECT_EQ(input.focus_type(), metrics::OmniboxFocusType::INTERACTION_FOCUS);
EXPECT_EQ(input.current_url(), page_url);
EXPECT_EQ(input.current_page_classification(),
metrics::OmniboxEventProto::CONTEXTUAL_SEARCHBOX);
EXPECT_EQ(input.lens_overlay_suggest_inputs()->encoded_image_signals(),
suggest_inputs.encoded_image_signals());
EXPECT_EQ(input.lens_overlay_suggest_inputs()->encoded_request_id(),
suggest_inputs.encoded_request_id());
EXPECT_EQ(input.lens_overlay_suggest_inputs()->search_session_id(),
suggest_inputs.search_session_id());
EXPECT_EQ(input.lens_overlay_suggest_inputs()
->encoded_visual_search_interaction_log_data(),
suggest_inputs.encoded_visual_search_interaction_log_data());
testing::Mock::VerifyAndClearExpectations(omnibox_edit_model_);
testing::Mock::VerifyAndClearExpectations(autocomplete_controller_);
testing::Mock::VerifyAndClearExpectations(lens_searchbox_client_.get());
}
{
SCOPED_TRACE("Non-empty input");
std::u16string input_text;
EXPECT_CALL(*omnibox_edit_model_, SetUserText(_))
.Times(1)
.WillOnce(SaveArg<0>(&input_text));
AutocompleteInput input;
EXPECT_CALL(*autocomplete_controller_, Start(_))
.Times(1)
.WillOnce(SaveArg<0>(&input));
EXPECT_CALL(*lens_searchbox_client_, GetPageClassification())
.Times(1)
.WillOnce(Return(metrics::OmniboxEventProto::CONTEXTUAL_SEARCHBOX));
GURL page_url("https://example.com");
EXPECT_CALL(*lens_searchbox_client_, GetPageURL())
.Times(1)
.WillOnce(ReturnRef(page_url));
lens::proto::LensOverlaySuggestInputs suggest_inputs;
suggest_inputs.set_encoded_image_signals("xyz");
suggest_inputs.set_encoded_request_id("abc");
suggest_inputs.set_search_session_id("123");
suggest_inputs.set_encoded_visual_search_interaction_log_data("321");
EXPECT_CALL(*lens_searchbox_client_, GetLensSuggestInputs())
.WillRepeatedly(ReturnRef(suggest_inputs));
handler_->QueryAutocomplete(u"a", /*prevent_inline_autocomplete=*/false);
EXPECT_EQ(input_text, u"a");
EXPECT_EQ(input.text(), u"a");
EXPECT_EQ(input.focus_type(),
metrics::OmniboxFocusType::INTERACTION_DEFAULT);
EXPECT_EQ(input.current_url(), page_url);
EXPECT_EQ(input.current_page_classification(),
metrics::OmniboxEventProto::CONTEXTUAL_SEARCHBOX);
EXPECT_EQ(input.lens_overlay_suggest_inputs()->encoded_image_signals(),
suggest_inputs.encoded_image_signals());
EXPECT_EQ(input.lens_overlay_suggest_inputs()->encoded_request_id(),
suggest_inputs.encoded_request_id());
EXPECT_EQ(input.lens_overlay_suggest_inputs()->search_session_id(),
suggest_inputs.search_session_id());
EXPECT_EQ(input.lens_overlay_suggest_inputs()
->encoded_visual_search_interaction_log_data(),
suggest_inputs.encoded_visual_search_interaction_log_data());
testing::Mock::VerifyAndClearExpectations(omnibox_edit_model_);
testing::Mock::VerifyAndClearExpectations(autocomplete_controller_);
testing::Mock::VerifyAndClearExpectations(lens_searchbox_client_.get());
}
{
SCOPED_TRACE("Icon override");
const char search_icon[] = "//resources/images/icon_search.svg";
const std::string& svg_name = handler_->AutocompleteIconToResourceName(
omnibox::kSubdirectoryArrowRightIcon);
EXPECT_EQ(svg_name, search_icon);
}
}
class WebuiOmniboxHandlerTest : public SearchboxHandlerTest {
public:
WebuiOmniboxHandlerTest() = default;
WebuiOmniboxHandlerTest(const WebuiOmniboxHandlerTest&) = delete;
WebuiOmniboxHandlerTest& operator=(const WebuiOmniboxHandlerTest&) = delete;
~WebuiOmniboxHandlerTest() override = default;
protected:
void SetUp() override {
SearchboxHandlerTest::SetUp();
omnibox_controller_ = std::make_unique<OmniboxController>(
std::make_unique<TestOmniboxClient>());
web_contents_ =
content::WebContentsTester::CreateTestWebContents(profile(), nullptr);
web_ui_.set_web_contents(web_contents_.get());
handler_ = std::make_unique<WebuiOmniboxHandler>(
mojo::PendingReceiver<searchbox::mojom::PageHandler>(),
/*metrics_reporter=*/nullptr, omnibox_controller_.get(), &web_ui_);
handler_->SetPage(page_.BindAndGetRemote());
}
void TearDown() override {
handler_.reset();
SearchboxHandlerTest::TearDown();
}
content::RenderViewHostTestEnabler test_render_host_factories_;
std::unique_ptr<content::WebContents> web_contents_;
content::TestWebUI web_ui_;
std::unique_ptr<OmniboxPopupUI> omnibox_popup_ui_;
std::unique_ptr<OmniboxController> omnibox_controller_;
std::unique_ptr<WebuiOmniboxHandler> handler_;
};
TEST_F(WebuiOmniboxHandlerTest, WebuiOmniboxUpdatesSelection) {
searchbox::mojom::OmniboxPopupSelectionPtr old_selection;
searchbox::mojom::OmniboxPopupSelectionPtr selection;
EXPECT_CALL(page_, UpdateSelection)
.Times(4)
.WillRepeatedly([&old_selection, &selection](
searchbox::mojom::OmniboxPopupSelectionPtr arg0,
searchbox::mojom::OmniboxPopupSelectionPtr arg1) {
old_selection = std::move(arg0);
selection = std::move(arg1);
});
handler_->OnSelectionChanged(
OmniboxPopupSelection(OmniboxPopupSelection::kNoMatch),
OmniboxPopupSelection(0, OmniboxPopupSelection::NORMAL));
page_.FlushForTesting();
EXPECT_EQ(0, selection->line);
EXPECT_EQ(searchbox::mojom::SelectionLineState::kNormal, selection->state);
handler_->OnSelectionChanged(
OmniboxPopupSelection(0, OmniboxPopupSelection::NORMAL),
OmniboxPopupSelection(1, OmniboxPopupSelection::KEYWORD_MODE));
page_.FlushForTesting();
EXPECT_EQ(1, selection->line);
EXPECT_EQ(searchbox::mojom::SelectionLineState::kKeywordMode,
selection->state);
handler_->OnSelectionChanged(
OmniboxPopupSelection(2, OmniboxPopupSelection::NORMAL),
OmniboxPopupSelection(2, OmniboxPopupSelection::FOCUSED_BUTTON_ACTION,
4));
page_.FlushForTesting();
EXPECT_EQ(2, selection->line);
EXPECT_EQ(4, selection->action_index);
EXPECT_EQ(searchbox::mojom::SelectionLineState::kFocusedButtonAction,
selection->state);
handler_->OnSelectionChanged(
OmniboxPopupSelection(3, OmniboxPopupSelection::FOCUSED_BUTTON_ACTION, 4),
OmniboxPopupSelection(
3, OmniboxPopupSelection::FOCUSED_BUTTON_REMOVE_SUGGESTION));
page_.FlushForTesting();
EXPECT_EQ(3, selection->line);
EXPECT_EQ(
searchbox::mojom::SelectionLineState::kFocusedButtonRemoveSuggestion,
selection->state);
}