522 lines
16 KiB
C
522 lines
16 KiB
C
/*
|
|
* Unit tests for imm32
|
|
*
|
|
* Copyright (c) 2008 Michael Jung
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "wine/test.h"
|
|
#include "winuser.h"
|
|
#include "wingdi.h"
|
|
#include "imm.h"
|
|
|
|
#define NUMELEMS(array) (sizeof((array))/sizeof((array)[0]))
|
|
|
|
static BOOL (WINAPI *pImmAssociateContextEx)(HWND,HIMC,DWORD);
|
|
|
|
/*
|
|
* msgspy - record and analyse message traces sent to a certain window
|
|
*/
|
|
static struct _msg_spy {
|
|
HWND hwnd;
|
|
HHOOK get_msg_hook;
|
|
HHOOK call_wnd_proc_hook;
|
|
CWPSTRUCT msgs[32];
|
|
unsigned int i_msg;
|
|
} msg_spy;
|
|
|
|
static LRESULT CALLBACK get_msg_filter(int nCode, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
if (HC_ACTION == nCode) {
|
|
MSG *msg = (MSG*)lParam;
|
|
|
|
if ((msg->hwnd == msg_spy.hwnd) &&
|
|
(msg_spy.i_msg < NUMELEMS(msg_spy.msgs)))
|
|
{
|
|
msg_spy.msgs[msg_spy.i_msg].hwnd = msg->hwnd;
|
|
msg_spy.msgs[msg_spy.i_msg].message = msg->message;
|
|
msg_spy.msgs[msg_spy.i_msg].wParam = msg->wParam;
|
|
msg_spy.msgs[msg_spy.i_msg].lParam = msg->lParam;
|
|
msg_spy.i_msg++;
|
|
}
|
|
}
|
|
|
|
return CallNextHookEx(msg_spy.get_msg_hook, nCode, wParam, lParam);
|
|
}
|
|
|
|
static LRESULT CALLBACK call_wnd_proc_filter(int nCode, WPARAM wParam,
|
|
LPARAM lParam)
|
|
{
|
|
if (HC_ACTION == nCode) {
|
|
CWPSTRUCT *cwp = (CWPSTRUCT*)lParam;
|
|
|
|
if ((cwp->hwnd == msg_spy.hwnd) &&
|
|
(msg_spy.i_msg < NUMELEMS(msg_spy.msgs)))
|
|
{
|
|
memcpy(&msg_spy.msgs[msg_spy.i_msg], cwp, sizeof(msg_spy.msgs[0]));
|
|
msg_spy.i_msg++;
|
|
}
|
|
}
|
|
|
|
return CallNextHookEx(msg_spy.call_wnd_proc_hook, nCode, wParam, lParam);
|
|
}
|
|
|
|
static void msg_spy_pump_msg_queue(void) {
|
|
MSG msg;
|
|
|
|
while(PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) {
|
|
TranslateMessage(&msg);
|
|
DispatchMessage(&msg);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
static void msg_spy_flush_msgs(void) {
|
|
msg_spy_pump_msg_queue();
|
|
msg_spy.i_msg = 0;
|
|
}
|
|
|
|
static CWPSTRUCT* msg_spy_find_msg(UINT message) {
|
|
UINT i;
|
|
|
|
msg_spy_pump_msg_queue();
|
|
|
|
if (msg_spy.i_msg >= NUMELEMS(msg_spy.msgs))
|
|
fprintf(stdout, "%s:%d: msg_spy: message buffer overflow!\n",
|
|
__FILE__, __LINE__);
|
|
|
|
for (i = 0; i < msg_spy.i_msg; i++)
|
|
if (msg_spy.msgs[i].message == message)
|
|
return &msg_spy.msgs[i];
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void msg_spy_init(HWND hwnd) {
|
|
msg_spy.hwnd = hwnd;
|
|
msg_spy.get_msg_hook =
|
|
SetWindowsHookEx(WH_GETMESSAGE, get_msg_filter, GetModuleHandle(0),
|
|
GetCurrentThreadId());
|
|
msg_spy.call_wnd_proc_hook =
|
|
SetWindowsHookEx(WH_CALLWNDPROC, call_wnd_proc_filter,
|
|
GetModuleHandle(0), GetCurrentThreadId());
|
|
msg_spy.i_msg = 0;
|
|
|
|
msg_spy_flush_msgs();
|
|
}
|
|
|
|
static void msg_spy_cleanup(void) {
|
|
if (msg_spy.get_msg_hook)
|
|
UnhookWindowsHookEx(msg_spy.get_msg_hook);
|
|
if (msg_spy.call_wnd_proc_hook)
|
|
UnhookWindowsHookEx(msg_spy.call_wnd_proc_hook);
|
|
memset(&msg_spy, 0, sizeof(msg_spy));
|
|
}
|
|
|
|
/*
|
|
* imm32 test cases - Issue some IMM commands on a dummy window and analyse the
|
|
* messages being sent to this window in response.
|
|
*/
|
|
static const char wndcls[] = "winetest_imm32_wndcls";
|
|
static HWND hwnd;
|
|
|
|
static LRESULT WINAPI wndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
switch (msg)
|
|
{
|
|
case WM_IME_SETCONTEXT:
|
|
case WM_NCCREATE:
|
|
case WM_CREATE:
|
|
return TRUE;
|
|
}
|
|
|
|
return DefWindowProcA(hwnd,msg,wParam,lParam);
|
|
}
|
|
|
|
static BOOL init(void) {
|
|
WNDCLASSEX wc;
|
|
HIMC imc;
|
|
HMODULE hmod;
|
|
|
|
hmod = GetModuleHandleA("imm32.dll");
|
|
pImmAssociateContextEx = (void*)GetProcAddress(hmod, "ImmAssociateContextEx");
|
|
|
|
wc.cbSize = sizeof(WNDCLASSEX);
|
|
wc.style = 0;
|
|
wc.lpfnWndProc = wndProc;
|
|
wc.cbClsExtra = 0;
|
|
wc.cbWndExtra = 0;
|
|
wc.hInstance = GetModuleHandle(0);
|
|
wc.hIcon = LoadIcon(NULL, IDI_APPLICATION);
|
|
wc.hCursor = LoadCursor(NULL, IDC_ARROW);
|
|
wc.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
|
|
wc.lpszMenuName = NULL;
|
|
wc.lpszClassName = wndcls;
|
|
wc.hIconSm = LoadIcon(NULL, IDI_APPLICATION);
|
|
|
|
if (!RegisterClassExA(&wc))
|
|
return FALSE;
|
|
|
|
hwnd = CreateWindowEx(WS_EX_CLIENTEDGE, wndcls, "Wine imm32.dll test",
|
|
WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT,
|
|
240, 120, NULL, NULL, GetModuleHandle(0), NULL);
|
|
if (!hwnd)
|
|
return FALSE;
|
|
|
|
imc = ImmGetContext(hwnd);
|
|
if (!imc)
|
|
{
|
|
win_skip("IME support not implemented\n");
|
|
return FALSE;
|
|
}
|
|
ImmReleaseContext(hwnd, imc);
|
|
|
|
ShowWindow(hwnd, SW_SHOWNORMAL);
|
|
UpdateWindow(hwnd);
|
|
|
|
msg_spy_init(hwnd);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void cleanup(void) {
|
|
msg_spy_cleanup();
|
|
if (hwnd)
|
|
DestroyWindow(hwnd);
|
|
UnregisterClass(wndcls, GetModuleHandle(0));
|
|
}
|
|
|
|
static void test_ImmNotifyIME(void) {
|
|
static const char string[] = "wine";
|
|
char resstr[16] = "";
|
|
HIMC imc;
|
|
BOOL ret;
|
|
|
|
imc = ImmGetContext(hwnd);
|
|
msg_spy_flush_msgs();
|
|
|
|
ret = ImmNotifyIME(imc, NI_COMPOSITIONSTR, CPS_CANCEL, 0);
|
|
ok(broken(!ret) ||
|
|
ret, /* Vista+ */
|
|
"Canceling an empty composition string should succeed.\n");
|
|
ok(!msg_spy_find_msg(WM_IME_COMPOSITION), "Windows does not post "
|
|
"WM_IME_COMPOSITION in response to NI_COMPOSITIONSTR / CPS_CANCEL, if "
|
|
"the composition string being canceled is empty.\n");
|
|
|
|
ImmSetCompositionString(imc, SCS_SETSTR, string, sizeof(string), NULL, 0);
|
|
msg_spy_flush_msgs();
|
|
|
|
ImmNotifyIME(imc, NI_COMPOSITIONSTR, CPS_CANCEL, 0);
|
|
msg_spy_flush_msgs();
|
|
|
|
/* behavior differs between win9x and NT */
|
|
ret = ImmGetCompositionString(imc, GCS_COMPSTR, resstr, sizeof(resstr));
|
|
ok(!ret, "After being cancelled the composition string is empty.\n");
|
|
|
|
msg_spy_flush_msgs();
|
|
|
|
ret = ImmNotifyIME(imc, NI_COMPOSITIONSTR, CPS_CANCEL, 0);
|
|
ok(broken(!ret) ||
|
|
ret, /* Vista+ */
|
|
"Canceling an empty composition string should succeed.\n");
|
|
ok(!msg_spy_find_msg(WM_IME_COMPOSITION), "Windows does not post "
|
|
"WM_IME_COMPOSITION in response to NI_COMPOSITIONSTR / CPS_CANCEL, if "
|
|
"the composition string being canceled is empty.\n");
|
|
|
|
msg_spy_flush_msgs();
|
|
ImmReleaseContext(hwnd, imc);
|
|
}
|
|
|
|
static void test_ImmGetCompositionString(void)
|
|
{
|
|
HIMC imc;
|
|
static const WCHAR string[] = {'w','i','n','e',0x65e5,0x672c,0x8a9e};
|
|
char cstring[20];
|
|
WCHAR wstring[20];
|
|
DWORD len;
|
|
DWORD alen,wlen;
|
|
|
|
imc = ImmGetContext(hwnd);
|
|
ImmSetCompositionStringW(imc, SCS_SETSTR, string, sizeof(string), NULL,0);
|
|
alen = ImmGetCompositionStringA(imc, GCS_COMPSTR, cstring, 20);
|
|
wlen = ImmGetCompositionStringW(imc, GCS_COMPSTR, wstring, 20);
|
|
/* windows machines without any IME installed just return 0 above */
|
|
if( alen && wlen)
|
|
{
|
|
len = ImmGetCompositionStringW(imc, GCS_COMPATTR, NULL, 0);
|
|
ok(len*sizeof(WCHAR)==wlen,"GCS_COMPATTR(W) not returning correct count\n");
|
|
len = ImmGetCompositionStringA(imc, GCS_COMPATTR, NULL, 0);
|
|
ok(len==alen,"GCS_COMPATTR(A) not returning correct count\n");
|
|
}
|
|
ImmReleaseContext(hwnd, imc);
|
|
}
|
|
|
|
static void test_ImmSetCompositionString(void)
|
|
{
|
|
HIMC imc;
|
|
BOOL ret;
|
|
|
|
SetLastError(0xdeadbeef);
|
|
imc = ImmGetContext(hwnd);
|
|
ok(imc != 0, "ImmGetContext() failed. Last error: %u\n", GetLastError());
|
|
if (!imc)
|
|
return;
|
|
|
|
ret = ImmSetCompositionStringW(imc, SCS_SETSTR, NULL, 0, NULL, 0);
|
|
ok(broken(!ret) ||
|
|
ret, /* Vista+ */
|
|
"ImmSetCompositionStringW() failed.\n");
|
|
|
|
ret = ImmSetCompositionStringW(imc, SCS_SETSTR | SCS_CHANGEATTR,
|
|
NULL, 0, NULL, 0);
|
|
ok(!ret, "ImmSetCompositionStringW() succeeded.\n");
|
|
|
|
ret = ImmSetCompositionStringW(imc, SCS_SETSTR | SCS_CHANGECLAUSE,
|
|
NULL, 0, NULL, 0);
|
|
ok(!ret, "ImmSetCompositionStringW() succeeded.\n");
|
|
|
|
ret = ImmSetCompositionStringW(imc, SCS_CHANGEATTR | SCS_CHANGECLAUSE,
|
|
NULL, 0, NULL, 0);
|
|
ok(!ret, "ImmSetCompositionStringW() succeeded.\n");
|
|
|
|
ret = ImmSetCompositionStringW(imc, SCS_SETSTR | SCS_CHANGEATTR | SCS_CHANGECLAUSE,
|
|
NULL, 0, NULL, 0);
|
|
ok(!ret, "ImmSetCompositionStringW() succeeded.\n");
|
|
|
|
ImmReleaseContext(hwnd, imc);
|
|
}
|
|
|
|
static void test_ImmIME(void)
|
|
{
|
|
HIMC imc;
|
|
|
|
imc = ImmGetContext(hwnd);
|
|
if (imc)
|
|
{
|
|
BOOL rc;
|
|
rc = ImmConfigureIMEA(imc, NULL, IME_CONFIG_REGISTERWORD, NULL);
|
|
ok (rc == 0, "ImmConfigureIMEA did not fail\n");
|
|
rc = ImmConfigureIMEW(imc, NULL, IME_CONFIG_REGISTERWORD, NULL);
|
|
ok (rc == 0, "ImmConfigureIMEW did not fail\n");
|
|
}
|
|
ImmReleaseContext(hwnd,imc);
|
|
}
|
|
|
|
static void test_ImmAssociateContextEx(void)
|
|
{
|
|
HIMC imc;
|
|
BOOL rc;
|
|
|
|
if (!pImmAssociateContextEx) return;
|
|
|
|
imc = ImmGetContext(hwnd);
|
|
if (imc)
|
|
{
|
|
HIMC retimc, newimc;
|
|
|
|
newimc = ImmCreateContext();
|
|
ok(newimc != imc, "handles should not be the same\n");
|
|
rc = pImmAssociateContextEx(NULL, NULL, 0);
|
|
ok(!rc, "ImmAssociateContextEx succeeded\n");
|
|
rc = pImmAssociateContextEx(hwnd, NULL, 0);
|
|
ok(rc, "ImmAssociateContextEx failed\n");
|
|
rc = pImmAssociateContextEx(NULL, imc, 0);
|
|
ok(!rc, "ImmAssociateContextEx succeeded\n");
|
|
|
|
rc = pImmAssociateContextEx(hwnd, imc, 0);
|
|
ok(rc, "ImmAssociateContextEx failed\n");
|
|
retimc = ImmGetContext(hwnd);
|
|
ok(retimc == imc, "handles should be the same\n");
|
|
ImmReleaseContext(hwnd,retimc);
|
|
|
|
rc = pImmAssociateContextEx(hwnd, newimc, 0);
|
|
ok(rc, "ImmAssociateContextEx failed\n");
|
|
retimc = ImmGetContext(hwnd);
|
|
ok(retimc == newimc, "handles should be the same\n");
|
|
ImmReleaseContext(hwnd,retimc);
|
|
|
|
rc = pImmAssociateContextEx(hwnd, NULL, IACE_DEFAULT);
|
|
ok(rc, "ImmAssociateContextEx failed\n");
|
|
}
|
|
ImmReleaseContext(hwnd,imc);
|
|
}
|
|
|
|
typedef struct _igc_threadinfo {
|
|
HWND hwnd;
|
|
HANDLE event;
|
|
HIMC himc;
|
|
} igc_threadinfo;
|
|
|
|
|
|
static DWORD WINAPI ImmGetContextThreadFunc( LPVOID lpParam)
|
|
{
|
|
HIMC h1,h2;
|
|
HWND hwnd2;
|
|
COMPOSITIONFORM cf;
|
|
POINT pt;
|
|
igc_threadinfo *info= (igc_threadinfo*)lpParam;
|
|
info->hwnd = CreateWindowEx(WS_EX_CLIENTEDGE, wndcls, "Wine imm32.dll test",
|
|
WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT,
|
|
240, 120, NULL, NULL, GetModuleHandle(0), NULL);
|
|
|
|
h1 = ImmGetContext(hwnd);
|
|
todo_wine ok(info->himc == h1, "hwnd context changed in new thread\n");
|
|
h2 = ImmGetContext(info->hwnd);
|
|
todo_wine ok(h2 != h1, "new hwnd in new thread should have different context\n");
|
|
info->himc = h2;
|
|
ImmReleaseContext(hwnd,h1);
|
|
|
|
hwnd2 = CreateWindowEx(WS_EX_CLIENTEDGE, wndcls, "Wine imm32.dll test",
|
|
WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT,
|
|
240, 120, NULL, NULL, GetModuleHandle(0), NULL);
|
|
h1 = ImmGetContext(hwnd2);
|
|
|
|
ok(h1 == h2, "Windows in same thread should have same default context\n");
|
|
ImmReleaseContext(hwnd2,h1);
|
|
ImmReleaseContext(info->hwnd,h2);
|
|
DestroyWindow(hwnd2);
|
|
|
|
/* priming for later tests */
|
|
ImmSetCompositionWindow(h1, &cf);
|
|
ImmSetStatusWindowPos(h1, &pt);
|
|
|
|
SetEvent(info->event);
|
|
Sleep(INFINITE);
|
|
return 1;
|
|
}
|
|
|
|
static void test_ImmThreads(void)
|
|
{
|
|
HIMC himc, otherHimc, h1;
|
|
igc_threadinfo threadinfo;
|
|
HANDLE hThread;
|
|
DWORD dwThreadId;
|
|
BOOL rc;
|
|
LOGFONT lf;
|
|
COMPOSITIONFORM cf;
|
|
DWORD status, sentence;
|
|
POINT pt;
|
|
|
|
himc = ImmGetContext(hwnd);
|
|
threadinfo.event = CreateEvent(NULL, TRUE, FALSE, NULL);
|
|
threadinfo.himc = himc;
|
|
hThread = CreateThread(NULL, 0, ImmGetContextThreadFunc, &threadinfo, 0, &dwThreadId );
|
|
WaitForSingleObject(threadinfo.event, INFINITE);
|
|
|
|
otherHimc = ImmGetContext(threadinfo.hwnd);
|
|
|
|
todo_wine ok(himc != otherHimc, "Windows from other threads should have different himc\n");
|
|
todo_wine ok(otherHimc == threadinfo.himc, "Context from other thread should not change in main thread\n");
|
|
|
|
if (0) /* FIXME: Causes wine to hang */
|
|
{
|
|
h1 = ImmAssociateContext(hwnd,otherHimc);
|
|
ok(h1 == NULL, "Should fail to be able to Associate a default context from a different thread\n");
|
|
h1 = ImmGetContext(hwnd);
|
|
ok(h1 == himc, "Context for window should remain unchanged\n");
|
|
ImmReleaseContext(hwnd,h1);
|
|
}
|
|
|
|
|
|
/* OpenStatus */
|
|
rc = ImmSetOpenStatus(himc, TRUE);
|
|
ok(rc != 0, "ImmSetOpenStatus failed\n");
|
|
rc = ImmGetOpenStatus(himc);
|
|
ok(rc != 0, "ImmGetOpenStatus failed\n");
|
|
rc = ImmSetOpenStatus(himc, FALSE);
|
|
ok(rc != 0, "ImmSetOpenStatus failed\n");
|
|
rc = ImmGetOpenStatus(himc);
|
|
ok(rc == 0, "ImmGetOpenStatus failed\n");
|
|
|
|
rc = ImmSetOpenStatus(otherHimc, TRUE);
|
|
todo_wine ok(rc == 0, "ImmSetOpenStatus should fail\n");
|
|
rc = ImmGetOpenStatus(otherHimc);
|
|
todo_wine ok(rc == 0, "ImmGetOpenStatus failed\n");
|
|
rc = ImmSetOpenStatus(otherHimc, FALSE);
|
|
todo_wine ok(rc == 0, "ImmSetOpenStatus should fail\n");
|
|
rc = ImmGetOpenStatus(otherHimc);
|
|
ok(rc == 0, "ImmGetOpenStatus failed\n");
|
|
|
|
/* CompositionFont */
|
|
rc = ImmGetCompositionFont(himc, &lf);
|
|
ok(rc != 0, "ImmGetCompositionFont failed\n");
|
|
rc = ImmSetCompositionFont(himc, &lf);
|
|
ok(rc != 0, "ImmSetCompositionFont failed\n");
|
|
|
|
rc = ImmGetCompositionFont(otherHimc, &lf);
|
|
ok(rc != 0 || broken(rc == 0), "ImmGetCompositionFont failed\n");
|
|
rc = ImmSetCompositionFont(otherHimc, &lf);
|
|
todo_wine ok(rc == 0, "ImmSetCompositionFont should fail\n");
|
|
|
|
/* CompositionWindow */
|
|
rc = ImmSetCompositionWindow(himc, &cf);
|
|
ok(rc != 0, "ImmSetCompositionWindow failed\n");
|
|
rc = ImmGetCompositionWindow(himc, &cf);
|
|
ok(rc != 0, "ImmGetCompositionWindow failed\n");
|
|
|
|
rc = ImmSetCompositionWindow(otherHimc, &cf);
|
|
todo_wine ok(rc == 0, "ImmSetCompositionWindow should fail\n");
|
|
rc = ImmGetCompositionWindow(otherHimc, &cf);
|
|
ok(rc != 0 || broken(rc == 0), "ImmGetCompositionWindow failed\n");
|
|
|
|
/* ConversionStatus */
|
|
rc = ImmGetConversionStatus(himc, &status, &sentence);
|
|
ok(rc != 0, "ImmGetConversionStatus failed\n");
|
|
rc = ImmSetConversionStatus(himc, status, sentence);
|
|
ok(rc != 0, "ImmSetConversionStatus failed\n");
|
|
|
|
rc = ImmGetConversionStatus(otherHimc, &status, &sentence);
|
|
ok(rc != 0 || broken(rc == 0), "ImmGetConversionStatus failed\n");
|
|
rc = ImmSetConversionStatus(otherHimc, status, sentence);
|
|
todo_wine ok(rc == 0, "ImmSetConversionStatus should fail\n");
|
|
|
|
/* StatusWindowPos */
|
|
rc = ImmSetStatusWindowPos(himc, &pt);
|
|
ok(rc != 0, "ImmSetStatusWindowPos failed\n");
|
|
rc = ImmGetStatusWindowPos(himc, &pt);
|
|
ok(rc != 0, "ImmGetStatusWindowPos failed\n");
|
|
|
|
rc = ImmSetStatusWindowPos(otherHimc, &pt);
|
|
todo_wine ok(rc == 0, "ImmSetStatusWindowPos should fail\n");
|
|
rc = ImmGetStatusWindowPos(otherHimc, &pt);
|
|
ok(rc != 0 || broken(rc == 0), "ImmGetStatusWindowPos failed\n");
|
|
|
|
ImmReleaseContext(threadinfo.hwnd,otherHimc);
|
|
ImmReleaseContext(hwnd,himc);
|
|
|
|
DestroyWindow(threadinfo.hwnd);
|
|
TerminateThread(hThread, 1);
|
|
|
|
himc = ImmGetContext(GetDesktopWindow());
|
|
todo_wine ok(himc == NULL, "Should not be able to get himc from other process window\n");
|
|
}
|
|
|
|
START_TEST(imm32) {
|
|
if (init())
|
|
{
|
|
test_ImmNotifyIME();
|
|
test_ImmGetCompositionString();
|
|
test_ImmSetCompositionString();
|
|
test_ImmIME();
|
|
test_ImmAssociateContextEx();
|
|
test_ImmThreads();
|
|
}
|
|
cleanup();
|
|
}
|