Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(91)

Side by Side Diff: ui/base/ime/input_method_auralinux.cc

Issue 1068093002: Refactoring for InputMethodAuraLinux. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: addressed nona's comments. Created 5 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 // Copyright 2013 The Chromium Authors. All rights reserved. 1 // Copyright 2013 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 "ui/base/ime/input_method_auralinux.h" 5 #include "ui/base/ime/input_method_auralinux.h"
6 6
7 #include "base/environment.h" 7 #include "base/environment.h"
8 #include "ui/base/ime/linux/linux_input_method_context_factory.h" 8 #include "ui/base/ime/linux/linux_input_method_context_factory.h"
9 #include "ui/base/ime/text_input_client.h" 9 #include "ui/base/ime/text_input_client.h"
10 #include "ui/events/event.h" 10 #include "ui/events/event.h"
11 11
12 namespace ui { 12 namespace ui {
13 13
14 namespace {
15
16 class ScopedBooleanFlipper {
yukawa 2015/04/08 22:26:48 Can we use base::AutoReset in //src/base/auto_rese
Shu Chen 2015/04/09 01:16:32 Done.
17 public:
18 ScopedBooleanFlipper(bool* v) : v_(v) { DCHECK(!*v_); *v_ = true; }
19 ~ScopedBooleanFlipper() { *v_ = false; }
20 private:
21 bool* v_;
22 };
23
24 } // namespace
25
14 InputMethodAuraLinux::InputMethodAuraLinux( 26 InputMethodAuraLinux::InputMethodAuraLinux(
15 internal::InputMethodDelegate* delegate) 27 internal::InputMethodDelegate* delegate)
16 : allowed_to_fire_vkey_process_key_(false), vkey_processkey_flags_(0) { 28 : context_focused_(false), handling_key_event_(false),
29 composing_text_(false), composition_changed_(false),
30 suppress_next_result_(false) {
17 SetDelegate(delegate); 31 SetDelegate(delegate);
32 context_ = LinuxInputMethodContextFactory::instance()->
33 CreateInputMethodContext(this, false);
34 context_simple_ = LinuxInputMethodContextFactory::instance()->
35 CreateInputMethodContext(this, true);
18 } 36 }
19 37
20 InputMethodAuraLinux::~InputMethodAuraLinux() {} 38 InputMethodAuraLinux::~InputMethodAuraLinux() {}
21 39
40 LinuxInputMethodContext* InputMethodAuraLinux::GetContextForTesting(
41 bool is_simple) {
42 return is_simple ? context_simple_.get() : context_.get();
43 }
44
22 // Overriden from InputMethod. 45 // Overriden from InputMethod.
23 46
24 void InputMethodAuraLinux::Init(bool focused) { 47 void InputMethodAuraLinux::Init(bool focused) {
25 CHECK(LinuxInputMethodContextFactory::instance())
26 << "This failure was likely caused because "
27 << "ui::InitializeInputMethod(ForTesting) was not called "
28 << "before instantiating this class.";
29 input_method_context_ =
30 LinuxInputMethodContextFactory::instance()->CreateInputMethodContext(
31 this);
32 CHECK(input_method_context_.get());
33
34 InputMethodBase::Init(focused); 48 InputMethodBase::Init(focused);
35 49
36 if (focused) { 50 UpdateContextFocusState();
37 input_method_context_->OnTextInputTypeChanged(
38 GetTextInputClient() ?
39 GetTextInputClient()->GetTextInputType() :
40 TEXT_INPUT_TYPE_TEXT);
41 }
42 } 51 }
43 52
44 bool InputMethodAuraLinux::OnUntranslatedIMEMessage( 53 bool InputMethodAuraLinux::OnUntranslatedIMEMessage(
45 const base::NativeEvent& event, 54 const base::NativeEvent& event,
46 NativeEventResult* result) { 55 NativeEventResult* result) {
47 return false; 56 return false;
48 } 57 }
49 58
50 bool InputMethodAuraLinux::DispatchKeyEvent(const ui::KeyEvent& event) { 59 bool InputMethodAuraLinux::DispatchKeyEvent(const ui::KeyEvent& event) {
51 DCHECK(event.type() == ET_KEY_PRESSED || event.type() == ET_KEY_RELEASED); 60 DCHECK(event.type() == ET_KEY_PRESSED || event.type() == ET_KEY_RELEASED);
52 DCHECK(system_toplevel_window_focused()); 61 DCHECK(system_toplevel_window_focused());
62 suppress_next_result_ = false;
53 63
54 // If no text input client, do nothing. 64 // If no text input client, do nothing.
55 if (!GetTextInputClient()) 65 if (!GetTextInputClient())
56 return DispatchKeyEventPostIME(event); 66 return DispatchKeyEventPostIME(event);
57 67
58 // Let an IME handle the key event first, and allow to fire a VKEY_PROCESSKEY 68 composition_changed_ = false;
59 // event for keydown events. Note that DOM Level 3 Events Sepc requires that 69 result_text_.clear();
60 // only keydown events fire keyCode=229 events and not for keyup events.
61 if (event.type() == ET_KEY_PRESSED &&
62 (event.flags() & ui::EF_IME_FABRICATED_KEY) == 0)
63 AllowToFireProcessKey(event);
64 if (input_method_context_->DispatchKeyEvent(event))
65 return true;
66 StopFiringProcessKey();
67 70
68 // Otherwise, insert the character. 71 bool filtered = false;
69 const bool handled = DispatchKeyEventPostIME(event); 72 {
70 if (event.type() == ET_KEY_PRESSED && GetTextInputClient()) { 73 ScopedBooleanFlipper flipper(&handling_key_event_);
71 const uint16 ch = event.GetCharacter(); 74 if (context_focused_)
72 if (ch) { 75 filtered = context_->DispatchKeyEvent(event);
73 GetTextInputClient()->InsertChar(ch, event.flags()); 76 else
74 return true; 77 filtered = context_simple_->DispatchKeyEvent(event);
78 }
79
80 if (event.type() == ui::ET_KEY_PRESSED && filtered) {
81 if (NeedInsertChar()) {
82 DispatchKeyEventPostIME(event);
83 } else {
84 ui::KeyEvent key(ui::ET_KEY_PRESSED, ui::VKEY_PROCESSKEY, event.flags());
85 DispatchKeyEventPostIME(key);
75 } 86 }
76 } 87 }
77 return handled; 88
89 // If has input result process the result.
90 if (result_text_.length() || composition_changed_)
yukawa 2015/04/08 22:26:49 I slightly prefer string::empty(). How about this?
Shu Chen 2015/04/09 01:16:32 Done.
91 ProcessInputMethodResult(event, filtered);
92
93 if (event.type() == ui::ET_KEY_PRESSED && !filtered) {
94 DispatchKeyEventPostIME(event);
95
96 // If a key event was not filtered by |context_| or |context_simple_|, then
97 // it means the key event didn't generate any result text. For some cases,
98 // the key event may still generate a valid character, eg. a control-key
99 // event (ctrl-a, return, tab, etc.). We need to send the character to the
100 // focused text input client by calling TextInputClient::InsertChar().
101 base::char16 ch = event.GetCharacter();
102 TextInputClient* client = GetTextInputClient();
103 if (ch && client)
104 client->InsertChar(ch, event.flags());
105 } else if (event.type() == ui::ET_KEY_RELEASED && !filtered) {
106 DispatchKeyEventPostIME(event);
107 }
James Su 2015/04/09 08:40:02 add a comment somewhere to clarify that filtered k
108
109 return true;
110 }
111
112 void InputMethodAuraLinux::UpdateContextFocusState() {
113 bool old_context_focused = context_focused_;
114 // Use switch here in case we are going to add more text input types.
115 switch (GetTextInputType()) {
116 case ui::TEXT_INPUT_TYPE_NONE:
117 case ui::TEXT_INPUT_TYPE_PASSWORD:
Seigo Nonaka 2015/04/08 17:15:40 Probably off the topic but I just noticed the issu
yukawa 2015/04/08 22:26:49 Sounds good. But I think it's OK to try it later
Shu Chen 2015/04/09 01:16:33 I would also prefer to consider it in a separated
118 context_focused_ = false;
119 break;
120 default:
121 context_focused_ = true;
122 break;
123 }
124
125 // We only focus in |context_| when the focus is in a normal textfield.
126 if (old_context_focused && !context_focused_)
127 context_->Blur();
128 else if (!old_context_focused && context_focused_)
129 context_->Focus();
130
131 // |context_simple_| can be used in any textfield, including password box, and
132 // even if the focused text input client's text input type is
133 // ui::TEXT_INPUT_TYPE_NONE.
134 if (GetTextInputClient())
135 context_simple_->Focus();
136 else
137 context_simple_->Blur();
78 } 138 }
79 139
80 void InputMethodAuraLinux::OnTextInputTypeChanged( 140 void InputMethodAuraLinux::OnTextInputTypeChanged(
81 const TextInputClient* client) { 141 const TextInputClient* client) {
82 if (!IsTextInputClientFocused(client)) 142 DCHECK(!composing_text_);
83 return; 143 UpdateContextFocusState();
84 input_method_context_->Reset();
85 // TODO(yoichio): Support inputmode HTML attribute. 144 // TODO(yoichio): Support inputmode HTML attribute.
86 input_method_context_->OnTextInputTypeChanged(client->GetTextInputType());
87 } 145 }
88 146
89 void InputMethodAuraLinux::OnCaretBoundsChanged(const TextInputClient* client) { 147 void InputMethodAuraLinux::OnCaretBoundsChanged(const TextInputClient* client) {
90 if (!IsTextInputClientFocused(client)) 148 if (!IsTextInputClientFocused(client))
91 return; 149 return;
92 input_method_context_->OnCaretBoundsChanged( 150 context_->SetCursorLocation(GetTextInputClient()->GetCaretBounds());
93 GetTextInputClient()->GetCaretBounds());
94 } 151 }
95 152
96 void InputMethodAuraLinux::CancelComposition(const TextInputClient* client) { 153 void InputMethodAuraLinux::CancelComposition(const TextInputClient* client) {
97 if (!IsTextInputClientFocused(client)) 154 if (!IsTextInputClientFocused(client))
98 return; 155 return;
99 input_method_context_->Reset(); 156 ResetContext();
100 input_method_context_->OnTextInputTypeChanged(client->GetTextInputType()); 157 }
158
159 void InputMethodAuraLinux::ResetContext() {
160 if (!GetTextInputClient())
161 return;
162
163 DCHECK(!handling_key_event_);
Seigo Nonaka 2015/04/08 17:15:40 ditto?
Shu Chen 2015/04/09 01:16:32 Done.
164
165 // To prevent any text from being committed when resetting the |context_|;
166 handling_key_event_ = true;
167 suppress_next_result_ = true;
168
169 context_->Reset();
170 context_simple_->Reset();
171
172 // Some input methods may not honour the reset call. Focusing out/in the
173 // |context_| to make sure it gets reset correctly.
174 if (context_focused_) {
175 context_->Blur();
176 context_->Focus();
177 }
178
179 composition_.Clear();
180 result_text_.clear();
181 handling_key_event_ = false;
182 composing_text_ = false;
183 composition_changed_ = false;
101 } 184 }
102 185
103 void InputMethodAuraLinux::OnInputLocaleChanged() { 186 void InputMethodAuraLinux::OnInputLocaleChanged() {
104 } 187 }
105 188
106 std::string InputMethodAuraLinux::GetInputLocale() { 189 std::string InputMethodAuraLinux::GetInputLocale() {
107 return ""; 190 return "";
108 } 191 }
109 192
110 bool InputMethodAuraLinux::IsActive() { 193 bool InputMethodAuraLinux::IsActive() {
111 // InputMethodAuraLinux is always ready and up. 194 // InputMethodAuraLinux is always ready and up.
112 return true; 195 return true;
113 } 196 }
114 197
115 bool InputMethodAuraLinux::IsCandidatePopupOpen() const { 198 bool InputMethodAuraLinux::IsCandidatePopupOpen() const {
116 // There seems no way to detect candidate windows or any popups. 199 // There seems no way to detect candidate windows or any popups.
117 return false; 200 return false;
118 } 201 }
119 202
120 // Overriden from ui::LinuxInputMethodContextDelegate 203 // Overriden from ui::LinuxInputMethodContextDelegate
121 204
122 void InputMethodAuraLinux::OnCommit(const base::string16& text) { 205 void InputMethodAuraLinux::OnCommit(const base::string16& text) {
123 MaybeFireProcessKey(); 206 if (suppress_next_result_) {
124 if (!IsTextInputTypeNone()) 207 suppress_next_result_ = false;
208 return;
209 }
210 if (!GetTextInputClient())
211 return;
212
213 if (!composing_text_ && !text.length())
yukawa 2015/04/08 22:26:48 I slightly prefer string::empty(). How about this?
Shu Chen 2015/04/09 01:16:32 Done.
214 return;
215
216 // Append the text to the buffer, because commit signal might be fired
217 // multiple times when processing a key event.
218 result_text_.append(text);
219
220 // If we are not handling key event, do not bother sending text result if the
221 // focused text input client does not support text input.
222 if (!handling_key_event_ && !IsTextInputTypeNone()) {
223 SendFakeProcessKeyEvent(true);
125 GetTextInputClient()->InsertText(text); 224 GetTextInputClient()->InsertText(text);
225 }
226 }
227
228 void InputMethodAuraLinux::OnPreeditStart() {
229 if (suppress_next_result_ || IsTextInputTypeNone())
230 return;
231
232 composing_text_ = true;
126 } 233 }
127 234
128 void InputMethodAuraLinux::OnPreeditChanged( 235 void InputMethodAuraLinux::OnPreeditChanged(
129 const CompositionText& composition_text) { 236 const CompositionText& composition_text) {
130 MaybeFireProcessKey(); 237 if (suppress_next_result_ || IsTextInputTypeNone())
131 TextInputClient* text_input_client = GetTextInputClient(); 238 return;
132 if (text_input_client) 239
133 text_input_client->SetCompositionText(composition_text); 240 if (!composition_.text.length() && !composition_text.text.length())
yukawa 2015/04/08 22:26:48 I slightly prefer string::empty(). How about this?
Shu Chen 2015/04/09 01:16:32 Done.
241 return;
242
243 composition_ = composition_text;
244 composition_changed_ = true;
245 if (composition_.text.length())
yukawa 2015/04/08 22:26:48 I slightly prefer string::empty(). How about this?
Shu Chen 2015/04/09 01:16:32 Done.
246 composing_text_ = true;
247
248 if (!handling_key_event_ && !IsTextInputTypeNone()) {
249 SendFakeProcessKeyEvent(true);
250 GetTextInputClient()->SetCompositionText(composition_);
251 }
134 } 252 }
135 253
136 void InputMethodAuraLinux::OnPreeditEnd() { 254 void InputMethodAuraLinux::OnPreeditEnd() {
137 MaybeFireProcessKey(); 255 if (composition_.text.empty() || IsTextInputTypeNone())
138 TextInputClient* text_input_client = GetTextInputClient(); 256 return;
139 if (text_input_client && text_input_client->HasCompositionText())
140 text_input_client->ClearCompositionText();
141 }
142 257
143 void InputMethodAuraLinux::OnPreeditStart() { 258 composition_changed_ = true;
144 MaybeFireProcessKey(); 259 composition_.Clear();
260
261 if (!handling_key_event_) {
262 TextInputClient* client = GetTextInputClient();
263 if (client && client->HasCompositionText())
264 client->ClearCompositionText();
265 }
145 } 266 }
146 267
147 // Overridden from InputMethodBase. 268 // Overridden from InputMethodBase.
148 269
270 void InputMethodAuraLinux::OnFocus() {
271 InputMethodBase::OnFocus();
272 UpdateContextFocusState();
273 }
274
275 void InputMethodAuraLinux::OnBlur() {
276 ConfirmCompositionText();
277 InputMethodBase::OnBlur();
278 UpdateContextFocusState();
279 }
280
281 void InputMethodAuraLinux::OnWillChangeFocusedClient(
282 TextInputClient* focused_before,
283 TextInputClient* focused) {
284 ConfirmCompositionText();
285 }
286
149 void InputMethodAuraLinux::OnDidChangeFocusedClient( 287 void InputMethodAuraLinux::OnDidChangeFocusedClient(
150 TextInputClient* focused_before, 288 TextInputClient* focused_before,
151 TextInputClient* focused) { 289 TextInputClient* focused) {
152 input_method_context_->Reset(); 290 UpdateContextFocusState();
153 input_method_context_->OnTextInputTypeChanged( 291
154 focused ? focused->GetTextInputType() : TEXT_INPUT_TYPE_NONE); 292 // Force to update caret bounds, in case the View thinks that the caret
293 // bounds has not changed.
294 if (context_focused_)
295 OnCaretBoundsChanged(GetTextInputClient());
155 296
156 InputMethodBase::OnDidChangeFocusedClient(focused_before, focused); 297 InputMethodBase::OnDidChangeFocusedClient(focused_before, focused);
157 } 298 }
158 299
159 // Helper functions to support VKEY_PROCESSKEY. 300 // private
160 301
161 void InputMethodAuraLinux::AllowToFireProcessKey(const ui::KeyEvent& event) { 302 void InputMethodAuraLinux::ProcessInputMethodResult(const KeyEvent& key,
162 allowed_to_fire_vkey_process_key_ = true; 303 bool filtered) {
163 vkey_processkey_flags_ = event.flags(); 304 TextInputClient* client = GetTextInputClient();
305 DCHECK(client);
306
307 if (result_text_.length()) {
yukawa 2015/04/08 22:26:48 I slightly prefer string::empty(). How about this?
Shu Chen 2015/04/09 01:16:32 Done.
308 if (filtered && NeedInsertChar()) {
309 for (base::string16::const_iterator i = result_text_.begin();
yukawa 2015/04/08 22:26:48 How about using range-based for? for (const auto
Shu Chen 2015/04/09 01:16:32 Done.
310 i != result_text_.end(); ++i) {
311 client->InsertChar(*i, key.flags());
312 }
313 } else {
314 client->InsertText(result_text_);
315 composing_text_ = false;
316 }
317 }
318
319 if (composition_changed_ && !IsTextInputTypeNone()) {
320 if (composition_.text.length()) {
yukawa 2015/04/08 22:26:49 I slightly prefer string::empty(). How about this?
Shu Chen 2015/04/09 01:16:32 Done.
321 composing_text_ = true;
322 client->SetCompositionText(composition_);
323 } else if (result_text_.empty()) {
324 client->ClearCompositionText();
325 }
326 }
164 } 327 }
165 328
166 void InputMethodAuraLinux::MaybeFireProcessKey() { 329 bool InputMethodAuraLinux::NeedInsertChar() const {
167 if (!allowed_to_fire_vkey_process_key_) 330 return IsTextInputTypeNone() ||
168 return; 331 (!composing_text_ && result_text_.length() == 1);
169
170 const ui::KeyEvent fabricated_event(ET_KEY_PRESSED,
171 VKEY_PROCESSKEY,
172 vkey_processkey_flags_);
173 DispatchKeyEventPostIME(fabricated_event);
174 StopFiringProcessKey();
175 } 332 }
176 333
177 void InputMethodAuraLinux::StopFiringProcessKey() { 334 void InputMethodAuraLinux::SendFakeProcessKeyEvent(bool pressed) const {
178 allowed_to_fire_vkey_process_key_ = false; 335 KeyEvent key(pressed ? ui::ET_KEY_PRESSED : ui::ET_KEY_RELEASED,
179 vkey_processkey_flags_ = 0; 336 ui::VKEY_PROCESSKEY, 0);
337 DispatchKeyEventPostIME(key);
338 }
339
340 void InputMethodAuraLinux::ConfirmCompositionText() {
341 TextInputClient* client = GetTextInputClient();
342 if (client && client->HasCompositionText())
343 client->ConfirmCompositionText();
344
345 ResetContext();
180 } 346 }
181 347
182 } // namespace ui 348 } // namespace ui
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698