Chromium Code Reviews| Index: chrome/android/javatests/src/org/chromium/chrome/browser/autofill/AutofillPopupWithKeyboardTest.java |
| diff --git a/chrome/android/javatests/src/org/chromium/chrome/browser/autofill/AutofillPopupWithKeyboardTest.java b/chrome/android/javatests/src/org/chromium/chrome/browser/autofill/AutofillPopupWithKeyboardTest.java |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..496be9a98b0c7718f882412a17625b7f745be2e2 |
| --- /dev/null |
| +++ b/chrome/android/javatests/src/org/chromium/chrome/browser/autofill/AutofillPopupWithKeyboardTest.java |
| @@ -0,0 +1,224 @@ |
| +// Copyright 2014 The Chromium Authors. All rights reserved. |
| +// Use of this source code is governed by a BSD-style license that can be |
| +// found in the LICENSE file. |
| + |
| +package org.chromium.chrome.browser.autofill; |
| + |
| +import android.test.suitebuilder.annotation.MediumTest; |
| +import android.view.View; |
| +import android.view.ViewGroup; |
| + |
| +import org.chromium.base.ThreadUtils; |
| +import org.chromium.base.test.util.Feature; |
| +import org.chromium.base.test.util.UrlUtils; |
| +import org.chromium.chrome.R; |
| +import org.chromium.chrome.browser.autofill.PersonalDataManager.AutofillProfile; |
| +import org.chromium.chrome.shell.ChromeShellActivity; |
| +import org.chromium.chrome.shell.ChromeShellTestBase; |
| +import org.chromium.content.browser.ContentViewCore; |
| +import org.chromium.content.browser.test.util.Criteria; |
| +import org.chromium.content.browser.test.util.CriteriaHelper; |
| +import org.chromium.content.browser.test.util.DOMUtils; |
| +import org.chromium.content_public.browser.WebContents; |
| +import org.chromium.ui.UiUtils; |
| +import org.chromium.ui.autofill.AutofillPopup; |
| + |
| +import java.util.concurrent.Callable; |
| +import java.util.concurrent.ExecutionException; |
| +import java.util.concurrent.TimeoutException; |
| + |
| +/** |
| + * Integration tests for interaction of the AutofillPopup and a keyboard. |
| + */ |
| +public class AutofillPopupWithKeyboardTest extends ChromeShellTestBase { |
| + /** |
| + * Test that showing autofill popup and keyboard will not hide the autofill popup. |
| + */ |
| + @MediumTest |
| + @Feature({"autofill-keyboard"}) |
| + public void testShowAutofillPopupAndKeyboardimultaneously() |
| + throws InterruptedException, ExecutionException, TimeoutException { |
| + launchChromeShellWithUrl(UrlUtils.encodeHtmlDataUri("<html><head>" |
| + + "<meta name=\"viewport\"" |
| + + "content=\"width=device-width, initial-scale=1.0, maximum-scale=1.0\" /></head>" |
| + + "<body><form method=\"POST\">" |
| + + "<input type=\"text\" id=\"fn\" autocomplete=\"given-name\" /><br>" |
| + + "<input type=\"text\" id=\"ln\" autocomplete=\"family-name\" /><br>" |
| + + "<textarea id=\"sa\" autocomplete=\"street-address\"></textarea><br>" |
| + + "<input type=\"text\" id=\"a1\" autocomplete=\"address-line1\" /><br>" |
| + + "<input type=\"text\" id=\"a2\" autocomplete=\"address-line2\" /><br>" |
| + + "<input type=\"text\" id=\"ct\" autocomplete=\"locality\" /><br>" |
| + + "<input type=\"text\" id=\"zc\" autocomplete=\"postal-code\" /><br>" |
| + + "<input type=\"text\" id=\"em\" autocomplete=\"email\" /><br>" |
| + + "<input type=\"text\" id=\"ph\" autocomplete=\"tel\" /><br>" |
| + + "<input type=\"text\" id=\"fx\" autocomplete=\"fax\" /><br>" |
| + + "<select id=\"co\" autocomplete=\"country\"><br>" |
| + + "<option value=\"BR\">Brazil</option>" |
| + + "<option value=\"US\">United States</option>" |
| + + "</select>" |
| + + "<input type=\"submit\" />" |
| + + "</form></body></html>")); |
| + assertTrue(waitForActiveShellToBeDoneLoading()); |
| + new AutofillTestHelper().setProfile(new AutofillProfile("", "https://www.example.com", |
| + "John Smith", "Acme Inc", "1 Main\nApt A", "CA", "San Francisco", "", "94102", "", |
| + "US", "(415) 888-9999", "john@acme.inc", "en")); |
| + class Info { |
|
newt (away)
2015/01/20 21:28:09
Another option would be to use AtomicReferences in
please use gerrit instead
2015/01/21 01:31:44
Done, that saves 2 lines :-).
|
| + public final ContentViewCore viewCore; |
| + public final WebContents webContents; |
| + public final ViewGroup view; |
| + public Info(ChromeShellActivity activity) { |
| + viewCore = activity.getActiveContentViewCore(); |
| + webContents = viewCore.getWebContents(); |
| + view = viewCore.getContainerView(); |
| + } |
| + } |
| + final Info info = ThreadUtils.runOnUiThreadBlocking(new Callable<Info>() { |
| + @Override |
| + public Info call() { |
| + return new Info(getActivity()); |
| + } |
| + }); |
| + assertTrue(DOMUtils.waitForNonZeroNodeBounds(info.webContents, "fn")); |
| + |
| + // Click on the unfocused input element the first time to focus on it. This brings up the |
| + // keyboard, but does not show the autofill popup. |
| + DOMUtils.clickNode(this, info.viewCore, "fn"); |
| + waitForKeyboardShownOnUiThreadBlocking(); |
| + notifyViewHeightReducedOnUiThreadBlocking(); |
| + |
| + // Hide the keyboard while still keeping the focus on the input field. |
| + ThreadUtils.runOnUiThreadBlocking(new Runnable() { |
| + @Override |
| + public void run() { |
| + UiUtils.hideKeyboard(info.view); |
| + } |
| + }); |
| + assertTrue("Keyboard was never hidden", CriteriaHelper.pollForCriteria(new Criteria() { |
| + @Override |
| + public boolean isSatisfied() { |
| + try { |
| + return !isKeyboardShowingOnUiThreadBlocking(); |
| + } catch (InterruptedException e1) { |
| + return false; |
| + } catch (ExecutionException e2) { |
| + return false; |
| + } |
| + } |
| + })); |
| + ThreadUtils.runOnUiThreadBlocking(new Runnable() { |
| + @Override |
| + public void run() { |
| + info.viewCore.onSizeChanged(info.viewCore.getViewportWidthPix(), |
| + info.viewCore.getViewportHeightPix() + 100, |
| + info.viewCore.getViewportWidthPix(), info.viewCore.getViewportHeightPix()); |
| + } |
| + }); |
| + |
| + // Click on the focused input element the second time. This brings up the autofill popup and |
| + // shows the keyboard at the same time. Showing the keyboard should not hide the autofill |
| + // popup. |
| + DOMUtils.clickNode(this, info.viewCore, "fn"); |
| + waitForKeyboardShownOnUiThreadBlocking(); |
| + notifyViewHeightReducedOnUiThreadBlocking(); |
| + |
| + // Verify that the autofill popup is showing. |
| + assertTrue("Autofill Popup anchor view was never added.", |
| + CriteriaHelper.pollForCriteria(new Criteria() { |
| + @Override |
| + public boolean isSatisfied() { |
| + try { |
| + return findAchorViewOnUiThreadBlocking(info.view) != null; |
| + } catch (InterruptedException e1) { |
| + return false; |
| + } catch (ExecutionException e2) { |
| + return false; |
| + } |
| + } |
| + })); |
| + final View anchorView = findAchorViewOnUiThreadBlocking(info.view); |
| + Object popupObject = ThreadUtils.runOnUiThreadBlocking(new Callable<Object>() { |
| + @Override |
| + public Object call() { |
| + return anchorView.getTag(); |
| + } |
| + }); |
| + assertTrue(popupObject instanceof AutofillPopup); |
| + final AutofillPopup popup = (AutofillPopup) popupObject; |
| + assertTrue( |
| + "Autofill Popup was never shown.", CriteriaHelper.pollForCriteria(new Criteria() { |
| + @Override |
| + public boolean isSatisfied() { |
| + try { |
| + return isPopupShowingOnUiThreadBlocking(popup); |
| + } catch (InterruptedException e1) { |
| + return false; |
| + } catch (ExecutionException e2) { |
| + return false; |
| + } |
| + } |
| + })); |
| + } |
| + |
| + // Wait until the keyboard is showing on the UI thread. |
| + private void waitForKeyboardShownOnUiThreadBlocking() throws InterruptedException { |
|
newt (away)
2015/01/20 21:28:10
you could add a "visible" parameter to this method
please use gerrit instead
2015/01/21 01:31:44
Done.
|
| + assertTrue("Keyboard was never shown", CriteriaHelper.pollForCriteria(new Criteria() { |
|
newt (away)
2015/01/20 21:28:09
use pollForUIThreadCriteria() here, then you can s
please use gerrit instead
2015/01/21 01:31:44
Done.
|
| + @Override |
| + public boolean isSatisfied() { |
| + try { |
| + return isKeyboardShowingOnUiThreadBlocking(); |
| + } catch (InterruptedException e1) { |
| + return false; |
| + } catch (ExecutionException e2) { |
| + return false; |
| + } |
| + } |
| + })); |
| + } |
| + |
| + // Notify the ContentViewCore that its height was reduced on the UI thread. |
| + private void notifyViewHeightReducedOnUiThreadBlocking() { |
|
newt (away)
2015/01/20 21:28:09
I'd remove "OnUiThreadBlocking" from these methods
please use gerrit instead
2015/01/21 01:31:44
Done.
|
| + ThreadUtils.runOnUiThreadBlocking(new Runnable() { |
| + @Override |
| + public void run() { |
| + ContentViewCore viewCore = getActivity().getActiveContentViewCore(); |
| + viewCore.onSizeChanged(viewCore.getViewportWidthPix(), |
| + viewCore.getViewportHeightPix() - 100, viewCore.getViewportWidthPix(), |
| + viewCore.getViewportHeightPix()); |
| + } |
| + }); |
| + } |
| + |
| + // Check whether the keyboard is showing on the UI thread. |
| + private Boolean isKeyboardShowingOnUiThreadBlocking() |
|
newt (away)
2015/01/20 21:28:09
This should return "boolean" (not Boolean). Same b
please use gerrit instead
2015/01/21 01:31:44
No longer relevant, as this method got inlined.
|
| + throws InterruptedException, ExecutionException { |
| + return ThreadUtils.runOnUiThreadBlocking(new Callable<Boolean>() { |
| + @Override |
| + public Boolean call() { |
| + return UiUtils.isKeyboardShowing( |
| + getActivity(), getActivity().getActiveContentViewCore().getContainerView()); |
| + } |
| + }); |
| + } |
| + |
| + // Find the autofill popup view on the UI thread. |
| + private View findAchorViewOnUiThreadBlocking(final ViewGroup view) |
| + throws InterruptedException, ExecutionException { |
| + return ThreadUtils.runOnUiThreadBlocking(new Callable<View>() { |
|
newt (away)
2015/01/20 21:28:09
I'd use runOnUiThreadBlockingNoException(). Then y
please use gerrit instead
2015/01/21 01:31:44
No longer necessary, as this got inlined.
|
| + @Override |
| + public View call() { |
| + return view.findViewById(R.id.dropdown_popup_window); |
| + } |
| + }); |
| + } |
| + |
| + // Check whether the autofill popup is showing on the UI thread. |
| + private Boolean isPopupShowingOnUiThreadBlocking(final AutofillPopup popup) |
| + throws InterruptedException, ExecutionException { |
| + return ThreadUtils.runOnUiThreadBlocking(new Callable<Boolean>() { |
| + @Override |
| + public Boolean call() { |
| + return popup.isShowing(); |
| + } |
| + }); |
| + } |
| +} |