| // Copyright 2016 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. |
| |
| #import "ios/chrome/test/earl_grey/chrome_earl_grey_ui.h" |
| |
| #include "components/strings/grit/components_strings.h" |
| #include "ios/chrome/browser/ui/authentication/signin_confirmation_view_controller.h" |
| #import "ios/chrome/browser/ui/settings/accounts_collection_view_controller.h" |
| #import "ios/chrome/browser/ui/settings/privacy_collection_view_controller.h" |
| #import "ios/chrome/browser/ui/settings/settings_collection_view_controller.h" |
| #import "ios/chrome/browser/ui/tools_menu/tools_menu_constants.h" |
| #import "ios/chrome/browser/ui/tools_menu/tools_popup_controller.h" |
| #import "ios/chrome/browser/ui/uikit_ui_util.h" |
| #include "ios/chrome/grit/ios_strings.h" |
| #import "ios/chrome/test/app/chrome_test_util.h" |
| #include "ios/chrome/test/app/navigation_test_util.h" |
| #import "ios/chrome/test/earl_grey/chrome_matchers.h" |
| #import "ios/testing/wait_util.h" |
| #import "ios/web/public/test/earl_grey/js_test_util.h" |
| #import "ios/web/public/test/earl_grey/web_view_matchers.h" |
| |
| #if !defined(__has_feature) || !__has_feature(objc_arc) |
| #error "This file requires ARC support." |
| #endif |
| |
| using chrome_test_util::AccountConsistencyConfirmationOkButton; |
| using chrome_test_util::AccountConsistencySetupSigninButton; |
| using chrome_test_util::ButtonWithAccessibilityLabel; |
| using chrome_test_util::ClearBrowsingDataCollectionView; |
| using chrome_test_util::SettingsMenuButton; |
| using testing::WaitUntilConditionOrTimeout; |
| using testing::kWaitForPageLoadTimeout; |
| |
| namespace { |
| |
| // Returns a GREYAction to scroll down (swipe up) for a reasonably small amount. |
| id<GREYAction> ScrollDown() { |
| // 150 is a reasonable value to ensure all menu items are seen, without too |
| // much delay. With a larger value, some menu items could be skipped while |
| // searching. A smaller value increses the area that is searched, but slows |
| // down the scroll. |
| CGFloat const kMenuScrollDisplacement = 150; |
| return grey_scrollInDirection(kGREYDirectionDown, kMenuScrollDisplacement); |
| } |
| } // namespace |
| |
| @implementation ChromeEarlGreyUI |
| |
| + (void)openToolsMenu { |
| // TODO(crbug.com/685570): Fix the tap instead of adding a delay. |
| GREYCondition* myCondition = [GREYCondition |
| conditionWithName:@"Delay to ensure the toolbar menu can be opened" |
| block:^BOOL { |
| return NO; |
| }]; |
| [myCondition waitWithTimeout:0.5]; |
| |
| // TODO(crbug.com/639524): Add logic to ensure the app is in the correct |
| // state, for example DCHECK if no tabs are displayed. |
| [[[EarlGrey |
| selectElementWithMatcher:grey_allOf(chrome_test_util::ToolsMenuButton(), |
| grey_sufficientlyVisible(), nil)] |
| usingSearchAction:grey_swipeSlowInDirection(kGREYDirectionDown) |
| onElementWithMatcher:web::WebViewScrollView( |
| chrome_test_util::GetCurrentWebState())] |
| performAction:grey_tap()]; |
| // TODO(crbug.com/639517): Add webViewScrollView matcher so we don't have |
| // to always find it. |
| } |
| |
| + (void)openSettingsMenu { |
| [ChromeEarlGreyUI openToolsMenu]; |
| id<GREYMatcher> interactableSettingsButton = |
| grey_allOf(SettingsMenuButton(), grey_interactable(), nil); |
| [[[EarlGrey selectElementWithMatcher:interactableSettingsButton] |
| usingSearchAction:ScrollDown() |
| onElementWithMatcher:grey_accessibilityID(kToolsMenuTableViewId)] |
| performAction:grey_tap()]; |
| } |
| |
| + (void)tapSettingsMenuButton:(id<GREYMatcher>)buttonMatcher { |
| id<GREYMatcher> interactableButtonMatcher = |
| grey_allOf(buttonMatcher, grey_interactable(), nil); |
| [[[EarlGrey selectElementWithMatcher:interactableButtonMatcher] |
| usingSearchAction:ScrollDown() |
| onElementWithMatcher:grey_accessibilityID(kSettingsCollectionViewId)] |
| performAction:grey_tap()]; |
| } |
| |
| + (void)tapClearBrowsingDataMenuButton:(id<GREYMatcher>)buttonMatcher { |
| id<GREYMatcher> interactableButtonMatcher = |
| grey_allOf(buttonMatcher, grey_interactable(), nil); |
| [[[EarlGrey selectElementWithMatcher:interactableButtonMatcher] |
| usingSearchAction:ScrollDown() |
| onElementWithMatcher:ClearBrowsingDataCollectionView()] |
| performAction:grey_tap()]; |
| } |
| |
| + (void)tapPrivacyMenuButton:(id<GREYMatcher>)buttonMatcher { |
| id<GREYMatcher> interactableButtonMatcher = |
| grey_allOf(buttonMatcher, grey_interactable(), nil); |
| [[[EarlGrey selectElementWithMatcher:interactableButtonMatcher] |
| usingSearchAction:ScrollDown() |
| onElementWithMatcher:grey_accessibilityID(kPrivacyCollectionViewId)] |
| performAction:grey_tap()]; |
| } |
| |
| + (void)tapAccountsMenuButton:(id<GREYMatcher>)buttonMatcher { |
| [[[EarlGrey selectElementWithMatcher:buttonMatcher] |
| usingSearchAction:ScrollDown() |
| onElementWithMatcher:grey_accessibilityID(kSettingsAccountsId)] |
| performAction:grey_tap()]; |
| } |
| |
| + (void)openNewTab { |
| [ChromeEarlGreyUI openToolsMenu]; |
| id<GREYMatcher> newTabButtonMatcher = |
| grey_accessibilityID(kToolsMenuNewTabId); |
| [[EarlGrey selectElementWithMatcher:newTabButtonMatcher] |
| performAction:grey_tap()]; |
| [[GREYUIThreadExecutor sharedInstance] drainUntilIdle]; |
| } |
| |
| + (void)openNewIncognitoTab { |
| [ChromeEarlGreyUI openToolsMenu]; |
| id<GREYMatcher> newIncognitoTabMatcher = |
| grey_accessibilityID(kToolsMenuNewIncognitoTabId); |
| [[EarlGrey selectElementWithMatcher:newIncognitoTabMatcher] |
| performAction:grey_tap()]; |
| [[GREYUIThreadExecutor sharedInstance] drainUntilIdle]; |
| } |
| |
| + (void)reload { |
| // On iPhone Reload button is a part of tools menu, so open it. |
| if (IsCompact()) { |
| [self openToolsMenu]; |
| } |
| [[EarlGrey selectElementWithMatcher:chrome_test_util::ReloadButton()] |
| performAction:grey_tap()]; |
| } |
| |
| + (void)openShareMenu { |
| if (IsCompact()) { |
| [ChromeEarlGreyUI openToolsMenu]; |
| } |
| [[EarlGrey selectElementWithMatcher:chrome_test_util::ShareButton()] |
| performAction:grey_tap()]; |
| } |
| |
| + (void)waitForToolbarVisible:(BOOL)isVisible { |
| const NSTimeInterval kWaitForToolbarAnimationTimeout = 1.0; |
| ConditionBlock condition = ^{ |
| NSError* error = nil; |
| id<GREYMatcher> visibleMatcher = isVisible ? grey_notNil() : grey_nil(); |
| [[EarlGrey selectElementWithMatcher:chrome_test_util::ToolsMenuButton()] |
| assertWithMatcher:visibleMatcher |
| error:&error]; |
| return error == nil; |
| }; |
| NSString* errorMessage = |
| isVisible ? @"Toolbar was not visible" : @"Toolbar was visible"; |
| GREYAssert(testing::WaitUntilConditionOrTimeout( |
| kWaitForToolbarAnimationTimeout, condition), |
| errorMessage); |
| } |
| |
| + (void)signInToIdentityByEmail:(NSString*)userEmail { |
| // Sign in to |userEmail|. |
| [[EarlGrey selectElementWithMatcher:ButtonWithAccessibilityLabel(userEmail)] |
| performAction:grey_tap()]; |
| [[EarlGrey selectElementWithMatcher:AccountConsistencySetupSigninButton()] |
| performAction:grey_tap()]; |
| } |
| |
| + (void)confirmSigninConfirmationDialog { |
| // Confirm sign in. "More" button is shown on short devices (e.g. iPhone 5s, |
| // iPhone SE), so needs to scroll first to dismiss the "More" button before |
| // taping on "OK". |
| // Cannot directly scroll on |kSignInConfirmationCollectionViewId| because it |
| // is a MDC collection view, not a UICollectionView, so itself is not |
| // scrollable. |
| id<GREYMatcher> signinUICollectionViewMatcher = grey_allOf( |
| grey_ancestor(grey_accessibilityID(kSigninConfirmationCollectionViewId)), |
| grey_kindOfClass([UICollectionView class]), nil); |
| [[EarlGrey selectElementWithMatcher:signinUICollectionViewMatcher] |
| performAction:grey_scrollToContentEdge(kGREYContentEdgeBottom)]; |
| |
| [[EarlGrey selectElementWithMatcher:AccountConsistencyConfirmationOkButton()] |
| performAction:grey_tap()]; |
| } |
| |
| @end |