blob: 1275851c70a967353bd6b62b589804264fda488b [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 "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