| // 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<algorithm> |
| #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/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/aura/window_observer.h" |
| #include"ui/base/mojom/menu_source_type.mojom.h" |
| #include"ui/base/mojom/ui_base_types.mojom-shared.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/menus/simple_menu_model.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" |
| #include"ui/wm/core/focus_controller.h" |
| #include"ui/wm/core/window_util.h" |
| |
| using aura::RootWindow; |
| |
| namespace ash{ |
| |
| namespace{ |
| |
| aura::Window*GetActiveDeskContainer(){ |
| returnShell::GetContainer(Shell::GetPrimaryRootWindow(), |
| desks_util::GetActiveDeskContainerId()); |
| } |
| |
| aura::Window*GetAlwaysOnTopContainer(){ |
| returnShell::GetContainer(Shell::GetPrimaryRootWindow(), |
| kShellWindowId_AlwaysOnTopContainer); |
| } |
| |
| // Expect ALL the containers! |
| voidExpectAllContainers(){ |
| 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); |
| |
| constint 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_CaptureModeSearchResultsPanel)); |
| 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)); |
| } |
| |
| classSimpleMenuDelegate:public ui::SimpleMenuModel::Delegate{ |
| public: |
| SimpleMenuDelegate()=default; |
| |
| SimpleMenuDelegate(constSimpleMenuDelegate&)=delete; |
| SimpleMenuDelegate&operator=(constSimpleMenuDelegate&)=delete; |
| |
| ~SimpleMenuDelegate() override=default; |
| |
| boolIsCommandIdChecked(int command_id)const override{returnfalse;} |
| |
| boolIsCommandIdEnabled(int command_id)const override{returntrue;} |
| |
| voidExecuteCommand(int command_id,int event_flags) override{} |
| }; |
| |
| }// namespace |
| |
| classShellTest:publicAshTestBase{ |
| public: |
| static std::unique_ptr<views::WidgetDelegateView> |
| CreateModalWidgetDelegate(){ |
| autodelegate= std::make_unique<views::WidgetDelegateView>( |
| views::WidgetDelegateView::CreatePassKey()); |
| delegate->SetCanResize(true); |
| delegate->SetModalType(ui::mojom::ModalType::kSystem); |
| delegate->SetOwnedByWidget(views::WidgetDelegate::OwnedByWidgetPassKey()); |
| delegate->SetTitle(u"Modal Window"); |
| returndelegate; |
| } |
| |
| voidTestCreateWindow(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(); |
| } |
| |
| voidLockScreenAndVerifyMenuClosed(){ |
| // 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. |
| TEST_F(ShellTest,CreateWindowWithPreferredSize){ |
| UpdateDisplay("1024x768,800x600"); |
| |
| aura::Window* secondary_root=Shell::GetAllRootWindows()[1]; |
| display::ScopedDisplayForNewWindows scoped_display(secondary_root); |
| |
| views::Widget::InitParams params( |
| views::Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET); |
| // Don't specify bounds, parent or context. |
| { |
| autodelegate= std::make_unique<views::WidgetDelegateView>( |
| views::WidgetDelegateView::CreatePassKey()); |
| 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::mojom::MenuSourceType::kMouse); |
| 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= |
| std::ranges::find(handlers, shell->mouse_cursor_filter()); |
| ui::EventHandlerList::const_iterator drag_drop= |
| std::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= |
| std::ranges::find(handlers, shell->accelerator_tracker()); |
| ui::EventHandlerList::const_iterator accelerator_filter= |
| std::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= |
| std::ranges::find(handlers, shell->mouse_cursor_filter()); |
| ui::EventHandlerList::const_iterator fullscreen_magnifier_filter= |
| std::ranges::find(handlers, shell->fullscreen_magnifier_controller()); |
| ui::EventHandlerList::const_iterator chromevox_filter= |
| std::ranges::find(handlers, shell->key_accessibility_enabler()); |
| ui::EventHandlerList::const_iterator select_to_speak_filter= |
| std::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= std::ranges::find(handlers, shell->mouse_cursor_filter()); |
| fullscreen_magnifier_filter= |
| std::ranges::find(handlers, shell->fullscreen_magnifier_controller()); |
| chromevox_filter= |
| std::ranges::find(handlers, shell->key_accessibility_enabler()); |
| select_to_speak_filter= std::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= std::ranges::find(handlers, shell->mouse_cursor_filter()); |
| fullscreen_magnifier_filter= |
| std::ranges::find(handlers, shell->fullscreen_magnifier_controller()); |
| chromevox_filter= |
| std::ranges::find(handlers, shell->key_accessibility_enabler()); |
| ui::EventHandlerList::const_iterator docked_magnifier_filter= |
| std::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(views::Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET); |
| |
| // 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()); |
| } |
| |
| // 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. |
| classShellTest2:publicAshTestBase{ |
| public: |
| ShellTest2()=default; |
| |
| ShellTest2(constShellTest2&)=delete; |
| ShellTest2&operator=(constShellTest2&)=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); |
| } |
| |
| usingShellLoginTest=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()); |
| } |
| |
| usingNoDuplicateShellContainerIdsTest=AshTestBase; |
| |
| TEST_F(NoDuplicateShellContainerIdsTest,ValidateContainersIds){ |
| ExpectAllContainers(); |
| } |
| |
| // A test fixture that host `ActivationChanger` to simulate window activation |
| // change during Shell shutdown and verifies no crash will happen. |
| classShellShutdownTest:publicAshTestBase{ |
| protected: |
| // Helper to activate `to_activate` when `to_observe` is destroyed. The |
| // intention is to simulate active window change during Shell destruction. |
| // The activation change needs to affect `CloseAllRootWindowChildWindows()` |
| // in Shell destructor. |
| classActivationChanger:public aura::WindowObserver{ |
| public: |
| ActivationChanger(aura::Window* to_observe, aura::Window* to_activate) |
| : to_observe_(to_observe), to_activate_(to_activate){ |
| // No `RemoveObsever` because `this` outlives `Shell` and all windows. |
| to_observe_->AddObserver(this); |
| } |
| |
| // aura::WindowObserver: |
| voidOnWindowDestroying(aura::Window* window) override{ |
| Shell::Get()->focus_controller()->ActivateWindow(to_activate_); |
| |
| // Clear out references to avoid `raw_ptr` dangling pointer warnings. |
| to_observe_=nullptr; |
| to_activate_=nullptr; |
| } |
| |
| private: |
| raw_ptr<aura::Window> to_observe_; |
| raw_ptr<aura::Window> to_activate_; |
| }; |
| |
| voidCreateActivationChanger(aura::Window* to_observe, |
| aura::Window* to_activate){ |
| activation_changer_= |
| std::make_unique<ActivationChanger>(to_observe, to_activate); |
| } |
| |
| private: |
| std::unique_ptr<ActivationChanger> activation_changer_; |
| }; |
| |
| TEST_F(ShellShutdownTest,ActivateWindow){ |
| aura::Window* to_observe= |
| CreateTestWindowInShellWithBounds(gfx::Rect(40,0,60,40)); |
| to_observe->Show(); |
| |
| aura::Window* to_activate= |
| CreateTestWindowInShellWithBounds(gfx::Rect(0,0,30,20)); |
| // Put `to_activate` in a container after desks containers so that its |
| // destruction (and activations of `to_activate`) comes after desk containers |
| // destruction. |
| Shell::GetPrimaryRootWindow() |
| ->GetChildById(kShellWindowId_FloatContainer) |
| ->AddChild(to_activate); |
| to_activate->Show(); |
| |
| wm::ActivateWindow(to_observe); |
| |
| // Creates an ActivationChanger to activate `to_activate` during shutdown. |
| CreateActivationChanger(to_observe, to_activate); |
| } |
| |
| }// namespace ash |