OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "chrome/browser/extensions/api/input/input.h" | 5 #include "chrome/browser/extensions/api/input/input.h" |
6 | 6 |
7 #include <string> | 7 #include <string> |
8 | 8 |
9 #include "base/lazy_instance.h" | 9 #include "base/lazy_instance.h" |
10 #include "base/string_util.h" | 10 #include "base/memory/scoped_ptr.h" |
11 #include "base/strings/string_number_conversions.h" | |
12 #include "base/values.h" | |
13 #include "chrome/browser/extensions/extension_function_registry.h" | 11 #include "chrome/browser/extensions/extension_function_registry.h" |
14 #include "chrome/browser/ui/top_level_widget.h" | |
15 #include "chrome/common/chrome_notification_types.h" | |
16 #include "content/public/browser/browser_thread.h" | 12 #include "content/public/browser/browser_thread.h" |
17 #include "ui/base/events/event.h" | 13 #include "ui/base/events/event.h" |
18 #include "ui/base/events/key_identifier_conversion.h" | |
19 | 14 |
20 #if defined(USE_ASH) && defined(USE_AURA) | 15 #if defined(USE_ASH) |
21 #include "ash/shell.h" | 16 #include "ash/shell.h" |
22 #include "ui/aura/root_window.h" | 17 #include "ui/aura/root_window.h" |
| 18 #include "ui/keyboard/keyboard_util.h" |
23 #endif | 19 #endif |
24 | 20 |
| 21 namespace { |
| 22 |
| 23 const char kNotYetImplementedError[] = |
| 24 "API is not implemented on this platform."; |
| 25 |
| 26 } // namespace |
| 27 |
25 namespace extensions { | 28 namespace extensions { |
26 | 29 |
27 namespace { | 30 bool SendKeyboardEventInputFunction::RunImpl() { |
| 31 #if defined(USE_ASH) |
| 32 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); |
28 | 33 |
29 // Keys. | 34 scoped_ptr<ui::KeyEvent> event( |
30 const char kType[] = "type"; | 35 keyboard::KeyEventFromArgs(args_.get(), &error_)); |
31 const char kKeyIdentifier[] = "keyIdentifier"; | 36 if (!event) |
32 const char kAlt[] = "altKey"; | 37 return false; |
33 const char kCtrl[] = "ctrlKey"; | |
34 const char kMeta[] = "metaKey"; | |
35 const char kShift[] = "shiftKey"; | |
36 const char kKeyDown[] = "keydown"; | |
37 const char kKeyUp[] = "keyup"; | |
38 | 38 |
39 // Errors. | 39 ash::Shell::GetActiveRootWindow()->AsRootWindowHostDelegate()->OnHostKeyEvent( |
40 const char kUnknownEventTypeError[] = "Unknown event type."; | 40 event.get()); |
41 const char kUnknownOrUnsupportedKeyIdentiferError[] = "Unknown or unsupported " | |
42 "key identifier."; | |
43 const char kUnsupportedModifier[] = "Unsupported modifier."; | |
44 const char kNoValidRecipientError[] = "No valid recipient for event."; | |
45 const char kKeyEventUnprocessedError[] = "Event was not handled."; | |
46 | 41 |
47 ui::EventType GetTypeFromString(const std::string& type) { | |
48 if (type == kKeyDown) { | |
49 return ui::ET_KEY_PRESSED; | |
50 } else if (type == kKeyUp) { | |
51 return ui::ET_KEY_RELEASED; | |
52 } | |
53 return ui::ET_UNKNOWN; | |
54 } | |
55 | |
56 // Converts a hex string "U+NNNN" to uint16. Returns 0 on error. | |
57 uint16 UnicodeIdentifierStringToInt(const std::string& key_identifier) { | |
58 int character = 0; | |
59 if ((key_identifier.length() == 6) && | |
60 (key_identifier.substr(0, 2) == "U+") && | |
61 (key_identifier.substr(2).find_first_not_of("0123456789abcdefABCDEF") == | |
62 std::string::npos)) { | |
63 const bool result = | |
64 base::HexStringToInt(key_identifier.substr(2), &character); | |
65 DCHECK(result) << key_identifier; | |
66 } | |
67 return character; | |
68 } | |
69 | |
70 } // namespace | |
71 | |
72 bool SendKeyboardEventInputFunction::RunImpl() { | |
73 DictionaryValue* args; | |
74 EXTENSION_FUNCTION_VALIDATE(args_->GetDictionary(0, &args)); | |
75 | |
76 std::string type_name; | |
77 EXTENSION_FUNCTION_VALIDATE(args->GetString(kType, &type_name)); | |
78 ui::EventType type = GetTypeFromString(type_name); | |
79 if (type == ui::ET_UNKNOWN) { | |
80 error_ = kUnknownEventTypeError; | |
81 return false; | |
82 } | |
83 | |
84 std::string identifier; | |
85 EXTENSION_FUNCTION_VALIDATE(args->GetString(kKeyIdentifier, &identifier)); | |
86 TrimWhitespaceASCII(identifier, TRIM_ALL, &identifier); | |
87 | |
88 DCHECK(content::BrowserThread::CurrentlyOn(content::BrowserThread::UI)); | |
89 const ui::KeyEvent& prototype_event = | |
90 ui::KeyEventFromKeyIdentifier(identifier); | |
91 uint16 character = 0; | |
92 if (prototype_event.key_code() == ui::VKEY_UNKNOWN) { | |
93 // Check if |identifier| is "U+NNNN" format. | |
94 character = UnicodeIdentifierStringToInt(identifier); | |
95 if (!character) { | |
96 error_ = kUnknownOrUnsupportedKeyIdentiferError; | |
97 return false; | |
98 } | |
99 } | |
100 | |
101 bool flag = false; | |
102 int flags = 0; | |
103 if (prototype_event.key_code() != ui::VKEY_UNKNOWN) | |
104 flags = prototype_event.flags(); | |
105 flags |= (args->GetBoolean(kAlt, &flag) && flag) ? ui::EF_ALT_DOWN : 0; | |
106 flags |= (args->GetBoolean(kCtrl, &flag) && flag) ? ui::EF_CONTROL_DOWN : 0; | |
107 flags |= (args->GetBoolean(kShift, &flag) && flag) ? ui::EF_SHIFT_DOWN : 0; | |
108 if (args->GetBoolean(kMeta, &flag) && flag) { | |
109 // Views does not have a Meta event flag, so return an error for now. | |
110 error_ = kUnsupportedModifier; | |
111 return false; | |
112 } | |
113 | |
114 ui::KeyEvent event(type, | |
115 prototype_event.key_code(), | |
116 flags, | |
117 prototype_event.is_char()); | |
118 if (character) { | |
119 event.set_character(character); | |
120 event.set_unmodified_character(character); | |
121 } | |
122 | |
123 #if defined(USE_ASH) && defined(USE_AURA) | |
124 ash::Shell::GetActiveRootWindow()->AsRootWindowHostDelegate()->OnHostKeyEvent( | |
125 &event); | |
126 return true; | 42 return true; |
127 #else | 43 #endif |
| 44 error_ = kNotYetImplementedError; |
128 return false; | 45 return false; |
129 #endif | |
130 } | 46 } |
131 | 47 |
132 InputAPI::InputAPI(Profile* profile) { | 48 InputAPI::InputAPI(Profile* profile) { |
133 ExtensionFunctionRegistry* registry = | 49 ExtensionFunctionRegistry* registry = |
134 ExtensionFunctionRegistry::GetInstance(); | 50 ExtensionFunctionRegistry::GetInstance(); |
135 registry->RegisterFunction<SendKeyboardEventInputFunction>(); | 51 registry->RegisterFunction<SendKeyboardEventInputFunction>(); |
136 } | 52 } |
137 | 53 |
138 InputAPI::~InputAPI() { | 54 InputAPI::~InputAPI() { |
139 } | 55 } |
140 | 56 |
141 static base::LazyInstance<ProfileKeyedAPIFactory<InputAPI> > | 57 static base::LazyInstance<ProfileKeyedAPIFactory<InputAPI> > |
142 g_factory = LAZY_INSTANCE_INITIALIZER; | 58 g_factory = LAZY_INSTANCE_INITIALIZER; |
143 | 59 |
144 // static | 60 // static |
145 ProfileKeyedAPIFactory<InputAPI>* InputAPI::GetFactoryInstance() { | 61 ProfileKeyedAPIFactory<InputAPI>* InputAPI::GetFactoryInstance() { |
146 return &g_factory.Get(); | 62 return &g_factory.Get(); |
147 } | 63 } |
148 | 64 |
149 } // namespace extensions | 65 } // namespace extensions |
OLD | NEW |