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. 4 5 #include "ash/accelerators/accelerator_dispatcher.h" 6 7 #if defined(USE_X11) 8 #include <X11/Xlib.h> 9 10 // Xlib defines RootWindow 11 #ifdef RootWindow 12 #undef RootWindow 13 #endif 14 #endif // defined(USE_X11) 15 16 #include "ash/accelerators/accelerator_controller.h" 17 #include "ash/shell.h" 18 #include "ash/wm/event_rewriter_event_filter.h" 19 #include "ui/aura/env.h" 20 #include "ui/aura/root_window.h" 21 #include "ui/base/accelerators/accelerator.h" 22 #include "ui/base/events/event.h" 23 #include "ui/base/events/event_constants.h" 24 #include "ui/base/events/event_utils.h" 25 #include "ui/views/controls/menu/menu_controller.h" 26 27 namespace ash { 28 namespace { 29 30 const int kModifierMask = (ui::EF_SHIFT_DOWN | 31 ui::EF_CONTROL_DOWN | 32 ui::EF_ALT_DOWN); 33 #if defined(OS_WIN) 34 bool IsKeyEvent(const MSG& msg) { 35 return 36 msg.message == WM_KEYDOWN || msg.message == WM_SYSKEYDOWN || 37 msg.message == WM_KEYUP || msg.message == WM_SYSKEYUP; 38 } 39 #elif defined(USE_X11) 40 bool IsKeyEvent(const XEvent* xev) { 41 return xev->type == KeyPress || xev->type == KeyRelease; 42 } 43 #endif 44 45 bool IsPossibleAcceleratorNotForMenu(const ui::KeyEvent& key_event) { 46 // For shortcuts generated by Ctrl or Alt plus a letter, number or 47 // the tab key, we want to exit the context menu first and then 48 // repost the event. That allows for the shortcut execution after 49 // the context menu has exited. 50 if (key_event.type() == ui::ET_KEY_PRESSED && 51 (key_event.flags() & (ui::EF_CONTROL_DOWN | ui::EF_ALT_DOWN))) { 52 const ui::KeyboardCode key_code = key_event.key_code(); 53 if ((key_code >= ui::VKEY_A && key_code <= ui::VKEY_Z) || 54 (key_code >= ui::VKEY_0 && key_code <= ui::VKEY_9) || 55 (key_code == ui::VKEY_TAB)) { 56 return true; 57 } 58 } 59 return false; 60 } 61 62 } // namespace 63 64 AcceleratorDispatcher::AcceleratorDispatcher( 65 base::MessageLoop::Dispatcher* nested_dispatcher, 66 aura::Window* associated_window) 67 : nested_dispatcher_(nested_dispatcher), 68 associated_window_(associated_window) { 69 DCHECK(nested_dispatcher_); 70 associated_window_->AddObserver(this); 71 } 72 73 AcceleratorDispatcher::~AcceleratorDispatcher() { 74 if (associated_window_) 75 associated_window_->RemoveObserver(this); 76 } 77 78 void AcceleratorDispatcher::OnWindowDestroying(aura::Window* window) { 79 if (associated_window_ == window) 80 associated_window_ = NULL; 81 } 82 83 bool AcceleratorDispatcher::Dispatch(const base::NativeEvent& event) { 84 if (!associated_window_) 85 return false; 86 if (!ui::IsNoopEvent(event) && !associated_window_->CanReceiveEvents()) 87 return aura::Env::GetInstance()->GetDispatcher()->Dispatch(event); 88 89 if (IsKeyEvent(event)) { 90 // Modifiers can be changed by the user preference, so we need to rewrite 91 // the event explicitly. 92 ui::KeyEvent key_event(event, false); 93 ui::EventHandler* event_rewriter = 94 ash::Shell::GetInstance()->event_rewriter_filter(); 95 DCHECK(event_rewriter); 96 event_rewriter->OnKeyEvent(&key_event); 97 if (key_event.stopped_propagation()) 98 return true; 99 100 if (IsPossibleAcceleratorNotForMenu(key_event)) { 101 if (views::MenuController* menu_controller = 102 views::MenuController::GetActiveInstance()) { 103 menu_controller->CancelAll(); 104 #if defined(USE_X11) 105 XPutBackEvent(event->xany.display, event); 106 #else 107 NOTIMPLEMENTED() << " Repost NativeEvent here."; 108 #endif 109 return false; 110 } 111 } 112 113 ash::AcceleratorController* accelerator_controller = 114 ash::Shell::GetInstance()->accelerator_controller(); 115 if (accelerator_controller) { 116 ui::Accelerator accelerator(key_event.key_code(), 117 key_event.flags() & kModifierMask); 118 if (key_event.type() == ui::ET_KEY_RELEASED) 119 accelerator.set_type(ui::ET_KEY_RELEASED); 120 // Fill out context object so AcceleratorController will know what 121 // was the previous accelerator or if the current accelerator is repeated. 122 Shell::GetInstance()->accelerator_controller()->context()-> 123 UpdateContext(accelerator); 124 if (accelerator_controller->Process(accelerator)) 125 return true; 126 } 127 128 return nested_dispatcher_->Dispatch(key_event.native_event()); 129 } 130 131 return nested_dispatcher_->Dispatch(event); 132 } 133 134 } // namespace ash 135