// Copyright (c) 2011 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 "content/test/mock_keyboard_driver_win.h" #include "base/basictypes.h" #include "base/logging.h" #include "content/test/mock_keyboard.h" namespace content { MockKeyboardDriverWin::MockKeyboardDriverWin() { // Save the keyboard layout and status of the application. // This class changes the keyboard layout and status of this application. // This change may break succeeding tests. To prevent this possible break, we // should save the layout and status here to restore when this instance is // destroyed. original_keyboard_layout_ = GetKeyboardLayout(0); active_keyboard_layout_ = original_keyboard_layout_; GetKeyboardState(&original_keyboard_states_[0]); const UINT num_keyboard_layouts = GetKeyboardLayoutList(0, NULL); DCHECK(num_keyboard_layouts > 0); orig_keyboard_layouts_list_.resize(num_keyboard_layouts); GetKeyboardLayoutList(num_keyboard_layouts, &orig_keyboard_layouts_list_[0]); memset(&keyboard_states_[0], 0, sizeof(keyboard_states_)); } MockKeyboardDriverWin::~MockKeyboardDriverWin() { // Unload the keyboard-layout driver, restore the keyboard state, and reset // the keyboard layout for succeeding tests. MaybeUnloadActiveLayout(); SetKeyboardState(&original_keyboard_states_[0]); ActivateKeyboardLayout(original_keyboard_layout_, KLF_RESET); } void MockKeyboardDriverWin::MaybeUnloadActiveLayout() { // Workaround for http://crbug.com/12093 // Only unload a keyboard layout if it was loaded by this mock driver. // Contrary to the documentation on MSDN unloading a keyboard layout // previously loaded by the system causes that layout to stop working. // We have confirmation of this behavior on XP & Vista. for (size_t i = 0; i < orig_keyboard_layouts_list_.size(); ++i) { if (orig_keyboard_layouts_list_[i] == active_keyboard_layout_) return; } // If we got here, this keyboard layout wasn't loaded by the system so it's // safe to unload it ourselve's. UnloadKeyboardLayout(active_keyboard_layout_); active_keyboard_layout_ = original_keyboard_layout_; } bool MockKeyboardDriverWin::SetLayout(int layout) { // Unload the current keyboard-layout driver and load a new keyboard-layout // driver for mapping a virtual key-code to a Unicode character. MaybeUnloadActiveLayout(); // Scan the mapping table and retrieve a Language ID for the input layout. // Load the keyboard-layout driver when we find a Language ID. // This Language IDs are copied from the registry // "HKLM\SYSTEM\CurrentControlSet\Control\Keyboard layouts". // TODO(hbono): Add more keyboard-layout drivers. static const struct { const wchar_t* language; MockKeyboard::Layout keyboard_layout; } kLanguageIDs[] = { {L"00000401", MockKeyboard::LAYOUT_ARABIC}, {L"00000402", MockKeyboard::LAYOUT_BULGARIAN}, {L"00000404", MockKeyboard::LAYOUT_CHINESE_TRADITIONAL}, {L"00000405", MockKeyboard::LAYOUT_CZECH}, {L"00000406", MockKeyboard::LAYOUT_DANISH}, {L"00000407", MockKeyboard::LAYOUT_GERMAN}, {L"00000408", MockKeyboard::LAYOUT_GREEK}, {L"00000409", MockKeyboard::LAYOUT_UNITED_STATES}, {L"0000040a", MockKeyboard::LAYOUT_SPANISH}, {L"0000040b", MockKeyboard::LAYOUT_FINNISH}, {L"0000040c", MockKeyboard::LAYOUT_FRENCH}, {L"0000040d", MockKeyboard::LAYOUT_HEBREW}, {L"0000040e", MockKeyboard::LAYOUT_HUNGARIAN}, {L"00000410", MockKeyboard::LAYOUT_ITALIAN}, {L"00000411", MockKeyboard::LAYOUT_JAPANESE}, {L"00000412", MockKeyboard::LAYOUT_KOREAN}, {L"00000415", MockKeyboard::LAYOUT_POLISH}, {L"00000416", MockKeyboard::LAYOUT_PORTUGUESE_BRAZILIAN}, {L"00000418", MockKeyboard::LAYOUT_ROMANIAN}, {L"00000419", MockKeyboard::LAYOUT_RUSSIAN}, {L"0000041a", MockKeyboard::LAYOUT_CROATIAN}, {L"0000041b", MockKeyboard::LAYOUT_SLOVAK}, {L"0000041e", MockKeyboard::LAYOUT_THAI}, {L"0000041d", MockKeyboard::LAYOUT_SWEDISH}, {L"0000041f", MockKeyboard::LAYOUT_TURKISH_Q}, {L"0000042a", MockKeyboard::LAYOUT_VIETNAMESE}, {L"00000439", MockKeyboard::LAYOUT_DEVANAGARI_INSCRIPT}, {L"00000816", MockKeyboard::LAYOUT_PORTUGUESE}, {L"00001409", MockKeyboard::LAYOUT_UNITED_STATES_DVORAK}, {L"00001009", MockKeyboard::LAYOUT_CANADIAN_FRENCH}, }; for (size_t i = 0; i < ARRAYSIZE_UNSAFE(kLanguageIDs); ++i) { if (layout == kLanguageIDs[i].keyboard_layout) { HKL new_keyboard_layout = LoadKeyboardLayout(kLanguageIDs[i].language, KLF_ACTIVATE); // loaded_keyboard_layout_ must always have a valid keyboard handle // so we only assign upon success. if (new_keyboard_layout) { active_keyboard_layout_ = new_keyboard_layout; return true; } return false; } } // Return false if there are not any matching drivers. return false; } bool MockKeyboardDriverWin::SetModifiers(int modifiers) { // Over-write the keyboard status with our modifier-key status. // WebInputEventFactory::keyboardEvent() uses GetKeyState() to retrive // modifier-key status. So, we update the modifier-key status with this // SetKeyboardState() call before creating NativeWebKeyboardEvent // instances. memset(&keyboard_states_[0], 0, sizeof(keyboard_states_)); static const struct { int key_code; int mask; } kModifierMasks[] = { {VK_SHIFT, MockKeyboard::LEFT_SHIFT | MockKeyboard::RIGHT_SHIFT}, {VK_CONTROL, MockKeyboard::LEFT_CONTROL | MockKeyboard::RIGHT_CONTROL}, {VK_MENU, MockKeyboard::LEFT_ALT | MockKeyboard::RIGHT_ALT}, {VK_LSHIFT, MockKeyboard::LEFT_SHIFT}, {VK_LCONTROL, MockKeyboard::LEFT_CONTROL}, {VK_LMENU, MockKeyboard::LEFT_ALT}, {VK_RSHIFT, MockKeyboard::RIGHT_SHIFT}, {VK_RCONTROL, MockKeyboard::RIGHT_CONTROL}, {VK_RMENU, MockKeyboard::RIGHT_ALT}, }; for (size_t i = 0; i < ARRAYSIZE_UNSAFE(kModifierMasks); ++i) { const int kKeyDownMask = 0x80; if (modifiers & kModifierMasks[i].mask) keyboard_states_[kModifierMasks[i].key_code] = kKeyDownMask; } SetKeyboardState(&keyboard_states_[0]); return true; } int MockKeyboardDriverWin::GetCharacters(int key_code, std::wstring* output) { // Retrieve Unicode characters composed from the input key-code and // the mofifiers. CHECK(output); wchar_t code[16]; int length = ToUnicodeEx(key_code, MapVirtualKey(key_code, 0), &keyboard_states_[0], &code[0], arraysize(code), 0, active_keyboard_layout_); if (length > 0) output->assign(code); return length; } } // namespace content