1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "ash/accelerators/accelerator_controller.h"
7 #include "ash/accelerators/accelerator_table.h"
8 #include "ash/accessibility_delegate.h"
9 #include "ash/ash_switches.h"
10 #include "ash/display/display_manager.h"
11 #include "ash/ime_control_delegate.h"
12 #include "ash/screen_util.h"
13 #include "ash/shell.h"
14 #include "ash/shell_window_ids.h"
15 #include "ash/system/brightness_control_delegate.h"
16 #include "ash/system/keyboard_brightness/keyboard_brightness_control_delegate.h"
17 #include "ash/system/tray/system_tray_delegate.h"
18 #include "ash/test/ash_test_base.h"
19 #include "ash/test/display_manager_test_api.h"
20 #include "ash/test/test_screenshot_delegate.h"
21 #include "ash/test/test_session_state_animator.h"
22 #include "ash/test/test_shelf_delegate.h"
23 #include "ash/test/test_shell_delegate.h"
24 #include "ash/test/test_volume_control_delegate.h"
25 #include "ash/volume_control_delegate.h"
26 #include "ash/wm/lock_state_controller.h"
27 #include "ash/wm/panels/panel_layout_manager.h"
28 #include "ash/wm/window_state.h"
29 #include "ash/wm/window_util.h"
30 #include "ash/wm/wm_event.h"
31 #include "base/command_line.h"
32 #include "ui/aura/client/aura_constants.h"
33 #include "ui/aura/test/test_window_delegate.h"
34 #include "ui/aura/test/test_windows.h"
35 #include "ui/aura/window.h"
36 #include "ui/events/event.h"
37 #include "ui/events/event_processor.h"
38 #include "ui/events/test/event_generator.h"
39 #include "ui/gfx/screen.h"
40 #include "ui/message_center/message_center.h"
41 #include "ui/views/widget/widget.h"
45 #include "ui/events/test/events_test_utils_x11.h"
52 class TestTarget
: public ui::AcceleratorTarget
{
54 TestTarget() : accelerator_pressed_count_(0), accelerator_repeat_count_(0) {}
55 ~TestTarget() override
{}
57 int accelerator_pressed_count() const {
58 return accelerator_pressed_count_
;
61 int accelerator_repeat_count() const { return accelerator_repeat_count_
; }
64 accelerator_pressed_count_
= 0;
65 accelerator_repeat_count_
= 0;
68 // Overridden from ui::AcceleratorTarget:
69 bool AcceleratorPressed(const ui::Accelerator
& accelerator
) override
;
70 bool CanHandleAccelerators() const override
;
73 int accelerator_pressed_count_
;
74 int accelerator_repeat_count_
;
76 DISALLOW_COPY_AND_ASSIGN(TestTarget
);
79 class ReleaseAccelerator
: public ui::Accelerator
{
81 ReleaseAccelerator(ui::KeyboardCode keycode
, int modifiers
)
82 : ui::Accelerator(keycode
, modifiers
) {
83 set_type(ui::ET_KEY_RELEASED
);
87 class DummyBrightnessControlDelegate
: public BrightnessControlDelegate
{
89 DummyBrightnessControlDelegate()
90 : handle_brightness_down_count_(0), handle_brightness_up_count_(0) {}
91 ~DummyBrightnessControlDelegate() override
{}
93 void HandleBrightnessDown(const ui::Accelerator
& accelerator
) override
{
94 ++handle_brightness_down_count_
;
95 last_accelerator_
= accelerator
;
97 void HandleBrightnessUp(const ui::Accelerator
& accelerator
) override
{
98 ++handle_brightness_up_count_
;
99 last_accelerator_
= accelerator
;
101 void SetBrightnessPercent(double percent
, bool gradual
) override
{}
102 void GetBrightnessPercent(
103 const base::Callback
<void(double)>& callback
) override
{
107 int handle_brightness_down_count() const {
108 return handle_brightness_down_count_
;
110 int handle_brightness_up_count() const {
111 return handle_brightness_up_count_
;
113 const ui::Accelerator
& last_accelerator() const {
114 return last_accelerator_
;
118 int handle_brightness_down_count_
;
119 int handle_brightness_up_count_
;
120 ui::Accelerator last_accelerator_
;
122 DISALLOW_COPY_AND_ASSIGN(DummyBrightnessControlDelegate
);
125 class DummyImeControlDelegate
: public ImeControlDelegate
{
127 DummyImeControlDelegate()
128 : handle_next_ime_count_(0),
129 handle_previous_ime_count_(0),
130 handle_switch_ime_count_(0) {}
131 ~DummyImeControlDelegate() override
{}
133 bool CanCycleIme() override
{ return true; }
134 void HandleNextIme() override
{ ++handle_next_ime_count_
; }
135 void HandlePreviousIme() override
{
136 ++handle_previous_ime_count_
;
138 bool CanSwitchIme(const ui::Accelerator
& accelerator
) override
{
141 void HandleSwitchIme(const ui::Accelerator
& accelerator
) override
{
142 ++handle_switch_ime_count_
;
145 int handle_next_ime_count() const {
146 return handle_next_ime_count_
;
148 int handle_previous_ime_count() const {
149 return handle_previous_ime_count_
;
151 int handle_switch_ime_count() const {
152 return handle_switch_ime_count_
;
154 ui::Accelerator
RemapAccelerator(
155 const ui::Accelerator
& accelerator
) override
{
156 return ui::Accelerator(accelerator
);
160 int handle_next_ime_count_
;
161 int handle_previous_ime_count_
;
162 int handle_switch_ime_count_
;
164 DISALLOW_COPY_AND_ASSIGN(DummyImeControlDelegate
);
167 class DummyKeyboardBrightnessControlDelegate
168 : public KeyboardBrightnessControlDelegate
{
170 DummyKeyboardBrightnessControlDelegate()
171 : handle_keyboard_brightness_down_count_(0),
172 handle_keyboard_brightness_up_count_(0) {}
173 ~DummyKeyboardBrightnessControlDelegate() override
{}
175 void HandleKeyboardBrightnessDown(
176 const ui::Accelerator
& accelerator
) override
{
177 ++handle_keyboard_brightness_down_count_
;
178 last_accelerator_
= accelerator
;
181 void HandleKeyboardBrightnessUp(const ui::Accelerator
& accelerator
) override
{
182 ++handle_keyboard_brightness_up_count_
;
183 last_accelerator_
= accelerator
;
186 int handle_keyboard_brightness_down_count() const {
187 return handle_keyboard_brightness_down_count_
;
190 int handle_keyboard_brightness_up_count() const {
191 return handle_keyboard_brightness_up_count_
;
194 const ui::Accelerator
& last_accelerator() const {
195 return last_accelerator_
;
199 int handle_keyboard_brightness_down_count_
;
200 int handle_keyboard_brightness_up_count_
;
201 ui::Accelerator last_accelerator_
;
203 DISALLOW_COPY_AND_ASSIGN(DummyKeyboardBrightnessControlDelegate
);
206 bool TestTarget::AcceleratorPressed(const ui::Accelerator
& accelerator
) {
207 if (accelerator
.IsRepeat())
208 ++accelerator_repeat_count_
;
210 ++accelerator_pressed_count_
;
214 bool TestTarget::CanHandleAccelerators() const {
220 class AcceleratorControllerTest
: public test::AshTestBase
{
222 AcceleratorControllerTest() {}
223 ~AcceleratorControllerTest() override
{}
226 void EnableInternalDisplay() {
227 test::DisplayManagerTestApi().SetFirstDisplayAsInternalDisplay();
230 static AcceleratorController
* GetController();
232 static bool ProcessInController(const ui::Accelerator
& accelerator
) {
233 GetController()->accelerator_history()->
234 StoreCurrentAccelerator(accelerator
);
235 return GetController()->Process(accelerator
);
238 static const ui::Accelerator
& GetPreviousAccelerator() {
239 return GetController()->accelerator_history()->
240 previous_accelerator();
243 static const ui::Accelerator
& GetCurrentAccelerator() {
244 return GetController()->accelerator_history()->
245 current_accelerator();
248 // Several functions to access ExitWarningHandler (as friend).
249 static void StubForTest(ExitWarningHandler
* ewh
) {
250 ewh
->stub_timer_for_test_
= true;
252 static void Reset(ExitWarningHandler
* ewh
) {
253 ewh
->state_
= ExitWarningHandler::IDLE
;
255 static void SimulateTimerExpired(ExitWarningHandler
* ewh
) {
258 static bool is_ui_shown(ExitWarningHandler
* ewh
) {
259 return !!ewh
->widget_
;
261 static bool is_idle(ExitWarningHandler
* ewh
) {
262 return ewh
->state_
== ExitWarningHandler::IDLE
;
264 static bool is_exiting(ExitWarningHandler
* ewh
) {
265 return ewh
->state_
== ExitWarningHandler::EXITING
;
267 aura::Window
* CreatePanel() {
268 aura::Window
* window
=
269 CreateTestWindowInShellWithDelegateAndType(NULL
,
270 ui::wm::WINDOW_TYPE_PANEL
, 0, gfx::Rect(5, 5, 20, 20));
271 test::TestShelfDelegate
* shelf_delegate
=
272 test::TestShelfDelegate::instance();
273 shelf_delegate
->AddShelfItem(window
);
274 PanelLayoutManager
* manager
= static_cast<PanelLayoutManager
*>(
275 Shell::GetContainer(window
->GetRootWindow(),
276 kShellWindowId_PanelContainer
)->layout_manager());
282 DISALLOW_COPY_AND_ASSIGN(AcceleratorControllerTest
);
285 AcceleratorController
* AcceleratorControllerTest::GetController() {
286 return Shell::GetInstance()->accelerator_controller();
290 // Double press of exit shortcut => exiting
291 TEST_F(AcceleratorControllerTest
, ExitWarningHandlerTestDoublePress
) {
292 ui::Accelerator
press(ui::VKEY_Q
, ui::EF_SHIFT_DOWN
| ui::EF_CONTROL_DOWN
);
293 ui::Accelerator
release(press
);
294 release
.set_type(ui::ET_KEY_RELEASED
);
295 ExitWarningHandler
* ewh
= GetController()->GetExitWarningHandlerForTest();
298 EXPECT_TRUE(is_idle(ewh
));
299 EXPECT_FALSE(is_ui_shown(ewh
));
300 EXPECT_TRUE(ProcessInController(press
));
301 EXPECT_FALSE(ProcessInController(release
));
302 EXPECT_FALSE(is_idle(ewh
));
303 EXPECT_TRUE(is_ui_shown(ewh
));
304 EXPECT_TRUE(ProcessInController(press
)); // second press before timer.
305 EXPECT_FALSE(ProcessInController(release
));
306 SimulateTimerExpired(ewh
);
307 EXPECT_TRUE(is_exiting(ewh
));
308 EXPECT_FALSE(is_ui_shown(ewh
));
312 // Single press of exit shortcut before timer => idle
313 TEST_F(AcceleratorControllerTest
, ExitWarningHandlerTestSinglePress
) {
314 ui::Accelerator
press(ui::VKEY_Q
, ui::EF_SHIFT_DOWN
| ui::EF_CONTROL_DOWN
);
315 ui::Accelerator
release(press
);
316 release
.set_type(ui::ET_KEY_RELEASED
);
317 ExitWarningHandler
* ewh
= GetController()->GetExitWarningHandlerForTest();
320 EXPECT_TRUE(is_idle(ewh
));
321 EXPECT_FALSE(is_ui_shown(ewh
));
322 EXPECT_TRUE(ProcessInController(press
));
323 EXPECT_FALSE(ProcessInController(release
));
324 EXPECT_FALSE(is_idle(ewh
));
325 EXPECT_TRUE(is_ui_shown(ewh
));
326 SimulateTimerExpired(ewh
);
327 EXPECT_TRUE(is_idle(ewh
));
328 EXPECT_FALSE(is_ui_shown(ewh
));
332 // Shutdown ash with exit warning bubble open should not crash.
333 TEST_F(AcceleratorControllerTest
, LingeringExitWarningBubble
) {
334 ExitWarningHandler
* ewh
= GetController()->GetExitWarningHandlerForTest();
338 // Trigger once to show the bubble.
339 ewh
->HandleAccelerator();
340 EXPECT_FALSE(is_idle(ewh
));
341 EXPECT_TRUE(is_ui_shown(ewh
));
343 // Exit ash and there should be no crash
345 #endif // !defined(OS_WIN)
347 TEST_F(AcceleratorControllerTest
, Register
) {
348 const ui::Accelerator
accelerator_a(ui::VKEY_A
, ui::EF_NONE
);
350 GetController()->Register(accelerator_a
, &target
);
352 // The registered accelerator is processed.
353 EXPECT_TRUE(ProcessInController(accelerator_a
));
354 EXPECT_EQ(1, target
.accelerator_pressed_count());
357 TEST_F(AcceleratorControllerTest
, RegisterMultipleTarget
) {
358 const ui::Accelerator
accelerator_a(ui::VKEY_A
, ui::EF_NONE
);
360 GetController()->Register(accelerator_a
, &target1
);
362 GetController()->Register(accelerator_a
, &target2
);
364 // If multiple targets are registered with the same accelerator, the target
365 // registered later processes the accelerator.
366 EXPECT_TRUE(ProcessInController(accelerator_a
));
367 EXPECT_EQ(0, target1
.accelerator_pressed_count());
368 EXPECT_EQ(1, target2
.accelerator_pressed_count());
371 TEST_F(AcceleratorControllerTest
, Unregister
) {
372 const ui::Accelerator
accelerator_a(ui::VKEY_A
, ui::EF_NONE
);
374 GetController()->Register(accelerator_a
, &target
);
375 const ui::Accelerator
accelerator_b(ui::VKEY_B
, ui::EF_NONE
);
376 GetController()->Register(accelerator_b
, &target
);
378 // Unregistering a different accelerator does not affect the other
380 GetController()->Unregister(accelerator_b
, &target
);
381 EXPECT_TRUE(ProcessInController(accelerator_a
));
382 EXPECT_EQ(1, target
.accelerator_pressed_count());
384 // The unregistered accelerator is no longer processed.
386 GetController()->Unregister(accelerator_a
, &target
);
387 EXPECT_FALSE(ProcessInController(accelerator_a
));
388 EXPECT_EQ(0, target
.accelerator_pressed_count());
391 TEST_F(AcceleratorControllerTest
, UnregisterAll
) {
392 const ui::Accelerator
accelerator_a(ui::VKEY_A
, ui::EF_NONE
);
394 GetController()->Register(accelerator_a
, &target1
);
395 const ui::Accelerator
accelerator_b(ui::VKEY_B
, ui::EF_NONE
);
396 GetController()->Register(accelerator_b
, &target1
);
397 const ui::Accelerator
accelerator_c(ui::VKEY_C
, ui::EF_NONE
);
399 GetController()->Register(accelerator_c
, &target2
);
400 GetController()->UnregisterAll(&target1
);
402 // All the accelerators registered for |target1| are no longer processed.
403 EXPECT_FALSE(ProcessInController(accelerator_a
));
404 EXPECT_FALSE(ProcessInController(accelerator_b
));
405 EXPECT_EQ(0, target1
.accelerator_pressed_count());
407 // UnregisterAll with a different target does not affect the other target.
408 EXPECT_TRUE(ProcessInController(accelerator_c
));
409 EXPECT_EQ(1, target2
.accelerator_pressed_count());
412 TEST_F(AcceleratorControllerTest
, Process
) {
413 const ui::Accelerator
accelerator_a(ui::VKEY_A
, ui::EF_NONE
);
415 GetController()->Register(accelerator_a
, &target1
);
417 // The registered accelerator is processed.
418 EXPECT_TRUE(ProcessInController(accelerator_a
));
419 EXPECT_EQ(1, target1
.accelerator_pressed_count());
421 // The non-registered accelerator is not processed.
422 const ui::Accelerator
accelerator_b(ui::VKEY_B
, ui::EF_NONE
);
423 EXPECT_FALSE(ProcessInController(accelerator_b
));
426 TEST_F(AcceleratorControllerTest
, IsRegistered
) {
427 const ui::Accelerator
accelerator_a(ui::VKEY_A
, ui::EF_NONE
);
428 const ui::Accelerator
accelerator_shift_a(ui::VKEY_A
, ui::EF_SHIFT_DOWN
);
430 GetController()->Register(accelerator_a
, &target
);
431 EXPECT_TRUE(GetController()->IsRegistered(accelerator_a
));
432 EXPECT_FALSE(GetController()->IsRegistered(accelerator_shift_a
));
433 GetController()->UnregisterAll(&target
);
434 EXPECT_FALSE(GetController()->IsRegistered(accelerator_a
));
437 TEST_F(AcceleratorControllerTest
, WindowSnap
) {
438 scoped_ptr
<aura::Window
> window(
439 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
440 wm::WindowState
* window_state
= wm::GetWindowState(window
.get());
442 window_state
->Activate();
445 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
446 gfx::Rect expected_bounds
= wm::GetDefaultLeftSnappedWindowBoundsInParent(
448 EXPECT_EQ(expected_bounds
.ToString(), window
->bounds().ToString());
451 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_RIGHT
);
452 gfx::Rect expected_bounds
= wm::GetDefaultRightSnappedWindowBoundsInParent(
454 EXPECT_EQ(expected_bounds
.ToString(), window
->bounds().ToString());
457 gfx::Rect normal_bounds
= window_state
->GetRestoreBoundsInParent();
459 GetController()->PerformActionIfEnabled(TOGGLE_MAXIMIZED
);
460 EXPECT_TRUE(window_state
->IsMaximized());
461 EXPECT_NE(normal_bounds
.ToString(), window
->bounds().ToString());
463 GetController()->PerformActionIfEnabled(TOGGLE_MAXIMIZED
);
464 EXPECT_FALSE(window_state
->IsMaximized());
465 // Window gets restored to its restore bounds since side-maximized state
466 // is treated as a "maximized" state.
467 EXPECT_EQ(normal_bounds
.ToString(), window
->bounds().ToString());
469 GetController()->PerformActionIfEnabled(TOGGLE_MAXIMIZED
);
470 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
471 EXPECT_FALSE(window_state
->IsMaximized());
473 GetController()->PerformActionIfEnabled(TOGGLE_MAXIMIZED
);
474 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_RIGHT
);
475 EXPECT_FALSE(window_state
->IsMaximized());
477 GetController()->PerformActionIfEnabled(TOGGLE_MAXIMIZED
);
478 EXPECT_TRUE(window_state
->IsMaximized());
479 GetController()->PerformActionIfEnabled(WINDOW_MINIMIZE
);
480 EXPECT_FALSE(window_state
->IsMaximized());
481 EXPECT_TRUE(window_state
->IsMinimized());
482 window_state
->Restore();
483 window_state
->Activate();
486 GetController()->PerformActionIfEnabled(WINDOW_MINIMIZE
);
487 EXPECT_TRUE(window_state
->IsMinimized());
491 TEST_F(AcceleratorControllerTest
, WindowSnapLeftDockLeftRestore
) {
492 scoped_ptr
<aura::Window
> window0(
493 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
494 scoped_ptr
<aura::Window
> window1(
495 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
496 wm::WindowState
* window1_state
= wm::GetWindowState(window1
.get());
497 window1_state
->Activate();
499 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
500 gfx::Rect normal_bounds
= window1_state
->GetRestoreBoundsInParent();
501 gfx::Rect expected_bounds
= wm::GetDefaultLeftSnappedWindowBoundsInParent(
503 EXPECT_EQ(expected_bounds
.ToString(), window1
->bounds().ToString());
504 EXPECT_TRUE(window1_state
->IsSnapped());
505 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
506 EXPECT_FALSE(window1_state
->IsNormalOrSnapped());
507 EXPECT_TRUE(window1_state
->IsDocked());
508 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
509 EXPECT_FALSE(window1_state
->IsDocked());
510 EXPECT_EQ(normal_bounds
.ToString(), window1
->bounds().ToString());
513 TEST_F(AcceleratorControllerTest
, WindowSnapRightDockRightRestore
) {
514 scoped_ptr
<aura::Window
> window0(
515 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
516 scoped_ptr
<aura::Window
> window1(
517 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
519 wm::WindowState
* window1_state
= wm::GetWindowState(window1
.get());
520 window1_state
->Activate();
522 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_RIGHT
);
523 gfx::Rect normal_bounds
= window1_state
->GetRestoreBoundsInParent();
524 gfx::Rect expected_bounds
=
525 wm::GetDefaultRightSnappedWindowBoundsInParent(window1
.get());
526 EXPECT_EQ(expected_bounds
.ToString(), window1
->bounds().ToString());
527 EXPECT_TRUE(window1_state
->IsSnapped());
528 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_RIGHT
);
529 EXPECT_FALSE(window1_state
->IsNormalOrSnapped());
530 EXPECT_TRUE(window1_state
->IsDocked());
531 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_RIGHT
);
532 EXPECT_FALSE(window1_state
->IsDocked());
533 EXPECT_EQ(normal_bounds
.ToString(), window1
->bounds().ToString());
536 TEST_F(AcceleratorControllerTest
, WindowSnapLeftDockLeftSnapRight
) {
537 scoped_ptr
<aura::Window
> window0(
538 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
539 scoped_ptr
<aura::Window
> window1(
540 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
542 wm::WindowState
* window1_state
= wm::GetWindowState(window1
.get());
543 window1_state
->Activate();
545 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
546 gfx::Rect expected_bounds
=
547 wm::GetDefaultLeftSnappedWindowBoundsInParent(window1
.get());
548 gfx::Rect expected_bounds2
=
549 wm::GetDefaultRightSnappedWindowBoundsInParent(window1
.get());
550 EXPECT_EQ(expected_bounds
.ToString(), window1
->bounds().ToString());
551 EXPECT_TRUE(window1_state
->IsSnapped());
552 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
553 EXPECT_FALSE(window1_state
->IsNormalOrSnapped());
554 EXPECT_TRUE(window1_state
->IsDocked());
555 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_RIGHT
);
556 EXPECT_FALSE(window1_state
->IsDocked());
557 EXPECT_TRUE(window1_state
->IsSnapped());
558 EXPECT_EQ(expected_bounds2
.ToString(), window1
->bounds().ToString());
561 TEST_F(AcceleratorControllerTest
, WindowDockLeftMinimizeWindowWithRestore
) {
562 scoped_ptr
<aura::Window
> window0(
563 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
564 scoped_ptr
<aura::Window
> window1(
565 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
567 wm::WindowState
* window1_state
= wm::GetWindowState(window1
.get());
568 window1_state
->Activate();
570 scoped_ptr
<aura::Window
> window2(
571 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
573 wm::WindowState
* window2_state
= wm::GetWindowState(window2
.get());
575 scoped_ptr
<aura::Window
> window3(
576 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
578 wm::WindowState
* window3_state
= wm::GetWindowState(window3
.get());
579 window3_state
->Activate();
581 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
582 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
583 gfx::Rect window3_docked_bounds
= window3
->bounds();
585 window2_state
->Activate();
586 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
587 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
588 window1_state
->Activate();
589 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
590 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
592 EXPECT_TRUE(window3_state
->IsDocked());
593 EXPECT_TRUE(window2_state
->IsDocked());
594 EXPECT_TRUE(window1_state
->IsDocked());
595 EXPECT_TRUE(window3_state
->IsMinimized());
597 window1_state
->Activate();
598 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
599 window2_state
->Activate();
600 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
601 window3_state
->Unminimize();
602 EXPECT_FALSE(window1_state
->IsDocked());
603 EXPECT_FALSE(window2_state
->IsDocked());
604 EXPECT_TRUE(window3_state
->IsDocked());
605 EXPECT_EQ(window3_docked_bounds
.ToString(), window3
->bounds().ToString());
608 TEST_F(AcceleratorControllerTest
, WindowPanelDockLeftDockRightRestore
) {
609 scoped_ptr
<aura::Window
> window0(
610 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
612 scoped_ptr
<aura::Window
> window(CreatePanel());
613 wm::WindowState
* window_state
= wm::GetWindowState(window
.get());
614 window_state
->Activate();
616 gfx::Rect window_restore_bounds2
= window
->bounds();
617 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_LEFT
);
618 gfx::Rect expected_bounds
=
619 wm::GetDefaultLeftSnappedWindowBoundsInParent(window
.get());
620 gfx::Rect window_restore_bounds
=
621 window_state
->GetRestoreBoundsInScreen();
622 EXPECT_NE(expected_bounds
.ToString(), window
->bounds().ToString());
623 EXPECT_FALSE(window_state
->IsSnapped());
624 EXPECT_FALSE(window_state
->IsNormalOrSnapped());
625 EXPECT_TRUE(window_state
->IsDocked());
626 window_state
->Restore();
627 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_RIGHT
);
628 EXPECT_TRUE(window_state
->IsDocked());
629 GetController()->PerformActionIfEnabled(WINDOW_CYCLE_SNAP_DOCK_RIGHT
);
630 EXPECT_FALSE(window_state
->IsDocked());
631 EXPECT_EQ(window_restore_bounds
.ToString(),
632 window_restore_bounds2
.ToString());
633 EXPECT_EQ(window_restore_bounds
.ToString(), window
->bounds().ToString());
636 TEST_F(AcceleratorControllerTest
, CenterWindowAccelerator
) {
637 scoped_ptr
<aura::Window
> window(
638 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
639 wm::WindowState
* window_state
= wm::GetWindowState(window
.get());
640 window_state
->Activate();
642 // Center the window using accelerator.
643 GetController()->PerformActionIfEnabled(WINDOW_POSITION_CENTER
);
644 gfx::Rect work_area
=
645 Shell::GetScreen()->GetDisplayNearestWindow(window
.get()).work_area();
646 gfx::Rect bounds
= window
->GetBoundsInScreen();
647 EXPECT_NEAR(bounds
.x() - work_area
.x(),
648 work_area
.right() - bounds
.right(),
650 EXPECT_NEAR(bounds
.y() - work_area
.y(),
651 work_area
.bottom() - bounds
.bottom(),
654 // Add the window to docked container and try to center it.
655 window
->SetBounds(gfx::Rect(0, 0, 20, 20));
656 const wm::WMEvent
event(wm::WM_EVENT_DOCK
);
657 wm::GetWindowState(window
.get())->OnWMEvent(&event
);
658 EXPECT_EQ(kShellWindowId_DockedContainer
, window
->parent()->id());
660 gfx::Rect docked_bounds
= window
->GetBoundsInScreen();
661 GetController()->PerformActionIfEnabled(WINDOW_POSITION_CENTER
);
662 // It should not get centered and should remain docked.
663 EXPECT_EQ(kShellWindowId_DockedContainer
, window
->parent()->id());
664 EXPECT_EQ(docked_bounds
.ToString(), window
->GetBoundsInScreen().ToString());
667 TEST_F(AcceleratorControllerTest
, AutoRepeat
) {
668 ui::Accelerator
accelerator_a(ui::VKEY_A
, ui::EF_CONTROL_DOWN
);
669 accelerator_a
.set_type(ui::ET_KEY_PRESSED
);
671 GetController()->Register(accelerator_a
, &target_a
);
672 ui::Accelerator
accelerator_b(ui::VKEY_B
, ui::EF_CONTROL_DOWN
);
673 accelerator_b
.set_type(ui::ET_KEY_PRESSED
);
675 GetController()->Register(accelerator_b
, &target_b
);
677 ui::test::EventGenerator
& generator
= GetEventGenerator();
678 generator
.PressKey(ui::VKEY_A
, ui::EF_CONTROL_DOWN
);
679 generator
.ReleaseKey(ui::VKEY_A
, ui::EF_CONTROL_DOWN
);
681 EXPECT_EQ(1, target_a
.accelerator_pressed_count());
682 EXPECT_EQ(0, target_a
.accelerator_repeat_count());
684 // Long press should generate one
685 generator
.PressKey(ui::VKEY_A
, ui::EF_CONTROL_DOWN
);
686 generator
.PressKey(ui::VKEY_A
, ui::EF_CONTROL_DOWN
| ui::EF_IS_REPEAT
);
687 EXPECT_EQ(2, target_a
.accelerator_pressed_count());
688 EXPECT_EQ(1, target_a
.accelerator_repeat_count());
689 generator
.PressKey(ui::VKEY_A
, ui::EF_CONTROL_DOWN
| ui::EF_IS_REPEAT
);
690 EXPECT_EQ(2, target_a
.accelerator_pressed_count());
691 EXPECT_EQ(2, target_a
.accelerator_repeat_count());
692 generator
.ReleaseKey(ui::VKEY_A
, ui::EF_CONTROL_DOWN
);
693 EXPECT_EQ(2, target_a
.accelerator_pressed_count());
694 EXPECT_EQ(2, target_a
.accelerator_repeat_count());
696 // Long press was intercepted by another key press.
697 generator
.PressKey(ui::VKEY_A
, ui::EF_CONTROL_DOWN
);
698 generator
.PressKey(ui::VKEY_A
, ui::EF_CONTROL_DOWN
| ui::EF_IS_REPEAT
);
699 generator
.PressKey(ui::VKEY_B
, ui::EF_CONTROL_DOWN
);
700 generator
.ReleaseKey(ui::VKEY_B
, ui::EF_CONTROL_DOWN
);
701 generator
.PressKey(ui::VKEY_A
, ui::EF_CONTROL_DOWN
);
702 generator
.PressKey(ui::VKEY_A
, ui::EF_CONTROL_DOWN
| ui::EF_IS_REPEAT
);
703 generator
.ReleaseKey(ui::VKEY_A
, ui::EF_CONTROL_DOWN
);
705 EXPECT_EQ(1, target_b
.accelerator_pressed_count());
706 EXPECT_EQ(0, target_b
.accelerator_repeat_count());
707 EXPECT_EQ(4, target_a
.accelerator_pressed_count());
708 EXPECT_EQ(4, target_a
.accelerator_repeat_count());
711 TEST_F(AcceleratorControllerTest
, Previous
) {
712 ui::test::EventGenerator
& generator
= GetEventGenerator();
713 generator
.PressKey(ui::VKEY_VOLUME_MUTE
, ui::EF_NONE
);
714 generator
.ReleaseKey(ui::VKEY_VOLUME_MUTE
, ui::EF_NONE
);
716 EXPECT_EQ(ui::VKEY_VOLUME_MUTE
,
717 GetPreviousAccelerator().key_code());
718 EXPECT_EQ(ui::EF_NONE
,
719 GetPreviousAccelerator().modifiers());
721 generator
.PressKey(ui::VKEY_TAB
, ui::EF_CONTROL_DOWN
);
722 generator
.ReleaseKey(ui::VKEY_TAB
, ui::EF_CONTROL_DOWN
);
724 EXPECT_EQ(ui::VKEY_TAB
,
725 GetPreviousAccelerator().key_code());
726 EXPECT_EQ(ui::EF_CONTROL_DOWN
,
727 GetPreviousAccelerator().modifiers());
730 TEST_F(AcceleratorControllerTest
, DontRepeatToggleFullscreen
) {
731 const AcceleratorData accelerators
[] = {
732 {true, ui::VKEY_J
, ui::EF_ALT_DOWN
, TOGGLE_FULLSCREEN
},
733 {true, ui::VKEY_K
, ui::EF_ALT_DOWN
, TOGGLE_FULLSCREEN
},
735 GetController()->RegisterAccelerators(accelerators
, arraysize(accelerators
));
737 views::Widget::InitParams
params(views::Widget::InitParams::TYPE_WINDOW
);
738 params
.context
= CurrentContext();
739 params
.bounds
= gfx::Rect(5, 5, 20, 20);
740 views::Widget
* widget
= new views::Widget
;
741 widget
->Init(params
);
744 widget
->GetNativeView()->SetProperty(aura::client::kCanMaximizeKey
, true);
746 ui::test::EventGenerator
& generator
= GetEventGenerator();
747 wm::WindowState
* window_state
= wm::GetWindowState(widget
->GetNativeView());
749 // Toggling not suppressed.
750 generator
.PressKey(ui::VKEY_J
, ui::EF_ALT_DOWN
);
751 EXPECT_TRUE(window_state
->IsFullscreen());
753 // The same accelerator - toggling suppressed.
754 generator
.PressKey(ui::VKEY_J
, ui::EF_ALT_DOWN
| ui::EF_IS_REPEAT
);
755 EXPECT_TRUE(window_state
->IsFullscreen());
757 // Different accelerator.
758 generator
.PressKey(ui::VKEY_K
, ui::EF_ALT_DOWN
);
759 EXPECT_FALSE(window_state
->IsFullscreen());
762 // TODO(oshima): Fix this test to use EventGenerator.
765 #define MAYBE_ProcessOnce DISABLED_ProcessOnce
767 #define MAYBE_ProcessOnce ProcessOnce
770 #if defined(OS_WIN) || defined(USE_X11)
771 TEST_F(AcceleratorControllerTest
, MAYBE_ProcessOnce
) {
772 // The IME event filter interferes with the basic key event propagation we
773 // attempt to do here, so we disable it.
775 ui::Accelerator
accelerator_a(ui::VKEY_A
, ui::EF_NONE
);
777 GetController()->Register(accelerator_a
, &target
);
779 // The accelerator is processed only once.
780 ui::EventProcessor
* dispatcher
=
781 Shell::GetPrimaryRootWindow()->GetHost()->event_processor();
783 MSG msg1
= { NULL
, WM_KEYDOWN
, ui::VKEY_A
, 0 };
784 ui::KeyEvent
key_event1(msg1
);
785 ui::EventDispatchDetails details
= dispatcher
->OnEventFromSource(&key_event1
);
786 EXPECT_TRUE(key_event1
.handled() || details
.dispatcher_destroyed
);
788 MSG msg2
= { NULL
, WM_CHAR
, L
'A', 0 };
789 ui::KeyEvent
key_event2(msg2
);
790 details
= dispatcher
->OnEventFromSource(&key_event2
);
791 EXPECT_FALSE(key_event2
.handled() || details
.dispatcher_destroyed
);
793 MSG msg3
= { NULL
, WM_KEYUP
, ui::VKEY_A
, 0 };
794 ui::KeyEvent
key_event3(msg3
);
795 details
= dispatcher
->OnEventFromSource(&key_event3
);
796 EXPECT_FALSE(key_event3
.handled() || details
.dispatcher_destroyed
);
797 #elif defined(USE_X11)
798 ui::ScopedXI2Event key_event
;
799 key_event
.InitKeyEvent(ui::ET_KEY_PRESSED
, ui::VKEY_A
, 0);
800 ui::KeyEvent
key_event1(key_event
);
801 ui::EventDispatchDetails details
= dispatcher
->OnEventFromSource(&key_event1
);
802 EXPECT_TRUE(key_event1
.handled() || details
.dispatcher_destroyed
);
804 ui::KeyEvent
key_event2('A', ui::VKEY_A
, ui::EF_NONE
);
805 details
= dispatcher
->OnEventFromSource(&key_event2
);
806 EXPECT_FALSE(key_event2
.handled() || details
.dispatcher_destroyed
);
808 key_event
.InitKeyEvent(ui::ET_KEY_RELEASED
, ui::VKEY_A
, 0);
809 ui::KeyEvent
key_event3(key_event
);
810 details
= dispatcher
->OnEventFromSource(&key_event3
);
811 EXPECT_FALSE(key_event3
.handled() || details
.dispatcher_destroyed
);
813 EXPECT_EQ(1, target
.accelerator_pressed_count());
817 TEST_F(AcceleratorControllerTest
, GlobalAccelerators
) {
819 EXPECT_TRUE(ProcessInController(
820 ui::Accelerator(ui::VKEY_TAB
, ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
)));
823 ProcessInController(ui::Accelerator(
824 ui::VKEY_TAB
, ui::EF_ALT_DOWN
)));
826 EXPECT_TRUE(ProcessInController(
827 ui::Accelerator(ui::VKEY_MEDIA_LAUNCH_APP1
, ui::EF_NONE
)));
829 #if defined(OS_CHROMEOS)
830 // The "Take Screenshot", "Take Partial Screenshot", volume, brightness, and
831 // keyboard brightness accelerators are only defined on ChromeOS.
833 test::TestScreenshotDelegate
* delegate
= GetScreenshotDelegate();
834 delegate
->set_can_take_screenshot(false);
835 EXPECT_TRUE(ProcessInController(
836 ui::Accelerator(ui::VKEY_MEDIA_LAUNCH_APP1
, ui::EF_CONTROL_DOWN
)));
838 ProcessInController(ui::Accelerator(
839 ui::VKEY_PRINT
, ui::EF_NONE
)));
840 EXPECT_TRUE(ProcessInController(ui::Accelerator(
841 ui::VKEY_MEDIA_LAUNCH_APP1
, ui::EF_SHIFT_DOWN
| ui::EF_CONTROL_DOWN
)));
843 delegate
->set_can_take_screenshot(true);
844 EXPECT_EQ(0, delegate
->handle_take_screenshot_count());
845 EXPECT_TRUE(ProcessInController(
846 ui::Accelerator(ui::VKEY_MEDIA_LAUNCH_APP1
, ui::EF_CONTROL_DOWN
)));
847 EXPECT_EQ(1, delegate
->handle_take_screenshot_count());
849 ProcessInController(ui::Accelerator(
850 ui::VKEY_PRINT
, ui::EF_NONE
)));
851 EXPECT_EQ(2, delegate
->handle_take_screenshot_count());
852 EXPECT_TRUE(ProcessInController(ui::Accelerator(
853 ui::VKEY_MEDIA_LAUNCH_APP1
, ui::EF_SHIFT_DOWN
| ui::EF_CONTROL_DOWN
)));
854 EXPECT_EQ(2, delegate
->handle_take_screenshot_count());
856 const ui::Accelerator
volume_mute(ui::VKEY_VOLUME_MUTE
, ui::EF_NONE
);
857 const ui::Accelerator
volume_down(ui::VKEY_VOLUME_DOWN
, ui::EF_NONE
);
858 const ui::Accelerator
volume_up(ui::VKEY_VOLUME_UP
, ui::EF_NONE
);
860 TestVolumeControlDelegate
* delegate
= new TestVolumeControlDelegate
;
861 ash::Shell::GetInstance()->system_tray_delegate()->SetVolumeControlDelegate(
862 scoped_ptr
<VolumeControlDelegate
>(delegate
).Pass());
863 EXPECT_EQ(0, delegate
->handle_volume_mute_count());
864 EXPECT_TRUE(ProcessInController(volume_mute
));
865 EXPECT_EQ(1, delegate
->handle_volume_mute_count());
866 EXPECT_EQ(volume_mute
, delegate
->last_accelerator());
867 EXPECT_EQ(0, delegate
->handle_volume_down_count());
868 EXPECT_TRUE(ProcessInController(volume_down
));
869 EXPECT_EQ(1, delegate
->handle_volume_down_count());
870 EXPECT_EQ(volume_down
, delegate
->last_accelerator());
871 EXPECT_EQ(0, delegate
->handle_volume_up_count());
872 EXPECT_TRUE(ProcessInController(volume_up
));
873 EXPECT_EQ(1, delegate
->handle_volume_up_count());
874 EXPECT_EQ(volume_up
, delegate
->last_accelerator());
877 // ui::VKEY_BRIGHTNESS_DOWN/UP are not defined on Windows.
878 const ui::Accelerator
brightness_down(ui::VKEY_BRIGHTNESS_DOWN
, ui::EF_NONE
);
879 const ui::Accelerator
brightness_up(ui::VKEY_BRIGHTNESS_UP
, ui::EF_NONE
);
881 DummyBrightnessControlDelegate
* delegate
=
882 new DummyBrightnessControlDelegate
;
883 GetController()->SetBrightnessControlDelegate(
884 scoped_ptr
<BrightnessControlDelegate
>(delegate
).Pass());
885 EXPECT_EQ(0, delegate
->handle_brightness_down_count());
886 EXPECT_TRUE(ProcessInController(brightness_down
));
887 EXPECT_EQ(1, delegate
->handle_brightness_down_count());
888 EXPECT_EQ(brightness_down
, delegate
->last_accelerator());
889 EXPECT_EQ(0, delegate
->handle_brightness_up_count());
890 EXPECT_TRUE(ProcessInController(brightness_up
));
891 EXPECT_EQ(1, delegate
->handle_brightness_up_count());
892 EXPECT_EQ(brightness_up
, delegate
->last_accelerator());
895 // Keyboard brightness
896 const ui::Accelerator
alt_brightness_down(ui::VKEY_BRIGHTNESS_DOWN
,
898 const ui::Accelerator
alt_brightness_up(ui::VKEY_BRIGHTNESS_UP
,
901 EXPECT_TRUE(ProcessInController(alt_brightness_down
));
902 EXPECT_TRUE(ProcessInController(alt_brightness_up
));
903 DummyKeyboardBrightnessControlDelegate
* delegate
=
904 new DummyKeyboardBrightnessControlDelegate
;
905 GetController()->SetKeyboardBrightnessControlDelegate(
906 scoped_ptr
<KeyboardBrightnessControlDelegate
>(delegate
).Pass());
907 EXPECT_EQ(0, delegate
->handle_keyboard_brightness_down_count());
908 EXPECT_TRUE(ProcessInController(alt_brightness_down
));
909 EXPECT_EQ(1, delegate
->handle_keyboard_brightness_down_count());
910 EXPECT_EQ(alt_brightness_down
, delegate
->last_accelerator());
911 EXPECT_EQ(0, delegate
->handle_keyboard_brightness_up_count());
912 EXPECT_TRUE(ProcessInController(alt_brightness_up
));
913 EXPECT_EQ(1, delegate
->handle_keyboard_brightness_up_count());
914 EXPECT_EQ(alt_brightness_up
, delegate
->last_accelerator());
920 ExitWarningHandler
* ewh
= GetController()->GetExitWarningHandlerForTest();
923 EXPECT_TRUE(is_idle(ewh
));
924 EXPECT_FALSE(is_ui_shown(ewh
));
925 EXPECT_TRUE(ProcessInController(
926 ui::Accelerator(ui::VKEY_Q
, ui::EF_SHIFT_DOWN
| ui::EF_CONTROL_DOWN
)));
927 EXPECT_FALSE(is_idle(ewh
));
928 EXPECT_TRUE(is_ui_shown(ewh
));
929 SimulateTimerExpired(ewh
);
930 EXPECT_TRUE(is_idle(ewh
));
931 EXPECT_FALSE(is_ui_shown(ewh
));
936 EXPECT_TRUE(ProcessInController(
937 ui::Accelerator(ui::VKEY_T
, ui::EF_CONTROL_DOWN
)));
939 // New incognito window
940 EXPECT_TRUE(ProcessInController(
941 ui::Accelerator(ui::VKEY_N
, ui::EF_SHIFT_DOWN
| ui::EF_CONTROL_DOWN
)));
944 EXPECT_TRUE(ProcessInController(
945 ui::Accelerator(ui::VKEY_N
, ui::EF_CONTROL_DOWN
)));
948 EXPECT_TRUE(ProcessInController(
949 ui::Accelerator(ui::VKEY_T
, ui::EF_SHIFT_DOWN
| ui::EF_CONTROL_DOWN
)));
952 EXPECT_TRUE(ProcessInController(
953 ui::Accelerator(ui::VKEY_ESCAPE
, ui::EF_SHIFT_DOWN
)));
955 #if defined(OS_CHROMEOS)
957 EXPECT_TRUE(ProcessInController(
958 ui::Accelerator(ui::VKEY_M
, ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
)));
961 // NOTE: Accelerators that do not work on the lock screen need to be
962 // tested before the sequence below is invoked because it causes a side
963 // effect of locking the screen.
964 EXPECT_TRUE(ProcessInController(
965 ui::Accelerator(ui::VKEY_L
, ui::EF_SHIFT_DOWN
| ui::EF_CONTROL_DOWN
)));
969 TEST_F(AcceleratorControllerTest
, GlobalAcceleratorsToggleAppList
) {
970 AccessibilityDelegate
* delegate
=
971 ash::Shell::GetInstance()->accessibility_delegate();
972 EXPECT_FALSE(ash::Shell::GetInstance()->GetAppListTargetVisibility());
974 // The press event should not open the AppList, the release should instead.
976 ProcessInController(ui::Accelerator(ui::VKEY_LWIN
, ui::EF_NONE
)));
977 EXPECT_EQ(ui::VKEY_LWIN
,
978 GetCurrentAccelerator().key_code());
980 EXPECT_FALSE(ash::Shell::GetInstance()->GetAppListTargetVisibility());
983 ProcessInController(ReleaseAccelerator(ui::VKEY_LWIN
, ui::EF_NONE
)));
984 EXPECT_TRUE(ash::Shell::GetInstance()->GetAppListTargetVisibility());
986 EXPECT_EQ(ui::VKEY_LWIN
,
987 GetPreviousAccelerator().key_code());
989 // When spoken feedback is on, the AppList should not toggle.
990 delegate
->ToggleSpokenFeedback(ui::A11Y_NOTIFICATION_NONE
);
992 ProcessInController(ui::Accelerator(ui::VKEY_LWIN
, ui::EF_NONE
)));
994 ProcessInController(ReleaseAccelerator(
995 ui::VKEY_LWIN
, ui::EF_NONE
)));
996 delegate
->ToggleSpokenFeedback(ui::A11Y_NOTIFICATION_NONE
);
997 EXPECT_TRUE(ash::Shell::GetInstance()->GetAppListTargetVisibility());
1000 ProcessInController(ui::Accelerator(ui::VKEY_LWIN
, ui::EF_NONE
)));
1002 ProcessInController(ReleaseAccelerator(
1003 ui::VKEY_LWIN
, ui::EF_NONE
)));
1004 EXPECT_FALSE(ash::Shell::GetInstance()->GetAppListTargetVisibility());
1006 // When spoken feedback is on, the AppList should not toggle.
1007 delegate
->ToggleSpokenFeedback(ui::A11Y_NOTIFICATION_NONE
);
1009 ProcessInController(ui::Accelerator(ui::VKEY_LWIN
, ui::EF_NONE
)));
1011 ProcessInController(ReleaseAccelerator(
1012 ui::VKEY_LWIN
, ui::EF_NONE
)));
1013 delegate
->ToggleSpokenFeedback(ui::A11Y_NOTIFICATION_NONE
);
1014 EXPECT_FALSE(ash::Shell::GetInstance()->GetAppListTargetVisibility());
1016 #if defined(OS_CHROMEOS)
1017 // The press of VKEY_BROWSER_SEARCH should toggle the AppList
1018 EXPECT_TRUE(ProcessInController(ui::Accelerator(ui::VKEY_BROWSER_SEARCH
,
1020 EXPECT_TRUE(ash::Shell::GetInstance()->GetAppListTargetVisibility());
1021 EXPECT_FALSE(ProcessInController(ReleaseAccelerator(ui::VKEY_BROWSER_SEARCH
,
1023 EXPECT_TRUE(ash::Shell::GetInstance()->GetAppListTargetVisibility());
1027 TEST_F(AcceleratorControllerTest
, ImeGlobalAccelerators
) {
1028 // Test IME shortcuts.
1030 ui::Accelerator
control_space_down(ui::VKEY_SPACE
, ui::EF_CONTROL_DOWN
);
1031 control_space_down
.set_type(ui::ET_KEY_PRESSED
);
1032 ui::Accelerator
control_space_up(ui::VKEY_SPACE
, ui::EF_CONTROL_DOWN
);
1033 control_space_up
.set_type(ui::ET_KEY_RELEASED
);
1034 const ui::Accelerator
convert(ui::VKEY_CONVERT
, ui::EF_NONE
);
1035 const ui::Accelerator
non_convert(ui::VKEY_NONCONVERT
, ui::EF_NONE
);
1036 const ui::Accelerator
wide_half_1(ui::VKEY_DBE_SBCSCHAR
, ui::EF_NONE
);
1037 const ui::Accelerator
wide_half_2(ui::VKEY_DBE_DBCSCHAR
, ui::EF_NONE
);
1038 const ui::Accelerator
hangul(ui::VKEY_HANGUL
, ui::EF_NONE
);
1039 EXPECT_FALSE(ProcessInController(control_space_down
));
1040 EXPECT_FALSE(ProcessInController(control_space_up
));
1041 EXPECT_FALSE(ProcessInController(convert
));
1042 EXPECT_FALSE(ProcessInController(non_convert
));
1043 EXPECT_FALSE(ProcessInController(wide_half_1
));
1044 EXPECT_FALSE(ProcessInController(wide_half_2
));
1045 EXPECT_FALSE(ProcessInController(hangul
));
1046 DummyImeControlDelegate
* delegate
= new DummyImeControlDelegate
;
1047 GetController()->SetImeControlDelegate(
1048 scoped_ptr
<ImeControlDelegate
>(delegate
).Pass());
1049 EXPECT_EQ(0, delegate
->handle_previous_ime_count());
1050 EXPECT_TRUE(ProcessInController(control_space_down
));
1051 EXPECT_EQ(1, delegate
->handle_previous_ime_count());
1052 EXPECT_TRUE(ProcessInController(control_space_up
));
1053 EXPECT_EQ(1, delegate
->handle_previous_ime_count());
1054 EXPECT_EQ(0, delegate
->handle_switch_ime_count());
1055 EXPECT_TRUE(ProcessInController(convert
));
1056 EXPECT_EQ(1, delegate
->handle_switch_ime_count());
1057 EXPECT_TRUE(ProcessInController(non_convert
));
1058 EXPECT_EQ(2, delegate
->handle_switch_ime_count());
1059 EXPECT_TRUE(ProcessInController(wide_half_1
));
1060 EXPECT_EQ(3, delegate
->handle_switch_ime_count());
1061 EXPECT_TRUE(ProcessInController(wide_half_2
));
1062 EXPECT_EQ(4, delegate
->handle_switch_ime_count());
1063 EXPECT_TRUE(ProcessInController(hangul
));
1064 EXPECT_EQ(5, delegate
->handle_switch_ime_count());
1067 // Test IME shortcuts that are triggered on key release.
1069 const ui::Accelerator
shift_alt_press(ui::VKEY_MENU
,
1070 ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
);
1071 const ReleaseAccelerator
shift_alt(ui::VKEY_MENU
, ui::EF_SHIFT_DOWN
);
1072 const ui::Accelerator
alt_shift_press(ui::VKEY_SHIFT
,
1073 ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
);
1074 const ReleaseAccelerator
alt_shift(ui::VKEY_SHIFT
, ui::EF_ALT_DOWN
);
1076 DummyImeControlDelegate
* delegate
= new DummyImeControlDelegate
;
1077 GetController()->SetImeControlDelegate(
1078 scoped_ptr
<ImeControlDelegate
>(delegate
).Pass());
1079 EXPECT_EQ(0, delegate
->handle_next_ime_count());
1080 EXPECT_FALSE(ProcessInController(shift_alt_press
));
1081 EXPECT_TRUE(ProcessInController(shift_alt
));
1082 EXPECT_EQ(1, delegate
->handle_next_ime_count());
1083 EXPECT_FALSE(ProcessInController(alt_shift_press
));
1084 EXPECT_TRUE(ProcessInController(alt_shift
));
1085 EXPECT_EQ(2, delegate
->handle_next_ime_count());
1087 // We should NOT switch IME when e.g. Shift+Alt+X is pressed and X is
1089 const ui::Accelerator
shift_alt_x_press(
1091 ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
);
1092 const ReleaseAccelerator
shift_alt_x(ui::VKEY_X
,
1093 ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
);
1095 EXPECT_FALSE(ProcessInController(shift_alt_press
));
1096 EXPECT_FALSE(ProcessInController(shift_alt_x_press
));
1097 EXPECT_FALSE(ProcessInController(shift_alt_x
));
1098 EXPECT_FALSE(ProcessInController(shift_alt
));
1099 EXPECT_EQ(2, delegate
->handle_next_ime_count());
1101 // But we _should_ if X is either VKEY_RETURN or VKEY_SPACE.
1102 // TODO(nona|mazda): Remove this when crbug.com/139556 in a better way.
1103 const ui::Accelerator
shift_alt_return_press(
1105 ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
);
1106 const ReleaseAccelerator
shift_alt_return(
1108 ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
);
1110 EXPECT_FALSE(ProcessInController(shift_alt_press
));
1111 EXPECT_FALSE(ProcessInController(shift_alt_return_press
));
1112 EXPECT_FALSE(ProcessInController(shift_alt_return
));
1113 EXPECT_TRUE(ProcessInController(shift_alt
));
1114 EXPECT_EQ(3, delegate
->handle_next_ime_count());
1116 const ui::Accelerator
shift_alt_space_press(
1118 ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
);
1119 const ReleaseAccelerator
shift_alt_space(
1121 ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
);
1123 EXPECT_FALSE(ProcessInController(shift_alt_press
));
1124 EXPECT_FALSE(ProcessInController(shift_alt_space_press
));
1125 EXPECT_FALSE(ProcessInController(shift_alt_space
));
1126 EXPECT_TRUE(ProcessInController(shift_alt
));
1127 EXPECT_EQ(4, delegate
->handle_next_ime_count());
1130 #if defined(OS_CHROMEOS)
1131 // Test IME shortcuts again with unnormalized accelerators (Chrome OS only).
1133 const ui::Accelerator
shift_alt_press(ui::VKEY_MENU
, ui::EF_SHIFT_DOWN
);
1134 const ReleaseAccelerator
shift_alt(ui::VKEY_MENU
, ui::EF_SHIFT_DOWN
);
1135 const ui::Accelerator
alt_shift_press(ui::VKEY_SHIFT
, ui::EF_ALT_DOWN
);
1136 const ReleaseAccelerator
alt_shift(ui::VKEY_SHIFT
, ui::EF_ALT_DOWN
);
1138 DummyImeControlDelegate
* delegate
= new DummyImeControlDelegate
;
1139 GetController()->SetImeControlDelegate(
1140 scoped_ptr
<ImeControlDelegate
>(delegate
).Pass());
1141 EXPECT_EQ(0, delegate
->handle_next_ime_count());
1142 EXPECT_FALSE(ProcessInController(shift_alt_press
));
1143 EXPECT_TRUE(ProcessInController(shift_alt
));
1144 EXPECT_EQ(1, delegate
->handle_next_ime_count());
1145 EXPECT_FALSE(ProcessInController(alt_shift_press
));
1146 EXPECT_TRUE(ProcessInController(alt_shift
));
1147 EXPECT_EQ(2, delegate
->handle_next_ime_count());
1149 // We should NOT switch IME when e.g. Shift+Alt+X is pressed and X is
1151 const ui::Accelerator
shift_alt_x_press(
1153 ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
);
1154 const ReleaseAccelerator
shift_alt_x(ui::VKEY_X
,
1155 ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
);
1157 EXPECT_FALSE(ProcessInController(shift_alt_press
));
1158 EXPECT_FALSE(ProcessInController(shift_alt_x_press
));
1159 EXPECT_FALSE(ProcessInController(shift_alt_x
));
1160 EXPECT_FALSE(ProcessInController(shift_alt
));
1161 EXPECT_EQ(2, delegate
->handle_next_ime_count());
1166 // TODO(nona|mazda): Remove this when crbug.com/139556 in a better way.
1167 TEST_F(AcceleratorControllerTest
, ImeGlobalAcceleratorsWorkaround139556
) {
1168 // The workaround for crbug.com/139556 depends on the fact that we don't
1169 // use Shift+Alt+Enter/Space with ET_KEY_PRESSED as an accelerator. Test it.
1170 const ui::Accelerator
shift_alt_return_press(
1172 ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
);
1173 EXPECT_FALSE(ProcessInController(shift_alt_return_press
));
1174 const ui::Accelerator
shift_alt_space_press(
1176 ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
);
1177 EXPECT_FALSE(ProcessInController(shift_alt_space_press
));
1180 TEST_F(AcceleratorControllerTest
, PreferredReservedAccelerators
) {
1181 #if defined(OS_CHROMEOS)
1182 // Power key is reserved on chromeos.
1183 EXPECT_TRUE(GetController()->IsReserved(
1184 ui::Accelerator(ui::VKEY_POWER
, ui::EF_NONE
)));
1185 EXPECT_FALSE(GetController()->IsPreferred(
1186 ui::Accelerator(ui::VKEY_POWER
, ui::EF_NONE
)));
1188 // ALT+Tab are not reserved but preferred.
1189 EXPECT_FALSE(GetController()->IsReserved(
1190 ui::Accelerator(ui::VKEY_TAB
, ui::EF_ALT_DOWN
)));
1191 EXPECT_FALSE(GetController()->IsReserved(
1192 ui::Accelerator(ui::VKEY_TAB
, ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
)));
1193 EXPECT_TRUE(GetController()->IsPreferred(
1194 ui::Accelerator(ui::VKEY_TAB
, ui::EF_ALT_DOWN
)));
1195 EXPECT_TRUE(GetController()->IsPreferred(
1196 ui::Accelerator(ui::VKEY_TAB
, ui::EF_SHIFT_DOWN
| ui::EF_ALT_DOWN
)));
1198 // Others are not reserved nor preferred
1199 EXPECT_FALSE(GetController()->IsReserved(
1200 ui::Accelerator(ui::VKEY_PRINT
, ui::EF_NONE
)));
1201 EXPECT_FALSE(GetController()->IsPreferred(
1202 ui::Accelerator(ui::VKEY_PRINT
, ui::EF_NONE
)));
1203 EXPECT_FALSE(GetController()->IsReserved(
1204 ui::Accelerator(ui::VKEY_TAB
, ui::EF_NONE
)));
1205 EXPECT_FALSE(GetController()->IsPreferred(
1206 ui::Accelerator(ui::VKEY_TAB
, ui::EF_NONE
)));
1207 EXPECT_FALSE(GetController()->IsReserved(
1208 ui::Accelerator(ui::VKEY_A
, ui::EF_NONE
)));
1209 EXPECT_FALSE(GetController()->IsPreferred(
1210 ui::Accelerator(ui::VKEY_A
, ui::EF_NONE
)));
1215 class PreferredReservedAcceleratorsTest
: public test::AshTestBase
{
1217 PreferredReservedAcceleratorsTest() {}
1218 ~PreferredReservedAcceleratorsTest() override
{}
1220 // test::AshTestBase:
1221 void SetUp() override
{
1222 AshTestBase::SetUp();
1223 Shell::GetInstance()->lock_state_controller()->
1224 set_animator_for_test(new test::TestSessionStateAnimator
);
1228 DISALLOW_COPY_AND_ASSIGN(PreferredReservedAcceleratorsTest
);
1233 TEST_F(PreferredReservedAcceleratorsTest
, AcceleratorsWithFullscreen
) {
1234 aura::Window
* w1
= CreateTestWindowInShellWithId(0);
1235 aura::Window
* w2
= CreateTestWindowInShellWithId(1);
1236 wm::ActivateWindow(w1
);
1238 wm::WMEvent
fullscreen(wm::WM_EVENT_FULLSCREEN
);
1239 wm::WindowState
* w1_state
= wm::GetWindowState(w1
);
1240 w1_state
->OnWMEvent(&fullscreen
);
1241 ASSERT_TRUE(w1_state
->IsFullscreen());
1243 ui::test::EventGenerator
& generator
= GetEventGenerator();
1244 #if defined(OS_CHROMEOS)
1245 // Power key (reserved) should always be handled.
1246 LockStateController::TestApi
test_api(
1247 Shell::GetInstance()->lock_state_controller());
1248 EXPECT_FALSE(test_api
.is_animating_lock());
1249 generator
.PressKey(ui::VKEY_POWER
, ui::EF_NONE
);
1250 EXPECT_TRUE(test_api
.is_animating_lock());
1253 // A fullscreen window can consume ALT-TAB (preferred).
1254 ASSERT_EQ(w1
, wm::GetActiveWindow());
1255 generator
.PressKey(ui::VKEY_TAB
, ui::EF_ALT_DOWN
);
1256 ASSERT_EQ(w1
, wm::GetActiveWindow());
1257 ASSERT_NE(w2
, wm::GetActiveWindow());
1259 // ALT-TAB is non repeatable. Press A to cancel the
1261 generator
.PressKey(ui::VKEY_A
, ui::EF_NONE
);
1262 generator
.ReleaseKey(ui::VKEY_A
, ui::EF_NONE
);
1264 // A normal window shouldn't consume preferred accelerator.
1265 wm::WMEvent
normal(wm::WM_EVENT_NORMAL
);
1266 w1_state
->OnWMEvent(&normal
);
1267 ASSERT_FALSE(w1_state
->IsFullscreen());
1269 EXPECT_EQ(w1
, wm::GetActiveWindow());
1270 generator
.PressKey(ui::VKEY_TAB
, ui::EF_ALT_DOWN
);
1271 ASSERT_NE(w1
, wm::GetActiveWindow());
1272 ASSERT_EQ(w2
, wm::GetActiveWindow());
1275 #if defined(OS_CHROMEOS)
1276 TEST_F(AcceleratorControllerTest
, DisallowedAtModalWindow
) {
1277 std::set
<AcceleratorAction
> all_actions
;
1278 for (size_t i
= 0 ; i
< kAcceleratorDataLength
; ++i
)
1279 all_actions
.insert(kAcceleratorData
[i
].action
);
1280 std::set
<AcceleratorAction
> all_debug_actions
;
1281 for (size_t i
= 0 ; i
< kDebugAcceleratorDataLength
; ++i
)
1282 all_debug_actions
.insert(kDebugAcceleratorData
[i
].action
);
1284 std::set
<AcceleratorAction
> actionsAllowedAtModalWindow
;
1285 for (size_t k
= 0 ; k
< kActionsAllowedAtModalWindowLength
; ++k
)
1286 actionsAllowedAtModalWindow
.insert(kActionsAllowedAtModalWindow
[k
]);
1287 for (std::set
<AcceleratorAction
>::const_iterator it
=
1288 actionsAllowedAtModalWindow
.begin();
1289 it
!= actionsAllowedAtModalWindow
.end(); ++it
) {
1290 EXPECT_TRUE(all_actions
.find(*it
) != all_actions
.end() ||
1291 all_debug_actions
.find(*it
) != all_debug_actions
.end())
1292 << " action from kActionsAllowedAtModalWindow"
1293 << " not found in kAcceleratorData or kDebugAcceleratorData. "
1294 << "action: " << *it
;
1296 scoped_ptr
<aura::Window
> window(
1297 CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
1298 wm::ActivateWindow(window
.get());
1299 Shell::GetInstance()->SimulateModalWindowOpenForTesting(true);
1300 for (std::set
<AcceleratorAction
>::const_iterator it
= all_actions
.begin();
1301 it
!= all_actions
.end(); ++it
) {
1302 if (actionsAllowedAtModalWindow
.find(*it
) ==
1303 actionsAllowedAtModalWindow
.end()) {
1304 EXPECT_TRUE(GetController()->PerformActionIfEnabled(*it
))
1305 << " for action (disallowed at modal window): " << *it
;
1308 // Testing of top row (F5-F10) accelerators that should still work
1309 // when a modal window is open
1313 test::TestScreenshotDelegate
* delegate
= GetScreenshotDelegate();
1314 delegate
->set_can_take_screenshot(false);
1315 EXPECT_TRUE(ProcessInController(
1316 ui::Accelerator(ui::VKEY_MEDIA_LAUNCH_APP1
, ui::EF_CONTROL_DOWN
)));
1318 ProcessInController(ui::Accelerator(
1319 ui::VKEY_PRINT
, ui::EF_NONE
)));
1320 EXPECT_TRUE(ProcessInController(ui::Accelerator(
1321 ui::VKEY_MEDIA_LAUNCH_APP1
, ui::EF_SHIFT_DOWN
| ui::EF_CONTROL_DOWN
)));
1322 delegate
->set_can_take_screenshot(true);
1323 EXPECT_EQ(0, delegate
->handle_take_screenshot_count());
1324 EXPECT_TRUE(ProcessInController(
1325 ui::Accelerator(ui::VKEY_MEDIA_LAUNCH_APP1
, ui::EF_CONTROL_DOWN
)));
1326 EXPECT_EQ(1, delegate
->handle_take_screenshot_count());
1328 ProcessInController(ui::Accelerator(
1329 ui::VKEY_PRINT
, ui::EF_NONE
)));
1330 EXPECT_EQ(2, delegate
->handle_take_screenshot_count());
1331 EXPECT_TRUE(ProcessInController(ui::Accelerator(
1332 ui::VKEY_MEDIA_LAUNCH_APP1
, ui::EF_SHIFT_DOWN
| ui::EF_CONTROL_DOWN
)));
1333 EXPECT_EQ(2, delegate
->handle_take_screenshot_count());
1336 const ui::Accelerator
brightness_down(ui::VKEY_BRIGHTNESS_DOWN
, ui::EF_NONE
);
1337 const ui::Accelerator
brightness_up(ui::VKEY_BRIGHTNESS_UP
, ui::EF_NONE
);
1339 DummyBrightnessControlDelegate
* delegate
=
1340 new DummyBrightnessControlDelegate
;
1341 GetController()->SetBrightnessControlDelegate(
1342 scoped_ptr
<BrightnessControlDelegate
>(delegate
).Pass());
1343 EXPECT_EQ(0, delegate
->handle_brightness_down_count());
1344 EXPECT_TRUE(ProcessInController(brightness_down
));
1345 EXPECT_EQ(1, delegate
->handle_brightness_down_count());
1346 EXPECT_EQ(brightness_down
, delegate
->last_accelerator());
1347 EXPECT_EQ(0, delegate
->handle_brightness_up_count());
1348 EXPECT_TRUE(ProcessInController(brightness_up
));
1349 EXPECT_EQ(1, delegate
->handle_brightness_up_count());
1350 EXPECT_EQ(brightness_up
, delegate
->last_accelerator());
1353 const ui::Accelerator
volume_mute(ui::VKEY_VOLUME_MUTE
, ui::EF_NONE
);
1354 const ui::Accelerator
volume_down(ui::VKEY_VOLUME_DOWN
, ui::EF_NONE
);
1355 const ui::Accelerator
volume_up(ui::VKEY_VOLUME_UP
, ui::EF_NONE
);
1357 EXPECT_TRUE(ProcessInController(volume_mute
));
1358 EXPECT_TRUE(ProcessInController(volume_down
));
1359 EXPECT_TRUE(ProcessInController(volume_up
));
1360 TestVolumeControlDelegate
* delegate
= new TestVolumeControlDelegate
;
1361 ash::Shell::GetInstance()->system_tray_delegate()->SetVolumeControlDelegate(
1362 scoped_ptr
<VolumeControlDelegate
>(delegate
).Pass());
1363 EXPECT_EQ(0, delegate
->handle_volume_mute_count());
1364 EXPECT_TRUE(ProcessInController(volume_mute
));
1365 EXPECT_EQ(1, delegate
->handle_volume_mute_count());
1366 EXPECT_EQ(volume_mute
, delegate
->last_accelerator());
1367 EXPECT_EQ(0, delegate
->handle_volume_down_count());
1368 EXPECT_TRUE(ProcessInController(volume_down
));
1369 EXPECT_EQ(1, delegate
->handle_volume_down_count());
1370 EXPECT_EQ(volume_down
, delegate
->last_accelerator());
1371 EXPECT_EQ(0, delegate
->handle_volume_up_count());
1372 EXPECT_TRUE(ProcessInController(volume_up
));
1373 EXPECT_EQ(1, delegate
->handle_volume_up_count());
1374 EXPECT_EQ(volume_up
, delegate
->last_accelerator());
1379 TEST_F(AcceleratorControllerTest
, DisallowedWithNoWindow
) {
1380 AccessibilityDelegate
* delegate
=
1381 ash::Shell::GetInstance()->accessibility_delegate();
1383 for (size_t i
= 0; i
< kActionsNeedingWindowLength
; ++i
) {
1384 delegate
->TriggerAccessibilityAlert(ui::A11Y_ALERT_NONE
);
1386 GetController()->PerformActionIfEnabled(kActionsNeedingWindow
[i
]));
1387 EXPECT_EQ(delegate
->GetLastAccessibilityAlert(),
1388 ui::A11Y_ALERT_WINDOW_NEEDED
);
1391 // Make sure we don't alert if we do have a window.
1392 scoped_ptr
<aura::Window
> window
;
1393 for (size_t i
= 0; i
< kActionsNeedingWindowLength
; ++i
) {
1394 window
.reset(CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
1395 wm::ActivateWindow(window
.get());
1396 delegate
->TriggerAccessibilityAlert(ui::A11Y_ALERT_NONE
);
1397 GetController()->PerformActionIfEnabled(kActionsNeedingWindow
[i
]);
1398 EXPECT_NE(delegate
->GetLastAccessibilityAlert(),
1399 ui::A11Y_ALERT_WINDOW_NEEDED
);
1402 // Don't alert if we have a minimized window either.
1403 for (size_t i
= 0; i
< kActionsNeedingWindowLength
; ++i
) {
1404 window
.reset(CreateTestWindowInShellWithBounds(gfx::Rect(5, 5, 20, 20)));
1405 wm::ActivateWindow(window
.get());
1406 GetController()->PerformActionIfEnabled(WINDOW_MINIMIZE
);
1407 delegate
->TriggerAccessibilityAlert(ui::A11Y_ALERT_NONE
);
1408 GetController()->PerformActionIfEnabled(kActionsNeedingWindow
[i
]);
1409 EXPECT_NE(delegate
->GetLastAccessibilityAlert(),
1410 ui::A11Y_ALERT_WINDOW_NEEDED
);
1414 #if defined(OS_CHROMEOS)
1417 // defines a class to test the behavior of deprecated accelerators.
1418 class DeprecatedAcceleratorTester
: public AcceleratorControllerTest
{
1420 DeprecatedAcceleratorTester() {}
1421 ~DeprecatedAcceleratorTester() override
{}
1423 ui::Accelerator
CreateAccelerator(const AcceleratorData
& data
) const {
1424 ui::Accelerator
result(data
.keycode
, data
.modifiers
);
1425 result
.set_type(data
.trigger_on_press
? ui::ET_KEY_PRESSED
1426 : ui::ET_KEY_RELEASED
);
1430 void ResetStateIfNeeded() {
1431 Shell
* shell
= Shell::GetInstance();
1432 if (shell
->session_state_delegate()->IsScreenLocked() ||
1433 shell
->session_state_delegate()->IsUserSessionBlocked()) {
1434 UnblockUserSession();
1438 bool ContainsDeprecatedAcceleratorNotification(const char* const id
) const {
1439 return nullptr != message_center()->FindVisibleNotificationById(id
);
1442 bool IsMessageCenterEmpty() const {
1443 return message_center()->GetVisibleNotifications().empty();
1446 void RemoveAllNotifications() const {
1447 message_center()->RemoveAllNotifications(false);
1450 message_center::MessageCenter
* message_center() const {
1451 return message_center::MessageCenter::Get();
1455 DISALLOW_COPY_AND_ASSIGN(DeprecatedAcceleratorTester
);
1460 TEST_F(DeprecatedAcceleratorTester
, TestDeprecatedAcceleratorsBehavior
) {
1461 for (size_t i
= 0; i
< kDeprecatedAcceleratorsLength
; ++i
) {
1462 const DeprecatedAcceleratorData
& data
= kDeprecatedAccelerators
[i
];
1464 EXPECT_TRUE(IsMessageCenterEmpty());
1466 ui::Accelerator deprecated_accelerator
=
1467 CreateAccelerator(data
.deprecated_accelerator
);
1468 if (data
.deprecated_enabled
)
1469 EXPECT_TRUE(ProcessInController(deprecated_accelerator
));
1471 EXPECT_FALSE(ProcessInController(deprecated_accelerator
));
1473 // We expect to see a notification in the message center.
1475 ContainsDeprecatedAcceleratorNotification(data
.uma_histogram_name
));
1476 RemoveAllNotifications();
1478 // If the action is LOCK_SCREEN, we must reset the state by unlocking the
1479 // screen before we proceed testing the rest of accelerators.
1480 ResetStateIfNeeded();
1484 TEST_F(DeprecatedAcceleratorTester
, TestNewAccelerators
) {
1485 // Add below the new accelerators that replaced the deprecated ones (if any).
1486 const AcceleratorData kNewAccelerators
[] = {
1487 {true, ui::VKEY_L
, ui::EF_COMMAND_DOWN
, LOCK_SCREEN
},
1488 {true, ui::VKEY_ESCAPE
, ui::EF_COMMAND_DOWN
, SHOW_TASK_MANAGER
},
1491 EXPECT_TRUE(IsMessageCenterEmpty());
1493 for (auto data
: kNewAccelerators
) {
1494 EXPECT_TRUE(ProcessInController(CreateAccelerator(data
)));
1496 // Expect no notifications from the new accelerators.
1497 EXPECT_TRUE(IsMessageCenterEmpty());
1499 // If the action is LOCK_SCREEN, we must reset the state by unlocking the
1500 // screen before we proceed testing the rest of accelerators.
1501 ResetStateIfNeeded();
1504 #endif // defined(OS_CHROMEOS)