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
3 // found in the LICENSE file.
4
5 #include "ui/base/ime/remote_input_method_win.h"
6
7 #include <InputScope.h>
8
9 #include <vector>
10
11 #include "base/memory/scoped_ptr.h"
12 #include "base/scoped_observer.h"
13 #include "base/strings/string16.h"
14 #include "testing/gtest/include/gtest/gtest.h"
15 #include "ui/base/ime/composition_text.h"
16 #include "ui/base/ime/dummy_text_input_client.h"
17 #include "ui/base/ime/input_method.h"
18 #include "ui/base/ime/input_method_delegate.h"
19 #include "ui/base/ime/input_method_observer.h"
20 #include "ui/base/ime/remote_input_method_delegate_win.h"
21 #include "ui/events/event.h"
22
23 namespace ui {
24 namespace {
25
26 class MockTextInputClient : public DummyTextInputClient {
27 public:
MockTextInputClient()28 MockTextInputClient()
29 : text_input_type_(TEXT_INPUT_TYPE_NONE),
30 text_input_mode_(TEXT_INPUT_MODE_DEFAULT),
31 call_count_set_composition_text_(0),
32 call_count_insert_char_(0),
33 call_count_insert_text_(0),
34 emulate_pepper_flash_(false),
35 is_candidate_window_shown_called_(false),
36 is_candidate_window_hidden_called_(false) {
37 }
38
call_count_set_composition_text() const39 size_t call_count_set_composition_text() const {
40 return call_count_set_composition_text_;
41 }
inserted_text() const42 const base::string16& inserted_text() const {
43 return inserted_text_;
44 }
call_count_insert_char() const45 size_t call_count_insert_char() const {
46 return call_count_insert_char_;
47 }
call_count_insert_text() const48 size_t call_count_insert_text() const {
49 return call_count_insert_text_;
50 }
is_candidate_window_shown_called() const51 bool is_candidate_window_shown_called() const {
52 return is_candidate_window_shown_called_;
53 }
is_candidate_window_hidden_called() const54 bool is_candidate_window_hidden_called() const {
55 return is_candidate_window_hidden_called_;
56 }
Reset()57 void Reset() {
58 text_input_type_ = TEXT_INPUT_TYPE_NONE;
59 text_input_mode_ = TEXT_INPUT_MODE_DEFAULT;
60 call_count_set_composition_text_ = 0;
61 inserted_text_.clear();
62 call_count_insert_char_ = 0;
63 call_count_insert_text_ = 0;
64 caret_bounds_ = gfx::Rect();
65 composition_character_bounds_.clear();
66 emulate_pepper_flash_ = false;
67 is_candidate_window_shown_called_ = false;
68 is_candidate_window_hidden_called_ = false;
69 }
set_text_input_type(ui::TextInputType type)70 void set_text_input_type(ui::TextInputType type) {
71 text_input_type_ = type;
72 }
set_text_input_mode(ui::TextInputMode mode)73 void set_text_input_mode(ui::TextInputMode mode) {
74 text_input_mode_ = mode;
75 }
set_caret_bounds(const gfx::Rect & caret_bounds)76 void set_caret_bounds(const gfx::Rect& caret_bounds) {
77 caret_bounds_ = caret_bounds;
78 }
set_composition_character_bounds(const std::vector<gfx::Rect> & composition_character_bounds)79 void set_composition_character_bounds(
80 const std::vector<gfx::Rect>& composition_character_bounds) {
81 composition_character_bounds_ = composition_character_bounds;
82 }
set_emulate_pepper_flash(bool enabled)83 void set_emulate_pepper_flash(bool enabled) {
84 emulate_pepper_flash_ = enabled;
85 }
86
87 private:
88 // Overriden from DummyTextInputClient.
SetCompositionText(const ui::CompositionText & composition)89 virtual void SetCompositionText(
90 const ui::CompositionText& composition) OVERRIDE {
91 ++call_count_set_composition_text_;
92 }
InsertChar(base::char16 ch,int flags)93 virtual void InsertChar(base::char16 ch, int flags) OVERRIDE {
94 inserted_text_.append(1, ch);
95 ++call_count_insert_char_;
96 }
InsertText(const base::string16 & text)97 virtual void InsertText(const base::string16& text) OVERRIDE {
98 inserted_text_.append(text);
99 ++call_count_insert_text_;
100 }
GetTextInputType() const101 virtual ui::TextInputType GetTextInputType() const OVERRIDE {
102 return text_input_type_;
103 }
GetTextInputMode() const104 virtual ui::TextInputMode GetTextInputMode() const OVERRIDE {
105 return text_input_mode_;
106 }
GetCaretBounds() const107 virtual gfx::Rect GetCaretBounds() const {
108 return caret_bounds_;
109 }
GetCompositionCharacterBounds(uint32 index,gfx::Rect * rect) const110 virtual bool GetCompositionCharacterBounds(uint32 index,
111 gfx::Rect* rect) const OVERRIDE {
112 // Emulate the situation of crbug.com/328237.
113 if (emulate_pepper_flash_)
114 return false;
115 if (!rect || composition_character_bounds_.size() <= index)
116 return false;
117 *rect = composition_character_bounds_[index];
118 return true;
119 }
HasCompositionText() const120 virtual bool HasCompositionText() const OVERRIDE {
121 return !composition_character_bounds_.empty();
122 }
GetCompositionTextRange(gfx::Range * range) const123 virtual bool GetCompositionTextRange(gfx::Range* range) const OVERRIDE {
124 if (composition_character_bounds_.empty())
125 return false;
126 *range = gfx::Range(0, composition_character_bounds_.size());
127 return true;
128 }
OnCandidateWindowShown()129 virtual void OnCandidateWindowShown() OVERRIDE {
130 is_candidate_window_shown_called_ = true;
131 }
OnCandidateWindowHidden()132 virtual void OnCandidateWindowHidden() OVERRIDE {
133 is_candidate_window_hidden_called_ = true;
134 }
135
136 ui::TextInputType text_input_type_;
137 ui::TextInputMode text_input_mode_;
138 gfx::Rect caret_bounds_;
139 std::vector<gfx::Rect> composition_character_bounds_;
140 base::string16 inserted_text_;
141 size_t call_count_set_composition_text_;
142 size_t call_count_insert_char_;
143 size_t call_count_insert_text_;
144 bool emulate_pepper_flash_;
145 bool is_candidate_window_shown_called_;
146 bool is_candidate_window_hidden_called_;
147 DISALLOW_COPY_AND_ASSIGN(MockTextInputClient);
148 };
149
150 class MockInputMethodDelegate : public internal::InputMethodDelegate {
151 public:
MockInputMethodDelegate()152 MockInputMethodDelegate() {}
153
fabricated_key_events() const154 const std::vector<ui::KeyboardCode>& fabricated_key_events() const {
155 return fabricated_key_events_;
156 }
Reset()157 void Reset() {
158 fabricated_key_events_.clear();
159 }
160
161 private:
DispatchKeyEventPostIME(const ui::KeyEvent & event)162 virtual bool DispatchKeyEventPostIME(const ui::KeyEvent& event) OVERRIDE {
163 EXPECT_FALSE(event.HasNativeEvent());
164 fabricated_key_events_.push_back(event.key_code());
165 return true;
166 }
167
168 std::vector<ui::KeyboardCode> fabricated_key_events_;
169 DISALLOW_COPY_AND_ASSIGN(MockInputMethodDelegate);
170 };
171
172 class MockRemoteInputMethodDelegateWin
173 : public internal::RemoteInputMethodDelegateWin {
174 public:
MockRemoteInputMethodDelegateWin()175 MockRemoteInputMethodDelegateWin()
176 : cancel_composition_called_(false),
177 text_input_client_updated_called_(false) {
178 }
179
cancel_composition_called() const180 bool cancel_composition_called() const {
181 return cancel_composition_called_;
182 }
text_input_client_updated_called() const183 bool text_input_client_updated_called() const {
184 return text_input_client_updated_called_;
185 }
input_scopes() const186 const std::vector<int32>& input_scopes() const {
187 return input_scopes_;
188 }
composition_character_bounds() const189 const std::vector<gfx::Rect>& composition_character_bounds() const {
190 return composition_character_bounds_;
191 }
Reset()192 void Reset() {
193 cancel_composition_called_ = false;
194 text_input_client_updated_called_ = false;
195 input_scopes_.clear();
196 composition_character_bounds_.clear();
197 }
198
199 private:
CancelComposition()200 virtual void CancelComposition() OVERRIDE {
201 cancel_composition_called_ = true;
202 }
203
OnTextInputClientUpdated(const std::vector<int32> & input_scopes,const std::vector<gfx::Rect> & composition_character_bounds)204 virtual void OnTextInputClientUpdated(
205 const std::vector<int32>& input_scopes,
206 const std::vector<gfx::Rect>& composition_character_bounds) OVERRIDE {
207 text_input_client_updated_called_ = true;
208 input_scopes_ = input_scopes;
209 composition_character_bounds_ = composition_character_bounds;
210 }
211
212 bool cancel_composition_called_;
213 bool text_input_client_updated_called_;
214 std::vector<int32> input_scopes_;
215 std::vector<gfx::Rect> composition_character_bounds_;
216 DISALLOW_COPY_AND_ASSIGN(MockRemoteInputMethodDelegateWin);
217 };
218
219 class MockInputMethodObserver : public InputMethodObserver {
220 public:
MockInputMethodObserver()221 MockInputMethodObserver()
222 : on_text_input_state_changed_(0),
223 on_input_method_destroyed_changed_(0) {
224 }
~MockInputMethodObserver()225 virtual ~MockInputMethodObserver() {
226 }
Reset()227 void Reset() {
228 on_text_input_state_changed_ = 0;
229 on_input_method_destroyed_changed_ = 0;
230 }
on_text_input_state_changed() const231 size_t on_text_input_state_changed() const {
232 return on_text_input_state_changed_;
233 }
on_input_method_destroyed_changed() const234 size_t on_input_method_destroyed_changed() const {
235 return on_input_method_destroyed_changed_;
236 }
237
238 private:
239 // Overriden from InputMethodObserver.
OnTextInputTypeChanged(const TextInputClient * client)240 virtual void OnTextInputTypeChanged(const TextInputClient* client) OVERRIDE {
241 }
OnFocus()242 virtual void OnFocus() OVERRIDE {
243 }
OnBlur()244 virtual void OnBlur() OVERRIDE {
245 }
OnCaretBoundsChanged(const TextInputClient * client)246 virtual void OnCaretBoundsChanged(const TextInputClient* client) OVERRIDE {
247 }
OnTextInputStateChanged(const TextInputClient * client)248 virtual void OnTextInputStateChanged(const TextInputClient* client) OVERRIDE {
249 ++on_text_input_state_changed_;
250 }
OnInputMethodDestroyed(const InputMethod * client)251 virtual void OnInputMethodDestroyed(const InputMethod* client) OVERRIDE {
252 ++on_input_method_destroyed_changed_;
253 }
OnShowImeIfNeeded()254 virtual void OnShowImeIfNeeded() {
255 }
256
257 size_t on_text_input_state_changed_;
258 size_t on_input_method_destroyed_changed_;
259 DISALLOW_COPY_AND_ASSIGN(MockInputMethodObserver);
260 };
261
262 typedef ScopedObserver<InputMethod, InputMethodObserver>
263 InputMethodScopedObserver;
264
TEST(RemoteInputMethodWinTest,RemoteInputMethodPrivateWin)265 TEST(RemoteInputMethodWinTest, RemoteInputMethodPrivateWin) {
266 InputMethod* other_ptr = static_cast<InputMethod*>(NULL) + 1;
267
268 // Use typed NULL to make EXPECT_NE happy until nullptr becomes available.
269 RemoteInputMethodPrivateWin* kNull =
270 static_cast<RemoteInputMethodPrivateWin*>(NULL);
271 EXPECT_EQ(kNull, RemoteInputMethodPrivateWin::Get(other_ptr));
272
273 MockInputMethodDelegate delegate_;
274 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
275 EXPECT_NE(kNull, RemoteInputMethodPrivateWin::Get(input_method.get()));
276
277 InputMethod* dangling_ptr = input_method.get();
278 input_method.reset(NULL);
279 EXPECT_EQ(kNull, RemoteInputMethodPrivateWin::Get(dangling_ptr));
280 }
281
TEST(RemoteInputMethodWinTest,OnInputSourceChanged)282 TEST(RemoteInputMethodWinTest, OnInputSourceChanged) {
283 MockInputMethodDelegate delegate_;
284 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
285 RemoteInputMethodPrivateWin* private_ptr =
286 RemoteInputMethodPrivateWin::Get(input_method.get());
287 ASSERT_TRUE(private_ptr != NULL);
288
289 private_ptr->OnInputSourceChanged(
290 MAKELANGID(LANG_JAPANESE, SUBLANG_JAPANESE_JAPAN), true);
291 EXPECT_EQ("ja-JP", input_method->GetInputLocale());
292
293 private_ptr->OnInputSourceChanged(
294 MAKELANGID(LANG_ARABIC, SUBLANG_ARABIC_QATAR), true);
295 EXPECT_EQ("ar-QA", input_method->GetInputLocale());
296 }
297
TEST(RemoteInputMethodWinTest,OnCandidatePopupChanged)298 TEST(RemoteInputMethodWinTest, OnCandidatePopupChanged) {
299 MockInputMethodDelegate delegate_;
300 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
301 RemoteInputMethodPrivateWin* private_ptr =
302 RemoteInputMethodPrivateWin::Get(input_method.get());
303 ASSERT_TRUE(private_ptr != NULL);
304
305 // Initial value
306 EXPECT_FALSE(input_method->IsCandidatePopupOpen());
307
308 // RemoteInputMethodWin::OnCandidatePopupChanged can be called even when the
309 // focused text input client is NULL.
310 ASSERT_TRUE(input_method->GetTextInputClient() == NULL);
311 private_ptr->OnCandidatePopupChanged(false);
312 private_ptr->OnCandidatePopupChanged(true);
313
314 MockTextInputClient mock_text_input_client;
315 input_method->SetFocusedTextInputClient(&mock_text_input_client);
316
317 ASSERT_FALSE(mock_text_input_client.is_candidate_window_shown_called());
318 ASSERT_FALSE(mock_text_input_client.is_candidate_window_hidden_called());
319 mock_text_input_client.Reset();
320
321 private_ptr->OnCandidatePopupChanged(true);
322 EXPECT_TRUE(input_method->IsCandidatePopupOpen());
323 EXPECT_TRUE(mock_text_input_client.is_candidate_window_shown_called());
324 EXPECT_FALSE(mock_text_input_client.is_candidate_window_hidden_called());
325
326 private_ptr->OnCandidatePopupChanged(false);
327 EXPECT_FALSE(input_method->IsCandidatePopupOpen());
328 EXPECT_TRUE(mock_text_input_client.is_candidate_window_shown_called());
329 EXPECT_TRUE(mock_text_input_client.is_candidate_window_hidden_called());
330 }
331
TEST(RemoteInputMethodWinTest,CancelComposition)332 TEST(RemoteInputMethodWinTest, CancelComposition) {
333 MockInputMethodDelegate delegate_;
334 MockTextInputClient mock_text_input_client;
335 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
336
337 // This must not cause a crash.
338 input_method->CancelComposition(&mock_text_input_client);
339
340 RemoteInputMethodPrivateWin* private_ptr =
341 RemoteInputMethodPrivateWin::Get(input_method.get());
342 ASSERT_TRUE(private_ptr != NULL);
343 MockRemoteInputMethodDelegateWin mock_remote_delegate;
344 private_ptr->SetRemoteDelegate(&mock_remote_delegate);
345
346 input_method->CancelComposition(&mock_text_input_client);
347 EXPECT_FALSE(mock_remote_delegate.cancel_composition_called());
348
349 input_method->SetFocusedTextInputClient(&mock_text_input_client);
350 input_method->CancelComposition(&mock_text_input_client);
351 EXPECT_TRUE(mock_remote_delegate.cancel_composition_called());
352 }
353
TEST(RemoteInputMethodWinTest,SetFocusedTextInputClient)354 TEST(RemoteInputMethodWinTest, SetFocusedTextInputClient) {
355 MockInputMethodDelegate delegate_;
356 MockTextInputClient mock_text_input_client;
357 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
358
359 mock_text_input_client.set_caret_bounds(gfx::Rect(10, 0, 10, 20));
360 mock_text_input_client.set_text_input_type(ui::TEXT_INPUT_TYPE_URL);
361 input_method->SetFocusedTextInputClient(&mock_text_input_client);
362
363 RemoteInputMethodPrivateWin* private_ptr =
364 RemoteInputMethodPrivateWin::Get(input_method.get());
365 ASSERT_TRUE(private_ptr != NULL);
366 MockRemoteInputMethodDelegateWin mock_remote_delegate;
367 private_ptr->SetRemoteDelegate(&mock_remote_delegate);
368
369 // Initial state must be synced.
370 EXPECT_TRUE(mock_remote_delegate.text_input_client_updated_called());
371 ASSERT_EQ(1, mock_remote_delegate.composition_character_bounds().size());
372 EXPECT_EQ(gfx::Rect(10, 0, 10, 20),
373 mock_remote_delegate.composition_character_bounds()[0]);
374 ASSERT_EQ(1, mock_remote_delegate.input_scopes().size());
375 EXPECT_EQ(IS_URL, mock_remote_delegate.input_scopes()[0]);
376
377 // State must be cleared by SetFocusedTextInputClient(NULL).
378 mock_remote_delegate.Reset();
379 input_method->SetFocusedTextInputClient(NULL);
380 EXPECT_TRUE(mock_remote_delegate.text_input_client_updated_called());
381 EXPECT_TRUE(mock_remote_delegate.composition_character_bounds().empty());
382 EXPECT_TRUE(mock_remote_delegate.input_scopes().empty());
383 }
384
TEST(RemoteInputMethodWinTest,DetachTextInputClient)385 TEST(RemoteInputMethodWinTest, DetachTextInputClient) {
386 MockInputMethodDelegate delegate_;
387 MockTextInputClient mock_text_input_client;
388 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
389
390 mock_text_input_client.set_caret_bounds(gfx::Rect(10, 0, 10, 20));
391 mock_text_input_client.set_text_input_type(ui::TEXT_INPUT_TYPE_URL);
392 input_method->SetFocusedTextInputClient(&mock_text_input_client);
393
394 RemoteInputMethodPrivateWin* private_ptr =
395 RemoteInputMethodPrivateWin::Get(input_method.get());
396 ASSERT_TRUE(private_ptr != NULL);
397 MockRemoteInputMethodDelegateWin mock_remote_delegate;
398 private_ptr->SetRemoteDelegate(&mock_remote_delegate);
399
400 // Initial state must be synced.
401 EXPECT_TRUE(mock_remote_delegate.text_input_client_updated_called());
402 ASSERT_EQ(1, mock_remote_delegate.composition_character_bounds().size());
403 EXPECT_EQ(gfx::Rect(10, 0, 10, 20),
404 mock_remote_delegate.composition_character_bounds()[0]);
405 ASSERT_EQ(1, mock_remote_delegate.input_scopes().size());
406 EXPECT_EQ(IS_URL, mock_remote_delegate.input_scopes()[0]);
407
408 // State must be cleared by DetachTextInputClient
409 mock_remote_delegate.Reset();
410 input_method->DetachTextInputClient(&mock_text_input_client);
411 EXPECT_TRUE(mock_remote_delegate.text_input_client_updated_called());
412 EXPECT_TRUE(mock_remote_delegate.composition_character_bounds().empty());
413 EXPECT_TRUE(mock_remote_delegate.input_scopes().empty());
414 }
415
TEST(RemoteInputMethodWinTest,OnCaretBoundsChanged)416 TEST(RemoteInputMethodWinTest, OnCaretBoundsChanged) {
417 MockInputMethodDelegate delegate_;
418 MockTextInputClient mock_text_input_client;
419 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
420
421 // This must not cause a crash.
422 input_method->OnCaretBoundsChanged(&mock_text_input_client);
423
424 mock_text_input_client.set_caret_bounds(gfx::Rect(10, 0, 10, 20));
425 input_method->SetFocusedTextInputClient(&mock_text_input_client);
426
427 RemoteInputMethodPrivateWin* private_ptr =
428 RemoteInputMethodPrivateWin::Get(input_method.get());
429 ASSERT_TRUE(private_ptr != NULL);
430 MockRemoteInputMethodDelegateWin mock_remote_delegate;
431 private_ptr->SetRemoteDelegate(&mock_remote_delegate);
432
433 // Initial state must be synced.
434 EXPECT_TRUE(mock_remote_delegate.text_input_client_updated_called());
435 ASSERT_EQ(1, mock_remote_delegate.composition_character_bounds().size());
436 EXPECT_EQ(gfx::Rect(10, 0, 10, 20),
437 mock_remote_delegate.composition_character_bounds()[0]);
438
439 // Redundant OnCaretBoundsChanged must be ignored.
440 mock_remote_delegate.Reset();
441 input_method->OnCaretBoundsChanged(&mock_text_input_client);
442 EXPECT_FALSE(mock_remote_delegate.text_input_client_updated_called());
443
444 // Check OnCaretBoundsChanged is handled. (w/o composition)
445 mock_remote_delegate.Reset();
446 mock_text_input_client.Reset();
447 mock_text_input_client.set_caret_bounds(gfx::Rect(10, 20, 30, 40));
448 input_method->OnCaretBoundsChanged(&mock_text_input_client);
449 EXPECT_TRUE(mock_remote_delegate.text_input_client_updated_called());
450 ASSERT_EQ(1, mock_remote_delegate.composition_character_bounds().size());
451 EXPECT_EQ(gfx::Rect(10, 20, 30, 40),
452 mock_remote_delegate.composition_character_bounds()[0]);
453
454 // Check OnCaretBoundsChanged is handled. (w/ composition)
455 {
456 mock_remote_delegate.Reset();
457 mock_text_input_client.Reset();
458
459 std::vector<gfx::Rect> bounds;
460 bounds.push_back(gfx::Rect(10, 20, 30, 40));
461 bounds.push_back(gfx::Rect(40, 30, 20, 10));
462 mock_text_input_client.set_composition_character_bounds(bounds);
463 input_method->OnCaretBoundsChanged(&mock_text_input_client);
464 EXPECT_TRUE(mock_remote_delegate.text_input_client_updated_called());
465 EXPECT_EQ(bounds, mock_remote_delegate.composition_character_bounds());
466 }
467 }
468
469 // Test case against crbug.com/328237.
TEST(RemoteInputMethodWinTest,OnCaretBoundsChangedForPepperFlash)470 TEST(RemoteInputMethodWinTest, OnCaretBoundsChangedForPepperFlash) {
471 MockInputMethodDelegate delegate_;
472 MockTextInputClient mock_text_input_client;
473 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
474 input_method->SetFocusedTextInputClient(&mock_text_input_client);
475
476 RemoteInputMethodPrivateWin* private_ptr =
477 RemoteInputMethodPrivateWin::Get(input_method.get());
478 ASSERT_TRUE(private_ptr != NULL);
479 MockRemoteInputMethodDelegateWin mock_remote_delegate;
480 private_ptr->SetRemoteDelegate(&mock_remote_delegate);
481
482 mock_remote_delegate.Reset();
483 mock_text_input_client.Reset();
484 mock_text_input_client.set_emulate_pepper_flash(true);
485
486 std::vector<gfx::Rect> caret_bounds;
487 caret_bounds.push_back(gfx::Rect(5, 15, 25, 35));
488 mock_text_input_client.set_caret_bounds(caret_bounds[0]);
489
490 std::vector<gfx::Rect> composition_bounds;
491 composition_bounds.push_back(gfx::Rect(10, 20, 30, 40));
492 composition_bounds.push_back(gfx::Rect(40, 30, 20, 10));
493 mock_text_input_client.set_composition_character_bounds(composition_bounds);
494 input_method->OnCaretBoundsChanged(&mock_text_input_client);
495 EXPECT_TRUE(mock_remote_delegate.text_input_client_updated_called());
496 // The caret bounds must be used when
497 // TextInputClient::GetCompositionCharacterBounds failed.
498 EXPECT_EQ(caret_bounds, mock_remote_delegate.composition_character_bounds());
499 }
500
TEST(RemoteInputMethodWinTest,OnTextInputTypeChanged)501 TEST(RemoteInputMethodWinTest, OnTextInputTypeChanged) {
502 MockInputMethodDelegate delegate_;
503 MockTextInputClient mock_text_input_client;
504 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
505
506 // This must not cause a crash.
507 input_method->OnCaretBoundsChanged(&mock_text_input_client);
508
509 mock_text_input_client.set_text_input_type(ui::TEXT_INPUT_TYPE_URL);
510 input_method->SetFocusedTextInputClient(&mock_text_input_client);
511
512 RemoteInputMethodPrivateWin* private_ptr =
513 RemoteInputMethodPrivateWin::Get(input_method.get());
514 ASSERT_TRUE(private_ptr != NULL);
515 MockRemoteInputMethodDelegateWin mock_remote_delegate;
516 private_ptr->SetRemoteDelegate(&mock_remote_delegate);
517
518 // Initial state must be synced.
519 EXPECT_TRUE(mock_remote_delegate.text_input_client_updated_called());
520 ASSERT_EQ(1, mock_remote_delegate.input_scopes().size());
521 EXPECT_EQ(IS_URL, mock_remote_delegate.input_scopes()[0]);
522
523 // Check TEXT_INPUT_TYPE_NONE is handled.
524 mock_remote_delegate.Reset();
525 mock_text_input_client.Reset();
526 mock_text_input_client.set_text_input_type(ui::TEXT_INPUT_TYPE_NONE);
527 mock_text_input_client.set_text_input_mode(ui::TEXT_INPUT_MODE_KATAKANA);
528 input_method->OnTextInputTypeChanged(&mock_text_input_client);
529 EXPECT_TRUE(mock_remote_delegate.text_input_client_updated_called());
530 EXPECT_TRUE(mock_remote_delegate.input_scopes().empty());
531
532 // Redundant OnTextInputTypeChanged must be ignored.
533 mock_remote_delegate.Reset();
534 input_method->OnTextInputTypeChanged(&mock_text_input_client);
535 EXPECT_FALSE(mock_remote_delegate.text_input_client_updated_called());
536
537 mock_remote_delegate.Reset();
538 mock_text_input_client.Reset();
539 mock_text_input_client.set_caret_bounds(gfx::Rect(10, 20, 30, 40));
540 input_method->OnCaretBoundsChanged(&mock_text_input_client);
541 }
542
TEST(RemoteInputMethodWinTest,DispatchKeyEvent_NativeKeyEvent)543 TEST(RemoteInputMethodWinTest, DispatchKeyEvent_NativeKeyEvent) {
544 // Basically RemoteInputMethodWin does not handle native keydown event.
545
546 MockInputMethodDelegate delegate_;
547 MockTextInputClient mock_text_input_client;
548 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
549
550 const MSG wm_keydown = { NULL, WM_KEYDOWN, ui::VKEY_A };
551 ui::KeyEvent native_keydown(wm_keydown);
552
553 // This must not cause a crash.
554 EXPECT_FALSE(input_method->DispatchKeyEvent(native_keydown));
555 EXPECT_TRUE(mock_text_input_client.inserted_text().empty());
556 EXPECT_TRUE(delegate_.fabricated_key_events().empty());
557 delegate_.Reset();
558 mock_text_input_client.Reset();
559
560 RemoteInputMethodPrivateWin* private_ptr =
561 RemoteInputMethodPrivateWin::Get(input_method.get());
562 ASSERT_TRUE(private_ptr != NULL);
563 MockRemoteInputMethodDelegateWin mock_remote_delegate;
564 private_ptr->SetRemoteDelegate(&mock_remote_delegate);
565
566 // TextInputClient is not focused yet here.
567
568 EXPECT_FALSE(input_method->DispatchKeyEvent(native_keydown));
569 EXPECT_TRUE(mock_text_input_client.inserted_text().empty());
570 EXPECT_TRUE(delegate_.fabricated_key_events().empty());
571 delegate_.Reset();
572 mock_text_input_client.Reset();
573
574 input_method->SetFocusedTextInputClient(&mock_text_input_client);
575
576 // TextInputClient is now focused here.
577
578 EXPECT_FALSE(input_method->DispatchKeyEvent(native_keydown));
579 EXPECT_TRUE(mock_text_input_client.inserted_text().empty());
580 EXPECT_TRUE(delegate_.fabricated_key_events().empty());
581 delegate_.Reset();
582 mock_text_input_client.Reset();
583 }
584
TEST(RemoteInputMethodWinTest,DispatchKeyEvent_NativeCharEvent)585 TEST(RemoteInputMethodWinTest, DispatchKeyEvent_NativeCharEvent) {
586 // RemoteInputMethodWin handles native char event if possible.
587
588 MockInputMethodDelegate delegate_;
589 MockTextInputClient mock_text_input_client;
590 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
591
592 const MSG wm_char = { NULL, WM_CHAR, 'A', 0 };
593 ui::KeyEvent native_char(wm_char);
594
595 // This must not cause a crash.
596 EXPECT_FALSE(input_method->DispatchKeyEvent(native_char));
597 EXPECT_TRUE(mock_text_input_client.inserted_text().empty());
598 EXPECT_TRUE(delegate_.fabricated_key_events().empty());
599 delegate_.Reset();
600 mock_text_input_client.Reset();
601
602 RemoteInputMethodPrivateWin* private_ptr =
603 RemoteInputMethodPrivateWin::Get(input_method.get());
604 ASSERT_TRUE(private_ptr != NULL);
605 MockRemoteInputMethodDelegateWin mock_remote_delegate;
606 private_ptr->SetRemoteDelegate(&mock_remote_delegate);
607
608 // TextInputClient is not focused yet here.
609
610 EXPECT_FALSE(input_method->DispatchKeyEvent(native_char));
611 EXPECT_TRUE(mock_text_input_client.inserted_text().empty());
612 EXPECT_TRUE(delegate_.fabricated_key_events().empty());
613 delegate_.Reset();
614 mock_text_input_client.Reset();
615
616 input_method->SetFocusedTextInputClient(&mock_text_input_client);
617
618 // TextInputClient is now focused here.
619
620 EXPECT_TRUE(input_method->DispatchKeyEvent(native_char));
621 EXPECT_EQ(L"A", mock_text_input_client.inserted_text());
622 EXPECT_TRUE(delegate_.fabricated_key_events().empty());
623 delegate_.Reset();
624 mock_text_input_client.Reset();
625 }
626
TEST(RemoteInputMethodWinTest,DispatchKeyEvent_FabricatedKeyDown)627 TEST(RemoteInputMethodWinTest, DispatchKeyEvent_FabricatedKeyDown) {
628 // Fabricated non-char event will be delegated to
629 // InputMethodDelegate::DispatchFabricatedKeyEventPostIME as long as the
630 // delegate is installed.
631
632 MockInputMethodDelegate delegate_;
633 MockTextInputClient mock_text_input_client;
634 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
635
636 ui::KeyEvent fabricated_keydown(ui::ET_KEY_PRESSED, ui::VKEY_A, ui::EF_NONE);
637 fabricated_keydown.set_character(L'A');
638
639 // This must not cause a crash.
640 EXPECT_TRUE(input_method->DispatchKeyEvent(fabricated_keydown));
641 EXPECT_TRUE(mock_text_input_client.inserted_text().empty());
642 ASSERT_EQ(1, delegate_.fabricated_key_events().size());
643 EXPECT_EQ(L'A', delegate_.fabricated_key_events()[0]);
644 delegate_.Reset();
645 mock_text_input_client.Reset();
646
647 RemoteInputMethodPrivateWin* private_ptr =
648 RemoteInputMethodPrivateWin::Get(input_method.get());
649 ASSERT_TRUE(private_ptr != NULL);
650 MockRemoteInputMethodDelegateWin mock_remote_delegate;
651 private_ptr->SetRemoteDelegate(&mock_remote_delegate);
652
653 // TextInputClient is not focused yet here.
654
655 EXPECT_TRUE(input_method->DispatchKeyEvent(fabricated_keydown));
656 EXPECT_TRUE(mock_text_input_client.inserted_text().empty());
657 ASSERT_EQ(1, delegate_.fabricated_key_events().size());
658 EXPECT_EQ(L'A', delegate_.fabricated_key_events()[0]);
659 delegate_.Reset();
660 mock_text_input_client.Reset();
661
662 input_method->SetFocusedTextInputClient(&mock_text_input_client);
663 // TextInputClient is now focused here.
664
665 EXPECT_TRUE(input_method->DispatchKeyEvent(fabricated_keydown));
666 EXPECT_TRUE(mock_text_input_client.inserted_text().empty());
667 ASSERT_EQ(1, delegate_.fabricated_key_events().size());
668 EXPECT_EQ(L'A', delegate_.fabricated_key_events()[0]);
669 delegate_.Reset();
670 mock_text_input_client.Reset();
671
672 input_method->SetDelegate(NULL);
673 // RemoteInputMethodDelegateWin is no longer set here.
674
675 EXPECT_FALSE(input_method->DispatchKeyEvent(fabricated_keydown));
676 EXPECT_TRUE(mock_text_input_client.inserted_text().empty());
677 }
678
TEST(RemoteInputMethodWinTest,DispatchKeyEvent_FabricatedChar)679 TEST(RemoteInputMethodWinTest, DispatchKeyEvent_FabricatedChar) {
680 // Note: RemoteInputMethodWin::DispatchKeyEvent should always return true
681 // for fabricated character events.
682
683 MockInputMethodDelegate delegate_;
684 MockTextInputClient mock_text_input_client;
685 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
686
687 ui::KeyEvent fabricated_char(L'A', ui::VKEY_A, ui::EF_NONE);
688
689 // This must not cause a crash.
690 EXPECT_TRUE(input_method->DispatchKeyEvent(fabricated_char));
691 EXPECT_TRUE(mock_text_input_client.inserted_text().empty());
692 EXPECT_TRUE(delegate_.fabricated_key_events().empty());
693 delegate_.Reset();
694 mock_text_input_client.Reset();
695
696 RemoteInputMethodPrivateWin* private_ptr =
697 RemoteInputMethodPrivateWin::Get(input_method.get());
698 ASSERT_TRUE(private_ptr != NULL);
699 MockRemoteInputMethodDelegateWin mock_remote_delegate;
700 private_ptr->SetRemoteDelegate(&mock_remote_delegate);
701
702 // TextInputClient is not focused yet here.
703
704 EXPECT_TRUE(input_method->DispatchKeyEvent(fabricated_char));
705 EXPECT_TRUE(mock_text_input_client.inserted_text().empty());
706 EXPECT_TRUE(delegate_.fabricated_key_events().empty());
707 delegate_.Reset();
708 mock_text_input_client.Reset();
709
710 input_method->SetFocusedTextInputClient(&mock_text_input_client);
711
712 // TextInputClient is now focused here.
713
714 EXPECT_TRUE(input_method->DispatchKeyEvent(fabricated_char));
715 EXPECT_EQ(L"A", mock_text_input_client.inserted_text());
716 EXPECT_TRUE(delegate_.fabricated_key_events().empty());
717 delegate_.Reset();
718 mock_text_input_client.Reset();
719 }
720
TEST(RemoteInputMethodWinTest,OnCompositionChanged)721 TEST(RemoteInputMethodWinTest, OnCompositionChanged) {
722 MockInputMethodDelegate delegate_;
723 MockTextInputClient mock_text_input_client;
724 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
725
726 RemoteInputMethodPrivateWin* private_ptr =
727 RemoteInputMethodPrivateWin::Get(input_method.get());
728 ASSERT_TRUE(private_ptr != NULL);
729 MockRemoteInputMethodDelegateWin mock_remote_delegate;
730 private_ptr->SetRemoteDelegate(&mock_remote_delegate);
731
732 CompositionText composition_text;
733
734 // TextInputClient is not focused yet here.
735
736 private_ptr->OnCompositionChanged(composition_text);
737 EXPECT_EQ(0, mock_text_input_client.call_count_set_composition_text());
738 delegate_.Reset();
739 mock_text_input_client.Reset();
740
741 input_method->SetFocusedTextInputClient(&mock_text_input_client);
742
743 // TextInputClient is now focused here.
744
745 private_ptr->OnCompositionChanged(composition_text);
746 EXPECT_EQ(1, mock_text_input_client.call_count_set_composition_text());
747 delegate_.Reset();
748 mock_text_input_client.Reset();
749 }
750
TEST(RemoteInputMethodWinTest,OnTextCommitted)751 TEST(RemoteInputMethodWinTest, OnTextCommitted) {
752 MockInputMethodDelegate delegate_;
753 MockTextInputClient mock_text_input_client;
754 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
755
756 RemoteInputMethodPrivateWin* private_ptr =
757 RemoteInputMethodPrivateWin::Get(input_method.get());
758 ASSERT_TRUE(private_ptr != NULL);
759 MockRemoteInputMethodDelegateWin mock_remote_delegate;
760 private_ptr->SetRemoteDelegate(&mock_remote_delegate);
761
762 base::string16 committed_text = L"Hello";
763
764 // TextInputClient is not focused yet here.
765
766 mock_text_input_client.set_text_input_type(TEXT_INPUT_TYPE_TEXT);
767 private_ptr->OnTextCommitted(committed_text);
768 EXPECT_EQ(0, mock_text_input_client.call_count_insert_char());
769 EXPECT_EQ(0, mock_text_input_client.call_count_insert_text());
770 EXPECT_EQ(L"", mock_text_input_client.inserted_text());
771 delegate_.Reset();
772 mock_text_input_client.Reset();
773
774 input_method->SetFocusedTextInputClient(&mock_text_input_client);
775
776 // TextInputClient is now focused here.
777
778 mock_text_input_client.set_text_input_type(TEXT_INPUT_TYPE_TEXT);
779 private_ptr->OnTextCommitted(committed_text);
780 EXPECT_EQ(0, mock_text_input_client.call_count_insert_char());
781 EXPECT_EQ(1, mock_text_input_client.call_count_insert_text());
782 EXPECT_EQ(committed_text, mock_text_input_client.inserted_text());
783 delegate_.Reset();
784 mock_text_input_client.Reset();
785
786 // When TextInputType is TEXT_INPUT_TYPE_NONE, TextInputClient::InsertText
787 // should not be used.
788 mock_text_input_client.set_text_input_type(TEXT_INPUT_TYPE_NONE);
789 private_ptr->OnTextCommitted(committed_text);
790 EXPECT_EQ(committed_text.size(),
791 mock_text_input_client.call_count_insert_char());
792 EXPECT_EQ(0, mock_text_input_client.call_count_insert_text());
793 EXPECT_EQ(committed_text, mock_text_input_client.inserted_text());
794 delegate_.Reset();
795 mock_text_input_client.Reset();
796 }
797
TEST(RemoteInputMethodWinTest,OnTextInputStateChanged_Observer)798 TEST(RemoteInputMethodWinTest, OnTextInputStateChanged_Observer) {
799 DummyTextInputClient text_input_client;
800 DummyTextInputClient text_input_client_the_other;
801
802 MockInputMethodObserver input_method_observer;
803 MockInputMethodDelegate delegate_;
804 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
805 InputMethodScopedObserver scoped_observer(&input_method_observer);
806 scoped_observer.Add(input_method.get());
807
808 input_method->SetFocusedTextInputClient(&text_input_client);
809 ASSERT_EQ(&text_input_client, input_method->GetTextInputClient());
810 EXPECT_EQ(1u, input_method_observer.on_text_input_state_changed());
811 input_method_observer.Reset();
812
813 input_method->SetFocusedTextInputClient(&text_input_client);
814 ASSERT_EQ(&text_input_client, input_method->GetTextInputClient());
815 EXPECT_EQ(0u, input_method_observer.on_text_input_state_changed());
816 input_method_observer.Reset();
817
818 input_method->SetFocusedTextInputClient(&text_input_client_the_other);
819 ASSERT_EQ(&text_input_client_the_other, input_method->GetTextInputClient());
820 EXPECT_EQ(1u, input_method_observer.on_text_input_state_changed());
821 input_method_observer.Reset();
822
823 input_method->DetachTextInputClient(&text_input_client_the_other);
824 ASSERT_TRUE(input_method->GetTextInputClient() == NULL);
825 EXPECT_EQ(1u, input_method_observer.on_text_input_state_changed());
826 input_method_observer.Reset();
827 }
828
TEST(RemoteInputMethodWinTest,OnInputMethodDestroyed_Observer)829 TEST(RemoteInputMethodWinTest, OnInputMethodDestroyed_Observer) {
830 DummyTextInputClient text_input_client;
831 DummyTextInputClient text_input_client_the_other;
832
833 MockInputMethodObserver input_method_observer;
834 InputMethodScopedObserver scoped_observer(&input_method_observer);
835
836 MockInputMethodDelegate delegate_;
837 scoped_ptr<InputMethod> input_method(CreateRemoteInputMethodWin(&delegate_));
838 input_method->AddObserver(&input_method_observer);
839
840 EXPECT_EQ(0u, input_method_observer.on_input_method_destroyed_changed());
841 input_method.reset();
842 EXPECT_EQ(1u, input_method_observer.on_input_method_destroyed_changed());
843 }
844
845 } // namespace
846 } // namespace ui
847