blob: 3c236f82caa91f2a49b034361bea5b8ca7a01450 [file] [log] [blame]
// 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