comctl32: Fixed height of comboboxex32.
[wine] / dlls / comctl32 / tests / comboex.c
1 /* Unit test suite for comboex control.
2  *
3  * Copyright 2005 Jason Edmeades
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Lesser General Public
7  * License as published by the Free Software Foundation; either
8  * version 2.1 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Lesser General Public License for more details.
14  *
15  * You should have received a copy of the GNU Lesser General Public
16  * License along with this library; if not, write to the Free Software
17  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
18  */
19
20 #include <assert.h>
21 #include <windows.h>
22 #include <commctrl.h>
23
24 #include "wine/test.h"
25 #include "msg.h"
26
27 #define EDITBOX_SEQ_INDEX  0
28 #define NUM_MSG_SEQUENCES  1
29
30 #define EDITBOX_ID         0
31
32 #define expect(expected, got) ok(got == expected, "Expected %d, got %d\n", expected, got)
33
34 static struct msg_sequence *sequences[NUM_MSG_SEQUENCES];
35
36 static HWND hComboExParentWnd;
37 static HINSTANCE hMainHinst;
38 static const char ComboExTestClass[] = "ComboExTestClass";
39
40 static BOOL (WINAPI *pSetWindowSubclass)(HWND, SUBCLASSPROC, UINT_PTR, DWORD_PTR);
41
42 #define MAX_CHARS 100
43 static char *textBuffer = NULL;
44
45 static HWND createComboEx(DWORD style) {
46    return CreateWindowExA(0, WC_COMBOBOXEXA, NULL, style, 0, 0, 300, 300,
47             hComboExParentWnd, NULL, hMainHinst, NULL);
48 }
49
50 static LONG addItem(HWND cbex, int idx, LPTSTR text) {
51     COMBOBOXEXITEM cbexItem;
52     memset(&cbexItem, 0x00, sizeof(cbexItem));
53     cbexItem.mask = CBEIF_TEXT;
54     cbexItem.iItem = idx;
55     cbexItem.pszText    = text;
56     cbexItem.cchTextMax = 0;
57     return SendMessage(cbex, CBEM_INSERTITEM, 0, (LPARAM)&cbexItem);
58 }
59
60 static LONG setItem(HWND cbex, int idx, LPTSTR text) {
61     COMBOBOXEXITEM cbexItem;
62     memset(&cbexItem, 0x00, sizeof(cbexItem));
63     cbexItem.mask = CBEIF_TEXT;
64     cbexItem.iItem = idx;
65     cbexItem.pszText    = text;
66     cbexItem.cchTextMax = 0;
67     return SendMessage(cbex, CBEM_SETITEM, 0, (LPARAM)&cbexItem);
68 }
69
70 static LONG delItem(HWND cbex, int idx) {
71     return SendMessage(cbex, CBEM_DELETEITEM, idx, 0);
72 }
73
74 static LONG getItem(HWND cbex, int idx, COMBOBOXEXITEM *cbItem) {
75     memset(cbItem, 0x00, sizeof(COMBOBOXEXITEM));
76     cbItem->mask = CBEIF_TEXT;
77     cbItem->pszText      = textBuffer;
78     cbItem->iItem        = idx;
79     cbItem->cchTextMax   = 100;
80     return SendMessage(cbex, CBEM_GETITEM, 0, (LPARAM)cbItem);
81 }
82
83 static LRESULT WINAPI editbox_subclass_proc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
84 {
85     WNDPROC oldproc = (WNDPROC)GetWindowLongPtrA(hwnd, GWLP_USERDATA);
86     static LONG defwndproc_counter = 0;
87     LRESULT ret;
88     struct message msg;
89
90     msg.message = message;
91     msg.flags = sent|wparam|lparam;
92     if (defwndproc_counter) msg.flags |= defwinproc;
93     msg.wParam = wParam;
94     msg.lParam = lParam;
95     msg.id     = EDITBOX_ID;
96
97     if (message != WM_PAINT &&
98         message != WM_ERASEBKGND &&
99         message != WM_NCPAINT &&
100         message != WM_NCHITTEST &&
101         message != WM_GETTEXT &&
102         message != WM_GETICON &&
103         message != WM_DEVICECHANGE)
104     {
105         add_message(sequences, EDITBOX_SEQ_INDEX, &msg);
106     }
107
108     defwndproc_counter++;
109     ret = CallWindowProcA(oldproc, hwnd, message, wParam, lParam);
110     defwndproc_counter--;
111     return ret;
112 }
113
114 static HWND subclass_editbox(HWND hwndComboEx)
115 {
116     WNDPROC oldproc;
117     HWND hwnd;
118
119     hwnd = (HWND)SendMessage(hwndComboEx, CBEM_GETEDITCONTROL, 0, 0);
120     oldproc = (WNDPROC)SetWindowLongPtrA(hwnd, GWLP_WNDPROC,
121                                          (LONG_PTR)editbox_subclass_proc);
122     SetWindowLongPtrA(hwnd, GWLP_USERDATA, (LONG_PTR)oldproc);
123
124     return hwnd;
125 }
126
127 static void test_comboboxex(void) {
128     HWND myHwnd = 0;
129     LONG res = -1;
130     COMBOBOXEXITEM cbexItem;
131     static TCHAR first_item[]        = {'F','i','r','s','t',' ','I','t','e','m',0},
132                  second_item[]       = {'S','e','c','o','n','d',' ','I','t','e','m',0},
133                  third_item[]        = {'T','h','i','r','d',' ','I','t','e','m',0},
134                  middle_item[]       = {'B','e','t','w','e','e','n',' ','F','i','r','s','t',' ','a','n','d',' ',
135                                         'S','e','c','o','n','d',' ','I','t','e','m','s',0},
136                  replacement_item[]  = {'B','e','t','w','e','e','n',' ','F','i','r','s','t',' ','a','n','d',' ',
137                                         'S','e','c','o','n','d',' ','I','t','e','m','s',0},
138                  out_of_range_item[] = {'O','u','t',' ','o','f',' ','R','a','n','g','e',' ','I','t','e','m',0};
139
140     /* Allocate space for result */
141     textBuffer = HeapAlloc(GetProcessHeap(), 0, MAX_CHARS);
142
143     /* Basic comboboxex test */
144     myHwnd = createComboEx(WS_BORDER | WS_VISIBLE | WS_CHILD | CBS_DROPDOWN);
145
146     /* Add items onto the end of the combobox */
147     res = addItem(myHwnd, -1, first_item);
148     ok(res == 0, "Adding simple item failed (%d)\n", res);
149     res = addItem(myHwnd, -1, second_item);
150     ok(res == 1, "Adding simple item failed (%d)\n", res);
151     res = addItem(myHwnd, 2, third_item);
152     ok(res == 2, "Adding simple item failed (%d)\n", res);
153     res = addItem(myHwnd, 1, middle_item);
154     ok(res == 1, "Inserting simple item failed (%d)\n", res);
155
156     /* Add an item completely out of range */
157     res = addItem(myHwnd, 99, out_of_range_item);
158     ok(res == -1, "Adding using out of range index worked unexpectedly (%d)\n", res);
159     res = addItem(myHwnd, 5, out_of_range_item);
160     ok(res == -1, "Adding using out of range index worked unexpectedly (%d)\n", res);
161     /* Removed: Causes traps on Windows XP
162        res = addItem(myHwnd, -2, "Out Of Range Item");
163        ok(res == -1, "Adding out of range worked unexpectedly (%ld)\n", res);
164      */
165
166     /* Get an item completely out of range */ 
167     res = getItem(myHwnd, 99, &cbexItem); 
168     ok(res == 0, "Getting item using out of range index worked unexpectedly (%d, %s)\n", res, cbexItem.pszText);
169     res = getItem(myHwnd, 4, &cbexItem); 
170     ok(res == 0, "Getting item using out of range index worked unexpectedly (%d, %s)\n", res, cbexItem.pszText);
171     res = getItem(myHwnd, -2, &cbexItem); 
172     ok(res == 0, "Getting item using out of range index worked unexpectedly (%d, %s)\n", res, cbexItem.pszText);
173
174     /* Get an item in range */ 
175     res = getItem(myHwnd, 0, &cbexItem); 
176     ok(res != 0, "Getting item using valid index failed unexpectedly (%d)\n", res);
177     ok(strcmp(first_item, cbexItem.pszText) == 0, "Getting item returned wrong string (%s)\n", cbexItem.pszText);
178
179     res = getItem(myHwnd, 1, &cbexItem); 
180     ok(res != 0, "Getting item using valid index failed unexpectedly (%d)\n", res);
181     ok(strcmp(middle_item, cbexItem.pszText) == 0, "Getting item returned wrong string (%s)\n", cbexItem.pszText);
182
183     res = getItem(myHwnd, 2, &cbexItem); 
184     ok(res != 0, "Getting item using valid index failed unexpectedly (%d)\n", res);
185     ok(strcmp(second_item, cbexItem.pszText) == 0, "Getting item returned wrong string (%s)\n", cbexItem.pszText);
186
187     res = getItem(myHwnd, 3, &cbexItem); 
188     ok(res != 0, "Getting item using valid index failed unexpectedly (%d)\n", res);
189     ok(strcmp(third_item, cbexItem.pszText) == 0, "Getting item returned wrong string (%s)\n", cbexItem.pszText);
190
191     /* Set an item completely out of range */ 
192     res = setItem(myHwnd, 99, replacement_item); 
193     ok(res == 0, "Setting item using out of range index worked unexpectedly (%d)\n", res);
194     res = setItem(myHwnd, 4, replacement_item); 
195     ok(res == 0, "Setting item using out of range index worked unexpectedly (%d)\n", res);
196     res = setItem(myHwnd, -2, replacement_item); 
197     ok(res == 0, "Setting item using out of range index worked unexpectedly (%d)\n", res);
198
199     /* Set an item in range */ 
200     res = setItem(myHwnd, 0, replacement_item);
201     ok(res != 0, "Setting first item failed (%d)\n", res);
202     res = setItem(myHwnd, 3, replacement_item);
203     ok(res != 0, "Setting last item failed (%d)\n", res);
204
205     /* Remove items completely out of range (4 items in control at this point) */
206     res = delItem(myHwnd, -1);
207     ok(res == CB_ERR, "Deleting using out of range index worked unexpectedly (%d)\n", res);
208     res = delItem(myHwnd, 4);
209     ok(res == CB_ERR, "Deleting using out of range index worked unexpectedly (%d)\n", res);
210
211     /* Remove items in range (4 items in control at this point) */
212     res = delItem(myHwnd, 3);
213     ok(res == 3, "Deleting using out of range index failed (%d)\n", res);
214     res = delItem(myHwnd, 0);
215     ok(res == 2, "Deleting using out of range index failed (%d)\n", res);
216     res = delItem(myHwnd, 0);
217     ok(res == 1, "Deleting using out of range index failed (%d)\n", res);
218     res = delItem(myHwnd, 0);
219     ok(res == 0, "Deleting using out of range index failed (%d)\n", res);
220
221     /* Remove from an empty box */
222     res = delItem(myHwnd, 0);
223     ok(res == CB_ERR, "Deleting using out of range index worked unexpectedly (%d)\n", res);
224
225
226     /* Cleanup */
227     HeapFree(GetProcessHeap(), 0, textBuffer);
228     DestroyWindow(myHwnd);
229 }
230
231 static void test_WM_LBUTTONDOWN(void)
232 {
233     HWND hComboEx, hCombo, hEdit, hList;
234     COMBOBOXINFO cbInfo;
235     UINT x, y, item_height;
236     LRESULT result;
237     UINT i;
238     int idx;
239     RECT rect;
240     WCHAR buffer[3];
241     static const UINT choices[] = {8,9,10,11,12,14,16,18,20,22,24,26,28,36,48,72};
242     static const WCHAR stringFormat[] = {'%','2','d','\0'};
243     BOOL (WINAPI *pGetComboBoxInfo)(HWND, PCOMBOBOXINFO);
244
245     pGetComboBoxInfo = (void*)GetProcAddress(GetModuleHandleA("user32.dll"), "GetComboBoxInfo");
246     if (!pGetComboBoxInfo){
247         win_skip("GetComboBoxInfo is not available\n");
248         return;
249     }
250
251     hComboEx = CreateWindowExA(0, WC_COMBOBOXEXA, NULL,
252             WS_VISIBLE|WS_CHILD|CBS_DROPDOWN, 0, 0, 200, 150,
253             hComboExParentWnd, NULL, hMainHinst, NULL);
254
255     for (i = 0; i < sizeof(choices)/sizeof(UINT); i++){
256         COMBOBOXEXITEMW cbexItem;
257         wsprintfW(buffer, stringFormat, choices[i]);
258
259         memset(&cbexItem, 0x00, sizeof(cbexItem));
260         cbexItem.mask = CBEIF_TEXT;
261         cbexItem.iItem = i;
262         cbexItem.pszText = buffer;
263         cbexItem.cchTextMax = 0;
264         ok(SendMessageW(hComboEx, CBEM_INSERTITEMW, 0, (LPARAM)&cbexItem) >= 0,
265            "Failed to add item %d\n", i);
266     }
267
268     hCombo = (HWND)SendMessage(hComboEx, CBEM_GETCOMBOCONTROL, 0, 0);
269     hEdit = (HWND)SendMessage(hComboEx, CBEM_GETEDITCONTROL, 0, 0);
270
271     cbInfo.cbSize = sizeof(COMBOBOXINFO);
272     result = pGetComboBoxInfo(hCombo, &cbInfo);
273     ok(result, "Failed to get combobox info structure. LastError=%d\n",
274        GetLastError());
275     hList = cbInfo.hwndList;
276
277     trace("hWnd=%p, hComboEx=%p, hCombo=%p, hList=%p, hEdit=%p\n",
278          hComboExParentWnd, hComboEx, hCombo, hList, hEdit);
279     ok(GetFocus() == hComboExParentWnd,
280        "Focus not on Main Window, instead on %p\n", GetFocus());
281
282     /* Click on the button to drop down the list */
283     x = cbInfo.rcButton.left + (cbInfo.rcButton.right-cbInfo.rcButton.left)/2;
284     y = cbInfo.rcButton.top + (cbInfo.rcButton.bottom-cbInfo.rcButton.top)/2;
285     result = SendMessage(hCombo, WM_LBUTTONDOWN, 0, MAKELPARAM(x, y));
286     ok(result, "WM_LBUTTONDOWN was not processed. LastError=%d\n",
287        GetLastError());
288     ok(GetFocus() == hCombo ||
289        broken(GetFocus() != hCombo), /* win98 */
290        "Focus not on ComboBoxEx's ComboBox Control, instead on %p\n",
291        GetFocus());
292     ok(SendMessage(hComboEx, CB_GETDROPPEDSTATE, 0, 0),
293        "The dropdown list should have appeared after clicking the button.\n");
294     idx = SendMessage(hCombo, CB_GETTOPINDEX, 0, 0);
295     ok(idx == 0, "For TopIndex expected %d, got %d\n", 0, idx);
296
297     result = SendMessage(hCombo, WM_LBUTTONUP, 0, MAKELPARAM(x, y));
298     ok(result, "WM_LBUTTONUP was not processed. LastError=%d\n",
299        GetLastError());
300     ok(GetFocus() == hCombo ||
301        broken(GetFocus() != hCombo), /* win98 */
302        "Focus not on ComboBoxEx's ComboBox Control, instead on %p\n",
303        GetFocus());
304
305     /* Click on the 5th item in the list */
306     item_height = SendMessage(hCombo, CB_GETITEMHEIGHT, 0, 0);
307     ok(GetClientRect(hList, &rect), "Failed to get list's client rect.\n");
308     x = rect.left + (rect.right-rect.left)/2;
309     y = item_height/2 + item_height*4;
310     result = SendMessage(hList, WM_MOUSEMOVE, 0, MAKELPARAM(x, y));
311     ok(!result, "WM_MOUSEMOVE was not processed. LastError=%d\n",
312        GetLastError());
313     ok(GetFocus() == hCombo ||
314        broken(GetFocus() != hCombo), /* win98 */
315        "Focus not on ComboBoxEx's ComboBox Control, instead on %p\n",
316        GetFocus());
317
318     result = SendMessage(hList, WM_LBUTTONDOWN, 0, MAKELPARAM(x, y));
319     ok(!result, "WM_LBUTTONDOWN was not processed. LastError=%d\n",
320        GetLastError());
321     ok(GetFocus() == hCombo ||
322        broken(GetFocus() != hCombo), /* win98 */
323        "Focus not on ComboBoxEx's ComboBox Control, instead on %p\n",
324        GetFocus());
325     ok(SendMessage(hComboEx, CB_GETDROPPEDSTATE, 0, 0),
326        "The dropdown list should still be visible.\n");
327
328     result = SendMessage(hList, WM_LBUTTONUP, 0, MAKELPARAM(x, y));
329     ok(!result, "WM_LBUTTONUP was not processed. LastError=%d\n",
330        GetLastError());
331     todo_wine ok(GetFocus() == hEdit ||
332        broken(GetFocus() == hCombo), /* win98 */
333        "Focus not on ComboBoxEx's Edit Control, instead on %p\n",
334        GetFocus());
335
336     result = SendMessage(hCombo, CB_GETDROPPEDSTATE, 0, 0);
337     ok(!result ||
338        broken(result != 0), /* win98 */
339        "The dropdown list should have been rolled up.\n");
340     idx = SendMessage(hComboEx, CB_GETCURSEL, 0, 0);
341     ok(idx == 4 ||
342        broken(idx == -1), /* win98 */
343        "Current Selection: expected %d, got %d\n", 4, idx);
344
345     DestroyWindow(hComboEx);
346 }
347
348 static void test_CB_GETLBTEXT(void)
349 {
350     HWND hCombo;
351     CHAR buff[1];
352     COMBOBOXEXITEMA item;
353     LRESULT ret;
354
355     hCombo = createComboEx(WS_BORDER | WS_VISIBLE | WS_CHILD | CBS_DROPDOWN);
356
357     /* set text to null */
358     addItem(hCombo, 0, NULL);
359
360     buff[0] = 'a';
361     item.mask = CBEIF_TEXT;
362     item.iItem = 0;
363     item.pszText = buff;
364     item.cchTextMax = 1;
365     ret = SendMessage(hCombo, CBEM_GETITEMA, 0, (LPARAM)&item);
366     ok(ret != 0, "CBEM_GETITEM failed\n");
367     ok(buff[0] == 0, "\n");
368
369     ret = SendMessage(hCombo, CB_GETLBTEXTLEN, 0, 0);
370     ok(ret == 0, "Expected zero length\n");
371
372     ret = SendMessage(hCombo, CB_GETLBTEXTLEN, 0, 0);
373     ok(ret == 0, "Expected zero length\n");
374
375     buff[0] = 'a';
376     ret = SendMessage(hCombo, CB_GETLBTEXT, 0, (LPARAM)buff);
377     ok(ret == 0, "Expected zero length\n");
378     ok(buff[0] == 0, "Expected null terminator as a string, got %s\n", buff);
379
380     DestroyWindow(hCombo);
381 }
382
383 static void test_WM_WINDOWPOSCHANGING(void)
384 {
385     HWND hCombo;
386     WINDOWPOS wp;
387     RECT rect;
388     int combo_height;
389     int ret;
390
391     hCombo = createComboEx(WS_BORDER | WS_VISIBLE | WS_CHILD | CBS_DROPDOWN);
392     ok(hCombo != NULL, "createComboEx failed\n");
393     ret = GetWindowRect(hCombo, &rect);
394     ok(ret, "GetWindowRect failed\n");
395     combo_height = rect.bottom - rect.top;
396     ok(combo_height > 0, "wrong combo height\n");
397
398     /* Test height > combo_height */
399     wp.x = rect.left;
400     wp.y = rect.top;
401     wp.cx = (rect.right - rect.left);
402     wp.cy = combo_height * 2;
403     wp.flags = 0;
404     wp.hwnd = hCombo;
405     wp.hwndInsertAfter = NULL;
406
407     ret = SendMessageA(hCombo, WM_WINDOWPOSCHANGING, 0, (LPARAM)&wp);
408     ok(ret == 0, "expected 0, got %x", ret);
409     ok(wp.cy == combo_height,
410             "Expected height %d, got %d\n", combo_height, wp.cy);
411
412     /* Test height < combo_height */
413     wp.x = rect.left;
414     wp.y = rect.top;
415     wp.cx = (rect.right - rect.left);
416     wp.cy = combo_height / 2;
417     wp.flags = 0;
418     wp.hwnd = hCombo;
419     wp.hwndInsertAfter = NULL;
420
421     ret = SendMessageA(hCombo, WM_WINDOWPOSCHANGING, 0, (LPARAM)&wp);
422     ok(ret == 0, "expected 0, got %x", ret);
423     ok(wp.cy == combo_height,
424             "Expected height %d, got %d\n", combo_height, wp.cy);
425
426     ret = DestroyWindow(hCombo);
427     ok(ret, "DestroyWindow failed\n");
428 }
429
430 static LRESULT CALLBACK ComboExTestWndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
431 {
432     switch(msg) {
433
434     case WM_DESTROY:
435         PostQuitMessage(0);
436         break;
437   
438     default:
439         return DefWindowProcA(hWnd, msg, wParam, lParam);
440     }
441     
442     return 0L;
443 }
444
445 static int init(void)
446 {
447     HMODULE hComctl32;
448     BOOL (WINAPI *pInitCommonControlsEx)(const INITCOMMONCONTROLSEX*);
449     WNDCLASSA wc;
450     INITCOMMONCONTROLSEX iccex;
451
452     hComctl32 = GetModuleHandleA("comctl32.dll");
453     pInitCommonControlsEx = (void*)GetProcAddress(hComctl32, "InitCommonControlsEx");
454     if (!pInitCommonControlsEx)
455     {
456         win_skip("InitCommonControlsEx() is missing. Skipping the tests\n");
457         return 0;
458     }
459     iccex.dwSize = sizeof(iccex);
460     iccex.dwICC  = ICC_USEREX_CLASSES;
461     pInitCommonControlsEx(&iccex);
462
463     pSetWindowSubclass = (void*)GetProcAddress(hComctl32, (LPSTR)410);
464
465     wc.style = CS_HREDRAW | CS_VREDRAW;
466     wc.cbClsExtra = 0;
467     wc.cbWndExtra = 0;
468     wc.hInstance = GetModuleHandleA(NULL);
469     wc.hIcon = NULL;
470     wc.hCursor = LoadCursorA(NULL, IDC_ARROW);
471     wc.hbrBackground = GetSysColorBrush(COLOR_WINDOW);
472     wc.lpszMenuName = NULL;
473     wc.lpszClassName = ComboExTestClass;
474     wc.lpfnWndProc = ComboExTestWndProc;
475     RegisterClassA(&wc);
476
477     hComboExParentWnd = CreateWindowExA(0, ComboExTestClass, "ComboEx test", WS_OVERLAPPEDWINDOW|WS_VISIBLE,
478       CW_USEDEFAULT, CW_USEDEFAULT, 680, 260, NULL, NULL, GetModuleHandleA(NULL), 0);
479     assert(hComboExParentWnd != NULL);
480
481     hMainHinst = GetModuleHandleA(NULL);
482     return 1;
483 }
484
485 static void cleanup(void)
486 {
487     MSG msg;
488     
489     PostMessageA(hComboExParentWnd, WM_CLOSE, 0, 0);
490     while (GetMessageA(&msg,0,0,0)) {
491         TranslateMessage(&msg);
492         DispatchMessageA(&msg);
493     }
494     
495     DestroyWindow(hComboExParentWnd);
496     UnregisterClassA(ComboExTestClass, GetModuleHandleA(NULL));
497 }
498
499 static void test_comboboxex_subclass(void)
500 {
501     HWND hComboEx, hCombo, hEdit;
502
503     hComboEx = createComboEx(WS_BORDER | WS_VISIBLE | WS_CHILD | CBS_DROPDOWN);
504
505     hCombo = (HWND)SendMessage(hComboEx, CBEM_GETCOMBOCONTROL, 0, 0);
506     ok(hCombo != NULL, "Failed to get internal combo\n");
507     hEdit = (HWND)SendMessage(hComboEx, CBEM_GETEDITCONTROL, 0, 0);
508     ok(hEdit != NULL, "Failed to get internal edit\n");
509
510     if (pSetWindowSubclass)
511     {
512         ok(GetPropA(hCombo, "CC32SubclassInfo") != NULL, "Expected CC32SubclassInfo property\n");
513         ok(GetPropA(hEdit, "CC32SubclassInfo") != NULL, "Expected CC32SubclassInfo property\n");
514     }
515
516     DestroyWindow(hComboEx);
517 }
518
519 static const struct message test_setitem_edit_seq[] = {
520     { WM_SETTEXT, sent|id, 0, 0, EDITBOX_ID },
521     { EM_SETSEL, sent|id|wparam|lparam, 0,  0, EDITBOX_ID },
522     { EM_SETSEL, sent|id|wparam|lparam, 0, -1, EDITBOX_ID },
523     { 0 }
524 };
525
526 static void test_get_set_item(void)
527 {
528     char textA[] = "test";
529     HWND hComboEx;
530     COMBOBOXEXITEMA item;
531     BOOL ret;
532
533     hComboEx = createComboEx(WS_BORDER | WS_VISIBLE | WS_CHILD | CBS_DROPDOWN);
534
535     subclass_editbox(hComboEx);
536
537     flush_sequences(sequences, NUM_MSG_SEQUENCES);
538
539     memset(&item, 0, sizeof(item));
540     item.mask = CBEIF_TEXT;
541     item.pszText = textA;
542     item.iItem = -1;
543     ret = SendMessage(hComboEx, CBEM_SETITEMA, 0, (LPARAM)&item);
544     expect(TRUE, ret);
545
546     ok_sequence(sequences, EDITBOX_SEQ_INDEX, test_setitem_edit_seq, "set item data for edit", FALSE);
547
548     /* get/set lParam */
549     item.mask = CBEIF_LPARAM;
550     item.iItem = -1;
551     item.lParam = 0xdeadbeef;
552     ret = SendMessage(hComboEx, CBEM_GETITEMA, 0, (LPARAM)&item);
553     expect(TRUE, ret);
554     ok(item.lParam == 0, "Expected zero, got %lx\n", item.lParam);
555
556     item.lParam = 0x1abe11ed;
557     ret = SendMessage(hComboEx, CBEM_SETITEMA, 0, (LPARAM)&item);
558     expect(TRUE, ret);
559
560     item.lParam = 0;
561     ret = SendMessage(hComboEx, CBEM_GETITEMA, 0, (LPARAM)&item);
562     expect(TRUE, ret);
563     ok(item.lParam == 0x1abe11ed, "Expected 0x1abe11ed, got %lx\n", item.lParam);
564
565     DestroyWindow(hComboEx);
566 }
567
568 START_TEST(comboex)
569 {
570     if (!init())
571         return;
572
573     init_msg_sequences(sequences, NUM_MSG_SEQUENCES);
574
575     test_comboboxex();
576     test_WM_LBUTTONDOWN();
577     test_CB_GETLBTEXT();
578     test_WM_WINDOWPOSCHANGING();
579     test_comboboxex_subclass();
580     test_get_set_item();
581
582     cleanup();
583 }