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