OLD | NEW |
| (Empty) |
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 "win8/metro_driver/ime/ime_popup_monitor.h" | |
6 | |
7 #include <windows.h> | |
8 | |
9 #include "base/logging.h" | |
10 #include "base/message_loop/message_loop.h" | |
11 #include "win8/metro_driver/ime/ime_popup_observer.h" | |
12 | |
13 namespace metro_driver { | |
14 namespace { | |
15 | |
16 ImePopupObserver* g_observer_ = NULL; | |
17 HWINEVENTHOOK g_hook_handle_ = NULL; | |
18 | |
19 void CALLBACK ImeEventCallback(HWINEVENTHOOK win_event_hook_handle, | |
20 DWORD event, | |
21 HWND window_handle, | |
22 LONG object_id, | |
23 LONG child_id, | |
24 DWORD event_thread, | |
25 DWORD event_time) { | |
26 // This function is registered to SetWinEventHook to be called back on the UI | |
27 // thread. | |
28 DCHECK(base::MessageLoopForUI::IsCurrent()); | |
29 | |
30 if (!g_observer_) | |
31 return; | |
32 switch (event) { | |
33 case EVENT_OBJECT_IME_SHOW: | |
34 g_observer_->OnImePopupChanged(ImePopupObserver::kPopupShown); | |
35 return; | |
36 case EVENT_OBJECT_IME_HIDE: | |
37 g_observer_->OnImePopupChanged(ImePopupObserver::kPopupHidden); | |
38 return; | |
39 case EVENT_OBJECT_IME_CHANGE: | |
40 g_observer_->OnImePopupChanged(ImePopupObserver::kPopupUpdated); | |
41 return; | |
42 } | |
43 } | |
44 | |
45 } // namespace | |
46 | |
47 void AddImePopupObserver(ImePopupObserver* observer) { | |
48 CHECK(g_observer_ == NULL) | |
49 << "Currently only one observer is supported at the same time."; | |
50 g_observer_ = observer; | |
51 | |
52 // IMEs running under immersive mode are supposed to generate WinEvent | |
53 // whenever their popup UI such as candidate window is shown, updated, and | |
54 // hidden to support accessibility applications. | |
55 // http://msdn.microsoft.com/en-us/library/windows/apps/hh967425.aspx#accessib
ility | |
56 // Note that there is another mechanism in TSF, called ITfUIElementSink, to | |
57 // subscribe when the visibility of an IME's UI element is changed. However, | |
58 // MS-IME running under immersive mode does not fully support this API. | |
59 // Thus, WinEvent is more reliable for this purpose. | |
60 g_hook_handle_ = SetWinEventHook( | |
61 EVENT_OBJECT_IME_SHOW, | |
62 EVENT_OBJECT_IME_CHANGE, | |
63 NULL, | |
64 ImeEventCallback, | |
65 GetCurrentProcessId(), // monitor the metro_driver process only | |
66 0, // hook all threads because MS-IME emits WinEvent in a worker thread | |
67 WINEVENT_OUTOFCONTEXT); // allows us to receive message in the UI thread | |
68 LOG_IF(ERROR, !g_hook_handle_) << "SetWinEventHook failed."; | |
69 } | |
70 | |
71 void RemoveImePopupObserver(ImePopupObserver* observer) { | |
72 if (g_observer_ != observer) | |
73 return; | |
74 g_observer_ = NULL; | |
75 if (!g_hook_handle_) | |
76 return; | |
77 const bool unhook_succeeded = !!UnhookWinEvent(g_hook_handle_); | |
78 LOG_IF(ERROR, !unhook_succeeded) << "UnhookWinEvent failed."; | |
79 g_hook_handle_ = NULL; | |
80 } | |
81 | |
82 } // namespace metro_driver | |
OLD | NEW |