blob: 0561f26b6f91db136784f31960ebd36ac062271e [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef ASH_TEST_ASH_TEST_BASE_H_
#define ASH_TEST_ASH_TEST_BASE_H_
#include <stdint.h>
#include <memory>
#include <string>
#include <utility>
#include "ash/constants/app_types.h"
#include "ash/public/cpp/shell_window_ids.h"
#include "ash/session/test_session_controller_client.h"
#include "ash/system/privacy_hub/sensor_disabled_notification_delegate.h"
#include "ash/test/pixel/ash_pixel_test_init_params.h"
#include "ash/wm/desks/desks_util.h"
#include "ash/wm/overview/overview_types.h"
#include "base/compiler_specific.h"
#include "base/test/task_environment.h"
#include "base/threading/thread.h"
#include "base/traits_bag.h"
#include "components/prefs/testing_pref_service.h"
#include "components/user_manager/user_type.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "third_party/skia/include/core/SkColor.h"
#include "ui/aura/client/window_types.h"
#include "ui/aura/env.h"
#include "ui/compositor/test/test_context_factories.h"
#include "ui/display/display.h"
#include "ui/events/event_constants.h"
#include "ui/events/test/event_generator.h"
namespace aura {
class Window;
class WindowDelegate;
} // namespace aura
namespace base {
namespace test {
class TaskEnvironment;
}
} // namespace base
namespace chromeos {
class FakePowerManagerClient;
}
namespace display {
class Display;
class DisplayManager;
namespace test {
class DisplayManagerTestApi;
} // namespace test
} // namespace display
namespace gfx {
class Rect;
}
namespace views {
class View;
class Widget;
class WidgetDelegate;
} // namespace views
namespace ash {
class AmbientAshTestHelper;
class AppListTestHelper;
class AshPixelDiffer;
class AshTestHelper;
class NotificationCenterTray;
class Shelf;
class TestAppListClient;
class TestShellDelegate;
class TestSystemTrayClient;
class UnifiedSystemTray;
class WorkAreaInsets;
// Base class for most tests in //ash. Constructs ash::Shell and all its
// dependencies. Provides a user login session (use NoSessionAshTestBase for
// tests that start at the login screen or need unusual user types). Sets
// animation durations to zero via AshTestHelper/AuraTestHelper.
class AshTestBase : public testing::Test {
public:
// Constructs an AshTestBase with |traits| being forwarded to its
// TaskEnvironment. MainThreadType always defaults to UI and must not be
// specified.
template <typename... TaskEnvironmentTraits>
NOINLINE explicit AshTestBase(TaskEnvironmentTraits&&... traits)
: AshTestBase(std::make_unique<base::test::TaskEnvironment>(
base::test::TaskEnvironment::MainThreadType::UI,
std::forward<TaskEnvironmentTraits>(traits)...)) {}
// Alternatively a subclass may pass a TaskEnvironment directly.
explicit AshTestBase(
std::unique_ptr<base::test::TaskEnvironment> task_environment);
AshTestBase(const AshTestBase&) = delete;
AshTestBase& operator=(const AshTestBase&) = delete;
~AshTestBase() override;
// testing::Test:
void SetUp() override;
void SetUp(std::unique_ptr<TestShellDelegate> delegate);
void TearDown() override;
// Returns the notification center tray on the primary display.
static NotificationCenterTray* GetPrimaryNotificationCenterTray();
// Returns the Shelf for the primary display.
static Shelf* GetPrimaryShelf();
// Returns the unified system tray on the primary display.
static UnifiedSystemTray* GetPrimaryUnifiedSystemTray();
// Returns WorkAreaInsets for the primary display.
static WorkAreaInsets* GetPrimaryWorkAreaInsets();
// Update the display configuration as given in |display_specs|.
// See ash::DisplayManagerTestApi::UpdateDisplay for more details.
// Note: To properly specify the radii of display's panel upon startup, set it
// via specifying the command line switch `ash-host-window-bounds`.
void UpdateDisplay(const std::string& display_specs,
bool from_native_platform = false,
bool generate_new_ids = false);
// Returns a root Window. Usually this is the active root Window, but that
// method can return NULL sometimes, and in those cases, we fall back on the
// primary root Window.
aura::Window* GetContext();
// Creates and shows a widget. See ash/public/cpp/shell_window_ids.h for
// values for |container_id|.
static std::unique_ptr<views::Widget> CreateTestWidget(
views::WidgetDelegate* delegate = nullptr,
int container_id = desks_util::GetActiveDeskContainerId(),
const gfx::Rect& bounds = gfx::Rect(),
bool show = true);
// Creates a frameless widget for testing.
static std::unique_ptr<views::Widget> CreateFramelessTestWidget();
// Creates a widget with a visible WINDOW_TYPE_NORMAL window with the given
// |app_type|. If |app_type| is AppType::NON_APP, this window is considered a
// non-app window.
// If |bounds_in_screen| is empty the window is added to the primary root
// window, otherwise the window is added to the display matching
// |bounds_in_screen|. |shell_window_id| is the shell window id to give to
// the new window.
// If |delegate| is empty, a new |TestWidgetDelegate| instance will be set as
// this widget's delegate.
std::unique_ptr<aura::Window> CreateAppWindow(
const gfx::Rect& bounds_in_screen = gfx::Rect(),
AppType app_type = AppType::SYSTEM_APP,
int shell_window_id = kShellWindowId_Invalid,
views::WidgetDelegate* delegate = nullptr);
// Creates a visible window in the appropriate container. If
// |bounds_in_screen| is empty the window is added to the primary root
// window, otherwise the window is added to the display matching
// |bounds_in_screen|. |shell_window_id| is the shell window id to give to
// the new window.
// If |type| is WINDOW_TYPE_NORMAL this creates a views::Widget, otherwise
// this creates an aura::Window.
std::unique_ptr<aura::Window> CreateTestWindow(
const gfx::Rect& bounds_in_screen = gfx::Rect(),
aura::client::WindowType type = aura::client::WINDOW_TYPE_NORMAL,
int shell_window_id = kShellWindowId_Invalid);
// Creates a visible top-level window with a delegate.
std::unique_ptr<aura::Window> CreateToplevelTestWindow(
const gfx::Rect& bounds_in_screen = gfx::Rect(),
int shell_window_id = kShellWindowId_Invalid);
// Versions of the functions in aura::test:: that go through our shell
// StackingController instead of taking a parent.
aura::Window* CreateTestWindowInShellWithId(int id);
aura::Window* CreateTestWindowInShellWithBounds(const gfx::Rect& bounds);
aura::Window* CreateTestWindowInShellWithDelegate(
aura::WindowDelegate* delegate,
int id,
const gfx::Rect& bounds);
aura::Window* CreateTestWindowInShellWithDelegateAndType(
aura::WindowDelegate* delegate,
aura::client::WindowType type,
int id,
const gfx::Rect& bounds);
// Attach |window| to the current shell's root window.
void ParentWindowInPrimaryRootWindow(aura::Window* window);
// Returns the raw pointer carried by `pixel_differ_`.
AshPixelDiffer* GetPixelDiffer();
// Stabilizes the variable UI components (such as the battery view). It should
// be called after the active user changes since some UI components are
// associated with the active account.
void StabilizeUIForPixelTest();
// Returns the EventGenerator that uses screen coordinates and works
// across multiple displays. It creates a new generator if it
// hasn't been created yet.
ui::test::EventGenerator* GetEventGenerator();
// Convenience method to return the DisplayManager.
display::DisplayManager* display_manager();
// Convenience method to return the FakePowerManagerClient.
chromeos::FakePowerManagerClient* power_manager_client() const;
// Test if moving a mouse to |point_in_screen| warps it to another
// display.
bool TestIfMouseWarpsAt(ui::test::EventGenerator* event_generator,
const gfx::Point& point_in_screen);
// Presses and releases a key to simulate typing one character.
void PressAndReleaseKey(ui::KeyboardCode key_code, int flags = ui::EF_NONE);
// Moves the mouse to the center of the view and generates a left mouse button
// click event.
void LeftClickOn(const views::View* view);
// Moves the mouse to the center of the view and generates a right mouse
// button click event.
void RightClickOn(const views::View* view);
// Generates a tap event on the center of `view`.
void GestureTapOn(const views::View* view);
// Enters/Exits overview mode with the given animation type `type`.
bool EnterOverview(
OverviewEnterExitType type = OverviewEnterExitType::kNormal);
bool ExitOverview(
OverviewEnterExitType type = OverviewEnterExitType::kNormal);
// Sets shelf animation duration for all displays.
void SetShelfAnimationDuration(base::TimeDelta duration);
// Waits for shelf animation in all displays.
void WaitForShelfAnimation();
// Execute a list of tasks during a drag and drop sequence in the apps grid.
// This method should be called after the drag is initiated by long pressing
// over an app but before actually moving the pointer to drag the item. When
// the drag and drop sequence is not handled by DragDropController, the list
// of tasks is just run sequentially outside the loop
void MaybeRunDragAndDropSequenceForAppList(
std::list<base::OnceClosure>* tasks,
bool is_touch);
protected:
enum UserSessionBlockReason {
FIRST_BLOCK_REASON,
BLOCKED_BY_LOCK_SCREEN = FIRST_BLOCK_REASON,
BLOCKED_BY_LOGIN_SCREEN,
BLOCKED_BY_USER_ADDING_SCREEN,
NUMBER_OF_BLOCK_REASONS
};
// Returns the rotation currentl active for the display |id|.
static display::Display::Rotation GetActiveDisplayRotation(int64_t id);
// Returns the rotation currently active for the internal display.
static display::Display::Rotation GetCurrentInternalDisplayRotation();
// Creates init params to set up a pixel test. If the test is not pixel
// related, returns `std::nullopt`. This function should be overridden by ash
// pixel tests.
virtual std::optional<pixel_test::InitParams> CreatePixelTestInitParams()
const;
void set_start_session(bool start_session) { start_session_ = start_session; }
void set_create_global_cras_audio_handler(
bool create_global_cras_audio_handler) {
create_global_cras_audio_handler_ = create_global_cras_audio_handler;
}
void set_create_quick_pair_mediator(bool create_quick_pair_mediator) {
create_quick_pair_mediator_ = create_quick_pair_mediator;
}
base::test::TaskEnvironment* task_environment() {
return task_environment_.get();
}
TestingPrefServiceSimple* local_state() { return &local_state_; }
AshTestHelper* ash_test_helper() { return ash_test_helper_.get(); }
// Returns nullptr before SetUp() is called.
ui::InProcessContextFactory* GetContextFactory() {
return test_context_factories_
? test_context_factories_->GetContextFactory()
: nullptr;
}
void SetUserPref(const std::string& user_email,
const std::string& path,
const base::Value& value);
TestSessionControllerClient* GetSessionControllerClient();
TestSystemTrayClient* GetSystemTrayClient();
AppListTestHelper* GetAppListTestHelper();
TestAppListClient* GetTestAppListClient();
AmbientAshTestHelper* GetAmbientAshTestHelper();
// Emulates an ash session that have |session_count| user sessions running.
// Note that existing user sessions will be cleared.
void CreateUserSessions(int session_count);
// Simulates a user sign-in. It creates a new user session, adds it to
// existing user sessions and makes it the active user session.
//
// For convenience |user_email| is used to create an |AccountId|. For testing
// behavior where |AccountId|s are compared, prefer the method of the same
// name that takes an |AccountId| created with a valid storage key instead.
// See the documentation for|AccountId::GetUserEmail| for discussion.
void SimulateUserLogin(
const std::string& user_email,
user_manager::UserType user_type = user_manager::UserType::kRegular);
// Simulates a user sign-in. It creates a new user session, adds it to
// existing user sessions and makes it the active user session.
void SimulateUserLogin(
const AccountId& account_id,
user_manager::UserType user_type = user_manager::UserType::kRegular);
// Simular to SimulateUserLogin but for a newly created user first ever login.
void SimulateNewUserFirstLogin(const std::string& user_email);
// Similar to SimulateUserLogin but for a guest user.
void SimulateGuestLogin();
// Simulates kiosk mode. |user_type| must correlate to a kiosk type user.
void SimulateKioskMode(user_manager::UserType user_type);
// Simulates setting height of the accessibility panel.
// Note: Accessibility panel widget needs to be setup first.
void SetAccessibilityPanelHeight(int panel_height);
// Clears all user sessions and resets to the primary login screen state.
void ClearLogin();
// Emulates whether the active user can lock screen.
void SetCanLockScreen(bool can_lock);
// Emulates whether the screen should be locked automatically.
void SetShouldLockScreenAutomatically(bool should_lock);
// Emulates whether the user adding screen is running.
void SetUserAddingScreenRunning(bool user_adding_screen_running);
// Methods to emulate blocking and unblocking user session with given
// |block_reason|.
void BlockUserSession(UserSessionBlockReason block_reason);
void UnblockUserSession();
// Enable or disable the virtual on-screen keyboard and run the message loop
// to allow observer operations to complete.
void SetVirtualKeyboardEnabled(bool enabled);
void DisableIME();
// Swap the primary display with the secondary.
void SwapPrimaryDisplay();
display::Display GetPrimaryDisplay() const;
display::Display GetSecondaryDisplay() const;
private:
void CreateWindowTreeIfNecessary();
// Prepares for pixel tests by enabling related flags and building
// `ash_test_helper_`.
void PrepareForPixelDiffTest();
bool setup_called_ = false;
bool teardown_called_ = false;
// SetUp() doesn't activate session if this is set to false.
bool start_session_ = true;
// `SetUp()` doesn't create a global `CrasAudioHandler` instance if this is
// set to false.
bool create_global_cras_audio_handler_ = true;
// `SetUp()` doesn't create a global `QuickPairMediator` instance if this is
// set to false.
bool create_quick_pair_mediator_ = true;
// |task_environment_| is initialized-once at construction time but
// subclasses may elect to provide their own.
std::unique_ptr<base::test::TaskEnvironment> task_environment_;
// A pref service used for local state.
TestingPrefServiceSimple local_state_;
// A helper class to take screen shots then compare with benchmarks. Set by
// `PrepareForPixelDiffTest()`.
std::unique_ptr<AshPixelDiffer> pixel_differ_;
std::unique_ptr<ui::TestContextFactories> test_context_factories_;
// Must be constructed after |task_environment_|.
std::unique_ptr<AshTestHelper> ash_test_helper_;
std::unique_ptr<ui::test::EventGenerator> event_generator_;
std::unique_ptr<ScopedSensorDisabledNotificationDelegateForTest>
scoped_disabled_notification_delegate_;
};
class NoSessionAshTestBase : public AshTestBase {
public:
NoSessionAshTestBase();
explicit NoSessionAshTestBase(
base::test::TaskEnvironment::TimeSource time_source);
NoSessionAshTestBase(const NoSessionAshTestBase&) = delete;
NoSessionAshTestBase& operator=(const NoSessionAshTestBase&) = delete;
~NoSessionAshTestBase() override;
};
} // namespace ash
#endif // ASH_TEST_ASH_TEST_BASE_H_