xref: /reactos/win32ss/user/user32/misc/imm.c (revision 426598c6)
1 /*
2  * COPYRIGHT:       See COPYING in the top level directory
3  * PROJECT:         ReactOS user32.dll
4  * FILE:            win32ss/user/user32/misc/imm.c
5  * PURPOSE:         User32.dll Imm functions
6  * PROGRAMMERS:     Dmitry Chapyshev (dmitry@reactos.org)
7  *                  Katayama Hirofumi MZ (katayama.hirofumi.mz@gmail.com)
8  */
9 
10 #include <user32.h>
11 #include <strsafe.h>
12 #include <ddk/immdev.h>
13 
14 WINE_DEFAULT_DEBUG_CHANNEL(user32);
15 
16 #define IMM_INIT_MAGIC 0x19650412
17 #define MAX_CANDIDATEFORM 4
18 
19 /* Is != NULL when we have loaded the IMM ourselves */
20 HINSTANCE ghImm32 = NULL; // Win: ghImm32
21 
22 BOOL gbImmInitializing = FALSE; // Win: bImmInitializing
23 
24 INT gfConIme = -1; // Win: gfConIme
25 
26 HWND FASTCALL IntGetTopLevelWindow(HWND hWnd)
27 {
28     DWORD style;
29 
30     for (; hWnd; hWnd = GetParent(hWnd))
31     {
32         style = (DWORD)GetWindowLongPtrW(hWnd, GWL_STYLE);
33         if (!(style & WS_CHILD))
34             break;
35     }
36 
37     return hWnd;
38 }
39 
40 /* define stub functions */
41 #undef DEFINE_IMM_ENTRY
42 #define DEFINE_IMM_ENTRY(type, name, params, retval, retkind) \
43     static type WINAPI IMMSTUB_##name params { IMM_RETURN_##retkind((type)retval); }
44 #include "immtable.h"
45 
46 // Win: gImmApiEntries
47 Imm32ApiTable gImmApiEntries = {
48 /* initialize by stubs */
49 #undef DEFINE_IMM_ENTRY
50 #define DEFINE_IMM_ENTRY(type, name, params, retval, retkind) \
51     IMMSTUB_##name,
52 #include "immtable.h"
53 };
54 
55 // Win: GetImmFileName
56 HRESULT
57 User32GetImmFileName(_Out_ LPWSTR lpBuffer, _In_ size_t cchBuffer)
58 {
59     UINT length = GetSystemDirectoryW(lpBuffer, cchBuffer);
60     if (length && length < cchBuffer)
61     {
62         StringCchCatW(lpBuffer, cchBuffer, L"\\");
63         return StringCchCatW(lpBuffer, cchBuffer, L"imm32.dll");
64     }
65     return StringCchCopyW(lpBuffer, cchBuffer, L"imm32.dll");
66 }
67 
68 // @unimplemented
69 // Win: _InitializeImmEntryTable
70 static BOOL IntInitializeImmEntryTable(VOID)
71 {
72     WCHAR ImmFile[MAX_PATH];
73     HMODULE imm32 = ghImm32;
74 
75     /* Check whether the IMM table has already been initialized */
76     if (IMM_FN(ImmWINNLSEnableIME) != IMMSTUB_ImmWINNLSEnableIME)
77         return TRUE;
78 
79     User32GetImmFileName(ImmFile, _countof(ImmFile));
80     TRACE("File %S\n", ImmFile);
81 
82     /* If IMM32 is already loaded, use it without increasing reference count. */
83     if (imm32 == NULL)
84         imm32 = GetModuleHandleW(ImmFile);
85 
86     /*
87      * Loading imm32.dll will call imm32!DllMain function.
88      * imm32!DllMain calls User32InitializeImmEntryTable.
89      * Thus, if imm32.dll was loaded, the table has been loaded.
90      */
91     if (imm32 == NULL)
92     {
93         imm32 = ghImm32 = LoadLibraryW(ImmFile);
94         if (imm32 == NULL)
95         {
96             ERR("Did not load imm32.dll!\n");
97             return FALSE;
98         }
99         return TRUE;
100     }
101 
102 /* load imm procedures */
103 #undef DEFINE_IMM_ENTRY
104 #define DEFINE_IMM_ENTRY(type, name, params, retval, retkind) \
105     do { \
106         FN_##name proc = (FN_##name)GetProcAddress(imm32, #name); \
107         if (!proc) { \
108             ERR("Could not load %s\n", #name); \
109             return FALSE; \
110         } \
111         IMM_FN(name) = proc; \
112     } while (0);
113 #include "immtable.h"
114 
115     return TRUE;
116 }
117 
118 // Win: InitializeImmEntryTable
119 BOOL WINAPI InitializeImmEntryTable(VOID)
120 {
121     gbImmInitializing = TRUE;
122     return IntInitializeImmEntryTable();
123 }
124 
125 // Win: User32InitializeImmEntryTable
126 BOOL WINAPI User32InitializeImmEntryTable(DWORD magic)
127 {
128     TRACE("Imm (%x)\n", magic);
129 
130     if (magic != IMM_INIT_MAGIC)
131         return FALSE;
132 
133     /* Check whether the IMM table has already been initialized */
134     if (IMM_FN(ImmWINNLSEnableIME) != IMMSTUB_ImmWINNLSEnableIME)
135         return TRUE;
136 
137     IntInitializeImmEntryTable();
138 
139     if (ghImm32 == NULL && !gbImmInitializing)
140     {
141         WCHAR ImmFile[MAX_PATH];
142         User32GetImmFileName(ImmFile, _countof(ImmFile));
143         ghImm32 = LoadLibraryW(ImmFile);
144         if (ghImm32 == NULL)
145         {
146             ERR("Did not load imm32.dll!\n");
147             return FALSE;
148         }
149     }
150 
151     return IMM_FN(ImmRegisterClient)(&gSharedInfo, ghImm32);
152 }
153 
154 // Win: ImeIsUsableContext
155 static BOOL User32CanSetImeWindowToImc(HIMC hIMC, HWND hImeWnd)
156 {
157     PIMC pIMC = ValidateHandle(hIMC, TYPE_INPUTCONTEXT);
158     return pIMC && (!pIMC->hImeWnd || pIMC->hImeWnd == hImeWnd || !ValidateHwnd(pIMC->hImeWnd));
159 }
160 
161 // Win: GetIMEShowStatus
162 static BOOL User32GetImeShowStatus(VOID)
163 {
164     return (BOOL)NtUserCallNoParam(NOPARAM_ROUTINE_GETIMESHOWSTATUS);
165 }
166 
167 /* Sends a message to the IME UI window. */
168 /* Win: SendMessageToUI(pimeui, uMsg, wParam, lParam, !unicode) */
169 static LRESULT
170 User32SendImeUIMessage(PIMEUI pimeui, UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL unicode)
171 {
172     LRESULT ret = 0;
173     HWND hwndUI = pimeui->hwndUI;
174     PWND pwnd, pwndUI;
175 
176     ASSERT(pimeui->spwnd != NULL);
177 
178     pwnd = pimeui->spwnd;
179     pwndUI = ValidateHwnd(hwndUI);
180     if (!pwnd || (pwnd->state & WNDS_DESTROYED) || (pwnd->state2 & WNDS2_INDESTROY) ||
181         !pwndUI || (pwndUI->state & WNDS_DESTROYED) || (pwndUI->state2 & WNDS2_INDESTROY))
182     {
183         return 0;
184     }
185 
186     InterlockedIncrement(&pimeui->nCntInIMEProc);
187 
188     if (unicode)
189         ret = SendMessageW(hwndUI, uMsg, wParam, lParam);
190     else
191         ret = SendMessageA(hwndUI, uMsg, wParam, lParam);
192 
193     InterlockedDecrement(&pimeui->nCntInIMEProc);
194 
195     return ret;
196 }
197 
198 // Win: SendOpenStatusNotify
199 static VOID User32NotifyOpenStatus(PIMEUI pimeui, HWND hwndIMC, BOOL bOpen)
200 {
201     WPARAM wParam = (bOpen ? IMN_OPENSTATUSWINDOW : IMN_CLOSESTATUSWINDOW);
202 
203     ASSERT(pimeui->spwnd != NULL);
204 
205     pimeui->fShowStatus = bOpen;
206 
207     if (LOWORD(GetWin32ClientInfo()->dwExpWinVer) >= 0x400)
208         SendMessageW(hwndIMC, WM_IME_NOTIFY, wParam, 0);
209     else
210         User32SendImeUIMessage(pimeui, WM_IME_NOTIFY, wParam, 0, TRUE);
211 }
212 
213 // Win: ImeMarkUsedContext
214 static VOID User32SetImeWindowOfImc(HIMC hIMC, HWND hImeWnd)
215 {
216     PIMC pIMC = ValidateHandle(hIMC, TYPE_INPUTCONTEXT);
217     if (!pIMC || pIMC->hImeWnd == hImeWnd)
218         return;
219 
220     NtUserUpdateInputContext(hIMC, UIC_IMEWINDOW, (ULONG_PTR)hImeWnd);
221 }
222 
223 // Win: ImeSetImc
224 static VOID User32UpdateImcOfImeUI(PIMEUI pimeui, HIMC hNewIMC)
225 {
226     HWND hImeWnd;
227     HIMC hOldIMC = pimeui->hIMC;
228 
229     ASSERT(pimeui->spwnd != NULL);
230     hImeWnd = UserHMGetHandle(pimeui->spwnd);
231 
232     if (hNewIMC == hOldIMC)
233         return;
234 
235     if (hOldIMC)
236         User32SetImeWindowOfImc(hOldIMC, NULL);
237 
238     pimeui->hIMC = hNewIMC;
239 
240     if (hNewIMC)
241         User32SetImeWindowOfImc(hNewIMC, hImeWnd);
242 }
243 
244 /* Handles WM_IME_NOTIFY message of the default IME window. */
245 /* Win: ImeNotifyHandler */
246 static LRESULT ImeWnd_OnImeNotify(PIMEUI pimeui, WPARAM wParam, LPARAM lParam)
247 {
248     LRESULT ret = 0;
249     HIMC hIMC;
250     LPINPUTCONTEXT pIC;
251     HWND hwndUI, hwndIMC, hImeWnd, hwndOwner;
252 
253     ASSERT(pimeui->spwnd != NULL);
254 
255     switch (wParam)
256     {
257         case IMN_SETCONVERSIONMODE:
258         case IMN_SETOPENSTATUS:
259             hIMC = pimeui->hIMC;
260             pIC = IMM_FN(ImmLockIMC)(hIMC);
261             if (pIC)
262             {
263                 hwndIMC = pimeui->hwndIMC;
264                 if (IsWindow(hwndIMC))
265                 {
266                     NtUserNotifyIMEStatus(hwndIMC, pIC->fOpen, pIC->fdwConversion);
267                 }
268                 else if (gfConIme == TRUE && pimeui->spwnd)
269                 {
270                     hImeWnd = UserHMGetHandle(pimeui->spwnd);
271                     hwndOwner = GetWindow(hImeWnd, GW_OWNER);
272                     if (hwndOwner)
273                     {
274                         NtUserNotifyIMEStatus(hwndOwner, pIC->fOpen, pIC->fdwConversion);
275                     }
276                 }
277 
278                 IMM_FN(ImmUnlockIMC)(hIMC);
279             }
280             /* FALL THROUGH */
281         default:
282             ret = User32SendImeUIMessage(pimeui, WM_IME_NOTIFY, wParam, lParam, TRUE);
283             break;
284 
285         case IMN_PRIVATE:
286             hwndUI = pimeui->hwndUI;
287             if (IsWindow(hwndUI))
288                 ret = SendMessageW(hwndUI, WM_IME_NOTIFY, wParam, lParam);
289             break;
290     }
291 
292     return ret;
293 }
294 
295 /* Creates the IME UI window. */
296 /* Win: CreateIMEUI */
297 static HWND User32CreateImeUIWindow(PIMEUI pimeui, HKL hKL)
298 {
299     IMEINFOEX ImeInfoEx;
300     PIMEDPI pImeDpi;
301     WNDCLASSW wc;
302     HWND hwndUI = NULL;
303     CHAR szUIClass[32];
304     PWND pwnd = pimeui->spwnd;
305 
306     ASSERT(pimeui->spwnd != NULL);
307 
308     if (!pwnd || !IMM_FN(ImmGetImeInfoEx)(&ImeInfoEx, ImeInfoExKeyboardLayout, &hKL))
309         return NULL;
310 
311     pImeDpi = IMM_FN(ImmLockImeDpi)(hKL);
312     if (!pImeDpi)
313         return NULL;
314 
315     if (!GetClassInfoW(pImeDpi->hInst, ImeInfoEx.wszUIClass, &wc))
316         goto Quit;
317 
318     if (ImeInfoEx.ImeInfo.fdwProperty & IME_PROP_UNICODE)
319     {
320         hwndUI = CreateWindowW(ImeInfoEx.wszUIClass, ImeInfoEx.wszUIClass, WS_POPUP | WS_DISABLED,
321                                0, 0, 0, 0, UserHMGetHandle(pwnd), 0, wc.hInstance, NULL);
322     }
323     else
324     {
325         WideCharToMultiByte(CP_ACP, 0, ImeInfoEx.wszUIClass, -1,
326                             szUIClass, _countof(szUIClass), NULL, NULL);
327         szUIClass[_countof(szUIClass) - 1] = 0;
328 
329         hwndUI = CreateWindowA(szUIClass, szUIClass, WS_POPUP | WS_DISABLED,
330                                0, 0, 0, 0, UserHMGetHandle(pwnd), 0, wc.hInstance, NULL);
331     }
332 
333     if (hwndUI)
334         NtUserSetWindowLongPtr(hwndUI, IMMGWLP_IMC, (LONG_PTR)pimeui->hIMC, FALSE);
335 
336 Quit:
337     IMM_FN(ImmUnlockImeDpi)(pImeDpi);
338     return hwndUI;
339 }
340 
341 /* Initializes the default IME window. */
342 /* Win: ImeWndCreateHandler */
343 static INT ImeWnd_OnCreate(PIMEUI pimeui, LPCREATESTRUCT lpCS)
344 {
345     PWND pParentWnd, pWnd = pimeui->spwnd;
346     HIMC hIMC = NULL;
347 
348     if (!pWnd || (pWnd->style & (WS_DISABLED | WS_POPUP)) != (WS_DISABLED | WS_POPUP))
349         return -1;
350 
351     pParentWnd = ValidateHwnd(lpCS->hwndParent);
352     if (pParentWnd)
353     {
354         hIMC = pParentWnd->hImc;
355         if (hIMC && !User32CanSetImeWindowToImc(hIMC, UserHMGetHandle(pWnd)))
356             hIMC = NULL;
357     }
358 
359     User32UpdateImcOfImeUI(pimeui, hIMC);
360 
361     pimeui->fShowStatus = FALSE;
362     pimeui->nCntInIMEProc = 0;
363     pimeui->fActivate = FALSE;
364     pimeui->fDestroy = FALSE;
365     pimeui->hwndIMC = NULL;
366     pimeui->hKL = GetWin32ClientInfo()->hKL;
367     pimeui->fCtrlShowStatus = TRUE;
368     pimeui->dwLastStatus = 0;
369 
370     return 0;
371 }
372 
373 /* Destroys the IME UI window. */
374 /* Win: DestroyIMEUI */
375 static VOID User32DestroyImeUIWindow(PIMEUI pimeui)
376 {
377     HWND hwndUI = pimeui->hwndUI;
378 
379     if (IsWindow(hwndUI))
380     {
381         pimeui->fDestroy = TRUE;
382         NtUserDestroyWindow(hwndUI);
383     }
384 
385     pimeui->fShowStatus = pimeui->fDestroy = FALSE;
386     pimeui->hwndUI = NULL;
387 }
388 
389 /* Handles WM_IME_SELECT message of the default IME window. */
390 /* Win: ImeSelectHandler */
391 static VOID ImeWnd_OnImeSelect(PIMEUI pimeui, WPARAM wParam, LPARAM lParam)
392 {
393     HKL hKL;
394     HWND hwndUI, hwndIMC = pimeui->hwndIMC;
395 
396     if (wParam)
397     {
398         pimeui->hKL = hKL = (HKL)lParam;
399 
400         if (!pimeui->fActivate)
401             return;
402 
403         pimeui->hwndUI = hwndUI = User32CreateImeUIWindow(pimeui, hKL);
404         if (hwndUI)
405             User32SendImeUIMessage(pimeui, WM_IME_SELECT, wParam, lParam, TRUE);
406 
407         if (User32GetImeShowStatus() && pimeui->fCtrlShowStatus)
408         {
409             if (!pimeui->fShowStatus && pimeui->fActivate && IsWindow(hwndIMC))
410                 User32NotifyOpenStatus(pimeui, hwndIMC, TRUE);
411         }
412     }
413     else
414     {
415         if (pimeui->fShowStatus && pimeui->fActivate && IsWindow(hwndIMC))
416             User32NotifyOpenStatus(pimeui, hwndIMC, FALSE);
417 
418         User32SendImeUIMessage(pimeui, WM_IME_SELECT, wParam, lParam, TRUE);
419         User32DestroyImeUIWindow(pimeui);
420         pimeui->hKL = NULL;
421     }
422 }
423 
424 /* Handles WM_IME_CONTROL message of the default IME window. */
425 /* Win: ImeControlHandler(pimeui, wParam, lParam, !unicode) */
426 static LRESULT
427 ImeWnd_OnImeControl(PIMEUI pimeui, WPARAM wParam, LPARAM lParam, BOOL unicode)
428 {
429     HIMC hIMC = pimeui->hIMC;
430     DWORD dwConversion, dwSentence;
431     POINT pt;
432 
433     if (IS_CICERO_MODE())
434     {
435         if (wParam == IMC_OPENSTATUSWINDOW)
436         {
437             IMM_FN(CtfImmRestoreToolbarWnd)(pimeui->dwLastStatus);
438             pimeui->dwLastStatus = 0;
439         }
440         else if (wParam == IMC_CLOSESTATUSWINDOW)
441         {
442             pimeui->dwLastStatus = IMM_FN(CtfImmHideToolbarWnd)();
443         }
444     }
445 
446     if (!hIMC)
447         return 0;
448 
449     switch (wParam)
450     {
451         case IMC_GETCONVERSIONMODE:
452             if (!IMM_FN(ImmGetConversionStatus)(hIMC, &dwConversion, &dwSentence))
453                 return 1;
454             return dwConversion;
455 
456         case IMC_GETSENTENCEMODE:
457             if (!IMM_FN(ImmGetConversionStatus)(hIMC, &dwConversion, &dwSentence))
458                 return 1;
459             return dwSentence;
460 
461         case IMC_GETOPENSTATUS:
462             return IMM_FN(ImmGetOpenStatus)(hIMC);
463 
464         case IMC_SETCONVERSIONMODE:
465             if (!IMM_FN(ImmGetConversionStatus)(hIMC, &dwConversion, &dwSentence) ||
466                 !IMM_FN(ImmSetConversionStatus)(hIMC, (DWORD)lParam, dwSentence))
467             {
468                 return 1;
469             }
470             break;
471 
472         case IMC_SETSENTENCEMODE:
473             if (!IMM_FN(ImmGetConversionStatus)(hIMC, &dwConversion, &dwSentence) ||
474                 !IMM_FN(ImmSetConversionStatus)(hIMC, dwConversion, (DWORD)lParam))
475             {
476                 return 1;
477             }
478             break;
479 
480         case IMC_SETOPENSTATUS:
481             if (!IMM_FN(ImmSetOpenStatus)(hIMC, (BOOL)lParam))
482                 return 1;
483             break;
484 
485         case IMC_GETCANDIDATEPOS:
486         case IMC_GETCOMPOSITIONWINDOW:
487         case IMC_GETSOFTKBDPOS:
488         case IMC_SETSOFTKBDPOS:
489         case IMC_GETSTATUSWINDOWPOS:
490             return User32SendImeUIMessage(pimeui, WM_IME_CONTROL, wParam, lParam, unicode);
491 
492         case IMC_SETCANDIDATEPOS:
493             if (!IMM_FN(ImmSetCandidateWindow)(hIMC, (LPCANDIDATEFORM)lParam))
494                 return 1;
495             break;
496 
497         case IMC_GETCOMPOSITIONFONT:
498             if (unicode)
499             {
500                 if (!IMM_FN(ImmGetCompositionFontW)(hIMC, (LPLOGFONTW)lParam))
501                     return 1;
502             }
503             else
504             {
505                 if (!IMM_FN(ImmGetCompositionFontA)(hIMC, (LPLOGFONTA)lParam))
506                     return 1;
507             }
508             break;
509 
510         case IMC_SETCOMPOSITIONFONT:
511             if (unicode)
512             {
513                 if (!IMM_FN(ImmSetCompositionFontW)(hIMC, (LPLOGFONTW)lParam))
514                     return 1;
515             }
516             else
517             {
518                 if (!IMM_FN(ImmSetCompositionFontA)(hIMC, (LPLOGFONTA)lParam))
519                     return 1;
520             }
521             break;
522 
523         case IMC_SETCOMPOSITIONWINDOW:
524             if (!IMM_FN(ImmSetCompositionWindow)(hIMC, (LPCOMPOSITIONFORM)lParam))
525                 return 1;
526             break;
527 
528         case IMC_SETSTATUSWINDOWPOS:
529             pt.x = GET_X_LPARAM(lParam);
530             pt.y = GET_Y_LPARAM(lParam);
531             if (!IMM_FN(ImmSetStatusWindowPos)(hIMC, &pt))
532                 return 1;
533             break;
534 
535         case IMC_CLOSESTATUSWINDOW:
536             if (pimeui->fShowStatus && User32GetImeShowStatus())
537             {
538                 pimeui->fShowStatus = FALSE;
539                 User32SendImeUIMessage(pimeui, WM_IME_NOTIFY, IMN_CLOSESTATUSWINDOW, 0, TRUE);
540             }
541             pimeui->fCtrlShowStatus = FALSE;
542             break;
543 
544         case IMC_OPENSTATUSWINDOW:
545             if (!pimeui->fShowStatus && User32GetImeShowStatus())
546             {
547                 pimeui->fShowStatus = TRUE;
548                 User32SendImeUIMessage(pimeui, WM_IME_NOTIFY, IMN_OPENSTATUSWINDOW, 0, TRUE);
549             }
550             pimeui->fCtrlShowStatus = TRUE;
551             break;
552 
553         default:
554             break;
555     }
556 
557     return 0;
558 }
559 
560 /* Modify the IME activation status. */
561 /* Win: FocusSetIMCContext */
562 static VOID FASTCALL User32SetImeActivenessOfWindow(HWND hWnd, BOOL bActive)
563 {
564     HIMC hIMC;
565 
566     if (!hWnd || !IsWindow(hWnd))
567     {
568         IMM_FN(ImmSetActiveContext)(NULL, NULL, bActive);
569         return;
570     }
571 
572     hIMC = IMM_FN(ImmGetContext)(hWnd);
573     IMM_FN(ImmSetActiveContext)(hWnd, hIMC, bActive);
574     IMM_FN(ImmReleaseContext)(hWnd, hIMC);
575 }
576 
577 /* Win: CtfLoadThreadLayout */
578 VOID FASTCALL CtfLoadThreadLayout(PIMEUI pimeui)
579 {
580     IMM_FN(CtfImmTIMActivate)(pimeui->hKL);
581     pimeui->hKL = GetWin32ClientInfo()->hKL;
582     IMM_FN(ImmLoadIME)(pimeui->hKL);
583     pimeui->hwndUI = NULL;
584 }
585 
586 /* Open the IME help or check the existence of the IME help. */
587 static LRESULT FASTCALL
588 User32DoImeHelp(PIMEUI pimeui, WPARAM wParam, LPARAM lParam)
589 {
590     WCHAR szHelpFile[MAX_PATH];
591     DWORD ret, dwEsc = IME_ESC_GETHELPFILENAME;
592     size_t cch;
593 
594     /* Is there any IME help file? */
595     ret = IMM_FN(ImmEscapeW)(pimeui->hKL, pimeui->hIMC, IME_ESC_QUERY_SUPPORT, &dwEsc);
596     if (!ret || !lParam)
597         return ret;
598 
599     /* Get the help filename */
600     if (IMM_FN(ImmEscapeW)(pimeui->hKL, pimeui->hIMC, IME_ESC_GETHELPFILENAME, szHelpFile))
601     {
602         /* Check filename extension */
603         cch = wcslen(szHelpFile);
604         if (cch > 4 && _wcsicmp(&szHelpFile[cch - 4], L".HLP") == 0)
605         {
606             /* Open the old-style help */
607             TRACE("szHelpFile: %s\n", debugstr_w(szHelpFile));
608             WinHelpW(NULL, szHelpFile, HELP_FINDER, 0);
609         }
610         else
611         {
612             /* Open the new-style help */
613             FIXME("(%p, %p, %p): %s\n", pimeui, wParam, lParam, debugstr_w(szHelpFile));
614             ret = FALSE;
615         }
616     }
617 
618     return ret;
619 }
620 
621 /* Handles WM_IME_SYSTEM message of the default IME window. */
622 /* Win: ImeSystemHandler */
623 static LRESULT ImeWnd_OnImeSystem(PIMEUI pimeui, WPARAM wParam, LPARAM lParam)
624 {
625     LRESULT ret = 0;
626     LPINPUTCONTEXTDX pIC;
627     HIMC hIMC = pimeui->hIMC;
628     LPCANDIDATEFORM pCandForm;
629     LPCOMPOSITIONFORM pCompForm;
630     DWORD dwConversion, dwSentence;
631     HWND hImeWnd;
632     BOOL bCompForm;
633     CANDIDATEFORM CandForm;
634     COMPOSITIONFORM CompForm;
635     UINT iCandForm;
636 
637     ASSERT(pimeui->spwnd != NULL);
638 
639     switch (wParam)
640     {
641         case IMS_NOTIFYIMESHOW:
642             if (User32GetImeShowStatus() == !lParam)
643             {
644                 hImeWnd = UserHMGetHandle(pimeui->spwnd);
645                 NtUserCallHwndParamLock(hImeWnd, lParam, TWOPARAM_ROUTINE_IMESHOWSTATUSCHANGE);
646             }
647             break;
648 
649         case IMS_UPDATEIMEUI:
650             if (!hIMC)
651                 break;
652 
653             bCompForm = TRUE;
654             pIC = IMM_FN(ImmLockIMC)(hIMC);
655             if (pIC)
656             {
657                 bCompForm = !(pIC->dwUIFlags & 0x2);
658                 IMM_FN(ImmUnlockIMC)(hIMC);
659             }
660 
661             if (!IsWindow(pimeui->hwndIMC))
662                 break;
663 
664             if (bCompForm && IMM_FN(ImmGetCompositionWindow)(hIMC, &CompForm))
665             {
666                 if (CompForm.dwStyle)
667                     IMM_FN(ImmSetCompositionWindow)(hIMC, &CompForm);
668             }
669 
670             for (iCandForm = 0; iCandForm < MAX_CANDIDATEFORM; ++iCandForm)
671             {
672                 if (IMM_FN(ImmGetCandidateWindow)(hIMC, iCandForm, &CandForm))
673                 {
674                     if (CandForm.dwStyle)
675                         IMM_FN(ImmSetCandidateWindow)(hIMC, &CandForm);
676                 }
677             }
678             break;
679 
680         case IMS_SETCANDFORM:
681             pIC = IMM_FN(ImmLockIMC)(hIMC);
682             if (!pIC)
683                 break;
684 
685             pCandForm = &pIC->cfCandForm[lParam];
686             IMM_FN(ImmSetCandidateWindow)(hIMC, pCandForm);
687             IMM_FN(ImmUnlockIMC)(hIMC);
688             break;
689 
690         case IMS_SETCOMPFONT:
691             pIC = IMM_FN(ImmLockIMC)(hIMC);
692             if (!pIC)
693                 break;
694 
695             IMM_FN(ImmSetCompositionFontW)(hIMC, &pIC->lfFont.W);
696             IMM_FN(ImmUnlockIMC)(hIMC);
697             break;
698 
699         case IMS_SETCOMPFORM:
700             pIC = IMM_FN(ImmLockIMC)(hIMC);
701             if (!pIC)
702                 break;
703 
704             pCompForm = &pIC->cfCompForm;
705             pIC->dwUIFlags |= 0x8;
706             IMM_FN(ImmSetCompositionWindow)(hIMC, pCompForm);
707             IMM_FN(ImmUnlockIMC)(hIMC);
708             break;
709 
710         case IMS_CONFIGURE:
711             IMM_FN(ImmConfigureIMEW)((HKL)lParam, pimeui->hwndIMC, IME_CONFIG_GENERAL, NULL);
712             break;
713 
714         case IMS_SETOPENSTATUS:
715             if (hIMC)
716                 IMM_FN(ImmSetOpenStatus)(hIMC, (BOOL)lParam);
717             break;
718 
719         case IMS_FREELAYOUT:
720             ret = IMM_FN(ImmFreeLayout)((DWORD)lParam);
721             break;
722 
723         case 0x13:
724             FIXME("\n");
725             break;
726 
727         case IMS_GETCONVSTATUS:
728             IMM_FN(ImmGetConversionStatus)(hIMC, &dwConversion, &dwSentence);
729             ret = dwConversion;
730             break;
731 
732         case IMS_IMEHELP:
733             return User32DoImeHelp(pimeui, wParam, lParam);
734 
735         case IMS_IMEACTIVATE:
736             User32SetImeActivenessOfWindow((HWND)lParam, TRUE);
737             break;
738 
739         case IMS_IMEDEACTIVATE:
740             User32SetImeActivenessOfWindow((HWND)lParam, FALSE);
741             break;
742 
743         case IMS_ACTIVATELAYOUT:
744             ret = IMM_FN(ImmActivateLayout)((HKL)lParam);
745             break;
746 
747         case IMS_GETIMEMENU:
748             ret = IMM_FN(ImmPutImeMenuItemsIntoMappedFile)((HIMC)lParam);
749             break;
750 
751         case 0x1D:
752             FIXME("\n");
753             break;
754 
755         case IMS_GETCONTEXT:
756             ret = (ULONG_PTR)IMM_FN(ImmGetContext)((HWND)lParam);
757             break;
758 
759         case IMS_SENDNOTIFICATION:
760         case IMS_COMPLETECOMPSTR:
761         case IMS_SETLANGBAND:
762         case IMS_UNSETLANGBAND:
763             ret = IMM_FN(ImmSystemHandler)(hIMC, wParam, lParam);
764             break;
765 
766         case IMS_LOADTHREADLAYOUT:
767             CtfLoadThreadLayout(pimeui);
768             break;
769 
770         default:
771             break;
772     }
773 
774     return ret;
775 }
776 
777 /* Handles WM_IME_SETCONTEXT message of the default IME window. */
778 /* Win: ImeSetContextHandler */
779 LRESULT ImeWnd_OnImeSetContext(PIMEUI pimeui, WPARAM wParam, LPARAM lParam)
780 {
781     LRESULT ret;
782     HIMC hIMC;
783     LPINPUTCONTEXTDX pIC;
784     HWND hwndFocus, hwndOldImc, hwndNewImc, hImeWnd, hwndActive, hwndOwner;
785     PWND pwndFocus, pImeWnd, pwndOwner;
786     COMPOSITIONFORM CompForm;
787 
788     pimeui->fActivate = !!wParam;
789     hwndOldImc = pimeui->hwndIMC;
790     ASSERT(pimeui->spwnd != NULL);
791 
792     if (wParam)
793     {
794         if (!pimeui->hwndUI)
795             pimeui->hwndUI = User32CreateImeUIWindow(pimeui, pimeui->hKL);
796 
797         if (gfConIme == -1)
798         {
799             gfConIme = (INT)NtUserGetThreadState(THREADSTATE_CHECKCONIME);
800             if (gfConIme)
801                 pimeui->fCtrlShowStatus = FALSE;
802         }
803 
804         hImeWnd = UserHMGetHandle(pimeui->spwnd);
805 
806         if (gfConIme)
807         {
808             hwndOwner = GetWindow(hImeWnd, GW_OWNER);
809             pwndOwner = ValidateHwnd(hwndOwner);
810             if (pwndOwner)
811             {
812                 User32UpdateImcOfImeUI(pimeui, pwndOwner->hImc);
813 
814                 if (pimeui->hwndUI)
815                     SetWindowLongPtrW(pimeui->hwndUI, IMMGWLP_IMC, (LONG_PTR)pwndOwner->hImc);
816             }
817 
818             return User32SendImeUIMessage(pimeui, WM_IME_SETCONTEXT, wParam, lParam, TRUE);
819         }
820 
821         hwndFocus = (HWND)NtUserQueryWindow(hImeWnd, QUERY_WINDOW_FOCUS);
822 
823         hIMC = IMM_FN(ImmGetContext)(hwndFocus);
824 
825         if (hIMC && !User32CanSetImeWindowToImc(hIMC, hImeWnd))
826         {
827             User32UpdateImcOfImeUI(pimeui, NULL);
828             return 0;
829         }
830 
831         User32UpdateImcOfImeUI(pimeui, hIMC);
832 
833         if (pimeui->hwndUI)
834             SetWindowLongPtrW(pimeui->hwndUI, IMMGWLP_IMC, (LONG_PTR)hIMC);
835 
836         if (hIMC)
837         {
838             pIC = IMM_FN(ImmLockIMC)(hIMC);
839             if (!pIC)
840                 return 0;
841 
842             if (hwndFocus != pIC->hWnd)
843             {
844                 IMM_FN(ImmUnlockIMC)(hIMC);
845                 return 0;
846             }
847 
848             if ((pIC->dwUIFlags & 0x40000) && hwndOldImc != hwndFocus)
849             {
850                 RtlZeroMemory(&CompForm, sizeof(CompForm));
851                 IMM_FN(ImmSetCompositionWindow)(hIMC, &CompForm);
852 
853                 pIC->dwUIFlags &= ~0x40000;
854             }
855 
856             IMM_FN(ImmUnlockIMC)(hIMC);
857 
858             hImeWnd = UserHMGetHandle(pimeui->spwnd);
859             if (NtUserSetImeOwnerWindow(hImeWnd, hwndFocus))
860                 pimeui->hwndIMC = hwndFocus;
861         }
862         else
863         {
864             pimeui->hwndIMC = hwndFocus;
865 
866             hImeWnd = UserHMGetHandle(pimeui->spwnd);
867             NtUserSetImeOwnerWindow(hImeWnd, NULL);
868         }
869     }
870 
871     ret = User32SendImeUIMessage(pimeui, WM_IME_SETCONTEXT, wParam, lParam, TRUE);
872 
873     if (!pimeui->spwnd)
874         return 0;
875 
876     if (!pimeui->fCtrlShowStatus || !User32GetImeShowStatus())
877         return ret;
878 
879     hImeWnd = UserHMGetHandle(pimeui->spwnd);
880     hwndFocus = (HWND)NtUserQueryWindow(hImeWnd, QUERY_WINDOW_FOCUS);
881     pwndFocus = ValidateHwnd(hwndFocus);
882 
883     if (wParam)
884     {
885         pImeWnd = ValidateHwnd(hImeWnd);
886         if (pwndFocus && pImeWnd && pImeWnd->head.pti == pwndFocus->head.pti)
887         {
888             hwndNewImc = pimeui->hwndIMC;
889             if (pimeui->fShowStatus)
890             {
891                 if (hwndOldImc && hwndNewImc && hwndOldImc != hwndNewImc &&
892                     IntGetTopLevelWindow(hwndOldImc) != IntGetTopLevelWindow(hwndNewImc))
893                 {
894                     User32NotifyOpenStatus(pimeui, hwndOldImc, FALSE);
895                     User32NotifyOpenStatus(pimeui, hwndNewImc, TRUE);
896                 }
897             }
898             else
899             {
900                 if (ValidateHwnd(hwndNewImc))
901                     User32NotifyOpenStatus(pimeui, hwndNewImc, TRUE);
902             }
903         }
904 
905         pImeWnd = pimeui->spwnd;
906         hImeWnd = (pImeWnd ? UserHMGetHandle(pImeWnd) : NULL);
907         if (hImeWnd)
908             NtUserCallHwndLock(hImeWnd, HWNDLOCK_ROUTINE_CHECKIMESHOWSTATUSINTHRD);
909     }
910     else
911     {
912         pImeWnd = pimeui->spwnd;
913         hImeWnd = UserHMGetHandle(pImeWnd);
914         hwndActive = (HWND)NtUserQueryWindow(hImeWnd, QUERY_WINDOW_ACTIVE);
915         if (!pwndFocus || !hwndActive || pImeWnd->head.pti != pwndFocus->head.pti)
916         {
917             if (IsWindow(hwndOldImc))
918             {
919                 User32NotifyOpenStatus(pimeui, hwndOldImc, FALSE);
920             }
921             else
922             {
923                 pimeui->fShowStatus = FALSE;
924                 User32SendImeUIMessage(pimeui, WM_IME_NOTIFY, IMN_CLOSESTATUSWINDOW, 0, TRUE);
925             }
926         }
927     }
928 
929     return ret;
930 }
931 
932 /* The window procedure of the default IME window */
933 /* Win: ImeWndProcWorker(hwnd, msg, wParam, lParam, !unicode) */
934 LRESULT WINAPI
935 ImeWndProc_common(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam, BOOL unicode) // ReactOS
936 {
937     PWND pWnd;
938     PIMEUI pimeui;
939     LRESULT ret;
940 
941     pWnd = ValidateHwnd(hwnd);
942     if (pWnd)
943     {
944         if (IS_CICERO_MODE())
945         {
946             ret = IMM_FN(CtfImmDispatchDefImeMessage)(hwnd, msg, wParam, lParam);
947             if (ret)
948                 return ret;
949         }
950 
951        if (!pWnd->fnid)
952        {
953           if (msg != WM_NCCREATE)
954           {
955              if (unicode)
956                 return DefWindowProcW(hwnd, msg, wParam, lParam);
957              return DefWindowProcA(hwnd, msg, wParam, lParam);
958           }
959           NtUserSetWindowFNID(hwnd, FNID_IME);
960           pimeui = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(IMEUI));
961           pimeui->spwnd = pWnd;
962           SetWindowLongPtrW(hwnd, 0, (LONG_PTR)pimeui);
963        }
964        else
965        {
966           if (pWnd->fnid != FNID_IME)
967           {
968              ERR("Wrong window class for Ime! fnId 0x%x\n",pWnd->fnid);
969              return 0;
970           }
971           pimeui = (PIMEUI)GetWindowLongPtrW(hwnd, 0);
972           if (pimeui == NULL)
973           {
974              ERR("Window is not set to IME!\n");
975              return 0;
976           }
977        }
978     }
979 
980     if (pimeui->nCntInIMEProc > 0)
981     {
982         switch (msg)
983         {
984             case WM_IME_CHAR:
985             case WM_IME_COMPOSITIONFULL:
986             case WM_IME_CONTROL:
987             case WM_IME_REQUEST:
988             case WM_IME_SELECT:
989             case WM_IME_SETCONTEXT:
990             case WM_IME_STARTCOMPOSITION:
991             case WM_IME_COMPOSITION:
992             case WM_IME_ENDCOMPOSITION:
993                 return 0;
994 
995             case WM_IME_NOTIFY:
996                 if (wParam < IMN_PRIVATE || IS_IME_HKL(pimeui->hKL) || !IS_CICERO_MODE())
997                     return 0;
998                 break;
999 
1000             case WM_IME_SYSTEM:
1001                 switch (wParam)
1002                 {
1003                     case 0x03:
1004                     case 0x10:
1005                     case 0x13:
1006                         break;
1007 
1008                     default:
1009                         return 0;
1010                 }
1011                 break;
1012 
1013             default:
1014             {
1015                 if (unicode)
1016                     return DefWindowProcW(hwnd, msg, wParam, lParam);
1017                 return DefWindowProcA(hwnd, msg, wParam, lParam);
1018             }
1019         }
1020     }
1021 
1022     switch (msg)
1023     {
1024         case WM_CREATE:
1025             return ImeWnd_OnCreate(pimeui, (LPCREATESTRUCT)lParam);
1026 
1027         case WM_DESTROY:
1028             User32DestroyImeUIWindow(pimeui);
1029             break;
1030 
1031         case WM_NCDESTROY:
1032             HeapFree(GetProcessHeap(), 0, pimeui);
1033             NtUserSetWindowFNID(hwnd, FNID_DESTROY);
1034             break;
1035 
1036         case WM_ERASEBKGND:
1037             return TRUE;
1038 
1039         case WM_PAINT:
1040             break;
1041 
1042         case WM_COPYDATA:
1043             // TODO:
1044             break;
1045 
1046         case WM_IME_STARTCOMPOSITION:
1047         case WM_IME_COMPOSITION:
1048         case WM_IME_ENDCOMPOSITION:
1049             return User32SendImeUIMessage(pimeui, msg, wParam, lParam, unicode);
1050 
1051         case WM_IME_CONTROL:
1052             return ImeWnd_OnImeControl(pimeui, wParam, lParam, unicode);
1053 
1054         case WM_IME_NOTIFY:
1055             return ImeWnd_OnImeNotify(pimeui, wParam, lParam);
1056 
1057         case WM_IME_REQUEST:
1058             break;
1059 
1060         case WM_IME_SELECT:
1061             ImeWnd_OnImeSelect(pimeui, wParam, lParam);
1062             break;
1063 
1064         case WM_IME_SETCONTEXT:
1065             return ImeWnd_OnImeSetContext(pimeui, wParam, lParam);
1066 
1067         case WM_IME_SYSTEM:
1068             return ImeWnd_OnImeSystem(pimeui, wParam, lParam);
1069 
1070         default:
1071         {
1072             if (unicode)
1073                 return DefWindowProcW(hwnd, msg, wParam, lParam);
1074             return DefWindowProcA(hwnd, msg, wParam, lParam);
1075         }
1076     }
1077 
1078     return 0;
1079 }
1080 
1081 // Win: ImeWndProcA
1082 LRESULT WINAPI ImeWndProcA( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam )
1083 {
1084     return ImeWndProc_common(hwnd, msg, wParam, lParam, FALSE);
1085 }
1086 
1087 // Win: ImeWndProcW
1088 LRESULT WINAPI ImeWndProcW( HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam )
1089 {
1090     return ImeWndProc_common(hwnd, msg, wParam, lParam, TRUE);
1091 }
1092 
1093 // Win: UpdatePerUserImmEnabling
1094 BOOL WINAPI UpdatePerUserImmEnabling(VOID)
1095 {
1096     HMODULE imm32;
1097     BOOL ret;
1098 
1099     ret = NtUserCallNoParam(NOPARAM_ROUTINE_UPDATEPERUSERIMMENABLING);
1100     if (!ret || !(gpsi->dwSRVIFlags & SRVINFO_IMM32))
1101         return FALSE;
1102 
1103     imm32 = GetModuleHandleW(L"imm32.dll");
1104     if (imm32)
1105         return TRUE;
1106 
1107     imm32 = LoadLibraryW(L"imm32.dll");
1108     if (imm32 == NULL)
1109     {
1110         ERR("Imm32 not installed!\n");
1111         ret = FALSE;
1112     }
1113 
1114     return ret;
1115 }
1116 
1117 BOOL
1118 WINAPI
1119 RegisterIMEClass(VOID)
1120 {
1121     ATOM atom;
1122     WNDCLASSEXW WndClass = { sizeof(WndClass) };
1123 
1124     WndClass.lpszClassName  = L"IME";
1125     WndClass.style          = CS_GLOBALCLASS;
1126     WndClass.lpfnWndProc    = ImeWndProcW;
1127     WndClass.cbWndExtra     = sizeof(LONG_PTR);
1128     WndClass.hCursor        = LoadCursorW(NULL, IDC_ARROW);
1129 
1130     atom = RegisterClassExWOWW(&WndClass, 0, FNID_IME, 0, FALSE);
1131     if (!atom)
1132     {
1133         ERR("Failed to register IME Class!\n");
1134         return FALSE;
1135     }
1136 
1137     RegisterDefaultClasses |= ICLASS_TO_MASK(ICLS_IME);
1138     TRACE("RegisterIMEClass atom = %u\n", atom);
1139     return TRUE;
1140 }
1141 
1142 /*
1143  * @implemented
1144  */
1145 BOOL
1146 WINAPI
1147 IMPSetIMEW(HWND hwnd, LPIMEPROW ime)
1148 {
1149     return IMM_FN(ImmIMPSetIMEW)(hwnd, ime);
1150 }
1151 
1152 /*
1153  * @implemented
1154  */
1155 BOOL
1156 WINAPI
1157 IMPQueryIMEW(LPIMEPROW ime)
1158 {
1159     return IMM_FN(ImmIMPQueryIMEW)(ime);
1160 }
1161 
1162 /*
1163  * @implemented
1164  */
1165 BOOL
1166 WINAPI
1167 IMPGetIMEW(HWND hwnd, LPIMEPROW ime)
1168 {
1169     return IMM_FN(ImmIMPGetIMEW)(hwnd, ime);
1170 }
1171 
1172 /*
1173  * @implemented
1174  */
1175 BOOL
1176 WINAPI
1177 IMPSetIMEA(HWND hwnd, LPIMEPROA ime)
1178 {
1179     return IMM_FN(ImmIMPSetIMEA)(hwnd, ime);
1180 }
1181 
1182 /*
1183  * @implemented
1184  */
1185 BOOL
1186 WINAPI
1187 IMPQueryIMEA(LPIMEPROA ime)
1188 {
1189     return IMM_FN(ImmIMPQueryIMEA)(ime);
1190 }
1191 
1192 /*
1193  * @implemented
1194  */
1195 BOOL
1196 WINAPI
1197 IMPGetIMEA(HWND hwnd, LPIMEPROA ime)
1198 {
1199     return IMM_FN(ImmIMPGetIMEA)(hwnd, ime);
1200 }
1201 
1202 /*
1203  * @implemented
1204  */
1205 LRESULT
1206 WINAPI
1207 SendIMEMessageExW(HWND hwnd, LPARAM lParam)
1208 {
1209     return IMM_FN(ImmSendIMEMessageExW)(hwnd, lParam);
1210 }
1211 
1212 /*
1213  * @implemented
1214  */
1215 LRESULT
1216 WINAPI
1217 SendIMEMessageExA(HWND hwnd, LPARAM lParam)
1218 {
1219     return IMM_FN(ImmSendIMEMessageExA)(hwnd, lParam);
1220 }
1221 
1222 /*
1223  * @implemented
1224  */
1225 BOOL
1226 WINAPI
1227 WINNLSEnableIME(HWND hwnd, BOOL enable)
1228 {
1229     return IMM_FN(ImmWINNLSEnableIME)(hwnd, enable);
1230 }
1231 
1232 /*
1233  * @implemented
1234  */
1235 BOOL
1236 WINAPI
1237 WINNLSGetEnableStatus(HWND hwnd)
1238 {
1239     return IMM_FN(ImmWINNLSGetEnableStatus)(hwnd);
1240 }
1241 
1242 /*
1243  * @implemented
1244  */
1245 UINT
1246 WINAPI
1247 WINNLSGetIMEHotkey(HWND hwnd)
1248 {
1249     return FALSE;
1250 }
1251