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