blob: 66187e19f96e3026e31dbde6113b8a27ea001974 [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.
#include "ash/shell.h"
#include <memory>
#include <queue>
#include <vector>
#include "ash/accelerators//accelerator_tracker.h"
#include "ash/accessibility/chromevox/key_accessibility_enabler.h"
#include "ash/accessibility/magnifier/fullscreen_magnifier_controller.h"
#include "ash/constants/ash_features.h"
#include "ash/constants/ash_switches.h"
#include "ash/display/mouse_cursor_event_filter.h"
#include "ash/drag_drop/drag_drop_controller.h"
#include "ash/drag_drop/drag_drop_controller_test_api.h"
#include "ash/keyboard/ui/keyboard_ui_controller.h"
#include "ash/keyboard/ui/keyboard_util.h"
#include "ash/public/cpp/ash_prefs.h"
#include "ash/public/cpp/keyboard/keyboard_switches.h"
#include "ash/public/cpp/shell_window_ids.h"
#include "ash/public/cpp/test/shell_test_api.h"
#include "ash/root_window_controller.h"
#include "ash/session/session_controller_impl.h"
#include "ash/session/test_session_controller_client.h"
#include "ash/shelf/home_button.h"
#include "ash/shelf/shelf.h"
#include "ash/shelf/shelf_layout_manager.h"
#include "ash/shelf/shelf_navigation_widget.h"
#include "ash/shelf/shelf_widget.h"
#include "ash/system/status_area_widget.h"
#include "ash/test/ash_test_base.h"
#include "ash/test/ash_test_helper.h"
#include "ash/test/test_widget_builder.h"
#include "ash/test_shell_delegate.h"
#include "ash/wallpaper/views/wallpaper_widget_controller.h"
#include "ash/wm/desks/desks_util.h"
#include "ash/wm/overview/overview_controller.h"
#include "base/command_line.h"
#include "base/containers/flat_set.h"
#include "base/ranges/algorithm.h"
#include "base/strings/utf_string_conversions.h"
#include "base/test/scoped_feature_list.h"
#include "components/account_id/account_id.h"
#include "ui/aura/env.h"
#include "ui/aura/window.h"
#include "ui/aura/window_event_dispatcher.h"
#include "ui/base/models/simple_menu_model.h"
#include "ui/display/scoped_display_for_new_windows.h"
#include "ui/events/test/event_generator.h"
#include "ui/events/test/events_test_utils.h"
#include "ui/events/test/test_event_handler.h"
#include "ui/gfx/geometry/size.h"
#include "ui/views/controls/menu/menu_controller.h"
#include "ui/views/controls/menu/menu_runner.h"
#include "ui/views/widget/widget.h"
#include "ui/views/widget/widget_delegate.h"
#include "ui/views/window/dialog_delegate.h"
#include "ui/wm/core/accelerator_filter.h"
using aura::RootWindow;
namespace ash {
namespace {
aura::Window* GetActiveDeskContainer() {
return Shell::GetContainer(Shell::GetPrimaryRootWindow(),
desks_util::GetActiveDeskContainerId());
}
aura::Window* GetAlwaysOnTopContainer() {
return Shell::GetContainer(Shell::GetPrimaryRootWindow(),
kShellWindowId_AlwaysOnTopContainer);
}
// Expect ALL the containers!
void ExpectAllContainers() {
aura::Window* root_window = Shell::GetPrimaryRootWindow();
// Validate no duplicate container IDs.
base::flat_set<int> container_ids;
std::queue<aura::Window*> window_queue;
window_queue.push(root_window);
while (!window_queue.empty()) {
aura::Window* current_window = window_queue.front();
window_queue.pop();
for (aura::Window* child : current_window->children())
window_queue.push(child);
const int id = current_window->GetId();
// Skip windows with no IDs.
if (id == aura::Window::kInitialId)
continue;
EXPECT_TRUE(container_ids.insert(id).second)
<< "Found duplicate ID: " << id
<< " at window: " << current_window->GetName();
}
EXPECT_TRUE(
Shell::GetContainer(root_window, kShellWindowId_WallpaperContainer));
for (int desk_id : desks_util::GetDesksContainersIds())
EXPECT_TRUE(Shell::GetContainer(root_window, desk_id));
EXPECT_TRUE(
Shell::GetContainer(root_window, kShellWindowId_AlwaysOnTopContainer));
EXPECT_TRUE(Shell::GetContainer(root_window, kShellWindowId_ShelfContainer));
EXPECT_TRUE(
Shell::GetContainer(root_window, kShellWindowId_SystemModalContainer));
EXPECT_TRUE(Shell::GetContainer(root_window,
kShellWindowId_LockScreenWallpaperContainer));
EXPECT_TRUE(
Shell::GetContainer(root_window, kShellWindowId_LockScreenContainer));
EXPECT_TRUE(Shell::GetContainer(root_window,
kShellWindowId_LockSystemModalContainer));
EXPECT_TRUE(Shell::GetContainer(root_window, kShellWindowId_MenuContainer));
EXPECT_TRUE(Shell::GetContainer(root_window,
kShellWindowId_DragImageAndTooltipContainer));
EXPECT_TRUE(
Shell::GetContainer(root_window, kShellWindowId_SettingBubbleContainer));
EXPECT_TRUE(
Shell::GetContainer(root_window, kShellWindowId_OverlayContainer));
EXPECT_TRUE(Shell::GetContainer(root_window,
kShellWindowId_ImeWindowParentContainer));
EXPECT_TRUE(Shell::GetContainer(root_window,
kShellWindowId_VirtualKeyboardContainer));
EXPECT_TRUE(
Shell::GetContainer(root_window, kShellWindowId_MouseCursorContainer));
// Phantom window is not a container.
EXPECT_EQ(0u, container_ids.count(kShellWindowId_PhantomWindow));
EXPECT_FALSE(Shell::GetContainer(root_window, kShellWindowId_PhantomWindow));
}
std::unique_ptr<views::WidgetDelegateView> CreateModalWidgetDelegate() {
auto delegate = std::make_unique<views::WidgetDelegateView>();
delegate->SetCanResize(true);
delegate->SetModalType(ui::MODAL_TYPE_SYSTEM);
delegate->SetOwnedByWidget(true);
delegate->SetTitle(u"Modal Window");
return delegate;
}
class SimpleMenuDelegate : public ui::SimpleMenuModel::Delegate {
public:
SimpleMenuDelegate() = default;
SimpleMenuDelegate(const SimpleMenuDelegate&) = delete;
SimpleMenuDelegate& operator=(const SimpleMenuDelegate&) = delete;
~SimpleMenuDelegate() override = default;
bool IsCommandIdChecked(int command_id) const override { return false; }
bool IsCommandIdEnabled(int command_id) const override { return true; }
void ExecuteCommand(int command_id, int event_flags) override {}
};
} // namespace
class ShellTest : public AshTestBase {
public:
void TestCreateWindow(views::Widget::InitParams::Type type,
bool always_on_top,
aura::Window* expected_container) {
TestWidgetBuilder builder;
if (always_on_top)
builder.SetZOrderLevel(ui::ZOrderLevel::kFloatingWindow);
views::Widget* widget =
builder.SetWidgetType(type).BuildOwnedByNativeWidget();
EXPECT_TRUE(
expected_container->Contains(widget->GetNativeWindow()->parent()))
<< "TestCreateWindow: type=" << type
<< ", always_on_top=" << always_on_top;
widget->Close();
}
void LockScreenAndVerifyMenuClosed() {
// Verify a menu is open before locking.
views::MenuController* menu_controller =
views::MenuController::GetActiveInstance();
DCHECK(menu_controller);
EXPECT_EQ(views::MenuController::ExitType::kNone,
menu_controller->exit_type());
// Create a LockScreen window.
views::Widget* lock_widget =
TestWidgetBuilder()
.SetWidgetType(views::Widget::InitParams::TYPE_WINDOW)
.SetShow(false)
.BuildOwnedByNativeWidget();
Shell::GetContainer(Shell::GetPrimaryRootWindow(),
kShellWindowId_LockScreenContainer)
->AddChild(lock_widget->GetNativeView());
lock_widget->Show();
// Simulate real screen locker to change session state to LOCKED
// when it is shown.
GetSessionControllerClient()->LockScreen();
SessionControllerImpl* controller = Shell::Get()->session_controller();
EXPECT_TRUE(controller->IsScreenLocked());
EXPECT_TRUE(lock_widget->GetNativeView()->HasFocus());
// Verify menu is closed.
EXPECT_EQ(nullptr, views::MenuController::GetActiveInstance());
lock_widget->Close();
GetSessionControllerClient()->UnlockScreen();
}
};
TEST_F(ShellTest, CreateWindow) {
// Normal window should be created in default container.
TestCreateWindow(views::Widget::InitParams::TYPE_WINDOW,
false, // always_on_top
GetActiveDeskContainer());
TestCreateWindow(views::Widget::InitParams::TYPE_POPUP,
false, // always_on_top
GetActiveDeskContainer());
// Always-on-top window and popup are created in always-on-top container.
TestCreateWindow(views::Widget::InitParams::TYPE_WINDOW,
true, // always_on_top
GetAlwaysOnTopContainer());
TestCreateWindow(views::Widget::InitParams::TYPE_POPUP,
true, // always_on_top
GetAlwaysOnTopContainer());
}
// Verifies that a window with a preferred size is created centered on the
// default display for new windows. Mojo apps like shortcut_viewer rely on this
// behavior.
TEST_F(ShellTest, CreateWindowWithPreferredSize) {
UpdateDisplay("1024x768,800x600");
aura::Window* secondary_root = Shell::GetAllRootWindows()[1];
display::ScopedDisplayForNewWindows scoped_display(secondary_root);
views::Widget::InitParams params;
params.ownership = views::Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET;
// Don't specify bounds, parent or context.
{
auto delegate = std::make_unique<views::WidgetDelegateView>();
delegate->SetPreferredSize(gfx::Size(400, 300));
params.delegate = delegate.release();
}
views::Widget widget;
params.context = GetContext();
widget.Init(std::move(params));
// Widget is centered on secondary display.
EXPECT_EQ(secondary_root, widget.GetNativeWindow()->GetRootWindow());
EXPECT_EQ(GetSecondaryDisplay().work_area().CenterPoint(),
widget.GetRestoredBounds().CenterPoint());
}
TEST_F(ShellTest, ChangeZOrderLevel) {
// Creates a normal window.
views::Widget* widget = TestWidgetBuilder().BuildOwnedByNativeWidget();
// It should be in the active desk container.
EXPECT_TRUE(
GetActiveDeskContainer()->Contains(widget->GetNativeWindow()->parent()));
// Set the z-order to float.
widget->SetZOrderLevel(ui::ZOrderLevel::kFloatingWindow);
// And it should in always on top container now.
EXPECT_EQ(GetAlwaysOnTopContainer(), widget->GetNativeWindow()->parent());
// Put the z-order back to normal.
widget->SetZOrderLevel(ui::ZOrderLevel::kNormal);
// It should go back to the active desk container.
EXPECT_TRUE(
GetActiveDeskContainer()->Contains(widget->GetNativeWindow()->parent()));
// Set the z-order again to the normal value.
widget->SetZOrderLevel(ui::ZOrderLevel::kNormal);
// Should have no effect and we are still in the the active desk container.
EXPECT_TRUE(
GetActiveDeskContainer()->Contains(widget->GetNativeWindow()->parent()));
widget->Close();
}
TEST_F(ShellTest, CreateModalWindow) {
// Create a normal window.
views::Widget* widget = TestWidgetBuilder().BuildOwnedByNativeWidget();
// It should be in the active desk container.
EXPECT_TRUE(
GetActiveDeskContainer()->Contains(widget->GetNativeWindow()->parent()));
// Create a modal window.
views::Widget* modal_widget = views::Widget::CreateWindowWithParent(
CreateModalWidgetDelegate(), widget->GetNativeView());
modal_widget->Show();
// It should be in modal container.
aura::Window* modal_container = Shell::GetContainer(
Shell::GetPrimaryRootWindow(), kShellWindowId_SystemModalContainer);
EXPECT_EQ(modal_container, modal_widget->GetNativeWindow()->parent());
modal_widget->Close();
widget->Close();
}
TEST_F(ShellTest, CreateLockScreenModalWindow) {
// Create a normal window.
views::Widget* widget = TestWidgetBuilder().BuildOwnedByNativeWidget();
EXPECT_TRUE(widget->GetNativeView()->HasFocus());
// It should be in the active desk container.
EXPECT_TRUE(
GetActiveDeskContainer()->Contains(widget->GetNativeWindow()->parent()));
GetSessionControllerClient()->LockScreen();
// Create a LockScreen window.
views::Widget* lock_widget =
TestWidgetBuilder().SetShow(false).BuildOwnedByNativeWidget();
Shell::GetContainer(Shell::GetPrimaryRootWindow(),
kShellWindowId_LockScreenContainer)
->AddChild(lock_widget->GetNativeView());
lock_widget->Show();
EXPECT_TRUE(lock_widget->GetNativeView()->HasFocus());
// It should be in LockScreen container.
aura::Window* lock_screen = Shell::GetContainer(
Shell::GetPrimaryRootWindow(), kShellWindowId_LockScreenContainer);
EXPECT_EQ(lock_screen, lock_widget->GetNativeWindow()->parent());
// Create a modal window with a lock window as parent.
views::Widget* lock_modal_widget = views::Widget::CreateWindowWithParent(
CreateModalWidgetDelegate(), lock_widget->GetNativeView());
lock_modal_widget->Show();
EXPECT_TRUE(lock_modal_widget->GetNativeView()->HasFocus());
// It should be in LockScreen modal container.
aura::Window* lock_modal_container =
Shell::GetContainer(Shell::GetPrimaryRootWindow(),
kShellWindowId_LockSystemModalContainer);
EXPECT_EQ(lock_modal_container,
lock_modal_widget->GetNativeWindow()->parent());
// Create a modal window with a normal window as parent.
views::Widget* modal_widget = views::Widget::CreateWindowWithParent(
CreateModalWidgetDelegate(), widget->GetNativeView());
modal_widget->Show();
// Window on lock screen shouldn't lost focus.
EXPECT_FALSE(modal_widget->GetNativeView()->HasFocus());
EXPECT_TRUE(lock_modal_widget->GetNativeView()->HasFocus());
// It should be in non-LockScreen modal container.
aura::Window* modal_container = Shell::GetContainer(
Shell::GetPrimaryRootWindow(), kShellWindowId_SystemModalContainer);
EXPECT_EQ(modal_container, modal_widget->GetNativeWindow()->parent());
// Modal widget without parent, caused crash see crbug.com/226141
views::Widget* modal_dialog = views::DialogDelegate::CreateDialogWidget(
CreateModalWidgetDelegate(), GetContext(), nullptr);
modal_dialog->Show();
EXPECT_FALSE(modal_dialog->GetNativeView()->HasFocus());
EXPECT_TRUE(lock_modal_widget->GetNativeView()->HasFocus());
modal_dialog->Close();
modal_widget->Close();
modal_widget->Close();
lock_modal_widget->Close();
lock_widget->Close();
widget->Close();
}
TEST_F(ShellTest, IsScreenLocked) {
SessionControllerImpl* controller = Shell::Get()->session_controller();
GetSessionControllerClient()->LockScreen();
EXPECT_TRUE(controller->IsScreenLocked());
GetSessionControllerClient()->UnlockScreen();
EXPECT_FALSE(controller->IsScreenLocked());
}
TEST_F(ShellTest, LockScreenClosesActiveMenu) {
SimpleMenuDelegate menu_delegate;
std::unique_ptr<ui::SimpleMenuModel> menu_model(
new ui::SimpleMenuModel(&menu_delegate));
menu_model->AddItem(0, u"Menu item");
views::Widget* widget = Shell::GetPrimaryRootWindowController()
->wallpaper_widget_controller()
->GetWidget();
std::unique_ptr<views::MenuRunner> menu_runner(
new views::MenuRunner(menu_model.get(), views::MenuRunner::CONTEXT_MENU));
menu_runner->RunMenuAt(widget, nullptr, gfx::Rect(),
views::MenuAnchorPosition::kTopLeft,
ui::MENU_SOURCE_MOUSE);
LockScreenAndVerifyMenuClosed();
}
TEST_F(ShellTest, ManagedWindowModeBasics) {
// We start with the usual window containers.
ExpectAllContainers();
// Shelf is visible.
ShelfWidget* shelf_widget = GetPrimaryShelf()->shelf_widget();
EXPECT_TRUE(shelf_widget->IsVisible());
// Shelf is at bottom-left of screen.
EXPECT_EQ(0, shelf_widget->GetWindowBoundsInScreen().x());
EXPECT_EQ(
Shell::GetPrimaryRootWindow()->GetHost()->GetBoundsInPixels().height(),
shelf_widget->GetWindowBoundsInScreen().bottom());
// We have a wallpaper but not a bare layer.
// TODO (antrim): enable once we find out why it fails component build.
// WallpaperWidgetController* wallpaper =
// Shell::GetPrimaryRootWindow()->
// GetProperty(kWindowDesktopComponent);
// EXPECT_TRUE(wallpaper);
// EXPECT_TRUE(wallpaper->widget());
// EXPECT_FALSE(wallpaper->layer());
// Create a normal window. It is not maximized.
views::Widget* widget = TestWidgetBuilder()
.SetBounds(gfx::Rect(11, 22, 300, 400))
.BuildOwnedByNativeWidget();
EXPECT_FALSE(widget->IsMaximized());
// Clean up.
widget->Close();
}
// Tests that the cursor-filter is ahead of the drag-drop controller in the
// pre-target list.
TEST_F(ShellTest, TestPreTargetHandlerOrder) {
Shell* shell = Shell::Get();
ui::EventTargetTestApi test_api(shell);
ShellTestApi shell_test_api;
ui::EventHandlerList handlers = test_api.GetPreTargetHandlers();
ui::EventHandlerList::const_iterator cursor_filter =
base::ranges::find(handlers, shell->mouse_cursor_filter());
ui::EventHandlerList::const_iterator drag_drop =
base::ranges::find(handlers, shell_test_api.drag_drop_controller());
EXPECT_NE(handlers.end(), cursor_filter);
EXPECT_NE(handlers.end(), drag_drop);
EXPECT_GT(drag_drop, cursor_filter);
}
// Tests that the accelerator_tracker is ahead of the accelerator_filter in the
// pre-target list to make sure the accelerators won't be filtered out before
// getting AcceleratorTracker.
TEST_F(ShellTest, AcceleratorPreTargetHandlerOrder) {
Shell* shell = Shell::Get();
ui::EventTargetTestApi test_api(shell);
ui::EventHandlerList handlers = test_api.GetPreTargetHandlers();
ui::EventHandlerList::const_iterator accelerator_tracker =
base::ranges::find(handlers, shell->accelerator_tracker());
ui::EventHandlerList::const_iterator accelerator_filter =
base::ranges::find(handlers, shell->accelerator_filter());
EXPECT_NE(handlers.end(), accelerator_tracker);
EXPECT_NE(handlers.end(), accelerator_filter);
EXPECT_GT(accelerator_filter, accelerator_tracker);
}
TEST_F(ShellTest, TestAccessibilityHandlerOrder) {
Shell* shell = Shell::Get();
ui::EventTargetTestApi test_api(shell);
ShellTestApi shell_test_api;
ui::EventHandler select_to_speak;
shell->AddAccessibilityEventHandler(
&select_to_speak,
AccessibilityEventHandlerManager::HandlerType::kSelectToSpeak);
// Check ordering.
ui::EventHandlerList handlers = test_api.GetPreTargetHandlers();
ui::EventHandlerList::const_iterator cursor_filter =
base::ranges::find(handlers, shell->mouse_cursor_filter());
ui::EventHandlerList::const_iterator fullscreen_magnifier_filter =
base::ranges::find(handlers, shell->fullscreen_magnifier_controller());
ui::EventHandlerList::const_iterator chromevox_filter =
base::ranges::find(handlers, shell->key_accessibility_enabler());
ui::EventHandlerList::const_iterator select_to_speak_filter =
base::ranges::find(handlers, &select_to_speak);
EXPECT_NE(handlers.end(), cursor_filter);
EXPECT_NE(handlers.end(), fullscreen_magnifier_filter);
EXPECT_NE(handlers.end(), chromevox_filter);
EXPECT_NE(handlers.end(), select_to_speak_filter);
EXPECT_LT(cursor_filter, fullscreen_magnifier_filter);
EXPECT_LT(fullscreen_magnifier_filter, chromevox_filter);
EXPECT_LT(chromevox_filter, select_to_speak_filter);
// Removing works.
shell->RemoveAccessibilityEventHandler(&select_to_speak);
handlers = test_api.GetPreTargetHandlers();
cursor_filter = base::ranges::find(handlers, shell->mouse_cursor_filter());
fullscreen_magnifier_filter =
base::ranges::find(handlers, shell->fullscreen_magnifier_controller());
chromevox_filter =
base::ranges::find(handlers, shell->key_accessibility_enabler());
select_to_speak_filter = base::ranges::find(handlers, &select_to_speak);
EXPECT_NE(handlers.end(), cursor_filter);
EXPECT_NE(handlers.end(), fullscreen_magnifier_filter);
EXPECT_NE(handlers.end(), chromevox_filter);
EXPECT_EQ(handlers.end(), select_to_speak_filter);
// Ordering still works.
EXPECT_LT(cursor_filter, fullscreen_magnifier_filter);
EXPECT_LT(fullscreen_magnifier_filter, chromevox_filter);
// Adding another is correct.
ui::EventHandler docked_magnifier;
shell->AddAccessibilityEventHandler(
&docked_magnifier,
AccessibilityEventHandlerManager::HandlerType::kDockedMagnifier);
handlers = test_api.GetPreTargetHandlers();
cursor_filter = base::ranges::find(handlers, shell->mouse_cursor_filter());
fullscreen_magnifier_filter =
base::ranges::find(handlers, shell->fullscreen_magnifier_controller());
chromevox_filter =
base::ranges::find(handlers, shell->key_accessibility_enabler());
ui::EventHandlerList::const_iterator docked_magnifier_filter =
base::ranges::find(handlers, &docked_magnifier);
EXPECT_NE(handlers.end(), cursor_filter);
EXPECT_NE(handlers.end(), fullscreen_magnifier_filter);
EXPECT_NE(handlers.end(), docked_magnifier_filter);
EXPECT_NE(handlers.end(), chromevox_filter);
// Inserted in proper order.
EXPECT_LT(cursor_filter, fullscreen_magnifier_filter);
EXPECT_LT(fullscreen_magnifier_filter, docked_magnifier_filter);
EXPECT_LT(docked_magnifier_filter, chromevox_filter);
}
// Verifies an EventHandler added to Env gets notified from EventGenerator.
TEST_F(ShellTest, EnvPreTargetHandler) {
ui::test::TestEventHandler event_handler;
aura::Env::GetInstance()->AddPreTargetHandler(&event_handler);
ui::test::EventGenerator generator(Shell::GetPrimaryRootWindow());
generator.MoveMouseBy(1, 1);
EXPECT_NE(0, event_handler.num_mouse_events());
aura::Env::GetInstance()->RemovePreTargetHandler(&event_handler);
}
// Verifies that pressing tab on an empty shell (one with no windows visible)
// will put focus on the shelf. This enables keyboard only users to get to the
// shelf without knowing the more obscure accelerators. Tab should move focus to
// the home button, shift + tab to the status widget. From there, normal shelf
// tab behaviour takes over, and the shell no longer catches that event.
TEST_F(ShellTest, NoWindowTabFocus) {
ExpectAllContainers();
StatusAreaWidget* status_area_widget =
GetPrimaryShelf()->status_area_widget();
ShelfNavigationWidget* home_button = GetPrimaryShelf()->navigation_widget();
// Create a normal window. It is not maximized.
auto widget = CreateTestWidget();
// Hit tab with window open, and expect that focus is not on the navigation
// widget or status widget.
PressAndReleaseKey(ui::VKEY_TAB);
EXPECT_FALSE(home_button->GetNativeView()->HasFocus());
EXPECT_FALSE(status_area_widget->GetNativeView()->HasFocus());
// Minimize the window, hit tab and expect that focus is on the launcher.
widget->Minimize();
PressAndReleaseKey(ui::VKEY_TAB);
EXPECT_TRUE(home_button->GetNativeView()->HasFocus());
// Show (to steal focus back before continuing testing) and close the window.
widget->Show();
widget->Close();
EXPECT_FALSE(home_button->GetNativeView()->HasFocus());
// Confirm that pressing tab when overview mode is open does not go to home
// button. Tab should be handled by overview mode and not hit the shell event
// handler.
EnterOverview();
PressAndReleaseKey(ui::VKEY_TAB);
EXPECT_FALSE(home_button->GetNativeView()->HasFocus());
ExitOverview();
// Hit shift tab and expect that focus is on status widget.
PressAndReleaseKey(ui::VKEY_TAB, ui::EF_SHIFT_DOWN);
EXPECT_TRUE(status_area_widget->GetNativeView()->HasFocus());
}
class ShellPickerDisabledTest : public AshTestBase {
public:
ShellPickerDisabledTest() {
base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
command_line->AppendSwitchASCII(switches::kPickerFeatureKey, "hello");
}
private:
base::test::ScopedFeatureList feature_list_{features::kPicker};
};
TEST_F(ShellPickerDisabledTest, NoPickerControllerIfFeatureKeyIsWrong) {
EXPECT_FALSE(Shell::Get()->picker_controller());
}
// This verifies WindowObservers are removed when a window is destroyed after
// the Shell is destroyed. This scenario (aura::Windows being deleted after the
// Shell) occurs if someone is holding a reference to an unparented Window, as
// is the case with a RenderWidgetHostViewAura that isn't on screen. As long as
// everything is ok, we won't crash. If there is a bug, window's destructor will
// notify some deleted object (say VideoDetector or ActivationController) and
// this will crash.
class ShellTest2 : public AshTestBase {
public:
ShellTest2() = default;
ShellTest2(const ShellTest2&) = delete;
ShellTest2& operator=(const ShellTest2&) = delete;
~ShellTest2() override = default;
protected:
std::unique_ptr<aura::Window> window_;
};
TEST_F(ShellTest2, DontCrashWhenWindowDeleted) {
window_ = std::make_unique<aura::Window>(nullptr,
aura::client::WINDOW_TYPE_UNKNOWN);
window_->Init(ui::LAYER_NOT_DRAWN);
}
using ShellLoginTest = NoSessionAshTestBase;
TEST_F(ShellLoginTest, DragAndDropDisabledBeforeLogin) {
DragDropController* drag_drop_controller =
ShellTestApi().drag_drop_controller();
DragDropControllerTestApi drag_drop_controller_test_api(drag_drop_controller);
EXPECT_FALSE(drag_drop_controller_test_api.enabled());
SimulateUserLogin("user1@test.com");
EXPECT_TRUE(drag_drop_controller_test_api.enabled());
}
using NoDuplicateShellContainerIdsTest = AshTestBase;
TEST_F(NoDuplicateShellContainerIdsTest, ValidateContainersIds) {
ExpectAllContainers();
}
} // namespace ash