Release 1.6-rc2.
[wine/testsucceed.git] / dlls / comctl32 / tests / progress.c
blob59a4b00804d76796a797f04fc91d9ad290ee6ead
1 /* Unit tests for the progress bar control.
3 * Copyright 2005 Michael Kaufmann
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20 #include <assert.h>
21 #include <stdarg.h>
23 #include "windef.h"
24 #include "winbase.h"
25 #include "wingdi.h"
26 #include "winuser.h"
27 #include "commctrl.h"
29 #include "wine/test.h"
32 static HWND hProgressParentWnd, hProgressWnd;
33 static const char progressTestClass[] = "ProgressBarTestClass";
35 static HWND create_progress(DWORD style)
37 return CreateWindowExA(0, PROGRESS_CLASSA, "", WS_VISIBLE | style,
38 0, 0, 100, 20, NULL, NULL, GetModuleHandleA(NULL), 0);
41 /* try to make sure pending X events have been processed before continuing */
42 static void flush_events(void)
44 MSG msg;
45 int diff = 100;
46 DWORD time = GetTickCount() + diff;
48 while (diff > 0)
50 if (MsgWaitForMultipleObjects( 0, NULL, FALSE, min(10,diff), QS_ALLINPUT ) == WAIT_TIMEOUT) break;
51 while (PeekMessage( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessage( &msg );
52 diff = time - GetTickCount();
56 static LRESULT CALLBACK ProgressTestWndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
58 switch(msg) {
60 case WM_DESTROY:
61 PostQuitMessage(0);
62 break;
64 default:
65 return DefWindowProcA(hWnd, msg, wParam, lParam);
68 return 0L;
71 static WNDPROC progress_wndproc;
72 static BOOL erased;
73 static RECT last_paint_rect;
75 static LRESULT CALLBACK ProgressSubclassProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
77 if (msg == WM_PAINT)
79 GetUpdateRect(hWnd, &last_paint_rect, FALSE);
81 else if (msg == WM_ERASEBKGND)
83 erased = TRUE;
85 return CallWindowProc(progress_wndproc, hWnd, msg, wParam, lParam);
89 static void update_window(HWND hWnd)
91 UpdateWindow(hWnd);
92 ok(!GetUpdateRect(hWnd, NULL, FALSE), "GetUpdateRect must return zero after UpdateWindow\n");
96 static void init(void)
98 HMODULE hComctl32;
99 BOOL (WINAPI *pInitCommonControlsEx)(const INITCOMMONCONTROLSEX*);
100 WNDCLASSA wc;
101 RECT rect;
103 hComctl32 = GetModuleHandleA("comctl32.dll");
104 pInitCommonControlsEx = (void*)GetProcAddress(hComctl32, "InitCommonControlsEx");
105 if (pInitCommonControlsEx)
107 INITCOMMONCONTROLSEX iccex;
108 iccex.dwSize = sizeof(iccex);
109 iccex.dwICC = ICC_PROGRESS_CLASS;
110 pInitCommonControlsEx(&iccex);
112 else
113 InitCommonControls();
115 wc.style = CS_HREDRAW | CS_VREDRAW;
116 wc.cbClsExtra = 0;
117 wc.cbWndExtra = 0;
118 wc.hInstance = GetModuleHandleA(NULL);
119 wc.hIcon = NULL;
120 wc.hCursor = LoadCursorA(NULL, IDC_ARROW);
121 wc.hbrBackground = GetSysColorBrush(COLOR_WINDOW);
122 wc.lpszMenuName = NULL;
123 wc.lpszClassName = progressTestClass;
124 wc.lpfnWndProc = ProgressTestWndProc;
125 RegisterClassA(&wc);
127 rect.left = 0;
128 rect.top = 0;
129 rect.right = 400;
130 rect.bottom = 20;
131 assert(AdjustWindowRect(&rect, WS_OVERLAPPEDWINDOW, FALSE));
133 hProgressParentWnd = CreateWindowExA(0, progressTestClass, "Progress Bar Test", WS_OVERLAPPEDWINDOW,
134 CW_USEDEFAULT, CW_USEDEFAULT, rect.right - rect.left, rect.bottom - rect.top, NULL, NULL, GetModuleHandleA(NULL), 0);
135 assert(hProgressParentWnd != NULL);
137 GetClientRect(hProgressParentWnd, &rect);
138 hProgressWnd = CreateWindowEx(0, PROGRESS_CLASS, "", WS_CHILD | WS_VISIBLE,
139 0, 0, rect.right, rect.bottom, hProgressParentWnd, NULL, GetModuleHandleA(NULL), 0);
140 assert(hProgressWnd != NULL);
141 progress_wndproc = (WNDPROC)SetWindowLongPtr(hProgressWnd, GWLP_WNDPROC, (LPARAM)ProgressSubclassProc);
143 ShowWindow(hProgressParentWnd, SW_SHOWNORMAL);
144 ok(GetUpdateRect(hProgressParentWnd, NULL, FALSE), "GetUpdateRect: There should be a region that needs to be updated\n");
145 flush_events();
146 update_window(hProgressParentWnd);
150 static void cleanup(void)
152 MSG msg;
154 PostMessageA(hProgressParentWnd, WM_CLOSE, 0, 0);
155 while (GetMessageA(&msg,0,0,0)) {
156 TranslateMessage(&msg);
157 DispatchMessageA(&msg);
160 UnregisterClassA(progressTestClass, GetModuleHandleA(NULL));
165 * Tests if a progress bar repaints itself immediately when it receives
166 * some specific messages.
168 static void test_redraw(void)
170 RECT client_rect;
171 LRESULT ret;
173 SendMessageA(hProgressWnd, PBM_SETRANGE, 0, MAKELPARAM(0, 100));
174 SendMessageA(hProgressWnd, PBM_SETPOS, 10, 0);
175 SendMessageA(hProgressWnd, PBM_SETSTEP, 20, 0);
176 update_window(hProgressWnd);
178 /* PBM_SETPOS */
179 ok(SendMessageA(hProgressWnd, PBM_SETPOS, 50, 0) == 10, "PBM_SETPOS must return the previous position\n");
180 ok(!GetUpdateRect(hProgressWnd, NULL, FALSE), "PBM_SETPOS: The progress bar should be redrawn immediately\n");
182 /* PBM_DELTAPOS */
183 ok(SendMessageA(hProgressWnd, PBM_DELTAPOS, 15, 0) == 50, "PBM_DELTAPOS must return the previous position\n");
184 ok(!GetUpdateRect(hProgressWnd, NULL, FALSE), "PBM_DELTAPOS: The progress bar should be redrawn immediately\n");
186 /* PBM_SETPOS */
187 ok(SendMessageA(hProgressWnd, PBM_SETPOS, 80, 0) == 65, "PBM_SETPOS must return the previous position\n");
188 ok(!GetUpdateRect(hProgressWnd, NULL, FALSE), "PBM_SETPOS: The progress bar should be redrawn immediately\n");
190 /* PBM_STEPIT */
191 ok(SendMessageA(hProgressWnd, PBM_STEPIT, 0, 0) == 80, "PBM_STEPIT must return the previous position\n");
192 ok(!GetUpdateRect(hProgressWnd, NULL, FALSE), "PBM_STEPIT: The progress bar should be redrawn immediately\n");
193 ret = SendMessageA(hProgressWnd, PBM_GETPOS, 0, 0);
194 if (ret == 0)
195 win_skip("PBM_GETPOS needs comctl32 > 4.70\n");
196 else
197 ok(ret == 100, "PBM_GETPOS returned a wrong position : %d\n", (UINT)ret);
199 /* PBM_SETRANGE and PBM_SETRANGE32:
200 Usually the progress bar doesn't repaint itself immediately. If the
201 position is not in the new range, it does.
202 Don't test this, it may change in future Windows versions. */
204 SendMessage(hProgressWnd, PBM_SETPOS, 0, 0);
205 update_window(hProgressWnd);
207 /* increase to 10 - no background erase required */
208 erased = FALSE;
209 SetRectEmpty(&last_paint_rect);
210 SendMessage(hProgressWnd, PBM_SETPOS, 10, 0);
211 GetClientRect(hProgressWnd, &client_rect);
212 ok(EqualRect(&last_paint_rect, &client_rect),
213 "last_paint_rect was { %d, %d, %d, %d } instead of { %d, %d, %d, %d }\n",
214 last_paint_rect.left, last_paint_rect.top, last_paint_rect.right, last_paint_rect.bottom,
215 client_rect.left, client_rect.top, client_rect.right, client_rect.bottom);
216 update_window(hProgressWnd);
217 ok(!erased, "Progress bar shouldn't have erased the background\n");
219 /* decrease to 0 - background erase will be required */
220 erased = FALSE;
221 SetRectEmpty(&last_paint_rect);
222 SendMessage(hProgressWnd, PBM_SETPOS, 0, 0);
223 GetClientRect(hProgressWnd, &client_rect);
224 ok(EqualRect(&last_paint_rect, &client_rect),
225 "last_paint_rect was { %d, %d, %d, %d } instead of { %d, %d, %d, %d }\n",
226 last_paint_rect.left, last_paint_rect.top, last_paint_rect.right, last_paint_rect.bottom,
227 client_rect.left, client_rect.top, client_rect.right, client_rect.bottom);
228 update_window(hProgressWnd);
229 ok(erased, "Progress bar should have erased the background\n");
232 static void test_setcolors(void)
234 HWND progress;
235 COLORREF clr;
237 progress = create_progress(PBS_SMOOTH);
239 clr = SendMessageA(progress, PBM_SETBARCOLOR, 0, 0);
240 ok(clr == CLR_DEFAULT, "got %x\n", clr);
242 clr = SendMessageA(progress, PBM_SETBARCOLOR, 0, RGB(0, 255, 0));
243 ok(clr == 0, "got %x\n", clr);
245 clr = SendMessageA(progress, PBM_SETBARCOLOR, 0, CLR_DEFAULT);
246 ok(clr == RGB(0, 255, 0), "got %x\n", clr);
248 clr = SendMessageA(progress, PBM_SETBKCOLOR, 0, 0);
249 ok(clr == CLR_DEFAULT, "got %x\n", clr);
251 clr = SendMessageA(progress, PBM_SETBKCOLOR, 0, RGB(255, 0, 0));
252 ok(clr == 0, "got %x\n", clr);
254 clr = SendMessageA(progress, PBM_SETBKCOLOR, 0, CLR_DEFAULT);
255 ok(clr == RGB(255, 0, 0), "got %x\n", clr);
257 DestroyWindow(progress);
260 START_TEST(progress)
262 init();
264 test_redraw();
265 test_setcolors();
267 cleanup();