blob: 0694f7433df5fe487bdfb2b2ff1643674247d13d [file] [log] [blame]
// 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 "ui/base/accelerators/accelerator.h"
#include <stdint.h>
#include "base/i18n/rtl.h"
#include "base/logging.h"
#include "base/strings/string_util.h"
#include "base/strings/utf_string_conversions.h"
#include "build/build_config.h"
#include "ui/base/l10n/l10n_util.h"
#include "ui/events/event.h"
#include "ui/strings/grit/ui_strings.h"
#if defined(OS_WIN)
#include <windows.h>
#endif
#if !defined(OS_WIN) && (defined(USE_AURA) || defined(OS_MACOSX))
#include "ui/events/keycodes/keyboard_code_conversion.h"
#endif
namespace ui {
namespace {
const int kModifierMask = EF_SHIFT_DOWN | EF_CONTROL_DOWN | EF_ALT_DOWN |
EF_COMMAND_DOWN | EF_ALTGR_DOWN;
const int kInterestingFlagsMask =
kModifierMask | EF_IS_SYNTHESIZED | EF_IS_REPEAT;
} // namespace
Accelerator::Accelerator() : Accelerator(VKEY_UNKNOWN, EF_NONE) {}
Accelerator::Accelerator(KeyboardCode key_code,
int modifiers,
KeyState key_state)
: key_code_(key_code),
key_state_(key_state),
modifiers_(modifiers & kInterestingFlagsMask),
interrupted_by_mouse_event_(false) {}
Accelerator::Accelerator(const KeyEvent& key_event)
: key_code_(key_event.key_code()),
key_state_(key_event.type() == ET_KEY_PRESSED ? KeyState::PRESSED
: KeyState::RELEASED),
// |modifiers_| may include the repeat flag.
modifiers_(key_event.flags() & kInterestingFlagsMask),
interrupted_by_mouse_event_(false) {}
Accelerator::Accelerator(const Accelerator& accelerator) {
key_code_ = accelerator.key_code_;
key_state_ = accelerator.key_state_;
modifiers_ = accelerator.modifiers_;
interrupted_by_mouse_event_ = accelerator.interrupted_by_mouse_event_;
if (accelerator.platform_accelerator_)
platform_accelerator_ = accelerator.platform_accelerator_->CreateCopy();
}
Accelerator::~Accelerator() {
}
// static
int Accelerator::MaskOutKeyEventFlags(int flags) {
return flags & kModifierMask;
}
KeyEvent Accelerator::ToKeyEvent() const {
return KeyEvent(key_state() == Accelerator::KeyState::PRESSED
? ET_KEY_PRESSED
: ET_KEY_RELEASED,
key_code(), modifiers());
}
Accelerator& Accelerator::operator=(const Accelerator& accelerator) {
if (this != &accelerator) {
key_code_ = accelerator.key_code_;
key_state_ = accelerator.key_state_;
modifiers_ = accelerator.modifiers_;
interrupted_by_mouse_event_ = accelerator.interrupted_by_mouse_event_;
if (accelerator.platform_accelerator_)
platform_accelerator_ = accelerator.platform_accelerator_->CreateCopy();
else
platform_accelerator_.reset();
}
return *this;
}
bool Accelerator::operator <(const Accelerator& rhs) const {
if (key_code_ != rhs.key_code_)
return key_code_ < rhs.key_code_;
if (key_state_ != rhs.key_state_) {
return static_cast<int32_t>(key_state_) <
static_cast<int32_t>(rhs.key_state_);
}
return MaskOutKeyEventFlags(modifiers_) <
MaskOutKeyEventFlags(rhs.modifiers_);
}
bool Accelerator::operator ==(const Accelerator& rhs) const {
return (key_code_ == rhs.key_code_) && (key_state_ == rhs.key_state_) &&
(MaskOutKeyEventFlags(modifiers_) ==
MaskOutKeyEventFlags(rhs.modifiers_)) &&
interrupted_by_mouse_event_ == rhs.interrupted_by_mouse_event_;
}
bool Accelerator::operator !=(const Accelerator& rhs) const {
return !(*this == rhs);
}
bool Accelerator::IsShiftDown() const {
return (modifiers_ & EF_SHIFT_DOWN) != 0;
}
bool Accelerator::IsCtrlDown() const {
return (modifiers_ & EF_CONTROL_DOWN) != 0;
}
bool Accelerator::IsAltDown() const {
return (modifiers_ & EF_ALT_DOWN) != 0;
}
bool Accelerator::IsCmdDown() const {
return (modifiers_ & EF_COMMAND_DOWN) != 0;
}
bool Accelerator::IsRepeat() const {
return (modifiers_ & EF_IS_REPEAT) != 0;
}
base::string16 Accelerator::GetShortcutText() const {
int string_id = 0;
switch (key_code_) {
case VKEY_TAB:
string_id = IDS_APP_TAB_KEY;
break;
case VKEY_RETURN:
string_id = IDS_APP_ENTER_KEY;
break;
case VKEY_ESCAPE:
string_id = IDS_APP_ESC_KEY;
break;
case VKEY_SPACE:
string_id = IDS_APP_SPACE_KEY;
break;
case VKEY_PRIOR:
string_id = IDS_APP_PAGEUP_KEY;
break;
case VKEY_NEXT:
string_id = IDS_APP_PAGEDOWN_KEY;
break;
case VKEY_END:
string_id = IDS_APP_END_KEY;
break;
case VKEY_HOME:
string_id = IDS_APP_HOME_KEY;
break;
case VKEY_INSERT:
string_id = IDS_APP_INSERT_KEY;
break;
case VKEY_DELETE:
string_id = IDS_APP_DELETE_KEY;
break;
case VKEY_LEFT:
string_id = IDS_APP_LEFT_ARROW_KEY;
break;
case VKEY_RIGHT:
string_id = IDS_APP_RIGHT_ARROW_KEY;
break;
case VKEY_UP:
string_id = IDS_APP_UP_ARROW_KEY;
break;
case VKEY_DOWN:
string_id = IDS_APP_DOWN_ARROW_KEY;
break;
case VKEY_BACK:
string_id = IDS_APP_BACKSPACE_KEY;
break;
case VKEY_F1:
string_id = IDS_APP_F1_KEY;
break;
case VKEY_F11:
string_id = IDS_APP_F11_KEY;
break;
case VKEY_OEM_COMMA:
string_id = IDS_APP_COMMA_KEY;
break;
case VKEY_OEM_PERIOD:
string_id = IDS_APP_PERIOD_KEY;
break;
case VKEY_MEDIA_NEXT_TRACK:
string_id = IDS_APP_MEDIA_NEXT_TRACK_KEY;
break;
case VKEY_MEDIA_PLAY_PAUSE:
string_id = IDS_APP_MEDIA_PLAY_PAUSE_KEY;
break;
case VKEY_MEDIA_PREV_TRACK:
string_id = IDS_APP_MEDIA_PREV_TRACK_KEY;
break;
case VKEY_MEDIA_STOP:
string_id = IDS_APP_MEDIA_STOP_KEY;
break;
default:
break;
}
base::string16 shortcut;
if (!string_id) {
#if defined(OS_WIN)
// Our fallback is to try translate the key code to a regular character
// unless it is one of digits (VK_0 to VK_9). Some keyboard
// layouts have characters other than digits assigned in
// an unshifted mode (e.g. French AZERY layout has 'a with grave
// accent' for '0'). For display in the menu (e.g. Ctrl-0 for the
// default zoom level), we leave VK_[0-9] alone without translation.
wchar_t key;
if (base::IsAsciiDigit(key_code_))
key = static_cast<wchar_t>(key_code_);
else
key = LOWORD(::MapVirtualKeyW(key_code_, MAPVK_VK_TO_CHAR));
shortcut += key;
#elif defined(USE_AURA) || defined(OS_MACOSX)
const uint16_t c = DomCodeToUsLayoutCharacter(
UsLayoutKeyboardCodeToDomCode(key_code_), false);
if (c != 0)
shortcut +=
static_cast<base::string16::value_type>(base::ToUpperASCII(c));
#endif
} else {
shortcut = l10n_util::GetStringUTF16(string_id);
}
// Checking whether the character used for the accelerator is alphanumeric.
// If it is not, then we need to adjust the string later on if the locale is
// right-to-left. See below for more information of why such adjustment is
// required.
base::string16 shortcut_rtl;
bool adjust_shortcut_for_rtl = false;
if (base::i18n::IsRTL() && shortcut.length() == 1 &&
!base::IsAsciiAlpha(shortcut[0]) && !base::IsAsciiDigit(shortcut[0])) {
adjust_shortcut_for_rtl = true;
shortcut_rtl.assign(shortcut);
}
if (IsShiftDown())
shortcut = l10n_util::GetStringFUTF16(IDS_APP_SHIFT_MODIFIER, shortcut);
// Note that we use 'else-if' in order to avoid using Ctrl+Alt as a shortcut.
// See http://blogs.msdn.com/oldnewthing/archive/2004/03/29/101121.aspx for
// more information.
if (IsCtrlDown())
shortcut = l10n_util::GetStringFUTF16(IDS_APP_CONTROL_MODIFIER, shortcut);
else if (IsAltDown())
shortcut = l10n_util::GetStringFUTF16(IDS_APP_ALT_MODIFIER, shortcut);
if (IsCmdDown()) {
#if defined(OS_MACOSX)
shortcut = l10n_util::GetStringFUTF16(IDS_APP_COMMAND_MODIFIER, shortcut);
#elif defined(OS_CHROMEOS)
shortcut = l10n_util::GetStringFUTF16(IDS_APP_SEARCH_MODIFIER, shortcut);
#else
NOTREACHED();
#endif
}
// For some reason, menus in Windows ignore standard Unicode directionality
// marks (such as LRE, PDF, etc.). On RTL locales, we use RTL menus and
// therefore any text we draw for the menu items is drawn in an RTL context.
// Thus, the text "Ctrl++" (which we currently use for the Zoom In option)
// appears as "++Ctrl" in RTL because the Unicode BiDi algorithm puts
// punctuations on the left when the context is right-to-left. Shortcuts that
// do not end with a punctuation mark (such as "Ctrl+H" do not have this
// problem).
//
// The only way to solve this problem is to adjust the string if the locale
// is RTL so that it is drawn correctly in an RTL context. Instead of
// returning "Ctrl++" in the above example, we return "++Ctrl". This will
// cause the text to appear as "Ctrl++" when Windows draws the string in an
// RTL context because the punctuation no longer appears at the end of the
// string.
//
// TODO(idana) bug# 1232732: this hack can be avoided if instead of using
// views::Menu we use views::MenuItemView because the latter is a View
// subclass and therefore it supports marking text as RTL or LTR using
// standard Unicode directionality marks.
if (adjust_shortcut_for_rtl) {
int key_length = static_cast<int>(shortcut_rtl.length());
DCHECK_GT(key_length, 0);
shortcut_rtl.append(base::ASCIIToUTF16("+"));
// Subtracting the size of the shortcut key and 1 for the '+' sign.
shortcut_rtl.append(shortcut, 0, shortcut.length() - key_length - 1);
shortcut.swap(shortcut_rtl);
}
return shortcut;
}
} // namespace ui