| // 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 "ash/test/ash_test_helper.h" |
| |
| #include <algorithm> |
| |
| #include "ash/accelerators/accelerator_controller_impl.h" |
| #include "ash/accelerometer/accelerometer_reader.h" |
| #include "ash/ambient/test/ambient_ash_test_helper.h" |
| #include "ash/app_list/test/app_list_test_helper.h" |
| #include "ash/assistant/assistant_controller_impl.h" |
| #include "ash/assistant/test/test_assistant_service.h" |
| #include "ash/constants/ash_features.h" |
| #include "ash/constants/ash_switches.h" |
| #include "ash/display/display_configuration_controller_test_api.h" |
| #include "ash/display/screen_ash.h" |
| #include "ash/host/ash_window_tree_host.h" |
| #include "ash/keyboard/keyboard_controller_impl.h" |
| #include "ash/keyboard/test_keyboard_ui.h" |
| #include "ash/public/cpp/test/test_keyboard_controller_observer.h" |
| #include "ash/public/cpp/test/test_new_window_delegate.h" |
| #include "ash/quick_pair/common/fake_quick_pair_browser_delegate.h" |
| #include "ash/quick_pair/keyed_service/fake_quick_pair_mediator_factory.h" |
| #include "ash/quick_pair/keyed_service/quick_pair_mediator.h" |
| #include "ash/session/test_session_controller_client.h" |
| #include "ash/shell.h" |
| #include "ash/shell_init_params.h" |
| #include "ash/style/dark_light_mode_controller_impl.h" |
| #include "ash/system/geolocation/test_geolocation_url_loader_factory.h" |
| #include "ash/system/notification_center/session_state_notification_blocker.h" |
| #include "ash/system/model/system_tray_model.h" |
| #include "ash/system/screen_layout_observer.h" |
| #include "ash/test/ash_test_views_delegate.h" |
| #include "ash/test/pixel/ash_pixel_test_helper.h" |
| #include "ash/test/toplevel_window.h" |
| #include "ash/test_shell_delegate.h" |
| #include "ash/wallpaper/test_wallpaper_controller_client.h" |
| #include "ash/wallpaper/wallpaper_controller_impl.h" |
| #include "ash/wm/desks/templates/saved_desk_test_helper.h" |
| #include "ash/wm/overview/overview_controller.h" |
| #include "ash/wm/tablet_mode/tablet_mode_controller.h" |
| #include "base/run_loop.h" |
| #include "base/system/sys_info.h" |
| #include "base/system/system_monitor.h" |
| #include "chromeos/ash/components/audio/cras_audio_handler.h" |
| #include "chromeos/ash/components/dbus/audio/cras_audio_client.h" |
| #include "chromeos/ash/components/dbus/rgbkbd/rgbkbd_client.h" |
| #include "chromeos/ash/components/dbus/typecd/typecd_client.h" |
| #include "chromeos/ash/components/login/login_state/login_state.h" |
| #include "chromeos/ash/services/bluetooth_config/in_process_instance.h" |
| #include "chromeos/ash/services/hotspot_config/public/cpp/cros_hotspot_config_test_helper.h" |
| #include "chromeos/dbus/power/power_policy_controller.h" |
| #include "chromeos/ui/frame/multitask_menu/multitask_menu_nudge_controller.h" |
| #include "device/bluetooth/bluetooth_adapter_factory.h" |
| #include "device/bluetooth/dbus/bluez_dbus_manager.h" |
| #include "device/bluetooth/floss/floss_dbus_manager.h" |
| #include "device/bluetooth/floss/floss_features.h" |
| #include "ui/aura/test/test_windows.h" |
| #include "ui/aura/window.h" |
| #include "ui/aura/window_tree_host.h" |
| #include "ui/base/ime/ash/mock_input_method_manager_impl.h" |
| #include "ui/color/color_provider_manager.h" |
| #include "ui/display/display_switches.h" |
| #include "ui/display/manager/display_manager.h" |
| #include "ui/display/manager/test/fake_display_delegate.h" |
| #include "ui/display/manager/util/display_manager_test_util.h" |
| #include "ui/display/test/display_manager_test_api.h" |
| #include "ui/display/util/display_util.h" |
| #include "ui/events/gesture_detection/gesture_configuration.h" |
| #include "ui/gfx/geometry/point.h" |
| #include "ui/platform_window/common/platform_window_defaults.h" |
| #include "ui/views/test/views_test_helper_aura.h" |
| #include "ui/wm/core/capture_controller.h" |
| #include "ui/wm/core/cursor_manager.h" |
| #include "ui/wm/core/focus_controller.h" |
| |
| namespace ash { |
| |
| namespace { |
| std::unique_ptr<views::TestViewsDelegate> MakeTestViewsDelegate() { |
| return std::make_unique<AshTestViewsDelegate>(); |
| } |
| } // namespace |
| |
| AshTestHelper::InitParams::InitParams() = default; |
| AshTestHelper::InitParams::InitParams(InitParams&&) = default; |
| AshTestHelper::InitParams::~InitParams() = default; |
| |
| class AshTestHelper::BluezDBusManagerInitializer { |
| public: |
| BluezDBusManagerInitializer() { bluez::BluezDBusManager::InitializeFake(); } |
| ~BluezDBusManagerInitializer() { |
| device::BluetoothAdapterFactory::Shutdown(); |
| bluez::BluezDBusManager::Shutdown(); |
| } |
| }; |
| |
| class AshTestHelper::FlossDBusManagerInitializer { |
| public: |
| FlossDBusManagerInitializer() { floss::FlossDBusManager::InitializeFake(); } |
| ~FlossDBusManagerInitializer() { |
| device::BluetoothAdapterFactory::Shutdown(); |
| floss::FlossDBusManager::Shutdown(); |
| } |
| }; |
| |
| class AshTestHelper::PowerPolicyControllerInitializer { |
| public: |
| PowerPolicyControllerInitializer() { |
| chromeos::PowerPolicyController::Initialize( |
| chromeos::PowerManagerClient::Get()); |
| } |
| ~PowerPolicyControllerInitializer() { |
| chromeos::PowerPolicyController::Shutdown(); |
| } |
| }; |
| |
| AshTestHelper::AshTestHelper(ui::ContextFactory* context_factory) |
| : AuraTestHelper(context_factory), |
| system_monitor_(std::make_unique<base::SystemMonitor>()), |
| scoped_fake_federated_service_connection_for_test_( |
| &fake_federated_service_connection_) { |
| views::ViewsTestHelperAura::SetFallbackTestViewsDelegateFactory( |
| &MakeTestViewsDelegate); |
| |
| // TODO(jamescook): Can we do this without changing command line? |
| // Use the origin (1,1) so that it doesn't overlap with the native mouse |
| // cursor. |
| if (!base::SysInfo::IsRunningOnChromeOS() && |
| !command_line_->GetProcessCommandLine()->HasSwitch( |
| ::switches::kHostWindowBounds)) { |
| // TODO(oshima): Disable native events instead of adding offset. |
| command_line_->GetProcessCommandLine()->AppendSwitchASCII( |
| ::switches::kHostWindowBounds, "10+10-800x600"); |
| } |
| |
| TabletModeController::SetUseScreenshotForTest(false); |
| |
| display::ResetDisplayIdForTest(); |
| display::SetInternalDisplayIds({}); |
| |
| // Reset the global state for the cursor manager. This includes the |
| // last cursor visibility state, etc. |
| wm::CursorManager::ResetCursorVisibilityStateForTest(); |
| |
| // Clears the saved state so that test doesn't use on the wrong |
| // default state. |
| shell::ToplevelWindow::ClearSavedStateForTest(); |
| |
| // SimpleGeolocationProvider has to be initialized before |
| // GeolocationController, which is constructed during Shell::Init(). |
| SimpleGeolocationProvider::Initialize( |
| base::MakeRefCounted<TestGeolocationUrlLoaderFactory>()); |
| } |
| |
| AshTestHelper::~AshTestHelper() { |
| if (app_list_test_helper_) { |
| TearDown(); |
| } |
| |
| SimpleGeolocationProvider::DestroyForTesting(); |
| |
| // Ensure the next test starts with a null display::Screen. This must be done |
| // here instead of in TearDown() since some tests test access to the Screen |
| // after the shell shuts down (which they use TearDown() to trigger). |
| ScreenAsh::DeleteScreenForShutdown(); |
| |
| // This should never have a meaningful effect, since either there is no |
| // ViewsTestHelperAura instance or the instance is currently in its |
| // destructor. |
| views::ViewsTestHelperAura::SetFallbackTestViewsDelegateFactory(nullptr); |
| } |
| |
| void AshTestHelper::SetUp() { |
| SetUp(InitParams()); |
| } |
| |
| void AshTestHelper::TearDown() { |
| saved_desk_test_helper_.reset(); |
| |
| ambient_ash_test_helper_.reset(); |
| |
| // The AppListTestHelper holds a pointer to the AppListController the Shell |
| // owns, so shut the test helper down first. |
| app_list_test_helper_.reset(); |
| |
| // Stop event dispatch like we do in ChromeBrowserMainExtraPartsAsh. |
| Shell::Get()->ShutdownEventDispatch(); |
| |
| Shell::DeleteInstance(); |
| // Suspend the tear down until all resources are returned via |
| // CompositorFrameSinkClient::ReclaimResources() |
| base::RunLoop().RunUntilIdle(); |
| |
| LoginState::Shutdown(); |
| |
| TypecdClient::Shutdown(); |
| |
| if (create_global_cras_audio_handler_) { |
| CrasAudioHandler::Shutdown(); |
| CrasAudioClient::Shutdown(); |
| } |
| |
| // The PowerPolicyController holds a pointer to the PowerManagementClient, so |
| // shut the controller down first. |
| power_policy_controller_initializer_.reset(); |
| chromeos::PowerManagerClient::Shutdown(); |
| RgbkbdClient::Shutdown(); |
| |
| TabletModeController::SetUseScreenshotForTest(true); |
| |
| // Destroy all owned objects to prevent tests from depending on their state |
| // after this returns. |
| cros_hotspot_config_test_helper_.reset(); |
| test_keyboard_controller_observer_.reset(); |
| session_controller_client_.reset(); |
| test_views_delegate_.reset(); |
| new_window_delegate_provider_.reset(); |
| bluez_dbus_manager_initializer_.reset(); |
| floss_dbus_manager_initializer_.reset(); |
| system_tray_client_.reset(); |
| assistant_service_.reset(); |
| notifier_settings_controller_.reset(); |
| prefs_provider_.reset(); |
| statistics_provider_.reset(); |
| command_line_.reset(); |
| quick_pair_browser_delegate_.reset(); |
| |
| // Purge ColorProviderManager between tests so that we don't accumulate |
| // ColorProviderInitializers. crbug.com/1349232. |
| ui::ColorProviderManager::ResetForTesting(); |
| |
| AuraTestHelper::TearDown(); |
| |
| // Cleanup the global state for InputMethodManager, but only if |
| // it was setup by this test helper. This allows tests to implement |
| // their own override, and in that case we shouldn't call Shutdown |
| // otherwise the global state will be deleted twice. |
| if (input_method_manager_) { |
| input_method::InputMethodManager::Shutdown(); |
| input_method_manager_ = nullptr; |
| } |
| } |
| |
| aura::Window* AshTestHelper::GetContext() { |
| aura::Window* root_window = Shell::GetRootWindowForNewWindows(); |
| if (!root_window) { |
| root_window = Shell::GetPrimaryRootWindow(); |
| } |
| DCHECK(root_window); |
| return root_window; |
| } |
| |
| aura::WindowTreeHost* AshTestHelper::GetHost() { |
| auto* manager = Shell::Get()->window_tree_host_manager(); |
| const int64_t id = manager->GetPrimaryDisplayId(); |
| return manager->GetAshWindowTreeHostForDisplayId(id)->AsWindowTreeHost(); |
| } |
| |
| aura::TestScreen* AshTestHelper::GetTestScreen() { |
| // If a test needs this, we may need to refactor TestScreen such that its |
| // methods can operate atop some sort of real screen/host/display, and hook |
| // them to the ones provided by the shell. For now, not bothering. |
| NOTIMPLEMENTED(); |
| return nullptr; |
| } |
| |
| aura::client::FocusClient* AshTestHelper::GetFocusClient() { |
| return Shell::Get()->focus_controller(); |
| } |
| |
| aura::client::CaptureClient* AshTestHelper::GetCaptureClient() { |
| return wm::CaptureController::Get(); |
| } |
| |
| void AshTestHelper::SetUp(InitParams init_params) { |
| create_global_cras_audio_handler_ = |
| init_params.create_global_cras_audio_handler; |
| create_quick_pair_mediator_ = init_params.create_quick_pair_mediator; |
| |
| if (create_global_cras_audio_handler_) { |
| // Create `CrasAudioHandler` for testing since `g_browser_process` is not |
| // created in `AshTestBase` tests. |
| CrasAudioClient::InitializeFake(); |
| CrasAudioHandler::InitializeForTesting(); |
| } |
| |
| // Build `pixel_test_helper_` only for a pixel diff test. |
| if (init_params.pixel_test_init_params) { |
| // Constructing `pixel_test_helper_` sets the locale. Therefore, building |
| // `pixel_test_helper_` before the code that establishes the Ash UI. |
| pixel_test_helper_ = std::make_unique<AshPixelTestHelper>( |
| std::move(*init_params.pixel_test_init_params)); |
| } |
| |
| // This block of objects are conditionally initialized here rather than in the |
| // constructor to make it easier for test classes to override them. |
| if (!input_method::InputMethodManager::Get()) { |
| // |input_method_manager_| is not owned and is cleaned up in TearDown() |
| // by calling InputMethodManager::Shutdown(). |
| input_method_manager_ = new input_method::MockInputMethodManagerImpl(); |
| input_method::InputMethodManager::Initialize(input_method_manager_); |
| } |
| if (floss::features::IsFlossEnabled()) { |
| if (!floss::FlossDBusManager::IsInitialized()) { |
| floss_dbus_manager_initializer_ = |
| std::make_unique<FlossDBusManagerInitializer>(); |
| } |
| } else { |
| if (!bluez::BluezDBusManager::IsInitialized()) { |
| bluez_dbus_manager_initializer_ = |
| std::make_unique<BluezDBusManagerInitializer>(); |
| } |
| } |
| |
| if (!RgbkbdClient::Get()) { |
| RgbkbdClient::InitializeFake(); |
| } |
| if (!chromeos::PowerManagerClient::Get()) { |
| chromeos::PowerManagerClient::InitializeFake(); |
| } |
| if (!chromeos::PowerPolicyController::IsInitialized()) { |
| power_policy_controller_initializer_ = |
| std::make_unique<PowerPolicyControllerInitializer>(); |
| } |
| |
| if (!TypecdClient::Get()) { |
| TypecdClient::InitializeFake(); |
| } |
| |
| if (!NewWindowDelegate::GetInstance()) { |
| new_window_delegate_provider_ = |
| std::make_unique<TestNewWindowDelegateProvider>( |
| std::make_unique<TestNewWindowDelegate>()); |
| } |
| if (!views::ViewsDelegate::GetInstance()) { |
| test_views_delegate_ = MakeTestViewsDelegate(); |
| } |
| |
| if (features::IsHotspotEnabled()) { |
| cros_hotspot_config_test_helper_ = |
| std::make_unique<hotspot_config::CrosHotspotConfigTestHelper>( |
| /*use_fake_implementation=*/true); |
| } |
| |
| LoginState::Initialize(); |
| |
| ambient_ash_test_helper_ = std::make_unique<AmbientAshTestHelper>(); |
| quick_pair_browser_delegate_ = |
| std::make_unique<quick_pair::FakeQuickPairBrowserDelegate>(); |
| |
| ShellInitParams shell_init_params; |
| shell_init_params.delegate = std::move(init_params.delegate); |
| if (!shell_init_params.delegate) { |
| shell_init_params.delegate = std::make_unique<TestShellDelegate>(); |
| } |
| shell_init_params.context_factory = GetContextFactory(); |
| shell_init_params.local_state = init_params.local_state; |
| shell_init_params.keyboard_ui_factory = |
| std::make_unique<TestKeyboardUIFactory>(); |
| if (create_quick_pair_mediator_) { |
| shell_init_params.quick_pair_mediator_factory = |
| std::make_unique<quick_pair::FakeQuickPairMediatorFactory>(); |
| } |
| shell_init_params.native_display_delegate = |
| std::make_unique<display::FakeDisplayDelegate>(); |
| Shell::CreateInstance(std::move(shell_init_params)); |
| Shell* shell = Shell::Get(); |
| |
| chromeos::MultitaskMenuNudgeController::SetSuppressNudgeForTesting(true); |
| |
| // Set up a test wallpaper controller client before signing in any users. At |
| // the time a user logs in, Wallpaper controller relies on |
| // WallpaperControllerClient to check if user data should be synced. |
| wallpaper_controller_client_ = |
| std::make_unique<TestWallpaperControllerClient>(); |
| shell->wallpaper_controller()->SetClient(wallpaper_controller_client_.get()); |
| |
| // Disable the notification delay timer used to prevent non system |
| // notifications from showing up right after login. This needs to be done |
| // before any user sessions are added since the delay timer starts right |
| // after that. |
| SessionStateNotificationBlocker::SetUseLoginNotificationDelayForTest(false); |
| |
| // Cursor is visible by default in tests. |
| shell->cursor_manager()->ShowCursor(); |
| |
| shell->assistant_controller()->SetAssistant(assistant_service_.get()); |
| |
| shell->system_tray_model()->SetClient(system_tray_client_.get()); |
| |
| session_controller_client_ = std::make_unique<TestSessionControllerClient>( |
| shell->session_controller(), prefs_provider_.get()); |
| session_controller_client_->InitializeAndSetClient(); |
| if (init_params.start_session) { |
| session_controller_client_->CreatePredefinedUserSessions(1); |
| } |
| |
| // Requires the AppListController the Shell creates. |
| app_list_test_helper_ = std::make_unique<AppListTestHelper>(); |
| |
| Shell::GetPrimaryRootWindow()->Show(); |
| Shell::GetPrimaryRootWindow()->GetHost()->Show(); |
| |
| // Don't change the display size due to host size resize. |
| display::test::DisplayManagerTestApi(shell->display_manager()) |
| .DisableChangeDisplayUponHostResize(); |
| |
| // Create the test keyboard controller observer to respond to |
| // OnLoadKeyboardContentsRequested(). |
| test_keyboard_controller_observer_ = |
| std::make_unique<TestKeyboardControllerObserver>( |
| shell->keyboard_controller()); |
| |
| // Tests that change the display configuration generally don't care about the |
| // notifications and the popup UI can interfere with things like cursors. |
| shell->screen_layout_observer()->set_show_notifications_for_testing(false); |
| |
| // Disable display change animations in unit tests. |
| DisplayConfigurationControllerTestApi( |
| shell->display_configuration_controller()) |
| .SetDisplayAnimator(false); |
| |
| // Remove the app dragging animations delay for testing purposes. |
| shell->overview_controller()->set_delayed_animation_task_delay_for_test( |
| base::TimeDelta()); |
| |
| // Tests expect empty wallpaper. |
| shell->wallpaper_controller()->CreateEmptyWallpaperForTesting(); |
| |
| // Native events and mouse movements are disabled by |
| // `ui::DisableNativeUiEventDispatchDisabled()`. Just make sure that the the |
| // mouse cursour is not on the screen by default. |
| aura::Env::GetInstance()->SetLastMouseLocation(gfx::Point(-1000, -1000)); |
| shell->cursor_manager()->EnableMouseEvents(); |
| |
| // Changing GestureConfiguration shouldn't make tests fail. These values |
| // prevent unexpected events from being generated during tests. Such as |
| // delayed events which create race conditions on slower tests. |
| ui::GestureConfiguration* gesture_config = |
| ui::GestureConfiguration::GetInstance(); |
| gesture_config->set_max_touch_down_duration_for_click_in_ms(800); |
| gesture_config->set_long_press_time_in_ms(1000); |
| gesture_config->set_max_touch_move_in_pixels_for_click(5); |
| |
| // Fake the |ec_lid_angle_driver_status_| in the unittests. |
| AccelerometerReader::GetInstance()->SetECLidAngleDriverStatusForTesting( |
| ECLidAngleDriverStatus::NOT_SUPPORTED); |
| |
| if (TabletMode::IsBoardTypeMarkedAsTabletCapable()) { |
| shell->tablet_mode_controller()->OnDeviceListsComplete(); |
| } |
| |
| // Call `StabilizeUIForPixelTest()` after the user session is activated (if |
| // any) in the test setup. |
| if (pixel_test_helper_) { |
| StabilizeUIForPixelTest(); |
| } |
| |
| saved_desk_test_helper_ = std::make_unique<SavedDeskTestHelper>(); |
| } |
| |
| display::Display AshTestHelper::GetSecondaryDisplay() const { |
| return display::test::DisplayManagerTestApi(Shell::Get()->display_manager()) |
| .GetSecondaryDisplay(); |
| } |
| |
| void AshTestHelper::SimulateUserLogin(const AccountId& account_id, |
| user_manager::UserType user_type, |
| bool is_new_profile) { |
| session_controller_client_->AddUserSession( |
| account_id, account_id.GetUserEmail(), user_type, |
| /*provide_pref_service=*/true, is_new_profile); |
| session_controller_client_->SwitchActiveUser(account_id); |
| session_controller_client_->SetSessionState( |
| session_manager::SessionState::ACTIVE); |
| |
| if (pixel_test_helper_) { |
| pixel_test_helper_->StabilizeUi(); |
| } |
| } |
| |
| void AshTestHelper::StabilizeUIForPixelTest() { |
| pixel_test_helper_->StabilizeUi(); |
| } |
| |
| } // namespace ash |