1 /* Unit test suite for status control.
2 *
3 * Copyright 2007 Google (Lei Zhang)
4 * Copyright 2007 Alex Arazi
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 <windows.h>
22 #include <commctrl.h>
23
24 #include "wine/test.h"
25
26 #define SUBCLASS_NAME "MyStatusBar"
27
28 #define expect(expected,got) ok (expected == got,"Expected %d, got %d\n",expected,got)
29 #define expect_rect(_left,_top,_right,_bottom,got) do { \
30 RECT exp = {abs(got.left - _left), abs(got.top - _top), \
31 abs(got.right - _right), abs(got.bottom - _bottom)}; \
32 ok(exp.left <= 2 && exp.top <= 2 && exp.right <= 2 && exp.bottom <= 2, \
33 "Expected rect (%d,%d)-(%d,%d), got %s\n", _left, _top, _right, _bottom, \
34 wine_dbgstr_rect(&(got))); } while (0)
35
36 static HINSTANCE hinst;
37 static WNDPROC g_status_wndproc;
38 static RECT g_rcCreated;
39 static HWND g_hMainWnd;
40 static int g_wmsize_count = 0;
41 static INT g_ysize;
42 static INT g_dpisize;
43 static int g_wmdrawitm_ctr;
44 static WNDPROC g_wndproc_saved;
45
46 static BOOL (WINAPI *pInitCommonControlsEx)(const INITCOMMONCONTROLSEX*);
47
create_status_control(DWORD style,DWORD exstyle)48 static HWND create_status_control(DWORD style, DWORD exstyle)
49 {
50 HWND hWndStatus;
51
52 /* make the control */
53 hWndStatus = CreateWindowExA(exstyle, STATUSCLASSNAMEA, NULL, style,
54 /* placement */
55 0, 0, 300, 20,
56 /* parent, etc */
57 NULL, NULL, hinst, NULL);
58 ok(hWndStatus != NULL, "failed to create status wnd\n");
59 return hWndStatus;
60 }
61
create_test_wndproc(HWND hwnd,UINT msg,WPARAM wParam,LPARAM lParam)62 static LRESULT WINAPI create_test_wndproc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
63 {
64 LRESULT ret;
65
66 if (msg == WM_CREATE)
67 {
68 CREATESTRUCTA *cs = (CREATESTRUCTA *)lParam;
69 ret = CallWindowProcA(g_status_wndproc, hwnd, msg, wParam, lParam);
70 GetWindowRect(hwnd, &g_rcCreated);
71 MapWindowPoints(HWND_DESKTOP, g_hMainWnd, (LPPOINT)&g_rcCreated, 2);
72 ok(cs->x == g_rcCreated.left, "CREATESTRUCT.x modified\n");
73 ok(cs->y == g_rcCreated.top, "CREATESTRUCT.y modified\n");
74 } else if (msg == WM_SIZE)
75 {
76 g_wmsize_count++;
77 ret = CallWindowProcA(g_status_wndproc, hwnd, msg, wParam, lParam);
78 }
79 else
80 ret = CallWindowProcA(g_status_wndproc, hwnd, msg, wParam, lParam);
81
82 return ret;
83 }
84
register_subclass(void)85 static void register_subclass(void)
86 {
87 WNDCLASSEXA cls;
88
89 cls.cbSize = sizeof(WNDCLASSEXA);
90 GetClassInfoExA(NULL, STATUSCLASSNAMEA, &cls);
91 g_status_wndproc = cls.lpfnWndProc;
92 cls.lpfnWndProc = create_test_wndproc;
93 cls.lpszClassName = SUBCLASS_NAME;
94 cls.hInstance = NULL;
95 ok(RegisterClassExA(&cls), "RegisterClassEx failed\n");
96 }
97
test_create(void)98 static void test_create(void)
99 {
100 RECT rc;
101 HWND hwnd;
102
103 ok((hwnd = CreateWindowA(SUBCLASS_NAME, "", WS_CHILD|WS_VISIBLE|SBARS_SIZEGRIP, 0, 0, 100, 100,
104 g_hMainWnd, NULL, NULL, 0)) != NULL, "CreateWindowA failed\n");
105 MapWindowPoints(HWND_DESKTOP, g_hMainWnd, (LPPOINT)&rc, 2);
106 GetWindowRect(hwnd, &rc);
107 MapWindowPoints(HWND_DESKTOP, g_hMainWnd, (LPPOINT)&rc, 2);
108 expect_rect(0, 0, 100, 100, g_rcCreated);
109 expect(0, rc.left);
110 expect(672, rc.right);
111 expect(226, rc.bottom);
112 /* we don't check rc.top as this may depend on user font settings */
113 DestroyWindow(hwnd);
114 }
115
check_height_font_enumproc(ENUMLOGFONTEXA * enumlf,NEWTEXTMETRICEXA * ntm,DWORD type,LPARAM lParam)116 static int CALLBACK check_height_font_enumproc(ENUMLOGFONTEXA *enumlf, NEWTEXTMETRICEXA *ntm, DWORD type, LPARAM lParam)
117 {
118 HWND hwndStatus = (HWND)lParam;
119 HDC hdc = GetDC(NULL);
120 static const int sizes[] = { 6, 7, 8, 9, 10, 11, 12, 13, 15, 16,
121 20, 22, 28, 36, 48, 72};
122 DWORD i;
123 INT y;
124 LPSTR facename = (CHAR *)enumlf->elfFullName;
125
126 /* on win9x, enumlf->elfFullName is only valid for truetype fonts */
127 if (type != TRUETYPE_FONTTYPE)
128 facename = enumlf->elfLogFont.lfFaceName;
129
130 for (i = 0; i < ARRAY_SIZE(sizes); i++)
131 {
132 HFONT hFont;
133 TEXTMETRICA tm;
134 HFONT hCtrlFont;
135 HFONT hOldFont;
136 RECT rcCtrl;
137
138 enumlf->elfLogFont.lfHeight = sizes[i];
139 hFont = CreateFontIndirectA(&enumlf->elfLogFont);
140 hCtrlFont = (HFONT)SendMessageA(hwndStatus, WM_SETFONT, (WPARAM)hFont, TRUE);
141 hOldFont = SelectObject(hdc, hFont);
142
143 GetClientRect(hwndStatus, &rcCtrl);
144 GetTextMetricsA(hdc, &tm);
145 y = tm.tmHeight + (tm.tmInternalLeading ? tm.tmInternalLeading : 2) + 4;
146
147 ok( (rcCtrl.bottom == max(y, g_ysize)) || (rcCtrl.bottom == max(y, g_dpisize)),
148 "got %d (expected %d or %d) for %s #%d\n",
149 rcCtrl.bottom, max(y, g_ysize), max(y, g_dpisize), facename, sizes[i]);
150
151 SelectObject(hdc, hOldFont);
152 SendMessageA(hwndStatus, WM_SETFONT, (WPARAM)hCtrlFont, TRUE);
153 DeleteObject(hFont);
154 }
155 ReleaseDC(NULL, hdc);
156 return 1;
157 }
158
check_height_family_enumproc(ENUMLOGFONTEXA * enumlf,NEWTEXTMETRICEXA * ntm,DWORD type,LPARAM lParam)159 static int CALLBACK check_height_family_enumproc(ENUMLOGFONTEXA *enumlf, NEWTEXTMETRICEXA *ntm, DWORD type, LPARAM lParam)
160 {
161 HDC hdc = GetDC(NULL);
162 enumlf->elfLogFont.lfHeight = 0;
163 EnumFontFamiliesExA(hdc, &enumlf->elfLogFont, (FONTENUMPROCA)check_height_font_enumproc, lParam, 0);
164 ReleaseDC(NULL, hdc);
165 return 1;
166 }
167
test_height(void)168 static void test_height(void)
169 {
170 LOGFONTA lf;
171 HFONT hFont, hFontSm;
172 RECT rc1, rc2;
173 HWND hwndStatus = CreateWindowA(SUBCLASS_NAME, NULL, WS_CHILD|WS_VISIBLE,
174 0, 0, 300, 20, g_hMainWnd, NULL, NULL, NULL);
175 HDC hdc;
176
177 GetClientRect(hwndStatus, &rc1);
178 hFont = CreateFontA(32, 0, 0, 0, FW_DONTCARE, FALSE, FALSE, FALSE, ANSI_CHARSET,
179 OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, FF_DONTCARE, "Tahoma");
180
181 g_wmsize_count = 0;
182 SendMessageA(hwndStatus, WM_SETFONT, (WPARAM)hFont, TRUE);
183 if (!g_wmsize_count)
184 {
185 skip("Status control not resized in win95, skipping broken tests.\n");
186 return;
187 }
188 ok(g_wmsize_count > 0, "WM_SETFONT should issue WM_SIZE\n");
189
190 GetClientRect(hwndStatus, &rc2);
191 expect_rect(0, 0, 672, 42, rc2); /* GetTextMetrics returns invalid tmInternalLeading for this font */
192
193 g_wmsize_count = 0;
194 SendMessageA(hwndStatus, WM_SETFONT, (WPARAM)hFont, TRUE);
195 ok(g_wmsize_count > 0, "WM_SETFONT should issue WM_SIZE\n");
196
197 GetClientRect(hwndStatus, &rc2);
198 expect_rect(0, 0, 672, 42, rc2);
199
200 /* minheight < fontsize - no effects*/
201 SendMessageA(hwndStatus, SB_SETMINHEIGHT, 12, 0);
202 SendMessageA(hwndStatus, WM_SIZE, 0, 0);
203 GetClientRect(hwndStatus, &rc2);
204 expect_rect(0, 0, 672, 42, rc2);
205
206 /* minheight > fontsize - has an effect after WM_SIZE */
207 SendMessageA(hwndStatus, SB_SETMINHEIGHT, 60, 0);
208 GetClientRect(hwndStatus, &rc2);
209 expect_rect(0, 0, 672, 42, rc2);
210 SendMessageA(hwndStatus, WM_SIZE, 0, 0);
211 GetClientRect(hwndStatus, &rc2);
212 expect_rect(0, 0, 672, 62, rc2);
213
214 /* font changed to smaller than minheight - has an effect */
215 SendMessageA(hwndStatus, SB_SETMINHEIGHT, 30, 0);
216 expect_rect(0, 0, 672, 62, rc2);
217 SendMessageA(hwndStatus, WM_SIZE, 0, 0);
218 GetClientRect(hwndStatus, &rc2);
219 expect_rect(0, 0, 672, 42, rc2);
220 hFontSm = CreateFontA(9, 0, 0, 0, FW_DONTCARE, FALSE, FALSE, FALSE, ANSI_CHARSET,
221 OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, FF_DONTCARE, "Tahoma");
222 SendMessageA(hwndStatus, WM_SETFONT, (WPARAM)hFontSm, TRUE);
223 GetClientRect(hwndStatus, &rc2);
224 expect_rect(0, 0, 672, 32, rc2);
225
226 /* test the height formula */
227 ZeroMemory(&lf, sizeof(lf));
228 SendMessageA(hwndStatus, SB_SETMINHEIGHT, 0, 0);
229 hdc = GetDC(NULL);
230
231 /* used only for some fonts (tahoma as example) */
232 g_ysize = GetSystemMetrics(SM_CYSIZE) + 2;
233 if (g_ysize & 1) g_ysize--; /* The min height is always even */
234
235 g_dpisize = MulDiv(18, GetDeviceCaps(hdc, LOGPIXELSY), 96) + 2;
236 if (g_dpisize & 1) g_dpisize--; /* The min height is always even */
237
238
239 trace("dpi=%d (min height: %d or %d) SM_CYSIZE: %d\n",
240 GetDeviceCaps(hdc, LOGPIXELSY), g_ysize, g_dpisize,
241 GetSystemMetrics(SM_CYSIZE));
242
243 EnumFontFamiliesExA(hdc, &lf, (FONTENUMPROCA)check_height_family_enumproc, (LPARAM)hwndStatus, 0);
244 ReleaseDC(NULL, hdc);
245
246 DestroyWindow(hwndStatus);
247 DeleteObject(hFont);
248 DeleteObject(hFontSm);
249 }
250
test_status_control(void)251 static void test_status_control(void)
252 {
253 HWND hWndStatus;
254 int r;
255 int nParts[] = {50, 150, -1};
256 int checkParts[] = {0, 0, 0};
257 int borders[] = {0, 0, 0};
258 RECT rc;
259 CHAR charArray[20];
260 HICON hIcon;
261 char ch;
262 char chstr[10] = "Inval id";
263 COLORREF crColor = RGB(0,0,0);
264
265 hWndStatus = create_status_control(WS_VISIBLE | SBT_TOOLTIPS, 0);
266
267 /* Divide into parts and set text */
268 r = SendMessageA(hWndStatus, SB_SETPARTS, 3, (LPARAM)nParts);
269 expect(TRUE,r);
270 r = SendMessageA(hWndStatus, SB_SETTEXTA, SBT_POPOUT|0, (LPARAM)"First");
271 expect(TRUE,r);
272 r = SendMessageA(hWndStatus, SB_SETTEXTA, SBT_OWNERDRAW|1, (LPARAM)"Second");
273 expect(TRUE,r);
274 r = SendMessageA(hWndStatus, SB_SETTEXTA, SBT_NOBORDERS|2, (LPARAM)"Third");
275 expect(TRUE,r);
276
277 /* Get RECT Information */
278 r = SendMessageA(hWndStatus, SB_GETRECT, 0, (LPARAM)&rc);
279 expect(TRUE,r);
280 expect(2,rc.top);
281 /* The rc.bottom test is system dependent
282 expect(22,rc.bottom); */
283 expect(0,rc.left);
284 expect(50,rc.right);
285 r = SendMessageA(hWndStatus, SB_GETRECT, -1, (LPARAM)&rc);
286 expect(FALSE,r);
287 r = SendMessageA(hWndStatus, SB_GETRECT, 3, (LPARAM)&rc);
288 expect(FALSE,r);
289 /* Get text length and text */
290 r = SendMessageA(hWndStatus, SB_GETTEXTLENGTHA, 0, 0);
291 expect(5,LOWORD(r));
292 expect(SBT_POPOUT,HIWORD(r));
293 r = SendMessageW(hWndStatus, WM_GETTEXTLENGTH, 0, 0);
294 ok(r == 5, "Expected 5, got %d\n", r);
295 r = SendMessageA(hWndStatus, SB_GETTEXTLENGTHA, 1, 0);
296 expect(0,LOWORD(r));
297 expect(SBT_OWNERDRAW,HIWORD(r));
298 r = SendMessageA(hWndStatus, SB_GETTEXTLENGTHA, 2, 0);
299 expect(5,LOWORD(r));
300 expect(SBT_NOBORDERS,HIWORD(r));
301 r = SendMessageA(hWndStatus, SB_GETTEXTA, 2, (LPARAM) charArray);
302 ok(strcmp(charArray,"Third") == 0, "Expected Third, got %s\n", charArray);
303 expect(5,LOWORD(r));
304 expect(SBT_NOBORDERS,HIWORD(r));
305
306 /* Get parts and borders */
307 r = SendMessageA(hWndStatus, SB_GETPARTS, 3, (LPARAM)checkParts);
308 ok(r == 3, "Expected 3, got %d\n", r);
309 expect(50,checkParts[0]);
310 expect(150,checkParts[1]);
311 expect(-1,checkParts[2]);
312 r = SendMessageA(hWndStatus, SB_GETBORDERS, 0, (LPARAM)borders);
313 ok(r == TRUE, "Expected TRUE, got %d\n", r);
314 expect(0,borders[0]);
315 expect(2,borders[1]);
316 expect(2,borders[2]);
317
318 /* Test resetting text with different characters */
319 r = SendMessageA(hWndStatus, SB_SETTEXTA, 0, (LPARAM)"First@Again");
320 expect(TRUE,r);
321 r = SendMessageA(hWndStatus, SB_SETTEXTA, 1, (LPARAM)"Invalid\tChars\\7\7");
322 expect(TRUE,r);
323 r = SendMessageA(hWndStatus, SB_SETTEXTA, 2, (LPARAM)"InvalidChars\\n\n");
324 expect(TRUE,r);
325
326 /* Get text again */
327 r = SendMessageA(hWndStatus, SB_GETTEXTA, 0, (LPARAM) charArray);
328 ok(strcmp(charArray,"First@Again") == 0, "Expected First@Again, got %s\n", charArray);
329 expect(11,LOWORD(r));
330 expect(0,HIWORD(r));
331 r = SendMessageA(hWndStatus, SB_GETTEXTA, 1, (LPARAM) charArray);
332 ok(strcmp(charArray,"Invalid\tChars\\7 ") == 0, "Expected Invalid\tChars\\7 , got %s\n", charArray);
333
334 expect(16,LOWORD(r));
335 expect(0,HIWORD(r));
336 r = SendMessageA(hWndStatus, SB_GETTEXTA, 2, (LPARAM) charArray);
337 ok(strcmp(charArray,"InvalidChars\\n ") == 0, "Expected InvalidChars\\n , got %s\n", charArray);
338
339 expect(15,LOWORD(r));
340 expect(0,HIWORD(r));
341
342 /* test more nonprintable chars */
343 for(ch = 0x00; ch < 0x7F; ch++) {
344 chstr[5] = ch;
345 r = SendMessageA(hWndStatus, SB_SETTEXTA, 0, (LPARAM)chstr);
346 expect(TRUE,r);
347 r = SendMessageA(hWndStatus, SB_GETTEXTA, 0, (LPARAM)charArray);
348 ok(r == strlen(charArray), "got %d\n", r);
349 /* substitution with single space */
350 if (ch > 0x00 && ch < 0x20 && ch != '\t')
351 chstr[5] = ' ';
352 ok(strcmp(charArray, chstr) == 0, "Expected %s, got %s\n", chstr, charArray);
353 }
354
355 /* Set background color */
356 crColor = SendMessageA(hWndStatus, SB_SETBKCOLOR , 0, RGB(255,0,0));
357 ok(crColor == CLR_DEFAULT ||
358 broken(crColor == RGB(0,0,0)), /* win95 */
359 "Expected 0x%.8x, got 0x%.8x\n", CLR_DEFAULT, crColor);
360 crColor = SendMessageA(hWndStatus, SB_SETBKCOLOR , 0, CLR_DEFAULT);
361 ok(crColor == RGB(255,0,0) ||
362 broken(crColor == RGB(0,0,0)), /* win95 */
363 "Expected 0x%.8x, got 0x%.8x\n", RGB(255,0,0), crColor);
364
365 /* Add an icon to the status bar */
366 hIcon = LoadIconA(NULL, (LPCSTR)IDI_QUESTION);
367 r = SendMessageA(hWndStatus, SB_SETICON, 1, 0);
368 ok(r != 0 ||
369 broken(r == 0), /* win95 */
370 "Expected non-zero, got %d\n", r);
371 r = SendMessageA(hWndStatus, SB_SETICON, 1, (LPARAM) hIcon);
372 ok(r != 0 ||
373 broken(r == 0), /* win95 */
374 "Expected non-zero, got %d\n", r);
375 r = SendMessageA(hWndStatus, SB_SETICON, 1, 0);
376 ok(r != 0 ||
377 broken(r == 0), /* win95 */
378 "Expected non-zero, got %d\n", r);
379
380 /* Set the Unicode format */
381 r = SendMessageA(hWndStatus, SB_SETUNICODEFORMAT, FALSE, 0);
382 expect(FALSE,r);
383 r = SendMessageA(hWndStatus, SB_GETUNICODEFORMAT, 0, 0);
384 expect(FALSE,r);
385 r = SendMessageA(hWndStatus, SB_SETUNICODEFORMAT, TRUE, 0);
386 expect(FALSE,r);
387 r = SendMessageA(hWndStatus, SB_GETUNICODEFORMAT, 0, 0);
388 ok(r == TRUE ||
389 broken(r == FALSE), /* win95 */
390 "Expected TRUE, got %d\n", r);
391
392 /* Reset number of parts */
393 r = SendMessageA(hWndStatus, SB_SETPARTS, 2, (LPARAM)nParts);
394 expect(TRUE,r);
395 r = SendMessageA(hWndStatus, SB_GETPARTS, 0, 0);
396 ok(r == 2, "Expected 2, got %d\n", r);
397 r = SendMessageA(hWndStatus, SB_SETPARTS, 0, 0);
398 expect(FALSE,r);
399 r = SendMessageA(hWndStatus, SB_GETPARTS, 0, 0);
400 ok(r == 2, "Expected 2, got %d\n", r);
401
402 /* Set the minimum height and get rectangle information again */
403 SendMessageA(hWndStatus, SB_SETMINHEIGHT, 50, 0);
404 r = SendMessageA(hWndStatus, WM_SIZE, 0, 0);
405 expect(0,r);
406 r = SendMessageA(hWndStatus, SB_GETRECT, 0, (LPARAM)&rc);
407 expect(TRUE,r);
408 expect(2,rc.top);
409 /* The rc.bottom test is system dependent
410 expect(22,rc.bottom); */
411 expect(0,rc.left);
412 expect(50,rc.right);
413 r = SendMessageA(hWndStatus, SB_GETRECT, -1, (LPARAM)&rc);
414 expect(FALSE,r);
415 r = SendMessageA(hWndStatus, SB_GETRECT, 3, (LPARAM)&rc);
416 expect(FALSE,r);
417
418 /* Set the ToolTip text */
419 SendMessageA(hWndStatus, SB_SETTIPTEXTA, 0,(LPARAM) "Tooltip Text");
420 lstrcpyA(charArray, "apple");
421 SendMessageA(hWndStatus, SB_GETTIPTEXTA, MAKEWPARAM (0, 20),(LPARAM) charArray);
422 ok(strcmp(charArray,"Tooltip Text") == 0 ||
423 broken(!strcmp(charArray, "apple")), /* win95 */
424 "Expected Tooltip Text, got %s\n", charArray);
425
426 /* Make simple */
427 SendMessageA(hWndStatus, SB_SIMPLE, TRUE, 0);
428 r = SendMessageA(hWndStatus, SB_ISSIMPLE, 0, 0);
429 ok(r == TRUE ||
430 broken(r == FALSE), /* win95 */
431 "Expected TRUE, got %d\n", r);
432
433 DestroyWindow(hWndStatus);
434 }
435
ownerdraw_test_wndproc(HWND hwnd,UINT msg,WPARAM wParam,LPARAM lParam)436 static LRESULT WINAPI ownerdraw_test_wndproc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
437 {
438 LRESULT ret;
439 if (msg == WM_DRAWITEM)
440 g_wmdrawitm_ctr++;
441 ret = CallWindowProcA(g_wndproc_saved, hwnd, msg, wParam, lParam);
442 return ret;
443 }
444
test_status_ownerdraw(void)445 static void test_status_ownerdraw(void)
446 {
447 HWND hWndStatus;
448 int r;
449 const char* statustext = "STATUS TEXT";
450 LONG oldstyle;
451
452 /* subclass the main window and make sure it is visible */
453 g_wndproc_saved = (WNDPROC) SetWindowLongPtrA( g_hMainWnd, GWLP_WNDPROC,
454 (LONG_PTR)ownerdraw_test_wndproc );
455 ok( g_wndproc_saved != 0, "failed to set the WndProc\n");
456 SetWindowPos( g_hMainWnd, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOSIZE|SWP_NOMOVE);
457 oldstyle = GetWindowLongA( g_hMainWnd, GWL_STYLE);
458 SetWindowLongA( g_hMainWnd, GWL_STYLE, oldstyle | WS_VISIBLE);
459 /* create a status child window */
460 ok((hWndStatus = CreateWindowA(SUBCLASS_NAME, "", WS_CHILD|WS_VISIBLE, 0, 0, 100, 100,
461 g_hMainWnd, NULL, NULL, 0)) != NULL, "CreateWindowA failed\n");
462 /* set text */
463 g_wmdrawitm_ctr = 0;
464 r = SendMessageA(hWndStatus, SB_SETTEXTA, 0, (LPARAM)statustext);
465 ok( r == TRUE, "Sendmessage returned %d, expected 1\n", r);
466 ok( 0 == g_wmdrawitm_ctr, "got %d drawitem messages expected none\n", g_wmdrawitm_ctr);
467 /* set same text, with ownerdraw flag */
468 g_wmdrawitm_ctr = 0;
469 r = SendMessageA(hWndStatus, SB_SETTEXTA, SBT_OWNERDRAW, (LPARAM)statustext);
470 ok( r == TRUE, "Sendmessage returned %d, expected 1\n", r);
471 ok( 1 == g_wmdrawitm_ctr, "got %d drawitem messages expected 1\n", g_wmdrawitm_ctr);
472 /* and again */
473 g_wmdrawitm_ctr = 0;
474 r = SendMessageA(hWndStatus, SB_SETTEXTA, SBT_OWNERDRAW, (LPARAM)statustext);
475 ok( r == TRUE, "Sendmessage returned %d, expected 1\n", r);
476 ok( 1 == g_wmdrawitm_ctr, "got %d drawitem messages expected 1\n", g_wmdrawitm_ctr);
477 /* clean up */
478 DestroyWindow(hWndStatus);
479 SetWindowLongA( g_hMainWnd, GWL_STYLE, oldstyle);
480 SetWindowLongPtrA( g_hMainWnd, GWLP_WNDPROC, (LONG_PTR)g_wndproc_saved );
481 }
482
test_gettext(void)483 static void test_gettext(void)
484 {
485 HWND hwndStatus = CreateWindowA(SUBCLASS_NAME, NULL, WS_CHILD|WS_VISIBLE,
486 0, 0, 300, 20, g_hMainWnd, NULL, NULL, NULL);
487 char buf[5];
488 int r;
489
490 r = SendMessageA(hwndStatus, SB_SETTEXTA, 0, (LPARAM)"Text");
491 expect(TRUE, r);
492 r = SendMessageA(hwndStatus, WM_GETTEXTLENGTH, 0, 0);
493 expect(4, r);
494 /* A size of 0 returns the length of the text */
495 r = SendMessageA(hwndStatus, WM_GETTEXT, 0, 0);
496 ok( r == 4 || broken(r == 2) /* win8 */, "Expected 4 got %d\n", r );
497 /* A size of 1 only stores the NULL terminator */
498 buf[0] = 0xa;
499 r = SendMessageA(hwndStatus, WM_GETTEXT, 1, (LPARAM)buf);
500 ok( r == 0 || broken(r == 4), "Expected 0 got %d\n", r );
501 if (!r) ok(!buf[0], "expected empty buffer\n");
502 /* A size of 2 returns a length 1 */
503 r = SendMessageA(hwndStatus, WM_GETTEXT, 2, (LPARAM)buf);
504 ok( r == 1 || broken(r == 4), "Expected 1 got %d\n", r );
505 r = SendMessageA(hwndStatus, WM_GETTEXT, sizeof(buf), (LPARAM)buf);
506 expect(4, r);
507 ok(!strcmp(buf, "Text"), "expected Text, got %s\n", buf);
508 DestroyWindow(hwndStatus);
509 }
510
511 /* Notify events to parent */
512 static BOOL g_got_dblclk;
513 static BOOL g_got_click;
514 static BOOL g_got_rdblclk;
515 static BOOL g_got_rclick;
516
517 /* Messages to parent */
518 static BOOL g_got_contextmenu;
519
test_notify_parent_proc(HWND hwnd,UINT msg,WPARAM wParam,LPARAM lParam)520 static LRESULT WINAPI test_notify_parent_proc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
521 {
522 switch(msg)
523 {
524 case WM_NOTIFY:
525 {
526 NMHDR *hdr = ((LPNMHDR)lParam);
527 switch(hdr->code)
528 {
529 case NM_DBLCLK: g_got_dblclk = TRUE; break;
530 case NM_CLICK: g_got_click = TRUE; break;
531 case NM_RDBLCLK: g_got_rdblclk = TRUE; break;
532 case NM_RCLICK: g_got_rclick = TRUE; break;
533 }
534
535 /* Return zero to indicate default processing */
536 return 0;
537 }
538
539 case WM_CONTEXTMENU: g_got_contextmenu = TRUE; return 0;
540
541 default:
542 return( DefWindowProcA(hwnd, msg, wParam, lParam));
543 }
544
545 return 0;
546 }
547
548 /* Test that WM_NOTIFY messages from the status control works correctly */
test_notify(void)549 static void test_notify(void)
550 {
551 HWND hwndParent;
552 HWND hwndStatus;
553 ATOM atom;
554 WNDCLASSA wclass = {0};
555 wclass.lpszClassName = "TestNotifyParentClass";
556 wclass.lpfnWndProc = test_notify_parent_proc;
557 atom = RegisterClassA(&wclass);
558 ok(atom, "RegisterClass failed\n");
559
560 /* create parent */
561 hwndParent = CreateWindowA(wclass.lpszClassName, "parent", WS_OVERLAPPEDWINDOW,
562 CW_USEDEFAULT, 0, 300, 20, NULL, NULL, NULL, NULL);
563 ok(hwndParent != NULL, "Parent creation failed!\n");
564
565 /* create status bar */
566 hwndStatus = CreateWindowA(STATUSCLASSNAMEA, NULL, WS_VISIBLE | WS_CHILD,
567 0, 0, 300, 20, hwndParent, NULL, NULL, NULL);
568 ok(hwndStatus != NULL, "Status creation failed!\n");
569
570 /* Send various mouse event, and check that we get them */
571 g_got_dblclk = FALSE;
572 SendMessageA(hwndStatus, WM_LBUTTONDBLCLK, 0, 0);
573 ok(g_got_dblclk, "WM_LBUTTONDBLCLK was not processed correctly!\n");
574 g_got_rdblclk = FALSE;
575 SendMessageA(hwndStatus, WM_RBUTTONDBLCLK, 0, 0);
576 ok(g_got_rdblclk, "WM_RBUTTONDBLCLK was not processed correctly!\n");
577 g_got_click = FALSE;
578 SendMessageA(hwndStatus, WM_LBUTTONUP, 0, 0);
579 ok(g_got_click, "WM_LBUTTONUP was not processed correctly!\n");
580
581 /* For R-UP, check that we also get the context menu from the default processing */
582 g_got_contextmenu = FALSE;
583 g_got_rclick = FALSE;
584 SendMessageA(hwndStatus, WM_RBUTTONUP, 0, 0);
585 ok(g_got_rclick, "WM_RBUTTONUP was not processed correctly!\n");
586 ok(g_got_contextmenu, "WM_RBUTTONUP did not activate the context menu!\n");
587 }
588
test_sizegrip(void)589 static void test_sizegrip(void)
590 {
591 HWND hwndStatus;
592 LONG style;
593 RECT rc, rcClient;
594 POINT pt;
595 int width, r;
596
597 hwndStatus = CreateWindowA(SUBCLASS_NAME, "", WS_CHILD|WS_VISIBLE|SBARS_SIZEGRIP,
598 0, 0, 100, 100, g_hMainWnd, NULL, NULL, NULL);
599
600 style = GetWindowLongPtrA(g_hMainWnd, GWL_STYLE);
601 width = GetSystemMetrics(SM_CXVSCROLL);
602
603 GetClientRect(hwndStatus, &rcClient);
604
605 pt.x = rcClient.right;
606 pt.y = rcClient.top;
607 ClientToScreen(hwndStatus, &pt);
608 rc.left = pt.x - width;
609 rc.right = pt.x;
610 rc.top = pt.y;
611
612 pt.y = rcClient.bottom;
613 ClientToScreen(hwndStatus, &pt);
614 rc.bottom = pt.y;
615
616 /* check bounds when not maximized */
617 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.left, rc.top));
618 expect(HTBOTTOMRIGHT, r);
619 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.left - 1, rc.top));
620 expect(HTCLIENT, r);
621 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.left, rc.top - 1));
622 expect(HTBOTTOMRIGHT, r);
623 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right, rc.bottom));
624 expect(HTBOTTOMRIGHT, r);
625 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right + 1, rc.bottom));
626 expect(HTBOTTOMRIGHT, r);
627 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right, rc.bottom + 1));
628 expect(HTBOTTOMRIGHT, r);
629 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right - 1, rc.bottom - 1));
630 expect(HTBOTTOMRIGHT, r);
631
632 /* not maximized and right-to-left */
633 SetWindowLongA(hwndStatus, GWL_EXSTYLE, WS_EX_LAYOUTRTL);
634
635 pt.x = rcClient.right;
636 ClientToScreen(hwndStatus, &pt);
637 rc.left = pt.x + width;
638 rc.right = pt.x;
639
640 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.left, rc.top));
641 expect(HTBOTTOMLEFT, r);
642 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.left + 1, rc.top));
643 expect(HTCLIENT, r);
644 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.left, rc.top - 1));
645 expect(HTBOTTOMLEFT, r);
646 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right, rc.bottom));
647 expect(HTBOTTOMLEFT, r);
648 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right - 1, rc.bottom));
649 expect(HTBOTTOMLEFT, r);
650 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right, rc.bottom + 1));
651 expect(HTBOTTOMLEFT, r);
652 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right + 1, rc.bottom - 1));
653 expect(HTBOTTOMLEFT, r);
654
655 /* maximize with left-to-right */
656 SetWindowLongA(g_hMainWnd, GWL_STYLE, style|WS_MAXIMIZE);
657 SetWindowLongA(hwndStatus, GWL_EXSTYLE, 0);
658
659 GetClientRect(hwndStatus, &rcClient);
660
661 pt.x = rcClient.right;
662 pt.y = rcClient.top;
663 ClientToScreen(hwndStatus, &pt);
664 rc.left = pt.x - width;
665 rc.right = pt.x;
666 rc.top = pt.y;
667
668 pt.y = rcClient.bottom;
669 ClientToScreen(hwndStatus, &pt);
670 rc.bottom = pt.y;
671
672 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.left, rc.top));
673 expect(HTCLIENT, r);
674 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.left - 1, rc.top));
675 expect(HTCLIENT, r);
676 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.left, rc.top - 1));
677 expect(HTNOWHERE, r);
678 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right, rc.bottom));
679 expect(HTNOWHERE, r);
680 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right + 1, rc.bottom));
681 expect(HTNOWHERE, r);
682 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right, rc.bottom + 1));
683 expect(HTNOWHERE, r);
684 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right - 1, rc.bottom - 1));
685 expect(HTCLIENT, r);
686
687 /* maximized with right-to-left */
688 SetWindowLongA(hwndStatus, GWL_EXSTYLE, WS_EX_LAYOUTRTL);
689
690 pt.x = rcClient.right;
691 ClientToScreen(hwndStatus, &pt);
692 rc.left = pt.x + width;
693 rc.right = pt.x;
694
695 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.left, rc.top));
696 expect(HTCLIENT, r);
697 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.left + 1, rc.top));
698 expect(HTCLIENT, r);
699 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.left, rc.top - 1));
700 expect(HTNOWHERE, r);
701 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right, rc.bottom));
702 expect(HTNOWHERE, r);
703 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right - 1, rc.bottom));
704 expect(HTNOWHERE, r);
705 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right, rc.bottom + 1));
706 expect(HTNOWHERE, r);
707 r = SendMessageA(hwndStatus, WM_NCHITTEST, 0, MAKELPARAM(rc.right + 1, rc.bottom - 1));
708 expect(HTCLIENT, r);
709
710 SetWindowLongA(g_hMainWnd, GWL_STYLE, style);
711 DestroyWindow(hwndStatus);
712 }
713
init_functions(void)714 static void init_functions(void)
715 {
716 HMODULE hComCtl32 = LoadLibraryA("comctl32.dll");
717
718 #define X(f) p##f = (void*)GetProcAddress(hComCtl32, #f);
719 X(InitCommonControlsEx);
720 #undef X
721 }
722
START_TEST(status)723 START_TEST(status)
724 {
725 INITCOMMONCONTROLSEX iccex;
726
727 init_functions();
728
729 hinst = GetModuleHandleA(NULL);
730
731 iccex.dwSize = sizeof(iccex);
732 iccex.dwICC = ICC_BAR_CLASSES;
733 pInitCommonControlsEx(&iccex);
734
735 g_hMainWnd = CreateWindowExA(0, WC_STATICA, "", WS_OVERLAPPEDWINDOW,
736 CW_USEDEFAULT, CW_USEDEFAULT, 672+2*GetSystemMetrics(SM_CXSIZEFRAME),
737 226+GetSystemMetrics(SM_CYCAPTION)+2*GetSystemMetrics(SM_CYSIZEFRAME),
738 NULL, NULL, GetModuleHandleA(NULL), 0);
739
740 register_subclass();
741
742 test_status_control();
743 test_create();
744 test_height();
745 test_status_ownerdraw();
746 test_gettext();
747 test_notify();
748 test_sizegrip();
749 }
750