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.
5 #include "ui/base/ime/input_method_base.h"
7 #include "base/gtest_prod_util.h"
8 #include "base/memory/scoped_ptr.h"
9 #include "base/message_loop/message_loop.h"
10 #include "base/run_loop.h"
11 #include "base/scoped_observer.h"
12 #include "testing/gtest/include/gtest/gtest.h"
13 #include "ui/base/ime/dummy_text_input_client.h"
14 #include "ui/base/ime/input_method_observer.h"
15 #include "ui/base/ime/text_input_focus_manager.h"
16 #include "ui/base/ui_base_switches_util.h"
17 #include "ui/events/event.h"
22 class ClientChangeVerifier
{
24 ClientChangeVerifier()
25 : previous_client_(NULL
),
27 call_expected_(false),
28 on_will_change_focused_client_called_(false),
29 on_did_change_focused_client_called_(false),
30 on_text_input_state_changed_(false) {
33 // Expects that focused text input client will not be changed.
34 void ExpectClientDoesNotChange() {
35 previous_client_
= NULL
;
37 call_expected_
= false;
38 on_will_change_focused_client_called_
= false;
39 on_did_change_focused_client_called_
= false;
40 on_text_input_state_changed_
= false;
43 // Expects that focused text input client will be changed from
44 // |previous_client| to |next_client|.
45 void ExpectClientChange(TextInputClient
* previous_client
,
46 TextInputClient
* next_client
) {
47 previous_client_
= previous_client
;
48 next_client_
= next_client
;
49 call_expected_
= true;
50 on_will_change_focused_client_called_
= false;
51 on_did_change_focused_client_called_
= false;
52 on_text_input_state_changed_
= false;
55 // Verifies the result satisfies the expectation or not.
57 if (switches::IsTextInputFocusManagerEnabled()) {
58 EXPECT_FALSE(on_will_change_focused_client_called_
);
59 EXPECT_FALSE(on_did_change_focused_client_called_
);
60 EXPECT_FALSE(on_text_input_state_changed_
);
62 EXPECT_EQ(call_expected_
, on_will_change_focused_client_called_
);
63 EXPECT_EQ(call_expected_
, on_did_change_focused_client_called_
);
64 EXPECT_EQ(call_expected_
, on_text_input_state_changed_
);
68 void OnWillChangeFocusedClient(TextInputClient
* focused_before
,
69 TextInputClient
* focused
) {
70 EXPECT_TRUE(call_expected_
);
73 EXPECT_EQ(previous_client_
, focused_before
);
74 EXPECT_EQ(next_client_
, focused
);
77 EXPECT_FALSE(on_will_change_focused_client_called_
);
78 EXPECT_FALSE(on_did_change_focused_client_called_
);
79 EXPECT_FALSE(on_text_input_state_changed_
);
81 on_will_change_focused_client_called_
= true;
84 void OnDidChangeFocusedClient(TextInputClient
* focused_before
,
85 TextInputClient
* focused
) {
86 EXPECT_TRUE(call_expected_
);
89 EXPECT_EQ(previous_client_
, focused_before
);
90 EXPECT_EQ(next_client_
, focused
);
93 EXPECT_TRUE(on_will_change_focused_client_called_
);
94 EXPECT_FALSE(on_did_change_focused_client_called_
);
95 EXPECT_FALSE(on_text_input_state_changed_
);
97 on_did_change_focused_client_called_
= true;
100 void OnTextInputStateChanged(const TextInputClient
* client
) {
101 EXPECT_TRUE(call_expected_
);
104 EXPECT_EQ(next_client_
, client
);
107 EXPECT_TRUE(on_will_change_focused_client_called_
);
108 EXPECT_TRUE(on_did_change_focused_client_called_
);
109 EXPECT_FALSE(on_text_input_state_changed_
);
111 on_text_input_state_changed_
= true;
115 TextInputClient
* previous_client_
;
116 TextInputClient
* next_client_
;
118 bool on_will_change_focused_client_called_
;
119 bool on_did_change_focused_client_called_
;
120 bool on_text_input_state_changed_
;
122 DISALLOW_COPY_AND_ASSIGN(ClientChangeVerifier
);
125 class InputMethodBaseTest
: public testing::Test
{
127 InputMethodBaseTest() {
129 virtual ~InputMethodBaseTest() {
132 virtual void SetUp() {
133 message_loop_
.reset(new base::MessageLoopForUI
);
136 virtual void TearDown() {
137 message_loop_
.reset();
141 scoped_ptr
<base::MessageLoop
> message_loop_
;
142 DISALLOW_COPY_AND_ASSIGN(InputMethodBaseTest
);
145 class MockInputMethodBase
: public InputMethodBase
{
147 // Note: this class does not take the ownership of |verifier|.
148 MockInputMethodBase(ClientChangeVerifier
* verifier
) : verifier_(verifier
) {
150 virtual ~MockInputMethodBase() {
154 // Overriden from InputMethod.
155 virtual bool OnUntranslatedIMEMessage(
156 const base::NativeEvent
& event
,
157 InputMethod::NativeEventResult
* result
) OVERRIDE
{
160 virtual bool DispatchKeyEvent(const ui::KeyEvent
&) OVERRIDE
{
163 virtual void OnCaretBoundsChanged(const TextInputClient
* client
) OVERRIDE
{
165 virtual void CancelComposition(const TextInputClient
* client
) OVERRIDE
{
167 virtual void OnInputLocaleChanged() OVERRIDE
{
169 virtual std::string
GetInputLocale() OVERRIDE
{
172 virtual bool IsActive() OVERRIDE
{
175 virtual bool IsCandidatePopupOpen() const OVERRIDE
{
178 // Overriden from InputMethodBase.
179 virtual void OnWillChangeFocusedClient(TextInputClient
* focused_before
,
180 TextInputClient
* focused
) OVERRIDE
{
181 verifier_
->OnWillChangeFocusedClient(focused_before
, focused
);
184 virtual void OnDidChangeFocusedClient(TextInputClient
* focused_before
,
185 TextInputClient
* focused
) OVERRIDE
{
186 verifier_
->OnDidChangeFocusedClient(focused_before
, focused
);
189 ClientChangeVerifier
* verifier_
;
191 FRIEND_TEST_ALL_PREFIXES(InputMethodBaseTest
, CandidateWindowEvents
);
192 DISALLOW_COPY_AND_ASSIGN(MockInputMethodBase
);
195 class MockInputMethodObserver
: public InputMethodObserver
{
197 // Note: this class does not take the ownership of |verifier|.
198 explicit MockInputMethodObserver(ClientChangeVerifier
* verifier
)
199 : verifier_(verifier
) {
201 virtual ~MockInputMethodObserver() {
205 virtual void OnTextInputTypeChanged(const TextInputClient
* client
) OVERRIDE
{
207 virtual void OnFocus() OVERRIDE
{
209 virtual void OnBlur() OVERRIDE
{
211 virtual void OnCaretBoundsChanged(const TextInputClient
* client
) OVERRIDE
{
213 virtual void OnTextInputStateChanged(const TextInputClient
* client
) OVERRIDE
{
214 verifier_
->OnTextInputStateChanged(client
);
216 virtual void OnShowImeIfNeeded() OVERRIDE
{
218 virtual void OnInputMethodDestroyed(const InputMethod
* client
) OVERRIDE
{
221 ClientChangeVerifier
* verifier_
;
222 DISALLOW_COPY_AND_ASSIGN(MockInputMethodObserver
);
225 class MockTextInputClient
: public DummyTextInputClient
{
227 MockTextInputClient()
228 : shown_event_count_(0), updated_event_count_(0), hidden_event_count_(0) {
230 virtual ~MockTextInputClient() {
233 virtual void OnCandidateWindowShown() OVERRIDE
{
234 ++shown_event_count_
;
236 virtual void OnCandidateWindowUpdated() OVERRIDE
{
237 ++updated_event_count_
;
239 virtual void OnCandidateWindowHidden() OVERRIDE
{
240 ++hidden_event_count_
;
243 int shown_event_count() const { return shown_event_count_
; }
244 int updated_event_count() const { return updated_event_count_
; }
245 int hidden_event_count() const { return hidden_event_count_
; }
248 int shown_event_count_
;
249 int updated_event_count_
;
250 int hidden_event_count_
;
253 typedef ScopedObserver
<InputMethod
, InputMethodObserver
>
254 InputMethodScopedObserver
;
256 void SetFocusedTextInputClient(InputMethod
* input_method
,
257 TextInputClient
* text_input_client
) {
258 if (switches::IsTextInputFocusManagerEnabled()) {
259 TextInputFocusManager::GetInstance()->FocusTextInputClient(
262 input_method
->SetFocusedTextInputClient(text_input_client
);
266 TEST_F(InputMethodBaseTest
, SetFocusedTextInputClient
) {
267 DummyTextInputClient text_input_client_1st
;
268 DummyTextInputClient text_input_client_2nd
;
270 ClientChangeVerifier verifier
;
271 MockInputMethodBase
input_method(&verifier
);
272 MockInputMethodObserver
input_method_observer(&verifier
);
273 InputMethodScopedObserver
scoped_observer(&input_method_observer
);
274 scoped_observer
.Add(&input_method
);
276 // Assume that the top-level-widget gains focus.
277 input_method
.OnFocus();
280 SCOPED_TRACE("Focus from NULL to 1st TextInputClient");
282 ASSERT_EQ(NULL
, input_method
.GetTextInputClient());
283 verifier
.ExpectClientChange(NULL
, &text_input_client_1st
);
284 SetFocusedTextInputClient(&input_method
, &text_input_client_1st
);
285 EXPECT_EQ(&text_input_client_1st
, input_method
.GetTextInputClient());
290 SCOPED_TRACE("Redundant focus events must be ignored");
291 verifier
.ExpectClientDoesNotChange();
292 SetFocusedTextInputClient(&input_method
, &text_input_client_1st
);
297 SCOPED_TRACE("Focus from 1st to 2nd TextInputClient");
299 ASSERT_EQ(&text_input_client_1st
, input_method
.GetTextInputClient());
300 verifier
.ExpectClientChange(&text_input_client_1st
,
301 &text_input_client_2nd
);
302 SetFocusedTextInputClient(&input_method
, &text_input_client_2nd
);
303 EXPECT_EQ(&text_input_client_2nd
, input_method
.GetTextInputClient());
308 SCOPED_TRACE("Focus from 2nd TextInputClient to NULL");
310 ASSERT_EQ(&text_input_client_2nd
, input_method
.GetTextInputClient());
311 verifier
.ExpectClientChange(&text_input_client_2nd
, NULL
);
312 SetFocusedTextInputClient(&input_method
, NULL
);
313 EXPECT_EQ(NULL
, input_method
.GetTextInputClient());
318 SCOPED_TRACE("Redundant focus events must be ignored");
319 verifier
.ExpectClientDoesNotChange();
320 SetFocusedTextInputClient(&input_method
, NULL
);
325 TEST_F(InputMethodBaseTest
, DetachTextInputClient
) {
326 // DetachTextInputClient is not supported when IsTextInputFocusManagerEnabled.
327 if (switches::IsTextInputFocusManagerEnabled())
330 DummyTextInputClient text_input_client
;
331 DummyTextInputClient text_input_client_the_other
;
333 ClientChangeVerifier verifier
;
334 MockInputMethodBase
input_method(&verifier
);
335 MockInputMethodObserver
input_method_observer(&verifier
);
336 InputMethodScopedObserver
scoped_observer(&input_method_observer
);
337 scoped_observer
.Add(&input_method
);
339 // Assume that the top-level-widget gains focus.
340 input_method
.OnFocus();
342 // Initialize for the next test.
344 verifier
.ExpectClientChange(NULL
, &text_input_client
);
345 input_method
.SetFocusedTextInputClient(&text_input_client
);
350 SCOPED_TRACE("DetachTextInputClient must be ignored for other clients");
351 ASSERT_EQ(&text_input_client
, input_method
.GetTextInputClient());
352 verifier
.ExpectClientDoesNotChange();
353 input_method
.DetachTextInputClient(&text_input_client_the_other
);
354 EXPECT_EQ(&text_input_client
, input_method
.GetTextInputClient());
359 SCOPED_TRACE("DetachTextInputClient must succeed even after the "
360 "top-level loses the focus");
362 ASSERT_EQ(&text_input_client
, input_method
.GetTextInputClient());
363 input_method
.OnBlur();
364 input_method
.OnFocus();
365 verifier
.ExpectClientChange(&text_input_client
, NULL
);
366 input_method
.DetachTextInputClient(&text_input_client
);
367 EXPECT_EQ(NULL
, input_method
.GetTextInputClient());
372 TEST_F(InputMethodBaseTest
, CandidateWindowEvents
) {
373 MockTextInputClient text_input_client
;
376 ClientChangeVerifier verifier
;
377 MockInputMethodBase
input_method_base(&verifier
);
378 input_method_base
.OnFocus();
380 verifier
.ExpectClientChange(NULL
, &text_input_client
);
381 SetFocusedTextInputClient(&input_method_base
, &text_input_client
);
383 EXPECT_EQ(0, text_input_client
.shown_event_count());
384 EXPECT_EQ(0, text_input_client
.updated_event_count());
385 EXPECT_EQ(0, text_input_client
.hidden_event_count());
387 input_method_base
.OnCandidateWindowShown();
388 base::RunLoop().RunUntilIdle();
390 EXPECT_EQ(1, text_input_client
.shown_event_count());
391 EXPECT_EQ(0, text_input_client
.updated_event_count());
392 EXPECT_EQ(0, text_input_client
.hidden_event_count());
394 input_method_base
.OnCandidateWindowUpdated();
395 base::RunLoop().RunUntilIdle();
397 EXPECT_EQ(1, text_input_client
.shown_event_count());
398 EXPECT_EQ(1, text_input_client
.updated_event_count());
399 EXPECT_EQ(0, text_input_client
.hidden_event_count());
401 input_method_base
.OnCandidateWindowHidden();
402 base::RunLoop().RunUntilIdle();
404 EXPECT_EQ(1, text_input_client
.shown_event_count());
405 EXPECT_EQ(1, text_input_client
.updated_event_count());
406 EXPECT_EQ(1, text_input_client
.hidden_event_count());
408 input_method_base
.OnCandidateWindowShown();
411 // If InputMethod is deleted immediately after an event happens, but before
412 // its callback is invoked, the callback will be cancelled.
413 base::RunLoop().RunUntilIdle();
414 EXPECT_EQ(1, text_input_client
.shown_event_count());
415 EXPECT_EQ(1, text_input_client
.updated_event_count());
416 EXPECT_EQ(1, text_input_client
.hidden_event_count());