comctl32: toolbar: Use correctly wParam in TB_ADDBITMAP.
[wine] / dlls / comctl32 / tests / toolbar.c
1 /* Unit tests for treeview.
2  *
3  * Copyright 2005 Krzysztof Foltman
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Lesser General Public
7  * License as published by the Free Software Foundation; either
8  * version 2.1 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Lesser General Public License for more details.
14  *
15  * You should have received a copy of the GNU Lesser General Public
16  * License along with this library; if not, write to the Free Software
17  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
18  */
19
20 #include <assert.h>
21 #include <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 HWND hMainWnd;
36  
37 static void MakeButton(TBBUTTON *p, int idCommand, int fsStyle, int nString) {
38   p->iBitmap = -2;
39   p->idCommand = idCommand;
40   p->fsState = TBSTATE_ENABLED;
41   p->fsStyle = fsStyle;
42   p->iString = nString;
43 }
44
45 static LRESULT CALLBACK MyWndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
46 {
47     return DefWindowProcA(hWnd, msg, wParam, lParam);
48 }
49
50 static void basic_test(void)
51 {
52     TBBUTTON buttons[9];
53     HWND hToolbar;
54     int i;
55     for (i=0; i<9; i++)
56         MakeButton(buttons+i, 1000+i, TBSTYLE_CHECKGROUP, 0);
57     MakeButton(buttons+3, 1003, TBSTYLE_SEP|TBSTYLE_GROUP, 0);
58     MakeButton(buttons+6, 1006, TBSTYLE_SEP, 0);
59
60     hToolbar = CreateToolbarEx(hMainWnd,
61         WS_VISIBLE | WS_CLIPCHILDREN | CCS_TOP |
62         WS_CHILD | TBSTYLE_LIST,
63         100,
64         0, NULL, (UINT)0,
65         buttons, sizeof(buttons)/sizeof(buttons[0]),
66         0, 0, 20, 16, sizeof(TBBUTTON));
67     ok(hToolbar != NULL, "Toolbar creation\n");
68     SendMessage(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)"test\000");
69
70     /* test for exclusion working inside a separator-separated :-) group */
71     SendMessage(hToolbar, TB_CHECKBUTTON, 1000, 1); /* press A1 */
72     ok(SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 pressed\n");
73     ok(!SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1001, 0), "A2 not pressed\n");
74
75     SendMessage(hToolbar, TB_CHECKBUTTON, 1004, 1); /* press A5, release A1 */
76     ok(SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1004, 0), "A5 pressed\n");
77     ok(!SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 not pressed anymore\n");
78
79     SendMessage(hToolbar, TB_CHECKBUTTON, 1005, 1); /* press A6, release A5 */
80     ok(SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1005, 0), "A6 pressed\n");
81     ok(!SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1004, 0), "A5 not pressed anymore\n");
82
83     /* test for inter-group crosstalk, ie. two radio groups interfering with each other */
84     SendMessage(hToolbar, TB_CHECKBUTTON, 1007, 1); /* press B2 */
85     ok(SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1005, 0), "A6 still pressed, no inter-group crosstalk\n");
86     ok(!SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 still not pressed\n");
87     ok(SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1007, 0), "B2 pressed\n");
88
89     SendMessage(hToolbar, TB_CHECKBUTTON, 1000, 1); /* press A1 and ensure B group didn't suffer */
90     ok(!SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1005, 0), "A6 not pressed anymore\n");
91     ok(SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 pressed\n");
92     ok(SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1007, 0), "B2 still pressed\n");
93
94     SendMessage(hToolbar, TB_CHECKBUTTON, 1008, 1); /* press B3, and ensure A group didn't suffer */
95     ok(!SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1005, 0), "A6 pressed\n");
96     ok(SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 pressed\n");
97     ok(!SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1007, 0), "B2 not pressed\n");
98     ok(SendMessage(hToolbar, TB_ISBUTTONCHECKED, 1008, 0), "B3 pressed\n");
99     DestroyWindow(hToolbar);
100 }
101
102 static void rebuild_toolbar(HWND *hToolbar)
103 {
104     if (*hToolbar != NULL)
105         DestroyWindow(*hToolbar);
106     *hToolbar = CreateWindowEx(0, TOOLBARCLASSNAME, NULL, WS_CHILD | WS_VISIBLE, 0, 0, 0, 0,
107         hMainWnd, (HMENU)5, GetModuleHandle(NULL), NULL);
108     ok(*hToolbar != NULL, "Toolbar creation problem\n");
109     ok(SendMessage(*hToolbar, TB_BUTTONSTRUCTSIZE, (WPARAM)sizeof(TBBUTTON), 0) == 0, "TB_BUTTONSTRUCTSIZE failed\n");
110     ok(SendMessage(*hToolbar, TB_AUTOSIZE, 0, 0) == 0, "TB_AUTOSIZE failed\n");
111 }
112
113 #define CHECK_IMAGELIST(count, dx, dy) { \
114     int cx, cy; \
115     HIMAGELIST himl = (HIMAGELIST)SendMessageA(hToolbar, TB_GETIMAGELIST, 0, 0); \
116     ok(himl != NULL, "No image list\n"); \
117     if (himl != NULL) {\
118         ok(ImageList_GetImageCount(himl) == count, "Images count mismatch - %d vs %d\n", count, ImageList_GetImageCount(himl)); \
119         ImageList_GetIconSize(himl, &cx, &cy); \
120         ok(cx == dx && cy == cy, "Icon size mismatch - %dx%d vs %dx%d\n", dx, dy, cx, cy); \
121     } \
122 }
123
124 #define CHECK_IMAGELIST_TODO_COUNT(count, dx, dy) { \
125     int cx, cy; \
126     HIMAGELIST himl = (HIMAGELIST)SendMessageA(hToolbar, TB_GETIMAGELIST, 0, 0); \
127     ok(himl != NULL, "No image list\n"); \
128     if (himl != NULL) {\
129         todo_wine ok(ImageList_GetImageCount(himl) == count, "Images count mismatch - %d vs %d\n", count, ImageList_GetImageCount(himl)); \
130         ImageList_GetIconSize(himl, &cx, &cy); \
131         ok(cx == dx && cy == cy, "Icon size mismatch - %dx%d vs %dx%d\n", dx, dy, cx, cy); \
132     } \
133 }
134
135 #define CHECK_IMAGELIST_TODO_COUNT_SIZE(count, dx, dy) { \
136     int cx, cy; \
137     HIMAGELIST himl = (HIMAGELIST)SendMessageA(hToolbar, TB_GETIMAGELIST, 0, 0); \
138     ok(himl != NULL, "No image list\n"); \
139     if (himl != NULL) {\
140         todo_wine ok(ImageList_GetImageCount(himl) == count, "Images count mismatch - %d vs %d\n", count, ImageList_GetImageCount(himl)); \
141         ImageList_GetIconSize(himl, &cx, &cy); \
142         todo_wine ok(cx == dx && cy == cy, "Icon size mismatch - %dx%d vs %dx%d\n", dx, dy, cx, cy); \
143     } \
144 }
145
146 static void test_add_bitmap(void)
147 {
148     HWND hToolbar = NULL;
149     TBADDBITMAP bmp128;
150     TBADDBITMAP bmp80;
151     TBADDBITMAP stdsmall;
152     INT ret;
153
154     /* empty 128x15 bitmap */
155     bmp128.hInst = GetModuleHandle(NULL);
156     bmp128.nID = IDB_BITMAP_128x15;
157
158     /* empty 80x15 bitmap */
159     bmp80.hInst = GetModuleHandle(NULL);
160     bmp80.nID = IDB_BITMAP_80x15;
161
162     /* standard bitmap - 240x15 pixels */
163     stdsmall.hInst = HINST_COMMCTRL;
164     stdsmall.nID = IDB_STD_SMALL_COLOR;
165
166     rebuild_toolbar(&hToolbar);
167     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 8, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n");
168     CHECK_IMAGELIST(8, 16, 15);
169     
170     /* adding more bitmaps */
171     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp80) == 8, "TB_ADDBITMAP - unexpected return\n");
172     CHECK_IMAGELIST(13, 16, 15);
173     /* adding the same bitmap will simply return the index of the already loaded block */
174     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 8, (LPARAM)&bmp128);
175     todo_wine ok(ret == 0, "TB_ADDBITMAP - unexpected return %d\n", ret);
176     CHECK_IMAGELIST_TODO_COUNT(13, 16, 15);
177     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp80);
178     todo_wine ok(ret == 8, "TB_ADDBITMAP - unexpected return %d\n", ret);
179     CHECK_IMAGELIST_TODO_COUNT(13, 16, 15);
180     /* even if we increase the wParam */
181     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 55, (LPARAM)&bmp80);
182     todo_wine ok(ret == 8, "TB_ADDBITMAP - unexpected return %d\n", ret);
183     CHECK_IMAGELIST_TODO_COUNT(13, 16, 15);
184
185     /* when the wParam is smaller than the bitmaps count but non-zero, all the bitmaps will be added*/
186     rebuild_toolbar(&hToolbar);
187     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 3, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n");
188     CHECK_IMAGELIST(8, 16, 15);
189     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp80);
190     todo_wine ok(ret == 3, "TB_ADDBITMAP - unexpected return %d\n", ret);
191     /* the returned value is misleading - id 8 is the id of the first icon from bmp80 */
192     CHECK_IMAGELIST(13, 16, 15);
193
194     /* the same for negative wParam */
195     rebuild_toolbar(&hToolbar);
196     ret = SendMessageA(hToolbar, TB_ADDBITMAP, -143, (LPARAM)&bmp128);
197     ok(ret == 0, "TB_ADDBITMAP - unexpected return %d\n", ret);
198     CHECK_IMAGELIST(8, 16, 15);
199     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&bmp80);
200     todo_wine ok(ret == -143, "TB_ADDBITMAP - unexpected return %d\n", ret);
201     CHECK_IMAGELIST(13, 16, 15);
202
203     /* for zero only one bitmap will be added */
204     rebuild_toolbar(&hToolbar);
205     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 0, (LPARAM)&bmp80);
206     ok(ret == 0, "TB_ADDBITMAP - unexpected return %d\n", ret);
207     CHECK_IMAGELIST(1, 16, 15);
208
209     /* if wParam is larger than the amount of icons, the list is grown */
210     rebuild_toolbar(&hToolbar);
211     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 100, (LPARAM)&bmp80) == 0, "TB_ADDBITMAP - unexpected return\n");
212     CHECK_IMAGELIST(100, 16, 15);
213     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 100, (LPARAM)&bmp128);
214     ok(ret == 100, "TB_ADDBITMAP - unexpected return %d\n", ret);
215     CHECK_IMAGELIST(200, 16, 15);
216
217     /* adding built-in items - the wParam is ignored */
218     rebuild_toolbar(&hToolbar);
219     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp80) == 0, "TB_ADDBITMAP - unexpected return\n");
220     CHECK_IMAGELIST(5, 16, 15);
221     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 0, (LPARAM)&stdsmall) == 5, "TB_ADDBITMAP - unexpected return\n");
222     CHECK_IMAGELIST(20, 16, 15);
223     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp128) == 20, "TB_ADDBITMAP - unexpected return\n");
224     CHECK_IMAGELIST(28, 16, 15);
225
226     /* when we increase the bitmap size, less icons will be created */
227     rebuild_toolbar(&hToolbar);
228     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(20, 20)) == TRUE, "TB_SETBITMAPSIZE failed\n");
229     ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n");
230     CHECK_IMAGELIST(6, 20, 20);
231     ret = SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&bmp80);
232     todo_wine ok(ret == 1, "TB_ADDBITMAP - unexpected return %d\n", ret);
233     CHECK_IMAGELIST(10, 20, 20);
234     /* the icons can be resized - an UpdateWindow is needed as this probably happens during WM_PAINT */
235     ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(8, 8)) == TRUE, "TB_SETBITMAPSIZE failed\n");
236     UpdateWindow(hToolbar);
237     CHECK_IMAGELIST_TODO_COUNT_SIZE(26, 8, 8);
238     /* loading a standard bitmaps automatically resizes the icons */
239     todo_wine ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&stdsmall) == 2, "TB_ADDBITMAP - unexpected return\n");
240     UpdateWindow(hToolbar);
241     CHECK_IMAGELIST_TODO_COUNT_SIZE(28, 16, 15);
242
243     DestroyWindow(hToolbar);
244 }
245
246 START_TEST(toolbar)
247 {
248     WNDCLASSA wc;
249     MSG msg;
250     RECT rc;
251   
252     InitCommonControls();
253   
254     wc.style = CS_HREDRAW | CS_VREDRAW;
255     wc.cbClsExtra = 0;
256     wc.cbWndExtra = 0;
257     wc.hInstance = GetModuleHandleA(NULL);
258     wc.hIcon = NULL;
259     wc.hCursor = LoadCursorA(NULL, MAKEINTRESOURCEA(IDC_IBEAM));
260     wc.hbrBackground = GetSysColorBrush(COLOR_WINDOW);
261     wc.lpszMenuName = NULL;
262     wc.lpszClassName = "MyTestWnd";
263     wc.lpfnWndProc = MyWndProc;
264     RegisterClassA(&wc);
265     
266     hMainWnd = CreateWindowExA(0, "MyTestWnd", "Blah", WS_OVERLAPPEDWINDOW, 
267       CW_USEDEFAULT, CW_USEDEFAULT, 680, 260, NULL, NULL, GetModuleHandleA(NULL), 0);
268     GetClientRect(hMainWnd, &rc);
269     ShowWindow(hMainWnd, SW_SHOW);
270
271     basic_test();
272     test_add_bitmap();
273
274     PostQuitMessage(0);
275     while(GetMessageA(&msg,0,0,0)) {
276         TranslateMessage(&msg);
277         DispatchMessageA(&msg);
278     }
279     DestroyWindow(hMainWnd);
280 }