summaryrefslogtreecommitdiff
path: root/chromium/ash/accelerators/accelerator_controller.cc
diff options
context:
space:
mode:
authorAndras Becsi <andras.becsi@digia.com>2014-03-18 13:16:26 +0100
committerFrederik Gladhorn <frederik.gladhorn@digia.com>2014-03-20 15:55:39 +0100
commit3f0f86b0caed75241fa71c95a5d73bc0164348c5 (patch)
tree92b9fb00f2e9e90b0be2262093876d4f43b6cd13 /chromium/ash/accelerators/accelerator_controller.cc
parente90d7c4b152c56919d963987e2503f9909a666d2 (diff)
downloadqtwebengine-chromium-3f0f86b0caed75241fa71c95a5d73bc0164348c5.tar.gz
Update to new stable branch 1750
This also includes an updated ninja and chromium dependencies needed on Windows. Change-Id: Icd597d80ed3fa4425933c9f1334c3c2e31291c42 Reviewed-by: Zoltan Arvai <zarvai@inf.u-szeged.hu> Reviewed-by: Zeno Albisser <zeno.albisser@digia.com>
Diffstat (limited to 'chromium/ash/accelerators/accelerator_controller.cc')
-rw-r--r--chromium/ash/accelerators/accelerator_controller.cc1215
1 files changed, 1215 insertions, 0 deletions
diff --git a/chromium/ash/accelerators/accelerator_controller.cc b/chromium/ash/accelerators/accelerator_controller.cc
new file mode 100644
index 00000000000..0ac3f7ef4b2
--- /dev/null
+++ b/chromium/ash/accelerators/accelerator_controller.cc
@@ -0,0 +1,1215 @@
+// Copyright (c) 2012 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "ash/accelerators/accelerator_controller.h"
+
+#include <algorithm>
+#include <cmath>
+#include <iostream>
+#include <string>
+
+#include "ash/accelerators/accelerator_commands.h"
+#include "ash/accelerators/accelerator_table.h"
+#include "ash/accelerators/debug_commands.h"
+#include "ash/ash_switches.h"
+#include "ash/caps_lock_delegate.h"
+#include "ash/debug.h"
+#include "ash/display/display_controller.h"
+#include "ash/display/display_manager.h"
+#include "ash/focus_cycler.h"
+#include "ash/ime_control_delegate.h"
+#include "ash/launcher/launcher.h"
+#include "ash/magnifier/magnification_controller.h"
+#include "ash/magnifier/partial_magnification_controller.h"
+#include "ash/media_delegate.h"
+#include "ash/multi_profile_uma.h"
+#include "ash/new_window_delegate.h"
+#include "ash/root_window_controller.h"
+#include "ash/rotator/screen_rotation.h"
+#include "ash/screenshot_delegate.h"
+#include "ash/session_state_delegate.h"
+#include "ash/shelf/shelf_delegate.h"
+#include "ash/shelf/shelf_model.h"
+#include "ash/shelf/shelf_widget.h"
+#include "ash/shell.h"
+#include "ash/shell_delegate.h"
+#include "ash/shell_window_ids.h"
+#include "ash/system/brightness_control_delegate.h"
+#include "ash/system/keyboard_brightness/keyboard_brightness_control_delegate.h"
+#include "ash/system/status_area_widget.h"
+#include "ash/system/tray/system_tray.h"
+#include "ash/system/tray/system_tray_delegate.h"
+#include "ash/system/tray/system_tray_notifier.h"
+#include "ash/system/web_notification/web_notification_tray.h"
+#include "ash/touch/touch_hud_debug.h"
+#include "ash/volume_control_delegate.h"
+#include "ash/wm/mru_window_tracker.h"
+#include "ash/wm/overview/window_selector_controller.h"
+#include "ash/wm/partial_screenshot_view.h"
+#include "ash/wm/power_button_controller.h"
+#include "ash/wm/window_cycle_controller.h"
+#include "ash/wm/window_state.h"
+#include "ash/wm/window_util.h"
+#include "ash/wm/workspace/snap_sizer.h"
+#include "base/bind.h"
+#include "base/command_line.h"
+#include "content/public/browser/gpu_data_manager.h"
+#include "content/public/browser/user_metrics.h"
+#include "ui/aura/env.h"
+#include "ui/aura/root_window.h"
+#include "ui/base/accelerators/accelerator.h"
+#include "ui/base/accelerators/accelerator_manager.h"
+#include "ui/compositor/debug_utils.h"
+#include "ui/compositor/layer.h"
+#include "ui/compositor/layer_animation_sequence.h"
+#include "ui/compositor/layer_animator.h"
+#include "ui/events/event.h"
+#include "ui/events/keycodes/keyboard_codes.h"
+#include "ui/gfx/screen.h"
+#include "ui/oak/oak.h"
+#include "ui/views/controls/webview/webview.h"
+#include "ui/views/debug_utils.h"
+#include "ui/views/widget/widget.h"
+
+#if defined(OS_CHROMEOS)
+#include "ash/session_state_delegate.h"
+#include "ash/system/chromeos/keyboard_brightness_controller.h"
+#include "base/sys_info.h"
+#endif // defined(OS_CHROMEOS)
+
+namespace ash {
+namespace {
+
+using internal::DisplayInfo;
+using content::UserMetricsAction;
+
+bool DebugShortcutsEnabled() {
+#if defined(NDEBUG)
+ return CommandLine::ForCurrentProcess()->HasSwitch(
+ switches::kAshDebugShortcuts);
+#else
+ return true;
+#endif
+}
+
+bool HandleAccessibleFocusCycle(bool reverse) {
+ if (reverse) {
+ content::RecordAction(UserMetricsAction("Accel_Accessible_Focus_Previous"));
+ } else {
+ content::RecordAction(UserMetricsAction("Accel_Accessible_Focus_Next"));
+ }
+
+ if (!Shell::GetInstance()->accessibility_delegate()->
+ IsSpokenFeedbackEnabled()) {
+ return false;
+ }
+ aura::Window* active_window = ash::wm::GetActiveWindow();
+ if (!active_window)
+ return false;
+ views::Widget* widget =
+ views::Widget::GetWidgetForNativeWindow(active_window);
+ if (!widget)
+ return false;
+ views::FocusManager* focus_manager = widget->GetFocusManager();
+ if (!focus_manager)
+ return false;
+ views::View* view = focus_manager->GetFocusedView();
+ if (!view)
+ return false;
+ if (!strcmp(view->GetClassName(), views::WebView::kViewClassName))
+ return false;
+
+ focus_manager->AdvanceFocus(reverse);
+ return true;
+}
+
+bool HandleCycleBackwardMRU(const ui::Accelerator& accelerator) {
+ Shell* shell = Shell::GetInstance();
+
+ if (accelerator.key_code() == ui::VKEY_TAB)
+ content::RecordAction(content::UserMetricsAction("Accel_PrevWindow_Tab"));
+
+ if (switches::UseOverviewMode()) {
+ shell->window_selector_controller()->HandleCycleWindow(
+ WindowSelector::BACKWARD);
+ return true;
+ }
+ shell->window_cycle_controller()->HandleCycleWindow(
+ WindowCycleController::BACKWARD, accelerator.IsAltDown());
+ return true;
+}
+
+bool HandleCycleForwardMRU(const ui::Accelerator& accelerator) {
+ Shell* shell = Shell::GetInstance();
+
+ if (accelerator.key_code() == ui::VKEY_TAB)
+ content::RecordAction(content::UserMetricsAction("Accel_NextWindow_Tab"));
+
+ if (switches::UseOverviewMode()) {
+ shell->window_selector_controller()->HandleCycleWindow(
+ WindowSelector::FORWARD);
+ return true;
+ }
+ shell->window_cycle_controller()->HandleCycleWindow(
+ WindowCycleController::FORWARD, accelerator.IsAltDown());
+ return true;
+}
+
+bool HandleCycleLinear(const ui::Accelerator& accelerator) {
+ Shell* shell = Shell::GetInstance();
+
+ // TODO(jamescook): When overview becomes the default the AcceleratorAction
+ // should be renamed from CYCLE_LINEAR to TOGGLE_OVERVIEW.
+ if (switches::UseOverviewMode()) {
+ content::RecordAction(content::UserMetricsAction("Accel_Overview_F5"));
+ shell->window_selector_controller()->ToggleOverview();
+ return true;
+ }
+ if (accelerator.key_code() == ui::VKEY_MEDIA_LAUNCH_APP1)
+ content::RecordAction(content::UserMetricsAction("Accel_NextWindow_F5"));
+ shell->window_cycle_controller()->HandleLinearCycleWindow();
+ return true;
+}
+
+bool HandleDisableCapsLock(ui::KeyboardCode key_code,
+ ui::EventType previous_event_type,
+ ui::KeyboardCode previous_key_code) {
+ Shell* shell = Shell::GetInstance();
+
+ if (previous_event_type == ui::ET_KEY_RELEASED ||
+ (previous_key_code != ui::VKEY_LSHIFT &&
+ previous_key_code != ui::VKEY_SHIFT &&
+ previous_key_code != ui::VKEY_RSHIFT)) {
+ // If something else was pressed between the Shift key being pressed
+ // and released, then ignore the release of the Shift key.
+ return false;
+ }
+ content::RecordAction(UserMetricsAction("Accel_Disable_Caps_Lock"));
+ if (shell->caps_lock_delegate()->IsCapsLockEnabled()) {
+ shell->caps_lock_delegate()->SetCapsLockEnabled(false);
+ return true;
+ }
+ return false;
+}
+
+bool HandleFocusLauncher() {
+ Shell* shell = Shell::GetInstance();
+ content::RecordAction(content::UserMetricsAction("Accel_Focus_Launcher"));
+ return shell->focus_cycler()->FocusWidget(
+ Launcher::ForPrimaryDisplay()->shelf_widget());
+}
+
+bool HandleLaunchAppN(int n) {
+ content::RecordAction(UserMetricsAction("Accel_Launch_App"));
+ Launcher::ForPrimaryDisplay()->LaunchAppIndexAt(n);
+ return true;
+}
+
+bool HandleLaunchLastApp() {
+ content::RecordAction(UserMetricsAction("Accel_Launch_Last_App"));
+ Launcher::ForPrimaryDisplay()->LaunchAppIndexAt(-1);
+ return true;
+}
+
+// Magnify the screen
+bool HandleMagnifyScreen(int delta_index) {
+ if (ash::Shell::GetInstance()->magnification_controller()->IsEnabled()) {
+ // TODO(yoshiki): Move the following logic to MagnificationController.
+ float scale =
+ ash::Shell::GetInstance()->magnification_controller()->GetScale();
+ // Calculate rounded logarithm (base kMagnificationScaleFactor) of scale.
+ int scale_index =
+ std::floor(std::log(scale) / std::log(kMagnificationScaleFactor) + 0.5);
+
+ int new_scale_index = std::max(0, std::min(8, scale_index + delta_index));
+
+ ash::Shell::GetInstance()->magnification_controller()->
+ SetScale(std::pow(kMagnificationScaleFactor, new_scale_index), true);
+ } else if (ash::Shell::GetInstance()->
+ partial_magnification_controller()->is_enabled()) {
+ float scale = delta_index > 0 ? kDefaultPartialMagnifiedScale : 1;
+ ash::Shell::GetInstance()->partial_magnification_controller()->
+ SetScale(scale);
+ }
+
+ return true;
+}
+
+bool HandleMediaNextTrack() {
+ Shell::GetInstance()->media_delegate()->HandleMediaNextTrack();
+ return true;
+}
+
+bool HandleMediaPlayPause() {
+ Shell::GetInstance()->media_delegate()->HandleMediaPlayPause();
+ return true;
+}
+
+bool HandleMediaPrevTrack() {
+ Shell::GetInstance()->media_delegate()->HandleMediaPrevTrack();
+ return true;
+}
+
+bool HandleNewIncognitoWindow() {
+ content::RecordAction(UserMetricsAction("Accel_New_Incognito_Window"));
+ bool incognito_allowed =
+ Shell::GetInstance()->delegate()->IsIncognitoAllowed();
+ if (incognito_allowed)
+ Shell::GetInstance()->new_window_delegate()->NewWindow(
+ true /* is_incognito */);
+ return incognito_allowed;
+}
+
+bool HandleNewTab(ui::KeyboardCode key_code) {
+ if (key_code == ui::VKEY_T)
+ content::RecordAction(content::UserMetricsAction("Accel_NewTab_T"));
+ Shell::GetInstance()->new_window_delegate()->NewTab();
+ return true;
+}
+
+bool HandleNewWindow() {
+ content::RecordAction(content::UserMetricsAction("Accel_New_Window"));
+ Shell::GetInstance()->new_window_delegate()->NewWindow(
+ false /* is_incognito */);
+ return true;
+}
+
+bool HandleNextIme(ImeControlDelegate* ime_control_delegate,
+ ui::EventType previous_event_type,
+ ui::KeyboardCode previous_key_code) {
+ // This check is necessary e.g. not to process the Shift+Alt+
+ // ET_KEY_RELEASED accelerator for Chrome OS (see ash/accelerators/
+ // accelerator_controller.cc) when Shift+Alt+Tab is pressed and then Tab
+ // is released.
+ if (previous_event_type == ui::ET_KEY_RELEASED &&
+ // Workaround for crbug.com/139556. CJK IME users tend to press
+ // Enter (or Space) and Shift+Alt almost at the same time to commit
+ // an IME string and then switch from the IME to the English layout.
+ // This workaround allows the user to trigger NEXT_IME even if the
+ // user presses Shift+Alt before releasing Enter.
+ // TODO(nona|mazda): Fix crbug.com/139556 in a cleaner way.
+ previous_key_code != ui::VKEY_RETURN &&
+ previous_key_code != ui::VKEY_SPACE) {
+ // We totally ignore this accelerator.
+ // TODO(mazda): Fix crbug.com/158217
+ return false;
+ }
+ content::RecordAction(UserMetricsAction("Accel_Next_Ime"));
+ if (ime_control_delegate)
+ return ime_control_delegate->HandleNextIme();
+ return false;
+}
+
+bool HandleOpenFeedbackPage() {
+ content::RecordAction(UserMetricsAction("Accel_Open_Feedback_Page"));
+ ash::Shell::GetInstance()->new_window_delegate()->OpenFeedbackPage();
+ return true;
+}
+
+bool HandlePositionCenter() {
+ content::RecordAction(UserMetricsAction("Accel_Window_Position_Center"));
+ aura::Window* window = wm::GetActiveWindow();
+ // Docked windows do not support centering and ignore accelerator.
+ if (window && !wm::GetWindowState(window)->IsDocked()) {
+ wm::CenterWindow(window);
+ return true;
+ }
+ return false;
+}
+
+bool HandlePreviousIme(ImeControlDelegate* ime_control_delegate,
+ const ui::Accelerator& accelerator) {
+ content::RecordAction(UserMetricsAction("Accel_Previous_Ime"));
+ if (ime_control_delegate)
+ return ime_control_delegate->HandlePreviousIme(accelerator);
+ return false;
+}
+
+bool HandleRestoreTab() {
+ content::RecordAction(content::UserMetricsAction("Accel_Restore_Tab"));
+ Shell::GetInstance()->new_window_delegate()->RestoreTab();
+ return true;
+}
+
+bool HandleRotatePaneFocus(Shell::Direction direction) {
+ Shell* shell = Shell::GetInstance();
+ switch (direction) {
+ // TODO(stevet): Not sure if this is the same as IDC_FOCUS_NEXT_PANE.
+ case Shell::FORWARD: {
+ content::RecordAction(UserMetricsAction("Accel_Focus_Next_Pane"));
+ shell->focus_cycler()->RotateFocus(internal::FocusCycler::FORWARD);
+ break;
+ }
+ case Shell::BACKWARD: {
+ content::RecordAction(UserMetricsAction("Accel_Focus_Previous_Pane"));
+ shell->focus_cycler()->RotateFocus(internal::FocusCycler::BACKWARD);
+ break;
+ }
+ }
+ return true;
+}
+
+// Rotate the active window.
+bool HandleRotateActiveWindow() {
+ content::RecordAction(UserMetricsAction("Accel_Rotate_Window"));
+ aura::Window* active_window = wm::GetActiveWindow();
+ if (active_window) {
+ // The rotation animation bases its target transform on the current
+ // rotation and position. Since there could be an animation in progress
+ // right now, queue this animation so when it starts it picks up a neutral
+ // rotation and position. Use replace so we only enqueue one at a time.
+ active_window->layer()->GetAnimator()->
+ set_preemption_strategy(ui::LayerAnimator::REPLACE_QUEUED_ANIMATIONS);
+ active_window->layer()->GetAnimator()->StartAnimation(
+ new ui::LayerAnimationSequence(
+ new ash::ScreenRotation(360, active_window->layer())));
+ }
+ return true;
+}
+
+gfx::Display::Rotation GetNextRotation(gfx::Display::Rotation current) {
+ switch (current) {
+ case gfx::Display::ROTATE_0:
+ return gfx::Display::ROTATE_90;
+ case gfx::Display::ROTATE_90:
+ return gfx::Display::ROTATE_180;
+ case gfx::Display::ROTATE_180:
+ return gfx::Display::ROTATE_270;
+ case gfx::Display::ROTATE_270:
+ return gfx::Display::ROTATE_0;
+ }
+ NOTREACHED() << "Unknown rotation:" << current;
+ return gfx::Display::ROTATE_0;
+}
+
+// Rotates the screen.
+bool HandleRotateScreen() {
+ content::RecordAction(UserMetricsAction("Accel_Rotate_Window"));
+ gfx::Point point = Shell::GetScreen()->GetCursorScreenPoint();
+ gfx::Display display = Shell::GetScreen()->GetDisplayNearestPoint(point);
+ const DisplayInfo& display_info =
+ Shell::GetInstance()->display_manager()->GetDisplayInfo(display.id());
+ Shell::GetInstance()->display_manager()->SetDisplayRotation(
+ display.id(), GetNextRotation(display_info.rotation()));
+ return true;
+}
+
+bool HandleScaleReset() {
+ internal::DisplayManager* display_manager =
+ Shell::GetInstance()->display_manager();
+ int64 display_id = display_manager->GetDisplayIdForUIScaling();
+ if (display_id == gfx::Display::kInvalidDisplayID)
+ return false;
+
+ content::RecordAction(UserMetricsAction("Accel_Scale_Ui_Reset"));
+
+ display_manager->SetDisplayUIScale(display_id, 1.0f);
+ return true;
+}
+
+bool HandleScaleUI(bool up) {
+ internal::DisplayManager* display_manager =
+ Shell::GetInstance()->display_manager();
+ int64 display_id = display_manager->GetDisplayIdForUIScaling();
+ if (display_id == gfx::Display::kInvalidDisplayID)
+ return false;
+
+ if (up) {
+ content::RecordAction(UserMetricsAction("Accel_Scale_Ui_Up"));
+ } else {
+ content::RecordAction(UserMetricsAction("Accel_Scale_Ui_Down"));
+ }
+
+ const DisplayInfo& display_info = display_manager->GetDisplayInfo(display_id);
+ float next_scale =
+ internal::DisplayManager::GetNextUIScale(display_info, up);
+ display_manager->SetDisplayUIScale(display_id, next_scale);
+ return true;
+}
+
+bool HandleSwapPrimaryDisplay() {
+ content::RecordAction(UserMetricsAction("Accel_Swap_Primary_Display"));
+ Shell::GetInstance()->display_controller()->SwapPrimaryDisplay();
+ return true;
+}
+
+bool HandleShowKeyboardOverlay() {
+ content::RecordAction(UserMetricsAction("Accel_Show_Keyboard_Overlay"));
+ ash::Shell::GetInstance()->new_window_delegate()->ShowKeyboardOverlay();
+
+ return true;
+}
+
+void HandleShowMessageCenterBubble() {
+ content::RecordAction(UserMetricsAction("Accel_Show_Message_Center_Bubble"));
+ internal::RootWindowController* controller =
+ internal::RootWindowController::ForTargetRootWindow();
+ internal::StatusAreaWidget* status_area_widget =
+ controller->shelf()->status_area_widget();
+ if (status_area_widget) {
+ WebNotificationTray* notification_tray =
+ status_area_widget->web_notification_tray();
+ if (notification_tray->visible())
+ notification_tray->ShowMessageCenterBubble();
+ }
+}
+
+bool HandleShowOak() {
+ if (CommandLine::ForCurrentProcess()->HasSwitch(
+ switches::kAshEnableOak)) {
+ oak::ShowOakWindowWithContext(Shell::GetPrimaryRootWindow());
+ return true;
+ }
+ return false;
+}
+
+bool HandleShowSystemTrayBubble() {
+ content::RecordAction(UserMetricsAction("Accel_Show_System_Tray_Bubble"));
+ internal::RootWindowController* controller =
+ internal::RootWindowController::ForTargetRootWindow();
+ if (!controller->GetSystemTray()->HasSystemBubble()) {
+ controller->GetSystemTray()->ShowDefaultView(BUBBLE_CREATE_NEW);
+ return true;
+ }
+ return false;
+}
+
+bool HandleShowTaskManager() {
+ content::RecordAction(UserMetricsAction("Accel_Show_Task_Manager"));
+ Shell::GetInstance()->new_window_delegate()->ShowTaskManager();
+ return true;
+}
+
+void HandleSilenceSpokenFeedback() {
+ content::RecordAction(UserMetricsAction("Accel_Silence_Spoken_Feedback"));
+
+ AccessibilityDelegate* delegate =
+ Shell::GetInstance()->accessibility_delegate();
+ if (!delegate->IsSpokenFeedbackEnabled())
+ return;
+ delegate->SilenceSpokenFeedback();
+}
+
+bool HandleSwitchIme(ImeControlDelegate* ime_control_delegate,
+ const ui::Accelerator& accelerator) {
+ content::RecordAction(UserMetricsAction("Accel_Switch_Ime"));
+ if (ime_control_delegate)
+ return ime_control_delegate->HandleSwitchIme(accelerator);
+ return false;
+}
+
+bool HandleTakePartialScreenshot(ScreenshotDelegate* screenshot_delegate) {
+ content::RecordAction(UserMetricsAction("Accel_Take_Partial_Screenshot"));
+ if (screenshot_delegate) {
+ ash::PartialScreenshotView::StartPartialScreenshot(
+ screenshot_delegate);
+ }
+ // Return true to prevent propagation of the key event because
+ // this key combination is reserved for partial screenshot.
+ return true;
+}
+
+bool HandleTakeScreenshot(ScreenshotDelegate* screenshot_delegate) {
+ content::RecordAction(UserMetricsAction("Accel_Take_Screenshot"));
+ if (screenshot_delegate &&
+ screenshot_delegate->CanTakeScreenshot()) {
+ screenshot_delegate->HandleTakeScreenshotForAllRootWindows();
+ }
+ // Return true to prevent propagation of the key event.
+ return true;
+}
+
+bool HandleToggleAppList(ui::KeyboardCode key_code,
+ ui::EventType previous_event_type,
+ ui::KeyboardCode previous_key_code,
+ const ui::Accelerator& accelerator) {
+ // If something else was pressed between the Search key (LWIN)
+ // being pressed and released, then ignore the release of the
+ // Search key.
+ if (key_code == ui::VKEY_LWIN &&
+ (previous_event_type == ui::ET_KEY_RELEASED ||
+ previous_key_code != ui::VKEY_LWIN))
+ return false;
+ if (key_code == ui::VKEY_LWIN)
+ content::RecordAction(content::UserMetricsAction("Accel_Search_LWin"));
+ // When spoken feedback is enabled, we should neither toggle the list nor
+ // consume the key since Search+Shift is one of the shortcuts the a11y
+ // feature uses. crbug.com/132296
+ DCHECK_EQ(ui::VKEY_LWIN, accelerator.key_code());
+ if (Shell::GetInstance()->accessibility_delegate()->
+ IsSpokenFeedbackEnabled())
+ return false;
+ ash::Shell::GetInstance()->ToggleAppList(NULL);
+ return true;
+}
+
+bool HandleToggleCapsLock(ui::KeyboardCode key_code,
+ ui::EventType previous_event_type,
+ ui::KeyboardCode previous_key_code) {
+ Shell* shell = Shell::GetInstance();
+ if (key_code == ui::VKEY_LWIN) {
+ // If something else was pressed between the Search key (LWIN)
+ // being pressed and released, then ignore the release of the
+ // Search key.
+ // TODO(danakj): Releasing Alt first breaks this: crbug.com/166495
+ if (previous_event_type == ui::ET_KEY_RELEASED ||
+ previous_key_code != ui::VKEY_LWIN)
+ return false;
+ }
+ content::RecordAction(UserMetricsAction("Accel_Toggle_Caps_Lock"));
+ shell->caps_lock_delegate()->ToggleCapsLock();
+ return true;
+}
+
+bool HandleToggleFullscreen(ui::KeyboardCode key_code) {
+ if (key_code == ui::VKEY_MEDIA_LAUNCH_APP2) {
+ content::RecordAction(UserMetricsAction("Accel_Fullscreen_F4"));
+ }
+ accelerators::ToggleFullscreen();
+ return true;
+}
+
+bool HandleToggleRootWindowFullScreen() {
+ Shell::GetPrimaryRootWindow()->GetDispatcher()->host()->ToggleFullScreen();
+ return true;
+}
+
+bool HandleWindowSnap(int action) {
+ wm::WindowState* window_state = wm::GetActiveWindowState();
+ // Disable window snapping shortcut key for full screen window due to
+ // http://crbug.com/135487.
+ if (!window_state ||
+ window_state->window()->type() != aura::client::WINDOW_TYPE_NORMAL ||
+ window_state->IsFullscreen()) {
+ return false;
+ }
+
+ if (action == WINDOW_SNAP_LEFT) {
+ content::RecordAction(UserMetricsAction("Accel_Window_Snap_Left"));
+ } else {
+ content::RecordAction(UserMetricsAction("Accel_Window_Snap_Right"));
+ }
+
+ internal::SnapSizer::SnapWindow(window_state,
+ action == WINDOW_SNAP_LEFT ? internal::SnapSizer::LEFT_EDGE :
+ internal::SnapSizer::RIGHT_EDGE);
+ return true;
+}
+
+bool HandleWindowMinimize() {
+ content::RecordAction(
+ content::UserMetricsAction("Accel_Toggle_Minimized_Minus"));
+ return accelerators::ToggleMinimized();
+}
+
+#if defined(OS_CHROMEOS)
+bool HandleAddRemoveDisplay() {
+ content::RecordAction(UserMetricsAction("Accel_Add_Remove_Display"));
+ Shell::GetInstance()->display_manager()->AddRemoveDisplay();
+ return true;
+}
+
+bool HandleCrosh() {
+ content::RecordAction(UserMetricsAction("Accel_Open_Crosh"));
+
+ Shell::GetInstance()->new_window_delegate()->OpenCrosh();
+ return true;
+}
+
+bool HandleFileManager() {
+ content::RecordAction(UserMetricsAction("Accel_Open_File_Manager"));
+
+ Shell::GetInstance()->new_window_delegate()->OpenFileManager();
+ return true;
+}
+
+bool HandleLock(ui::KeyboardCode key_code) {
+ content::RecordAction(UserMetricsAction("Accel_LockScreen_L"));
+ Shell::GetInstance()->session_state_delegate()->LockScreen();
+ return true;
+}
+
+bool HandleCycleUser(SessionStateDelegate::CycleUser cycle_user) {
+ if (!Shell::GetInstance()->delegate()->IsMultiProfilesEnabled())
+ return false;
+ ash::SessionStateDelegate* delegate =
+ ash::Shell::GetInstance()->session_state_delegate();
+ if (delegate->NumberOfLoggedInUsers() <= 1)
+ return false;
+ MultiProfileUMA::RecordSwitchActiveUser(
+ MultiProfileUMA::SWITCH_ACTIVE_USER_BY_ACCELERATOR);
+ switch (cycle_user) {
+ case SessionStateDelegate::CYCLE_TO_NEXT_USER:
+ content::RecordAction(UserMetricsAction("Accel_Switch_To_Next_User"));
+ break;
+ case SessionStateDelegate::CYCLE_TO_PREVIOUS_USER:
+ content::RecordAction(UserMetricsAction("Accel_Switch_To_Previous_User"));
+ break;
+ }
+ delegate->CycleActiveUser(cycle_user);
+ return true;
+}
+
+bool HandleToggleMirrorMode() {
+ content::RecordAction(UserMetricsAction("Accel_Toggle_Mirror_Mode"));
+ Shell::GetInstance()->display_controller()->ToggleMirrorMode();
+ return true;
+}
+
+bool HandleToggleSpokenFeedback() {
+ content::RecordAction(UserMetricsAction("Accel_Toggle_Spoken_Feedback"));
+
+ Shell::GetInstance()->accessibility_delegate()->
+ ToggleSpokenFeedback(A11Y_NOTIFICATION_SHOW);
+ return true;
+}
+
+bool HandleTouchHudClear() {
+ internal::RootWindowController* controller =
+ internal::RootWindowController::ForTargetRootWindow();
+ if (controller->touch_hud_debug()) {
+ controller->touch_hud_debug()->Clear();
+ return true;
+ }
+ return false;
+}
+
+bool HandleTouchHudModeChange() {
+ internal::RootWindowController* controller =
+ internal::RootWindowController::ForTargetRootWindow();
+ if (controller->touch_hud_debug()) {
+ controller->touch_hud_debug()->ChangeToNextMode();
+ return true;
+ }
+ return false;
+}
+
+bool HandleTouchHudProjectToggle() {
+ content::RecordAction(UserMetricsAction("Accel_Touch_Hud_Clear"));
+ bool enabled = Shell::GetInstance()->is_touch_hud_projection_enabled();
+ Shell::GetInstance()->SetTouchHudProjectionEnabled(!enabled);
+ return true;
+}
+
+#endif // defined(OS_CHROMEOS)
+
+// Debug print methods.
+
+bool HandlePrintLayerHierarchy() {
+ aura::Window::Windows root_windows = Shell::GetAllRootWindows();
+ for (size_t i = 0; i < root_windows.size(); ++i) {
+ ui::PrintLayerHierarchy(
+ root_windows[i]->layer(),
+ root_windows[i]->GetDispatcher()->GetLastMouseLocationInRoot());
+ }
+ return true;
+}
+
+bool HandlePrintViewHierarchy() {
+ aura::Window* active_window = ash::wm::GetActiveWindow();
+ if (!active_window)
+ return true;
+ views::Widget* browser_widget =
+ views::Widget::GetWidgetForNativeWindow(active_window);
+ if (!browser_widget)
+ return true;
+ views::PrintViewHierarchy(browser_widget->GetRootView());
+ return true;
+}
+
+void PrintWindowHierarchy(aura::Window* window,
+ int indent,
+ std::ostringstream* out) {
+ std::string indent_str(indent, ' ');
+ std::string name(window->name());
+ if (name.empty())
+ name = "\"\"";
+ *out << indent_str << name << " (" << window << ")"
+ << " type=" << window->type()
+ << (wm::IsActiveWindow(window) ? " [active] " : " ")
+ << (window->IsVisible() ? " visible " : " ")
+ << window->bounds().ToString()
+ << '\n';
+
+ for (size_t i = 0; i < window->children().size(); ++i)
+ PrintWindowHierarchy(window->children()[i], indent + 3, out);
+}
+
+bool HandlePrintWindowHierarchy() {
+ Shell::RootWindowControllerList controllers =
+ Shell::GetAllRootWindowControllers();
+ for (size_t i = 0; i < controllers.size(); ++i) {
+ std::ostringstream out;
+ out << "RootWindow " << i << ":\n";
+ PrintWindowHierarchy(controllers[i]->root_window(), 0, &out);
+ // Error so logs can be collected from end-users.
+ LOG(ERROR) << out.str();
+ }
+ return true;
+}
+
+bool HandlePrintUIHierarchies() {
+ // This is a separate command so the user only has to hit one key to generate
+ // all the logs. Developers use the individual dumps repeatedly, so keep
+ // those as separate commands to avoid spamming their logs.
+ HandlePrintLayerHierarchy();
+ HandlePrintWindowHierarchy();
+ HandlePrintViewHierarchy();
+ return true;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////////////
+// AcceleratorControllerContext, public:
+
+AcceleratorControllerContext::AcceleratorControllerContext() {
+ current_accelerator_.set_type(ui::ET_UNKNOWN);
+ previous_accelerator_.set_type(ui::ET_UNKNOWN);
+}
+
+void AcceleratorControllerContext::UpdateContext(
+ const ui::Accelerator& accelerator) {
+ previous_accelerator_ = current_accelerator_;
+ current_accelerator_ = accelerator;
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// AcceleratorController, public:
+
+AcceleratorController::AcceleratorController()
+ : accelerator_manager_(new ui::AcceleratorManager) {
+ Init();
+}
+
+AcceleratorController::~AcceleratorController() {
+}
+
+void AcceleratorController::Init() {
+ for (size_t i = 0; i < kActionsAllowedAtLoginOrLockScreenLength; ++i) {
+ actions_allowed_at_login_screen_.insert(
+ kActionsAllowedAtLoginOrLockScreen[i]);
+ actions_allowed_at_lock_screen_.insert(
+ kActionsAllowedAtLoginOrLockScreen[i]);
+ }
+ for (size_t i = 0; i < kActionsAllowedAtLockScreenLength; ++i)
+ actions_allowed_at_lock_screen_.insert(kActionsAllowedAtLockScreen[i]);
+ for (size_t i = 0; i < kActionsAllowedAtModalWindowLength; ++i)
+ actions_allowed_at_modal_window_.insert(kActionsAllowedAtModalWindow[i]);
+ for (size_t i = 0; i < kReservedActionsLength; ++i)
+ reserved_actions_.insert(kReservedActions[i]);
+ for (size_t i = 0; i < kNonrepeatableActionsLength; ++i)
+ nonrepeatable_actions_.insert(kNonrepeatableActions[i]);
+ for (size_t i = 0; i < kActionsAllowedInAppModeLength; ++i)
+ actions_allowed_in_app_mode_.insert(kActionsAllowedInAppMode[i]);
+ for (size_t i = 0; i < kActionsNeedingWindowLength; ++i)
+ actions_needing_window_.insert(kActionsNeedingWindow[i]);
+
+ RegisterAccelerators(kAcceleratorData, kAcceleratorDataLength);
+
+#if !defined(NDEBUG)
+ RegisterAccelerators(kDesktopAcceleratorData, kDesktopAcceleratorDataLength);
+#endif
+
+ if (DebugShortcutsEnabled()) {
+ RegisterAccelerators(kDebugAcceleratorData, kDebugAcceleratorDataLength);
+ for (size_t i = 0; i < kReservedDebugActionsLength; ++i)
+ reserved_actions_.insert(kReservedDebugActions[i]);
+ }
+
+#if defined(OS_CHROMEOS)
+ keyboard_brightness_control_delegate_.reset(
+ new KeyboardBrightnessController());
+#endif
+}
+
+void AcceleratorController::Register(const ui::Accelerator& accelerator,
+ ui::AcceleratorTarget* target) {
+ accelerator_manager_->Register(accelerator,
+ ui::AcceleratorManager::kNormalPriority,
+ target);
+}
+
+void AcceleratorController::Unregister(const ui::Accelerator& accelerator,
+ ui::AcceleratorTarget* target) {
+ accelerator_manager_->Unregister(accelerator, target);
+}
+
+void AcceleratorController::UnregisterAll(ui::AcceleratorTarget* target) {
+ accelerator_manager_->UnregisterAll(target);
+}
+
+bool AcceleratorController::Process(const ui::Accelerator& accelerator) {
+ if (ime_control_delegate_) {
+ return accelerator_manager_->Process(
+ ime_control_delegate_->RemapAccelerator(accelerator));
+ }
+ return accelerator_manager_->Process(accelerator);
+}
+
+bool AcceleratorController::IsRegistered(
+ const ui::Accelerator& accelerator) const {
+ return accelerator_manager_->GetCurrentTarget(accelerator) != NULL;
+}
+
+bool AcceleratorController::IsReservedAccelerator(
+ const ui::Accelerator& accelerator) const {
+ const ui::Accelerator remapped_accelerator = ime_control_delegate_.get() ?
+ ime_control_delegate_->RemapAccelerator(accelerator) : accelerator;
+
+ std::map<ui::Accelerator, int>::const_iterator iter =
+ accelerators_.find(remapped_accelerator);
+ if (iter == accelerators_.end())
+ return false; // not an accelerator.
+
+ return reserved_actions_.find(iter->second) != reserved_actions_.end();
+}
+
+bool AcceleratorController::PerformAction(int action,
+ const ui::Accelerator& accelerator) {
+ ash::Shell* shell = ash::Shell::GetInstance();
+ if (!shell->session_state_delegate()->IsActiveUserSessionStarted() &&
+ actions_allowed_at_login_screen_.find(action) ==
+ actions_allowed_at_login_screen_.end()) {
+ return false;
+ }
+ if (shell->session_state_delegate()->IsScreenLocked() &&
+ actions_allowed_at_lock_screen_.find(action) ==
+ actions_allowed_at_lock_screen_.end()) {
+ return false;
+ }
+ if (shell->IsSystemModalWindowOpen() &&
+ actions_allowed_at_modal_window_.find(action) ==
+ actions_allowed_at_modal_window_.end()) {
+ // Note: we return true. This indicates the shortcut is handled
+ // and will not be passed to the modal window. This is important
+ // for things like Alt+Tab that would cause an undesired effect
+ // in the modal window by cycling through its window elements.
+ return true;
+ }
+ if (shell->delegate()->IsRunningInForcedAppMode() &&
+ actions_allowed_in_app_mode_.find(action) ==
+ actions_allowed_in_app_mode_.end()) {
+ return false;
+ }
+ if (MruWindowTracker::BuildWindowList(false).empty() &&
+ actions_needing_window_.find(action) != actions_needing_window_.end()) {
+ Shell::GetInstance()->accessibility_delegate()->TriggerAccessibilityAlert(
+ A11Y_ALERT_WINDOW_NEEDED);
+ return true;
+ }
+
+ const ui::KeyboardCode key_code = accelerator.key_code();
+ // PerformAction() is performed from gesture controllers and passes
+ // empty Accelerator() instance as the second argument. Such events
+ // should never be suspended.
+ const bool gesture_event = key_code == ui::VKEY_UNKNOWN;
+
+ // Ignore accelerators invoked as repeated (while holding a key for a long
+ // time, if their handling is nonrepeatable.
+ if (nonrepeatable_actions_.find(action) != nonrepeatable_actions_.end() &&
+ context_.repeated() && !gesture_event) {
+ return true;
+ }
+ // Type of the previous accelerator. Used by NEXT_IME and DISABLE_CAPS_LOCK.
+ const ui::EventType previous_event_type =
+ context_.previous_accelerator().type();
+ const ui::KeyboardCode previous_key_code =
+ context_.previous_accelerator().key_code();
+
+ // You *MUST* return true when some action is performed. Otherwise, this
+ // function might be called *twice*, via BrowserView::PreHandleKeyboardEvent
+ // and BrowserView::HandleKeyboardEvent, for a single accelerator press.
+ //
+ // If your accelerator invokes more than one line of code, please either
+ // implement it in your module's controller code (like TOGGLE_MIRROR_MODE
+ // below) or pull it into a HandleFoo() function above.
+ switch (action) {
+ case ACCESSIBLE_FOCUS_NEXT:
+ return HandleAccessibleFocusCycle(false);
+ case ACCESSIBLE_FOCUS_PREVIOUS:
+ return HandleAccessibleFocusCycle(true);
+ case CYCLE_BACKWARD_MRU:
+ return HandleCycleBackwardMRU(accelerator);
+ case CYCLE_FORWARD_MRU:
+ return HandleCycleForwardMRU(accelerator);
+ case CYCLE_LINEAR:
+ return HandleCycleLinear(accelerator);
+#if defined(OS_CHROMEOS)
+ case ADD_REMOVE_DISPLAY:
+ return HandleAddRemoveDisplay();
+ case TOGGLE_MIRROR_MODE:
+ return HandleToggleMirrorMode();
+ case LOCK_SCREEN:
+ return HandleLock(key_code);
+ case OPEN_FILE_MANAGER:
+ return HandleFileManager();
+ case OPEN_CROSH:
+ return HandleCrosh();
+ case SILENCE_SPOKEN_FEEDBACK:
+ HandleSilenceSpokenFeedback();
+ break;
+ case SWAP_PRIMARY_DISPLAY:
+ return HandleSwapPrimaryDisplay();
+ case SWITCH_TO_NEXT_USER:
+ return HandleCycleUser(SessionStateDelegate::CYCLE_TO_NEXT_USER);
+ case SWITCH_TO_PREVIOUS_USER:
+ return HandleCycleUser(SessionStateDelegate::CYCLE_TO_PREVIOUS_USER);
+ case TOGGLE_SPOKEN_FEEDBACK:
+ return HandleToggleSpokenFeedback();
+ case TOGGLE_WIFI:
+ Shell::GetInstance()->system_tray_notifier()->NotifyRequestToggleWifi();
+ return true;
+ case TOUCH_HUD_CLEAR:
+ return HandleTouchHudClear();
+ case TOUCH_HUD_MODE_CHANGE:
+ return HandleTouchHudModeChange();
+ case TOUCH_HUD_PROJECTION_TOGGLE:
+ return HandleTouchHudProjectToggle();
+ case DISABLE_GPU_WATCHDOG:
+ content::GpuDataManager::GetInstance()->DisableGpuWatchdog();
+ return true;
+#endif // OS_CHROMEOS
+ case OPEN_FEEDBACK_PAGE:
+ return HandleOpenFeedbackPage();
+ case EXIT:
+ // UMA metrics are recorded in the handler.
+ exit_warning_handler_.HandleAccelerator();
+ return true;
+ case NEW_INCOGNITO_WINDOW:
+ return HandleNewIncognitoWindow();
+ case NEW_TAB:
+ return HandleNewTab(key_code);
+ case NEW_WINDOW:
+ return HandleNewWindow();
+ case RESTORE_TAB:
+ return HandleRestoreTab();
+ case TAKE_SCREENSHOT:
+ return HandleTakeScreenshot(screenshot_delegate_.get());
+ case TAKE_PARTIAL_SCREENSHOT:
+ return HandleTakePartialScreenshot(screenshot_delegate_.get());
+ case TOGGLE_APP_LIST:
+ return HandleToggleAppList(
+ key_code, previous_event_type, previous_key_code, accelerator);
+ case DISABLE_CAPS_LOCK:
+ return HandleDisableCapsLock(
+ key_code, previous_event_type, previous_key_code);
+ case TOGGLE_CAPS_LOCK:
+ return HandleToggleCapsLock(
+ key_code, previous_event_type, previous_key_code);
+ case BRIGHTNESS_DOWN:
+ if (brightness_control_delegate_)
+ return brightness_control_delegate_->HandleBrightnessDown(accelerator);
+ break;
+ case BRIGHTNESS_UP:
+ if (brightness_control_delegate_)
+ return brightness_control_delegate_->HandleBrightnessUp(accelerator);
+ break;
+ case KEYBOARD_BRIGHTNESS_DOWN:
+ if (keyboard_brightness_control_delegate_)
+ return keyboard_brightness_control_delegate_->
+ HandleKeyboardBrightnessDown(accelerator);
+ break;
+ case KEYBOARD_BRIGHTNESS_UP:
+ if (keyboard_brightness_control_delegate_)
+ return keyboard_brightness_control_delegate_->
+ HandleKeyboardBrightnessUp(accelerator);
+ break;
+ case VOLUME_MUTE: {
+ ash::VolumeControlDelegate* volume_delegate =
+ shell->system_tray_delegate()->GetVolumeControlDelegate();
+ return volume_delegate && volume_delegate->HandleVolumeMute(accelerator);
+ }
+ case VOLUME_DOWN: {
+ ash::VolumeControlDelegate* volume_delegate =
+ shell->system_tray_delegate()->GetVolumeControlDelegate();
+ return volume_delegate && volume_delegate->HandleVolumeDown(accelerator);
+ }
+ case VOLUME_UP: {
+ ash::VolumeControlDelegate* volume_delegate =
+ shell->system_tray_delegate()->GetVolumeControlDelegate();
+ return volume_delegate && volume_delegate->HandleVolumeUp(accelerator);
+ }
+ case FOCUS_LAUNCHER:
+ return HandleFocusLauncher();
+ case FOCUS_NEXT_PANE:
+ return HandleRotatePaneFocus(Shell::FORWARD);
+ case FOCUS_PREVIOUS_PANE:
+ return HandleRotatePaneFocus(Shell::BACKWARD);
+ case SHOW_KEYBOARD_OVERLAY:
+ return HandleShowKeyboardOverlay();
+ case SHOW_OAK:
+ return HandleShowOak();
+ case SHOW_SYSTEM_TRAY_BUBBLE:
+ return HandleShowSystemTrayBubble();
+ case SHOW_MESSAGE_CENTER_BUBBLE:
+ HandleShowMessageCenterBubble();
+ break;
+ case SHOW_TASK_MANAGER:
+ return HandleShowTaskManager();
+ case NEXT_IME:
+ return HandleNextIme(
+ ime_control_delegate_.get(), previous_event_type, previous_key_code);
+ case PREVIOUS_IME:
+ return HandlePreviousIme(ime_control_delegate_.get(), accelerator);
+ case PRINT_UI_HIERARCHIES:
+ return HandlePrintUIHierarchies();
+ case SWITCH_IME:
+ return HandleSwitchIme(ime_control_delegate_.get(), accelerator);
+ case LAUNCH_APP_0:
+ return HandleLaunchAppN(0);
+ case LAUNCH_APP_1:
+ return HandleLaunchAppN(1);
+ case LAUNCH_APP_2:
+ return HandleLaunchAppN(2);
+ case LAUNCH_APP_3:
+ return HandleLaunchAppN(3);
+ case LAUNCH_APP_4:
+ return HandleLaunchAppN(4);
+ case LAUNCH_APP_5:
+ return HandleLaunchAppN(5);
+ case LAUNCH_APP_6:
+ return HandleLaunchAppN(6);
+ case LAUNCH_APP_7:
+ return HandleLaunchAppN(7);
+ case LAUNCH_LAST_APP:
+ return HandleLaunchLastApp();
+ case WINDOW_SNAP_LEFT:
+ case WINDOW_SNAP_RIGHT:
+ return HandleWindowSnap(action);
+ case WINDOW_MINIMIZE:
+ return HandleWindowMinimize();
+ case TOGGLE_FULLSCREEN:
+ return HandleToggleFullscreen(key_code);
+ case TOGGLE_MAXIMIZED:
+ accelerators::ToggleMaximized();
+ return true;
+ case WINDOW_POSITION_CENTER:
+ return HandlePositionCenter();
+ case SCALE_UI_UP:
+ return HandleScaleUI(true /* up */);
+ case SCALE_UI_DOWN:
+ return HandleScaleUI(false /* down */);
+ case SCALE_UI_RESET:
+ return HandleScaleReset();
+ case ROTATE_WINDOW:
+ return HandleRotateActiveWindow();
+ case ROTATE_SCREEN:
+ return HandleRotateScreen();
+ case TOGGLE_DESKTOP_BACKGROUND_MODE:
+ return debug::CycleDesktopBackgroundMode();
+ case TOGGLE_ROOT_WINDOW_FULL_SCREEN:
+ return HandleToggleRootWindowFullScreen();
+ case DEBUG_TOGGLE_DEVICE_SCALE_FACTOR:
+ Shell::GetInstance()->display_manager()->ToggleDisplayScaleFactor();
+ return true;
+ case DEBUG_TOGGLE_SHOW_DEBUG_BORDERS:
+ ash::debug::ToggleShowDebugBorders();
+ return true;
+ case DEBUG_TOGGLE_SHOW_FPS_COUNTER:
+ ash::debug::ToggleShowFpsCounter();
+ return true;
+ case DEBUG_TOGGLE_SHOW_PAINT_RECTS:
+ ash::debug::ToggleShowPaintRects();
+ return true;
+ case MAGNIFY_SCREEN_ZOOM_IN:
+ return HandleMagnifyScreen(1);
+ case MAGNIFY_SCREEN_ZOOM_OUT:
+ return HandleMagnifyScreen(-1);
+ case MEDIA_NEXT_TRACK:
+ return HandleMediaNextTrack();
+ case MEDIA_PLAY_PAUSE:
+ return HandleMediaPlayPause();
+ case MEDIA_PREV_TRACK:
+ return HandleMediaPrevTrack();
+ case POWER_PRESSED: // fallthrough
+ case POWER_RELEASED:
+#if defined(OS_CHROMEOS)
+ if (!base::SysInfo::IsRunningOnChromeOS()) {
+ // There is no powerd in linux desktop, so call the
+ // PowerButtonController here.
+ Shell::GetInstance()->power_button_controller()->
+ OnPowerButtonEvent(action == POWER_PRESSED, base::TimeTicks());
+ }
+#endif
+ // We don't do anything with these at present on the device,
+ // (power button events are reported to us from powerm via
+ // D-BUS), but we consume them to prevent them from getting
+ // passed to apps -- see http://crbug.com/146609.
+ return true;
+ case LOCK_PRESSED:
+ case LOCK_RELEASED:
+ Shell::GetInstance()->power_button_controller()->
+ OnLockButtonEvent(action == LOCK_PRESSED, base::TimeTicks());
+ return true;
+ case PRINT_LAYER_HIERARCHY:
+ return HandlePrintLayerHierarchy();
+ case PRINT_VIEW_HIERARCHY:
+ return HandlePrintViewHierarchy();
+ case PRINT_WINDOW_HIERARCHY:
+ return HandlePrintWindowHierarchy();
+ default:
+ NOTREACHED() << "Unhandled action " << action;
+ }
+ return false;
+}
+
+void AcceleratorController::SetBrightnessControlDelegate(
+ scoped_ptr<BrightnessControlDelegate> brightness_control_delegate) {
+ // Install brightness control delegate only when internal
+ // display exists.
+ if (Shell::GetInstance()->display_manager()->HasInternalDisplay() ||
+ CommandLine::ForCurrentProcess()->HasSwitch(
+ switches::kAshEnableBrightnessControl)) {
+ brightness_control_delegate_ = brightness_control_delegate.Pass();
+ }
+}
+
+void AcceleratorController::SetImeControlDelegate(
+ scoped_ptr<ImeControlDelegate> ime_control_delegate) {
+ ime_control_delegate_ = ime_control_delegate.Pass();
+}
+
+void AcceleratorController::SetScreenshotDelegate(
+ scoped_ptr<ScreenshotDelegate> screenshot_delegate) {
+ screenshot_delegate_ = screenshot_delegate.Pass();
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// AcceleratorController, ui::AcceleratorTarget implementation:
+
+bool AcceleratorController::AcceleratorPressed(
+ const ui::Accelerator& accelerator) {
+ std::map<ui::Accelerator, int>::const_iterator it =
+ accelerators_.find(accelerator);
+ DCHECK(it != accelerators_.end());
+ return PerformAction(static_cast<AcceleratorAction>(it->second), accelerator);
+}
+
+void AcceleratorController::RegisterAccelerators(
+ const AcceleratorData accelerators[],
+ size_t accelerators_length) {
+ for (size_t i = 0; i < accelerators_length; ++i) {
+ ui::Accelerator accelerator(accelerators[i].keycode,
+ accelerators[i].modifiers);
+ accelerator.set_type(accelerators[i].trigger_on_press ?
+ ui::ET_KEY_PRESSED : ui::ET_KEY_RELEASED);
+ Register(accelerator, this);
+ accelerators_.insert(
+ std::make_pair(accelerator, accelerators[i].action));
+ }
+}
+
+void AcceleratorController::SetKeyboardBrightnessControlDelegate(
+ scoped_ptr<KeyboardBrightnessControlDelegate>
+ keyboard_brightness_control_delegate) {
+ keyboard_brightness_control_delegate_ =
+ keyboard_brightness_control_delegate.Pass();
+}
+
+bool AcceleratorController::CanHandleAccelerators() const {
+ return true;
+}
+
+} // namespace ash