1 // Copyright 2015 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "base/strings/stringprintf.h"
6 #include "chrome/browser/autofill/autofill_uitest_util.h"
7 #include "chrome/browser/chrome_notification_types.h"
8 #include "chrome/browser/ui/browser_window.h"
9 #include "chrome/browser/ui/chrome_pages.h"
10 #include "chrome/browser/ui/tabs/tab_strip_model.h"
11 #include "chrome/common/url_constants.h"
12 #include "chrome/test/base/in_process_browser_test.h"
13 #include "chrome/test/base/interactive_test_utils.h"
14 #include "components/autofill/core/browser/autofill_profile.h"
15 #include "components/autofill/core/browser/autofill_test_utils.h"
16 #include "components/autofill/core/browser/personal_data_manager.h"
17 #include "content/public/test/browser_test_utils.h"
21 // This class tests the Autofill options settings.
22 // This test is part of the interactive_ui_tests instead of browser_tests
23 // because it is necessary to emulate pushing the tab key.
24 class AutofillOptionsWebUITest
: public InProcessBrowserTest
{
26 AutofillOptionsWebUITest() {}
28 // Navigate to the autofillEditAddress page.
29 void SetUpOnMainThread() override
{
30 const GURL url
= chrome::GetSettingsUrl("autofillEditAddress");
31 ui_test_utils::NavigateToURL(browser(), url
);
35 const std::string kEditAddressOverlaySelector
=
36 "#autofill-edit-address-overlay";
38 content::RenderFrameHost
* GetActiveFrame() {
39 return GetActiveWebContents()->GetFocusedFrame();
42 content::RenderViewHost
* GetRenderViewHost() {
43 return GetActiveWebContents()->GetRenderViewHost();
46 content::WebContents
* GetActiveWebContents() {
47 return browser()->tab_strip_model()->GetActiveWebContents();
50 void CreateTestProfile() {
51 autofill::AddTestProfile(browser(), autofill::test::GetFullProfile());
54 // Returns true if element contains document.activeElement.
55 bool ContainsActiveElement(const std::string
& element_selector
) {
56 const std::string script
= base::StringPrintf(
57 "domAutomationController.send("
58 "document.querySelector('%s').contains(document.activeElement));",
59 element_selector
.c_str());
61 EXPECT_TRUE(content::ExecuteScriptAndExtractBool(
68 // Returns the number of items in the list.
69 int GetListSize(const std::string
& list_selector
) {
70 const std::string script
= base::StringPrintf(
71 "domAutomationController.send("
72 "document.querySelector('%s').items.length);",
73 list_selector
.c_str());
75 EXPECT_TRUE(content::ExecuteScriptAndExtractInt(
82 // Focus the first input field of the first list item.
83 void FocusFirstListItemInput(const std::string
& list_selector
) {
84 const std::string script
= base::StringPrintf(
85 "document.querySelector('%s input').focus();",
86 list_selector
.c_str());
87 EXPECT_TRUE(content::ExecuteScript(GetActiveFrame(), script
));
90 // Returns the text of the first item in the list.
91 std::string
GetFirstListItemText(const std::string
& list_selector
) {
92 // EXPECT_TRUE will fail if there is no first item or first item does not
94 const std::string script
= base::StringPrintf(
95 "domAutomationController.send("
96 "document.querySelector('%s input').value);",
97 list_selector
.c_str());
99 EXPECT_TRUE(content::ExecuteScriptAndExtractString(
106 // Returns true if the first item in the list has 'selected' attribute.
107 bool GetFirstListItemSelected(const std::string
& list_selector
) {
108 // EXPECT_TRUE will fail if there is no first item.
109 const std::string script
= base::StringPrintf(
110 "domAutomationController.send("
111 "document.querySelector('%s').items[0].hasAttribute('selected'));",
112 list_selector
.c_str());
114 EXPECT_TRUE(content::ExecuteScriptAndExtractBool(
121 // Returns true if a row delete button ('X' button) is focused.
122 bool GetDeleteButtonFocused() {
123 const std::string script
=
124 "domAutomationController.send("
125 "document.activeElement.classList.contains('row-delete-button'));";
127 EXPECT_TRUE(content::ExecuteScriptAndExtractBool(
134 // Insert text into currently focused element.
135 void InsertText(const std::string
& text
) {
136 ASSERT_EQ(std::string::npos
, text
.find("'"));
137 const std::string script
= base::StringPrintf(
138 "document.execCommand('insertText', false, '%s');",
140 EXPECT_TRUE(content::ExecuteScript(GetActiveFrame(), script
));
143 // Press and release tab key in the browser. This will wait for the element on
144 // the page to change.
145 bool PressTab(bool shift
) {
146 return ui_test_utils::SendKeyPressAndWait(
153 content::NOTIFICATION_FOCUS_CHANGED_IN_PAGE
,
154 content::Source
<content::RenderViewHost
>(GetRenderViewHost()));
157 void InitializeDomMessageQueue() {
158 dom_message_queue_
.reset(new content::DOMMessageQueue
);
161 // Wait for a message from the DOM automation controller.
162 void WaitForDomMessage(const std::string
& message
) {
163 const std::string expected
= "\"" + message
+ "\"";
164 std::string received
;
166 ASSERT_TRUE(dom_message_queue_
->WaitForMessage(&received
));
167 } while (received
!= expected
);
170 void ListenForFirstItemSelected(const std::string
& list_selector
) {
171 const std::string script
= base::StringPrintf(
172 "document.querySelector('%s').items[0].addEventListener("
173 "'selectedChange', function(e) {"
175 "domAutomationController.setAutomationId(0);"
176 "domAutomationController.send('first item selected');"
179 list_selector
.c_str());
181 EXPECT_TRUE(content::ExecuteScript(
186 void ListenForCommitEdit(const std::string
& list_selector
) {
187 const std::string script
= base::StringPrintf(
188 "document.querySelector('%s').addEventListener("
189 "'commitedit', function() {"
190 "domAutomationController.setAutomationId(0);"
191 "domAutomationController.send('done commitedit');"
193 list_selector
.c_str());
195 EXPECT_TRUE(content::ExecuteScript(
200 // Add an event listener to send a DOM automation controller message from
201 // JavaScript each time validation completes for the list.
202 void ListenForDoneValidating(const std::string
& list_selector
) {
203 // doneValidating will execute the 'then' function immediately if no
204 // validations are pending, so wait for 'commitedit' event before calling
206 const std::string script
= base::StringPrintf(
207 "document.querySelector('%s').addEventListener('commitedit',"
209 "document.querySelector('%s').doneValidating().then(function() {"
210 "domAutomationController.setAutomationId(0);"
211 "domAutomationController.send('done validating');"
214 list_selector
.c_str(),
215 list_selector
.c_str());
217 EXPECT_TRUE(content::ExecuteScript(
222 // Verifies that everything is the way it should be after list item is
224 void VerifyEditAddressListPostConditions(const std::string
& list_selector
,
225 const std::string
& input_text
,
226 bool list_requires_validation
) {
227 // Verify that neither the list nor any of its children still have focus.
228 EXPECT_FALSE(ContainsActiveElement(list_selector
));
230 // Verify that focus moved to a different element of the overlay.
231 EXPECT_TRUE(ContainsActiveElement(kEditAddressOverlaySelector
));
233 // Verify that list has exactly two items. They will be the item that was
234 // just added/modified + the placeholder.
235 EXPECT_EQ(2, GetListSize(list_selector
));
237 // Verify that the first list item has the string that was inserted.
238 EXPECT_EQ(input_text
, GetFirstListItemText(list_selector
));
240 // TODO(bondd): phone list doesn't select first item after validation.
241 // It becomes selected later when the list is given focus.
242 if (!list_requires_validation
) {
243 // Verify that the first list item is the selected item in the list.
244 EXPECT_TRUE(GetFirstListItemSelected(list_selector
));
248 // Make sure that when text is entered in the placeholder of an empty list and
249 // the tab key is pressed:
250 // + Focus leaves the list and goes to a different element on the page.
251 // + The text stays added and a new placeholder is created.
252 // + The list item with the newly added text is the selected list item (not
255 // Added to prevent http://crbug.com/440760 from regressing again.
256 void TestEditAddressListTabKeyAddItem(const std::string
& list_selector
,
257 const std::string
& input_text
,
258 bool list_requires_validation
) {
259 // Focus the input field and insert test string.
260 FocusFirstListItemInput(list_selector
);
261 WaitForDomMessage("first item selected");
263 InsertText(input_text
);
265 // Press tab key to move to next element after the list.
268 if (list_requires_validation
)
269 WaitForDomMessage("done validating");
271 WaitForDomMessage("done commitedit");
273 // Make sure everything ended up the way it should be.
274 VerifyEditAddressListPostConditions(list_selector
, input_text
,
275 list_requires_validation
);
278 // Depends on state set up by TestEditAddressListTabKeyAddItem. Should be
279 // called immediately after that method.
281 // Make sure that when a list item's text is edited and the tab key is
283 // + After the first tab press the item's delete button is focused.
284 // + After the second tab press focus leaves the list and goes to a
285 // different element on the page.
286 // + The edited text persists.
287 // + The edited list item is the selected list item.
289 // Added to prevent http://crbug.com/443491 from regressing again.
290 void TestEditAddressListTabKeyEditItem(const std::string
& list_selector
,
291 const std::string
& input_text
,
292 bool list_requires_validation
,
293 bool skip_ok_button
) {
297 // Press shift+tab to move back to the first list item's delete button.
299 EXPECT_TRUE(GetDeleteButtonFocused());
301 // Press shift+tab to move back to the first list item's input field.
303 // Verify that the first item in the list is focused.
304 EXPECT_TRUE(ContainsActiveElement(list_selector
+ " input"));
306 // Insert modified text in the first list item.
307 std::string second_input
= "second" + input_text
;
308 InsertText(second_input
);
310 // Press tab key to focus the list item's delete button.
312 EXPECT_TRUE(GetDeleteButtonFocused());
314 // Press tab key again to move to next element after the list.
317 if (list_requires_validation
)
318 WaitForDomMessage("done validating");
320 WaitForDomMessage("done commitedit");
322 // Make sure everything ended up the way it should be.
323 VerifyEditAddressListPostConditions(list_selector
, second_input
,
324 list_requires_validation
);
327 void TestEditAddressListTabKey(const std::string
& field_name
,
328 const std::string
& input_text
,
329 bool list_requires_validation
,
330 bool skip_ok_button
) {
331 const std::string list_selector
= kEditAddressOverlaySelector
+ " [field=" +
334 InitializeDomMessageQueue();
335 ListenForFirstItemSelected(list_selector
);
336 if (list_requires_validation
)
337 ListenForDoneValidating(list_selector
);
339 ListenForCommitEdit(list_selector
);
341 TestEditAddressListTabKeyAddItem(list_selector
, input_text
,
342 list_requires_validation
);
343 TestEditAddressListTabKeyEditItem(list_selector
, input_text
,
344 list_requires_validation
, skip_ok_button
);
348 scoped_ptr
<content::DOMMessageQueue
> dom_message_queue_
;
350 DISALLOW_COPY_AND_ASSIGN(AutofillOptionsWebUITest
);
355 // Test the 'fullName' InlineEditableItemList in autofillEditAddress overlay.
356 IN_PROC_BROWSER_TEST_F(AutofillOptionsWebUITest
,
357 TestEditAddressNameTabKey
) {
358 TestEditAddressListTabKey("fullName", "Test Name", false, false);
361 // Test the 'phone' InlineEditableItemList in autofillEditAddress overlay.
362 IN_PROC_BROWSER_TEST_F(AutofillOptionsWebUITest
,
363 TestEditAddressPhoneTabKey
) {
365 TestEditAddressListTabKey("phone", "123-456-7890", true, false);
368 // Test the 'email' InlineEditableItemList in autofillEditAddress overlay.
369 IN_PROC_BROWSER_TEST_F(AutofillOptionsWebUITest
,
370 TestEditAddressEmailTabKey
) {
371 #if defined(OS_WIN) || defined(OS_CHROMEOS)
372 // Button strip order is 'Cancel' and then 'OK' on Linux and Mac. 'OK' and
373 // then 'Cancel' on Windows and CrOS.
374 bool skip_ok_button
= true;
376 bool skip_ok_button
= false;
379 TestEditAddressListTabKey("email", "test@example.com", false, skip_ok_button
);