1 /* Unit tests for toolbar.
2  *
3  * Copyright 2005 Krzysztof Foltman
4  * Copyright 2007 Mikolaj Zalewski
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 <stdarg.h>
22 
23 #include "windef.h"
24 #include "winbase.h"
25 #include "wingdi.h"
26 #include "winuser.h"
27 #include "winnls.h"
28 #include "winreg.h"
29 #include "commctrl.h"
30 
31 #include "resources.h"
32 
33 #include "wine/test.h"
34 
35 #include "msg.h"
36 
37 #define PARENT_SEQ_INDEX       0
38 #define NUM_MSG_SEQUENCES      1
39 
40 static HWND (WINAPI *pCreateToolbarEx)(HWND, DWORD, UINT, INT, HINSTANCE, UINT_PTR, const TBBUTTON *,
41     INT, INT, INT, INT, INT, UINT);
42 static BOOL (WINAPI *pImageList_Destroy)(HIMAGELIST);
43 static INT (WINAPI *pImageList_GetImageCount)(HIMAGELIST);
44 static BOOL (WINAPI *pImageList_GetIconSize)(HIMAGELIST, int *, int *);
45 static HIMAGELIST (WINAPI *pImageList_LoadImageA)(HINSTANCE, LPCSTR, int, int, COLORREF, UINT, UINT);
46 
47 static struct msg_sequence *sequences[NUM_MSG_SEQUENCES];
48 
49 static HWND hMainWnd;
50 static BOOL g_fBlockHotItemChange;
51 static BOOL g_fReceivedHotItemChange;
52 static BOOL g_fExpectedHotItemOld;
53 static BOOL g_fExpectedHotItemNew;
54 static DWORD g_dwExpectedDispInfoMask;
55 static BOOL g_ResetDispTextPtr;
56 
57 static const struct message ttgetdispinfo_parent_seq[] = {
58     { WM_NOTIFY, sent|id, 0, 0, TBN_GETINFOTIPA },
59     { WM_NOTIFY, sent|id, 0, 0, TTN_GETDISPINFOA },
60     { 0 }
61 };
62 
63 static const struct message save_parent_seq[] = {
64     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, -1 },
65     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 0 },
66     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 1 },
67     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 2 },
68     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 3 },
69     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 4 },
70     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 5 },
71     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 6 },
72     { 0 }
73 };
74 
75 static const struct message restore_parent_seq[] = {
76     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, -1 },
77     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 0 },
78     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 1 },
79     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 2 },
80     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 3 },
81     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 4 },
82     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 5 },
83     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 6 },
84     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 7 },
85     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 8 },
86     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 9 },
87     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 0xa },
88     { WM_NOTIFY, sent|id, 0, 0, TBN_BEGINADJUST },
89     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_GETBUTTONINFOA, 0 },
90     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_GETBUTTONINFOA, 1 },
91     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_GETBUTTONINFOA, 2 },
92     { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_GETBUTTONINFOA, 3 },
93     { WM_NOTIFY, sent|id, 0, 0, TBN_ENDADJUST },
94     { 0 }
95 };
96 
97 #define DEFINE_EXPECT(func) \
98     static BOOL expect_ ## func = FALSE, called_ ## func = FALSE
99 
100 #define CHECK_EXPECT2(func) \
101     do { \
102         ok(expect_ ##func, "unexpected call " #func "\n"); \
103         called_ ## func = TRUE; \
104     }while(0)
105 
106 #define CHECK_CALLED(func) \
107     do { \
108         ok(called_ ## func, "expected " #func "\n"); \
109         expect_ ## func = called_ ## func = FALSE; \
110     }while(0)
111 
112 #define SET_EXPECT(func) \
113     expect_ ## func = TRUE
114 
115 #define expect(EXPECTED,GOT) ok((GOT)==(EXPECTED), "Expected %d, got %d\n", (EXPECTED), (GOT))
116 
117 #define check_rect(name, val, exp, ...) ok(EqualRect(&val, &exp), \
118     "invalid rect %s - expected %s - (" name ")\n", \
119     wine_dbgstr_rect(&val), wine_dbgstr_rect(&exp), __VA_ARGS__);
120 
121 #define compare(val, exp, format) ok((val) == (exp), #val " value " format " expected " format "\n", (val), (exp));
122 
123 #define check_button_size(handle, width, height, ...) {\
124     LRESULT bsize = SendMessageA(handle, TB_GETBUTTONSIZE, 0, 0);\
125     ok(bsize == MAKELONG(width, height), "Unexpected button size - got size (%d, %d), expected (%d, %d)\n", LOWORD(bsize), HIWORD(bsize), width, height);\
126     }
127 
128 static void MakeButton(TBBUTTON *p, int idCommand, int fsStyle, int nString) {
129   p->iBitmap = -2;
130   p->idCommand = idCommand;
131   p->fsState = TBSTATE_ENABLED;
132   p->fsStyle = fsStyle;
133   p->iString = nString;
134 }
135 
136 static void *alloced_str;
137 
138 static LRESULT parent_wnd_notify(LPARAM lParam)
139 {
140     NMHDR *hdr = (NMHDR *)lParam;
141     NMTBHOTITEM *nmhi;
142     NMTBDISPINFOA *nmdisp;
143     switch (hdr->code)
144     {
145         case TBN_HOTITEMCHANGE:
146             nmhi = (NMTBHOTITEM *)lParam;
147             g_fReceivedHotItemChange = TRUE;
148             if (g_fExpectedHotItemOld != g_fExpectedHotItemNew)
149             {
150                 compare(nmhi->idOld, g_fExpectedHotItemOld, "%d");
151                 compare(nmhi->idNew, g_fExpectedHotItemNew, "%d");
152             }
153             if (g_fBlockHotItemChange)
154                 return 1;
155             break;
156 
157         case TBN_GETDISPINFOA:
158             ok(FALSE, "TBN_GETDISPINFOA received\n");
159             break;
160 
161         case TBN_GETINFOTIPA:
162         {
163             NMTBGETINFOTIPA *tbgit = (NMTBGETINFOTIPA*)lParam;
164 
165             if (g_ResetDispTextPtr)
166             {
167                 tbgit->pszText = NULL;
168                 return 0;
169             }
170             break;
171         }
172         case TBN_GETDISPINFOW:
173             nmdisp = (NMTBDISPINFOA *)lParam;
174 
175             compare(nmdisp->dwMask, g_dwExpectedDispInfoMask, "%x");
176             ok(nmdisp->pszText == NULL, "pszText is not NULL\n");
177         break;
178         case TBN_SAVE:
179         {
180             NMTBSAVE *save = (NMTBSAVE *)lParam;
181             if (save->iItem == -1)
182             {
183                 save->cbData = save->cbData * 2 + 11 * sizeof(DWORD);
184                 save->pData = heap_alloc( save->cbData );
185                 save->pData[0] = 0xcafe;
186                 save->pCurrent = save->pData + 1;
187             }
188             else
189             {
190                 save->pCurrent[0] = 0xcafe0000 + save->iItem;
191                 save->pCurrent++;
192             }
193 
194             /* Add on 5 more pseudo buttons. */
195             if (save->iItem == save->cButtons - 1)
196             {
197                 save->pCurrent[0] = 0xffffffff;
198                 save->pCurrent[1] = 0xcafe0007;
199                 save->pCurrent[2] = 0xfffffffe;
200                 save->pCurrent[3] = 0xcafe0008;
201                 save->pCurrent[4] = 0x80000000;
202                 save->pCurrent[5] = 0xcafe0009;
203                 save->pCurrent[6] = 0x7fffffff;
204                 save->pCurrent[7] = 0xcafe000a;
205                 save->pCurrent[8] = 0x100;
206                 save->pCurrent[9] = 0xcafe000b;
207             }
208 
209             /* Return value is ignored */
210             return 1;
211         }
212         case TBN_RESTORE:
213         {
214             NMTBRESTORE *restore = (NMTBRESTORE *)lParam;
215 
216             if (restore->iItem == -1)
217             {
218                 ok( restore->cButtons == 25, "got %d\n", restore->cButtons );
219                 ok( *restore->pCurrent == 0xcafe, "got %08x\n", *restore->pCurrent );
220                 /* Skip the last one */
221                 restore->cButtons = 11;
222                 restore->pCurrent++;
223                 /* BytesPerRecord is ignored */
224                 restore->cbBytesPerRecord = 10;
225             }
226             else
227             {
228                 ok( *restore->pCurrent == 0xcafe0000 + restore->iItem, "got %08x\n", *restore->pCurrent );
229                 if (restore->iItem < 7 || restore->iItem == 10)
230                 {
231                     ok( restore->tbButton.iBitmap == -1, "got %08x\n", restore->tbButton.iBitmap );
232                     if (restore->iItem < 7)
233                         ok( restore->tbButton.idCommand == restore->iItem * 2 + 1, "%d: got %08x\n", restore->iItem, restore->tbButton.idCommand );
234                     else
235                         ok( restore->tbButton.idCommand == 0x7fffffff, "%d: got %08x\n", restore->iItem, restore->tbButton.idCommand );
236                     ok( restore->tbButton.fsState == 0, "%d: got %02x\n", restore->iItem, restore->tbButton.fsState );
237                     ok( restore->tbButton.fsStyle == 0, "%d: got %02x\n", restore->iItem, restore->tbButton.fsStyle );
238                 }
239                 else
240                 {
241                     ok( restore->tbButton.iBitmap == 8, "got %08x\n", restore->tbButton.iBitmap );
242                     ok( restore->tbButton.idCommand == 0, "%d: got %08x\n", restore->iItem, restore->tbButton.idCommand );
243                     if (restore->iItem == 7)
244                         ok( restore->tbButton.fsState == 0, "%d: got %02x\n", restore->iItem, restore->tbButton.fsState );
245                     else
246                         ok( restore->tbButton.fsState == TBSTATE_HIDDEN, "%d: got %02x\n", restore->iItem, restore->tbButton.fsState );
247                     ok( restore->tbButton.fsStyle == BTNS_SEP, "%d: got %02x\n", restore->iItem, restore->tbButton.fsStyle );
248                 }
249 
250                 ok( restore->tbButton.dwData == 0, "got %08lx\n", restore->tbButton.dwData );
251                 ok( restore->tbButton.iString == 0, "got %08lx\n", restore->tbButton.iString );
252 
253                 restore->tbButton.iBitmap = 0;
254                 restore->tbButton.fsState = TBSTATE_ENABLED;
255                 restore->tbButton.fsStyle = 0;
256                 restore->tbButton.dwData = restore->iItem;
257 
258                 if (restore->iItem == 0)
259                 {
260                     restore->tbButton.iString = (INT_PTR)heap_alloc_zero( 8 );
261                     strcpy( (char *)restore->tbButton.iString, "foo" );
262                 }
263                 else if (restore->iItem == 1)
264                     restore->tbButton.iString = 2;
265                 else
266                     restore->tbButton.iString = -1;
267 
268                 restore->pCurrent++;
269                 /* Altering cButtons after the 1st call makes no difference. */
270                 restore->cButtons--;
271             }
272 
273             /* Returning non-zero from the 1st call aborts the restore,
274                otherwise the return value is ignored. */
275             if (restore->iItem == -1) return 0;
276             return 1;
277         }
278         case TBN_GETBUTTONINFOA:
279         {
280             NMTOOLBARA *tb = (NMTOOLBARA *)lParam;
281             tb->tbButton.iBitmap = 0;
282             tb->tbButton.fsState = 0;
283             tb->tbButton.fsStyle = 0;
284             tb->tbButton.dwData = 0;
285             ok( tb->cchText == 128, "got %d\n", tb->cchText );
286             switch (tb->iItem)
287             {
288             case 0:
289                 tb->tbButton.idCommand = 7;
290                 alloced_str = heap_alloc_zero( 8 );
291                 strcpy( alloced_str, "foo" );
292                 tb->tbButton.iString = (INT_PTR)alloced_str;
293                 return 1;
294             case 1:
295                 tb->tbButton.idCommand = 9;
296                 tb->tbButton.iString = 0;
297                 /* tb->pszText is ignored */
298                 strcpy( tb->pszText, "foo" );
299                 return 1;
300            case 2:
301                 tb->tbButton.idCommand = 11;
302                 tb->tbButton.iString = 3;
303                 return 1;
304             }
305             return 0;
306         }
307     }
308     return 0;
309 }
310 
311 static LRESULT CALLBACK parent_wnd_proc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
312 {
313     static LONG defwndproc_counter = 0;
314     struct message msg;
315     LRESULT ret;
316 
317     msg.message = message;
318     msg.flags = sent|wparam|lparam;
319     if (defwndproc_counter) msg.flags |= defwinproc;
320     msg.wParam = wParam;
321     msg.lParam = lParam;
322     if (message == WM_NOTIFY && lParam)
323     {
324         msg.id = ((NMHDR*)lParam)->code;
325         switch (msg.id)
326         {
327         case TBN_SAVE:
328         {
329             NMTBSAVE *save = (NMTBSAVE *)lParam;
330             msg.stage = save->iItem;
331         }
332         break;
333         case TBN_RESTORE:
334         {
335             NMTBRESTORE *restore = (NMTBRESTORE *)lParam;
336             msg.stage = restore->iItem;
337         }
338         break;
339         case TBN_GETBUTTONINFOA:
340         {
341             NMTOOLBARA *tb = (NMTOOLBARA *)lParam;
342             msg.stage = tb->iItem;
343         }
344         break;
345         }
346     }
347 
348     /* log system messages, except for painting */
349     if (message < WM_USER &&
350         message != WM_PAINT &&
351         message != WM_ERASEBKGND &&
352         message != WM_NCPAINT &&
353         message != WM_NCHITTEST &&
354         message != WM_GETTEXT &&
355         message != WM_GETICON &&
356         message != WM_DEVICECHANGE)
357     {
358         add_message(sequences, PARENT_SEQ_INDEX, &msg);
359     }
360 
361     switch (message)
362     {
363         case WM_NOTIFY:
364             return parent_wnd_notify(lParam);
365     }
366 
367     defwndproc_counter++;
368     ret = DefWindowProcA(hWnd, message, wParam, lParam);
369     defwndproc_counter--;
370 
371     return ret;
372 }
373 
374 static void basic_test(void)
375 {
376     TBBUTTON buttons[9];
377     HWND hToolbar;
378     int i;
379 
380     for (i=0; i<9; i++)
381         MakeButton(buttons+i, 1000+i, TBSTYLE_CHECKGROUP, 0);
382     MakeButton(buttons+3, 1003, TBSTYLE_SEP|TBSTYLE_GROUP, 0);
383     MakeButton(buttons+6, 1006, TBSTYLE_SEP, 0);
384 
385     hToolbar = pCreateToolbarEx(hMainWnd,
386         WS_VISIBLE | WS_CLIPCHILDREN | CCS_TOP |
387         WS_CHILD | TBSTYLE_LIST,
388         100,
389         0, NULL, 0,
390         buttons, ARRAY_SIZE(buttons),
391         0, 0, 20, 16, sizeof(TBBUTTON));
392     ok(hToolbar != NULL, "Toolbar creation\n");
393     SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)"test\000");
394 
395     /* test for exclusion working inside a separator-separated :-) group */
396     SendMessageA(hToolbar, TB_CHECKBUTTON, 1000, 1); /* press A1 */
397     ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 pressed\n");
398     ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1001, 0), "A2 not pressed\n");
399 
400     SendMessageA(hToolbar, TB_CHECKBUTTON, 1004, 1); /* press A5, release A1 */
401     ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1004, 0), "A5 pressed\n");
402     ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 not pressed anymore\n");
403 
404     SendMessageA(hToolbar, TB_CHECKBUTTON, 1005, 1); /* press A6, release A5 */
405     ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1005, 0), "A6 pressed\n");
406     ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1004, 0), "A5 not pressed anymore\n");
407 
408     /* test for inter-group crosstalk, i.e. two radio groups interfering with each other */
409     SendMessageA(hToolbar, TB_CHECKBUTTON, 1007, 1); /* press B2 */
410     ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1005, 0), "A6 still pressed, no inter-group crosstalk\n");
411     ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 still not pressed\n");
412     ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1007, 0), "B2 pressed\n");
413 
414     SendMessageA(hToolbar, TB_CHECKBUTTON, 1000, 1); /* press A1 and ensure B group didn't suffer */
415     ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1005, 0), "A6 not pressed anymore\n");
416     ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 pressed\n");
417     ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1007, 0), "B2 still pressed\n");
418 
419     SendMessageA(hToolbar, TB_CHECKBUTTON, 1008, 1); /* press B3, and ensure A group didn't suffer */
420     ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1005, 0), "A6 pressed\n");
421     ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 pressed\n");
422     ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1007, 0), "B2 not pressed\n");
423     ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1008, 0), "B3 pressed\n");
424 
425     /* tests with invalid index */
426     compare(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 0xdeadbeef, 0), -1L, "%ld");
427     compare(SendMessageA(hToolbar, TB_ISBUTTONPRESSED, 0xdeadbeef, 0), -1L, "%ld");
428     compare(SendMessageA(hToolbar, TB_ISBUTTONENABLED, 0xdeadbeef, 0), -1L, "%ld");
429     compare(SendMessageA(hToolbar, TB_ISBUTTONINDETERMINATE, 0xdeadbeef, 0), -1L, "%ld");
430     compare(SendMessageA(hToolbar, TB_ISBUTTONHIGHLIGHTED, 0xdeadbeef, 0), -1L, "%ld");
431     compare(SendMessageA(hToolbar, TB_ISBUTTONHIDDEN, 0xdeadbeef, 0), -1L, "%ld");
432 
433     DestroyWindow(hToolbar);
434 }
435 
436 static void rebuild_toolbar(HWND *hToolbar)
437 {
438     if (*hToolbar)
439         DestroyWindow(*hToolbar);
440     *hToolbar = CreateWindowExA(0, TOOLBARCLASSNAMEA, NULL, WS_CHILD | WS_VISIBLE, 0, 0, 0, 0,
441         hMainWnd, (HMENU)5, GetModuleHandleA(NULL), NULL);
442     ok(*hToolbar != NULL, "Toolbar creation problem\n");
443     ok(SendMessageA(*hToolbar, TB_BUTTONSTRUCTSIZE, sizeof(TBBUTTON), 0) == 0, "TB_BUTTONSTRUCTSIZE failed\n");
444     ok(SendMessageA(*hToolbar, TB_AUTOSIZE, 0, 0) == 0, "TB_AUTOSIZE failed\n");
445     ok(SendMessageA(*hToolbar, WM_SETFONT, (WPARAM)GetStockObject(SYSTEM_FONT), 0)==1, "WM_SETFONT\n");
446 }
447 
448 static void rebuild_toolbar_with_buttons(HWND *hToolbar)
449 {
450     TBBUTTON buttons[5];
451     rebuild_toolbar(hToolbar);
452 
453     ZeroMemory(&buttons, sizeof(buttons));
454     buttons[0].idCommand = 1;
455     buttons[0].fsStyle = BTNS_BUTTON;
456     buttons[0].fsState = TBSTATE_ENABLED;
457     buttons[0].iString = -1;
458     buttons[1].idCommand = 3;
459     buttons[1].fsStyle = BTNS_BUTTON;
460     buttons[1].fsState = TBSTATE_ENABLED;
461     buttons[1].iString = -1;
462     buttons[2].idCommand = 5;
463     buttons[2].fsStyle = BTNS_SEP;
464     buttons[2].fsState = TBSTATE_ENABLED;
465     buttons[2].iString = -1;
466     buttons[3].idCommand = 7;
467     buttons[3].fsStyle = BTNS_BUTTON;
468     buttons[3].fsState = TBSTATE_ENABLED;
469     buttons[3].iString = -1;
470     buttons[4].idCommand = 9;
471     buttons[4].fsStyle = BTNS_BUTTON;
472     buttons[4].fsState = 0;  /* disabled */
473     buttons[4].iString = -1;
474     ok(SendMessageA(*hToolbar, TB_ADDBUTTONSA, 5, (LPARAM)buttons) == 1, "TB_ADDBUTTONSA failed\n");
475     ok(SendMessageA(*hToolbar, TB_AUTOSIZE, 0, 0) == 0, "TB_AUTOSIZE failed\n");
476 }
477 
478 static void add_128x15_bitmap(HWND hToolbar, int nCmds)
479 {
480     TBADDBITMAP bmp128;
481     bmp128.hInst = GetModuleHandleA(NULL);
482     bmp128.nID = IDB_BITMAP_128x15;
483     ok(SendMessageA(hToolbar, TB_ADDBITMAP, nCmds, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n");
484 }
485 
486 #define CHECK_IMAGELIST(count, dx, dy) { \
487     int cx, cy; \
488     HIMAGELIST himl = (HIMAGELIST)SendMessageA(hToolbar, TB_GETIMAGELIST, 0, 0); \
489     ok(himl != NULL, "No image list\n"); \
490     if (himl != NULL) {\
491         ok(pImageList_GetImageCount(himl) == count, "Images count mismatch - %d vs %d\n", count, pImageList_GetImageCount(himl)); \
492         pImageList_GetIconSize(himl, &cx, &cy); \
493         ok(cx == dx && cy == dy, "Icon size mismatch - %dx%d vs %dx%d\n", dx, dy, cx, cy); \
494     } \
495 }
496 
497 static void test_add_bitmap(void)
498 {
499     TBADDBITMAP stdsmall, std;
500     HWND hToolbar = NULL;
501     TBADDBITMAP bmp128;
502     TBADDBITMAP bmp80;
503     TBADDBITMAP addbmp;
504     HIMAGELIST himl;
505     INT ret, id;
506 
507     /* Test default bitmaps range */
508     for (id = IDB_STD_SMALL_COLOR; id < IDB_HIST_LARGE_COLOR; id++)
509     {
510         HIMAGELIST himl;
511         int cx, cy, count;
512 
513         rebuild_toolbar(&hToolbar);
514 
515         std.hInst = HINST_COMMCTRL;
516         std.nID = id;
517 
518         ret = SendMessageA(hToolbar, TB_ADDBITMAP, 0, (LPARAM)&std);
519         ok(ret == 0, "Got %d\n", ret);
520 
521         himl = (HIMAGELIST)SendMessageA(hToolbar, TB_GETIMAGELIST, 0, 0);
522         ok(himl != NULL, "Got %p\n", himl);
523 
524         ret = pImageList_GetIconSize(himl, &cx, &cy);
525         ok(ret, "Got %d\n", ret);
526         ok(cx == cy, "Got %d x %d\n", cx, cy);
527 
528         count = pImageList_GetImageCount(himl);
529 
530         /* Image count */
531         switch (id)
532         {
533         case IDB_STD_SMALL_COLOR:
534         case IDB_STD_LARGE_COLOR:
535         case 2:
536         case 3:
537             ok(count == 15, "got count %d\n", count);
538             break;
539         case IDB_VIEW_SMALL_COLOR:
540         case IDB_VIEW_LARGE_COLOR:
541         case 6:
542         case 7:
543             ok(count == 12, "got count %d\n", count);
544             break;
545         case IDB_HIST_SMALL_COLOR:
546         case IDB_HIST_LARGE_COLOR:
547             ok(count == 5, "got count %d\n", count);
548             break;
549         default:
550             ok(0, "id %d, count %d\n", id, count);
551         }
552 
553         /* Image sizes */
554         switch (id)
555         {
556         case IDB_STD_SMALL_COLOR:
557         case 2:
558         case IDB_VIEW_SMALL_COLOR:
559         case 6:
560         case IDB_HIST_SMALL_COLOR:
561             ok(cx == 16, "got size %d\n", cx);
562             break;
563         case IDB_STD_LARGE_COLOR:
564         case 3:
565         case IDB_VIEW_LARGE_COLOR:
566         case 7:
567         case IDB_HIST_LARGE_COLOR:
568             ok(cx == 24, "got size %d\n", cx);
569             break;
570         default:
571             ok(0, "id %d, size %d\n", id, cx);
572         }
573     }
574 
575     /* empty 128x15 bitmap */
576     bmp128.hInst = GetModuleHandleA(NULL);
577     bmp128.nID = IDB_BITMAP_128x15;
578 
579     /* empty 80x15 bitmap */
580     bmp80.hInst = GetModuleHandleA(NULL);
581     bmp80.nID = IDB_BITMAP_80x15;
582 
583     /* standard bitmap - 240x15 pixels */
584     stdsmall.hInst = HINST_COMMCTRL;
585     stdsmall.nID = IDB_STD_SMALL_COLOR;
586 
587     rebuild_toolbar(&hToolbar);
588     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 8, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n");
589     CHECK_IMAGELIST(8, 16, 16);
590 
591     /* adding more bitmaps */
592     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp80) == 8, "TB_ADDBITMAP - unexpected return\n");
593     CHECK_IMAGELIST(13, 16, 16);
594     /* adding the same bitmap will simply return the index of the already loaded block */
595     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 8, (LPARAM)&bmp128);
596     ok(ret == 0, "TB_ADDBITMAP - unexpected return %d\n", ret);
597     CHECK_IMAGELIST(13, 16, 16);
598     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp80);
599     ok(ret == 8, "TB_ADDBITMAP - unexpected return %d\n", ret);
600     CHECK_IMAGELIST(13, 16, 16);
601     /* even if we increase the wParam */
602     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 55, (LPARAM)&bmp80);
603     ok(ret == 8, "TB_ADDBITMAP - unexpected return %d\n", ret);
604     CHECK_IMAGELIST(13, 16, 16);
605 
606     /* when the wParam is smaller than the bitmaps count but non-zero, all the bitmaps will be added*/
607     rebuild_toolbar(&hToolbar);
608     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 3, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n");
609     CHECK_IMAGELIST(8, 16, 16);
610     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp80);
611     ok(ret == 3, "TB_ADDBITMAP - unexpected return %d\n", ret);
612     /* the returned value is misleading - id 8 is the id of the first icon from bmp80 */
613     CHECK_IMAGELIST(13, 16, 16);
614 
615     /* the same for negative wParam */
616     rebuild_toolbar(&hToolbar);
617     ret = SendMessageA(hToolbar, TB_ADDBITMAP, -143, (LPARAM)&bmp128);
618     ok(ret == 0, "TB_ADDBITMAP - unexpected return %d\n", ret);
619     CHECK_IMAGELIST(8, 16, 16);
620     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&bmp80);
621     ok(ret == -143, "TB_ADDBITMAP - unexpected return %d\n", ret);
622     CHECK_IMAGELIST(13, 16, 16);
623 
624     /* for zero only one bitmap will be added */
625     rebuild_toolbar(&hToolbar);
626     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 0, (LPARAM)&bmp80);
627     ok(ret == 0, "TB_ADDBITMAP - unexpected return %d\n", ret);
628     CHECK_IMAGELIST(1, 16, 16);
629 
630     /* if wParam is larger than the amount of icons, the list is grown */
631     rebuild_toolbar(&hToolbar);
632     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 100, (LPARAM)&bmp80) == 0, "TB_ADDBITMAP - unexpected return\n");
633     CHECK_IMAGELIST(100, 16, 16);
634     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 100, (LPARAM)&bmp128);
635     ok(ret == 100, "TB_ADDBITMAP - unexpected return %d\n", ret);
636     CHECK_IMAGELIST(200, 16, 16);
637 
638     /* adding built-in items - the wParam is ignored */
639     rebuild_toolbar(&hToolbar);
640     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp80) == 0, "TB_ADDBITMAP - unexpected return\n");
641     CHECK_IMAGELIST(5, 16, 16);
642     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 0, (LPARAM)&stdsmall) == 5, "TB_ADDBITMAP - unexpected return\n");
643     CHECK_IMAGELIST(20, 16, 16);
644     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp128) == 20, "TB_ADDBITMAP - unexpected return\n");
645     CHECK_IMAGELIST(28, 16, 16);
646 
647     /* when we increase the bitmap size, less icons will be created */
648     rebuild_toolbar(&hToolbar);
649     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(20, 20)) == TRUE, "TB_SETBITMAPSIZE failed\n");
650     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n");
651     CHECK_IMAGELIST(6, 20, 20);
652     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&bmp80);
653     ok(ret == 1, "TB_ADDBITMAP - unexpected return %d\n", ret);
654     CHECK_IMAGELIST(10, 20, 20);
655     /* the icons can be resized - an UpdateWindow is needed as this probably happens during WM_PAINT */
656     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(8, 8)) == TRUE, "TB_SETBITMAPSIZE failed\n");
657     UpdateWindow(hToolbar);
658     CHECK_IMAGELIST(26, 8, 8);
659     /* loading a standard bitmaps automatically resizes the icons */
660     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&stdsmall) == 2, "TB_ADDBITMAP - unexpected return\n");
661     UpdateWindow(hToolbar);
662     CHECK_IMAGELIST(28, 16, 16);
663 
664     /* two more SETBITMAPSIZE tests */
665     rebuild_toolbar(&hToolbar);
666     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 100, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n");
667     CHECK_IMAGELIST(100, 16, 16);
668     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 100, (LPARAM)&bmp80) == 100, "TB_ADDBITMAP - unexpected return\n");
669     CHECK_IMAGELIST(200, 16, 16);
670     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(8, 8)) == TRUE, "TB_SETBITMAPSIZE failed\n");
671     UpdateWindow(hToolbar);
672     CHECK_IMAGELIST(200, 8, 8);
673     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(30, 30)) == TRUE, "TB_SETBITMAPSIZE failed\n");
674     UpdateWindow(hToolbar);
675     CHECK_IMAGELIST(200, 30, 30);
676     rebuild_toolbar(&hToolbar);
677     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n");
678     CHECK_IMAGELIST(8, 16, 16);
679     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 3, (LPARAM)&bmp80) == 5, "TB_ADDBITMAP - unexpected return\n");
680     CHECK_IMAGELIST(13, 16, 16);
681     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(30, 30)) == TRUE, "TB_SETBITMAPSIZE failed\n");
682     UpdateWindow(hToolbar);
683     CHECK_IMAGELIST(8, 30, 30);
684     /* when the width or height is zero, set it to 1 */
685     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(0, 0)) == TRUE, "TB_SETBITMAPSIZE failed\n");
686     UpdateWindow(hToolbar);
687     CHECK_IMAGELIST(208, 1, 1);
688     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(0, 5)) == TRUE, "TB_SETBITMAPSIZE failed\n");
689     UpdateWindow(hToolbar);
690     CHECK_IMAGELIST(208, 1, 5);
691     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(5, 0)) == TRUE, "TB_SETBITMAPSIZE failed\n");
692     UpdateWindow(hToolbar);
693     CHECK_IMAGELIST(41, 5, 1);
694 
695     /* the control can add bitmaps to an existing image list */
696     rebuild_toolbar(&hToolbar);
697     himl = pImageList_LoadImageA(GetModuleHandleA(NULL), (LPCSTR)MAKEINTRESOURCE(IDB_BITMAP_80x15),
698                                 20, 2, CLR_NONE, IMAGE_BITMAP, LR_DEFAULTCOLOR);
699     ok(himl != NULL, "failed to create imagelist\n");
700     ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, (LPARAM)himl) == 0, "TB_SETIMAGELIST failed\n");
701     CHECK_IMAGELIST(4, 20, 15);
702     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n");
703     CHECK_IMAGELIST(10, 20, 15);
704     /* however TB_SETBITMAPSIZE/add std bitmap won't change the image size (the button size does change) */
705     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(8, 8)) == TRUE, "TB_SETBITMAPSIZE failed\n");
706     UpdateWindow(hToolbar);
707     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(15, 14), "%x");
708     CHECK_IMAGELIST(10, 20, 15);
709     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 0, (LPARAM)&stdsmall) == 1, "TB_SETBITMAPSIZE failed\n");
710     UpdateWindow(hToolbar);
711     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(23, 22), "%x");
712     CHECK_IMAGELIST(22, 20, 15);
713 
714     /* check standard bitmaps */
715     addbmp.hInst = HINST_COMMCTRL;
716     addbmp.nID = IDB_STD_SMALL_COLOR;
717     rebuild_toolbar(&hToolbar);
718     pImageList_Destroy(himl);
719 
720     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&addbmp) == 0, "TB_ADDBITMAP - unexpected return\n");
721     CHECK_IMAGELIST(15, 16, 16);
722     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(23, 22), "%x");
723     addbmp.nID = IDB_STD_LARGE_COLOR;
724     rebuild_toolbar(&hToolbar);
725     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&addbmp) == 0, "TB_ADDBITMAP - unexpected return\n");
726     CHECK_IMAGELIST(15, 24, 24);
727     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(31, 30), "%x");
728 
729     addbmp.nID = IDB_VIEW_SMALL_COLOR;
730     rebuild_toolbar(&hToolbar);
731     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&addbmp) == 0, "TB_ADDBITMAP - unexpected return\n");
732     CHECK_IMAGELIST(12, 16, 16);
733     addbmp.nID = IDB_VIEW_LARGE_COLOR;
734     rebuild_toolbar(&hToolbar);
735     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&addbmp) == 0, "TB_ADDBITMAP - unexpected return\n");
736     CHECK_IMAGELIST(12, 24, 24);
737 
738     addbmp.nID = IDB_HIST_SMALL_COLOR;
739     rebuild_toolbar(&hToolbar);
740     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&addbmp) == 0, "TB_ADDBITMAP - unexpected return\n");
741     CHECK_IMAGELIST(5, 16, 16);
742     addbmp.nID = IDB_HIST_LARGE_COLOR;
743     rebuild_toolbar(&hToolbar);
744     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&addbmp) == 0, "TB_ADDBITMAP - unexpected return\n");
745     CHECK_IMAGELIST(5, 24, 24);
746 
747 
748     DestroyWindow(hToolbar);
749 }
750 
751 #define CHECK_STRING_TABLE(count, tab) { \
752         INT _i; \
753         CHAR _buf[260]; \
754         for (_i = 0; _i < (count); _i++) {\
755             ret = SendMessageA(hToolbar, TB_GETSTRINGA, MAKEWPARAM(260, _i), (LPARAM)_buf); \
756             ok(ret >= 0, "TB_GETSTRINGA - unexpected return %d while checking string %d\n", ret, _i); \
757             if (ret >= 0) \
758                 ok(strcmp(_buf, (tab)[_i]) == 0, "Invalid string #%d - '%s' vs '%s'\n", _i, (tab)[_i], _buf); \
759         } \
760         ok(SendMessageA(hToolbar, TB_GETSTRINGA, MAKEWPARAM(260, (count)), (LPARAM)_buf) == -1, \
761             "Too many strings in table\n"); \
762     }
763 
764 static void test_add_string(void)
765 {
766     LPCSTR test1 = "a\0b\0";
767     LPCSTR test2 = "|a|b||\0";
768     LPCSTR ret1[] = {"a", "b"};
769     LPCSTR ret2[] = {"a", "b", "|a|b||"};
770     LPCSTR ret3[] = {"a", "b", "|a|b||", "p", "q"};
771     LPCSTR ret4[] = {"a", "b", "|a|b||", "p", "q", "p"};
772     LPCSTR ret5[] = {"a", "b", "|a|b||", "p", "q", "p", "p", "q"};
773     LPCSTR ret6[] = {"a", "b", "|a|b||", "p", "q", "p", "p", "q", "p", "", "q"};
774     LPCSTR ret7[] = {"a", "b", "|a|b||", "p", "q", "p", "p", "q", "p", "", "q", "br", "c", "d"};
775     HWND hToolbar = NULL;
776     TBBUTTON button;
777     int ret;
778     CHAR buf[260];
779 
780     rebuild_toolbar(&hToolbar);
781     ret = SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)test1);
782     ok(ret == 0, "TB_ADDSTRINGA - unexpected return %d\n", ret);
783     ret = SendMessageA(hToolbar, TB_GETSTRINGA, MAKEWPARAM(260, 1), (LPARAM)buf);
784     if (ret == 0)
785     {
786         win_skip("TB_GETSTRINGA needs 5.80\n");
787         return;
788     }
789     CHECK_STRING_TABLE(2, ret1);
790     ret = SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)test2);
791     ok(ret == 2, "TB_ADDSTRINGA - unexpected return %d\n", ret);
792     CHECK_STRING_TABLE(3, ret2);
793 
794     /* null instance handle */
795     ret = SendMessageA(hToolbar, TB_ADDSTRINGA, 0, IDS_TBADD1);
796     ok(ret == -1, "TB_ADDSTRINGA - unexpected return %d\n", ret);
797 
798     /* invalid instance handle */
799     ret = SendMessageA(hToolbar, TB_ADDSTRINGA, 0xdeadbeef, IDS_TBADD1);
800     ok(ret == -1, "TB_ADDSTRINGA - unexpected return %d\n", ret);
801 
802     ret = SendMessageA(hToolbar, TB_ADDSTRINGA, (WPARAM)GetModuleHandleA(NULL), IDS_TBADD1);
803     ok(ret == 3, "TB_ADDSTRINGA - unexpected return %d\n", ret);
804     CHECK_STRING_TABLE(3, ret2);
805     ret = SendMessageA(hToolbar, TB_ADDSTRINGA, (WPARAM)GetModuleHandleA(NULL), IDS_TBADD2);
806     ok(ret == 3, "TB_ADDSTRINGA - unexpected return %d\n", ret);
807     CHECK_STRING_TABLE(5, ret3);
808     ret = SendMessageA(hToolbar, TB_ADDSTRINGA, (WPARAM)GetModuleHandleA(NULL), IDS_TBADD3);
809     ok(ret == 5, "TB_ADDSTRINGA - unexpected return %d\n", ret);
810     CHECK_STRING_TABLE(6, ret4);
811     ret = SendMessageA(hToolbar, TB_ADDSTRINGA, (WPARAM)GetModuleHandleA(NULL), IDS_TBADD4);
812     ok(ret == 6, "TB_ADDSTRINGA - unexpected return %d\n", ret);
813     CHECK_STRING_TABLE(8, ret5);
814     ret = SendMessageA(hToolbar, TB_ADDSTRINGA, (WPARAM)GetModuleHandleA(NULL), IDS_TBADD5);
815     ok(ret == 8, "TB_ADDSTRINGA - unexpected return %d\n", ret);
816     CHECK_STRING_TABLE(11, ret6);
817     ret = SendMessageA(hToolbar, TB_ADDSTRINGA, (WPARAM)GetModuleHandleA(NULL), IDS_TBADD7);
818     ok(ret == 11, "TB_ADDSTRINGA - unexpected return %d\n", ret);
819     CHECK_STRING_TABLE(14, ret7);
820 
821     ZeroMemory(&button, sizeof(button));
822     button.iString = (UINT_PTR)"Test";
823     SendMessageA(hToolbar, TB_INSERTBUTTONA, 0, (LPARAM)&button);
824     CHECK_STRING_TABLE(14, ret7);
825     SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&button);
826     CHECK_STRING_TABLE(14, ret7);
827 
828     DestroyWindow(hToolbar);
829 }
830 
831 static void expect_hot_notify(int idold, int idnew)
832 {
833     g_fExpectedHotItemOld = idold;
834     g_fExpectedHotItemNew = idnew;
835     g_fReceivedHotItemChange = FALSE;
836 }
837 
838 #define check_hot_notify() \
839     ok(g_fReceivedHotItemChange, "TBN_HOTITEMCHANGE not received\n"); \
840     g_fExpectedHotItemOld = g_fExpectedHotItemNew = 0;
841 
842 static void test_hotitem(void)
843 {
844     HWND hToolbar = NULL;
845     TBBUTTONINFOA tbinfo;
846     LRESULT ret;
847 
848     g_fBlockHotItemChange = FALSE;
849 
850     rebuild_toolbar_with_buttons(&hToolbar);
851     /* set TBSTYLE_FLAT. comctl5 allows hot items only for such toolbars.
852      * comctl6 doesn't have this requirement even when theme == NULL */
853     SetWindowLongA(hToolbar, GWL_STYLE, TBSTYLE_FLAT | GetWindowLongA(hToolbar, GWL_STYLE));
854     ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0);
855     ok(ret == -1, "Hot item: %ld, expected -1\n", ret);
856     ret = SendMessageA(hToolbar, TB_SETHOTITEM, 1, 0);
857     ok(ret == -1, "TB_SETHOTITEM returned %ld, expected -1\n", ret);
858     ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0);
859     ok(ret == 1, "Hot item: %ld, expected 1\n", ret);
860     ret = SendMessageA(hToolbar, TB_SETHOTITEM, 2, 0);
861     ok(ret == 1, "TB_SETHOTITEM returned %ld, expected 1\n", ret);
862 
863     ret = SendMessageA(hToolbar, TB_SETHOTITEM, 0xbeef, 0);
864     ok(ret == 2, "TB_SETHOTITEM returned %ld, expected 2\n", ret);
865     ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0);
866     ok(ret == 2, "Hot item: %lx, expected 2\n", ret);
867     ret = SendMessageA(hToolbar, TB_SETHOTITEM, -0xbeef, 0);
868     ok(ret == 2, "TB_SETHOTITEM returned %ld, expected 2\n", ret);
869     ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0);
870     ok(ret == -1, "Hot item: %lx, expected -1\n", ret);
871 
872     expect_hot_notify(0, 7);
873     ret = SendMessageA(hToolbar, TB_SETHOTITEM, 3, 0);
874     ok(ret == -1, "TB_SETHOTITEM returned %ld, expected -1\n", ret);
875     check_hot_notify();
876     ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0);
877     ok(ret == 3, "Hot item: %lx, expected 3\n", ret);
878     g_fBlockHotItemChange = TRUE;
879     ret = SendMessageA(hToolbar, TB_SETHOTITEM, 2, 0);
880     ok(ret == 3, "TB_SETHOTITEM returned %ld, expected 3\n", ret);
881     ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0);
882     ok(ret == 3, "Hot item: %lx, expected 3\n", ret);
883     g_fBlockHotItemChange = FALSE;
884 
885     g_fReceivedHotItemChange = FALSE;
886     ret = SendMessageA(hToolbar, TB_SETHOTITEM, 0xbeaf, 0);
887     ok(ret == 3, "TB_SETHOTITEM returned %ld, expected 3\n", ret);
888     ok(g_fReceivedHotItemChange == FALSE, "TBN_HOTITEMCHANGE received for invalid parameter\n");
889 
890     g_fReceivedHotItemChange = FALSE;
891     ret = SendMessageA(hToolbar, TB_SETHOTITEM, 3, 0);
892     ok(ret == 3, "TB_SETHOTITEM returned %ld, expected 3\n", ret);
893     ok(g_fReceivedHotItemChange == FALSE, "TBN_HOTITEMCHANGE received after a duplication\n");
894 
895     expect_hot_notify(7, 0);
896     ret = SendMessageA(hToolbar, TB_SETHOTITEM, -0xbeaf, 0);
897     ok(ret == 3, "TB_SETHOTITEM returned %ld, expected 3\n", ret);
898     check_hot_notify();
899     SendMessageA(hToolbar, TB_SETHOTITEM, 3, 0);
900 
901     /* setting disabled buttons will generate a notify with the button id but no button will be hot */
902     expect_hot_notify(7, 9);
903     ret = SendMessageA(hToolbar, TB_SETHOTITEM, 4, 0);
904     ok(ret == 3, "TB_SETHOTITEM returned %ld, expected 3\n", ret);
905     check_hot_notify();
906     ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0);
907     ok(ret == -1, "Hot item: %lx, expected -1\n", ret);
908     /* enabling the button won't change that */
909     SendMessageA(hToolbar, TB_ENABLEBUTTON, 9, TRUE);
910     ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0);
911     ok(ret == -1, "TB_GETHOTITEM returned %ld, expected -1\n", ret);
912 
913     /* disabling a hot button works */
914     ret = SendMessageA(hToolbar, TB_SETHOTITEM, 3, 0);
915     ok(ret == -1, "TB_SETHOTITEM returned %ld, expected -1\n", ret);
916     g_fReceivedHotItemChange = FALSE;
917     SendMessageA(hToolbar, TB_ENABLEBUTTON, 7, FALSE);
918     ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0);
919     ok(ret == 3, "TB_GETHOTITEM returned %ld, expected 3\n", ret);
920     ok(g_fReceivedHotItemChange == FALSE, "Unexpected TBN_HOTITEMCHANGE\n");
921 
922     SendMessageA(hToolbar, TB_SETHOTITEM, 1, 0);
923     tbinfo.cbSize = sizeof(TBBUTTONINFOA);
924     tbinfo.dwMask = TBIF_STATE;
925     tbinfo.fsState = 0;  /* disabled */
926     g_fReceivedHotItemChange = FALSE;
927     ok(SendMessageA(hToolbar, TB_SETBUTTONINFOA, 1, (LPARAM)&tbinfo) == TRUE, "TB_SETBUTTONINFOA failed\n");
928     ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0);
929     ok(ret == 1, "TB_GETHOTITEM returned %ld, expected 1\n", ret);
930     ok(g_fReceivedHotItemChange == FALSE, "Unexpected TBN_HOTITEMCHANGE\n");
931 
932     /* deleting a button unsets the hot item */
933     ret = SendMessageA(hToolbar, TB_SETHOTITEM, 0, 0);
934     ok(ret == 1, "TB_SETHOTITEM returned %ld, expected 1\n", ret);
935     g_fReceivedHotItemChange = FALSE;
936     ret = SendMessageA(hToolbar, TB_DELETEBUTTON, 1, 0);
937     ok(ret == TRUE, "TB_DELETEBUTTON returned %ld, expected TRUE\n", ret);
938     ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0);
939     ok(ret == -1, "TB_GETHOTITEM returned %ld, expected -1\n", ret);
940     ok(g_fReceivedHotItemChange == FALSE, "Unexpected TBN_HOTITEMCHANGE\n");
941 
942     DestroyWindow(hToolbar);
943 }
944 
945 #if 0  /* use this to generate more tests*/
946 
947 static void dump_sizes(HWND hToolbar)
948 {
949     SIZE sz;
950     RECT r;
951     int count = SendMessageA(hToolbar, TB_BUTTONCOUNT, 0, 0);
952     int i;
953 
954     GetClientRect(hToolbar, &r);
955     SendMessageA(hToolbar, TB_GETMAXSIZE, 0, &sz);
956     printf("  { {%d, %d, %d, %d}, {%d, %d}, %d, {", r.left, r.top, r.right, r.bottom,
957         sz.cx, sz.cy, count);
958     for (i=0; i<count; i++)
959     {
960         SendMessageA(hToolbar, TB_GETITEMRECT, i, &r);
961         printf("%s{%3d, %3d, %3d, %3d}, ", (i%3==0 ? "\n    " : ""), r.left, r.top, r.right, r.bottom);
962     }
963     printf("\n  }, },\n");
964 }
965 
966 #define check_sizes() dump_sizes(hToolbar);
967 #define check_sizes_todo(todomask) dump_sizes(hToolbar);
968 
969 #else
970 
971 static int system_font_height(void) {
972     HDC hDC;
973     TEXTMETRICA tm;
974 
975     hDC = CreateCompatibleDC(NULL);
976     GetTextMetricsA(hDC, &tm);
977     DeleteDC(NULL);
978 
979     return tm.tmHeight;
980 }
981 
982 static int string_width(const CHAR *s) {
983     SIZE sz;
984     HDC hdc;
985 
986     hdc = CreateCompatibleDC(NULL);
987     GetTextExtentPoint32A(hdc, s, strlen(s), &sz);
988     DeleteDC(hdc);
989 
990     return sz.cx;
991 }
992 
993 typedef struct
994 {
995     RECT rcClient;
996     SIZE szMin;
997     INT nButtons;
998     RECT *prcButtons;
999 } tbsize_result_t;
1000 
1001 static tbsize_result_t init_tbsize_result(int nButtonsAlloc, int cleft, int ctop, int cright, int cbottom, int minx, int miny) {
1002     tbsize_result_t ret;
1003 
1004     SetRect(&ret.rcClient, cleft, ctop, cright, cbottom);
1005     ret.szMin.cx = minx;
1006     ret.szMin.cy = miny;
1007     ret.nButtons = 0;
1008     ret.prcButtons = heap_alloc_zero(nButtonsAlloc * sizeof(*ret.prcButtons));
1009 
1010     return ret;
1011 }
1012 
1013 static void tbsize_addbutton(tbsize_result_t *tbsr, int left, int top, int right, int bottom) {
1014     SetRect(&tbsr->prcButtons[tbsr->nButtons], left, top, right, bottom);
1015     tbsr->nButtons++;
1016 }
1017 
1018 #define STRING0 "A"
1019 #define STRING1 "MMMMMMMMMMMMM"
1020 #define STRING2 "Tst"
1021 
1022 static tbsize_result_t *tbsize_results;
1023 
1024 #define tbsize_results_num 28
1025 
1026 static void init_tbsize_results(void) {
1027     int fontheight = system_font_height();
1028     int buttonwidth;
1029 
1030     tbsize_results = heap_alloc_zero(tbsize_results_num * sizeof(*tbsize_results));
1031 
1032     tbsize_results[0] = init_tbsize_result(5, 0, 0 ,672 ,26, 100 ,22);
1033     tbsize_addbutton(&tbsize_results[0],   0,   2,  23,  24);
1034     tbsize_addbutton(&tbsize_results[0],  23,   2,  46,  24);
1035     tbsize_addbutton(&tbsize_results[0],  46,   2,  54,  24);
1036     tbsize_addbutton(&tbsize_results[0],  54,   2,  77,  24);
1037     tbsize_addbutton(&tbsize_results[0],  77,   2, 100,  24);
1038 
1039     tbsize_results[1] = init_tbsize_result(7, 0, 0, 672, 26, 146, 22);
1040     tbsize_addbutton(&tbsize_results[1],   0,   2,  23,  24);
1041     tbsize_addbutton(&tbsize_results[1],  23,   2,  46,  24);
1042     tbsize_addbutton(&tbsize_results[1],  46,   2,  54,  24);
1043     tbsize_addbutton(&tbsize_results[1],  54,   2,  77,  24);
1044     tbsize_addbutton(&tbsize_results[1],  77,   2, 100,  24);
1045     tbsize_addbutton(&tbsize_results[1], 100,   2, 123,  24);
1046     tbsize_addbutton(&tbsize_results[1],   0,   24, 23,  46);
1047 
1048     tbsize_results[2] = init_tbsize_result(7, 0, 0, 672, 26, 146, 22);
1049     tbsize_addbutton(&tbsize_results[2],   0,   2,  23,  24);
1050     tbsize_addbutton(&tbsize_results[2],  23,   2,  46,  24);
1051     tbsize_addbutton(&tbsize_results[2],  46,   2,  54,  24);
1052     tbsize_addbutton(&tbsize_results[2],  54,   2,  77,  24);
1053     tbsize_addbutton(&tbsize_results[2],  77,   2, 100,  24);
1054     tbsize_addbutton(&tbsize_results[2], 100,   2, 123,  24);
1055     tbsize_addbutton(&tbsize_results[2],   0,   24, 23,  46);
1056 
1057     tbsize_results[3] = init_tbsize_result(7, 0, 0, 672, 26, 146, 22);
1058     tbsize_addbutton(&tbsize_results[3],   0,   2,  23,  24);
1059     tbsize_addbutton(&tbsize_results[3],  23,   2,  46,  24);
1060     tbsize_addbutton(&tbsize_results[3],  46,   2,  54,  24);
1061     tbsize_addbutton(&tbsize_results[3],  54,   2,  77,  24);
1062     tbsize_addbutton(&tbsize_results[3],  77,   2, 100,  24);
1063     tbsize_addbutton(&tbsize_results[3], 100,   2, 123,  24);
1064     tbsize_addbutton(&tbsize_results[3], 123,   2, 146,  24);
1065 
1066     tbsize_results[4] = init_tbsize_result(9, 0, 0, 672, 26, 192, 22);
1067     tbsize_addbutton(&tbsize_results[4],   0,   2,  23,  24);
1068     tbsize_addbutton(&tbsize_results[4],  23,   2,  46,  24);
1069     tbsize_addbutton(&tbsize_results[4],  46,   2,  54,  24);
1070     tbsize_addbutton(&tbsize_results[4],  54,   2,  77,  24);
1071     tbsize_addbutton(&tbsize_results[4],  77,   2, 100,  24);
1072     tbsize_addbutton(&tbsize_results[4], 100,   2, 123,  24);
1073     tbsize_addbutton(&tbsize_results[4], 123,   2, 146,  24);
1074     tbsize_addbutton(&tbsize_results[4], 146,   2, 169,  24);
1075     tbsize_addbutton(&tbsize_results[4], 169,   2, 192,  24);
1076 
1077     tbsize_results[5] = init_tbsize_result(39, 0, 0, 672, 92, 882, 22);
1078     tbsize_addbutton(&tbsize_results[5],   0,   2,  23,  24);
1079     tbsize_addbutton(&tbsize_results[5],  23,   2,  46,  24);
1080     tbsize_addbutton(&tbsize_results[5],   0,   2,   8,  29);
1081     tbsize_addbutton(&tbsize_results[5],   0,  29,  23,  51);
1082     tbsize_addbutton(&tbsize_results[5],  23,  29,  46,  51);
1083     tbsize_addbutton(&tbsize_results[5],  46,  29,  69,  51);
1084     tbsize_addbutton(&tbsize_results[5],  69,  29,  92,  51);
1085     tbsize_addbutton(&tbsize_results[5],  92,  29, 115,  51);
1086     tbsize_addbutton(&tbsize_results[5], 115,  29, 138,  51);
1087     tbsize_addbutton(&tbsize_results[5], 138,  29, 161,  51);
1088     tbsize_addbutton(&tbsize_results[5], 161,  29, 184,  51);
1089     tbsize_addbutton(&tbsize_results[5], 184,  29, 207,  51);
1090     tbsize_addbutton(&tbsize_results[5], 207,  29, 230,  51);
1091     tbsize_addbutton(&tbsize_results[5], 230,  29, 253,  51);
1092     tbsize_addbutton(&tbsize_results[5], 253,  29, 276,  51);
1093     tbsize_addbutton(&tbsize_results[5], 276,  29, 299,  51);
1094     tbsize_addbutton(&tbsize_results[5], 299,  29, 322,  51);
1095     tbsize_addbutton(&tbsize_results[5], 322,  29, 345,  51);
1096     tbsize_addbutton(&tbsize_results[5], 345,  29, 368,  51);
1097     tbsize_addbutton(&tbsize_results[5], 368,  29, 391,  51);
1098     tbsize_addbutton(&tbsize_results[5], 391,  29, 414,  51);
1099     tbsize_addbutton(&tbsize_results[5], 414,  29, 437,  51);
1100     tbsize_addbutton(&tbsize_results[5], 437,  29, 460,  51);
1101     tbsize_addbutton(&tbsize_results[5], 460,  29, 483,  51);
1102     tbsize_addbutton(&tbsize_results[5], 483,  29, 506,  51);
1103     tbsize_addbutton(&tbsize_results[5], 506,  29, 529,  51);
1104     tbsize_addbutton(&tbsize_results[5], 529,  29, 552,  51);
1105     tbsize_addbutton(&tbsize_results[5], 552,  29, 575,  51);
1106     tbsize_addbutton(&tbsize_results[5], 575,  29, 598,  51);
1107     tbsize_addbutton(&tbsize_results[5], 598,  29, 621,  51);
1108     tbsize_addbutton(&tbsize_results[5], 621,  29, 644,  51);
1109     tbsize_addbutton(&tbsize_results[5], 644,  29, 667,  51);
1110     tbsize_addbutton(&tbsize_results[5],   0,  51,  23,  73);
1111     tbsize_addbutton(&tbsize_results[5],  23,  51,  46,  73);
1112     tbsize_addbutton(&tbsize_results[5],  46,  51,  69,  73);
1113     tbsize_addbutton(&tbsize_results[5],  69,  51,  92,  73);
1114     tbsize_addbutton(&tbsize_results[5],  92,  51, 115,  73);
1115     tbsize_addbutton(&tbsize_results[5], 115,  51, 138,  73);
1116     tbsize_addbutton(&tbsize_results[5], 138,  51, 161,  73);
1117 
1118     tbsize_results[6] = init_tbsize_result(7, 0, 0, 48, 226, 23, 140);
1119     tbsize_addbutton(&tbsize_results[6],   0,   2,  23,  24);
1120     tbsize_addbutton(&tbsize_results[6],  23,   2,  46,  24);
1121     tbsize_addbutton(&tbsize_results[6],  46,   2,  94,  24);
1122     tbsize_addbutton(&tbsize_results[6],  94,   2, 117,  24);
1123     tbsize_addbutton(&tbsize_results[6], 117,   2, 140,  24);
1124     tbsize_addbutton(&tbsize_results[6], 140,   2, 163,  24);
1125     tbsize_addbutton(&tbsize_results[6],   0,  24,  23,  46);
1126 
1127     tbsize_results[7] = init_tbsize_result(7, 0, 0, 92, 226, 23, 140);
1128     tbsize_addbutton(&tbsize_results[7],   0,   2,  23,  24);
1129     tbsize_addbutton(&tbsize_results[7],  23,   2,  46,  24);
1130     tbsize_addbutton(&tbsize_results[7],   0,  24,  92,  32);
1131     tbsize_addbutton(&tbsize_results[7],   0,  32,  23,  54);
1132     tbsize_addbutton(&tbsize_results[7],  23,  32,  46,  54);
1133     tbsize_addbutton(&tbsize_results[7],  46,  32,  69,  54);
1134     tbsize_addbutton(&tbsize_results[7],  69,  32,  92,  54);
1135 
1136     tbsize_results[8] = init_tbsize_result(7, 0, 0, 672, 26, 194, 30);
1137     tbsize_addbutton(&tbsize_results[8],   0,   2,  31,  32);
1138     tbsize_addbutton(&tbsize_results[8],  31,   2,  62,  32);
1139     tbsize_addbutton(&tbsize_results[8],  62,   2,  70,  32);
1140     tbsize_addbutton(&tbsize_results[8],  70,   2, 101,  32);
1141     tbsize_addbutton(&tbsize_results[8], 101,   2, 132,  32);
1142     tbsize_addbutton(&tbsize_results[8], 132,   2, 163,  32);
1143     tbsize_addbutton(&tbsize_results[8],   0,  32,  31,  62);
1144 
1145     tbsize_results[9] = init_tbsize_result(7, 0, 0, 672, 64, 194, 30);
1146     tbsize_addbutton(&tbsize_results[9],   0,   2,  31,  32);
1147     tbsize_addbutton(&tbsize_results[9],  31,   2,  62,  32);
1148     tbsize_addbutton(&tbsize_results[9],  62,   2,  70,  32);
1149     tbsize_addbutton(&tbsize_results[9],  70,   2, 101,  32);
1150     tbsize_addbutton(&tbsize_results[9], 101,   2, 132,  32);
1151     tbsize_addbutton(&tbsize_results[9], 132,   2, 163,  32);
1152     tbsize_addbutton(&tbsize_results[9],   0,  32,  31,  62);
1153 
1154     tbsize_results[10] = init_tbsize_result(7, 0, 0, 672, 64, 194, 30);
1155     tbsize_addbutton(&tbsize_results[10],   0,   0,  31,  30);
1156     tbsize_addbutton(&tbsize_results[10],  31,   0,  62,  30);
1157     tbsize_addbutton(&tbsize_results[10],  62,   0,  70,  30);
1158     tbsize_addbutton(&tbsize_results[10],  70,   0, 101,  30);
1159     tbsize_addbutton(&tbsize_results[10], 101,   0, 132,  30);
1160     tbsize_addbutton(&tbsize_results[10], 132,   0, 163,  30);
1161     tbsize_addbutton(&tbsize_results[10],   0,  30,  31,  60);
1162 
1163     tbsize_results[11] = init_tbsize_result(7, 0, 0, 124, 226, 31, 188);
1164     tbsize_addbutton(&tbsize_results[11],   0,    0,  31,  30);
1165     tbsize_addbutton(&tbsize_results[11],  31,    0,  62,  30);
1166     tbsize_addbutton(&tbsize_results[11],   0,   30, 124,  38);
1167     tbsize_addbutton(&tbsize_results[11],   0,   38,  31,  68);
1168     tbsize_addbutton(&tbsize_results[11],  31,   38,  62,  68);
1169     tbsize_addbutton(&tbsize_results[11],  62,   38,  93,  68);
1170     tbsize_addbutton(&tbsize_results[11],  93,   38, 124,  68);
1171 
1172     tbsize_results[12] = init_tbsize_result(7, 0, 0, 672, 26, 146, 22);
1173     tbsize_addbutton(&tbsize_results[12],   0,   2,  23,  24);
1174     tbsize_addbutton(&tbsize_results[12],  23,   2,  46,  24);
1175     tbsize_addbutton(&tbsize_results[12],  46,   2,  54,  24);
1176     tbsize_addbutton(&tbsize_results[12],  54,   2,  77,  24);
1177     tbsize_addbutton(&tbsize_results[12],  77,   2, 100,  24);
1178     tbsize_addbutton(&tbsize_results[12], 100,   2, 123,  24);
1179     tbsize_addbutton(&tbsize_results[12], 123,   2, 146,  24);
1180 
1181     tbsize_results[13] = init_tbsize_result(7, 0, 0, 672, 26, 146, 100);
1182     tbsize_addbutton(&tbsize_results[13],   0,   0,  23, 100);
1183     tbsize_addbutton(&tbsize_results[13],  23,   0,  46, 100);
1184     tbsize_addbutton(&tbsize_results[13],  46,   0,  54, 100);
1185     tbsize_addbutton(&tbsize_results[13],  54,   0,  77, 100);
1186     tbsize_addbutton(&tbsize_results[13],  77,   0, 100, 100);
1187     tbsize_addbutton(&tbsize_results[13], 100,   0, 123, 100);
1188     tbsize_addbutton(&tbsize_results[13], 123,   0, 146, 100);
1189 
1190     tbsize_results[14] = init_tbsize_result(10, 0, 0, 672, 26, 146, 100);
1191     tbsize_addbutton(&tbsize_results[14],   0,   0,  23, 100);
1192     tbsize_addbutton(&tbsize_results[14],  23,   0,  46, 100);
1193     tbsize_addbutton(&tbsize_results[14],  46,   0,  54, 100);
1194     tbsize_addbutton(&tbsize_results[14],  54,   0,  77, 100);
1195     tbsize_addbutton(&tbsize_results[14],  77,   0, 100, 100);
1196     tbsize_addbutton(&tbsize_results[14], 100,   0, 123, 100);
1197     tbsize_addbutton(&tbsize_results[14], 123,   0, 146, 100);
1198     tbsize_addbutton(&tbsize_results[14], 146,   0, 169, 100);
1199     tbsize_addbutton(&tbsize_results[14], 169,   0, 192, 100);
1200     tbsize_addbutton(&tbsize_results[14], 192,   0, 215, 100);
1201 
1202     tbsize_results[15] = init_tbsize_result(11, 0, 0, 672, 26, 238, 39);
1203     tbsize_addbutton(&tbsize_results[15],   0,   0,  23,  23 + fontheight);
1204     tbsize_addbutton(&tbsize_results[15],  23,   0,  46,  23 + fontheight);
1205     tbsize_addbutton(&tbsize_results[15],  46,   0,  54,  23 + fontheight);
1206     tbsize_addbutton(&tbsize_results[15],  54,   0,  77,  23 + fontheight);
1207     tbsize_addbutton(&tbsize_results[15],  77,   0, 100,  23 + fontheight);
1208     tbsize_addbutton(&tbsize_results[15], 100,   0, 123,  23 + fontheight);
1209     tbsize_addbutton(&tbsize_results[15], 123,   0, 146,  23 + fontheight);
1210     tbsize_addbutton(&tbsize_results[15], 146,   0, 169,  23 + fontheight);
1211     tbsize_addbutton(&tbsize_results[15], 169,   0, 192,  23 + fontheight);
1212     tbsize_addbutton(&tbsize_results[15], 192,   0, 215,  23 + fontheight);
1213     tbsize_addbutton(&tbsize_results[15], 215,   0, 238,  23 + fontheight);
1214 
1215     tbsize_results[16] = init_tbsize_result(11, 0, 0, 672, 26, 239, 22);
1216     tbsize_addbutton(&tbsize_results[16],   0,   0,  23,  22);
1217     tbsize_addbutton(&tbsize_results[16],  23,   0,  46,  22);
1218     tbsize_addbutton(&tbsize_results[16],  46,   0,  54,  22);
1219     tbsize_addbutton(&tbsize_results[16],  54,   0,  77,  22);
1220     tbsize_addbutton(&tbsize_results[16],  77,   0, 100,  22);
1221     tbsize_addbutton(&tbsize_results[16], 100,   0, 123,  22);
1222     tbsize_addbutton(&tbsize_results[16], 123,   0, 146,  22);
1223     tbsize_addbutton(&tbsize_results[16], 146,   0, 169,  22);
1224     tbsize_addbutton(&tbsize_results[16], 169,   0, 192,  22);
1225     tbsize_addbutton(&tbsize_results[16], 192,   0, 215,  22);
1226     tbsize_addbutton(&tbsize_results[16], 215,   0, 238,  22);
1227 
1228     buttonwidth = 7 + string_width(STRING1);
1229 
1230     tbsize_results[17] = init_tbsize_result(3, 0, 0, 672, 26, 489, 39);
1231     tbsize_addbutton(&tbsize_results[17],   0,   2, buttonwidth,  25 + fontheight);
1232     tbsize_addbutton(&tbsize_results[17], buttonwidth,   2, 2*buttonwidth + 4,  25 + fontheight);
1233     tbsize_addbutton(&tbsize_results[17], 2*buttonwidth + 4,   2, 3*buttonwidth + 4,  25 + fontheight);
1234 
1235     tbsize_results[18] = init_tbsize_result(6, 0, 0, 672, 104, 978, 24);
1236     tbsize_addbutton(&tbsize_results[18],   0,   2, buttonwidth,  10 + fontheight);
1237     tbsize_addbutton(&tbsize_results[18], buttonwidth,   2, 2*buttonwidth,  10 + fontheight);
1238     tbsize_addbutton(&tbsize_results[18], 2*buttonwidth,   2, 3*buttonwidth,  10 + fontheight);
1239     tbsize_addbutton(&tbsize_results[18], 3*buttonwidth,   2, 4*buttonwidth,  10 + fontheight);
1240     tbsize_addbutton(&tbsize_results[18], 4*buttonwidth,   2, 5*buttonwidth + 4,  10 + fontheight);
1241     tbsize_addbutton(&tbsize_results[18], 5*buttonwidth + 4,   2, 5*buttonwidth + 4 + string_width(STRING2) + 11,  10 + fontheight);
1242 
1243     tbsize_results[19] = init_tbsize_result(6, 0, 0, 672, 28, 978, 38);
1244     tbsize_addbutton(&tbsize_results[19],   0,   0, buttonwidth,  22 + fontheight);
1245     tbsize_addbutton(&tbsize_results[19], buttonwidth,   0, 2*buttonwidth,  22 + fontheight);
1246     tbsize_addbutton(&tbsize_results[19], 2*buttonwidth,   0, 3*buttonwidth,  22 + fontheight);
1247     tbsize_addbutton(&tbsize_results[19], 3*buttonwidth,   0, 4*buttonwidth,  22 + fontheight);
1248     tbsize_addbutton(&tbsize_results[19], 4*buttonwidth,   0, 5*buttonwidth + 4,  22 + fontheight);
1249     tbsize_addbutton(&tbsize_results[19], 5*buttonwidth + 4,   0, 5*buttonwidth + 4 + string_width(STRING2) + 11,  22 + fontheight);
1250 
1251     tbsize_results[20] = init_tbsize_result(3, 0, 0, 672, 100, 239, 102);
1252     tbsize_addbutton(&tbsize_results[20],   0,   2, 100, 102);
1253     tbsize_addbutton(&tbsize_results[20], 100,   2, 139, 102);
1254     tbsize_addbutton(&tbsize_results[20], 139,   2, 239, 102);
1255 
1256     tbsize_results[21] = init_tbsize_result(3, 0, 0, 672, 42, 185, 40);
1257     tbsize_addbutton(&tbsize_results[21],   0,   2,  75,  40);
1258     tbsize_addbutton(&tbsize_results[21],  75,   2, 118,  40);
1259     tbsize_addbutton(&tbsize_results[21], 118,   2, 165 + string_width(STRING2),  40);
1260 
1261     tbsize_results[22] = init_tbsize_result(1, 0, 0, 672, 42, 67, 40);
1262     tbsize_addbutton(&tbsize_results[22],   0,   2,  47 + string_width(STRING2),  40);
1263 
1264     tbsize_results[23] = init_tbsize_result(2, 0, 0, 672, 42, 67, 41);
1265     tbsize_addbutton(&tbsize_results[23],   0,   2, 672,  25 + fontheight);
1266     tbsize_addbutton(&tbsize_results[23],   0,  25 + fontheight, 672,  48 + 2*fontheight);
1267 
1268     tbsize_results[24] = init_tbsize_result(1, 0, 0, 672, 42, 67, 40);
1269     tbsize_addbutton(&tbsize_results[24],   0,   2,  11 + string_width(STRING2),  24);
1270 
1271     tbsize_results[25] = init_tbsize_result(1, 0, 0, 672, 42, 67, 40);
1272     tbsize_addbutton(&tbsize_results[25],   0,   2,  40,  24);
1273 
1274     tbsize_results[26] = init_tbsize_result(1, 0, 0, 672, 42, 67, 40);
1275     tbsize_addbutton(&tbsize_results[26],   0,   2,  40,  24);
1276 
1277     tbsize_results[27] = init_tbsize_result(1, 0, 0, 672, 42, 67, 40);
1278     tbsize_addbutton(&tbsize_results[27],   0,   2,  40,  24);
1279 }
1280 
1281 static void free_tbsize_results(void) {
1282     int i;
1283 
1284     for (i = 0; i < tbsize_results_num; i++)
1285         heap_free(tbsize_results[i].prcButtons);
1286     heap_free(tbsize_results);
1287     tbsize_results = NULL;
1288 }
1289 
1290 static int tbsize_numtests = 0;
1291 
1292 typedef struct
1293 {
1294     int test_num;
1295     int rect_index;
1296     RECT rcButton;
1297 } tbsize_alt_result_t;
1298 
1299 static tbsize_alt_result_t tbsize_alt_results[] =
1300 {
1301   { 5, 2, { 0, 24, 8, 29 } },
1302   { 20, 1, { 100, 2, 107, 102 } },
1303   { 20, 2, { 107, 2, 207, 102 } }
1304 };
1305 
1306 static DWORD tbsize_alt_numtests = 0;
1307 
1308 #define check_sizes_todo(todomask) { \
1309         RECT rc; \
1310         int buttonCount, i, mask=(todomask); \
1311         tbsize_result_t *res = &tbsize_results[tbsize_numtests]; \
1312         GetClientRect(hToolbar, &rc); \
1313         /*check_rect("client", rc, res->rcClient);*/ \
1314         buttonCount = SendMessageA(hToolbar, TB_BUTTONCOUNT, 0, 0); \
1315         compare(buttonCount, res->nButtons, "%d"); \
1316         for (i=0; i<min(buttonCount, res->nButtons); i++) { \
1317             ok(SendMessageA(hToolbar, TB_GETITEMRECT, i, (LPARAM)&rc) == 1, "TB_GETITEMRECT\n"); \
1318             if (broken(tbsize_alt_numtests < ARRAY_SIZE(tbsize_alt_results) && \
1319                        EqualRect(&rc, &tbsize_alt_results[tbsize_alt_numtests].rcButton))) { \
1320                 win_skip("Alternate rect found\n"); \
1321                 tbsize_alt_numtests++; \
1322             } else todo_wine_if(mask&1) \
1323                 check_rect("button = %d, tbsize_numtests = %d", rc, res->prcButtons[i], i, tbsize_numtests); \
1324             mask >>= 1; \
1325         } \
1326         tbsize_numtests++; \
1327     }
1328 
1329 #define check_sizes() check_sizes_todo(0)
1330 
1331 #endif
1332 
1333 static TBBUTTON buttons1[] = {
1334     {0, 10, TBSTATE_WRAP|TBSTATE_ENABLED, 0, {0, }, 0, -1},
1335     {0, 11, 0, 0, {0, }, 0, -1},
1336 };
1337 static TBBUTTON buttons2[] = {
1338     {0, 20, TBSTATE_ENABLED, 0, {0, }, 0, -1},
1339     {0, 21, TBSTATE_ENABLED, 0, {0, }, 0, -1},
1340 };
1341 static TBBUTTON buttons3[] = {
1342     {0, 30, TBSTATE_ENABLED, 0, {0, }, 0, 0},
1343     {0, 31, TBSTATE_ENABLED, 0, {0, }, 0, 1},
1344     {0, 32, TBSTATE_ENABLED, BTNS_AUTOSIZE, {0, }, 0, 1},
1345     {0, 33, TBSTATE_ENABLED, BTNS_AUTOSIZE, {0, }, 0, (UINT_PTR)STRING2}
1346 };
1347 static TBBUTTON buttons4[] = {
1348     {0, 40, TBSTATE_ENABLED, BTNS_AUTOSIZE, {0, }, 0, (UINT_PTR)STRING2},
1349     {0, 41, TBSTATE_ENABLED, 0, {0, }, 0, (UINT_PTR)STRING2},
1350     {0, 41, TBSTATE_ENABLED, BTNS_SHOWTEXT, {0, }, 0, (UINT_PTR)STRING2}
1351 };
1352 
1353 static void test_sizes(void)
1354 {
1355     HWND hToolbar = NULL;
1356     HIMAGELIST himl, himl2;
1357     TBBUTTONINFOA tbinfo;
1358     TBBUTTON button;
1359     int style;
1360     int i;
1361     int fontheight = system_font_height();
1362 
1363     init_tbsize_results();
1364 
1365     rebuild_toolbar_with_buttons(&hToolbar);
1366     style = GetWindowLongA(hToolbar, GWL_STYLE);
1367     ok(style == (WS_CHILD|WS_VISIBLE|CCS_TOP), "Invalid style %x\n", style);
1368     check_sizes();
1369     /* the TBSTATE_WRAP makes a second row */
1370     SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons1);
1371     check_sizes();
1372     SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0);
1373     check_sizes();
1374     SendMessageA(hToolbar, TB_GETBUTTON, 5, (LPARAM)&button);
1375     ok(button.fsState == (TBSTATE_WRAP|TBSTATE_ENABLED), "got %08x\n", button.fsState);
1376     /* after setting the TBSTYLE_WRAPABLE the TBSTATE_WRAP is ignored */
1377     SetWindowLongA(hToolbar, GWL_STYLE, style|TBSTYLE_WRAPABLE);
1378     check_sizes();
1379     SendMessageA(hToolbar, TB_GETBUTTON, 5, (LPARAM)&button);
1380     ok(button.fsState == TBSTATE_ENABLED, "got %08x\n", button.fsState);
1381     /* adding new buttons with TBSTYLE_WRAPABLE doesn't add a new row */
1382     SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons1);
1383     check_sizes();
1384     /* only after adding enough buttons the bar will be wrapped on a
1385      * separator and then on the first button */
1386     for (i=0; i<15; i++)
1387         SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons1);
1388     check_sizes_todo(0x4);
1389     SendMessageA(hToolbar, TB_GETBUTTON, 31, (LPARAM)&button);
1390     ok(button.fsState == (TBSTATE_WRAP|TBSTATE_ENABLED), "got %08x\n", button.fsState);
1391     SetWindowLongA(hToolbar, GWL_STYLE, style);
1392     SendMessageA(hToolbar, TB_GETBUTTON, 31, (LPARAM)&button);
1393     ok(button.fsState == TBSTATE_ENABLED, "got %08x\n", button.fsState);
1394 
1395     rebuild_toolbar_with_buttons(&hToolbar);
1396     SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons1);
1397     /* setting the buttons vertical will only change the window client size */
1398     SetWindowLongA(hToolbar, GWL_STYLE, style | CCS_VERT);
1399     SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0);
1400     check_sizes_todo(0x3c);
1401     /* with a TBSTYLE_WRAPABLE a wrapping will occur on the separator */
1402     SetWindowLongA(hToolbar, GWL_STYLE, style | TBSTYLE_WRAPABLE | CCS_VERT);
1403     SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0);
1404     check_sizes_todo(0x7c);
1405 
1406     rebuild_toolbar_with_buttons(&hToolbar);
1407     SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons1);
1408     /* a TB_SETBITMAPSIZE changes button sizes*/
1409     SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(24, 24));
1410     check_sizes();
1411 
1412     /* setting a TBSTYLE_FLAT doesn't change anything - even after a TB_AUTOSIZE */
1413     SetWindowLongA(hToolbar, GWL_STYLE, style | TBSTYLE_FLAT);
1414     SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0);
1415     check_sizes();
1416     /* but after a TB_SETBITMAPSIZE the top margins is changed */
1417     SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(20, 20));
1418     SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(24, 24));
1419     check_sizes();
1420     /* some vertical toolbar sizes */
1421     SetWindowLongA(hToolbar, GWL_STYLE, style | TBSTYLE_FLAT | TBSTYLE_WRAPABLE | CCS_VERT);
1422     check_sizes_todo(0x7c);
1423 
1424     rebuild_toolbar_with_buttons(&hToolbar);
1425     SetWindowLongA(hToolbar, GWL_STYLE, style | TBSTYLE_FLAT);
1426     /* newly added buttons will be use the previous margin */
1427     SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons2);
1428     check_sizes();
1429     /* TB_SETBUTTONSIZE can't be used to reduce the size of a button below the default */
1430     check_button_size(hToolbar, 23, 22);
1431     ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(22, 21))==1, "TB_SETBUTTONSIZE\n");
1432     check_button_size(hToolbar, 23, 22);
1433     ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(5, 100))==1, "TB_SETBUTTONSIZE\n");
1434     check_button_size(hToolbar, 23, 100);
1435     ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(3, 3))==1, "TB_SETBUTTONSIZE\n");
1436     check_button_size(hToolbar, 23, 22);
1437     ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(5, 100))==1, "TB_SETBUTTONSIZE\n");
1438     check_button_size(hToolbar, 23, 100);
1439     check_sizes();
1440     /* add some buttons with non-default sizes */
1441     SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons2);
1442     SendMessageA(hToolbar, TB_INSERTBUTTONA, -1, (LPARAM)&buttons2[0]);
1443     check_sizes();
1444     SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[0]);
1445     /* TB_ADDSTRINGA resets the size */
1446     SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM) STRING0 "\0" STRING1 "\0");
1447     check_button_size(hToolbar, 23, 23 + fontheight);
1448     check_sizes();
1449     /* TB_SETBUTTONSIZE can be used to crop the text */
1450     SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(3, 3));
1451     check_button_size(hToolbar, 23, 22);
1452     check_sizes();
1453     /* the default size is bitmap size + padding */
1454     SendMessageA(hToolbar, TB_SETPADDING, 0, MAKELONG(1, 1));
1455     SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(3, 3));
1456     check_button_size(hToolbar, 17, 17);
1457     SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(3, 3));
1458     SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(3, 3));
1459     check_button_size(hToolbar, 4, 4);
1460 
1461     rebuild_toolbar(&hToolbar);
1462     /* sending a TB_SETBITMAPSIZE with the same sizes is enough to make the button smaller */
1463     check_button_size(hToolbar, 23, 22);
1464     SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(16, 15));
1465     check_button_size(hToolbar, 23, 21);
1466     /* -1 in TB_SETBITMAPSIZE is a special code meaning that the coordinate shouldn't be changed */
1467     add_128x15_bitmap(hToolbar, 16);
1468     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(14, -1)), "TB_SETBITMAPSIZE failed\n");
1469     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(21, 21), "%x");
1470     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(-1, 12)), "TB_SETBITMAPSIZE failed\n");
1471     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(21, 18), "%x");
1472     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(-1, -1)), "TB_SETBITMAPSIZE failed\n");
1473     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(21, 18), "%x");
1474     /* check the imagelist */
1475     InvalidateRect(hToolbar, NULL, TRUE);
1476     UpdateWindow(hToolbar);
1477     CHECK_IMAGELIST(16, 14, 12);
1478 
1479     rebuild_toolbar(&hToolbar);
1480     SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)STRING0 "\0" STRING1 "\0");
1481     /* the height is increased after a TB_ADDSTRINGA */
1482     check_button_size(hToolbar, 23, 23 + fontheight);
1483     SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(100, 100));
1484     /* if a string is in the pool, even adding a button without a string resets the size */
1485     SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons2[0]);
1486     check_button_size(hToolbar, 23, 22);
1487     SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(100, 100));
1488     /* an BTNS_AUTOSIZE button is also considered when computing the new size */
1489     SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[2]);
1490     check_button_size(hToolbar, 7 + string_width(STRING1), 23 + fontheight);
1491     SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[0]);
1492     check_sizes();
1493     /* delete button doesn't change the buttons size */
1494     SendMessageA(hToolbar, TB_DELETEBUTTON, 2, 0);
1495     SendMessageA(hToolbar, TB_DELETEBUTTON, 1, 0);
1496     check_button_size(hToolbar, 7 + string_width(STRING1), 23 + fontheight);
1497     /* TB_INSERTBUTTONAS will */
1498     SendMessageA(hToolbar, TB_INSERTBUTTONA, 1, (LPARAM)&buttons2[0]);
1499     check_button_size(hToolbar, 23, 22);
1500 
1501     /* TB_HIDEBUTTON and TB_MOVEBUTTON doesn't force a recalc */
1502     SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(100, 100));
1503     ok(SendMessageA(hToolbar, TB_MOVEBUTTON, 0, 1), "TB_MOVEBUTTON failed\n");
1504     check_button_size(hToolbar, 100, 100);
1505     ok(SendMessageA(hToolbar, TB_HIDEBUTTON, 20, TRUE), "TB_HIDEBUTTON failed\n");
1506     check_button_size(hToolbar, 100, 100);
1507     /* however changing the hidden flag with TB_SETSTATE does */
1508     ok(SendMessageA(hToolbar, TB_SETSTATE, 20, TBSTATE_ENABLED|TBSTATE_HIDDEN), "TB_SETSTATE failed\n");
1509     check_button_size(hToolbar, 100, 100);
1510     ok(SendMessageA(hToolbar, TB_SETSTATE, 20, TBSTATE_ENABLED), "TB_SETSTATE failed\n");
1511     check_button_size(hToolbar, 23, 22);
1512 
1513     /* TB_SETIMAGELIST always changes the height but the width only if necessary */
1514     SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(100, 100));
1515     himl = pImageList_LoadImageA(GetModuleHandleA(NULL), (LPCSTR)MAKEINTRESOURCE(IDB_BITMAP_80x15),
1516                                 20, 2, CLR_NONE, IMAGE_BITMAP, LR_DEFAULTCOLOR);
1517     ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, (LPARAM)himl) == 0, "TB_SETIMAGELIST failed\n");
1518     check_button_size(hToolbar, 100, 21);
1519     SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(100, 100));
1520     check_button_size(hToolbar, 100, 100);
1521     /* But there are no update when we change imagelist, and image sizes are the same */
1522     himl2 = pImageList_LoadImageA(GetModuleHandleA(NULL), (LPCSTR)MAKEINTRESOURCE(IDB_BITMAP_128x15),
1523                                  20, 2, CLR_NONE, IMAGE_BITMAP, LR_DEFAULTCOLOR);
1524     ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, (LRESULT)himl2) == (LRESULT)himl, "TB_SETIMAGELIST failed\n");
1525     check_button_size(hToolbar, 100, 100);
1526     SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(1, 1));
1527     check_button_size(hToolbar, 27, 21);
1528     ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, 0) == (LRESULT)himl2, "TB_SETIMAGELIST failed\n");
1529     check_button_size(hToolbar, 27, 7);
1530     SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(1, 1));
1531     check_button_size(hToolbar, 8, 7)
1532     ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, (LPARAM)himl) == 0, "TB_SETIMAGELIST failed\n");
1533     check_button_size(hToolbar, 27, 21)
1534     /* the text is taken into account */
1535     SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)STRING0 "\0" STRING1 "\0");
1536     SendMessageA(hToolbar, TB_ADDBUTTONSA, 4, (LPARAM)buttons3);
1537     check_button_size(hToolbar, 7 + string_width(STRING1), 22 + fontheight);
1538     ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, 0) == (LRESULT)himl, "TB_SETIMAGELIST failed\n");
1539     check_button_size(hToolbar, 7 + string_width(STRING1), 8 + fontheight);
1540     /* the style change also comes into effect */
1541     check_sizes();
1542     SetWindowLongA(hToolbar, GWL_STYLE, GetWindowLongA(hToolbar, GWL_STYLE) | TBSTYLE_FLAT);
1543     ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, (LPARAM)himl) == 0, "TB_SETIMAGELIST failed\n");
1544     check_sizes_todo(0x30);     /* some small problems with BTNS_AUTOSIZE button sizes */
1545 
1546     rebuild_toolbar(&hToolbar);
1547     pImageList_Destroy(himl);
1548     pImageList_Destroy(himl2);
1549 
1550     SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[3]);
1551     check_button_size(hToolbar, 7 + string_width(STRING2), 23 + fontheight);
1552     SendMessageA(hToolbar, TB_DELETEBUTTON, 0, 0);
1553     check_button_size(hToolbar, 7 + string_width(STRING2), 23 + fontheight);
1554 
1555     rebuild_toolbar(&hToolbar);
1556 
1557     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELPARAM(32, 32)) == 1, "TB_SETBITMAPSIZE failed\n");
1558     ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(100, 100)) == 1, "TB_SETBUTTONSIZE failed\n");
1559     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons2[0]) == 1, "TB_ADDBUTTONSA failed\n");
1560     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[2]) == 1, "TB_ADDBUTTONSA failed\n");
1561     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[0]) == 1, "TB_ADDBUTTONSA failed\n");
1562     SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 );
1563     check_sizes();
1564 
1565     rebuild_toolbar(&hToolbar);
1566     SetWindowLongA(hToolbar, GWL_STYLE, TBSTYLE_LIST | GetWindowLongA(hToolbar, GWL_STYLE));
1567     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELPARAM(32, 32)) == 1, "TB_SETBITMAPSIZE failed\n");
1568     ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(100, 100)) == 1, "TB_SETBUTTONSIZE failed\n");
1569     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons2[0]) == 1, "TB_ADDBUTTONSA failed\n");
1570     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[2]) == 1, "TB_ADDBUTTONSA failed\n");
1571     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[3]) == 1, "TB_ADDBUTTONSA failed\n");
1572     SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 );
1573     check_sizes_todo(0xff);
1574 
1575     rebuild_toolbar(&hToolbar);
1576     SetWindowLongA(hToolbar, GWL_STYLE, TBSTYLE_LIST | GetWindowLongA(hToolbar, GWL_STYLE));
1577     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELPARAM(32, 32)) == 1, "TB_SETBITMAPSIZE failed\n");
1578     ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(100, 100)) == 1, "TB_SETBUTTONSIZE failed\n");
1579     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[3]) == 1, "TB_ADDBUTTONSA failed\n");
1580     SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 );
1581     check_sizes();
1582 
1583     rebuild_toolbar(&hToolbar);
1584     SetWindowLongA(hToolbar, GWL_STYLE, TBSTYLE_WRAPABLE | GetWindowLongA(hToolbar, GWL_STYLE));
1585     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[3]) == 1, "TB_ADDBUTTONSA failed\n");
1586     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[3]) == 1, "TB_ADDBUTTONSA failed\n");
1587     tbinfo.cx = 672;
1588     tbinfo.cbSize = sizeof(TBBUTTONINFOA);
1589     tbinfo.dwMask = TBIF_SIZE | TBIF_BYINDEX;
1590     if (SendMessageA(hToolbar, TB_SETBUTTONINFOA, 0, (LPARAM)&tbinfo))
1591     {
1592         ok(SendMessageA(hToolbar, TB_SETBUTTONINFOA, 1, (LPARAM)&tbinfo) != 0, "TB_SETBUTTONINFOA failed\n");
1593         SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0);
1594         check_sizes();
1595     }
1596     else  /* TBIF_BYINDEX probably not supported, confirm that this was the reason for the failure */
1597     {
1598         tbinfo.dwMask = TBIF_SIZE;
1599         ok(SendMessageA(hToolbar, TB_SETBUTTONINFOA, 33, (LPARAM)&tbinfo) != 0, "TB_SETBUTTONINFOA failed\n");
1600         tbsize_numtests++;
1601     }
1602 
1603     /* Single BTNS_AUTOSIZE button with string. */
1604     rebuild_toolbar(&hToolbar);
1605     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons4[0]) == 1, "TB_ADDBUTTONSA failed\n");
1606     ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(40, 20)) == 1, "TB_SETBUTTONSIZE failed\n");
1607     SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 );
1608     check_sizes();
1609 
1610     /* Single non-BTNS_AUTOSIZE button with string. */
1611     rebuild_toolbar(&hToolbar);
1612     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons4[1]) == 1, "TB_ADDBUTTONSA failed\n");
1613     ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(40, 20)) == 1, "TB_SETBUTTONSIZE failed\n");
1614     SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 );
1615     check_sizes();
1616 
1617     /* Single non-BTNS_AUTOSIZE button with string with TBSTYLE_EX_MIXEDBUTTONS set. */
1618     rebuild_toolbar(&hToolbar);
1619     SendMessageA(hToolbar, TB_SETEXTENDEDSTYLE, 0, TBSTYLE_EX_MIXEDBUTTONS);
1620     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons4[1]) == 1, "TB_ADDBUTTONSA failed\n");
1621     ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(40, 20)) == 1, "TB_SETBUTTONSIZE failed\n");
1622     SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 );
1623     check_sizes();
1624 
1625     /* Single non-BTNS_AUTOSIZE, BTNS_SHOWTEXT button with string with TBSTYLE_EX_MIXEDBUTTONS set. */
1626     rebuild_toolbar(&hToolbar);
1627     SendMessageA(hToolbar, TB_SETEXTENDEDSTYLE, 0, TBSTYLE_EX_MIXEDBUTTONS);
1628     ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons4[2]) == 1, "TB_ADDBUTTONSA failed\n");
1629     ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(40, 20)) == 1, "TB_SETBUTTONSIZE failed\n");
1630     SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 );
1631     check_sizes();
1632 
1633     free_tbsize_results();
1634     DestroyWindow(hToolbar);
1635 }
1636 
1637 /* Toolbar control has two ways of reacting to a change. We call them a
1638  * relayout and recalc. A recalc forces a recompute of values like button size
1639  * and top margin (the latter in comctl32 <v6), while a relayout uses the cached
1640  * values. This functions creates a flat toolbar with a top margin of a non-flat
1641  * toolbar. We will notice a recalc, as it will recompte the top margin and
1642  * change it to zero*/
1643 static void prepare_recalc_test(HWND *phToolbar)
1644 {
1645     RECT rect;
1646     rebuild_toolbar_with_buttons(phToolbar);
1647     SetWindowLongA(*phToolbar, GWL_STYLE,
1648         GetWindowLongA(*phToolbar, GWL_STYLE) | TBSTYLE_FLAT);
1649     SendMessageA(*phToolbar, TB_GETITEMRECT, 1, (LPARAM)&rect);
1650     ok(rect.top == 2, "Test will make no sense because initial top is %d instead of 2\n",
1651         rect.top);
1652 }
1653 
1654 static BOOL did_recalc(HWND hToolbar)
1655 {
1656     RECT rect;
1657     SendMessageA(hToolbar, TB_GETITEMRECT, 1, (LPARAM)&rect);
1658     ok(rect.top == 2 || rect.top == 0, "Unexpected top margin %d in recalc test\n",
1659         rect.top);
1660     return (rect.top == 0);
1661 }
1662 
1663 /* call after a recalc did happen to return to an unstable state */
1664 static void restore_recalc_state(HWND hToolbar)
1665 {
1666     RECT rect;
1667     /* return to style with a 2px top margin */
1668     SetWindowLongA(hToolbar, GWL_STYLE,
1669                    SendMessageA(hToolbar, TB_GETSTYLE, 0, 0) & ~TBSTYLE_FLAT);
1670     /* recalc */
1671     SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[3]);
1672     /* top margin will be 0px if a recalc occurs */
1673     SetWindowLongA(hToolbar, GWL_STYLE,
1674                    SendMessageA(hToolbar, TB_GETSTYLE, 0, 0) | TBSTYLE_FLAT);
1675     /* safety check */
1676     SendMessageA(hToolbar, TB_GETITEMRECT, 1, (LPARAM)&rect);
1677     ok(rect.top == 2, "Test will make no sense because initial top is %d instead of 2\n",
1678         rect.top);
1679 }
1680 
1681 static void test_recalc(void)
1682 {
1683     HWND hToolbar = NULL;
1684     TBBUTTONINFOA bi;
1685     CHAR test[] = "Test";
1686     const int EX_STYLES_COUNT = 5;
1687     int i;
1688     BOOL recalc;
1689     DWORD style;
1690 
1691     /* Like TB_ADDBUTTONSA tested in test_sized, inserting a button without text
1692      * results in a relayout, while adding one with text forces a recalc */
1693     prepare_recalc_test(&hToolbar);
1694     SendMessageA(hToolbar, TB_INSERTBUTTONA, 1, (LPARAM)&buttons3[0]);
1695     recalc = did_recalc(hToolbar);
1696     ok(!recalc, "Unexpected recalc - adding button without text\n");
1697 
1698     prepare_recalc_test(&hToolbar);
1699     SendMessageA(hToolbar, TB_INSERTBUTTONA, 1, (LPARAM)&buttons3[3]);
1700     recalc = did_recalc(hToolbar);
1701     ok(recalc, "Expected a recalc - adding button with text\n");
1702 
1703     /* TB_SETBUTTONINFOA, even when adding a text, results only in a relayout */
1704     prepare_recalc_test(&hToolbar);
1705     bi.cbSize = sizeof(bi);
1706     bi.dwMask = TBIF_TEXT;
1707     bi.pszText = test;
1708     SendMessageA(hToolbar, TB_SETBUTTONINFOA, 1, (LPARAM)&bi);
1709     recalc = did_recalc(hToolbar);
1710     ok(!recalc, "Unexpected recalc - setting a button text\n");
1711 
1712     /* most extended styled doesn't force a recalc (testing all the bits gives
1713      * the same results, but prints some ERRs while testing) */
1714     for (i = 0; i < EX_STYLES_COUNT; i++)
1715     {
1716         if (i == 1 || i == 3)  /* an undoc style and TBSTYLE_EX_MIXEDBUTTONS */
1717             continue;
1718         prepare_recalc_test(&hToolbar);
1719         expect(0, (int)SendMessageA(hToolbar, TB_GETEXTENDEDSTYLE, 0, 0));
1720         SendMessageA(hToolbar, TB_SETEXTENDEDSTYLE, 0, (1 << i));
1721         recalc = did_recalc(hToolbar);
1722         ok(!recalc, "Unexpected recalc - setting bit %d\n", i);
1723         SendMessageA(hToolbar, TB_SETEXTENDEDSTYLE, 0, 0);
1724         recalc = did_recalc(hToolbar);
1725         ok(!recalc, "Unexpected recalc - clearing bit %d\n", i);
1726         expect(0, (int)SendMessageA(hToolbar, TB_GETEXTENDEDSTYLE, 0, 0));
1727     }
1728 
1729     /* TBSTYLE_EX_MIXEDBUTTONS does a recalc on change */
1730     prepare_recalc_test(&hToolbar);
1731     SendMessageA(hToolbar, TB_SETEXTENDEDSTYLE, 0, TBSTYLE_EX_MIXEDBUTTONS);
1732     recalc = did_recalc(hToolbar);
1733     if (recalc)
1734     {
1735         ok(recalc, "Expected a recalc - setting TBSTYLE_EX_MIXEDBUTTONS\n");
1736         restore_recalc_state(hToolbar);
1737         SendMessageA(hToolbar, TB_SETEXTENDEDSTYLE, 0, TBSTYLE_EX_MIXEDBUTTONS);
1738         recalc = did_recalc(hToolbar);
1739         ok(!recalc, "Unexpected recalc - setting TBSTYLE_EX_MIXEDBUTTONS again\n");
1740         restore_recalc_state(hToolbar);
1741         SendMessageA(hToolbar, TB_SETEXTENDEDSTYLE, 0, 0);
1742         recalc = did_recalc(hToolbar);
1743         ok(recalc, "Expected a recalc - clearing TBSTYLE_EX_MIXEDBUTTONS\n");
1744     }
1745     else win_skip( "No recalc on TBSTYLE_EX_MIXEDBUTTONS\n" );
1746 
1747     /* undocumented exstyle 0x2 seems to change the top margin, which
1748      * interferes with these tests */
1749 
1750     /* Show that a change in TBSTYLE_WRAPABLE causes a recalc */
1751     prepare_recalc_test(&hToolbar);
1752     style = SendMessageA(hToolbar, TB_GETSTYLE, 0, 0);
1753     SendMessageA(hToolbar, TB_SETSTYLE, 0, style);
1754     recalc = did_recalc(hToolbar);
1755     ok(!recalc, "recalc %d\n", recalc);
1756 
1757     SendMessageA(hToolbar, TB_SETSTYLE, 0, style | TBSTYLE_TOOLTIPS | TBSTYLE_TRANSPARENT | CCS_BOTTOM);
1758     recalc = did_recalc(hToolbar);
1759     ok(!recalc, "recalc %d\n", recalc);
1760 
1761     SendMessageA(hToolbar, TB_SETSTYLE, 0, style | TBSTYLE_WRAPABLE);
1762     recalc = did_recalc(hToolbar);
1763     ok(recalc, "recalc %d\n", recalc);
1764     restore_recalc_state(hToolbar);
1765 
1766     SendMessageA(hToolbar, TB_SETSTYLE, 0, style | TBSTYLE_WRAPABLE);
1767     recalc = did_recalc(hToolbar);
1768     ok(!recalc, "recalc %d\n", recalc);
1769 
1770     SendMessageA(hToolbar, TB_SETSTYLE, 0, style);
1771     recalc = did_recalc(hToolbar);
1772     ok(recalc, "recalc %d\n", recalc);
1773     restore_recalc_state(hToolbar);
1774 
1775     /* Changing CCS_VERT does not recalc */
1776     SendMessageA(hToolbar, TB_SETSTYLE, 0, style | CCS_VERT);
1777     recalc = did_recalc(hToolbar);
1778     ok(!recalc, "recalc %d\n", recalc);
1779     restore_recalc_state(hToolbar);
1780 
1781     SendMessageA(hToolbar, TB_SETSTYLE, 0, style);
1782     recalc = did_recalc(hToolbar);
1783     ok(!recalc, "recalc %d\n", recalc);
1784     restore_recalc_state(hToolbar);
1785 
1786     /* Setting the window's style directly also causes recalc */
1787     SetWindowLongA(hToolbar, GWL_STYLE, style | TBSTYLE_WRAPABLE);
1788     recalc = did_recalc(hToolbar);
1789     ok(recalc, "recalc %d\n", recalc);
1790 
1791     DestroyWindow(hToolbar);
1792 }
1793 
1794 static void test_getbuttoninfo(void)
1795 {
1796     HWND hToolbar = NULL;
1797     TBBUTTONINFOW tbiW;
1798     TBBUTTONINFOA tbi;
1799     int i;
1800 
1801     rebuild_toolbar_with_buttons(&hToolbar);
1802     for (i = 0; i < 128; i++)
1803     {
1804         int ret;
1805 
1806         tbi.cbSize = i;
1807         tbi.dwMask = TBIF_COMMAND;
1808         ret = (int)SendMessageA(hToolbar, TB_GETBUTTONINFOA, 1, (LPARAM)&tbi);
1809         if (i == sizeof(TBBUTTONINFOA)) {
1810             compare(ret, 0, "%d");
1811         } else {
1812             compare(ret, -1, "%d");
1813         }
1814     }
1815 
1816     /* TBIF_TEXT with NULL pszText */
1817     memset(&tbiW, 0, sizeof(tbiW));
1818     tbiW.cbSize = sizeof(tbiW);
1819     tbiW.dwMask = TBIF_BYINDEX | TBIF_STYLE | TBIF_COMMAND | TBIF_TEXT;
1820     i = SendMessageA(hToolbar, TB_GETBUTTONINFOW, 1, (LPARAM)&tbiW);
1821     ok(i == 1, "Got index %d\n", i);
1822 
1823     DestroyWindow(hToolbar);
1824 }
1825 
1826 static void test_createtoolbarex(void)
1827 {
1828     HWND hToolbar;
1829     TBBUTTON btns[3];
1830     ZeroMemory(&btns, sizeof(btns));
1831 
1832     hToolbar = pCreateToolbarEx(hMainWnd, WS_VISIBLE, 1, 16, GetModuleHandleA(NULL), IDB_BITMAP_128x15, btns,
1833         3, 20, 20, 16, 16, sizeof(TBBUTTON));
1834     CHECK_IMAGELIST(16, 20, 20);
1835     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), 0x1a001b, "%x");
1836     DestroyWindow(hToolbar);
1837 
1838     hToolbar = pCreateToolbarEx(hMainWnd, WS_VISIBLE, 1, 16, GetModuleHandleA(NULL), IDB_BITMAP_128x15, btns,
1839         3, 4, 4, 16, 16, sizeof(TBBUTTON));
1840     CHECK_IMAGELIST(32, 4, 4);
1841     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), 0xa000b, "%x");
1842     DestroyWindow(hToolbar);
1843 
1844     hToolbar = pCreateToolbarEx(hMainWnd, WS_VISIBLE, 1, 16, GetModuleHandleA(NULL), IDB_BITMAP_128x15, btns,
1845         3, 0, 8, 12, 12, sizeof(TBBUTTON));
1846     CHECK_IMAGELIST(16, 12, 12);
1847     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), 0x120013, "%x");
1848     DestroyWindow(hToolbar);
1849 
1850     hToolbar = pCreateToolbarEx(hMainWnd, WS_VISIBLE, 1, 16, GetModuleHandleA(NULL), IDB_BITMAP_128x15, btns,
1851         3, -1, 8, 12, 12, sizeof(TBBUTTON));
1852     CHECK_IMAGELIST(16, 12, 8);
1853     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), 0xe0013, "%x");
1854     DestroyWindow(hToolbar);
1855 
1856     hToolbar = pCreateToolbarEx(hMainWnd, WS_VISIBLE, 1, 16, GetModuleHandleA(NULL), IDB_BITMAP_128x15, btns,
1857         3, -1, 8, -1, 12, sizeof(TBBUTTON));
1858     CHECK_IMAGELIST(16, 16, 8);
1859     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), 0xe0017, "%x");
1860     DestroyWindow(hToolbar);
1861 
1862     hToolbar = pCreateToolbarEx(hMainWnd, WS_VISIBLE, 1, 16, GetModuleHandleA(NULL), IDB_BITMAP_128x15, btns,
1863         3, 0, 0, 12, -1, sizeof(TBBUTTON));
1864     CHECK_IMAGELIST(16, 12, 16);
1865     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), 0x160013, "%x");
1866     DestroyWindow(hToolbar);
1867 
1868     hToolbar = pCreateToolbarEx(hMainWnd, WS_VISIBLE, 1, 16, GetModuleHandleA(NULL), IDB_BITMAP_128x15, btns,
1869         3, 0, 0, 0, 12, sizeof(TBBUTTON));
1870     CHECK_IMAGELIST(16, 16, 16);
1871     compare((int)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), 0x160017, "%x");
1872     DestroyWindow(hToolbar);
1873 }
1874 
1875 static void test_dispinfo(void)
1876 {
1877     HWND hToolbar = NULL;
1878     const TBBUTTON buttons_disp[] = {
1879         {-1, 20, TBSTATE_ENABLED, 0, {0, }, 0, -1},
1880         {0,  21, TBSTATE_ENABLED, 0, {0, }, 0, -1},
1881     };
1882     BOOL ret;
1883 
1884     rebuild_toolbar(&hToolbar);
1885     SendMessageA(hToolbar, TB_LOADIMAGES, IDB_HIST_SMALL_COLOR, (LPARAM)HINST_COMMCTRL);
1886     SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons_disp);
1887     g_dwExpectedDispInfoMask = TBNF_IMAGE;
1888     /* Some TBN_GETDISPINFO tests will be done in MyWnd_Notify function.
1889      * We will receive TBN_GETDISPINFOW even if the control is ANSI */
1890     compare((BOOL)SendMessageA(hToolbar, CCM_GETUNICODEFORMAT, 0, 0), 0, "%d");
1891     ShowWindow(hToolbar, SW_SHOW);
1892     UpdateWindow(hToolbar);
1893 
1894     ret = (BOOL)SendMessageA(hToolbar, CCM_SETUNICODEFORMAT, TRUE, 0);
1895     compare(ret, FALSE, "%d");
1896     compare(SendMessageA(hToolbar, CCM_GETUNICODEFORMAT, 0, 0), 1L, "%ld");
1897     InvalidateRect(hToolbar, NULL, FALSE);
1898     UpdateWindow(hToolbar);
1899 
1900     ret = (BOOL)SendMessageA(hToolbar, CCM_SETUNICODEFORMAT, FALSE, 0);
1901     compare(ret, TRUE, "%d");
1902     compare(SendMessageA(hToolbar, CCM_GETUNICODEFORMAT, 0, 0), 0L, "%ld");
1903     InvalidateRect(hToolbar, NULL, FALSE);
1904     UpdateWindow(hToolbar);
1905 
1906     DestroyWindow(hToolbar);
1907     g_dwExpectedDispInfoMask = 0;
1908 }
1909 
1910 typedef struct
1911 {
1912     int  nRows;
1913     BOOL bLarger;
1914     int  expectedRows;
1915 } tbrows_result_t;
1916 
1917 static tbrows_result_t tbrows_results[] =
1918 {
1919     {1, TRUE,  1}, /* 0: Simple case 9 in a row */
1920     {2, TRUE,  2}, /* 1: Another simple case 5 on one row, 4 on another*/
1921     {3, FALSE, 3}, /* 2: 3 lines - should be 3 lines of 3 buttons */
1922     {8, FALSE, 5}, /* 3: 8 lines - should be 5 lines of 2 buttons */
1923     {8, TRUE,  9}, /* 4: 8 lines but grow - should be 9 lines */
1924     {1, TRUE,  1}  /* 5: Back to simple case */
1925 };
1926 
1927 static void test_setrows(void)
1928 {
1929     TBBUTTON buttons[9];
1930     HWND hToolbar;
1931     DWORD i;
1932 
1933     for (i=0; i<9; i++)
1934         MakeButton(buttons+i, 1000+i, TBSTYLE_FLAT | TBSTYLE_CHECKGROUP, 0);
1935 
1936     /* Test 1 - 9 buttons */
1937     hToolbar = pCreateToolbarEx(hMainWnd,
1938         WS_VISIBLE | WS_CLIPCHILDREN | WS_CHILD | CCS_NORESIZE | CCS_NOPARENTALIGN
1939         | CCS_NOMOVEY | CCS_TOP,
1940         0,
1941         0, NULL, 0,
1942         buttons, ARRAY_SIZE(buttons),
1943         20, 20, 0, 0, sizeof(TBBUTTON));
1944     ok(hToolbar != NULL, "Toolbar creation\n");
1945     ok(SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0) == 0, "TB_AUTOSIZE failed\n");
1946 
1947     /* test setting rows to each of 1-10 with bLarger true and false */
1948     for (i=0; i<ARRAY_SIZE(tbrows_results); i++) {
1949         RECT rc;
1950         int rows;
1951 
1952         memset(&rc, 0xCC, sizeof(rc));
1953         SendMessageA(hToolbar, TB_SETROWS,
1954                      MAKELONG(tbrows_results[i].nRows, tbrows_results[i].bLarger),
1955                      (LPARAM) &rc);
1956 
1957         rows = SendMessageA(hToolbar, TB_GETROWS, MAKELONG(0,0), MAKELONG(0,0));
1958         ok(rows == tbrows_results[i].expectedRows,
1959                    "[%d] Unexpected number of rows %d (expected %d)\n", i, rows,
1960                    tbrows_results[i].expectedRows);
1961     }
1962 
1963     DestroyWindow(hToolbar);
1964 }
1965 
1966 static void test_getstring(void)
1967 {
1968     HWND hToolbar = NULL;
1969     char str[10];
1970     WCHAR strW[10];
1971     static const char answer[] = "STR";
1972     static const WCHAR answerW[] = { 'S','T','R',0 };
1973     INT r;
1974 
1975     hToolbar = CreateWindowExA(0, TOOLBARCLASSNAMEA, NULL, WS_CHILD | WS_VISIBLE, 0, 0, 0, 0, hMainWnd, (HMENU)5, GetModuleHandleA(NULL), NULL);
1976     ok(hToolbar != NULL, "Toolbar creation problem\n");
1977 
1978     r = SendMessageA(hToolbar, TB_GETSTRINGA, MAKEWPARAM(0, 0), 0);
1979     if (r == 0)
1980     {
1981         win_skip("TB_GETSTRINGA and TB_GETSTRINGW need 5.80\n");
1982         DestroyWindow(hToolbar);
1983         return;
1984     }
1985     expect(-1, r);
1986     r = SendMessageW(hToolbar, TB_GETSTRINGW, MAKEWPARAM(0, 0), 0);
1987     expect(-1, r);
1988     r = SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)answer);
1989     expect(0, r);
1990     r = SendMessageA(hToolbar, TB_GETSTRINGA, MAKEWPARAM(0, 0), 0);
1991     expect(lstrlenA(answer), r);
1992     r = SendMessageW(hToolbar, TB_GETSTRINGW, MAKEWPARAM(0, 0), 0);
1993     expect(lstrlenA(answer), r);
1994     r = SendMessageA(hToolbar, TB_GETSTRINGA, MAKEWPARAM(sizeof(str), 0), (LPARAM)str);
1995     expect(lstrlenA(answer), r);
1996     expect(0, lstrcmpA(answer, str));
1997     r = SendMessageW(hToolbar, TB_GETSTRINGW, MAKEWPARAM(sizeof(strW), 0), (LPARAM)strW);
1998     expect(lstrlenA(answer), r);
1999     expect(0, lstrcmpW(answerW, strW));
2000 
2001     DestroyWindow(hToolbar);
2002 }
2003 
2004 static void test_tooltip(void)
2005 {
2006     HWND hToolbar = NULL;
2007     const TBBUTTON buttons_disp[] = {
2008         {-1, 20, TBSTATE_ENABLED, 0, {0, }, 0, -1},
2009         {0,  21, TBSTATE_ENABLED, 0, {0, }, 0, -1},
2010     };
2011     NMTTDISPINFOW nmtti;
2012     HWND tooltip;
2013 
2014     rebuild_toolbar(&hToolbar);
2015 
2016     SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons_disp);
2017 
2018     /* W used to get through toolbar code that assumes tooltip is always Unicode */
2019     memset(&nmtti, 0, sizeof(nmtti));
2020     nmtti.hdr.code = TTN_GETDISPINFOW;
2021     nmtti.hdr.idFrom = 20;
2022 
2023     SendMessageA(hToolbar, CCM_SETUNICODEFORMAT, FALSE, 0);
2024 
2025     flush_sequences(sequences, NUM_MSG_SEQUENCES);
2026     SendMessageA(hToolbar, WM_NOTIFY, 0, (LPARAM)&nmtti);
2027     ok_sequence(sequences, PARENT_SEQ_INDEX, ttgetdispinfo_parent_seq,
2028                 "dispinfo from tooltip", FALSE);
2029 
2030     g_ResetDispTextPtr = TRUE;
2031     SendMessageA(hToolbar, WM_NOTIFY, 0, (LPARAM)&nmtti);
2032     g_ResetDispTextPtr = FALSE;
2033 
2034     DestroyWindow(hToolbar);
2035 
2036     /* TBSTYLE_TOOLTIPS */
2037     hToolbar = CreateWindowExA(0, TOOLBARCLASSNAMEA, NULL, WS_CHILD | WS_VISIBLE, 0, 0, 0, 0,
2038         hMainWnd, (HMENU)5, GetModuleHandleA(NULL), NULL);
2039     tooltip = (HWND)SendMessageA(hToolbar, TB_GETTOOLTIPS, 0, 0);
2040     ok(tooltip == NULL, "got %p\n", tooltip);
2041     DestroyWindow(hToolbar);
2042 }
2043 
2044 static void test_get_set_style(void)
2045 {
2046     TBBUTTON buttons[9];
2047     DWORD style, style2, ret;
2048     HWND hToolbar;
2049     int i;
2050 
2051     for (i=0; i<9; i++)
2052         MakeButton(buttons+i, 1000+i, TBSTYLE_CHECKGROUP, 0);
2053     MakeButton(buttons+3, 1003, TBSTYLE_SEP|TBSTYLE_GROUP, 0);
2054     MakeButton(buttons+6, 1006, TBSTYLE_SEP, 0);
2055 
2056     hToolbar = pCreateToolbarEx(hMainWnd,
2057         WS_VISIBLE | WS_CLIPCHILDREN | CCS_TOP |
2058         WS_CHILD | TBSTYLE_LIST,
2059         100,
2060         0, NULL, 0,
2061         buttons, ARRAY_SIZE(buttons),
2062         0, 0, 20, 16, sizeof(TBBUTTON));
2063     ok(hToolbar != NULL, "Toolbar creation\n");
2064     SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)"test\000");
2065 
2066     style = SendMessageA(hToolbar, TB_GETSTYLE, 0, 0);
2067     style2 = GetWindowLongA(hToolbar, GWL_STYLE);
2068 todo_wine
2069     ok(style == style2, "got 0x%08x, expected 0x%08x\n", style, style2);
2070 
2071     /* try to alter common window bits */
2072     style2 |= WS_BORDER;
2073     ret = SendMessageA(hToolbar, TB_SETSTYLE, 0, style2);
2074     ok(ret == 0, "got %d\n", ret);
2075     style = SendMessageA(hToolbar, TB_GETSTYLE, 0, 0);
2076     style2 = GetWindowLongA(hToolbar, GWL_STYLE);
2077     ok((style != style2) && (style == (style2 | WS_BORDER)),
2078         "got 0x%08x, expected 0x%08x\n", style, style2);
2079     ok(style & WS_BORDER, "got 0x%08x\n", style);
2080 
2081     /* now styles are the same, alter window style */
2082     ret = SendMessageA(hToolbar, TB_SETSTYLE, 0, style2);
2083     ok(ret == 0, "got %d\n", ret);
2084     style2 |= WS_BORDER;
2085     SetWindowLongA(hToolbar, GWL_STYLE, style2);
2086     style = SendMessageA(hToolbar, TB_GETSTYLE, 0, 0);
2087     ok(style == style2, "got 0x%08x, expected 0x%08x\n", style, style2);
2088 
2089     DestroyWindow(hToolbar);
2090 }
2091 
2092 static HHOOK g_tbhook;
2093 static HWND g_toolbar;
2094 
2095 DEFINE_EXPECT(g_hook_create);
2096 DEFINE_EXPECT(g_hook_WM_NCCREATE);
2097 DEFINE_EXPECT(g_hook_WM_CREATE);
2098 
2099 static LRESULT WINAPI toolbar_subclass_proc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
2100 {
2101     WNDPROC oldproc = (WNDPROC)GetWindowLongPtrA(hwnd, GWLP_USERDATA);
2102     LRESULT ret;
2103     DWORD style;
2104 
2105     if (msg == WM_NCCREATE)
2106     {
2107         if (g_toolbar == hwnd)
2108         {
2109             CHECK_EXPECT2(g_hook_WM_NCCREATE);
2110             g_toolbar = hwnd;
2111             ret = CallWindowProcA(oldproc, hwnd, msg, wParam, lParam);
2112 
2113             /* control is already set up */
2114             style = SendMessageA(hwnd, TB_GETSTYLE, 0, 0);
2115             ok(style != 0, "got %x\n", style);
2116 
2117             style = GetWindowLongA(hwnd, GWL_STYLE);
2118             ok((style & TBSTYLE_TOOLTIPS) == 0, "got 0x%08x\n", style);
2119             SetWindowLongA(hwnd, GWL_STYLE, style|TBSTYLE_TOOLTIPS);
2120             style = GetWindowLongA(hwnd, GWL_STYLE);
2121             ok((style & TBSTYLE_TOOLTIPS) == TBSTYLE_TOOLTIPS, "got 0x%08x\n", style);
2122 
2123             return ret;
2124         }
2125     }
2126     else if (msg == WM_CREATE)
2127     {
2128         CREATESTRUCTA *cs = (CREATESTRUCTA*)lParam;
2129 
2130         if (g_toolbar == hwnd)
2131         {
2132             CHECK_EXPECT2(g_hook_WM_CREATE);
2133 
2134             style = GetWindowLongA(hwnd, GWL_STYLE);
2135             ok((style & TBSTYLE_TOOLTIPS) == TBSTYLE_TOOLTIPS, "got 0x%08x\n", style);
2136 
2137             /* test if toolbar-specific messages are already working before WM_CREATE */
2138             style = SendMessageA(hwnd, TB_GETSTYLE, 0, 0);
2139             ok(style != 0, "got %x\n", style);
2140             ok((style & TBSTYLE_TOOLTIPS) == TBSTYLE_TOOLTIPS, "got 0x%x\n", style);
2141             ok((cs->style & TBSTYLE_TOOLTIPS) == 0, "0x%08x\n", cs->style);
2142 
2143             ret = CallWindowProcA(oldproc, hwnd, msg, wParam, lParam);
2144 
2145             style = GetWindowLongA(hwnd, GWL_STYLE);
2146             ok((style & TBSTYLE_TOOLTIPS) == TBSTYLE_TOOLTIPS, "got 0x%08x\n", style);
2147 
2148             /* test if toolbar-specific messages are already working before WM_CREATE */
2149             style = SendMessageA(hwnd, TB_GETSTYLE, 0, 0);
2150             ok(style != 0, "got %x\n", style);
2151             ok((style & TBSTYLE_TOOLTIPS) == TBSTYLE_TOOLTIPS, "got 0x%x\n", style);
2152 
2153             return ret;
2154         }
2155     }
2156 
2157     return CallWindowProcA(oldproc, hwnd, msg, wParam, lParam);
2158 }
2159 
2160 static LRESULT CALLBACK cbt_hook_proc(int code, WPARAM wParam, LPARAM lParam)
2161 {
2162     if (code == HCBT_CREATEWND)
2163     {
2164         HWND hwnd = (HWND)wParam;
2165 
2166         if (!g_toolbar)
2167         {
2168             WNDPROC oldproc;
2169 
2170             CHECK_EXPECT2(g_hook_create);
2171             g_toolbar = hwnd;
2172             /* subclass */
2173             oldproc = (WNDPROC)SetWindowLongPtrA(hwnd, GWLP_WNDPROC, (LONG_PTR)toolbar_subclass_proc);
2174             SetWindowLongPtrA(hwnd, GWLP_USERDATA, (LONG_PTR)oldproc);
2175         }
2176         return 0;
2177     }
2178 
2179     return CallNextHookEx(g_tbhook, code, wParam, lParam);
2180 }
2181 
2182 static void test_create(void)
2183 {
2184     HWND hwnd, tooltip;
2185     DWORD style;
2186 
2187     g_tbhook = SetWindowsHookA(WH_CBT, cbt_hook_proc);
2188 
2189     SET_EXPECT(g_hook_create);
2190     SET_EXPECT(g_hook_WM_NCCREATE);
2191     SET_EXPECT(g_hook_WM_CREATE);
2192 
2193     hwnd = CreateWindowExA(0, TOOLBARCLASSNAMEA, NULL, WS_CHILD | WS_VISIBLE, 0, 0, 0, 0,
2194         hMainWnd, (HMENU)5, GetModuleHandleA(NULL), NULL);
2195 
2196     CHECK_CALLED(g_hook_create);
2197     CHECK_CALLED(g_hook_WM_NCCREATE);
2198     CHECK_CALLED(g_hook_WM_CREATE);
2199 
2200     style = GetWindowLongA(hwnd, GWL_STYLE);
2201     ok((style & TBSTYLE_TOOLTIPS) == TBSTYLE_TOOLTIPS, "got 0x%08x\n", style);
2202 
2203     tooltip = (HWND)SendMessageA(hwnd, TB_GETTOOLTIPS, 0, 0);
2204     ok(tooltip != NULL, "got %p\n", tooltip);
2205     ok(GetParent(tooltip) == hMainWnd, "got %p, %p\n", hMainWnd, hwnd);
2206 
2207     DestroyWindow(hwnd);
2208     UnhookWindowsHook(WH_CBT, cbt_hook_proc);
2209 
2210     /* TBSTYLE_TRANSPARENT */
2211     hwnd = CreateWindowExA(0, TOOLBARCLASSNAMEA, NULL,
2212         WS_CHILD|WS_VISIBLE|WS_CLIPSIBLINGS|TBSTYLE_FLAT|TBSTYLE_TOOLTIPS|TBSTYLE_GROUP,
2213         0, 0, 0, 0, hMainWnd, (HMENU)5, GetModuleHandleA(NULL), NULL);
2214 
2215     style = GetWindowLongA(hwnd, GWL_STYLE);
2216     ok((style & TBSTYLE_TRANSPARENT) == TBSTYLE_TRANSPARENT, "got 0x%08x\n", style);
2217 
2218     style = SendMessageA(hwnd, TB_GETSTYLE, 0, 0);
2219     ok((style & TBSTYLE_TRANSPARENT) == TBSTYLE_TRANSPARENT, "got 0x%08x\n", style);
2220 
2221     DestroyWindow(hwnd);
2222 }
2223 
2224 typedef struct {
2225     DWORD mask;
2226     DWORD style;
2227     DWORD style_set;
2228 } extended_style_t;
2229 
2230 static const extended_style_t extended_style_test[] = {
2231     {
2232       TBSTYLE_EX_DRAWDDARROWS | TBSTYLE_EX_HIDECLIPPEDBUTTONS | TBSTYLE_EX_DOUBLEBUFFER,
2233       TBSTYLE_EX_DRAWDDARROWS | TBSTYLE_EX_HIDECLIPPEDBUTTONS | TBSTYLE_EX_DOUBLEBUFFER,
2234       TBSTYLE_EX_DRAWDDARROWS | TBSTYLE_EX_HIDECLIPPEDBUTTONS | TBSTYLE_EX_DOUBLEBUFFER
2235     },
2236     {
2237       TBSTYLE_EX_MIXEDBUTTONS, TBSTYLE_EX_MIXEDBUTTONS,
2238       TBSTYLE_EX_DRAWDDARROWS | TBSTYLE_EX_HIDECLIPPEDBUTTONS | TBSTYLE_EX_DOUBLEBUFFER | TBSTYLE_EX_MIXEDBUTTONS
2239     },
2240 
2241     { 0, TBSTYLE_EX_MIXEDBUTTONS, TBSTYLE_EX_MIXEDBUTTONS },
2242     { 0, 0, 0 },
2243     { 0, TBSTYLE_EX_DRAWDDARROWS, TBSTYLE_EX_DRAWDDARROWS },
2244     { 0, TBSTYLE_EX_HIDECLIPPEDBUTTONS, TBSTYLE_EX_HIDECLIPPEDBUTTONS },
2245 
2246     { 0, 0, 0 },
2247     { TBSTYLE_EX_HIDECLIPPEDBUTTONS, TBSTYLE_EX_MIXEDBUTTONS, 0 },
2248     { TBSTYLE_EX_MIXEDBUTTONS, TBSTYLE_EX_HIDECLIPPEDBUTTONS, 0 },
2249     { TBSTYLE_EX_DOUBLEBUFFER, TBSTYLE_EX_MIXEDBUTTONS, 0 },
2250 
2251     {
2252       TBSTYLE_EX_DOUBLEBUFFER | TBSTYLE_EX_MIXEDBUTTONS,
2253       TBSTYLE_EX_MIXEDBUTTONS, TBSTYLE_EX_MIXEDBUTTONS
2254     },
2255     {
2256       TBSTYLE_EX_DOUBLEBUFFER | TBSTYLE_EX_MIXEDBUTTONS,
2257       TBSTYLE_EX_DOUBLEBUFFER, TBSTYLE_EX_DOUBLEBUFFER
2258     }
2259 };
2260 
2261 static void test_TB_GET_SET_EXTENDEDSTYLE(void)
2262 {
2263     DWORD style, oldstyle, oldstyle2;
2264     const extended_style_t *ptr;
2265     HWND hwnd = NULL;
2266     int i;
2267 
2268     rebuild_toolbar(&hwnd);
2269 
2270     SendMessageA(hwnd, TB_SETEXTENDEDSTYLE, TBSTYLE_EX_DOUBLEBUFFER, TBSTYLE_EX_MIXEDBUTTONS);
2271     style = SendMessageA(hwnd, TB_GETEXTENDEDSTYLE, 0, 0);
2272     if (style == TBSTYLE_EX_MIXEDBUTTONS)
2273     {
2274         win_skip("Some extended style bits are not supported\n");
2275         DestroyWindow(hwnd);
2276         return;
2277     }
2278 
2279     for (i = 0; i < ARRAY_SIZE(extended_style_test); i++)
2280     {
2281         ptr = &extended_style_test[i];
2282 
2283         oldstyle2 = SendMessageA(hwnd, TB_GETEXTENDEDSTYLE, 0, 0);
2284 
2285         oldstyle = SendMessageA(hwnd, TB_SETEXTENDEDSTYLE, ptr->mask, ptr->style);
2286         ok(oldstyle == oldstyle2, "%d: got old style 0x%08x, expected 0x%08x\n", i, oldstyle, oldstyle2);
2287         style = SendMessageA(hwnd, TB_GETEXTENDEDSTYLE, 0, 0);
2288         ok(style == ptr->style_set, "%d: got style 0x%08x, expected 0x%08x\n", i, style, ptr->style_set);
2289     }
2290 
2291     /* Windows sets CCS_VERT when TB_GETEXTENDEDSTYLE is set */
2292     oldstyle2 = SendMessageA(hwnd, TB_GETEXTENDEDSTYLE, 0, 0);
2293     oldstyle = SendMessageA(hwnd, TB_SETEXTENDEDSTYLE, 0, TBSTYLE_EX_VERTICAL);
2294     ok(oldstyle == oldstyle2, "got old style 0x%08x, expected 0x%08x\n", oldstyle, oldstyle2);
2295     style = SendMessageA(hwnd, TB_GETEXTENDEDSTYLE, 0, 0);
2296     ok(style == TBSTYLE_EX_VERTICAL, "got style 0x%08x, expected 0x%08x\n", style, TBSTYLE_EX_VERTICAL);
2297     style = SendMessageA(hwnd, TB_GETSTYLE, 0, 0);
2298  todo_wine
2299     ok(style == CCS_VERT, "got style 0x%08x, expected CCS_VERT\n", style);
2300 
2301     DestroyWindow(hwnd);
2302 }
2303 
2304 static void test_noresize(void)
2305 {
2306     HWND wnd;
2307     int i;
2308     TBBUTTON button = {0, 10, TBSTATE_ENABLED, 0, {0, }, 0, -1};
2309 
2310     wnd = CreateWindowExA(0, TOOLBARCLASSNAMEA, NULL, WS_CHILD | WS_VISIBLE | CCS_NORESIZE | TBSTYLE_WRAPABLE, 0, 0, 100, 20,
2311                           hMainWnd, (HMENU)5, GetModuleHandleA(NULL), NULL);
2312     SendMessageA(wnd, TB_BUTTONSTRUCTSIZE, sizeof(TBBUTTON), 0);
2313 
2314     for (i=0; i<30; i++)
2315     {
2316         button.idCommand = 10 + i;
2317         SendMessageA(wnd, TB_ADDBUTTONSA, 1, (LPARAM)&button);
2318     }
2319 
2320     SendMessageA(wnd, TB_SETSTATE, 10, TBSTATE_WRAP|TBSTATE_ENABLED);
2321 
2322     /* autosize clears the wrap on button 0 */
2323     SendMessageA(wnd, TB_AUTOSIZE, 0, 0);
2324     for (i=0; i<30; i++)
2325     {
2326         SendMessageA(wnd, TB_GETBUTTON, i, (LPARAM)&button);
2327         if (i % 4 == 3)
2328             ok(button.fsState == (TBSTATE_WRAP|TBSTATE_ENABLED), "%d: got %08x\n", i, button.fsState);
2329         else
2330             ok(button.fsState == TBSTATE_ENABLED, "%d: got %08x\n", i, button.fsState);
2331     }
2332 
2333     /* changing the parent doesn't do anything */
2334     MoveWindow(hMainWnd, 0,0, 400, 200, FALSE);
2335     for (i=0; i<30; i++)
2336     {
2337         SendMessageA(wnd, TB_GETBUTTON, i, (LPARAM)&button);
2338         if (i % 4 == 3)
2339             ok(button.fsState == (TBSTATE_WRAP|TBSTATE_ENABLED), "%d: got %08x\n", i, button.fsState);
2340         else
2341             ok(button.fsState == TBSTATE_ENABLED, "%d: got %08x\n", i, button.fsState);
2342     }
2343 
2344     /* again nothing here */
2345     SendMessageA(wnd, TB_AUTOSIZE, 0, 0);
2346     for (i=0; i<30; i++)
2347     {
2348         SendMessageA(wnd, TB_GETBUTTON, i, (LPARAM)&button);
2349         if (i % 4 == 3)
2350             ok(button.fsState == (TBSTATE_WRAP|TBSTATE_ENABLED), "%d: got %08x\n", i, button.fsState);
2351         else
2352             ok(button.fsState == TBSTATE_ENABLED, "%d: got %08x\n", i, button.fsState);
2353     }
2354 
2355     DestroyWindow(wnd);
2356 
2357 }
2358 
2359 static void test_save(void)
2360 {
2361     HWND wnd = NULL;
2362     TBSAVEPARAMSW params;
2363     static const WCHAR subkey[] = {'S','o','f','t','w','a','r','e','\\','W','i','n','e','\\',
2364                                    'W','i','n','e','T','e','s','t',0};
2365     static const WCHAR value[] = {'t','o','o','l','b','a','r','t','e','s','t',0};
2366     LONG res;
2367     HKEY key;
2368     BYTE data[100];
2369     DWORD size = sizeof(data), type, i, count;
2370     TBBUTTON tb;
2371     static const TBBUTTON more_btns[2] =
2372         {
2373             {0, 11, TBSTATE_HIDDEN, BTNS_BUTTON, {0}, 0, -1},
2374             {0, 13, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 0, -1}
2375         };
2376     static const DWORD expect[] = {0xcafe, 1, 0xcafe0000, 3, 0xcafe0001, 5, 0xcafe0002, 7, 0xcafe0003,
2377                                    9, 0xcafe0004, 11, 0xcafe0005, 13, 0xcafe0006, 0xffffffff, 0xcafe0007,
2378                                    0xfffffffe, 0xcafe0008, 0x80000000, 0xcafe0009, 0x7fffffff, 0xcafe000a,
2379                                    0x100, 0xcafe000b};
2380     static const TBBUTTON expect_btns[] =
2381     {
2382         {0, 1, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 0, 0},
2383         {0, 3, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 1, 2},
2384         {0, 5, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 2, 0},
2385         {0, 7, 0, BTNS_BUTTON, {0}, 0, (INT_PTR)"foo"},
2386         {0, 9, 0, BTNS_BUTTON, {0}, 0, 0},
2387         {0, 11, 0, BTNS_BUTTON, {0}, 0, 3},
2388         {0, 13, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 6, 0},
2389         {0, 0, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 7, 0},
2390         {0, 0, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 8, 0},
2391         {0, 0, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 9, 0},
2392         {0, 0x7fffffff, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 0xa, 0},
2393     };
2394 
2395     params.hkr = HKEY_CURRENT_USER;
2396     params.pszSubKey = subkey;
2397     params.pszValueName = value;
2398 
2399     rebuild_toolbar_with_buttons( &wnd );
2400     SendMessageW(wnd, TB_ADDBUTTONSW, ARRAY_SIZE(more_btns), (LPARAM)more_btns);
2401 
2402     flush_sequences(sequences, NUM_MSG_SEQUENCES);
2403     res = SendMessageW( wnd, TB_SAVERESTOREW, TRUE, (LPARAM)&params );
2404     ok( res, "saving failed\n" );
2405     ok_sequence(sequences, PARENT_SEQ_INDEX, save_parent_seq, "save", FALSE);
2406     DestroyWindow( wnd );
2407 
2408     res = RegOpenKeyW( HKEY_CURRENT_USER, subkey, &key );
2409     ok( !res, "got %08x\n", res );
2410     res = RegQueryValueExW( key, value, NULL, &type, data, &size );
2411     ok( !res, "got %08x\n", res );
2412     ok( type == REG_BINARY, "got %08x\n", type );
2413     ok( size == sizeof(expect), "got %08x\n", size );
2414     ok( !memcmp( data, expect, size ), "mismatch\n" );
2415 
2416     RegCloseKey( key );
2417 
2418     wnd = NULL;
2419     rebuild_toolbar( &wnd );
2420 
2421     flush_sequences(sequences, NUM_MSG_SEQUENCES);
2422     res = SendMessageW( wnd, TB_SAVERESTOREW, FALSE, (LPARAM)&params );
2423     ok( res, "restoring failed\n" );
2424     ok_sequence(sequences, PARENT_SEQ_INDEX, restore_parent_seq, "restore", FALSE);
2425     count = SendMessageW( wnd, TB_BUTTONCOUNT, 0, 0 );
2426     ok( count == ARRAY_SIZE(expect_btns), "got %d\n", count );
2427 
2428     for (i = 0; i < count; i++)
2429     {
2430         res = SendMessageW( wnd, TB_GETBUTTON, i, (LPARAM)&tb );
2431         ok( res, "got %d\n", res );
2432 
2433         ok( tb.iBitmap == expect_btns[i].iBitmap, "%d: got %d\n", i, tb.iBitmap );
2434         ok( tb.idCommand == expect_btns[i].idCommand, "%d: got %d\n", i, tb.idCommand );
2435         ok( tb.fsState == expect_btns[i].fsState, "%d: got %02x\n", i, tb.fsState );
2436         ok( tb.fsStyle == expect_btns[i].fsStyle, "%d: got %02x\n", i, tb.fsStyle );
2437         ok( tb.dwData == expect_btns[i].dwData, "%d: got %lx\n", i, tb.dwData );
2438         if (IS_INTRESOURCE(expect_btns[i].iString))
2439             ok( tb.iString == expect_btns[i].iString, "%d: got %lx\n", i, tb.iString );
2440         else
2441             ok( !strcmp( (char *)tb.iString, (char *)expect_btns[i].iString ),
2442                 "%d: got %s\n", i, (char *)tb.iString );
2443 
2444         /* In fact the ptr value set in TBN_GETBUTTONINFOA is simply copied */
2445         if (tb.idCommand == 7)
2446             ok( tb.iString == (INT_PTR)alloced_str, "string not set\n");
2447     }
2448 
2449     DestroyWindow( wnd );
2450     RegOpenKeyW( HKEY_CURRENT_USER, subkey, &key );
2451     RegDeleteValueW( key, value );
2452     RegCloseKey( key );
2453 }
2454 
2455 static void test_drawtext_flags(void)
2456 {
2457     HWND hwnd = NULL;
2458     UINT flags;
2459 
2460     rebuild_toolbar(&hwnd);
2461 
2462     flags = SendMessageA(hwnd, TB_SETDRAWTEXTFLAGS, 0, 0);
2463 todo_wine
2464     ok(flags == 0, "Unexpected draw text flags %#x\n", flags);
2465 
2466     /* zero mask, flags are retained */
2467     flags = SendMessageA(hwnd, TB_SETDRAWTEXTFLAGS, 0, DT_BOTTOM);
2468 todo_wine
2469     ok(flags == 0, "Unexpected draw text flags %#x\n", flags);
2470     ok(!(flags & DT_BOTTOM), "Unexpected DT_BOTTOM style\n");
2471 
2472     flags = SendMessageA(hwnd, TB_SETDRAWTEXTFLAGS, 0, 0);
2473 todo_wine
2474     ok(flags == 0, "Unexpected draw text flags %#x\n", flags);
2475     ok(!(flags & DT_BOTTOM), "Unexpected DT_BOTTOM style\n");
2476 
2477     /* set/remove */
2478     flags = SendMessageA(hwnd, TB_SETDRAWTEXTFLAGS, DT_BOTTOM, DT_BOTTOM);
2479 todo_wine
2480     ok(flags == 0, "Unexpected draw text flags %#x\n", flags);
2481     ok(!(flags & DT_BOTTOM), "Unexpected DT_BOTTOM style\n");
2482 
2483     flags = SendMessageA(hwnd, TB_SETDRAWTEXTFLAGS, DT_BOTTOM, 0);
2484 todo_wine
2485     ok(flags == DT_BOTTOM, "Unexpected draw text flags %#x\n", flags);
2486     ok(flags & DT_BOTTOM, "Expected DT_BOTTOM style, %#x\n", flags);
2487 
2488     flags = SendMessageA(hwnd, TB_SETDRAWTEXTFLAGS, DT_BOTTOM, 0);
2489 todo_wine
2490     ok(flags == 0, "Unexpected draw text flags %#x\n", flags);
2491     ok(!(flags & DT_BOTTOM), "Unexpected DT_BOTTOM style\n");
2492 
2493     DestroyWindow(hwnd);
2494 }
2495 
2496 static void test_imagelist(void)
2497 {
2498     HIMAGELIST imagelist;
2499     HWND hwnd = NULL;
2500     int ret;
2501 
2502     rebuild_toolbar(&hwnd);
2503 
2504     imagelist = (HIMAGELIST)SendMessageA(hwnd, TB_GETIMAGELIST, 0, 0);
2505     ok(imagelist == NULL, "got %p\n", imagelist);
2506 
2507     ret = SendMessageA(hwnd, TB_SETBITMAPSIZE, 0, MAKELONG(16, 16));
2508     ok(ret, "got %d\n", ret);
2509 
2510     imagelist = (HIMAGELIST)SendMessageA(hwnd, TB_GETIMAGELIST, 0, 0);
2511     ok(imagelist == NULL, "got %p\n", imagelist);
2512 
2513     DestroyWindow(hwnd);
2514 }
2515 
2516 static void init_functions(void)
2517 {
2518     HMODULE hComCtl32 = LoadLibraryA("comctl32.dll");
2519 
2520 #define X(f) p##f = (void*)GetProcAddress(hComCtl32, #f);
2521     X(CreateToolbarEx);
2522     X(ImageList_GetIconSize);
2523     X(ImageList_GetImageCount);
2524     X(ImageList_LoadImageA);
2525     X(ImageList_Destroy);
2526 #undef X
2527 }
2528 
2529 START_TEST(toolbar)
2530 {
2531     WNDCLASSA wc;
2532     MSG msg;
2533     RECT rc;
2534 
2535     init_msg_sequences(sequences, NUM_MSG_SEQUENCES);
2536     init_functions();
2537 
2538     wc.style = CS_HREDRAW | CS_VREDRAW;
2539     wc.cbClsExtra = 0;
2540     wc.cbWndExtra = 0;
2541     wc.hInstance = GetModuleHandleA(NULL);
2542     wc.hIcon = NULL;
2543     wc.hCursor = LoadCursorA(NULL, (LPCSTR)IDC_IBEAM);
2544     wc.hbrBackground = GetSysColorBrush(COLOR_WINDOW);
2545     wc.lpszMenuName = NULL;
2546     wc.lpszClassName = "Toolbar test parent";
2547     wc.lpfnWndProc = parent_wnd_proc;
2548     RegisterClassA(&wc);
2549 
2550     hMainWnd = CreateWindowExA(0, "Toolbar test parent", "Blah", WS_OVERLAPPEDWINDOW,
2551       CW_USEDEFAULT, CW_USEDEFAULT, 680, 260, NULL, NULL, GetModuleHandleA(NULL), 0);
2552     GetClientRect(hMainWnd, &rc);
2553     ShowWindow(hMainWnd, SW_SHOW);
2554 
2555     basic_test();
2556     test_add_bitmap();
2557     test_add_string();
2558     test_hotitem();
2559     test_sizes();
2560     test_recalc();
2561     test_getbuttoninfo();
2562     test_createtoolbarex();
2563     test_dispinfo();
2564     test_setrows();
2565     test_getstring();
2566     test_tooltip();
2567     test_get_set_style();
2568     test_create();
2569     test_TB_GET_SET_EXTENDEDSTYLE();
2570     test_noresize();
2571     test_save();
2572     test_drawtext_flags();
2573     test_imagelist();
2574 
2575     PostQuitMessage(0);
2576     while(GetMessageA(&msg,0,0,0)) {
2577         TranslateMessage(&msg);
2578         DispatchMessageA(&msg);
2579     }
2580     DestroyWindow(hMainWnd);
2581 }
2582