global_keyboard_shortcuts_mac.mm revision 201ade2fbba22bfb27ae029f4d23fca6ded109a0
1// Copyright (c) 2010 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 <AppKit/NSEvent.h>
6#include <Carbon/Carbon.h>
7
8#include "chrome/browser/global_keyboard_shortcuts_mac.h"
9
10#include "base/basictypes.h"
11#include "base/logging.h"
12#include "chrome/app/chrome_command_ids.h"
13
14// Basically, there are two kinds of keyboard shortcuts: Ones that should work
15// only if the tab contents is focused (BrowserKeyboardShortcut), and ones that
16// should work in all other cases (WindowKeyboardShortcut). In the latter case,
17// we differentiate between shortcuts that are checked before any other view
18// gets the chance to handle them (WindowKeyboardShortcut) or after all views
19// had a chance but did not handle the keypress event
20// (DelayedWindowKeyboardShortcut).
21
22const KeyboardShortcutData* GetWindowKeyboardShortcutTable(
23    size_t* num_entries) {
24  static const KeyboardShortcutData keyboard_shortcuts[] = {
25    //cmd   shift  cntrl  option
26    //---   -----  -----  ------
27    // '{' / '}' characters should be matched earlier than virtual key code
28    // (therefore we can match alt-8 as '{' on german keyboards).
29    {true,  false, false, false, 0,             '}', IDC_SELECT_NEXT_TAB},
30    {true,  false, false, false, 0,             '{', IDC_SELECT_PREVIOUS_TAB},
31    {false, false, true,  false, kVK_PageDown,  0,   IDC_SELECT_NEXT_TAB},
32    {false, false, true,  false, kVK_Tab,       0,   IDC_SELECT_NEXT_TAB},
33    {false, false, true,  false, kVK_PageUp,    0,   IDC_SELECT_PREVIOUS_TAB},
34    {false, true,  true,  false, kVK_Tab,       0,   IDC_SELECT_PREVIOUS_TAB},
35    // Cmd-0..8 select the Nth tab, with cmd-9 being "last tab".
36    {true,  false, false, false, kVK_ANSI_1,          0, IDC_SELECT_TAB_0},
37    {true,  false, false, false, kVK_ANSI_Keypad1,    0, IDC_SELECT_TAB_0},
38    {true,  false, false, false, kVK_ANSI_2,          0, IDC_SELECT_TAB_1},
39    {true,  false, false, false, kVK_ANSI_Keypad2,    0, IDC_SELECT_TAB_1},
40    {true,  false, false, false, kVK_ANSI_3,          0, IDC_SELECT_TAB_2},
41    {true,  false, false, false, kVK_ANSI_Keypad3,    0, IDC_SELECT_TAB_2},
42    {true,  false, false, false, kVK_ANSI_4,          0, IDC_SELECT_TAB_3},
43    {true,  false, false, false, kVK_ANSI_Keypad4,    0, IDC_SELECT_TAB_3},
44    {true,  false, false, false, kVK_ANSI_5,          0, IDC_SELECT_TAB_4},
45    {true,  false, false, false, kVK_ANSI_Keypad5,    0, IDC_SELECT_TAB_4},
46    {true,  false, false, false, kVK_ANSI_6,          0, IDC_SELECT_TAB_5},
47    {true,  false, false, false, kVK_ANSI_Keypad6,    0, IDC_SELECT_TAB_5},
48    {true,  false, false, false, kVK_ANSI_7,          0, IDC_SELECT_TAB_6},
49    {true,  false, false, false, kVK_ANSI_Keypad7,    0, IDC_SELECT_TAB_6},
50    {true,  false, false, false, kVK_ANSI_8,          0, IDC_SELECT_TAB_7},
51    {true,  false, false, false, kVK_ANSI_Keypad8,    0, IDC_SELECT_TAB_7},
52    {true,  false, false, false, kVK_ANSI_9,          0, IDC_SELECT_LAST_TAB},
53    {true,  false, false, false, kVK_ANSI_Keypad9,    0, IDC_SELECT_LAST_TAB},
54  };
55
56  *num_entries = arraysize(keyboard_shortcuts);
57
58  return keyboard_shortcuts;
59}
60
61const KeyboardShortcutData* GetDelayedWindowKeyboardShortcutTable(
62    size_t* num_entries) {
63  static const KeyboardShortcutData keyboard_shortcuts[] = {
64    //cmd   shift  cntrl  option
65    //---   -----  -----  ------
66    {false, false, false, false, kVK_Escape,        0, IDC_STOP},
67  };
68
69  *num_entries = arraysize(keyboard_shortcuts);
70
71  return keyboard_shortcuts;
72}
73
74const KeyboardShortcutData* GetBrowserKeyboardShortcutTable(
75    size_t* num_entries) {
76  static const KeyboardShortcutData keyboard_shortcuts[] = {
77    //cmd   shift  cntrl  option
78    //---   -----  -----  ------
79    {true,  false, false, false, kVK_LeftArrow,    0,   IDC_BACK},
80    {true,  false, false, false, kVK_RightArrow,   0,   IDC_FORWARD},
81    {false, false, false, false, kVK_Delete,       0,   IDC_BACK},
82    {false, true,  false, false, kVK_Delete,       0,   IDC_FORWARD},
83    {true,  true,  false, false, 0,                'c', IDC_DEV_TOOLS_INSPECT},
84  };
85
86  *num_entries = arraysize(keyboard_shortcuts);
87
88  return keyboard_shortcuts;
89}
90
91static bool MatchesEventForKeyboardShortcut(
92    const KeyboardShortcutData& shortcut,
93    bool command_key, bool shift_key, bool cntrl_key, bool opt_key,
94    int vkey_code, unichar key_char) {
95  // Expects that one of |key_char| or |vkey_code| is 0.
96  DCHECK((shortcut.key_char == 0) ^ (shortcut.vkey_code == 0));
97  if (shortcut.key_char) {
98    // The given shortcut key is to be matched by a keyboard character.
99    // In this case we ignore shift and opt (alt) key modifiers, because
100    // the character may be generated by a combination with those keys.
101    if (shortcut.command_key == command_key &&
102        shortcut.cntrl_key == cntrl_key &&
103        shortcut.key_char == key_char)
104      return true;
105  } else if (shortcut.vkey_code) {
106    // The given shortcut key is to be matched by a virtual key code.
107    if (shortcut.command_key == command_key &&
108        shortcut.shift_key == shift_key &&
109        shortcut.cntrl_key == cntrl_key &&
110        shortcut.opt_key == opt_key &&
111        shortcut.vkey_code == vkey_code)
112      return true;
113  } else {
114    NOTREACHED();  // Shouldn't happen.
115  }
116  return false;
117}
118
119static int CommandForKeyboardShortcut(
120    const KeyboardShortcutData* (*get_keyboard_shortcut_table)(size_t*),
121    bool command_key, bool shift_key, bool cntrl_key, bool opt_key,
122    int vkey_code, unichar key_char) {
123
124  // Scan through keycodes and see if it corresponds to one of the global
125  // shortcuts on file.
126  //
127  // TODO(jeremy): Change this into a hash table once we get enough
128  // entries in the array to make a difference.
129  // (When turning this into a hash table, note that the current behavior
130  // relies on the order of the table (see the comment for '{' / '}' above).
131  size_t num_shortcuts = 0;
132  const KeyboardShortcutData *it = get_keyboard_shortcut_table(&num_shortcuts);
133  for (size_t i = 0; i < num_shortcuts; ++i, ++it) {
134    if (MatchesEventForKeyboardShortcut(*it, command_key, shift_key, cntrl_key,
135                                        opt_key, vkey_code, key_char))
136      return it->chrome_command;
137  }
138
139  return -1;
140}
141
142int CommandForWindowKeyboardShortcut(
143    bool command_key, bool shift_key, bool cntrl_key, bool opt_key,
144    int vkey_code, unichar key_char) {
145  return CommandForKeyboardShortcut(GetWindowKeyboardShortcutTable,
146                                    command_key, shift_key,
147                                    cntrl_key, opt_key, vkey_code,
148                                    key_char);
149}
150
151int CommandForDelayedWindowKeyboardShortcut(
152    bool command_key, bool shift_key, bool cntrl_key, bool opt_key,
153    int vkey_code, unichar key_char) {
154  return CommandForKeyboardShortcut(GetDelayedWindowKeyboardShortcutTable,
155                                    command_key, shift_key,
156                                    cntrl_key, opt_key, vkey_code,
157                                    key_char);
158}
159
160int CommandForBrowserKeyboardShortcut(
161    bool command_key, bool shift_key, bool cntrl_key, bool opt_key,
162    int vkey_code, unichar key_char) {
163  return CommandForKeyboardShortcut(GetBrowserKeyboardShortcutTable,
164                                    command_key, shift_key,
165                                    cntrl_key, opt_key, vkey_code,
166                                    key_char);
167}
168
169unichar KeyCharacterForEvent(NSEvent* event) {
170  NSString* eventString = [event charactersIgnoringModifiers];
171  NSString* characters = [event characters];
172
173  // Character pairs that undergo BiDi mirrored.
174  // There are actually many more such pairs, but these are the ones that
175  // are likely to show up in keyboard shortcuts.
176  const struct {
177    unichar a;
178    unichar b;
179  } kMirroredBiDiChars[] = {
180    {'{', '}'},
181    {'[', ']'},
182    {'(', ')'},
183  };
184
185  if ([eventString length] != 1)
186    return 0;
187
188  if ([characters length] != 1)
189    return [eventString characterAtIndex:0];
190
191  unichar noModifiersChar = [eventString characterAtIndex:0];
192  unichar rawChar = [characters characterAtIndex:0];
193  // When both |characters| and |charactersIgnoringModifiers| are ascii,
194  // return the first character of |characters|, if...
195  if (isascii(noModifiersChar) && isascii(rawChar)) {
196    // |characters| is an alphabet (mainly for dvorak-qwerty layout), or
197    if (isalpha(rawChar))
198      return rawChar;
199
200    // http://crbug.com/42517
201    // In RTL keyboard layouts, Cocoa mirrors characters in the string
202    // returned by [event charactersIgnoringModifiers].  In this case, return
203    // the raw (unmirrored) char.
204    // FIXME: If there is a need to add any more characters to the
205    // kMirroredBiDiChars table, then it's probably better to use ICU's
206    // u_charMirror() function to perform this test.
207    for (size_t i = 0; i < ARRAYSIZE_UNSAFE(kMirroredBiDiChars); ++i) {
208      const unichar& a = kMirroredBiDiChars[i].a;
209      const unichar& b = kMirroredBiDiChars[i].b;
210      if ((rawChar == a && noModifiersChar == b) ||
211          (rawChar == b && noModifiersChar == a))
212          return rawChar;
213    }
214
215    // opt/alt modifier is set (e.g. on german layout we want '{' for opt-8).
216    if ([event modifierFlags] & NSAlternateKeyMask)
217      return [characters characterAtIndex:0];
218  }
219
220  return [eventString characterAtIndex:0];
221}
222