| // Copyright 2013 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include <memory> |
| |
| #include "base/memory/raw_ptr.h" |
| #include "build/build_config.h" |
| #include "chrome/browser/autofill/autofill_uitest_util.h" |
| #include "chrome/browser/ui/autofill/autofill_popup_controller_impl.h" |
| #include "chrome/browser/ui/autofill/autofill_popup_view.h" |
| #include "chrome/browser/ui/autofill/chrome_autofill_client.h" |
| #include "chrome/browser/ui/browser.h" |
| #include "chrome/browser/ui/browser_window.h" |
| #include "chrome/browser/ui/tabs/tab_strip_model.h" |
| #include "chrome/test/base/in_process_browser_test.h" |
| #include "chrome/test/base/ui_test_utils.h" |
| #include "components/autofill/content/browser/content_autofill_driver.h" |
| #include "components/autofill/content/browser/content_autofill_driver_factory.h" |
| #include "components/autofill/core/browser/autofill_test_utils.h" |
| #include "components/autofill/core/browser/browser_autofill_manager.h" |
| #include "components/autofill/core/browser/test_autofill_external_delegate.h" |
| #include "content/public/browser/render_frame_host.h" |
| #include "content/public/browser/web_contents.h" |
| #include "content/public/test/browser_test.h" |
| #include "net/dns/mock_host_resolver.h" |
| #include "third_party/blink/public/mojom/frame/fullscreen.mojom.h" |
| #include "ui/compositor/scoped_animation_duration_scale_mode.h" |
| #include "ui/gfx/geometry/rect.h" |
| #include "ui/gfx/geometry/vector2d.h" |
| |
| namespace autofill { |
| |
| class AutofillPopupControllerBrowserTest : public InProcessBrowserTest { |
| public: |
| AutofillPopupControllerBrowserTest() = default; |
| ~AutofillPopupControllerBrowserTest() override = default; |
| |
| void SetUpOnMainThread() override { |
| web_contents()->Focus(); |
| |
| // The test cases mock the entire forms by directly calling |
| // ContentAutofillDriver functions. Nonetheless we set up an HTTP server and |
| // open an empty page. Otherwise, the FormData::url would be about:blank and |
| // FormStructure::ShouldBeParsed() would be false, so the form wouldn't be |
| // even parsed by AutofillManager. |
| ASSERT_TRUE(embedded_test_server()->InitializeAndListen()); |
| host_resolver()->AddRule("*", "127.0.0.1"); |
| embedded_test_server()->RegisterRequestHandler(base::BindRepeating( |
| [](const net::test_server::HttpRequest& request) |
| -> std::unique_ptr<net::test_server::HttpResponse> { |
| auto response = |
| std::make_unique<net::test_server::BasicHttpResponse>(); |
| response->set_code(net::HTTP_OK); |
| response->set_content_type("text/html;charset=utf-8"); |
| response->set_content(""); |
| return response; |
| })); |
| embedded_test_server()->StartAcceptingConnections(); |
| ASSERT_TRUE(ui_test_utils::NavigateToURL( |
| browser(), embedded_test_server()->GetURL("/test.html"))); |
| |
| autofill_driver_ = |
| ContentAutofillDriverFactory::FromWebContents(web_contents()) |
| ->DriverForFrame(main_rfh()); |
| auto autofill_external_delegate = |
| std::make_unique<TestAutofillExternalDelegate>( |
| &autofill_manager(), autofill_driver_, |
| /*call_parent_methods=*/true); |
| autofill_external_delegate_ = autofill_external_delegate.get(); |
| autofill_manager().SetExternalDelegateForTest( |
| std::move(autofill_external_delegate)); |
| |
| disable_animation_ = std::make_unique<ui::ScopedAnimationDurationScaleMode>( |
| ui::ScopedAnimationDurationScaleMode::ZERO_DURATION); |
| } |
| |
| void TearDownOnMainThread() override { |
| // Explicitly null these raw pointers to avoid that they become dangling. |
| autofill_external_delegate_ = nullptr; |
| autofill_driver_ = nullptr; |
| |
| InProcessBrowserTest::TearDownOnMainThread(); |
| } |
| |
| protected: |
| content::WebContents* web_contents() { |
| return browser()->tab_strip_model()->GetActiveWebContents(); |
| } |
| |
| content::RenderFrameHost* main_rfh() { |
| return web_contents()->GetPrimaryMainFrame(); |
| } |
| |
| BrowserAutofillManager& autofill_manager() { |
| return static_cast<BrowserAutofillManager&>( |
| *autofill_driver_->autofill_manager()); |
| } |
| |
| test::AutofillBrowserTestEnvironment autofill_test_environment_; |
| raw_ptr<ContentAutofillDriver> autofill_driver_ = nullptr; |
| raw_ptr<TestAutofillExternalDelegate> autofill_external_delegate_ = nullptr; |
| std::unique_ptr<ui::ScopedAnimationDurationScaleMode> disable_animation_; |
| }; |
| |
| #if BUILDFLAG(IS_MAC) |
| // Fails on Mac OS. http://crbug.com/453256 |
| #define MAYBE_HidePopupOnWindowMove DISABLED_HidePopupOnWindowMove |
| #else |
| #define MAYBE_HidePopupOnWindowMove HidePopupOnWindowMove |
| #endif |
| IN_PROC_BROWSER_TEST_F(AutofillPopupControllerBrowserTest, |
| MAYBE_HidePopupOnWindowMove) { |
| GenerateTestAutofillPopup(autofill_external_delegate_); |
| |
| EXPECT_FALSE(autofill_external_delegate_->popup_hidden()); |
| |
| // Move the window, which should close the popup. |
| gfx::Rect new_bounds = browser()->window()->GetBounds() - gfx::Vector2d(1, 1); |
| browser()->window()->SetBounds(new_bounds); |
| |
| autofill_external_delegate_->WaitForPopupHidden(); |
| EXPECT_TRUE(autofill_external_delegate_->popup_hidden()); |
| } |
| |
| IN_PROC_BROWSER_TEST_F(AutofillPopupControllerBrowserTest, |
| HidePopupOnWindowResize) { |
| GenerateTestAutofillPopup(autofill_external_delegate_); |
| |
| EXPECT_FALSE(autofill_external_delegate_->popup_hidden()); |
| |
| // Resize the window, which should cause the popup to hide. |
| gfx::Rect new_bounds = browser()->window()->GetBounds(); |
| new_bounds.Inset(1); |
| browser()->window()->SetBounds(new_bounds); |
| |
| autofill_external_delegate_->WaitForPopupHidden(); |
| EXPECT_TRUE(autofill_external_delegate_->popup_hidden()); |
| } |
| |
| // Tests that entering fullscreen hides the popup and, in particular, does not |
| // crash (crbug.com/1267047). |
| IN_PROC_BROWSER_TEST_F(AutofillPopupControllerBrowserTest, |
| HidePopupOnWindowEnterFullscreen) { |
| GenerateTestAutofillPopup(autofill_external_delegate_); |
| |
| EXPECT_FALSE(autofill_external_delegate_->popup_hidden()); |
| |
| // Enter fullscreen, which should cause the popup to hide. |
| ASSERT_FALSE(browser()->window()->IsFullscreen()); |
| content::WebContentsDelegate* wcd = browser(); |
| wcd->EnterFullscreenModeForTab(main_rfh(), {}); |
| ASSERT_TRUE(browser()->window()->IsFullscreen()); |
| |
| autofill_external_delegate_->WaitForPopupHidden(); |
| EXPECT_TRUE(autofill_external_delegate_->popup_hidden()); |
| } |
| |
| // Tests that exiting fullscreen hides the popup and, in particular, does not |
| // crash (crbug.com/1267047). |
| IN_PROC_BROWSER_TEST_F(AutofillPopupControllerBrowserTest, |
| HidePopupOnWindowExitFullscreen) { |
| content::WebContentsDelegate* wcd = browser(); |
| wcd->EnterFullscreenModeForTab(main_rfh(), {}); |
| |
| GenerateTestAutofillPopup(autofill_external_delegate_); |
| |
| EXPECT_FALSE(autofill_external_delegate_->popup_hidden()); |
| |
| // Exit fullscreen, which should cause the popup to hide. |
| ASSERT_TRUE(browser()->window()->IsFullscreen()); |
| wcd->ExitFullscreenModeForTab(web_contents()); |
| ASSERT_FALSE(browser()->window()->IsFullscreen()); |
| |
| autofill_external_delegate_->WaitForPopupHidden(); |
| EXPECT_TRUE(autofill_external_delegate_->popup_hidden()); |
| } |
| |
| // This test checks that the browser doesn't crash if the delegate is deleted |
| // before the popup is hidden. |
| #if BUILDFLAG(IS_MAC) |
| // Flaky on Mac 10.9 in debug mode. http://crbug.com/710439 |
| #define MAYBE_DeleteDelegateBeforePopupHidden \ |
| DISABLED_DeleteDelegateBeforePopupHidden |
| #else |
| #define MAYBE_DeleteDelegateBeforePopupHidden DeleteDelegateBeforePopupHidden |
| #endif |
| IN_PROC_BROWSER_TEST_F(AutofillPopupControllerBrowserTest, |
| MAYBE_DeleteDelegateBeforePopupHidden) { |
| GenerateTestAutofillPopup(autofill_external_delegate_); |
| |
| // Delete the external delegate here so that is gets deleted before popup is |
| // hidden. This can happen if the web_contents are destroyed before the popup |
| // is hidden. See http://crbug.com/232475 |
| autofill_external_delegate_ = nullptr; |
| autofill_manager().SetExternalDelegateForTest(nullptr); |
| autofill_driver_->set_autofill_manager(nullptr); |
| } |
| |
| // crbug.com/965025 |
| IN_PROC_BROWSER_TEST_F(AutofillPopupControllerBrowserTest, ResetSelectedLine) { |
| GenerateTestAutofillPopup(autofill_external_delegate_); |
| |
| auto* client = |
| autofill::ChromeAutofillClient::FromWebContentsForTesting(web_contents()); |
| AutofillPopupController* controller = |
| client->popup_controller_for_testing().get(); |
| ASSERT_TRUE(controller); |
| |
| // Push some suggestions and select the line #3. |
| std::vector<std::u16string> rows = {u"suggestion1", u"suggestion2", |
| u"suggestion3", u"suggestion4"}; |
| client->UpdateAutofillPopupDataListValues(rows, rows); |
| int original_suggestions_count = controller->GetLineCount(); |
| controller->SelectSuggestion(3u); |
| |
| // Replace the list with the smaller one. |
| rows = {u"suggestion1"}; |
| client->UpdateAutofillPopupDataListValues(rows, rows); |
| // Make sure that previously selected line #3 doesn't exist. |
| ASSERT_LT(controller->GetLineCount(), original_suggestions_count); |
| // Selecting a new line should not crash. |
| controller->SelectSuggestion(0u); |
| } |
| |
| } // namespace autofill |