2 * comctl32 month calendar unit tests
4 * Copyright (C) 2006 Vitaliy Margolen
5 * Copyright (C) 2007 Farshad Agah
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
30 #include "wine/test.h"
36 #define expect(expected, got) ok(expected == got, "Expected %d, got %d\n", expected, got);
37 #define expect_hex(expected, got) ok(expected == got, "Expected %x, got %x\n", expected, got);
38 #define expect_d(expected, got) ok(abs((expected) - (got)) <= 2, "Expected %d, got %d\n", expected, got);
40 #define NUM_MSG_SEQUENCES 2
41 #define PARENT_SEQ_INDEX 0
42 #define MONTHCAL_SEQ_INDEX 1
44 static struct msg_sequence
*sequences
[NUM_MSG_SEQUENCES
];
46 static HWND parent_wnd
;
48 static const struct message create_parent_window_seq
[] = {
49 { WM_GETMINMAXINFO
, sent
},
50 { WM_NCCREATE
, sent
},
51 { WM_NCCALCSIZE
, sent
|wparam
, 0 },
53 { WM_SHOWWINDOW
, sent
|wparam
, 1 },
54 { WM_WINDOWPOSCHANGING
, sent
|wparam
, 0 },
55 { WM_QUERYNEWPALETTE
, sent
|optional
},
56 { WM_WINDOWPOSCHANGING
, sent
|wparam
|optional
, 0 },
57 { WM_WINDOWPOSCHANGED
, sent
|optional
},
58 { WM_ACTIVATEAPP
, sent
|wparam
, 1 },
59 { WM_NCACTIVATE
, sent
},
60 { WM_ACTIVATE
, sent
|wparam
, 1 },
61 { WM_IME_SETCONTEXT
, sent
|wparam
|defwinproc
|optional
, 1 },
62 { WM_IME_NOTIFY
, sent
|defwinproc
|optional
},
63 { WM_SETFOCUS
, sent
|wparam
|defwinproc
, 0 },
64 /* Win9x adds SWP_NOZORDER below */
65 { WM_WINDOWPOSCHANGED
, sent
, /*|wparam, SWP_SHOWWINDOW|SWP_NOSIZE|SWP_NOMOVE|SWP_NOCLIENTSIZE|SWP_NOCLIENTMOVE*/ },
66 { WM_NCCALCSIZE
, sent
|wparam
|optional
, 1 },
72 static const struct message create_monthcal_control_seq
[] = {
73 { WM_NOTIFYFORMAT
, sent
|lparam
, 0, NF_QUERY
},
74 { WM_QUERYUISTATE
, sent
|optional
},
76 { WM_PARENTNOTIFY
, sent
|wparam
, WM_CREATE
},
80 static const struct message create_monthcal_multi_sel_style_seq
[] = {
81 { WM_NOTIFYFORMAT
, sent
|lparam
, 0, NF_QUERY
},
82 { WM_QUERYUISTATE
, sent
|optional
},
84 { WM_PARENTNOTIFY
, sent
},
88 static const struct message monthcal_curr_date_seq
[] = {
89 { MCM_SETCURSEL
, sent
|wparam
, 0},
90 { WM_PAINT
, sent
|wparam
|lparam
|defwinproc
, 0, 0},
91 { MCM_SETCURSEL
, sent
|wparam
, 0},
92 { MCM_SETCURSEL
, sent
|wparam
, 0},
93 { MCM_GETCURSEL
, sent
|wparam
, 0},
94 { MCM_GETCURSEL
, sent
|wparam
|lparam
, 0, 0},
98 static const struct message monthcal_first_day_seq
[] = {
99 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
101 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, -5},
102 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
104 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, -4},
105 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
107 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, -3},
108 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
110 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, -2},
111 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
113 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, -1},
114 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
116 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
117 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
119 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 1},
120 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
122 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 2},
123 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
125 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 3},
126 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
128 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 4},
129 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
131 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 5},
132 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
134 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 6},
135 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
137 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 7},
138 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
140 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 8},
141 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
143 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 9},
144 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
146 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 10},
147 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
149 { MCM_SETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 11},
150 { MCM_GETFIRSTDAYOFWEEK
, sent
|wparam
|lparam
, 0, 0},
154 static const struct message monthcal_unicode_seq
[] = {
155 { MCM_GETUNICODEFORMAT
, sent
|wparam
|lparam
, 0, 0},
156 { MCM_SETUNICODEFORMAT
, sent
|wparam
|lparam
, 1, 0},
157 { MCM_GETUNICODEFORMAT
, sent
|wparam
|lparam
, 0, 0},
158 { MCM_SETUNICODEFORMAT
, sent
|wparam
|lparam
, 0, 0},
159 { MCM_GETUNICODEFORMAT
, sent
|wparam
|lparam
, 0, 0},
160 { MCM_SETUNICODEFORMAT
, sent
|wparam
|lparam
, 1, 0},
164 static const struct message monthcal_hit_test_seq
[] = {
165 { MCM_SETCURSEL
, sent
|wparam
, 0},
166 { WM_PAINT
, sent
|wparam
|lparam
|defwinproc
, 0, 0},
167 { MCM_HITTEST
, sent
|wparam
, 0},
168 { MCM_HITTEST
, sent
|wparam
, 0},
169 { MCM_HITTEST
, sent
|wparam
, 0},
170 { MCM_HITTEST
, sent
|wparam
, 0},
171 { MCM_HITTEST
, sent
|wparam
, 0},
172 { MCM_HITTEST
, sent
|wparam
, 0},
173 { MCM_HITTEST
, sent
|wparam
, 0},
174 { MCM_HITTEST
, sent
|wparam
, 0},
175 { MCM_HITTEST
, sent
|wparam
, 0},
176 { MCM_HITTEST
, sent
|wparam
, 0},
180 static const struct message monthcal_todaylink_seq
[] = {
181 { MCM_HITTEST
, sent
|wparam
, 0},
182 { MCM_SETTODAY
, sent
|wparam
, 0},
183 { WM_PAINT
, sent
|wparam
|lparam
|defwinproc
, 0, 0},
184 { MCM_GETTODAY
, sent
|wparam
, 0},
185 { WM_LBUTTONDOWN
, sent
|wparam
, MK_LBUTTON
},
186 { WM_CAPTURECHANGED
, sent
|wparam
|lparam
|defwinproc
, 0, 0},
187 { WM_PAINT
, sent
|wparam
|lparam
|defwinproc
, 0, 0},
188 { MCM_GETCURSEL
, sent
|wparam
, 0},
192 static const struct message monthcal_today_seq
[] = {
193 { MCM_SETTODAY
, sent
|wparam
, 0},
194 { WM_PAINT
, sent
|wparam
|lparam
|defwinproc
, 0, 0},
195 { MCM_GETTODAY
, sent
|wparam
, 0},
196 { MCM_SETTODAY
, sent
|wparam
, 0},
197 { WM_PAINT
, sent
|wparam
|lparam
|defwinproc
, 0, 0},
198 { MCM_GETTODAY
, sent
|wparam
, 0},
202 static const struct message monthcal_scroll_seq
[] = {
203 { MCM_SETMONTHDELTA
, sent
|wparam
|lparam
, 2, 0},
204 { MCM_SETMONTHDELTA
, sent
|wparam
|lparam
, 3, 0},
205 { MCM_GETMONTHDELTA
, sent
|wparam
|lparam
, 0, 0},
206 { MCM_SETMONTHDELTA
, sent
|wparam
|lparam
, 12, 0},
207 { MCM_GETMONTHDELTA
, sent
|wparam
|lparam
, 0, 0},
208 { MCM_SETMONTHDELTA
, sent
|wparam
|lparam
, 15, 0},
209 { MCM_GETMONTHDELTA
, sent
|wparam
|lparam
, 0, 0},
210 { MCM_SETMONTHDELTA
, sent
|wparam
|lparam
, -5, 0},
211 { MCM_GETMONTHDELTA
, sent
|wparam
|lparam
, 0, 0},
215 static const struct message monthcal_monthrange_seq
[] = {
216 { MCM_GETMONTHRANGE
, sent
|wparam
, GMR_VISIBLE
},
217 { MCM_GETMONTHRANGE
, sent
|wparam
, GMR_DAYSTATE
},
221 static const struct message monthcal_max_sel_day_seq
[] = {
222 { MCM_SETMAXSELCOUNT
, sent
|wparam
|lparam
, 5, 0},
223 { MCM_GETMAXSELCOUNT
, sent
|wparam
|lparam
, 0, 0},
224 { MCM_SETMAXSELCOUNT
, sent
|wparam
|lparam
, 15, 0},
225 { MCM_GETMAXSELCOUNT
, sent
|wparam
|lparam
, 0, 0},
226 { MCM_SETMAXSELCOUNT
, sent
|wparam
|lparam
, -1, 0},
227 { MCM_GETMAXSELCOUNT
, sent
|wparam
|lparam
, 0, 0},
231 /* expected message sequence for parent*/
232 static const struct message destroy_monthcal_parent_msgs_seq
[] = {
233 { WM_PARENTNOTIFY
, sent
|wparam
, WM_DESTROY
},
237 /* expected message sequence for child*/
238 static const struct message destroy_monthcal_child_msgs_seq
[] = {
239 { 0x0090, sent
|optional
}, /* Vista */
240 { WM_SHOWWINDOW
, sent
|wparam
|lparam
, 0, 0},
241 { WM_WINDOWPOSCHANGING
, sent
|wparam
, 0},
242 { WM_WINDOWPOSCHANGED
, sent
|wparam
, 0},
243 { WM_DESTROY
, sent
|wparam
|lparam
, 0, 0},
244 { WM_NCDESTROY
, sent
|wparam
|lparam
, 0, 0},
248 static const struct message destroy_monthcal_multi_sel_style_seq
[] = {
249 { 0x0090, sent
|optional
}, /* Vista */
250 { WM_SHOWWINDOW
, sent
|wparam
|lparam
, 0, 0},
251 { WM_WINDOWPOSCHANGING
, sent
|wparam
, 0},
252 { WM_WINDOWPOSCHANGED
, sent
|wparam
, 0},
253 { WM_DESTROY
, sent
|wparam
|lparam
, 0, 0},
254 { WM_NCDESTROY
, sent
|wparam
|lparam
, 0, 0},
258 static void test_monthcal(void)
261 SYSTEMTIME st
[2], st1
[2], today
;
262 int res
, month_range
;
265 hwnd
= CreateWindowA(MONTHCAL_CLASSA
, "MonthCal", WS_POPUP
| WS_VISIBLE
, CW_USEDEFAULT
,
266 0, 300, 300, 0, 0, NULL
, NULL
);
267 ok(hwnd
!= NULL
, "Failed to create MonthCal\n");
269 /* test range just after creation */
270 memset(&st
, 0xcc, sizeof(st
));
271 limits
= SendMessage(hwnd
, MCM_GETRANGE
, 0, (LPARAM
)st
);
273 broken(limits
== GDTR_MIN
), /* comctl32 <= 4.70 */
274 "No limits should be set (%d)\n", limits
);
275 if (limits
== GDTR_MIN
)
277 win_skip("comctl32 <= 4.70 is broken\n");
282 ok(0 == st
[0].wYear
||
283 broken(1752 == st
[0].wYear
), /* comctl32 <= 4.72 */
284 "Expected 0, got %d\n", st
[0].wYear
);
285 ok(0 == st
[0].wMonth
||
286 broken(9 == st
[0].wMonth
), /* comctl32 <= 4.72 */
287 "Expected 0, got %d\n", st
[0].wMonth
);
288 ok(0 == st
[0].wDay
||
289 broken(14 == st
[0].wDay
), /* comctl32 <= 4.72 */
290 "Expected 0, got %d\n", st
[0].wDay
);
291 expect(0, st
[0].wDayOfWeek
);
292 expect(0, st
[0].wHour
);
293 expect(0, st
[0].wMinute
);
294 expect(0, st
[0].wSecond
);
295 expect(0, st
[0].wMilliseconds
);
297 expect(0, st
[1].wYear
);
298 expect(0, st
[1].wMonth
);
299 expect(0, st
[1].wDay
);
300 expect(0, st
[1].wDayOfWeek
);
301 expect(0, st
[1].wHour
);
302 expect(0, st
[1].wMinute
);
303 expect(0, st
[1].wSecond
);
304 expect(0, st
[1].wMilliseconds
);
306 GetSystemTime(&st
[0]);
309 SendMessage(hwnd
, MCM_GETTODAY
, 0, (LPARAM
)&today
);
311 /* Invalid date/time */
313 /* Time should not matter */
314 st
[1].wHour
= st
[1].wMinute
= st
[1].wSecond
= 70;
315 st
[1].wMilliseconds
= 1200;
316 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MAX
, (LPARAM
)st
), "Failed to set MAX limit\n");
317 /* invalid timestamp is written back with today data and msecs untouched */
318 expect(today
.wHour
, st
[1].wHour
);
319 expect(today
.wMinute
, st
[1].wMinute
);
320 expect(today
.wSecond
, st
[1].wSecond
);
321 expect(1200, st
[1].wMilliseconds
);
323 ok(SendMessage(hwnd
, MCM_GETRANGE
, 0, (LPARAM
)st1
) == GDTR_MAX
, "No limits should be set\n");
324 ok(st1
[0].wYear
!= 2000, "Lower limit changed\n");
325 /* invalid timestamp should be replaced with today data, except msecs */
326 expect(today
.wHour
, st1
[1].wHour
);
327 expect(today
.wMinute
, st1
[1].wMinute
);
328 expect(today
.wSecond
, st1
[1].wSecond
);
329 expect(1200, st1
[1].wMilliseconds
);
331 /* Invalid date/time with invalid milliseconds only */
332 GetSystemTime(&st
[0]);
334 /* Time should not matter */
335 st
[1].wMilliseconds
= 1200;
336 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MAX
, (LPARAM
)st
), "Failed to set MAX limit\n");
337 /* invalid milliseconds field doesn't lead to invalid timestamp */
338 expect(st
[0].wHour
, st
[1].wHour
);
339 expect(st
[0].wMinute
, st
[1].wMinute
);
340 expect(st
[0].wSecond
, st
[1].wSecond
);
341 expect(1200, st
[1].wMilliseconds
);
343 GetSystemTime(&st
[0]);
346 ok(!SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MIN
| GDTR_MAX
, (LPARAM
)st
), "Should have failed to set limits\n");
347 ok(SendMessage(hwnd
, MCM_GETRANGE
, 0, (LPARAM
)st1
) == GDTR_MAX
, "No limits should be set\n");
348 ok(st1
[0].wYear
!= 2000, "Lower limit changed\n");
349 ok(!SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MAX
, (LPARAM
)st
), "Should have failed to set MAX limit\n");
350 ok(SendMessage(hwnd
, MCM_GETRANGE
, 0, (LPARAM
)st1
) == GDTR_MAX
, "No limits should be set\n");
351 ok(st1
[0].wYear
!= 2000, "Lower limit changed\n");
353 GetSystemTime(&st
[0]);
358 month_range
= SendMessage(hwnd
, MCM_GETMONTHRANGE
, GMR_VISIBLE
, (LPARAM
)st1
);
360 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MIN
| GDTR_MAX
, (LPARAM
)st
), "Failed to set both min and max limits\n");
361 res
= SendMessage(hwnd
, MCM_GETMONTHRANGE
, GMR_VISIBLE
, (LPARAM
)st1
);
362 ok(res
== month_range
, "Invalid month range (%d)\n", res
);
363 ok(SendMessage(hwnd
, MCM_GETRANGE
, 0, (LPARAM
)st1
) == (GDTR_MIN
|GDTR_MAX
), "Limits should be set\n");
366 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MIN
| GDTR_MAX
, (LPARAM
)st
), "Failed to set both min and max limits\n");
367 res
= SendMessage(hwnd
, MCM_GETMONTHRANGE
, GMR_VISIBLE
, (LPARAM
)st1
);
368 ok(res
== month_range
, "Invalid month range (%d)\n", res
);
371 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MIN
| GDTR_MAX
, (LPARAM
)st
), "Failed to set both min and max limits\n");
373 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MIN
| GDTR_MAX
, (LPARAM
)st
), "Failed to set both min and max limits\n");
376 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MAX
, (LPARAM
)st
), "Failed to set max limit\n");
377 ok(SendMessage(hwnd
, MCM_GETRANGE
, 0, (LPARAM
)st1
) == GDTR_MAX
, "Only MAX limit should be set\n");
379 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MAX
, (LPARAM
)st
), "Failed to set max limit\n");
381 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MAX
, (LPARAM
)st
), "Failed to set max limit\n");
383 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MAX
, (LPARAM
)st
), "Failed to set max limit\n");
384 ok(SendMessage(hwnd
, MCM_GETRANGE
, 0, (LPARAM
)st1
) == GDTR_MAX
, "Only MAX limit should be set\n");
386 /* set both limits, then set max < min */
387 GetSystemTime(&st
[0]);
390 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MIN
|GDTR_MAX
, (LPARAM
)st
), "Failed to set limits\n");
391 ok(SendMessage(hwnd
, MCM_GETRANGE
, 0, (LPARAM
)st1
) == (GDTR_MIN
|GDTR_MAX
), "Min limit expected\n");
393 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MAX
, (LPARAM
)st
), "Failed to set limits\n");
394 ok(SendMessage(hwnd
, MCM_GETRANGE
, 0, (LPARAM
)st1
) == GDTR_MAX
, "Max limit expected\n");
396 expect(0, st1
[0].wYear
);
397 expect(0, st1
[0].wMonth
);
398 expect(0, st1
[0].wDay
);
399 expect(0, st1
[0].wDayOfWeek
);
400 expect(0, st1
[0].wHour
);
401 expect(0, st1
[0].wMinute
);
402 expect(0, st1
[0].wSecond
);
403 expect(0, st1
[0].wMilliseconds
);
405 expect(st
[1].wYear
, st1
[1].wYear
);
406 expect(st
[1].wMonth
, st1
[1].wMonth
);
407 expect(st
[1].wDay
, st1
[1].wDay
);
408 expect(st
[1].wDayOfWeek
, st1
[1].wDayOfWeek
);
409 expect(st
[1].wHour
, st1
[1].wHour
);
410 expect(st
[1].wMinute
, st1
[1].wMinute
);
411 expect(st
[1].wSecond
, st1
[1].wSecond
);
412 expect(st
[1].wMilliseconds
, st1
[1].wMilliseconds
);
416 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MIN
|GDTR_MAX
, (LPARAM
)st
), "Failed to set limits\n");
417 ok(SendMessage(hwnd
, MCM_GETRANGE
, 0, (LPARAM
)st1
) == (GDTR_MIN
|GDTR_MAX
), "Min limit expected\n");
418 st
[0].wYear
++; /* start == end now */
419 ok(SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MIN
, (LPARAM
)st
), "Failed to set limits\n");
420 ok(SendMessage(hwnd
, MCM_GETRANGE
, 0, (LPARAM
)st1
) == GDTR_MIN
, "Min limit expected\n");
422 expect(st
[0].wYear
, st1
[0].wYear
);
423 expect(st
[0].wMonth
, st1
[0].wMonth
);
424 expect(st
[0].wDay
, st1
[0].wDay
);
425 expect(st
[0].wDayOfWeek
, st1
[0].wDayOfWeek
);
426 expect(st
[0].wHour
, st1
[0].wHour
);
427 expect(st
[0].wMinute
, st1
[0].wMinute
);
428 expect(st
[0].wSecond
, st1
[0].wSecond
);
429 expect(st
[0].wMilliseconds
, st1
[0].wMilliseconds
);
431 expect(0, st1
[1].wYear
);
432 expect(0, st1
[1].wMonth
);
433 expect(0, st1
[1].wDay
);
434 expect(0, st1
[1].wDayOfWeek
);
435 expect(0, st1
[1].wHour
);
436 expect(0, st1
[1].wMinute
);
437 expect(0, st1
[1].wSecond
);
438 expect(0, st1
[1].wMilliseconds
);
443 static LRESULT WINAPI
parent_wnd_proc(HWND hwnd
, UINT message
, WPARAM wParam
, LPARAM lParam
)
445 static LONG defwndproc_counter
= 0;
449 /* log system messages, except for painting */
450 if (message
< WM_USER
&&
451 message
!= WM_PAINT
&&
452 message
!= WM_ERASEBKGND
&&
453 message
!= WM_NCPAINT
&&
454 message
!= WM_NCHITTEST
&&
455 message
!= WM_GETTEXT
&&
456 message
!= WM_GETICON
&&
457 message
!= WM_DEVICECHANGE
)
459 trace("parent: %p, %04x, %08lx, %08lx\n", hwnd
, message
, wParam
, lParam
);
461 msg
.message
= message
;
462 msg
.flags
= sent
|wparam
|lparam
;
463 if (defwndproc_counter
) msg
.flags
|= defwinproc
;
466 add_message(sequences
, PARENT_SEQ_INDEX
, &msg
);
469 defwndproc_counter
++;
470 ret
= DefWindowProcA(hwnd
, message
, wParam
, lParam
);
471 defwndproc_counter
--;
476 static BOOL
register_parent_wnd_class(void)
481 cls
.lpfnWndProc
= parent_wnd_proc
;
484 cls
.hInstance
= GetModuleHandleA(NULL
);
486 cls
.hCursor
= LoadCursorA(0, IDC_ARROW
);
487 cls
.hbrBackground
= GetStockObject(WHITE_BRUSH
);
488 cls
.lpszMenuName
= NULL
;
489 cls
.lpszClassName
= "Month-Cal test parent class";
490 return RegisterClassA(&cls
);
493 static HWND
create_parent_window(void)
497 InitCommonControls();
499 /* flush message sequences, so we can check the new sequence by the end of function */
500 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
502 if (!register_parent_wnd_class())
505 hwnd
= CreateWindowEx(0, "Month-Cal test parent class",
506 "Month-Cal test parent window",
507 WS_CAPTION
| WS_SYSMENU
| WS_MINIMIZEBOX
|
508 WS_MAXIMIZEBOX
| WS_VISIBLE
,
510 GetDesktopWindow(), NULL
, GetModuleHandleA(NULL
), NULL
);
514 /* check for message sequences */
515 ok_sequence(sequences
, PARENT_SEQ_INDEX
, create_parent_window_seq
, "create parent window", FALSE
);
520 static LRESULT WINAPI
monthcal_subclass_proc(HWND hwnd
, UINT message
, WPARAM wParam
, LPARAM lParam
)
522 WNDPROC oldproc
= (WNDPROC
)GetWindowLongPtrA(hwnd
, GWLP_USERDATA
);
523 static LONG defwndproc_counter
= 0;
527 msg
.message
= message
;
528 msg
.flags
= sent
|wparam
|lparam
;
529 if (defwndproc_counter
) msg
.flags
|= defwinproc
;
532 add_message(sequences
, MONTHCAL_SEQ_INDEX
, &msg
);
534 /* some debug output for style changing */
535 if ((message
== WM_STYLECHANGING
||
536 message
== WM_STYLECHANGED
) && lParam
)
538 STYLESTRUCT
*style
= (STYLESTRUCT
*)lParam
;
539 trace("\told style: 0x%08x, new style: 0x%08x\n", style
->styleOld
, style
->styleNew
);
542 defwndproc_counter
++;
543 ret
= CallWindowProcA(oldproc
, hwnd
, message
, wParam
, lParam
);
544 defwndproc_counter
--;
549 static HWND
create_monthcal_control(DWORD style
)
554 hwnd
= CreateWindowEx(0,
557 WS_CHILD
| WS_BORDER
| WS_VISIBLE
| style
,
559 parent_wnd
, NULL
, GetModuleHandleA(NULL
), NULL
);
561 if (!hwnd
) return NULL
;
563 oldproc
= (WNDPROC
)SetWindowLongPtrA(hwnd
, GWLP_WNDPROC
,
564 (LONG_PTR
)monthcal_subclass_proc
);
565 SetWindowLongPtrA(hwnd
, GWLP_USERDATA
, (LONG_PTR
)oldproc
);
567 SendMessage(hwnd
, WM_SETFONT
, (WPARAM
)GetStockObject(SYSTEM_FONT
), 0);
573 /* Setter and Getters Tests */
575 static void test_color(void)
577 COLORREF color
, prev
;
580 hwnd
= create_monthcal_control(0);
582 /* invalid color index */
583 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TRAILINGTEXT
+ 1, 0);
585 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_TRAILINGTEXT
+ 1, RGB(255,255,255));
588 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_BACKGROUND
, 0);
589 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_BACKGROUND
, RGB(0,0,0));
591 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_BACKGROUND
, 0);
592 expect(RGB(0,0,0), color
);
593 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_BACKGROUND
, RGB(255,255,255));
595 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_BACKGROUND
, 0);
596 expect(RGB(255,255,255), color
);
598 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_MONTHBK
, 0);
599 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_MONTHBK
, RGB(0,0,0));
601 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_MONTHBK
, 0);
602 expect(RGB(0,0,0), color
);
603 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_MONTHBK
, RGB(255,255,255));
605 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_MONTHBK
, 0);
606 expect(RGB(255,255,255), color
);
608 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TEXT
, 0);
609 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_TEXT
, RGB(0,0,0));
611 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TEXT
, 0);
612 expect(RGB(0,0,0), color
);
613 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_TEXT
, RGB(255,255,255));
615 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TEXT
, 0);
616 expect(RGB(255,255,255), color
);
618 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TITLEBK
, 0);
619 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_TITLEBK
, RGB(0,0,0));
621 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TITLEBK
, 0);
622 expect(RGB(0,0,0), color
);
623 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_TITLEBK
, RGB(255,255,255));
625 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TITLEBK
, 0);
626 expect(RGB(255,255,255), color
);
628 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TITLETEXT
, 0);
629 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_TITLETEXT
, RGB(0,0,0));
631 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TITLETEXT
, 0);
632 expect(RGB(0,0,0), color
);
633 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_TITLETEXT
, RGB(255,255,255));
635 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TITLETEXT
, 0);
636 expect(RGB(255,255,255), color
);
638 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TRAILINGTEXT
, 0);
639 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_TRAILINGTEXT
, RGB(0,0,0));
641 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TRAILINGTEXT
, 0);
642 expect(RGB(0,0,0), color
);
643 prev
= SendMessage(hwnd
, MCM_SETCOLOR
, MCSC_TRAILINGTEXT
, RGB(255,255,255));
645 color
= SendMessage(hwnd
, MCM_GETCOLOR
, MCSC_TRAILINGTEXT
, 0);
646 expect(RGB(255,255,255), color
);
651 static void test_currdate(void)
653 SYSTEMTIME st_original
, st_new
, st_test
;
657 hwnd
= create_monthcal_control(0);
659 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
661 /* Setter and Getters for current date selected */
662 st_original
.wYear
= 2000;
663 st_original
.wMonth
= 11;
664 st_original
.wDay
= 28;
665 st_original
.wHour
= 11;
666 st_original
.wMinute
= 59;
667 st_original
.wSecond
= 30;
668 st_original
.wMilliseconds
= 0;
669 st_original
.wDayOfWeek
= 0;
671 st_new
= st_test
= st_original
;
673 /* Should not validate the time */
674 res
= SendMessage(hwnd
, MCM_SETCURSEL
, 0, (LPARAM
)&st_test
);
677 /* Overflow matters, check for wDay */
679 res
= SendMessage(hwnd
, MCM_SETCURSEL
, 0, (LPARAM
)&st_test
);
682 /* correct wDay before checking for wMonth */
684 expect(st_original
.wDay
, st_test
.wDay
);
686 /* Overflow matters, check for wMonth */
688 res
= SendMessage(hwnd
, MCM_SETCURSEL
, 0, (LPARAM
)&st_test
);
691 /* checking if gets the information right, modify st_new */
699 res
= SendMessage(hwnd
, MCM_GETCURSEL
, 0, (LPARAM
)&st_new
);
702 /* st_new change to st_origin, above settings with overflow */
703 /* should not change the current settings */
704 expect(st_original
.wYear
, st_new
.wYear
);
705 expect(st_original
.wMonth
, st_new
.wMonth
);
706 expect(st_original
.wDay
, st_new
.wDay
);
707 ok(st_original
.wHour
== st_new
.wHour
||
708 broken(0 == st_new
.wHour
), /* comctl32 <= 4.70 */
709 "Expected %d, got %d\n", st_original
.wHour
, st_new
.wHour
);
710 ok(st_original
.wMinute
== st_new
.wMinute
||
711 broken(0 == st_new
.wMinute
), /* comctl32 <= 4.70 */
712 "Expected %d, got %d\n", st_original
.wMinute
, st_new
.wMinute
);
713 ok(st_original
.wSecond
== st_new
.wSecond
||
714 broken(0 == st_new
.wSecond
), /* comctl32 <= 4.70 */
715 "Expected %d, got %d\n", st_original
.wSecond
, st_new
.wSecond
);
717 /* lparam cannot be NULL */
718 res
= SendMessage(hwnd
, MCM_GETCURSEL
, 0, 0);
721 ok_sequence(sequences
, MONTHCAL_SEQ_INDEX
, monthcal_curr_date_seq
, "monthcal currDate", TRUE
);
723 /* December, 31, 9999 is the maximum allowed date */
724 memset(&st_new
, 0, sizeof(st_new
));
728 res
= SendMessage(hwnd
, MCM_SETCURSEL
, 0, (LPARAM
)&st_new
);
730 memset(&st_test
, 0, sizeof(st_test
));
731 res
= SendMessage(hwnd
, MCM_GETCURSEL
, 0, (LPARAM
)&st_test
);
733 expect(st_new
.wYear
, st_test
.wYear
);
734 expect(st_new
.wMonth
, st_test
.wMonth
);
735 expect(st_new
.wDay
, st_test
.wDay
);
736 expect(st_new
.wHour
, st_test
.wHour
);
737 expect(st_new
.wMinute
, st_test
.wMinute
);
738 expect(st_new
.wSecond
, st_test
.wSecond
);
739 /* try one day later */
740 st_original
= st_new
;
741 st_new
.wYear
= 10000;
744 res
= SendMessage(hwnd
, MCM_SETCURSEL
, 0, (LPARAM
)&st_new
);
746 broken(1 == res
), /* comctl32 <= 4.72 */
747 "Expected 0, got %d\n", res
);
750 memset(&st_test
, 0, sizeof(st_test
));
751 res
= SendMessage(hwnd
, MCM_GETCURSEL
, 0, (LPARAM
)&st_test
);
753 expect(st_original
.wYear
, st_test
.wYear
);
754 expect(st_original
.wMonth
, st_test
.wMonth
);
755 expect(st_original
.wDay
, st_test
.wDay
);
756 expect(st_original
.wHour
, st_test
.wHour
);
757 expect(st_original
.wMinute
, st_test
.wMinute
);
758 expect(st_original
.wSecond
, st_test
.wSecond
);
761 /* setting selection equal to current reports success even if out range */
762 memset(&st_new
, 0, sizeof(st_new
));
766 res
= SendMessage(hwnd
, MCM_SETCURSEL
, 0, (LPARAM
)&st_new
);
768 memset(&st_test
, 0, sizeof(st_test
));
769 st_test
.wYear
= 2009;
772 res
= SendMessage(hwnd
, MCM_SETRANGE
, GDTR_MIN
, (LPARAM
)&st_test
);
774 /* set to current again */
775 res
= SendMessage(hwnd
, MCM_SETCURSEL
, 0, (LPARAM
)&st_new
);
778 /* set with invalid day of week */
779 memset(&st_test
, 0, sizeof(st_test
));
780 st_test
.wYear
= 2009;
783 st_test
.wDayOfWeek
= 100;
784 res
= SendMessage(hwnd
, MCM_SETCURSEL
, 0, (LPARAM
)&st_test
);
787 memset(&st_test
, 0, sizeof(st_test
));
788 res
= SendMessage(hwnd
, MCM_GETCURSEL
, 0, (LPARAM
)&st_test
);
790 expect(2009, st_test
.wYear
);
791 expect(7, st_test
.wDay
);
792 expect(10, st_test
.wMonth
);
793 expect(3, st_test
.wDayOfWeek
);
798 static void test_firstDay(void)
800 int res
, fday
, i
, prev
;
801 CHAR b
[128], caltype
[3];
802 LCID lcid
= LOCALE_USER_DEFAULT
;
806 SetLastError(0xdeadbeef);
807 ret
= GetLocaleInfoA(lcid
, LOCALE_ICALENDARTYPE
, caltype
, 3);
809 skip("Must know local calendar type (%x)\n", GetLastError());
811 } else if (atoi(caltype
) != CAL_GREGORIAN
) {
812 skip("MonthCalendar Control only supports Gregorian calendar (type: %s)\n", caltype
);
816 hwnd
= create_monthcal_control(0);
818 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
820 /* Setter and Getters for first day of week */
821 /* check for locale first day */
822 if(GetLocaleInfoA(lcid
, LOCALE_IFIRSTDAYOFWEEK
, b
, 128)){
824 trace("fday: %d\n", fday
);
825 res
= SendMessage(hwnd
, MCM_GETFIRSTDAYOFWEEK
, 0, 0);
829 /* checking for the values that actually will be stored as */
830 /* current first day when we set a new value */
831 for (i
= -5; i
< 12; i
++){
832 res
= SendMessage(hwnd
, MCM_SETFIRSTDAYOFWEEK
, 0, i
);
834 res
= SendMessage(hwnd
, MCM_GETFIRSTDAYOFWEEK
, 0, 0);
838 expect(MAKELONG(fday
, FALSE
), res
);
840 /* out of range sets max first day of week, locale is ignored */
841 expect(MAKELONG(6, TRUE
), res
);
843 expect(MAKELONG(i
, TRUE
), res
);
847 ok_sequence(sequences
, MONTHCAL_SEQ_INDEX
, monthcal_first_day_seq
, "monthcal firstDay", FALSE
);
850 skip("Cannot retrieve first day of the week\n");
856 static void test_unicode(void)
861 hwnd
= create_monthcal_control(0);
863 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
865 /* Setter and Getters for Unicode format */
867 /* getting the current settings */
868 temp
= SendMessage(hwnd
, MCM_GETUNICODEFORMAT
, 0, 0);
870 /* setting to 1, should return previous settings */
871 res
= SendMessage(hwnd
, MCM_SETUNICODEFORMAT
, 1, 0);
874 /* current setting is 1, so, should return 1 */
875 res
= SendMessage(hwnd
, MCM_GETUNICODEFORMAT
, 0, 0);
877 broken(0 == res
), /* comctl32 <= 4.70 */
878 "Expected 1, got %d\n", res
);
880 /* setting to 0, should return previous settings */
881 res
= SendMessage(hwnd
, MCM_SETUNICODEFORMAT
, 0, 0);
883 broken(0 == res
), /* comctl32 <= 4.70 */
884 "Expected 1, got %d\n", res
);
886 /* current setting is 0, so, it should return 0 */
887 res
= SendMessage(hwnd
, MCM_GETUNICODEFORMAT
, 0, 0);
890 /* should return previous settings */
891 res
= SendMessage(hwnd
, MCM_SETUNICODEFORMAT
, 1, 0);
894 ok_sequence(sequences
, MONTHCAL_SEQ_INDEX
, monthcal_unicode_seq
, "monthcal unicode", FALSE
);
899 static void test_hittest(void)
901 typedef struct hittest_test
907 static const hittest_test_t title_hits
[] = {
908 /* Start is the same everywhere */
910 { MCHT_TITLEBTNPREV
, 0 },
911 /* The middle piece is only tested for presence of items */
912 /* End is the same everywhere */
913 { MCHT_TITLEBTNNEXT
, 0 },
925 char yearmonth
[80], *locale_month
, *locale_year
;
926 int month_count
, year_count
;
929 memset(&mchit
, 0, sizeof(MCHITTESTINFO
));
931 hwnd
= create_monthcal_control(0);
933 /* test with invalid structure size */
934 mchit
.cbSize
= MCHITTESTINFO_V1_SIZE
- 1;
937 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
)&mchit
);
938 expect(0, mchit
.pt
.x
);
939 expect(0, mchit
.pt
.y
);
941 expect(0, mchit
.uHit
);
942 /* test with invalid pointer */
943 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, 0);
946 /* resize control to display single Calendar */
947 res
= SendMessage(hwnd
, MCM_GETMINREQRECT
, 0, (LPARAM
)&r
);
950 win_skip("Message MCM_GETMINREQRECT unsupported. Skipping.\n");
954 MoveWindow(hwnd
, 0, 0, r
.right
, r
.bottom
, FALSE
);
956 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
964 st
.wMilliseconds
= 0;
967 res
= SendMessage(hwnd
, MCM_SETCURSEL
, 0, (LPARAM
)&st
);
970 /* (0, 0) is the top left of the control - title */
971 mchit
.cbSize
= MCHITTESTINFO_V1_SIZE
;
974 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
)&mchit
);
975 expect(0, mchit
.pt
.x
);
976 expect(0, mchit
.pt
.y
);
977 expect(mchit
.uHit
, res
);
978 expect_hex(MCHT_TITLE
, res
);
980 /* bottom right of the control and should not be active */
981 mchit
.pt
.x
= r
.right
;
982 mchit
.pt
.y
= r
.bottom
;
983 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
)&mchit
);
984 expect(r
.right
, mchit
.pt
.x
);
985 expect(r
.bottom
, mchit
.pt
.y
);
986 expect(mchit
.uHit
, res
);
987 todo_wine
expect_hex(MCHT_NOWHERE
, res
);
989 /* completely out of the control, should not be active */
990 mchit
.pt
.x
= 2 * r
.right
;
991 mchit
.pt
.y
= 2 * r
.bottom
;
992 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
) & mchit
);
993 expect(2 * r
.right
, mchit
.pt
.x
);
994 expect(2 * r
.bottom
, mchit
.pt
.y
);
995 expect(mchit
.uHit
, res
);
996 todo_wine
expect_hex(MCHT_NOWHERE
, res
);
998 /* in active area - day of the week */
999 mchit
.pt
.x
= r
.right
/ 2;
1000 mchit
.pt
.y
= r
.bottom
/ 2;
1001 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
) & mchit
);
1002 expect(r
.right
/ 2, mchit
.pt
.x
);
1003 expect(r
.bottom
/ 2, mchit
.pt
.y
);
1004 expect(mchit
.uHit
, res
);
1005 expect_hex(MCHT_CALENDARDATE
, res
);
1007 /* in active area - day of the week #2 */
1008 mchit
.pt
.x
= r
.right
/ 14; /* half of first day rect */
1009 mchit
.pt
.y
= r
.bottom
/ 2;
1010 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
) & mchit
);
1011 expect(r
.right
/ 14, mchit
.pt
.x
);
1012 expect(r
.bottom
/ 2, mchit
.pt
.y
);
1013 expect(mchit
.uHit
, res
);
1014 expect_hex(MCHT_CALENDARDATE
, res
);
1016 /* in active area - date from prev month */
1017 mchit
.pt
.x
= r
.right
/ 14; /* half of first day rect */
1018 mchit
.pt
.y
= 6 * r
.bottom
/ 19;
1019 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
) & mchit
);
1020 expect(r
.right
/ 14, mchit
.pt
.x
);
1021 expect(6 * r
.bottom
/ 19, mchit
.pt
.y
);
1022 expect(mchit
.uHit
, res
);
1023 expect_hex(MCHT_CALENDARDATEPREV
, res
);
1026 /* (125, 115) is in active area - date from this month */
1029 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
) & mchit
);
1030 expect(125, mchit
.pt
.x
);
1031 expect(115, mchit
.pt
.y
);
1032 expect(mchit
.uHit
, res
);
1033 expect(MCHT_CALENDARDATE
, res
);
1036 /* in active area - date from next month */
1037 mchit
.pt
.x
= 11 * r
.right
/ 14;
1038 mchit
.pt
.y
= 16 * r
.bottom
/ 19;
1039 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
) & mchit
);
1040 expect(11 * r
.right
/ 14, mchit
.pt
.x
);
1041 expect(16 * r
.bottom
/ 19, mchit
.pt
.y
);
1042 expect(mchit
.uHit
, res
);
1043 expect_hex(MCHT_CALENDARDATENEXT
, res
);
1045 /* in active area - today link */
1046 mchit
.pt
.x
= r
.right
/ 14;
1047 mchit
.pt
.y
= 18 * r
.bottom
/ 19;
1048 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
) & mchit
);
1049 expect(r
.right
/ 14, mchit
.pt
.x
);
1050 expect(18 * r
.bottom
/ 19, mchit
.pt
.y
);
1051 expect(mchit
.uHit
, res
);
1052 expect_hex(MCHT_TODAYLINK
, res
);
1054 /* in active area - today link */
1055 mchit
.pt
.x
= r
.right
/ 2;
1056 mchit
.pt
.y
= 18 * r
.bottom
/ 19;
1057 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
) & mchit
);
1058 expect(r
.right
/ 2, mchit
.pt
.x
);
1059 expect(18 * r
.bottom
/ 19, mchit
.pt
.y
);
1060 expect(mchit
.uHit
, res
);
1061 expect_hex(MCHT_TODAYLINK
, res
);
1063 /* in active area - today link */
1064 mchit
.pt
.x
= r
.right
/ 10;
1065 mchit
.pt
.y
= 18 * r
.bottom
/ 19;
1066 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
) & mchit
);
1067 expect(r
.right
/ 10, mchit
.pt
.x
);
1068 expect(18 * r
.bottom
/ 19, mchit
.pt
.y
);
1069 expect(mchit
.uHit
, res
);
1070 expect_hex(MCHT_TODAYLINK
, res
);
1072 ok_sequence(sequences
, MONTHCAL_SEQ_INDEX
, monthcal_hit_test_seq
, "monthcal hit test", TRUE
);
1074 /* The horizontal position of title bar elements depends on locale (y pos
1075 is constant), so we sample across a horizontal line and make sure we
1076 find all elements. */
1078 /* Get the format of the title */
1079 GetLocaleInfo(LOCALE_USER_DEFAULT
, LOCALE_SYEARMONTH
, yearmonth
, 80);
1080 /* Find out if we have a month and/or year */
1081 locale_year
= strstr(yearmonth
, "y");
1082 locale_month
= strstr(yearmonth
, "M");
1085 mchit
.pt
.y
= (5/2) * r
.bottom
/ 19;
1087 old_res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
) & mchit
);
1088 expect_hex(title_hits
[title_index
].ht
, old_res
);
1090 in_the_middle
= FALSE
;
1091 month_count
= year_count
= 0;
1092 for (x
= 0; x
< r
.right
; x
++){
1094 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
) & mchit
);
1095 expect(x
, mchit
.pt
.x
);
1096 expect((5/2) * r
.bottom
/ 19, mchit
.pt
.y
);
1097 expect(mchit
.uHit
, res
);
1098 if (res
!= old_res
) {
1100 if (old_res
== MCHT_TITLEBTNPREV
)
1101 in_the_middle
= TRUE
;
1103 if (res
== MCHT_TITLEBTNNEXT
)
1104 in_the_middle
= FALSE
;
1106 if (in_the_middle
) {
1107 if (res
== MCHT_TITLEMONTH
)
1109 else if (res
== MCHT_TITLEYEAR
)
1114 if (sizeof(title_hits
) / sizeof(title_hits
[0]) <= title_index
)
1117 if (title_hits
[title_index
].todo
) {
1119 ok(title_hits
[title_index
].ht
== res
, "Expected %x, got %x, pos %d\n",
1120 title_hits
[title_index
].ht
, res
, x
);
1122 ok(title_hits
[title_index
].ht
== res
, "Expected %x, got %x, pos %d\n",
1123 title_hits
[title_index
].ht
, res
, x
);
1130 /* There are some limits, even if LOCALE_SYEARMONTH contains rubbish
1131 * or no month/year indicators at all */
1133 todo_wine
ok(month_count
== 1, "Expected 1 month item, got %d\n", month_count
);
1135 ok(month_count
<= 1, "Too many month items: %d\n", month_count
);
1138 todo_wine
ok(year_count
== 1, "Expected 1 year item, got %d\n", year_count
);
1140 ok(year_count
<= 1, "Too many year items: %d\n", year_count
);
1142 todo_wine
ok(month_count
+ year_count
>= 1, "Not enough month and year items\n");
1144 ok(r
.right
<= x
&& title_index
+ 1 == sizeof(title_hits
) / sizeof(title_hits
[0]),
1145 "Wrong title layout\n");
1147 DestroyWindow(hwnd
);
1150 static void test_todaylink(void)
1152 MCHITTESTINFO mchit
;
1153 SYSTEMTIME st_test
, st_new
;
1158 memset(&mchit
, 0, sizeof(MCHITTESTINFO
));
1160 hwnd
= create_monthcal_control(0);
1162 res
= SendMessage(hwnd
, MCM_GETMINREQRECT
, 0, (LPARAM
)&r
);
1164 MoveWindow(hwnd
, 0, 0, r
.right
, r
.bottom
, FALSE
);
1166 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
1168 /* hit active area - today link */
1169 mchit
.cbSize
= MCHITTESTINFO_V1_SIZE
;
1170 mchit
.pt
.x
= r
.right
/ 14;
1171 mchit
.pt
.y
= 18 * r
.bottom
/ 19;
1172 res
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
) & mchit
);
1173 expect(r
.right
/ 14, mchit
.pt
.x
);
1174 expect(18 * r
.bottom
/ 19, mchit
.pt
.y
);
1175 expect(mchit
.uHit
, res
);
1176 expect(MCHT_TODAYLINK
, res
);
1180 st_test
.wYear
= 2005;
1182 res
= SendMessage(hwnd
, MCM_SETTODAY
, 0, (LPARAM
)&st_test
);
1185 memset(&st_new
, 0, sizeof(st_new
));
1186 res
= SendMessage(hwnd
, MCM_GETTODAY
, 0, (LPARAM
)&st_new
);
1188 expect(1, st_new
.wDay
);
1189 expect(1, st_new
.wMonth
);
1190 expect(2005, st_new
.wYear
);
1192 res
= SendMessage(hwnd
, WM_LBUTTONDOWN
, MK_LBUTTON
, MAKELONG(mchit
.pt
.x
, mchit
.pt
.y
));
1195 memset(&st_new
, 0, sizeof(st_new
));
1196 res
= SendMessage(hwnd
, MCM_GETCURSEL
, 0, (LPARAM
)&st_new
);
1198 expect(1, st_new
.wDay
);
1199 expect(1, st_new
.wMonth
);
1200 expect(2005, st_new
.wYear
);
1202 ok_sequence(sequences
, MONTHCAL_SEQ_INDEX
, monthcal_todaylink_seq
, "monthcal hit test", TRUE
);
1204 DestroyWindow(hwnd
);
1207 static void test_today(void)
1209 SYSTEMTIME st_test
, st_new
;
1213 hwnd
= create_monthcal_control(0);
1215 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
1217 /* Setter and Getters for "today" information */
1219 /* check for overflow, should be ok */
1220 memset(&st_test
, 0, sizeof(st_test
));
1222 st_test
.wMonth
= 38;
1227 res
= SendMessage(hwnd
, MCM_SETTODAY
, 0, (LPARAM
)&st_test
);
1230 res
= SendMessage(hwnd
, MCM_GETTODAY
, 0, (LPARAM
)&st_new
);
1233 /* st_test should not change */
1234 expect(38, st_test
.wDay
);
1235 expect(38, st_test
.wMonth
);
1237 /* st_new should change, overflow does not matter */
1238 expect(38, st_new
.wDay
);
1239 expect(38, st_new
.wMonth
);
1241 /* check for zero, should be ok*/
1245 res
= SendMessage(hwnd
, MCM_SETTODAY
, 0, (LPARAM
)&st_test
);
1248 res
= SendMessage(hwnd
, MCM_GETTODAY
, 0, (LPARAM
)&st_new
);
1251 /* st_test should not change */
1252 expect(0, st_test
.wDay
);
1253 expect(0, st_test
.wMonth
);
1255 /* st_new should change to zero*/
1256 expect(0, st_new
.wDay
);
1257 expect(0, st_new
.wMonth
);
1259 ok_sequence(sequences
, MONTHCAL_SEQ_INDEX
, monthcal_today_seq
, "monthcal today", TRUE
);
1261 DestroyWindow(hwnd
);
1264 static void test_scroll(void)
1269 hwnd
= create_monthcal_control(0);
1271 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
1273 /* Setter and Getters for scroll rate */
1274 res
= SendMessage(hwnd
, MCM_SETMONTHDELTA
, 2, 0);
1277 res
= SendMessage(hwnd
, MCM_SETMONTHDELTA
, 3, 0);
1279 res
= SendMessage(hwnd
, MCM_GETMONTHDELTA
, 0, 0);
1282 res
= SendMessage(hwnd
, MCM_SETMONTHDELTA
, 12, 0);
1284 res
= SendMessage(hwnd
, MCM_GETMONTHDELTA
, 0, 0);
1287 res
= SendMessage(hwnd
, MCM_SETMONTHDELTA
, 15, 0);
1289 res
= SendMessage(hwnd
, MCM_GETMONTHDELTA
, 0, 0);
1292 res
= SendMessage(hwnd
, MCM_SETMONTHDELTA
, -5, 0);
1294 res
= SendMessage(hwnd
, MCM_GETMONTHDELTA
, 0, 0);
1297 ok_sequence(sequences
, MONTHCAL_SEQ_INDEX
, monthcal_scroll_seq
, "monthcal scroll", FALSE
);
1299 DestroyWindow(hwnd
);
1302 static void test_monthrange(void)
1305 SYSTEMTIME st_visible
[2], st_daystate
[2], st
;
1309 hwnd
= create_monthcal_control(0);
1311 st_visible
[0].wYear
= 0;
1312 st_visible
[0].wMonth
= 0;
1313 st_visible
[0].wDay
= 0;
1314 st_daystate
[1] = st_daystate
[0] = st_visible
[1] = st_visible
[0];
1322 st
.wMilliseconds
= 0;
1325 res
= SendMessage(hwnd
, MCM_SETCURSEL
, 0, (LPARAM
)&st
);
1328 /* to be locale independent */
1329 SendMessage(hwnd
, MCM_SETFIRSTDAYOFWEEK
, 0, (LPARAM
)6);
1331 res
= SendMessage(hwnd
, MCM_GETMINREQRECT
, 0, (LPARAM
)&r
);
1333 /* resize control to display two Calendars */
1334 MoveWindow(hwnd
, 0, 0, r
.right
, (5/2)*r
.bottom
, FALSE
);
1336 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
1338 res
= SendMessage(hwnd
, MCM_GETMONTHRANGE
, GMR_VISIBLE
, (LPARAM
)st_visible
);
1340 expect(2000, st_visible
[0].wYear
);
1341 expect(11, st_visible
[0].wMonth
);
1342 expect(1, st_visible
[0].wDay
);
1343 expect(2000, st_visible
[1].wYear
);
1344 expect(12, st_visible
[1].wMonth
);
1345 expect(31, st_visible
[1].wDay
);
1347 res
= SendMessage(hwnd
, MCM_GETMONTHRANGE
, GMR_DAYSTATE
, (LPARAM
)st_daystate
);
1349 expect(2000, st_daystate
[0].wYear
);
1350 expect(10, st_daystate
[0].wMonth
);
1351 expect(29, st_daystate
[0].wDay
);
1352 expect(2001, st_daystate
[1].wYear
);
1353 expect(1, st_daystate
[1].wMonth
);
1354 expect(6, st_daystate
[1].wDay
);
1356 ok_sequence(sequences
, MONTHCAL_SEQ_INDEX
, monthcal_monthrange_seq
, "monthcal monthrange", FALSE
);
1358 /* with null date array parameter */
1359 res
= SendMessage(hwnd
, MCM_GETMONTHRANGE
, GMR_VISIBLE
, 0);
1362 res
= SendMessage(hwnd
, MCM_GETMONTHRANGE
, GMR_DAYSTATE
, 0);
1365 /* resize control to display single Calendar */
1366 MoveWindow(hwnd
, 0, 0, r
.right
, r
.bottom
, FALSE
);
1368 memset(&st
, 0, sizeof(st
));
1373 res
= SendMessage(hwnd
, MCM_SETCURSEL
, 0, (LPARAM
)&st
);
1376 /* September 1752 has 19 days */
1377 res
= SendMessage(hwnd
, MCM_GETMONTHRANGE
, GMR_VISIBLE
, (LPARAM
)st_visible
);
1380 expect(1752, st_visible
[0].wYear
);
1381 expect(9, st_visible
[0].wMonth
);
1382 ok(14 == st_visible
[0].wDay
||
1383 broken(1 == st_visible
[0].wDay
), /* comctl32 <= 4.72 */
1384 "Expected 14, got %d\n", st_visible
[0].wDay
);
1386 expect(1752, st_visible
[1].wYear
);
1387 expect(9, st_visible
[1].wMonth
);
1388 expect(19, st_visible
[1].wDay
);
1390 DestroyWindow(hwnd
);
1393 static void test_maxselday(void)
1399 hwnd
= create_monthcal_control(0);
1400 /* if no style specified default to 1 */
1401 res
= SendMessage(hwnd
, MCM_GETMAXSELCOUNT
, 0, 0);
1403 res
= SendMessage(hwnd
, MCM_SETMAXSELCOUNT
, 5, 0);
1405 res
= SendMessage(hwnd
, MCM_GETMAXSELCOUNT
, 0, 0);
1408 /* try to set style */
1409 style
= GetWindowLong(hwnd
, GWL_STYLE
);
1410 SetWindowLong(hwnd
, GWL_STYLE
, style
| MCS_MULTISELECT
);
1411 style
= GetWindowLong(hwnd
, GWL_STYLE
);
1412 ok(!(style
& MCS_MULTISELECT
), "Expected MCS_MULTISELECT not to be set\n");
1413 DestroyWindow(hwnd
);
1415 hwnd
= create_monthcal_control(MCS_MULTISELECT
);
1416 /* try to remove style */
1417 style
= GetWindowLong(hwnd
, GWL_STYLE
);
1418 SetWindowLong(hwnd
, GWL_STYLE
, style
& ~MCS_MULTISELECT
);
1419 style
= GetWindowLong(hwnd
, GWL_STYLE
);
1420 ok(style
& MCS_MULTISELECT
, "Expected MCS_MULTISELECT to be set\n");
1421 DestroyWindow(hwnd
);
1423 hwnd
= create_monthcal_control(MCS_MULTISELECT
);
1425 /* default width is a week */
1426 res
= SendMessage(hwnd
, MCM_GETMAXSELCOUNT
, 0, 0);
1429 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
1431 /* Setter and Getters for max selected days */
1432 res
= SendMessage(hwnd
, MCM_SETMAXSELCOUNT
, 5, 0);
1434 res
= SendMessage(hwnd
, MCM_GETMAXSELCOUNT
, 0, 0);
1437 res
= SendMessage(hwnd
, MCM_SETMAXSELCOUNT
, 15, 0);
1439 res
= SendMessage(hwnd
, MCM_GETMAXSELCOUNT
, 0, 0);
1442 /* test invalid value */
1443 res
= SendMessage(hwnd
, MCM_SETMAXSELCOUNT
, -1, 0);
1445 res
= SendMessage(hwnd
, MCM_GETMAXSELCOUNT
, 0, 0);
1448 ok_sequence(sequences
, MONTHCAL_SEQ_INDEX
, monthcal_max_sel_day_seq
, "monthcal MaxSelDay", FALSE
);
1450 /* zero value is invalid too */
1451 res
= SendMessage(hwnd
, MCM_SETMAXSELCOUNT
, 0, 0);
1453 res
= SendMessage(hwnd
, MCM_GETMAXSELCOUNT
, 0, 0);
1456 DestroyWindow(hwnd
);
1459 static void test_size(void)
1463 HFONT hFont1
, hFont2
;
1467 hwnd
= create_monthcal_control(0);
1469 lstrcpyA(logfont
.lfFaceName
, "Arial");
1470 memset(&logfont
, 0, sizeof(logfont
));
1471 logfont
.lfHeight
= 12;
1472 hFont1
= CreateFontIndirectA(&logfont
);
1474 logfont
.lfHeight
= 24;
1475 hFont2
= CreateFontIndirectA(&logfont
);
1477 /* initialize to a font we can compare against */
1478 SendMessage(hwnd
, WM_SETFONT
, (WPARAM
)hFont1
, 0);
1479 res
= SendMessage(hwnd
, MCM_GETMINREQRECT
, 0, (LPARAM
)&r1
);
1480 ok(res
, "SendMessage(MCM_GETMINREQRECT) failed\n");
1482 /* check that setting a larger font results in an larger rect */
1483 SendMessage(hwnd
, WM_SETFONT
, (WPARAM
)hFont2
, 0);
1484 res
= SendMessage(hwnd
, MCM_GETMINREQRECT
, 0, (LPARAM
)&r2
);
1485 ok(res
, "SendMessage(MCM_GETMINREQRECT) failed\n");
1487 OffsetRect(&r1
, -r1
.left
, -r1
.top
);
1488 OffsetRect(&r2
, -r2
.left
, -r2
.top
);
1490 ok(r1
.bottom
< r2
.bottom
, "Failed to get larger rect with larger font\n");
1492 DestroyWindow(hwnd
);
1495 static void test_create(void)
1499 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
1501 hwnd
= create_monthcal_control(0);
1502 ok_sequence(sequences
, PARENT_SEQ_INDEX
, create_monthcal_control_seq
, "create monthcal control", TRUE
);
1504 DestroyWindow(hwnd
);
1506 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
1507 hwnd
= create_monthcal_control(MCS_MULTISELECT
);
1508 ok_sequence(sequences
, PARENT_SEQ_INDEX
, create_monthcal_multi_sel_style_seq
, "create monthcal (multi sel style)", TRUE
);
1509 DestroyWindow(hwnd
);
1512 static void test_destroy(void)
1516 hwnd
= create_monthcal_control(0);
1517 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
1518 DestroyWindow(hwnd
);
1519 ok_sequence(sequences
, PARENT_SEQ_INDEX
, destroy_monthcal_parent_msgs_seq
, "Destroy monthcal (parent msg)", FALSE
);
1520 ok_sequence(sequences
, MONTHCAL_SEQ_INDEX
, destroy_monthcal_child_msgs_seq
, "Destroy monthcal (child msg)", FALSE
);
1522 /* MCS_MULTISELECT */
1523 hwnd
= create_monthcal_control(MCS_MULTISELECT
);
1524 flush_sequences(sequences
, NUM_MSG_SEQUENCES
);
1525 DestroyWindow(hwnd
);
1526 ok_sequence(sequences
, MONTHCAL_SEQ_INDEX
, destroy_monthcal_multi_sel_style_seq
, "Destroy monthcal (multi sel style)", FALSE
);
1529 static void test_selrange(void)
1532 SYSTEMTIME st
, range
[2], range2
[2];
1533 BOOL ret
, old_comctl32
= FALSE
;
1535 hwnd
= create_monthcal_control(MCS_MULTISELECT
);
1537 /* just after creation selection should start and end today */
1538 ret
= SendMessage(hwnd
, MCM_GETTODAY
, 0, (LPARAM
)&st
);
1541 memset(range
, 0xcc, sizeof(range
));
1542 ret
= SendMessage(hwnd
, MCM_GETSELRANGE
, 0, (LPARAM
)range
);
1544 expect(st
.wYear
, range
[0].wYear
);
1545 expect(st
.wMonth
, range
[0].wMonth
);
1546 expect(st
.wDay
, range
[0].wDay
);
1547 if (range
[0].wDayOfWeek
!= st
.wDayOfWeek
)
1549 win_skip("comctl32 <= 4.70 doesn't set some values\n");
1550 old_comctl32
= TRUE
;
1554 expect(st
.wDayOfWeek
, range
[0].wDayOfWeek
);
1555 expect(st
.wHour
, range
[0].wHour
);
1556 expect(st
.wMinute
, range
[0].wMinute
);
1557 expect(st
.wSecond
, range
[0].wSecond
);
1558 expect(st
.wMilliseconds
, range
[0].wMilliseconds
);
1561 expect(st
.wYear
, range
[1].wYear
);
1562 expect(st
.wMonth
, range
[1].wMonth
);
1563 expect(st
.wDay
, range
[1].wDay
);
1566 expect(st
.wDayOfWeek
, range
[1].wDayOfWeek
);
1567 expect(st
.wHour
, range
[1].wHour
);
1568 expect(st
.wMinute
, range
[1].wMinute
);
1569 expect(st
.wSecond
, range
[1].wSecond
);
1570 expect(st
.wMilliseconds
, range
[1].wMilliseconds
);
1573 /* bounds are swapped if min > max */
1574 memset(&range
[0], 0, sizeof(range
[0]));
1575 range
[0].wYear
= 2009;
1576 range
[0].wMonth
= 10;
1578 range
[1] = range
[0];
1581 ret
= SendMessage(hwnd
, MCM_SETSELRANGE
, 0, (LPARAM
)range
);
1584 ret
= SendMessage(hwnd
, MCM_GETSELRANGE
, 0, (LPARAM
)range2
);
1587 expect(range
[1].wYear
, range2
[0].wYear
);
1588 expect(range
[1].wMonth
, range2
[0].wMonth
);
1589 expect(range
[1].wDay
, range2
[0].wDay
);
1590 expect(6, range2
[0].wDayOfWeek
);
1591 expect(range
[1].wHour
, range2
[0].wHour
);
1592 expect(range
[1].wMinute
, range2
[0].wMinute
);
1593 expect(range
[1].wSecond
, range2
[0].wSecond
);
1594 expect(range
[1].wMilliseconds
, range2
[0].wMilliseconds
);
1596 expect(range
[0].wYear
, range2
[1].wYear
);
1597 expect(range
[0].wMonth
, range2
[1].wMonth
);
1598 expect(range
[0].wDay
, range2
[1].wDay
);
1599 expect(1, range2
[1].wDayOfWeek
);
1600 expect(range
[0].wHour
, range2
[1].wHour
);
1601 expect(range
[0].wMinute
, range2
[1].wMinute
);
1602 expect(range
[0].wSecond
, range2
[1].wSecond
);
1603 expect(range
[0].wMilliseconds
, range2
[1].wMilliseconds
);
1605 /* try with range larger than maximum configured */
1606 memset(&range
[0], 0, sizeof(range
[0]));
1607 range
[0].wYear
= 2009;
1608 range
[0].wMonth
= 10;
1610 range
[1] = range
[0];
1612 ret
= SendMessage(hwnd
, MCM_SETSELRANGE
, 0, (LPARAM
)range
);
1615 range
[1] = range
[0];
1616 /* default max. range is 7 days */
1619 ret
= SendMessage(hwnd
, MCM_SETSELRANGE
, 0, (LPARAM
)range
);
1622 ret
= SendMessage(hwnd
, MCM_GETSELRANGE
, 0, (LPARAM
)range2
);
1625 expect(range
[0].wYear
, range2
[0].wYear
);
1626 expect(range
[0].wMonth
, range2
[0].wMonth
);
1627 expect(range
[0].wDay
, range2
[0].wDay
);
1628 expect(range
[0].wYear
, range2
[1].wYear
);
1629 expect(range
[0].wMonth
, range2
[1].wMonth
);
1630 expect(range
[0].wDay
, range2
[1].wDay
);
1632 DestroyWindow(hwnd
);
1635 static void test_killfocus(void)
1640 hwnd
= create_monthcal_control(0);
1642 /* make parent invisible */
1643 style
= GetWindowLong(parent_wnd
, GWL_STYLE
);
1644 SetWindowLong(parent_wnd
, GWL_STYLE
, style
&~ WS_VISIBLE
);
1646 SendMessage(hwnd
, WM_KILLFOCUS
, (WPARAM
)GetDesktopWindow(), 0);
1648 style
= GetWindowLong(hwnd
, GWL_STYLE
);
1649 ok(style
& WS_VISIBLE
, "Expected WS_VISIBLE to be set\n");
1651 style
= GetWindowLong(parent_wnd
, GWL_STYLE
);
1652 SetWindowLong(parent_wnd
, GWL_STYLE
, style
| WS_VISIBLE
);
1654 DestroyWindow(hwnd
);
1657 static void test_hittest_v6(void)
1659 MCHITTESTINFO mchit
;
1664 hwnd
= create_monthcal_control(0);
1665 SendMessage(hwnd
, MCM_SETCALENDARBORDER
, TRUE
, 0);
1667 SendMessage(hwnd
, MCM_GETMINREQRECT
, 0, (LPARAM
)&r
);
1668 /* reserving some area around calendar */
1669 MoveWindow(hwnd
, 0, 0, r
.right
* 3 / 2, r
.bottom
* 3 / 2, FALSE
);
1670 mchit
.cbSize
= sizeof(MCHITTESTINFO
);
1671 mchit
.pt
.x
= mchit
.pt
.y
= 0;
1675 ret
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
)&mchit
);
1678 win_skip("Only MCHITTESTINFO_V1 supported\n");
1679 DestroyWindow(hwnd
);
1682 todo_wine
expect_hex(MCHT_NOWHERE
, ret
);
1683 expect(-1, mchit
.iOffset
);
1684 expect(-1, mchit
.iRow
);
1685 expect(-1, mchit
.iCol
);
1687 MoveWindow(hwnd
, 0, 0, r
.right
, r
.bottom
, FALSE
);
1688 mchit
.pt
.x
= r
.right
/ 2;
1689 mchit
.pt
.y
= r
.bottom
/ 2;
1691 ret
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
)&mchit
);
1692 expect_hex(MCHT_CALENDARDATE
, ret
);
1693 expect(0, mchit
.iOffset
);
1696 mchit
.pt
.x
= r
.right
/ (7*2);
1697 mchit
.pt
.y
= r
.bottom
/ 2;
1699 mchit
.iCol
= mchit
.iRow
= -1;
1701 mchit
.rc
.left
= mchit
.rc
.right
= mchit
.rc
.top
= mchit
.rc
.bottom
= -1;
1702 ret
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
)&mchit
);
1703 expect_hex(MCHT_CALENDARDATE
, ret
);
1704 expect_hex(MCHT_CALENDARDATE
, mchit
.uHit
);
1705 expect(0, mchit
.iOffset
);
1706 expect(2, mchit
.iRow
);
1707 expect(0, mchit
.iCol
);
1708 /* returned a one day rectangle */
1709 expect_d(r
.right
/ 7, mchit
.rc
.right
- mchit
.rc
.left
);
1710 expect_d(r
.bottom
/ 10, mchit
.rc
.bottom
- mchit
.rc
.top
);
1716 mchit
.iCol
= mchit
.iRow
= -1;
1718 mchit
.rc
.left
= mchit
.rc
.right
= mchit
.rc
.top
= mchit
.rc
.bottom
= -1;
1719 ret
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
)&mchit
);
1720 expect_hex(MCHT_TITLE
, ret
);
1721 expect_hex(MCHT_TITLE
, mchit
.uHit
);
1722 expect(0, mchit
.iOffset
);
1723 expect(-1, mchit
.iRow
);
1724 expect(-1, mchit
.iCol
);
1725 expect(0, mchit
.rc
.left
);
1726 expect(0, mchit
.rc
.top
);
1727 expect_d(r
.right
, mchit
.rc
.right
);
1728 ok(mchit
.rc
.bottom
> 0, "got %d\n", mchit
.rc
.bottom
);
1730 /* between two calendars */
1731 MoveWindow(hwnd
, 0, 0, r
.right
* 5/2, r
.bottom
, FALSE
);
1732 mchit
.pt
.x
= r
.right
/ (5*4);
1733 mchit
.pt
.y
= r
.bottom
/ 2;
1735 mchit
.iCol
= mchit
.iRow
= -2;
1737 mchit
.rc
.left
= mchit
.rc
.right
= mchit
.rc
.top
= mchit
.rc
.bottom
= -1;
1738 ret
= SendMessage(hwnd
, MCM_HITTEST
, 0, (LPARAM
)&mchit
);
1739 todo_wine
expect_hex(MCHT_NOWHERE
, ret
);
1740 todo_wine
expect_hex(MCHT_NOWHERE
, mchit
.uHit
);
1741 expect(-2, mchit
.iOffset
);
1742 expect(-2, mchit
.iRow
);
1743 expect(-2, mchit
.iCol
);
1744 todo_wine
expect(0, mchit
.rc
.left
);
1745 todo_wine
expect(0, mchit
.rc
.top
);
1746 todo_wine
expect_d(r
.right
* 5/2, mchit
.rc
.right
);
1747 todo_wine
expect_d(r
.bottom
, mchit
.rc
.bottom
);
1749 DestroyWindow(hwnd
);
1752 static void test_get_set_border(void)
1757 hwnd
= create_monthcal_control(0);
1759 /* a non-default value */
1760 ret
= SendMessage(hwnd
, MCM_SETCALENDARBORDER
, TRUE
, 10);
1763 ret
= SendMessage(hwnd
, MCM_GETCALENDARBORDER
, 0, 0);
1767 skip("MCM_GET/SETCALENDARBORDER not supported\n");
1768 DestroyWindow(hwnd
);
1774 DestroyWindow(hwnd
);
1777 static void test_MCM_SIZERECTTOMIN(void)
1783 hwnd
= create_monthcal_control(0);
1785 ret
= SendMessageA(hwnd
, MCM_GETMINREQRECT
, 0, (LPARAM
)&r2
);
1788 win_skip("Message MCM_GETMINREQRECT unsupported. Skipping.\n");
1789 DestroyWindow(hwnd
);
1793 ret
= SendMessageA(hwnd
, MCM_SIZERECTTOMIN
, 0, 0);
1794 ok(ret
== 0, "got %d\n", ret
);
1796 r
.left
= r
.right
= r
.top
= r
.bottom
= 0;
1797 ret
= SendMessageA(hwnd
, MCM_SIZERECTTOMIN
, 0, (LPARAM
)&r
);
1800 skip("Message MCM_SIZERECTTOMIN unsupported. Skipping.\n");
1801 DestroyWindow(hwnd
);
1804 ok(ret
== 1, "got %d\n", ret
);
1805 ok(r
.left
== 0 && r
.right
> 0, "got %d, %d\n", r
.left
, r
.right
);
1808 ret
= SendMessageA(hwnd
, MCM_SIZERECTTOMIN
, 0, (LPARAM
)&r
);
1809 ok(ret
== 1, "got %d\n", ret
);
1811 r2
.right
= (r2
.right
- r2
.left
) * 3;
1812 r2
.bottom
= (r2
.bottom
- r2
.top
) * 3;
1813 r2
.left
= r2
.top
= 0;
1814 ret
= SendMessageA(hwnd
, MCM_SIZERECTTOMIN
, 0, (LPARAM
)&r2
);
1815 ok(ret
== 1, "got %d\n", ret
);
1817 DestroyWindow(hwnd
);
1820 static void test_MCM_GETCALENDARCOUNT(void)
1825 hwnd
= create_monthcal_control(0);
1827 ret
= SendMessageA(hwnd
, MCM_GETCALENDARCOUNT
, 0, 0);
1830 win_skip("Message MCM_GETCALENDARCOUNT unsupported. Skipping.\n");
1831 DestroyWindow(hwnd
);
1837 DestroyWindow(hwnd
);
1840 static void test_daystate(void)
1842 MONTHDAYSTATE state
[4];
1846 /* without MCS_DAYSTATE */
1847 hwnd
= create_monthcal_control(0);
1849 ret
= SendMessageA(hwnd
, MCM_GETMONTHRANGE
, GMR_DAYSTATE
, 0);
1852 ret
= SendMessageA(hwnd
, MCM_SETDAYSTATE
, 4, (LPARAM
)&state
);
1855 ret
= SendMessageA(hwnd
, MCM_SETDAYSTATE
, 2, (LPARAM
)&state
);
1858 ret
= SendMessageA(hwnd
, MCM_SETDAYSTATE
, 0, 0);
1861 /* try to switch on */
1862 SetWindowLongA(hwnd
, GWL_STYLE
, GetWindowLongA(hwnd
, GWL_STYLE
) | MCS_DAYSTATE
);
1863 style
= GetWindowLongA(hwnd
, GWL_STYLE
);
1864 ok((style
& MCS_DAYSTATE
) == 0, "got 0x%08x\n", style
);
1866 DestroyWindow(hwnd
);
1868 /* with MCS_DAYSTATE */
1869 hwnd
= create_monthcal_control(MCS_DAYSTATE
);
1871 ret
= SendMessageA(hwnd
, MCM_GETMONTHRANGE
, GMR_DAYSTATE
, 0);
1874 ret
= SendMessageA(hwnd
, MCM_SETDAYSTATE
, 4, (LPARAM
)&state
);
1877 ret
= SendMessageA(hwnd
, MCM_SETDAYSTATE
, 2, (LPARAM
)&state
);
1880 ret
= SendMessageA(hwnd
, MCM_SETDAYSTATE
, 0, 0);
1883 /* try to switch off */
1884 SetWindowLongA(hwnd
, GWL_STYLE
, GetWindowLongA(hwnd
, GWL_STYLE
) & ~MCS_DAYSTATE
);
1885 style
= GetWindowLongA(hwnd
, GWL_STYLE
);
1886 ok((style
& MCS_DAYSTATE
) == MCS_DAYSTATE
, "got 0x%08x\n", style
);
1888 DestroyWindow(hwnd
);
1891 START_TEST(monthcal
)
1893 BOOL (WINAPI
*pInitCommonControlsEx
)(const INITCOMMONCONTROLSEX
*);
1894 INITCOMMONCONTROLSEX iccex
;
1898 ULONG_PTR ctx_cookie
;
1901 hComctl32
= GetModuleHandleA("comctl32.dll");
1902 pInitCommonControlsEx
= (void*)GetProcAddress(hComctl32
, "InitCommonControlsEx");
1903 if (!pInitCommonControlsEx
)
1905 skip("InitCommonControlsEx() is missing. Skipping the tests\n");
1908 iccex
.dwSize
= sizeof(iccex
);
1909 iccex
.dwICC
= ICC_DATE_CLASSES
;
1910 pInitCommonControlsEx(&iccex
);
1914 init_msg_sequences(sequences
, NUM_MSG_SEQUENCES
);
1916 parent_wnd
= create_parent_window();
1935 if (!load_v6_module(&ctx_cookie
, &hCtx
))
1937 DestroyWindow(parent_wnd
);
1941 /* this is a XP SP3 failure workaround */
1942 hwnd
= CreateWindowExA(0, MONTHCAL_CLASSA
, "foo",
1943 WS_CHILD
| WS_BORDER
| WS_VISIBLE
,
1945 parent_wnd
, NULL
, GetModuleHandleA(NULL
), NULL
);
1946 if (!IsWindow(hwnd
))
1948 win_skip("FIXME: failed to create Monthcal window.\n");
1949 unload_v6_module(ctx_cookie
, hCtx
);
1950 DestroyWindow(parent_wnd
);
1954 DestroyWindow(hwnd
);
1957 test_get_set_border();
1958 test_MCM_SIZERECTTOMIN();
1959 test_MCM_GETCALENDARCOUNT();
1961 unload_v6_module(ctx_cookie
, hCtx
);
1963 DestroyWindow(parent_wnd
);