1 /*
2  * Unit tests for scrollbar
3  *
4  * Copyright 2008 Lyutin Anatoly (Etersoft)
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with this library; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19  */
20 
21 #include <assert.h>
22 #include <stdarg.h>
23 #include <stdio.h>
24 #include <windows.h>
25 
26 #include "wine/test.h"
27 
28 static HWND hScroll;
29 static BOOL bThemeActive = FALSE;
30 
31 static LRESULT CALLBACK MyWndProc( HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam )
32 {
33     switch(msg)
34     {
35 
36     case WM_CREATE:
37     {
38         hScroll = CreateWindowA( "SCROLLBAR", "", WS_CHILD | WS_VISIBLE, 0, 0, 120, 100, hWnd, (HMENU)100, GetModuleHandleA(0), 0 );
39 
40         return 0;
41     }
42     case WM_DESTROY:
43         PostQuitMessage(0);
44         break;
45     case WM_HSCROLL:
46     case WM_VSCROLL:
47         /* stop tracking */
48         ReleaseCapture();
49         return 0;
50     default:
51         return DefWindowProcA(hWnd, msg, wParam, lParam);
52     }
53     return 0;
54 }
55 
56 static HWND create_main_test_wnd(void)
57 {
58     HWND hMainWnd;
59 
60     hScroll = NULL;
61     hMainWnd = CreateWindowExA( 0, "MyTestWnd", "Scroll",
62       WS_OVERLAPPEDWINDOW|WS_VSCROLL|WS_HSCROLL,
63       CW_USEDEFAULT, CW_USEDEFAULT, 100, 100, NULL, NULL, GetModuleHandleA(NULL), 0 );
64     ok(hMainWnd != NULL, "Failed to create parent window. Tests aborted.\n");
65     ok(hScroll != NULL, "got NULL scroll bar handle\n");
66 
67     return hMainWnd;
68 }
69 
70 static void scrollbar_test_track(void)
71 {
72     HWND mainwnd;
73 
74     mainwnd = create_main_test_wnd();
75 
76     /* test that scrollbar tracking is terminated when
77      * the control loses mouse capture */
78     SendMessageA( hScroll, WM_LBUTTONDOWN, 0, MAKELPARAM( 1, 1));
79     /* a normal return from SendMessage */
80     /* not normal for instances such as closing the window */
81     ok( IsWindow( hScroll), "Scrollbar has gone!\n");
82 
83     DestroyWindow(hScroll);
84     DestroyWindow(mainwnd);
85 }
86 
87 static void test_EnableScrollBar(void)
88 {
89     HWND mainwnd;
90     BOOL ret;
91 
92     mainwnd = create_main_test_wnd();
93 
94     ret = EnableScrollBar( hScroll, SB_CTL, ESB_DISABLE_BOTH );
95     ok( ret, "The scrollbar should be disabled.\n" );
96     ok( !IsWindowEnabled( hScroll ), "The scrollbar window should be disabled.\n" );
97 
98     ret = EnableScrollBar( hScroll, SB_CTL, ESB_ENABLE_BOTH );
99     ok( ret, "The scrollbar should be enabled.\n" );
100     ok( IsWindowEnabled( hScroll ), "The scrollbar window should be enabled.\n" );
101 
102     /* test buttons separately */
103     ret = EnableScrollBar( hScroll, SB_CTL, ESB_DISABLE_LTUP );
104     ok( ret, "The scrollbar LTUP button should be disabled.\n" );
105     ok( IsWindowEnabled( hScroll ), "The scrollbar window should be enabled.\n" );
106     ret = EnableScrollBar( hScroll, SB_CTL, ESB_ENABLE_BOTH );
107     ok( ret, "The scrollbar should be enabled.\n" );
108     ok( IsWindowEnabled( hScroll ), "The scrollbar window should be enabled.\n" );
109 
110     ret = EnableScrollBar( hScroll, SB_CTL, ESB_DISABLE_RTDN );
111     ok( ret, "The scrollbar RTDN button should be disabled.\n" );
112     ok( IsWindowEnabled( hScroll ), "The scrollbar window should be enabled.\n" );
113     ret = EnableScrollBar( hScroll, SB_CTL, ESB_ENABLE_BOTH );
114     ok( ret, "The scrollbar should be enabled.\n" );
115     ok( IsWindowEnabled( hScroll ), "The scrollbar window should be enabled.\n" );
116 
117     SetLastError( 0xdeadbeef );
118     ret = EnableScrollBar( mainwnd, SB_CTL, ESB_ENABLE_BOTH );
119     ok( !ret, "EnableScrollBar should fail.\n" );
120     todo_wine
121     ok( GetLastError() == ERROR_INVALID_PARAMETER
122         || broken(GetLastError() == 0xdeadbeef), /* winxp */
123         "GetLastError() = %u\n", GetLastError() );
124 
125     /* disable window, try to re-enable */
126     ret = EnableWindow( hScroll, FALSE );
127     ok( !ret, "got %d\n", ret );
128     ok( !IsWindowEnabled( hScroll ), "The scrollbar window should be disabled.\n" );
129 
130     ret = EnableScrollBar( hScroll, SB_CTL, ESB_ENABLE_BOTH );
131     ok( ret, "got %d\n", ret );
132     ok( IsWindowEnabled( hScroll ), "The scrollbar window should be disabled.\n" );
133 
134     DestroyWindow(hScroll);
135     DestroyWindow(mainwnd);
136 }
137 
138 static void test_SetScrollPos(void)
139 {
140     HWND mainwnd;
141     int ret;
142 
143     mainwnd = create_main_test_wnd();
144 
145     EnableWindow( hScroll, FALSE );
146     ok( !IsWindowEnabled( hScroll ), "The scroll should be disabled.\n" );
147 
148     ret = SetScrollPos( hScroll, SB_CTL, 30, TRUE);
149     ok( !ret, "The position should not be set.\n" );
150 
151     ret = GetScrollPos( hScroll, SB_CTL);
152     ok( !ret, "The position should be equal to zero\n");
153 
154     ret = SetScrollRange( hScroll, SB_CTL, 0, 100, TRUE );
155     ok( ret, "The range should be set.\n" );
156 
157     ret = SetScrollPos( hScroll, SB_CTL, 30, TRUE);
158     ok( !ret , "The position should not be set.\n" );
159 
160     ret = GetScrollPos( hScroll, SB_CTL);
161     ok( ret == 30, "The position should be set!!!\n");
162 
163     EnableWindow( hScroll, TRUE );
164     ok( IsWindowEnabled( hScroll ), "The scroll should be enabled.\n" );
165 
166     ret = SetScrollPos( hScroll, SB_CTL, 30, TRUE);
167     ok( ret == 30, "The position should be set.\n" );
168 
169     ret = GetScrollPos( hScroll, SB_CTL);
170     ok( ret == 30, "The position should not be equal to zero\n");
171 
172     ret = SetScrollRange( hScroll, SB_CTL, 0, 100, TRUE );
173     ok( ret, "The range should be set.\n" );
174 
175     ret = SetScrollPos( hScroll, SB_CTL, 30, TRUE);
176     ok( ret == 30, "The position should be set.\n" );
177 
178     ret = GetScrollPos( hScroll, SB_CTL);
179     ok( ret == 30, "The position should not be equal to zero\n");
180 
181     SetLastError( 0xdeadbeef );
182     ret = SetScrollPos( mainwnd, SB_CTL, 30, TRUE );
183     ok( !ret, "The position should not be set.\n" );
184     ok( GetLastError() == 0xdeadbeef, "GetLastError() = %u\n", GetLastError() );
185 
186     SetLastError( 0xdeadbeef );
187     ret = GetScrollPos( mainwnd, SB_CTL );
188     ok( !ret, "The position should be equal to zero\n");
189     ok( GetLastError() == 0xdeadbeef, "GetLastError() = %u\n", GetLastError() );
190 
191     DestroyWindow(hScroll);
192     DestroyWindow(mainwnd);
193 }
194 
195 static void test_ShowScrollBar(void)
196 {
197     HWND mainwnd;
198     BOOL    ret;
199 
200     mainwnd = create_main_test_wnd();
201 
202     ret = ShowScrollBar( hScroll, SB_CTL, FALSE );
203     ok( ret, "The ShowScrollBar() should not failed.\n" );
204     ok( !IsWindowVisible( hScroll ), "The scrollbar window should not be visible\n" );
205 
206     ret = ShowScrollBar( hScroll, SB_CTL, TRUE );
207     ok( ret, "The ShowScrollBar() should not failed.\n" );
208     ok( !IsWindowVisible( hScroll ), "The scrollbar window should be visible\n" );
209 
210     ret = ShowScrollBar( NULL, SB_CTL, TRUE );
211     ok( !ret, "The ShowScrollBar() should failed.\n" );
212 
213     ret = ShowScrollBar( mainwnd, SB_CTL, TRUE );
214     ok( ret, "The ShowScrollBar() should not fail.\n" );
215 
216     DestroyWindow(hScroll);
217     DestroyWindow(mainwnd);
218 }
219 
220 static void test_GetScrollBarInfo(void)
221 {
222     HWND hMainWnd;
223     BOOL ret;
224     SCROLLBARINFO sbi;
225     RECT rect;
226     BOOL (WINAPI *pGetScrollBarInfo)(HWND, LONG, LPSCROLLBARINFO);
227 
228     pGetScrollBarInfo = (void*)GetProcAddress(GetModuleHandleA("user32.dll"), "GetScrollBarInfo");
229     if (!pGetScrollBarInfo)
230     {
231         win_skip("GetScrollBarInfo is not available\n");
232         return;
233     }
234 
235     hMainWnd = create_main_test_wnd();
236 
237     /* Test GetScrollBarInfo to make sure it returns rcScrollBar in screen
238      * coordinates. */
239     sbi.cbSize = sizeof(sbi);
240     ret = pGetScrollBarInfo( hScroll, OBJID_CLIENT, &sbi);
241     ok( ret, "The GetScrollBarInfo() call should not fail.\n" );
242     GetWindowRect( hScroll, &rect );
243     ok( ret, "The GetWindowRect() call should not fail.\n" );
244     ok( !(sbi.rgstate[0] & (STATE_SYSTEM_INVISIBLE|STATE_SYSTEM_OFFSCREEN)),
245         "unexpected rgstate(0x%x)\n", sbi.rgstate[0]);
246     ok(EqualRect(&rect, &sbi.rcScrollBar), "WindowRect %s != rcScrollBar %s\n",
247        wine_dbgstr_rect(&rect), wine_dbgstr_rect(&sbi.rcScrollBar));
248 
249     /* Test windows horizontal and vertical scrollbar to make sure rcScrollBar
250      * is still returned in screen coordinates by moving the window, and
251      * making sure that it shifts the rcScrollBar value. */
252     ShowWindow( hMainWnd, SW_SHOW );
253     sbi.cbSize = sizeof(sbi);
254     ret = pGetScrollBarInfo( hMainWnd, OBJID_HSCROLL, &sbi);
255     ok( ret, "The GetScrollBarInfo() call should not fail.\n" );
256     GetWindowRect( hMainWnd, &rect );
257     ok( ret, "The GetWindowRect() call should not fail.\n" );
258     MoveWindow( hMainWnd, rect.left+5, rect.top+5,
259                 rect.right-rect.left, rect.bottom-rect.top, TRUE );
260     rect = sbi.rcScrollBar;
261     OffsetRect(&rect, 5, 5);
262     ret = pGetScrollBarInfo( hMainWnd, OBJID_HSCROLL, &sbi);
263     ok( ret, "The GetScrollBarInfo() call should not fail.\n" );
264     ok(EqualRect(&rect, &sbi.rcScrollBar), "PreviousRect %s != CurrentRect %s\n",
265        wine_dbgstr_rect(&rect), wine_dbgstr_rect(&sbi.rcScrollBar));
266 
267     sbi.cbSize = sizeof(sbi);
268     ret = pGetScrollBarInfo( hMainWnd, OBJID_VSCROLL, &sbi);
269     ok( ret, "The GetScrollBarInfo() call should not fail.\n" );
270     GetWindowRect( hMainWnd, &rect );
271     ok( ret, "The GetWindowRect() call should not fail.\n" );
272     MoveWindow( hMainWnd, rect.left+5, rect.top+5,
273                 rect.right-rect.left, rect.bottom-rect.top, TRUE );
274     rect = sbi.rcScrollBar;
275     OffsetRect(&rect, 5, 5);
276     ret = pGetScrollBarInfo( hMainWnd, OBJID_VSCROLL, &sbi);
277     ok( ret, "The GetScrollBarInfo() call should not fail.\n" );
278     ok(EqualRect(&rect, &sbi.rcScrollBar), "PreviousRect %s != CurrentRect %s\n",
279        wine_dbgstr_rect(&rect), wine_dbgstr_rect(&sbi.rcScrollBar));
280 
281     DestroyWindow(hScroll);
282     DestroyWindow(hMainWnd);
283 }
284 
285 /* some tests designed to show that Horizontal and Vertical
286  * window scroll bar info are not created independently */
287 static void scrollbar_test_default( DWORD style)
288 {
289     INT min, max, ret;
290     DWORD winstyle;
291     HWND hwnd;
292     SCROLLINFO si = { sizeof( SCROLLINFO), SIF_TRACKPOS };
293 
294     hwnd = CreateWindowExA( 0, "static", "", WS_POPUP | style,
295                 0, 0, 10, 10, 0, 0, 0, NULL);
296     assert( hwnd != 0);
297 
298     ret = GetScrollRange( hwnd, SB_VERT, &min, &max);
299     ok( ret ||
300             broken( !ret) /* Win 9x/ME */ , "GetScrollRange failed.\n");
301     /* range is 0,0 if there are no H or V scroll bars. 0,100 otherwise */
302     if( !( style & ( WS_VSCROLL | WS_HSCROLL)))
303         ok( min == 0 && max == 0,
304                 "Scroll bar range is %d,%d. Expected 0,0. Style %08x\n", min, max, style);
305     else
306         ok(( min == 0 && max == 100) ||
307                 broken( min == 0 && max == 0), /* Win 9x/ME */
308                 "Scroll bar range is %d,%d. Expected 0,100. Style %08x\n", min, max, style);
309     ret = GetScrollRange( hwnd, SB_HORZ, &min, &max);
310     ok( ret ||
311             broken( !ret) /* Win 9x/ME */ , "GetScrollRange failed.\n");
312     /* range is 0,0 if there are no H or V scroll bars. 0,100 otherwise */
313     if( !( style & ( WS_VSCROLL | WS_HSCROLL)))
314         ok( min == 0 && max == 0,
315                 "Scroll bar range is %d,%d. Expected 0,0. Style %08x\n", min, max, style);
316     else
317         ok(( min == 0 && max == 100) ||
318                 broken( min == 0 && max == 0), /* Win 9x/ME */
319                 "Scroll bar range is %d,%d. Expected 0,100. Style %08x\n", min, max, style);
320     /* test GetScrollInfo, vist for vertical SB */
321     ret = GetScrollInfo( hwnd, SB_VERT, &si);
322     /* should fail if no H or V scroll bar styles are present. Succeed otherwise */
323     if( !( style & ( WS_VSCROLL | WS_HSCROLL)))
324         ok( !ret, "GetScrollInfo succeeded unexpectedly. Style is %08x\n", style);
325     else
326         ok( ret ||
327                 broken( !ret), /* Win 9x/ME */
328                 "GetScrollInfo failed unexpectedly. Style is %08x\n", style);
329     /* Same for Horizontal SB */
330     ret = GetScrollInfo( hwnd, SB_HORZ, &si);
331     /* should fail if no H or V scroll bar styles are present. Succeed otherwise */
332     if( !( style & ( WS_VSCROLL | WS_HSCROLL)))
333         ok( !ret, "GetScrollInfo succeeded unexpectedly. Style is %08x\n", style);
334     else
335         ok( ret ||
336                 broken( !ret), /* Win 9x/ME */
337                 "GetScrollInfo failed unexpectedly. Style is %08x\n", style);
338     /* now set the Vertical Scroll range to something that could be the default value it
339      * already has */;
340     ret = SetScrollRange( hwnd, SB_VERT, 0, 100, FALSE);
341     ok( ret, "SetScrollRange failed.\n");
342     /* and request the Horizontal range */
343     ret = GetScrollRange( hwnd, SB_HORZ, &min, &max);
344     ok( ret, "GetScrollRange failed.\n");
345     /* now the range should be 0,100 in ALL cases */
346     ok( min == 0 && max == 100,
347             "Scroll bar range is %d,%d. Expected 0,100. Style %08x\n", min, max, style);
348     /* See what is different now for GetScrollRange */
349     ret = GetScrollInfo( hwnd, SB_HORZ, &si);
350     /* should succeed in ALL cases */
351     ok( ret, "GetScrollInfo failed unexpectedly. Style is %08x\n", style);
352     ret = GetScrollInfo( hwnd, SB_VERT, &si);
353     /* should succeed in ALL cases */
354     ok( ret, "GetScrollInfo failed unexpectedly. Style is %08x\n", style);
355     /* report the windows style */
356     winstyle = GetWindowLongA( hwnd, GWL_STYLE );
357     /* WS_VSCROLL added to the window style */
358     if( !(style & WS_VSCROLL))
359     {
360         if (bThemeActive || style != WS_HSCROLL)
361 todo_wine
362             ok( (winstyle & (WS_HSCROLL|WS_VSCROLL)) == ( style | WS_VSCROLL),
363                 "unexpected style change %08x/%08x\n", winstyle, style);
364         else
365             ok( (winstyle & (WS_HSCROLL|WS_VSCROLL)) == style,
366                 "unexpected style change %08x/%08x\n", winstyle, style);
367     }
368     /* do the test again with H and V reversed.
369      * Start with a clean window */
370     DestroyWindow( hwnd);
371     hwnd = CreateWindowExA( 0, "static", "", WS_POPUP | style,
372                 0, 0, 10, 10, 0, 0, 0, NULL);
373     assert( hwnd != 0);
374     /* Set Horizontal Scroll range to something that could be the default value it
375      * already has */;
376     ret = SetScrollRange( hwnd, SB_HORZ, 0, 100, FALSE);
377     ok( ret, "SetScrollRange failed.\n");
378     /* and request the Vertical range */
379     ret = GetScrollRange( hwnd, SB_VERT, &min, &max);
380     ok( ret, "GetScrollRange failed.\n");
381     /* now the range should be 0,100 in ALL cases */
382     ok( min == 0 && max == 100,
383             "Scroll bar range is %d,%d. Expected 0,100. Style %08x\n", min, max, style);
384     /* See what is different now for GetScrollRange */
385     ret = GetScrollInfo( hwnd, SB_HORZ, &si);
386     /* should succeed in ALL cases */
387     ok( ret, "GetScrollInfo failed unexpectedly. Style is %08x\n", style);
388     ret = GetScrollInfo( hwnd, SB_VERT, &si);
389     /* should succeed in ALL cases */
390     ok( ret, "GetScrollInfo failed unexpectedly. Style is %08x\n", style);
391     /* report the windows style */
392     winstyle = GetWindowLongA( hwnd, GWL_STYLE );
393     /* WS_HSCROLL added to the window style */
394     if( !(style & WS_HSCROLL))
395     {
396         if (bThemeActive || style != WS_VSCROLL)
397 todo_wine
398             ok( (winstyle & (WS_HSCROLL|WS_VSCROLL)) == ( style | WS_HSCROLL),
399                 "unexpected style change %08x/%08x\n", winstyle, style);
400         else
401             ok( (winstyle & (WS_HSCROLL|WS_VSCROLL)) == style,
402                 "unexpected style change %08x/%08x\n", winstyle, style);
403     }
404     /* Slightly change the test to use SetScrollInfo
405      * Start with a clean window */
406     DestroyWindow( hwnd);
407     hwnd = CreateWindowExA( 0, "static", "", WS_POPUP | style,
408                 0, 0, 10, 10, 0, 0, 0, NULL);
409     assert( hwnd != 0);
410     /* set Horizontal position with SetScrollInfo */
411     si.nPos = 0;
412     si.nMin = 11;
413     si.nMax = 22;
414     si.fMask |= SIF_RANGE;
415     ret = SetScrollInfo( hwnd, SB_HORZ, &si, FALSE);
416     ok( ret, "SetScrollInfo failed. Style is %08x\n", style);
417     /* and request the Vertical range */
418     ret = GetScrollRange( hwnd, SB_VERT, &min, &max);
419     ok( ret, "GetScrollRange failed.\n");
420     /* now the range should be 0,100 in ALL cases */
421     ok( min == 0 && max == 100,
422             "Scroll bar range is %d,%d. Expected 0,100. Style %08x\n", min, max, style);
423     /* See what is different now for GetScrollRange */
424     ret = GetScrollInfo( hwnd, SB_HORZ, &si);
425     /* should succeed in ALL cases */
426     ok( ret, "GetScrollInfo failed unexpectedly. Style is %08x\n", style);
427     ret = GetScrollInfo( hwnd, SB_VERT, &si);
428     /* should succeed in ALL cases */
429     ok( ret, "GetScrollInfo failed unexpectedly. Style is %08x\n", style);
430     /* also test if the window scroll bars are enabled */
431     ret = EnableScrollBar( hwnd, SB_VERT, ESB_ENABLE_BOTH);
432     ok( !ret, "Vertical window scroll bar was not enabled\n");
433     ret = EnableScrollBar( hwnd, SB_HORZ, ESB_ENABLE_BOTH);
434     ok( !ret, "Horizontal window scroll bar was not enabled\n");
435     DestroyWindow( hwnd);
436     /* finally, check if adding a WS_[HV]SCROLL style of a window makes the scroll info
437      * available */
438     if( style & (WS_HSCROLL | WS_VSCROLL)) return;/* only test if not yet set */
439     /* Start with a clean window */
440     DestroyWindow( hwnd);
441     hwnd = CreateWindowExA( 0, "static", "", WS_POPUP ,
442                 0, 0, 10, 10, 0, 0, 0, NULL);
443     assert( hwnd != 0);
444     ret = GetScrollInfo( hwnd, SB_VERT, &si);
445     /* should fail */
446     ok( !ret, "GetScrollInfo succeeded unexpectedly. Style is %08x\n", style);
447     /* add scroll styles */
448     winstyle = GetWindowLongA( hwnd, GWL_STYLE );
449     SetWindowLongW( hwnd, GWL_STYLE, winstyle | WS_VSCROLL | WS_HSCROLL);
450     ret = GetScrollInfo( hwnd, SB_VERT, &si);
451     /* should still fail */
452     ok( !ret, "GetScrollInfo succeeded unexpectedly. Style is %08x\n", style);
453     /* clean up */
454     DestroyWindow( hwnd);
455 }
456 
457 static LRESULT CALLBACK scroll_init_proc(HWND hwnd, UINT msg,
458                                         WPARAM wparam, LPARAM lparam)
459 {
460     SCROLLINFO horz, vert;
461     CREATESTRUCTA *cs = (CREATESTRUCTA *)lparam;
462     BOOL h_ret, v_ret;
463 
464     switch(msg)
465     {
466         case WM_NCCREATE:
467             return cs->lpCreateParams ? DefWindowProcA(hwnd, msg, wparam, lparam) :
468                                         TRUE;
469 
470         case WM_CREATE:
471             horz.cbSize = sizeof horz;
472             horz.fMask  = SIF_ALL;
473             horz.nMin   = 0xdeadbeaf;
474             horz.nMax   = 0xbaadc0de;
475             vert = horz;
476             h_ret = GetScrollInfo(hwnd, SB_HORZ, &horz);
477             v_ret = GetScrollInfo(hwnd, SB_VERT, &vert);
478 
479             if(cs->lpCreateParams)
480             {
481                 /* WM_NCCREATE was passed to DefWindowProc */
482                 if(cs->style & (WS_VSCROLL | WS_HSCROLL))
483                 {
484                     ok(h_ret && v_ret, "GetScrollInfo() should return NON-zero "
485                             "but got h_ret=%d v_ret=%d\n", h_ret, v_ret);
486                     ok(vert.nMin == 0 && vert.nMax == 100,
487                             "unexpected init values(SB_VERT): min=%d max=%d\n",
488                             vert.nMin, vert.nMax);
489                     ok(horz.nMin == 0 && horz.nMax == 100,
490                             "unexpected init values(SB_HORZ): min=%d max=%d\n",
491                             horz.nMin, horz.nMax);
492                 }
493                 else
494                 {
495                     ok(!h_ret && !v_ret, "GetScrollInfo() should return zeru, "
496                             "but got h_ret=%d v_ret=%d\n", h_ret, v_ret);
497                     ok(vert.nMin == 0xdeadbeaf && vert.nMax == 0xbaadc0de,
498                             "unexpected  initialization(SB_VERT): min=%d max=%d\n",
499                             vert.nMin, vert.nMax);
500                     ok(horz.nMin == 0xdeadbeaf && horz.nMax == 0xbaadc0de,
501                             "unexpected  initialization(SB_HORZ): min=%d max=%d\n",
502                             horz.nMin, horz.nMax);
503                 }
504             }
505             else
506             {
507                 ok(!h_ret && !v_ret, "GetScrollInfo() should return zeru, "
508                     "but got h_ret=%d v_ret=%d\n", h_ret, v_ret);
509                 ok(horz.nMin == 0xdeadbeaf && horz.nMax == 0xbaadc0de &&
510                     vert.nMin == 0xdeadbeaf && vert.nMax == 0xbaadc0de,
511                         "unexpected initialization\n");
512             }
513             return FALSE; /* abort creation */
514 
515         default:
516             /* need for WM_GETMINMAXINFO, which precedes WM_NCCREATE */
517             return 0;
518     }
519 }
520 
521 static void scrollbar_test_init(void)
522 {
523     WNDCLASSEXA wc;
524     CHAR cls_name[] = "scroll_test_class";
525     LONG style[] = {WS_VSCROLL, WS_HSCROLL, WS_VSCROLL | WS_HSCROLL, 0};
526     int i;
527 
528     memset( &wc, 0, sizeof wc );
529     wc.cbSize        = sizeof wc;
530     wc.style         = CS_VREDRAW | CS_HREDRAW;
531     wc.hInstance     = GetModuleHandleA(0);
532     wc.hCursor       = LoadCursorA(NULL, (LPCSTR)IDC_ARROW);
533     wc.hbrBackground = GetStockObject(WHITE_BRUSH);
534     wc.lpszClassName = cls_name;
535     wc.lpfnWndProc   = scroll_init_proc;
536     RegisterClassExA(&wc);
537 
538     for(i = 0; i < ARRAY_SIZE(style); i++)
539     {
540         /* need not to destroy these windows due creation abort */
541         CreateWindowExA(0, cls_name, NULL, style[i],
542                 100, 100, 100, 100, NULL, NULL, wc.hInstance, (LPVOID)TRUE);
543         CreateWindowExA(0, cls_name, NULL, style[i],
544                 100, 100, 100, 100, NULL, NULL, wc.hInstance, (LPVOID)FALSE);
545     }
546     UnregisterClassA(cls_name, wc.hInstance);
547 }
548 
549 static void test_SetScrollInfo(void)
550 {
551     SCROLLINFO si;
552     HWND mainwnd;
553     BOOL ret;
554 
555     mainwnd = create_main_test_wnd();
556 
557     ret = IsWindowEnabled(hScroll);
558     ok(ret, "scroll bar disabled\n");
559 
560     EnableScrollBar(hScroll, SB_CTL, ESB_DISABLE_BOTH);
561 
562     ret = IsWindowEnabled(hScroll);
563     ok(!ret, "scroll bar enabled\n");
564 
565     memset(&si, 0, sizeof(si));
566     si.cbSize = sizeof(si);
567     si.fMask = 0xf;
568     ret = GetScrollInfo(hScroll, SB_CTL, &si);
569     ok(ret, "got %d\n", ret);
570 
571     /* SetScrollInfo */
572     memset(&si, 0, sizeof(si));
573     si.cbSize = sizeof(si);
574     ret = IsWindowEnabled(hScroll);
575     ok(!ret, "scroll bar enabled\n");
576     si.fMask = SIF_POS|SIF_RANGE|SIF_PAGE|SIF_DISABLENOSCROLL;
577     si.nMax = 100;
578     si.nMin = 10;
579     si.nPos = 0;
580     si.nPage = 100;
581     SetScrollInfo(hScroll, SB_CTL, &si, TRUE);
582     ret = IsWindowEnabled(hScroll);
583     ok(!ret, "scroll bar enabled\n");
584 
585     si.fMask = 0xf;
586     ret = GetScrollInfo(hScroll, SB_CTL, &si);
587     ok(ret, "got %d\n", ret);
588 
589     EnableScrollBar(hScroll, SB_CTL, ESB_ENABLE_BOTH);
590     ok(IsWindowEnabled(hScroll), "expected enabled scrollbar\n");
591 
592     si.fMask = SIF_POS|SIF_RANGE|SIF_PAGE|SIF_DISABLENOSCROLL;
593     si.nMax = 10;
594     si.nMin = 100;
595     si.nPos = 0;
596     si.nPage = 100;
597     SetScrollInfo(hScroll, SB_CTL, &si, TRUE);
598     ret = IsWindowEnabled(hScroll);
599     ok(ret, "scroll bar disabled\n");
600 
601     DestroyWindow(hScroll);
602     DestroyWindow(mainwnd);
603 }
604 
605 static WNDPROC scrollbar_wndproc;
606 
607 static SCROLLINFO set_scrollinfo;
608 
609 static LRESULT CALLBACK subclass_proc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam)
610 {
611     if (msg == WM_CREATE && ((CREATESTRUCTA*)lparam)->lpCreateParams)
612         return DefWindowProcA(hwnd, msg, wparam, lparam);
613 
614     if (msg == SBM_SETSCROLLINFO)
615         set_scrollinfo = *(SCROLLINFO*)lparam;
616 
617     return CallWindowProcA(scrollbar_wndproc, hwnd, msg, wparam, lparam);
618 }
619 
620 static void test_subclass(void)
621 {
622     SCROLLBARINFO scroll_info;
623     WNDCLASSEXA class_info;
624     WNDCLASSA wc;
625     LRESULT res;
626     HWND hwnd;
627     BOOL r;
628 
629     r = GetClassInfoExA(GetModuleHandleA(NULL), "SCROLLBAR", &class_info);
630     ok(r, "GetClassInfoEx failed: %u\n", GetLastError());
631     scrollbar_wndproc = class_info.lpfnWndProc;
632 
633     memset(&wc, 0, sizeof(wc));
634     wc.cbWndExtra = class_info.cbWndExtra + 3; /* more space than needed works */
635     wc.hInstance = GetModuleHandleA(NULL);
636     wc.lpszClassName = "MyTestSubclass";
637     wc.lpfnWndProc = subclass_proc;
638     r = RegisterClassA(&wc);
639     ok(r, "RegisterClass failed: %u\n", GetLastError());
640 
641     hwnd = CreateWindowExA( 0, "MyTestSubclass", "Scroll", WS_OVERLAPPEDWINDOW,
642                             CW_USEDEFAULT, CW_USEDEFAULT, 100, 100, NULL, NULL, GetModuleHandleA(NULL), 0 );
643     ok(hwnd != NULL, "Failed to create window: %u\n", GetLastError());
644 
645     r = SetScrollRange(hwnd, SB_CTL, 0, 100, TRUE);
646     ok(r, "SetScrollRange failed: %u\n", GetLastError());
647 
648     res = SetScrollPos(hwnd, SB_CTL, 2, FALSE);
649     ok(!res, "SetScrollPos returned %lu\n", res);
650 
651     memset(&set_scrollinfo, 0xcc, sizeof(set_scrollinfo));
652     res = SetScrollPos(hwnd, SB_CTL, 1, FALSE);
653     ok(res == 2, "SetScrollPos returned %lu\n", res);
654     ok(set_scrollinfo.cbSize == sizeof(SCROLLINFO), "cbSize = %u\n", set_scrollinfo.cbSize);
655     todo_wine
656     ok(set_scrollinfo.fMask == (0x1000 | SIF_POS), "fMask = %x\n", set_scrollinfo.fMask);
657     ok(set_scrollinfo.nPos == 1, "nPos = %x\n", set_scrollinfo.nPos);
658 
659     memset(&scroll_info, 0xcc, sizeof(scroll_info));
660     scroll_info.cbSize = sizeof(scroll_info);
661     res = SendMessageA(hwnd, SBM_GETSCROLLBARINFO, 0, (LPARAM)&scroll_info);
662     ok(res == 1, "SBM_GETSCROLLBARINFO returned %lu\n", res);
663 
664     DestroyWindow(hwnd);
665 
666     /* if we skip calling wndproc for WM_CREATE, window is not considered a scrollbar */
667     hwnd = CreateWindowExA( 0, "MyTestSubclass", "Scroll", WS_OVERLAPPEDWINDOW,
668                             CW_USEDEFAULT, CW_USEDEFAULT, 100, 100, NULL, NULL, GetModuleHandleA(NULL), (void *)1 );
669     ok(hwnd != NULL, "Failed to create window: %u\n", GetLastError());
670 
671     memset(&scroll_info, 0xcc, sizeof(scroll_info));
672     scroll_info.cbSize = sizeof(scroll_info);
673     res = SendMessageA(hwnd, SBM_GETSCROLLBARINFO, 0, (LPARAM)&scroll_info);
674     ok(!res, "SBM_GETSCROLLBARINFO returned %lu\n", res);
675 
676     DestroyWindow(hwnd);
677 
678     /* not enough space in extra data */
679     wc.cbWndExtra = class_info.cbWndExtra - 1;
680     wc.lpszClassName = "MyTestSubclass2";
681     r = RegisterClassA(&wc);
682     ok(r, "RegisterClass failed: %u\n", GetLastError());
683 
684     hwnd = CreateWindowExA( 0, "MyTestSubclass2", "Scroll", WS_OVERLAPPEDWINDOW,
685                             CW_USEDEFAULT, CW_USEDEFAULT, 100, 100, NULL, NULL, GetModuleHandleA(NULL), 0 );
686     ok(hwnd != NULL, "Failed to create window: %u\n", GetLastError());
687 
688     memset(&scroll_info, 0xcc, sizeof(scroll_info));
689     scroll_info.cbSize = sizeof(scroll_info);
690     res = SendMessageA(hwnd, SBM_GETSCROLLBARINFO, 0, (LPARAM)&scroll_info);
691     ok(!res, "SBM_GETSCROLLBARINFO returned %lu\n", res);
692 
693     memset(&set_scrollinfo, 0xcc, sizeof(set_scrollinfo));
694     res = SetScrollPos(hwnd, SB_CTL, 1, FALSE);
695     ok(res == 0, "SetScrollPos returned %lu\n", res);
696     ok(set_scrollinfo.cbSize == sizeof(SCROLLINFO), "cbSize = %u\n", set_scrollinfo.cbSize);
697     todo_wine
698     ok(set_scrollinfo.fMask == (0x1000 | SIF_POS), "fMask = %x\n", set_scrollinfo.fMask);
699     ok(set_scrollinfo.nPos == 1, "nPos = %x\n", set_scrollinfo.nPos);
700 
701     DestroyWindow(hwnd);
702 }
703 
704 START_TEST ( scroll )
705 {
706     WNDCLASSA wc;
707     HMODULE hUxtheme;
708     BOOL (WINAPI * pIsThemeActive)(VOID);
709 
710     wc.style = CS_HREDRAW | CS_VREDRAW;
711     wc.cbClsExtra = 0;
712     wc.cbWndExtra = 0;
713     wc.hInstance = GetModuleHandleA(NULL);
714     wc.hIcon = NULL;
715     wc.hCursor = LoadCursorA(NULL, (LPCSTR)IDC_IBEAM);
716     wc.hbrBackground = GetSysColorBrush(COLOR_WINDOW);
717     wc.lpszMenuName = NULL;
718     wc.lpszClassName = "MyTestWnd";
719     wc.lpfnWndProc = MyWndProc;
720     RegisterClassA(&wc);
721 
722     test_EnableScrollBar();
723     test_SetScrollPos();
724     test_ShowScrollBar();
725     test_GetScrollBarInfo();
726     scrollbar_test_track();
727     test_SetScrollInfo();
728     test_subclass();
729 
730     /* Some test results vary depending of theming being active or not */
731     hUxtheme = LoadLibraryA("uxtheme.dll");
732     if (hUxtheme)
733     {
734         pIsThemeActive = (void*)GetProcAddress(hUxtheme, "IsThemeActive");
735         if (pIsThemeActive)
736             bThemeActive = pIsThemeActive();
737         FreeLibrary(hUxtheme);
738     }
739 
740     scrollbar_test_default( 0);
741     scrollbar_test_default( WS_HSCROLL);
742     scrollbar_test_default( WS_VSCROLL);
743     scrollbar_test_default( WS_HSCROLL | WS_VSCROLL);
744 
745     scrollbar_test_init();
746 }
747