comctl32/tab: Init TCITEM fields in TCM_GETITEM when invalid index passed.
[wine] / dlls / comctl32 / tests / tab.c
1 /* Unit test suite for tab control.
2  *
3  * Copyright 2003 Vitaliy Margolen
4  * Copyright 2007 Hagop Hagopian
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19  */
20
21 #include <assert.h>
22 #include <windows.h>
23 #include <commctrl.h>
24 #include <stdio.h>
25
26 #include "wine/test.h"
27 #include "msg.h"
28
29 #define DEFAULT_MIN_TAB_WIDTH 54
30 #define TAB_PADDING_X 6
31 #define EXTRA_ICON_PADDING 3
32 #define MAX_TABLEN 32
33
34 #define NUM_MSG_SEQUENCES  2
35 #define PARENT_SEQ_INDEX   0
36 #define TAB_SEQ_INDEX      1
37
38 #define expect(expected, got) ok ( expected == got, "Expected %d, got %d\n", expected, got)
39 #define expect_str(expected, got)\
40  ok ( strcmp(expected, got) == 0, "Expected '%s', got '%s'\n", expected, got)
41
42 #define TabWidthPadded(padd_x, num) (DEFAULT_MIN_TAB_WIDTH - (TAB_PADDING_X - (padd_x)) * num)
43
44 #define TabCheckSetSize(hwnd, SetWidth, SetHeight, ExpWidth, ExpHeight, Msg)\
45     SendMessage (hwnd, TCM_SETITEMSIZE, 0,\
46         (LPARAM) MAKELPARAM((SetWidth >= 0) ? SetWidth:0, (SetHeight >= 0) ? SetHeight:0));\
47     if (winetest_interactive) RedrawWindow (hwnd, NULL, 0, RDW_UPDATENOW);\
48     CheckSize(hwnd, ExpWidth, ExpHeight, Msg);
49
50 #define CheckSize(hwnd,width,height,msg)\
51     SendMessage (hwnd, TCM_GETITEMRECT, 0, (LPARAM) &rTab);\
52     if ((width  >= 0) && (height < 0))\
53         ok (width  == rTab.right  - rTab.left, "%s: Expected width [%d] got [%d]\n",\
54         msg, (int)width,  rTab.right  - rTab.left);\
55     else if ((height >= 0) && (width  < 0))\
56         ok (height == rTab.bottom - rTab.top,  "%s: Expected height [%d] got [%d]\n",\
57         msg, (int)height, rTab.bottom - rTab.top);\
58     else\
59         ok ((width  == rTab.right  - rTab.left) &&\
60             (height == rTab.bottom - rTab.top ),\
61             "%s: Expected [%d,%d] got [%d,%d]\n", msg, (int)width, (int)height,\
62             rTab.right - rTab.left, rTab.bottom - rTab.top);
63
64 static HFONT hFont = 0;
65
66 static struct msg_sequence *sequences[NUM_MSG_SEQUENCES];
67
68 static const struct message create_parent_wnd_seq[] = {
69     { WM_GETMINMAXINFO, sent },
70     { WM_NCCREATE, sent },
71     { WM_NCCALCSIZE, sent|wparam, 0 },
72     { WM_CREATE, sent },
73     { WM_SHOWWINDOW, sent|wparam, 1 },
74     { WM_WINDOWPOSCHANGING, sent|wparam, 0 },
75     { WM_WINDOWPOSCHANGING, sent|wparam, 0 },
76     { WM_ACTIVATEAPP, sent|wparam, 1 },
77     { WM_NCACTIVATE, sent|wparam, 1 },
78     { WM_ACTIVATE, sent|wparam, 1 },
79     { WM_IME_SETCONTEXT, sent|wparam|defwinproc|optional, 1 },
80     { WM_IME_NOTIFY, sent|defwinproc|optional },
81     { WM_SETFOCUS, sent|wparam|defwinproc, 0 },
82     /* Win9x adds SWP_NOZORDER below */
83     { WM_WINDOWPOSCHANGED, sent},
84     { WM_NCCALCSIZE, sent|wparam|optional, 1 },
85     { WM_SIZE, sent },
86     { WM_MOVE, sent },
87     { 0 }
88 };
89
90 static const struct message add_tab_to_parent[] = {
91     { TCM_INSERTITEMA, sent },
92     { TCM_INSERTITEMA, sent|optional },
93     { WM_NOTIFYFORMAT, sent|defwinproc },
94     { WM_QUERYUISTATE, sent|wparam|lparam|defwinproc|optional, 0, 0 },
95     { WM_PARENTNOTIFY, sent|defwinproc },
96     { TCM_INSERTITEMA, sent },
97     { TCM_INSERTITEMA, sent },
98     { TCM_INSERTITEMA, sent },
99     { TCM_INSERTITEMA, sent|optional },
100     { 0 }
101 };
102
103 static const struct message add_tab_to_parent_interactive[] = {
104     { TCM_INSERTITEMA, sent },
105     { TCM_INSERTITEMA, sent },
106     { WM_NOTIFYFORMAT, sent|defwinproc },
107     { WM_QUERYUISTATE, sent|wparam|lparam|defwinproc, 0, 0 },
108     { WM_PARENTNOTIFY, sent|defwinproc },
109     { TCM_INSERTITEMA, sent },
110     { TCM_INSERTITEMA, sent },
111     { TCM_INSERTITEMA, sent },
112     { WM_SHOWWINDOW, sent},
113     { WM_WINDOWPOSCHANGING, sent},
114     { WM_WINDOWPOSCHANGING, sent},
115     { WM_NCACTIVATE, sent},
116     { WM_ACTIVATE, sent},
117     { WM_IME_SETCONTEXT, sent|defwinproc|optional},
118     { WM_IME_NOTIFY, sent|defwinproc|optional},
119     { WM_SETFOCUS, sent|defwinproc},
120     { WM_WINDOWPOSCHANGED, sent},
121     { WM_SIZE, sent},
122     { WM_MOVE, sent},
123     { 0 }
124 };
125
126 static const struct message add_tab_control_parent_seq[] = {
127     { WM_NOTIFYFORMAT, sent },
128     { WM_QUERYUISTATE, sent|wparam|lparam|optional, 0, 0 },
129     { 0 }
130 };
131
132 static const struct message add_tab_control_parent_seq_interactive[] = {
133     { WM_NOTIFYFORMAT, sent },
134     { WM_QUERYUISTATE, sent|wparam|lparam, 0, 0 },
135     { WM_WINDOWPOSCHANGING, sent|optional},
136     { WM_NCACTIVATE, sent},
137     { WM_ACTIVATE, sent},
138     { WM_WINDOWPOSCHANGING, sent|optional},
139     { WM_KILLFOCUS, sent},
140     { WM_IME_SETCONTEXT, sent|optional},
141     { WM_IME_NOTIFY, sent|optional},
142     { 0 }
143 };
144
145 static const struct message empty_sequence[] = {
146     { 0 }
147 };
148
149 static const struct message set_min_tab_width_seq[] = {
150     { TCM_SETMINTABWIDTH, sent|wparam, 0 },
151     { TCM_SETMINTABWIDTH, sent|wparam, 0 },
152     { 0 }
153 };
154
155 static const struct message get_item_count_seq[] = {
156     { TCM_GETITEMCOUNT, sent|wparam|lparam, 0, 0 },
157     { 0 }
158 };
159
160 static const struct message get_row_count_seq[] = {
161     { TCM_GETROWCOUNT, sent|wparam|lparam, 0, 0 },
162     { 0 }
163 };
164
165 static const struct message get_item_rect_seq[] = {
166     { TCM_GETITEMRECT, sent },
167     { TCM_GETITEMRECT, sent },
168     { 0 }
169 };
170
171 static const struct message getset_cur_focus_seq[] = {
172     { TCM_SETCURFOCUS, sent|lparam, 0 },
173     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
174     { TCM_SETCURFOCUS, sent|lparam, 0 },
175     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
176     { TCM_SETCURSEL, sent|lparam, 0 },
177     { TCM_SETCURFOCUS, sent|lparam, 0 },
178     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
179     { 0 }
180 };
181
182 static const struct message getset_cur_sel_seq[] = {
183     { TCM_SETCURSEL, sent|lparam, 0 },
184     { TCM_GETCURSEL, sent|wparam|lparam, 0, 0 },
185     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
186     { TCM_SETCURSEL, sent|lparam, 0 },
187     { TCM_GETCURSEL, sent|wparam|lparam, 0, 0 },
188     { TCM_SETCURSEL, sent|lparam, 0 },
189     { TCM_SETCURSEL, sent|lparam, 0 },
190     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
191     { 0 }
192 };
193
194 static const struct message getset_extended_style_seq[] = {
195     { TCM_GETEXTENDEDSTYLE, sent|wparam|lparam, 0, 0 },
196     { TCM_SETEXTENDEDSTYLE, sent },
197     { TCM_GETEXTENDEDSTYLE, sent|wparam|lparam, 0, 0 },
198     { TCM_SETEXTENDEDSTYLE, sent },
199     { TCM_GETEXTENDEDSTYLE, sent|wparam|lparam, 0, 0 },
200     { 0 }
201 };
202
203 static const struct message getset_unicode_format_seq[] = {
204     { CCM_SETUNICODEFORMAT, sent|lparam, 0 },
205     { CCM_GETUNICODEFORMAT, sent|wparam|lparam, 0, 0 },
206     { CCM_SETUNICODEFORMAT, sent|lparam, 0 },
207     { CCM_GETUNICODEFORMAT, sent|wparam|lparam, 0, 0 },
208     { CCM_SETUNICODEFORMAT, sent|lparam, 0 },
209     { 0 }
210 };
211
212 static const struct message getset_item_seq[] = {
213     { TCM_SETITEMA, sent },
214     { TCM_GETITEMA, sent },
215     { TCM_GETITEMA, sent },
216     { 0 }
217 };
218
219 static const struct message getset_tooltip_seq[] = {
220     { WM_NOTIFYFORMAT, sent|optional },
221     { WM_QUERYUISTATE, sent|wparam|lparam|optional, 0, 0 },
222     { WM_WINDOWPOSCHANGING, sent|wparam, 0 },
223     { WM_NOTIFYFORMAT, sent|optional },
224     { TCM_SETTOOLTIPS, sent|lparam, 0 },
225     { TCM_GETTOOLTIPS, sent|wparam|lparam, 0, 0 },
226     { TCM_SETTOOLTIPS, sent|lparam, 0 },
227     { TCM_GETTOOLTIPS, sent|wparam|lparam, 0, 0 },
228     { 0 }
229 };
230
231 static const struct message getset_tooltip_parent_seq[] = {
232     { WM_WINDOWPOSCHANGING, sent|wparam, 0 },
233     { 0 }
234 };
235
236 static const struct message insert_focus_seq[] = {
237     { TCM_GETITEMCOUNT, sent|wparam|lparam, 0, 0 },
238     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
239     { TCM_INSERTITEM, sent|wparam, 1 },
240     { WM_NOTIFYFORMAT, sent|defwinproc|optional },
241     { WM_QUERYUISTATE, sent|defwinproc|optional },
242     { WM_PARENTNOTIFY, sent|defwinproc|optional },
243     { TCM_GETITEMCOUNT, sent|wparam|lparam, 0, 0 },
244     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
245     { TCM_INSERTITEM, sent|wparam, 2 },
246     { WM_NOTIFYFORMAT, sent|defwinproc|optional },
247     { WM_QUERYUISTATE, sent|defwinproc|optional, },
248     { WM_PARENTNOTIFY, sent|defwinproc|optional },
249     { TCM_GETITEMCOUNT, sent|wparam|lparam, 0, 0 },
250     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
251     { TCM_SETCURFOCUS, sent|wparam|lparam, -1, 0 },
252     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
253     { TCM_INSERTITEM, sent|wparam, 3 },
254     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
255     { 0 }
256 };
257
258 static const struct message delete_focus_seq[] = {
259     { TCM_GETITEMCOUNT, sent|wparam|lparam, 0, 0 },
260     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
261     { TCM_DELETEITEM, sent|wparam|lparam, 1, 0 },
262     { TCM_GETITEMCOUNT, sent|wparam|lparam, 0, 0 },
263     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
264     { TCM_SETCURFOCUS, sent|wparam|lparam, -1, 0 },
265     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
266     { TCM_DELETEITEM, sent|wparam|lparam, 0, 0 },
267     { TCM_GETITEMCOUNT, sent|wparam|lparam, 0, 0 },
268     { TCM_GETCURFOCUS, sent|wparam|lparam, 0, 0 },
269     { 0 }
270 };
271
272
273 static HWND
274 create_tabcontrol (DWORD style, DWORD mask)
275 {
276     HWND handle;
277     TCITEM tcNewTab;
278     static char text1[] = "Tab 1",
279     text2[] = "Wide Tab 2",
280     text3[] = "T 3";
281
282     handle = CreateWindow (
283         WC_TABCONTROLA,
284         "TestTab",
285         WS_CLIPSIBLINGS | WS_CLIPCHILDREN | TCS_FOCUSNEVER | style,
286         10, 10, 300, 100,
287         NULL, NULL, NULL, 0);
288
289     assert (handle);
290
291     SetWindowLong(handle, GWL_STYLE, WS_CLIPSIBLINGS | WS_CLIPCHILDREN | TCS_FOCUSNEVER | style);
292     SendMessage (handle, WM_SETFONT, 0, (LPARAM) hFont);
293
294     tcNewTab.mask = mask;
295     tcNewTab.pszText = text1;
296     tcNewTab.iImage = 0;
297     SendMessage (handle, TCM_INSERTITEM, 0, (LPARAM) &tcNewTab);
298     tcNewTab.pszText = text2;
299     tcNewTab.iImage = 1;
300     SendMessage (handle, TCM_INSERTITEM, 1, (LPARAM) &tcNewTab);
301     tcNewTab.pszText = text3;
302     tcNewTab.iImage = 2;
303     SendMessage (handle, TCM_INSERTITEM, 2, (LPARAM) &tcNewTab);
304
305     if (winetest_interactive)
306     {
307         ShowWindow (handle, SW_SHOW);
308         RedrawWindow (handle, NULL, 0, RDW_UPDATENOW);
309         Sleep (1000);
310     }
311
312     return handle;
313 }
314
315 static LRESULT WINAPI parentWindowProcess(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
316 {
317     static LONG defwndproc_counter = 0;
318     LRESULT ret;
319     struct message msg;
320
321     /* do not log painting messages */
322     if (message != WM_PAINT &&
323         message != WM_ERASEBKGND &&
324         message != WM_NCPAINT &&
325         message != WM_NCHITTEST &&
326         message != WM_GETTEXT &&
327         message != WM_GETICON &&
328         message != WM_DEVICECHANGE)
329     {
330         trace("parent: %p, %04x, %08lx, %08lx\n", hwnd, message, wParam, lParam);
331
332         msg.message = message;
333         msg.flags = sent|wparam|lparam;
334         if (defwndproc_counter) msg.flags |= defwinproc;
335         msg.wParam = wParam;
336         msg.lParam = lParam;
337         add_message(sequences, PARENT_SEQ_INDEX, &msg);
338     }
339
340     defwndproc_counter++;
341     ret = DefWindowProcA(hwnd, message, wParam, lParam);
342     defwndproc_counter--;
343
344     return ret;
345 }
346
347 static BOOL registerParentWindowClass(void)
348 {
349     WNDCLASSA cls;
350
351     cls.style = 0;
352     cls.lpfnWndProc = parentWindowProcess;
353     cls.cbClsExtra = 0;
354     cls.cbWndExtra = 0;
355     cls.hInstance = GetModuleHandleA(NULL);
356     cls.hIcon = 0;
357     cls.hCursor = LoadCursorA(0, IDC_ARROW);
358     cls.hbrBackground = GetStockObject(WHITE_BRUSH);
359     cls.lpszMenuName = NULL;
360     cls.lpszClassName = "Tab test parent class";
361     return RegisterClassA(&cls);
362 }
363
364 static HWND createParentWindow(void)
365 {
366     if (!registerParentWindowClass())
367         return NULL;
368
369     return CreateWindowEx(0, "Tab test parent class",
370                           "Tab test parent window",
371                           WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX |
372                           WS_MAXIMIZEBOX | WS_VISIBLE,
373                           0, 0, 100, 100,
374                           GetDesktopWindow(), NULL, GetModuleHandleA(NULL), NULL);
375 }
376
377 struct subclass_info
378 {
379     WNDPROC oldproc;
380 };
381
382 static LRESULT WINAPI tabSubclassProcess(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
383 {
384     struct subclass_info *info = (struct subclass_info *)GetWindowLongPtrA(hwnd, GWLP_USERDATA);
385     static LONG defwndproc_counter = 0;
386     LRESULT ret;
387     struct message msg;
388
389     /* do not log painting messages */
390     if (message != WM_PAINT &&
391         message != WM_ERASEBKGND &&
392         message != WM_NCPAINT &&
393         message != WM_NCHITTEST &&
394         message != WM_GETTEXT &&
395         message != WM_GETICON &&
396         message != WM_DEVICECHANGE)
397     {
398         trace("tab: %p, %04x, %08lx, %08lx\n", hwnd, message, wParam, lParam);
399
400         msg.message = message;
401         msg.flags = sent|wparam|lparam;
402         if (defwndproc_counter) msg.flags |= defwinproc;
403         msg.wParam = wParam;
404         msg.lParam = lParam;
405         add_message(sequences, TAB_SEQ_INDEX, &msg);
406     }
407
408     defwndproc_counter++;
409     ret = CallWindowProcA(info->oldproc, hwnd, message, wParam, lParam);
410     defwndproc_counter--;
411
412     return ret;
413 }
414
415 static HWND createFilledTabControl(HWND parent_wnd, DWORD style, DWORD mask, INT nTabs)
416 {
417     HWND tabHandle;
418     TCITEM tcNewTab;
419     struct subclass_info *info;
420     RECT rect;
421     INT i;
422
423     info = HeapAlloc(GetProcessHeap(), 0, sizeof(struct subclass_info));
424     if (!info)
425         return NULL;
426
427     GetClientRect(parent_wnd, &rect);
428
429     tabHandle = CreateWindow (
430         WC_TABCONTROLA,
431         "TestTab",
432         WS_CLIPSIBLINGS | WS_CLIPCHILDREN | TCS_FOCUSNEVER | style,
433         0, 0, rect.right, rect.bottom,
434         parent_wnd, NULL, NULL, 0);
435
436     assert(tabHandle);
437
438     info->oldproc = (WNDPROC)SetWindowLongPtrA(tabHandle, GWLP_WNDPROC, (LONG_PTR)tabSubclassProcess);
439     SetWindowLongPtrA(tabHandle, GWLP_USERDATA, (LONG_PTR)info);
440
441     tcNewTab.mask = mask;
442
443     for (i = 0; i < nTabs; i++)
444     {
445         char tabName[MAX_TABLEN];
446
447         sprintf(tabName, "Tab %d", i+1);
448         tcNewTab.pszText = tabName;
449         tcNewTab.iImage = i;
450         SendMessage (tabHandle, TCM_INSERTITEM, i, (LPARAM) &tcNewTab);
451     }
452
453     if (winetest_interactive)
454     {
455         ShowWindow (tabHandle, SW_SHOW);
456         RedrawWindow (tabHandle, NULL, 0, RDW_UPDATENOW);
457         Sleep (1000);
458     }
459
460     return tabHandle;
461 }
462
463 static HWND create_tooltip (HWND hTab, char toolTipText[])
464 {
465     HWND hwndTT;
466
467     TOOLINFO ti;
468     LPTSTR lptstr = toolTipText;
469     RECT rect;
470
471     /* Creating a tooltip window*/
472     hwndTT = CreateWindowEx(
473         WS_EX_TOPMOST,
474         TOOLTIPS_CLASS,
475         NULL,
476         WS_POPUP | TTS_NOPREFIX | TTS_ALWAYSTIP,
477         CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
478         hTab, NULL, 0, NULL);
479
480     SetWindowPos(
481         hwndTT,
482         HWND_TOPMOST,
483         0, 0, 0, 0,
484         SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
485
486     GetClientRect (hTab, &rect);
487
488     /* Initialize members of toolinfo*/
489     ti.cbSize = sizeof(TOOLINFO);
490     ti.uFlags = TTF_SUBCLASS;
491     ti.hwnd = hTab;
492     ti.hinst = 0;
493     ti.uId = 0;
494     ti.lpszText = lptstr;
495
496     ti.rect = rect;
497
498     /* Add toolinfo structure to the tooltip control */
499     SendMessage(hwndTT, TTM_ADDTOOL, 0, (LPARAM) &ti);
500
501     return hwndTT;
502 }
503
504 static void test_tab(INT nMinTabWidth)
505 {
506     HWND hwTab;
507     RECT rTab;
508     HIMAGELIST himl = ImageList_Create(21, 21, ILC_COLOR, 3, 4);
509     SIZE size;
510     HDC hdc;
511     HFONT hOldFont;
512     INT i, dpi, exp;
513
514     hwTab = create_tabcontrol(TCS_FIXEDWIDTH, TCIF_TEXT|TCIF_IMAGE);
515     SendMessage(hwTab, TCM_SETMINTABWIDTH, 0, nMinTabWidth);
516     /* Get System default MinTabWidth */
517     if (nMinTabWidth < 0)
518         nMinTabWidth = SendMessage(hwTab, TCM_SETMINTABWIDTH, 0, nMinTabWidth);
519
520     hdc = GetDC(hwTab);
521     dpi = GetDeviceCaps(hdc, LOGPIXELSX);
522     hOldFont = SelectObject(hdc, (HFONT)SendMessage(hwTab, WM_GETFONT, 0, 0));
523     GetTextExtentPoint32A(hdc, "Tab 1", strlen("Tab 1"), &size);
524     trace("Tab1 text size: size.cx=%d size.cy=%d\n", size.cx, size.cy);
525     SelectObject(hdc, hOldFont);
526     ReleaseDC(hwTab, hdc);
527
528     trace ("  TCS_FIXEDWIDTH tabs no icon...\n");
529     CheckSize(hwTab, dpi, -1, "default width");
530     TabCheckSetSize(hwTab, 50, 20, 50, 20, "set size");
531     TabCheckSetSize(hwTab, 0, 1, 0, 1, "min size");
532
533     SendMessage(hwTab, TCM_SETIMAGELIST, 0, (LPARAM)himl);
534
535     trace ("  TCS_FIXEDWIDTH tabs with icon...\n");
536     TabCheckSetSize(hwTab, 50, 30, 50, 30, "set size > icon");
537     TabCheckSetSize(hwTab, 20, 20, 25, 20, "set size < icon");
538     TabCheckSetSize(hwTab, 0, 1, 25, 1, "min size");
539
540     DestroyWindow (hwTab);
541
542     hwTab = create_tabcontrol(TCS_FIXEDWIDTH | TCS_BUTTONS, TCIF_TEXT|TCIF_IMAGE);
543     SendMessage(hwTab, TCM_SETMINTABWIDTH, 0, nMinTabWidth);
544
545     hdc = GetDC(hwTab);
546     dpi = GetDeviceCaps(hdc, LOGPIXELSX);
547     ReleaseDC(hwTab, hdc);
548     trace ("  TCS_FIXEDWIDTH buttons no icon...\n");
549     CheckSize(hwTab, dpi, -1, "default width");
550     TabCheckSetSize(hwTab, 20, 20, 20, 20, "set size 1");
551     TabCheckSetSize(hwTab, 10, 50, 10, 50, "set size 2");
552     TabCheckSetSize(hwTab, 0, 1, 0, 1, "min size");
553
554     SendMessage(hwTab, TCM_SETIMAGELIST, 0, (LPARAM)himl);
555
556     trace ("  TCS_FIXEDWIDTH buttons with icon...\n");
557     TabCheckSetSize(hwTab, 50, 30, 50, 30, "set size > icon");
558     TabCheckSetSize(hwTab, 20, 20, 25, 20, "set size < icon");
559     TabCheckSetSize(hwTab, 0, 1, 25, 1, "min size");
560     SendMessage(hwTab, TCM_SETPADDING, 0, MAKELPARAM(4,4));
561     TabCheckSetSize(hwTab, 0, 1, 25, 1, "set padding, min size");
562
563     DestroyWindow (hwTab);
564
565     hwTab = create_tabcontrol(TCS_FIXEDWIDTH | TCS_BOTTOM, TCIF_TEXT|TCIF_IMAGE);
566     SendMessage(hwTab, TCM_SETMINTABWIDTH, 0, nMinTabWidth);
567
568     hdc = GetDC(hwTab);
569     dpi = GetDeviceCaps(hdc, LOGPIXELSX);
570     ReleaseDC(hwTab, hdc);
571     trace ("  TCS_FIXEDWIDTH | TCS_BOTTOM tabs...\n");
572     CheckSize(hwTab, dpi, -1, "no icon, default width");
573
574     TabCheckSetSize(hwTab, 20, 20, 20, 20, "no icon, set size 1");
575     TabCheckSetSize(hwTab, 10, 50, 10, 50, "no icon, set size 2");
576     TabCheckSetSize(hwTab, 0, 1, 0, 1, "no icon, min size");
577
578     SendMessage(hwTab, TCM_SETIMAGELIST, 0, (LPARAM)himl);
579
580     TabCheckSetSize(hwTab, 50, 30, 50, 30, "with icon, set size > icon");
581     TabCheckSetSize(hwTab, 20, 20, 25, 20, "with icon, set size < icon");
582     TabCheckSetSize(hwTab, 0, 1, 25, 1, "with icon, min size");
583     SendMessage(hwTab, TCM_SETPADDING, 0, MAKELPARAM(4,4));
584     TabCheckSetSize(hwTab, 0, 1, 25, 1, "set padding, min size");
585
586     DestroyWindow (hwTab);
587
588     hwTab = create_tabcontrol(0, TCIF_TEXT|TCIF_IMAGE);
589     SendMessage(hwTab, TCM_SETMINTABWIDTH, 0, nMinTabWidth);
590
591     trace ("  non fixed width, with text...\n");
592     exp = max(size.cx +TAB_PADDING_X*2, (nMinTabWidth < 0) ? DEFAULT_MIN_TAB_WIDTH : nMinTabWidth);
593     SendMessage( hwTab, TCM_GETITEMRECT, 0, (LPARAM)&rTab );
594     ok( rTab.right  - rTab.left == exp || broken(rTab.right  - rTab.left == DEFAULT_MIN_TAB_WIDTH),
595         "no icon, default width: Expected width [%d] got [%d]\n", exp, rTab.right - rTab.left );
596
597     for (i=0; i<8; i++)
598     {
599         INT nTabWidth = (nMinTabWidth < 0) ? TabWidthPadded(i, 2) : nMinTabWidth;
600
601         SendMessage(hwTab, TCM_SETIMAGELIST, 0, 0);
602         SendMessage(hwTab, TCM_SETPADDING, 0, MAKELPARAM(i,i));
603
604         TabCheckSetSize(hwTab, 50, 20, max(size.cx + i*2, nTabWidth), 20, "no icon, set size");
605         TabCheckSetSize(hwTab, 0, 1, max(size.cx + i*2, nTabWidth), 1, "no icon, min size");
606
607         SendMessage(hwTab, TCM_SETIMAGELIST, 0, (LPARAM)himl);
608         nTabWidth = (nMinTabWidth < 0) ? TabWidthPadded(i, 3) : nMinTabWidth;
609
610         TabCheckSetSize(hwTab, 50, 30, max(size.cx + 21 + i*3, nTabWidth), 30, "with icon, set size > icon");
611         TabCheckSetSize(hwTab, 20, 20, max(size.cx + 21 + i*3, nTabWidth), 20, "with icon, set size < icon");
612         TabCheckSetSize(hwTab, 0, 1, max(size.cx + 21 + i*3, nTabWidth), 1, "with icon, min size");
613     }
614     DestroyWindow (hwTab);
615
616     hwTab = create_tabcontrol(0, TCIF_IMAGE);
617     SendMessage(hwTab, TCM_SETMINTABWIDTH, 0, nMinTabWidth);
618
619     trace ("  non fixed width, no text...\n");
620     exp = (nMinTabWidth < 0) ? DEFAULT_MIN_TAB_WIDTH : nMinTabWidth;
621     SendMessage( hwTab, TCM_GETITEMRECT, 0, (LPARAM)&rTab );
622     ok( rTab.right  - rTab.left == exp || broken(rTab.right  - rTab.left == DEFAULT_MIN_TAB_WIDTH),
623         "no icon, default width: Expected width [%d] got [%d]\n", exp, rTab.right - rTab.left );
624
625     for (i=0; i<8; i++)
626     {
627         INT nTabWidth = (nMinTabWidth < 0) ? TabWidthPadded(i, 2) : nMinTabWidth;
628
629         SendMessage(hwTab, TCM_SETIMAGELIST, 0, 0);
630         SendMessage(hwTab, TCM_SETPADDING, 0, MAKELPARAM(i,i));
631
632         TabCheckSetSize(hwTab, 50, 20, nTabWidth, 20, "no icon, set size");
633         TabCheckSetSize(hwTab, 0, 1, nTabWidth, 1, "no icon, min size");
634
635         SendMessage(hwTab, TCM_SETIMAGELIST, 0, (LPARAM)himl);
636         if (i > 1 && nMinTabWidth > 0 && nMinTabWidth < DEFAULT_MIN_TAB_WIDTH)
637             nTabWidth += EXTRA_ICON_PADDING *(i-1);
638
639         TabCheckSetSize(hwTab, 50, 30, nTabWidth, 30, "with icon, set size > icon");
640         TabCheckSetSize(hwTab, 20, 20, nTabWidth, 20, "with icon, set size < icon");
641         TabCheckSetSize(hwTab, 0, 1, nTabWidth, 1, "with icon, min size");
642     }
643
644     DestroyWindow (hwTab);
645
646     ImageList_Destroy(himl);
647     DeleteObject(hFont);
648 }
649
650 static void test_getters_setters(HWND parent_wnd, INT nTabs)
651 {
652     HWND hTab;
653     RECT rTab;
654     INT nTabsRetrieved;
655     INT rowCount;
656     INT dpi;
657     HDC hdc;
658
659     ok(parent_wnd != NULL, "no parent window!\n");
660     flush_sequences(sequences, NUM_MSG_SEQUENCES);
661
662     hTab = createFilledTabControl(parent_wnd, TCS_FIXEDWIDTH, TCIF_TEXT|TCIF_IMAGE, nTabs);
663     ok(hTab != NULL, "Failed to create tab control\n");
664
665     if(!winetest_interactive)
666         ok_sequence(sequences, TAB_SEQ_INDEX, add_tab_to_parent,
667                     "Tab sequence, after adding tab control to parent", TRUE);
668     else
669         ok_sequence(sequences, TAB_SEQ_INDEX, add_tab_to_parent_interactive,
670                     "Tab sequence, after adding tab control to parent", TRUE);
671
672     if(!winetest_interactive)
673         ok_sequence(sequences, PARENT_SEQ_INDEX, add_tab_control_parent_seq,
674                     "Parent after sequence, adding tab control to parent", TRUE);
675     else
676         ok_sequence(sequences, PARENT_SEQ_INDEX, add_tab_control_parent_seq_interactive,
677                     "Parent after sequence, adding tab control to parent", TRUE);
678
679     flush_sequences(sequences, NUM_MSG_SEQUENCES);
680     ok(SendMessage(hTab, TCM_SETMINTABWIDTH, 0, -1) > 0,"TCM_SETMINTABWIDTH returned < 0\n");
681     ok_sequence(sequences, PARENT_SEQ_INDEX, empty_sequence, "Set minTabWidth test parent sequence", FALSE);
682
683     /* Testing GetItemCount */
684     flush_sequences(sequences, NUM_MSG_SEQUENCES);
685     nTabsRetrieved = SendMessage(hTab, TCM_GETITEMCOUNT, 0, 0);
686     expect(nTabs, nTabsRetrieved);
687     ok_sequence(sequences, TAB_SEQ_INDEX, get_item_count_seq, "Get itemCount test sequence", FALSE);
688     ok_sequence(sequences, PARENT_SEQ_INDEX, empty_sequence, "Getset itemCount test parent sequence", FALSE);
689
690     /* Testing GetRowCount */
691     flush_sequences(sequences, NUM_MSG_SEQUENCES);
692     rowCount = SendMessage(hTab, TCM_GETROWCOUNT, 0, 0);
693     expect(1, rowCount);
694     ok_sequence(sequences, TAB_SEQ_INDEX, get_row_count_seq, "Get rowCount test sequence", FALSE);
695     ok_sequence(sequences, PARENT_SEQ_INDEX, empty_sequence, "Get rowCount test parent sequence", FALSE);
696
697     /* Testing GetItemRect */
698     flush_sequences(sequences, NUM_MSG_SEQUENCES);
699     ok(SendMessage(hTab, TCM_GETITEMRECT, 0, (LPARAM) &rTab), "GetItemRect failed.\n");
700
701     hdc = GetDC(hTab);
702     dpi = GetDeviceCaps(hdc, LOGPIXELSX);
703     ReleaseDC(hTab, hdc);
704     CheckSize(hTab, dpi, -1 , "Default Width");
705     ok_sequence(sequences, TAB_SEQ_INDEX, get_item_rect_seq, "Get itemRect test sequence", FALSE);
706     ok_sequence(sequences, PARENT_SEQ_INDEX, empty_sequence, "Get itemRect test parent sequence", FALSE);
707
708     /* Testing CurFocus */
709     {
710         INT focusIndex;
711
712         flush_sequences(sequences, NUM_MSG_SEQUENCES);
713
714         /* Testing CurFocus with largest appropriate value */
715         SendMessage(hTab, TCM_SETCURFOCUS, nTabs-1, 0);
716         focusIndex = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
717             expect(nTabs-1, focusIndex);
718
719         /* Testing CurFocus with negative value */
720         SendMessage(hTab, TCM_SETCURFOCUS, -10, 0);
721         focusIndex = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
722             expect(-1, focusIndex);
723
724         /* Testing CurFocus with value larger than number of tabs */
725         focusIndex = SendMessage(hTab, TCM_SETCURSEL, 1, 0);
726         todo_wine{
727             expect(-1, focusIndex);
728         }
729
730         SendMessage(hTab, TCM_SETCURFOCUS, nTabs+1, 0);
731         focusIndex = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
732             expect(1, focusIndex);
733
734         ok_sequence(sequences, TAB_SEQ_INDEX, getset_cur_focus_seq, "Getset curFoc test sequence", FALSE);
735     }
736
737     /* Testing CurSel */
738     {
739         INT selectionIndex;
740         INT focusIndex;
741         TCITEM tcItem;
742
743         flush_sequences(sequences, NUM_MSG_SEQUENCES);
744
745         /* Testing CurSel with largest appropriate value */
746         selectionIndex = SendMessage(hTab, TCM_SETCURSEL, nTabs-1, 0);
747             expect(1, selectionIndex);
748         selectionIndex = SendMessage(hTab, TCM_GETCURSEL, 0, 0);
749             expect(nTabs-1, selectionIndex);
750
751         /* Focus should switch with selection */
752         focusIndex = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
753             expect(nTabs-1, focusIndex);
754
755         /* Testing CurSel with negative value */
756         SendMessage(hTab, TCM_SETCURSEL, -10, 0);
757         selectionIndex = SendMessage(hTab, TCM_GETCURSEL, 0, 0);
758             expect(-1, selectionIndex);
759
760         /* Testing CurSel with value larger than number of tabs */
761         selectionIndex = SendMessage(hTab, TCM_SETCURSEL, 1, 0);
762             expect(-1, selectionIndex);
763
764         selectionIndex = SendMessage(hTab, TCM_SETCURSEL, nTabs+1, 0);
765             expect(-1, selectionIndex);
766         selectionIndex = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
767             expect(1, selectionIndex);
768
769         ok_sequence(sequences, TAB_SEQ_INDEX, getset_cur_sel_seq, "Getset curSel test sequence", FALSE);
770         ok_sequence(sequences, PARENT_SEQ_INDEX, empty_sequence, "Getset curSel test parent sequence", FALSE);
771
772         /* selected item should have TCIS_BUTTONPRESSED state
773            It doesn't depend on button state */
774         memset(&tcItem, 0, sizeof(TCITEM));
775         tcItem.mask = TCIF_STATE;
776         tcItem.dwStateMask = TCIS_BUTTONPRESSED;
777         selectionIndex = SendMessage(hTab, TCM_GETCURSEL, 0, 0);
778         SendMessage(hTab, TCM_GETITEM, selectionIndex, (LPARAM) &tcItem);
779         ok (tcItem.dwState & TCIS_BUTTONPRESSED, "Selected item should have TCIS_BUTTONPRESSED\n");
780     }
781
782     /* Testing ExtendedStyle */
783     {
784         DWORD prevExtendedStyle;
785         DWORD extendedStyle;
786
787         flush_sequences(sequences, NUM_MSG_SEQUENCES);
788
789         /* Testing Flat Separators */
790         extendedStyle = SendMessage(hTab, TCM_GETEXTENDEDSTYLE, 0, 0);
791         prevExtendedStyle = SendMessage(hTab, TCM_SETEXTENDEDSTYLE, 0, TCS_EX_FLATSEPARATORS);
792         expect(extendedStyle, prevExtendedStyle);
793
794         extendedStyle = SendMessage(hTab, TCM_GETEXTENDEDSTYLE, 0, 0);
795         expect(TCS_EX_FLATSEPARATORS, extendedStyle);
796
797         /* Testing Register Drop */
798         prevExtendedStyle = SendMessage(hTab, TCM_SETEXTENDEDSTYLE, 0, TCS_EX_REGISTERDROP);
799             expect(extendedStyle, prevExtendedStyle);
800
801         extendedStyle = SendMessage(hTab, TCM_GETEXTENDEDSTYLE, 0, 0);
802         todo_wine{
803             expect(TCS_EX_REGISTERDROP, extendedStyle);
804         }
805
806         ok_sequence(sequences, TAB_SEQ_INDEX, getset_extended_style_seq, "Getset extendedStyle test sequence", FALSE);
807         ok_sequence(sequences, PARENT_SEQ_INDEX, empty_sequence, "Getset extendedStyle test parent sequence", FALSE);
808     }
809
810     /* Testing UnicodeFormat */
811     {
812         INT unicodeFormat;
813
814         flush_sequences(sequences, NUM_MSG_SEQUENCES);
815
816         unicodeFormat = SendMessage(hTab, TCM_SETUNICODEFORMAT, TRUE, 0);
817         todo_wine{
818             expect(0, unicodeFormat);
819         }
820         unicodeFormat = SendMessage(hTab, TCM_GETUNICODEFORMAT, 0, 0);
821             expect(1, unicodeFormat);
822
823         unicodeFormat = SendMessage(hTab, TCM_SETUNICODEFORMAT, FALSE, 0);
824             expect(1, unicodeFormat);
825         unicodeFormat = SendMessage(hTab, TCM_GETUNICODEFORMAT, 0, 0);
826             expect(0, unicodeFormat);
827
828         unicodeFormat = SendMessage(hTab, TCM_SETUNICODEFORMAT, TRUE, 0);
829             expect(0, unicodeFormat);
830
831         ok_sequence(sequences, TAB_SEQ_INDEX, getset_unicode_format_seq, "Getset unicodeFormat test sequence", FALSE);
832         ok_sequence(sequences, PARENT_SEQ_INDEX, empty_sequence, "Getset unicodeFormat test parent sequence", FALSE);
833     }
834
835     /* Testing GetSet Item */
836     {
837         TCITEM tcItem;
838         DWORD ret;
839         char szText[32] = "New Label";
840
841         /* TCM_GETITEM with null dest pointer */
842         ret = SendMessage(hTab, TCM_GETITEM, 0, (LPARAM)NULL);
843         expect(FALSE, ret);
844
845         /* passing invalid index should result in initialization to zero
846            for members mentioned in mask requested */
847
848         /* valid range here is [0,4] */
849         memset(&tcItem, 0xcc, sizeof(tcItem));
850         tcItem.mask = TCIF_PARAM;
851         ret = SendMessage(hTab, TCM_GETITEM, 5, (LPARAM)&tcItem);
852         expect(FALSE, ret);
853         ok(tcItem.lParam == 0, "Expected zero lParam, got %lu\n", tcItem.lParam);
854
855         memset(&tcItem, 0xcc, sizeof(tcItem));
856         tcItem.mask = TCIF_IMAGE;
857         ret = SendMessage(hTab, TCM_GETITEM, 5, (LPARAM)&tcItem);
858         expect(FALSE, ret);
859         expect(0, tcItem.iImage);
860
861         memset(&tcItem, 0xcc, sizeof(tcItem));
862         tcItem.mask = TCIF_TEXT;
863         tcItem.pszText = szText;
864         szText[0] = 'a';
865         ret = SendMessage(hTab, TCM_GETITEM, 5, (LPARAM)&tcItem);
866         expect(FALSE, ret);
867         expect('a', szText[0]);
868
869         memset(&tcItem, 0xcc, sizeof(tcItem));
870         tcItem.mask = TCIF_STATE;
871         tcItem.dwStateMask = 0;
872         tcItem.dwState = TCIS_BUTTONPRESSED;
873         ret = SendMessage(hTab, TCM_GETITEM, 5, (LPARAM)&tcItem);
874         expect(FALSE, ret);
875         ok(tcItem.dwState == 0, "Expected zero dwState, got %u\n", tcItem.dwState);
876
877         memset(&tcItem, 0xcc, sizeof(tcItem));
878         tcItem.mask = TCIF_STATE;
879         tcItem.dwStateMask = TCIS_BUTTONPRESSED;
880         tcItem.dwState = TCIS_BUTTONPRESSED;
881         ret = SendMessage(hTab, TCM_GETITEM, 5, (LPARAM)&tcItem);
882         expect(FALSE, ret);
883         ok(tcItem.dwState == 0, "Expected zero dwState\n");
884
885         /* check with negative index to be sure */
886         memset(&tcItem, 0xcc, sizeof(tcItem));
887         tcItem.mask = TCIF_PARAM;
888         ret = SendMessage(hTab, TCM_GETITEM, -1, (LPARAM)&tcItem);
889         expect(FALSE, ret);
890         ok(tcItem.lParam == 0, "Expected zero lParam, got %lu\n", tcItem.lParam);
891
892         memset(&tcItem, 0xcc, sizeof(tcItem));
893         tcItem.mask = TCIF_PARAM;
894         ret = SendMessage(hTab, TCM_GETITEM, -2, (LPARAM)&tcItem);
895         expect(FALSE, ret);
896         ok(tcItem.lParam == 0, "Expected zero lParam, got %lu\n", tcItem.lParam);
897
898         flush_sequences(sequences, NUM_MSG_SEQUENCES);
899
900         tcItem.mask = TCIF_TEXT;
901         tcItem.pszText = &szText[0];
902         tcItem.cchTextMax = sizeof(szText);
903
904         strcpy(szText, "New Label");
905         ok ( SendMessage(hTab, TCM_SETITEM, 0, (LPARAM) &tcItem), "Setting new item failed.\n");
906         ok ( SendMessage(hTab, TCM_GETITEM, 0, (LPARAM) &tcItem), "Getting item failed.\n");
907         expect_str("New Label", tcItem.pszText);
908
909         ok ( SendMessage(hTab, TCM_GETITEM, 1, (LPARAM) &tcItem), "Getting item failed.\n");
910         expect_str("Tab 2", tcItem.pszText);
911
912         ok_sequence(sequences, TAB_SEQ_INDEX, getset_item_seq, "Getset item test sequence", FALSE);
913         ok_sequence(sequences, PARENT_SEQ_INDEX, empty_sequence, "Getset item test parent sequence", FALSE);
914
915         /* TCIS_BUTTONPRESSED doesn't depend on tab style */
916         memset(&tcItem, 0, sizeof(tcItem));
917         tcItem.mask = TCIF_STATE;
918         tcItem.dwStateMask = TCIS_BUTTONPRESSED;
919         tcItem.dwState = TCIS_BUTTONPRESSED;
920         ok ( SendMessage(hTab, TCM_SETITEM, 0, (LPARAM) &tcItem), "Setting new item failed.\n");
921         tcItem.dwState = 0;
922         ok ( SendMessage(hTab, TCM_GETITEM, 0, (LPARAM) &tcItem), "Getting item failed.\n");
923         ok (tcItem.dwState == TCIS_BUTTONPRESSED, "TCIS_BUTTONPRESSED should be set.\n");
924         /* next highlight item, test that dwStateMask actually masks */
925         tcItem.mask = TCIF_STATE;
926         tcItem.dwStateMask = TCIS_HIGHLIGHTED;
927         tcItem.dwState = TCIS_HIGHLIGHTED;
928         ok ( SendMessage(hTab, TCM_SETITEM, 0, (LPARAM) &tcItem), "Setting new item failed.\n");
929         tcItem.dwState = 0;
930         ok ( SendMessage(hTab, TCM_GETITEM, 0, (LPARAM) &tcItem), "Getting item failed.\n");
931         ok (tcItem.dwState == TCIS_HIGHLIGHTED, "TCIS_HIGHLIGHTED should be set.\n");
932         tcItem.mask = TCIF_STATE;
933         tcItem.dwStateMask = TCIS_BUTTONPRESSED;
934         tcItem.dwState = 0;
935         ok ( SendMessage(hTab, TCM_GETITEM, 0, (LPARAM) &tcItem), "Getting item failed.\n");
936         ok (tcItem.dwState == TCIS_BUTTONPRESSED, "TCIS_BUTTONPRESSED should be set.\n");
937     }
938
939     /* Testing GetSet ToolTip */
940     {
941         HWND toolTip;
942         char toolTipText[32] = "ToolTip Text Test";
943
944         flush_sequences(sequences, NUM_MSG_SEQUENCES);
945
946         toolTip = create_tooltip(hTab, toolTipText);
947         SendMessage(hTab, TCM_SETTOOLTIPS, (LPARAM) toolTip, 0);
948         ok (toolTip == (HWND) SendMessage(hTab,TCM_GETTOOLTIPS,0,0), "ToolTip was set incorrectly.\n");
949
950         SendMessage(hTab, TCM_SETTOOLTIPS, 0, 0);
951         ok (NULL  == (HWND) SendMessage(hTab,TCM_GETTOOLTIPS,0,0), "ToolTip was set incorrectly.\n");
952
953         ok_sequence(sequences, TAB_SEQ_INDEX, getset_tooltip_seq, "Getset tooltip test sequence", TRUE);
954         ok_sequence(sequences, PARENT_SEQ_INDEX, getset_tooltip_parent_seq, "Getset tooltip test parent sequence", TRUE);
955     }
956
957     DestroyWindow(hTab);
958 }
959
960 static void test_adjustrect(HWND parent_wnd)
961 {
962     HWND hTab;
963     INT r;
964
965     ok(parent_wnd != NULL, "no parent window!\n");
966
967     hTab = createFilledTabControl(parent_wnd, TCS_FIXEDWIDTH, 0, 0);
968     ok(hTab != NULL, "Failed to create tab control\n");
969
970     r = SendMessage(hTab, TCM_ADJUSTRECT, FALSE, 0);
971     expect(-1, r);
972
973     r = SendMessage(hTab, TCM_ADJUSTRECT, TRUE, 0);
974     expect(-1, r);
975 }
976 static void test_insert_focus(HWND parent_wnd)
977 {
978     HWND hTab;
979     INT nTabsRetrieved;
980     INT r;
981     TCITEM tcNewTab;
982     DWORD mask = TCIF_TEXT|TCIF_IMAGE;
983     static char tabName[] = "TAB";
984     tcNewTab.mask = mask;
985     tcNewTab.pszText = tabName;
986
987     ok(parent_wnd != NULL, "no parent window!\n");
988
989     hTab = createFilledTabControl(parent_wnd, TCS_FIXEDWIDTH, mask, 0);
990     ok(hTab != NULL, "Failed to create tab control\n");
991
992     flush_sequences(sequences, NUM_MSG_SEQUENCES);
993
994     nTabsRetrieved = SendMessage(hTab, TCM_GETITEMCOUNT, 0, 0);
995     expect(0, nTabsRetrieved);
996
997     r = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
998     expect(-1, r);
999
1000     tcNewTab.iImage = 1;
1001     r = SendMessage(hTab, TCM_INSERTITEM, 1, (LPARAM) &tcNewTab);
1002     expect(0, r);
1003
1004     nTabsRetrieved = SendMessage(hTab, TCM_GETITEMCOUNT, 0, 0);
1005     expect(1, nTabsRetrieved);
1006
1007     r = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
1008     expect(0, r);
1009
1010     tcNewTab.iImage = 2;
1011     r = SendMessage(hTab, TCM_INSERTITEM, 2, (LPARAM) &tcNewTab);
1012     expect(1, r);
1013
1014     nTabsRetrieved = SendMessage(hTab, TCM_GETITEMCOUNT, 0, 0);
1015     expect(2, nTabsRetrieved);
1016
1017     r = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
1018     expect(0, r);
1019
1020     r = SendMessage(hTab, TCM_SETCURFOCUS, -1, 0);
1021     expect(0, r);
1022
1023     r = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
1024     expect(-1, r);
1025
1026     tcNewTab.iImage = 3;
1027     r = SendMessage(hTab, TCM_INSERTITEM, 3, (LPARAM) &tcNewTab);
1028     expect(2, r);
1029
1030     r = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
1031     expect(2, r);
1032
1033     ok_sequence(sequences, TAB_SEQ_INDEX, insert_focus_seq, "insert_focus test sequence", FALSE);
1034     ok_sequence(sequences, PARENT_SEQ_INDEX, empty_sequence, "insert_focus parent test sequence", FALSE);
1035
1036     DestroyWindow(hTab);
1037 }
1038
1039 static void test_delete_focus(HWND parent_wnd)
1040 {
1041     HWND hTab;
1042     INT nTabsRetrieved;
1043     INT r;
1044
1045     ok(parent_wnd != NULL, "no parent window!\n");
1046
1047     hTab = createFilledTabControl(parent_wnd, TCS_FIXEDWIDTH, TCIF_TEXT|TCIF_IMAGE, 2);
1048     ok(hTab != NULL, "Failed to create tab control\n");
1049
1050     flush_sequences(sequences, NUM_MSG_SEQUENCES);
1051
1052     nTabsRetrieved = SendMessage(hTab, TCM_GETITEMCOUNT, 0, 0);
1053     expect(2, nTabsRetrieved);
1054
1055     r = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
1056     expect(0, r);
1057
1058     r = SendMessage(hTab, TCM_DELETEITEM, 1, 0);
1059     expect(1, r);
1060
1061     nTabsRetrieved = SendMessage(hTab, TCM_GETITEMCOUNT, 0, 0);
1062     expect(1, nTabsRetrieved);
1063
1064     r = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
1065     expect(0, r);
1066
1067     r = SendMessage(hTab, TCM_SETCURFOCUS, -1, 0);
1068     expect(0, r);
1069
1070     r = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
1071     expect(-1, r);
1072
1073     r = SendMessage(hTab, TCM_DELETEITEM, 0, 0);
1074     expect(1, r);
1075
1076     nTabsRetrieved = SendMessage(hTab, TCM_GETITEMCOUNT, 0, 0);
1077     expect(0, nTabsRetrieved);
1078
1079     r = SendMessage(hTab, TCM_GETCURFOCUS, 0, 0);
1080     expect(-1, r);
1081
1082     ok_sequence(sequences, TAB_SEQ_INDEX, delete_focus_seq, "delete_focus test sequence", FALSE);
1083     ok_sequence(sequences, PARENT_SEQ_INDEX, empty_sequence, "delete_focus parent test sequence", FALSE);
1084
1085     DestroyWindow(hTab);
1086 }
1087
1088 static void test_removeimage(HWND parent_wnd)
1089 {
1090     static const BYTE bits[32];
1091     HWND hwTab;
1092     INT i;
1093     TCITEM item;
1094     HICON hicon;
1095     HIMAGELIST himl = ImageList_Create(16, 16, ILC_COLOR, 3, 4);
1096
1097     hicon = CreateIcon(NULL, 16, 16, 1, 1, bits, bits);
1098     ImageList_AddIcon(himl, hicon);
1099     ImageList_AddIcon(himl, hicon);
1100     ImageList_AddIcon(himl, hicon);
1101
1102     hwTab = create_tabcontrol(TCS_FIXEDWIDTH, TCIF_TEXT|TCIF_IMAGE);
1103     SendMessage(hwTab, TCM_SETIMAGELIST, 0, (LPARAM)himl);
1104
1105     memset(&item, 0, sizeof(TCITEM));
1106     item.mask = TCIF_IMAGE;
1107
1108     for(i = 0; i < 3; i++) {
1109         SendMessage(hwTab, TCM_GETITEM, i, (LPARAM)&item);
1110         expect(i, item.iImage);
1111     }
1112
1113     /* remove image middle image */
1114     SendMessage(hwTab, TCM_REMOVEIMAGE, 1, 0);
1115     expect(2, ImageList_GetImageCount(himl));
1116     item.iImage = -1;
1117     SendMessage(hwTab, TCM_GETITEM, 0, (LPARAM)&item);
1118     expect(0, item.iImage);
1119     item.iImage = 0;
1120     SendMessage(hwTab, TCM_GETITEM, 1, (LPARAM)&item);
1121     expect(-1, item.iImage);
1122     item.iImage = 0;
1123     SendMessage(hwTab, TCM_GETITEM, 2, (LPARAM)&item);
1124     expect(1, item.iImage);
1125     /* remove first image */
1126     SendMessage(hwTab, TCM_REMOVEIMAGE, 0, 0);
1127     expect(1, ImageList_GetImageCount(himl));
1128     item.iImage = 0;
1129     SendMessage(hwTab, TCM_GETITEM, 0, (LPARAM)&item);
1130     expect(-1, item.iImage);
1131     item.iImage = 0;
1132     SendMessage(hwTab, TCM_GETITEM, 1, (LPARAM)&item);
1133     expect(-1, item.iImage);
1134     item.iImage = -1;
1135     SendMessage(hwTab, TCM_GETITEM, 2, (LPARAM)&item);
1136     expect(0, item.iImage);
1137     /* remove the last one */
1138     SendMessage(hwTab, TCM_REMOVEIMAGE, 0, 0);
1139     expect(0, ImageList_GetImageCount(himl));
1140     for(i = 0; i < 3; i++) {
1141         item.iImage = 0;
1142         SendMessage(hwTab, TCM_GETITEM, i, (LPARAM)&item);
1143         expect(-1, item.iImage);
1144     }
1145
1146     DestroyWindow(hwTab);
1147     ImageList_Destroy(himl);
1148     DestroyIcon(hicon);
1149 }
1150
1151 START_TEST(tab)
1152 {
1153     HWND parent_wnd;
1154     LOGFONTA logfont;
1155
1156     lstrcpyA(logfont.lfFaceName, "Arial");
1157     memset(&logfont, 0, sizeof(logfont));
1158     logfont.lfHeight = -12;
1159     logfont.lfWeight = FW_NORMAL;
1160     logfont.lfCharSet = ANSI_CHARSET;
1161     hFont = CreateFontIndirectA(&logfont);
1162
1163     InitCommonControls();
1164
1165     trace ("Testing with default MinWidth\n");
1166     test_tab(-1);
1167     trace ("Testing with MinWidth set to -3\n");
1168     test_tab(-3);
1169     trace ("Testing with MinWidth set to 24\n");
1170     test_tab(24);
1171     trace ("Testing with MinWidth set to 54\n");
1172     test_tab(54);
1173     trace ("Testing with MinWidth set to 94\n");
1174     test_tab(94);
1175
1176     init_msg_sequences(sequences, NUM_MSG_SEQUENCES);
1177
1178     parent_wnd = createParentWindow();
1179     ok(parent_wnd != NULL, "Failed to create parent window!\n");
1180
1181     /* Testing getters and setters with 5 tabs */
1182     test_getters_setters(parent_wnd, 5);
1183
1184     test_adjustrect(parent_wnd);
1185
1186     test_insert_focus(parent_wnd);
1187     test_delete_focus(parent_wnd);
1188     test_removeimage(parent_wnd);
1189
1190     DestroyWindow(parent_wnd);
1191 }