imm32: Added some unit test cases.
diff --git a/.gitignore b/.gitignore
index eee1787..77c6829 100644
--- a/.gitignore
+++ b/.gitignore
@@ -189,6 +189,9 @@
dlls/imagehlp/libimagehlp.def
dlls/imm.dll16
dlls/imm32/libimm32.def
+dlls/imm32/tests/*.ok
+dlls/imm32/tests/imm32_crosstest.exe
+dlls/imm32/tests/testlist.c
dlls/imm32/version.res
dlls/inetcomm/libinetcomm.def
dlls/inetcomm/tests/*.ok
@@ -737,6 +740,7 @@
programs/winetest/gdi32_test.exe
programs/winetest/gdiplus_test.exe
programs/winetest/hlink_test.exe
+programs/winetest/imm32_test.exe
programs/winetest/inetcomm_test.exe
programs/winetest/infosoft_test.exe
programs/winetest/iphlpapi_test.exe
diff --git a/Makefile.in b/Makefile.in
index 18d1b60..fc4d43a 100644
--- a/Makefile.in
+++ b/Makefile.in
@@ -258,6 +258,7 @@
dlls/imaadp32.acm/Makefile \
dlls/imagehlp/Makefile \
dlls/imm32/Makefile \
+ dlls/imm32/tests/Makefile \
dlls/inetcomm/Makefile \
dlls/inetcomm/tests/Makefile \
dlls/infosoft/Makefile \
@@ -641,6 +642,7 @@
dlls/imaadp32.acm/Makefile: dlls/imaadp32.acm/Makefile.in dlls/Makedll.rules
dlls/imagehlp/Makefile: dlls/imagehlp/Makefile.in dlls/Makedll.rules
dlls/imm32/Makefile: dlls/imm32/Makefile.in dlls/Makedll.rules
+dlls/imm32/tests/Makefile: dlls/imm32/tests/Makefile.in dlls/Maketest.rules
dlls/inetcomm/Makefile: dlls/inetcomm/Makefile.in dlls/Makedll.rules
dlls/inetcomm/tests/Makefile: dlls/inetcomm/tests/Makefile.in dlls/Maketest.rules
dlls/infosoft/Makefile: dlls/infosoft/Makefile.in dlls/Makedll.rules
diff --git a/configure b/configure
index a5d8209..3b59e50 100755
--- a/configure
+++ b/configure
@@ -20808,6 +20808,8 @@
ac_config_files="$ac_config_files dlls/imm32/Makefile"
+ac_config_files="$ac_config_files dlls/imm32/tests/Makefile"
+
ac_config_files="$ac_config_files dlls/inetcomm/Makefile"
ac_config_files="$ac_config_files dlls/inetcomm/tests/Makefile"
@@ -22010,6 +22012,7 @@
"dlls/imaadp32.acm/Makefile") CONFIG_FILES="$CONFIG_FILES dlls/imaadp32.acm/Makefile" ;;
"dlls/imagehlp/Makefile") CONFIG_FILES="$CONFIG_FILES dlls/imagehlp/Makefile" ;;
"dlls/imm32/Makefile") CONFIG_FILES="$CONFIG_FILES dlls/imm32/Makefile" ;;
+ "dlls/imm32/tests/Makefile") CONFIG_FILES="$CONFIG_FILES dlls/imm32/tests/Makefile" ;;
"dlls/inetcomm/Makefile") CONFIG_FILES="$CONFIG_FILES dlls/inetcomm/Makefile" ;;
"dlls/inetcomm/tests/Makefile") CONFIG_FILES="$CONFIG_FILES dlls/inetcomm/tests/Makefile" ;;
"dlls/infosoft/Makefile") CONFIG_FILES="$CONFIG_FILES dlls/infosoft/Makefile" ;;
diff --git a/configure.ac b/configure.ac
index e30ba97..3ca94d4 100644
--- a/configure.ac
+++ b/configure.ac
@@ -1684,6 +1684,7 @@
AC_CONFIG_FILES([dlls/imaadp32.acm/Makefile])
AC_CONFIG_FILES([dlls/imagehlp/Makefile])
AC_CONFIG_FILES([dlls/imm32/Makefile])
+AC_CONFIG_FILES([dlls/imm32/tests/Makefile])
AC_CONFIG_FILES([dlls/inetcomm/Makefile])
AC_CONFIG_FILES([dlls/inetcomm/tests/Makefile])
AC_CONFIG_FILES([dlls/infosoft/Makefile])
diff --git a/dlls/Makefile.in b/dlls/Makefile.in
index d7b0976..cfb3247 100644
--- a/dlls/Makefile.in
+++ b/dlls/Makefile.in
@@ -271,6 +271,7 @@
gdi32/tests \
gdiplus/tests \
hlink/tests \
+ imm32/tests \
inetcomm/tests \
infosoft/tests \
iphlpapi/tests \
diff --git a/dlls/imm32/tests/Makefile.in b/dlls/imm32/tests/Makefile.in
new file mode 100644
index 0000000..29a2c4f
--- /dev/null
+++ b/dlls/imm32/tests/Makefile.in
@@ -0,0 +1,13 @@
+TOPSRCDIR = @top_srcdir@
+TOPOBJDIR = ../../..
+SRCDIR = @srcdir@
+VPATH = @srcdir@
+TESTDLL = imm32.dll
+IMPORTS = imm32 user32 kernel32
+
+CTESTS = \
+ imm32.c
+
+@MAKE_TEST_RULES@
+
+@DEPENDENCIES@ # everything below this line is overwritten by make depend
diff --git a/dlls/imm32/tests/imm32.c b/dlls/imm32/tests/imm32.c
new file mode 100644
index 0000000..895c801
--- /dev/null
+++ b/dlls/imm32/tests/imm32.c
@@ -0,0 +1,222 @@
+/*
+ * 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 "imm.h"
+
+#define NUMELEMS(array) (sizeof((array))/sizeof((array)[0]))
+
+/*
+ * 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) {
+ int 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() {
+ 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 int init(void) {
+ WNDCLASSEX wc;
+
+ wc.cbSize = sizeof(WNDCLASSEX);
+ wc.style = 0;
+ wc.lpfnWndProc = DefWindowProc;
+ 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 0;
+
+ 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 0;
+
+ ShowWindow(hwnd, SW_SHOWNORMAL);
+ UpdateWindow(hwnd);
+
+ msg_spy_init(hwnd);
+
+ return 1;
+}
+
+static void cleanup(void) {
+ msg_spy_cleanup();
+ if (hwnd)
+ DestroyWindow(hwnd);
+ UnregisterClass(wndcls, GetModuleHandle(0));
+}
+
+static int test_ImmNotifyIME(void) {
+ static const char string[] = "wine";
+ char resstr[16] = "";
+ HIMC imc;
+
+ imc = ImmGetContext(hwnd);
+ msg_spy_flush_msgs();
+
+ ok(ImmNotifyIME(imc, NI_COMPOSITIONSTR, CPS_CANCEL, 0), "Canceling an "
+ "empty composition string succeeds.\n");
+ todo_wine {
+ 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();
+
+ ImmSetCompositionString(imc, SCS_SETSTR, string, sizeof(string), NULL, 0);
+ ImmNotifyIME(imc, NI_COMPOSITIONSTR, CPS_CANCEL, 0);
+ ok(msg_spy_find_msg(WM_IME_COMPOSITION) != NULL, "Windows does post "
+ "WM_IME_COMPOSITION in response to NI_COMPOSITIONSTR / CPS_CANCEL, if "
+ "the composition string being canceled is non empty.\n");
+ ok(!ImmGetCompositionString(imc, GCS_COMPSTR, resstr, sizeof(resstr)),
+ "After being canceled the composition string is empty.\n");
+
+ msg_spy_flush_msgs();
+
+ ok(ImmNotifyIME(imc, NI_COMPOSITIONSTR, CPS_CANCEL, 0), "Canceling an "
+ "empty composition string succeeds.\n");
+ todo_wine {
+ 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);
+
+ return 0;
+}
+
+START_TEST(imm32) {
+ if (init())
+ test_ImmNotifyIME();
+ cleanup();
+}
diff --git a/programs/winetest/Makefile.in b/programs/winetest/Makefile.in
index 2cc17d1..d3681a3 100644
--- a/programs/winetest/Makefile.in
+++ b/programs/winetest/Makefile.in
@@ -42,6 +42,7 @@
gdi32_test.exe \
gdiplus_test.exe \
hlink_test.exe \
+ imm32_test.exe \
inetcomm_test.exe \
infosoft_test.exe \
iphlpapi_test.exe \
@@ -138,6 +139,8 @@
cp $(DLLDIR)/gdiplus/tests/gdiplus_test.exe$(DLLEXT) $@ && $(STRIP) $@
hlink_test.exe: $(DLLDIR)/hlink/tests/hlink_test.exe$(DLLEXT)
cp $(DLLDIR)/hlink/tests/hlink_test.exe$(DLLEXT) $@ && $(STRIP) $@
+imm32_test.exe: $(DLLDIR)/imm32/tests/imm32_test.exe$(DLLEXT)
+ cp $(DLLDIR)/imm32/tests/imm32_test.exe$(DLLEXT) $@ && $(STRIP) $@
inetcomm_test.exe: $(DLLDIR)/inetcomm/tests/inetcomm_test.exe$(DLLEXT)
cp $(DLLDIR)/inetcomm/tests/inetcomm_test.exe$(DLLEXT) $@ && $(STRIP) $@
infosoft_test.exe: $(DLLDIR)/infosoft/tests/infosoft_test.exe$(DLLEXT)
diff --git a/programs/winetest/winetest.rc b/programs/winetest/winetest.rc
index 58ae062..4b67bd2 100644
--- a/programs/winetest/winetest.rc
+++ b/programs/winetest/winetest.rc
@@ -100,6 +100,7 @@
gdi32_test.exe TESTRES "gdi32_test.exe"
gdiplus_test.exe TESTRES "gdiplus_test.exe"
hlink_test.exe TESTRES "hlink_test.exe"
+imm32_test.exe TESTRES "imm32_test.exe"
inetcomm_test.exe TESTRES "inetcomm_test.exe"
infosoft_test.exe TESTRES "infosoft_test.exe"
iphlpapi_test.exe TESTRES "iphlpapi_test.exe"