ntoskrnl.exe: Add a stub implementation of KeInitializeTimer.
[wine] / dlls / comctl32 / monthcal.c
1 /* Month calendar control
2
3  *
4  * Copyright 1998, 1999 Eric Kohl (ekohl@abo.rhein-zeitung.de)
5  * Copyright 1999 Alex Priem (alexp@sci.kun.nl)
6  * Copyright 1999 Chris Morgan <cmorgan@wpi.edu> and
7  *                James Abbatiello <abbeyj@wpi.edu>
8  * Copyright 2000 Uwe Bonnes <bon@elektron.ikp.physik.tu-darmstadt.de>
9  *
10  * This library is free software; you can redistribute it and/or
11  * modify it under the terms of the GNU Lesser General Public
12  * License as published by the Free Software Foundation; either
13  * version 2.1 of the License, or (at your option) any later version.
14  *
15  * This library is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
18  * Lesser General Public License for more details.
19  *
20  * You should have received a copy of the GNU Lesser General Public
21  * License along with this library; if not, write to the Free Software
22  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
23  *
24  * NOTE
25  * 
26  * This code was audited for completeness against the documented features
27  * of Comctl32.dll version 6.0 on Oct. 20, 2004, by Dimitrie O. Paun.
28  * 
29  * Unless otherwise noted, we believe this code to be complete, as per
30  * the specification mentioned above.
31  * If you discover missing features, or bugs, please note them below.
32  * 
33  * TODO:
34  *    -- MCM_[GS]ETUNICODEFORMAT
35  *    -- MONTHCAL_GetMonthRange
36  *    -- handle resources better (doesn't work now); 
37  *    -- take care of internationalization.
38  *    -- keyboard handling.
39  *    -- GetRange: At the moment, we copy ranges anyway, regardless of
40  *                 infoPtr->rangeValid; an invalid range is simply filled 
41  *                 with zeros in SetRange.  Is this the right behavior?
42  *    -- search for FIXME
43  */
44
45 #include <math.h>
46 #include <stdarg.h>
47 #include <stdio.h>
48 #include <stdlib.h>
49 #include <string.h>
50
51 #include "windef.h"
52 #include "winbase.h"
53 #include "wingdi.h"
54 #include "winuser.h"
55 #include "winnls.h"
56 #include "commctrl.h"
57 #include "comctl32.h"
58 #include "uxtheme.h"
59 #include "tmschema.h"
60 #include "wine/unicode.h"
61 #include "wine/debug.h"
62
63 WINE_DEFAULT_DEBUG_CHANNEL(monthcal);
64
65 #define MC_SEL_LBUTUP       1   /* Left button released */
66 #define MC_SEL_LBUTDOWN     2   /* Left button pressed in calendar */
67 #define MC_PREVPRESSED      4   /* Prev month button pressed */
68 #define MC_NEXTPRESSED      8   /* Next month button pressed */
69 #define MC_NEXTMONTHDELAY   350 /* when continuously pressing `next */
70                                                                                 /* month', wait 500 ms before going */
71                                                                                 /* to the next month */
72 #define MC_NEXTMONTHTIMER   1                   /* Timer ID's */
73 #define MC_PREVMONTHTIMER   2
74
75 #define countof(arr) (sizeof(arr)/sizeof(arr[0]))
76
77 typedef struct
78 {
79     HWND hwndSelf;
80     COLORREF    bk;
81     COLORREF    txt;
82     COLORREF    titlebk;
83     COLORREF    titletxt;
84     COLORREF    monthbk;
85     COLORREF    trailingtxt;
86     HFONT       hFont;
87     HFONT       hBoldFont;
88     int         textHeight;
89     int         textWidth;
90     int         height_increment;
91     int         width_increment;
92     int         firstDayplace; /* place of the first day of the current month */
93     int         delta;  /* scroll rate; # of months that the */
94                         /* control moves when user clicks a scroll button */
95     int         visible;        /* # of months visible */
96     int         firstDay;       /* Start month calendar with firstDay's day */
97     int         firstDayHighWord;    /* High word only used externally */
98     int         monthRange;
99     MONTHDAYSTATE *monthdayState;
100     SYSTEMTIME  todaysDate;
101     DWORD       currentMonth;
102     DWORD       currentYear;
103     int         status;         /* See MC_SEL flags */
104     int         curSelDay;      /* current selected day */
105     int         firstSelDay;    /* first selected day */
106     int         maxSelCount;
107     SYSTEMTIME  minSel;
108     SYSTEMTIME  maxSel;
109     DWORD       rangeValid;
110     SYSTEMTIME  minDate;
111     SYSTEMTIME  maxDate;
112
113     RECT title;         /* rect for the header above the calendar */
114     RECT titlebtnnext;  /* the `next month' button in the header */
115     RECT titlebtnprev;  /* the `prev month' button in the header */
116     RECT titlemonth;    /* the `month name' txt in the header */
117     RECT titleyear;     /* the `year number' txt in the header */
118     RECT wdays;         /* week days at top */
119     RECT days;          /* calendar area */
120     RECT weeknums;      /* week numbers at left side */
121     RECT todayrect;     /* `today: xx/xx/xx' text rect */
122     HWND hwndNotify;    /* Window to receive the notifications */
123     HWND hWndYearEdit;  /* Window Handle of edit box to handle years */
124     HWND hWndYearUpDown;/* Window Handle of updown box to handle years */
125 } MONTHCAL_INFO, *LPMONTHCAL_INFO;
126
127
128 /* Offsets of days in the week to the weekday of january 1 in a leap year */
129 static const int DayOfWeekTable[] = {0, 3, 2, 5, 0, 3, 5, 1, 4, 6, 2, 4};
130
131 static const WCHAR themeClass[] = { 'S','c','r','o','l','l','b','a','r',0 };
132
133 #define MONTHCAL_GetInfoPtr(hwnd) ((MONTHCAL_INFO *)GetWindowLongPtrW(hwnd, 0))
134
135 /* helper functions  */
136
137 /* returns the number of days in any given month, checking for leap days */
138 /* january is 1, december is 12 */
139 int MONTHCAL_MonthLength(int month, int year)
140 {
141   const int mdays[] = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31, 0};
142   /*Wrap around, this eases handling*/
143   if(month == 0)
144     month = 12;
145   if(month == 13)
146     month = 1;
147
148   /* if we have a leap year add 1 day to February */
149   /* a leap year is a year either divisible by 400 */
150   /* or divisible by 4 and not by 100 */
151   if(month == 2) { /* February */
152     return mdays[month - 1] + ((year%400 == 0) ? 1 : ((year%100 != 0) &&
153      (year%4 == 0)) ? 1 : 0);
154   }
155   else {
156     return mdays[month - 1];
157   }
158 }
159
160
161 /* make sure that time is valid */
162 static int MONTHCAL_ValidateTime(SYSTEMTIME time)
163 {
164   if(time.wMonth < 1 || time.wMonth > 12 ) return FALSE;
165   if(time.wDayOfWeek > 6) return FALSE;
166   if(time.wDay > MONTHCAL_MonthLength(time.wMonth, time.wYear))
167           return FALSE;
168
169   return TRUE;
170 }
171
172
173 /* Note:Depending on DST, this may be offset by a day.
174    Need to find out if we're on a DST place & adjust the clock accordingly.
175    Above function assumes we have a valid data.
176    Valid for year>1752;  1 <= d <= 31, 1 <= m <= 12.
177    0 = Sunday.
178 */
179
180 /* returns the day in the week(0 == sunday, 6 == saturday) */
181 /* day(1 == 1st, 2 == 2nd... etc), year is the  year value */
182 static int MONTHCAL_CalculateDayOfWeek(DWORD day, DWORD month, DWORD year)
183 {
184   year-=(month < 3);
185
186   return((year + year/4 - year/100 + year/400 +
187          DayOfWeekTable[month-1] + day ) % 7);
188 }
189
190 /* From a given point, calculate the row (weekpos), column(daypos)
191    and day in the calendar. day== 0 mean the last day of tha last month
192 */
193 static int MONTHCAL_CalcDayFromPos(const MONTHCAL_INFO *infoPtr, int x, int y,
194                                    int *daypos,int *weekpos)
195 {
196   int retval, firstDay;
197   RECT rcClient;
198
199   GetClientRect(infoPtr->hwndSelf, &rcClient);
200
201   /* if the point is outside the x bounds of the window put
202   it at the boundary */
203   if (x > rcClient.right)
204     x = rcClient.right;
205
206
207   *daypos = (x - infoPtr->days.left ) / infoPtr->width_increment;
208   *weekpos = (y - infoPtr->days.top ) / infoPtr->height_increment;
209
210   firstDay = (MONTHCAL_CalculateDayOfWeek(1, infoPtr->currentMonth, infoPtr->currentYear)+6 - infoPtr->firstDay)%7;
211   retval = *daypos + (7 * *weekpos) - firstDay;
212   return retval;
213 }
214
215 /* day is the day of the month, 1 == 1st day of the month */
216 /* sets x and y to be the position of the day */
217 /* x == day, y == week where(0,0) == firstDay, 1st week */
218 static void MONTHCAL_CalcDayXY(const MONTHCAL_INFO *infoPtr, int day, int month,
219                                  int *x, int *y)
220 {
221   int firstDay, prevMonth;
222
223   firstDay = (MONTHCAL_CalculateDayOfWeek(1, infoPtr->currentMonth, infoPtr->currentYear) +6 - infoPtr->firstDay)%7;
224
225   if(month==infoPtr->currentMonth) {
226     *x = (day + firstDay) % 7;
227     *y = (day + firstDay - *x) / 7;
228     return;
229   }
230   if(month < infoPtr->currentMonth) {
231     prevMonth = month - 1;
232     if(prevMonth==0)
233        prevMonth = 12;
234
235     *x = (MONTHCAL_MonthLength(prevMonth, infoPtr->currentYear) - firstDay) % 7;
236     *y = 0;
237     return;
238   }
239
240   *y = MONTHCAL_MonthLength(month, infoPtr->currentYear - 1) / 7;
241   *x = (day + firstDay + MONTHCAL_MonthLength(month,
242        infoPtr->currentYear)) % 7;
243 }
244
245
246 /* x: column(day), y: row(week) */
247 static void MONTHCAL_CalcDayRect(const MONTHCAL_INFO *infoPtr, RECT *r, int x, int y)
248 {
249   r->left = infoPtr->days.left + x * infoPtr->width_increment;
250   r->right = r->left + infoPtr->width_increment;
251   r->top  = infoPtr->days.top  + y * infoPtr->height_increment;
252   r->bottom = r->top + infoPtr->textHeight;
253 }
254
255
256 /* sets the RECT struct r to the rectangle around the day and month */
257 /* day is the day value of the month(1 == 1st), month is the month */
258 /* value(january == 1, december == 12) */
259 static inline void MONTHCAL_CalcPosFromDay(const MONTHCAL_INFO *infoPtr,
260                                             int day, int month, RECT *r)
261 {
262   int x, y;
263
264   MONTHCAL_CalcDayXY(infoPtr, day, month, &x, &y);
265   MONTHCAL_CalcDayRect(infoPtr, r, x, y);
266 }
267
268
269 /* day is the day in the month(1 == 1st of the month) */
270 /* month is the month value(1 == january, 12 == december) */
271 static void MONTHCAL_CircleDay(const MONTHCAL_INFO *infoPtr, HDC hdc, int day, int month)
272 {
273   HPEN hRedPen = CreatePen(PS_SOLID, 2, RGB(255, 0, 0));
274   HPEN hOldPen2 = SelectObject(hdc, hRedPen);
275   POINT points[13];
276   int x, y;
277   RECT day_rect;
278
279
280   MONTHCAL_CalcPosFromDay(infoPtr, day, month, &day_rect);
281
282   x = day_rect.left;
283   y = day_rect.top;
284
285   points[0].x = x;
286   points[0].y = y - 1;
287   points[1].x = x + 0.8 * infoPtr->width_increment;
288   points[1].y = y - 1;
289   points[2].x = x + 0.9 * infoPtr->width_increment;
290   points[2].y = y;
291   points[3].x = x + infoPtr->width_increment;
292   points[3].y = y + 0.5 * infoPtr->height_increment;
293
294   points[4].x = x + infoPtr->width_increment;
295   points[4].y = y + 0.9 * infoPtr->height_increment;
296   points[5].x = x + 0.6 * infoPtr->width_increment;
297   points[5].y = y + 0.9 * infoPtr->height_increment;
298   points[6].x = x + 0.5 * infoPtr->width_increment;
299   points[6].y = y + 0.9 * infoPtr->height_increment; /* bring the bottom up just
300                                 a hair to fit inside the day rectangle */
301
302   points[7].x = x + 0.2 * infoPtr->width_increment;
303   points[7].y = y + 0.8 * infoPtr->height_increment;
304   points[8].x = x + 0.1 * infoPtr->width_increment;
305   points[8].y = y + 0.8 * infoPtr->height_increment;
306   points[9].x = x;
307   points[9].y = y + 0.5 * infoPtr->height_increment;
308
309   points[10].x = x + 0.1 * infoPtr->width_increment;
310   points[10].y = y + 0.2 * infoPtr->height_increment;
311   points[11].x = x + 0.2 * infoPtr->width_increment;
312   points[11].y = y + 0.3 * infoPtr->height_increment;
313   points[12].x = x + 0.4 * infoPtr->width_increment;
314   points[12].y = y + 0.2 * infoPtr->height_increment;
315
316   PolyBezier(hdc, points, 13);
317   DeleteObject(hRedPen);
318   SelectObject(hdc, hOldPen2);
319 }
320
321
322 static void MONTHCAL_DrawDay(const MONTHCAL_INFO *infoPtr, HDC hdc, int day, int month,
323                              int x, int y, int bold)
324 {
325   static const WCHAR fmtW[] = { '%','d',0 };
326   WCHAR buf[10];
327   RECT r;
328   static int haveBoldFont, haveSelectedDay = FALSE;
329   HBRUSH hbr;
330   COLORREF oldCol = 0;
331   COLORREF oldBk = 0;
332
333   wsprintfW(buf, fmtW, day);
334
335 /* No need to check styles: when selection is not valid, it is set to zero.
336  * 1<day<31, so evertyhing's OK.
337  */
338
339   MONTHCAL_CalcDayRect(infoPtr, &r, x, y);
340
341   if((day>=infoPtr->minSel.wDay) && (day<=infoPtr->maxSel.wDay)
342        && (month==infoPtr->currentMonth)) {
343     HRGN hrgn;
344     RECT r2;
345
346     TRACE("%d %d %d\n",day, infoPtr->minSel.wDay, infoPtr->maxSel.wDay);
347     TRACE("%d %d %d %d\n", r.left, r.top, r.right, r.bottom);
348     oldCol = SetTextColor(hdc, infoPtr->monthbk);
349     oldBk = SetBkColor(hdc, infoPtr->trailingtxt);
350     hbr = GetSysColorBrush(COLOR_GRAYTEXT);
351     hrgn = CreateEllipticRgn(r.left, r.top, r.right, r.bottom);
352     FillRgn(hdc, hrgn, hbr);
353
354     /* FIXME: this may need to be changed now b/c of the other
355         drawing changes 11/3/99 CMM */
356     r2.left   = r.left - 0.25 * infoPtr->textWidth;
357     r2.top    = r.top;
358     r2.right  = r.left + 0.5 * infoPtr->textWidth;
359     r2.bottom = r.bottom;
360     if(haveSelectedDay) FillRect(hdc, &r2, hbr);
361       haveSelectedDay = TRUE;
362   } else {
363     haveSelectedDay = FALSE;
364   }
365
366   /* need to add some code for multiple selections */
367
368   if((bold) &&(!haveBoldFont)) {
369     SelectObject(hdc, infoPtr->hBoldFont);
370     haveBoldFont = TRUE;
371   }
372   if((!bold) &&(haveBoldFont)) {
373     SelectObject(hdc, infoPtr->hFont);
374     haveBoldFont = FALSE;
375   }
376
377   if(haveSelectedDay) {
378     SetTextColor(hdc, oldCol);
379     SetBkColor(hdc, oldBk);
380   }
381
382   SetBkMode(hdc,TRANSPARENT);
383   DrawTextW(hdc, buf, -1, &r, DT_CENTER | DT_VCENTER | DT_SINGLELINE );
384
385   /* draw a rectangle around the currently selected days text */
386   if((day==infoPtr->curSelDay) && (month==infoPtr->currentMonth))
387     DrawFocusRect(hdc, &r);
388 }
389
390
391 static void paint_button (const MONTHCAL_INFO *infoPtr, HDC hdc, BOOL btnNext,
392                           BOOL pressed, RECT* r)
393 {
394     HTHEME theme = GetWindowTheme (infoPtr->hwndSelf);
395     
396     if (theme)
397     {
398         static const int states[] = {
399             /* Prev button */
400             ABS_LEFTNORMAL,  ABS_LEFTPRESSED,  ABS_LEFTDISABLED,
401             /* Next button */
402             ABS_RIGHTNORMAL, ABS_RIGHTPRESSED, ABS_RIGHTDISABLED
403         };
404         int stateNum = btnNext ? 3 : 0;
405         if (pressed)
406             stateNum += 1;
407         else
408         {
409             DWORD dwStyle = GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE);
410             if (dwStyle & WS_DISABLED) stateNum += 2;
411         }
412         DrawThemeBackground (theme, hdc, SBP_ARROWBTN, states[stateNum], r, NULL);
413     }
414     else
415     {
416         int style = btnNext ? DFCS_SCROLLRIGHT : DFCS_SCROLLLEFT;
417         if (pressed)
418             style |= DFCS_PUSHED;
419         else
420         {
421             DWORD dwStyle = GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE);
422             if (dwStyle & WS_DISABLED) style |= DFCS_INACTIVE;
423         }
424         
425         DrawFrameControl(hdc, r, DFC_SCROLL, style);
426     }
427 }
428
429
430 static void MONTHCAL_Refresh(MONTHCAL_INFO *infoPtr, HDC hdc, const PAINTSTRUCT *ps)
431 {
432   static const WCHAR todayW[] = { 'T','o','d','a','y',':',0 };
433   static const WCHAR fmt1W[] = { '%','s',' ','%','l','d',0 };
434   static const WCHAR fmt2W[] = { '%','s',' ','%','s',0 };
435   static const WCHAR fmt3W[] = { '%','d',0 };
436   RECT *title=&infoPtr->title;
437   RECT *prev=&infoPtr->titlebtnprev;
438   RECT *next=&infoPtr->titlebtnnext;
439   RECT *titlemonth=&infoPtr->titlemonth;
440   RECT *titleyear=&infoPtr->titleyear;
441   RECT dayrect;
442   RECT *days=&dayrect;
443   RECT rtoday;
444   int i, j, m, mask, day, firstDay, weeknum, weeknum1,prevMonth;
445   int textHeight = infoPtr->textHeight, textWidth = infoPtr->textWidth;
446   SIZE size;
447   HBRUSH hbr;
448   HFONT currentFont;
449   WCHAR buf[20];
450   WCHAR buf1[20];
451   WCHAR buf2[32];
452   COLORREF oldTextColor, oldBkColor;
453   DWORD dwStyle = GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE);
454   RECT rcTemp;
455   RECT rcDay; /* used in MONTHCAL_CalcDayRect() */
456   SYSTEMTIME localtime;
457   int startofprescal;
458
459   oldTextColor = SetTextColor(hdc, GetSysColor(COLOR_WINDOWTEXT));
460
461   /* fill background */
462   hbr = CreateSolidBrush (infoPtr->bk);
463   FillRect(hdc, &ps->rcPaint, hbr);
464   DeleteObject(hbr);
465
466   /* draw header */
467   if(IntersectRect(&rcTemp, &(ps->rcPaint), title))
468   {
469     hbr =  CreateSolidBrush(infoPtr->titlebk);
470     FillRect(hdc, title, hbr);
471     DeleteObject(hbr);
472   }
473
474   /* if the previous button is pressed draw it depressed */
475   if(IntersectRect(&rcTemp, &(ps->rcPaint), prev))
476     paint_button (infoPtr, hdc, FALSE, infoPtr->status & MC_PREVPRESSED, prev);
477
478   /* if next button is depressed draw it depressed */
479   if(IntersectRect(&rcTemp, &(ps->rcPaint), next))
480     paint_button (infoPtr, hdc, TRUE, infoPtr->status & MC_NEXTPRESSED, next);
481
482   oldBkColor = SetBkColor(hdc, infoPtr->titlebk);
483   SetTextColor(hdc, infoPtr->titletxt);
484   currentFont = SelectObject(hdc, infoPtr->hBoldFont);
485
486   GetLocaleInfoW( LOCALE_USER_DEFAULT,LOCALE_SMONTHNAME1+infoPtr->currentMonth -1,
487                   buf1,countof(buf1));
488   wsprintfW(buf, fmt1W, buf1, infoPtr->currentYear);
489
490   if(IntersectRect(&rcTemp, &(ps->rcPaint), title))
491   {
492     DrawTextW(hdc, buf, strlenW(buf), title,
493                         DT_CENTER | DT_VCENTER | DT_SINGLELINE);
494   }
495
496 /* titlemonth left/right contained rect for whole titletxt('June  1999')
497   * MCM_HitTestInfo wants month & year rects, so prepare these now.
498   *(no, we can't draw them separately; the whole text is centered)
499   */
500   GetTextExtentPoint32W(hdc, buf, strlenW(buf), &size);
501   titlemonth->left = title->right / 2 + title->left / 2 - size.cx / 2;
502   titleyear->right = title->right / 2 + title->left / 2 + size.cx / 2;
503   GetTextExtentPoint32W(hdc, buf1, strlenW(buf1), &size);
504   titlemonth->right = titlemonth->left + size.cx;
505   titleyear->left = titlemonth->right;
506
507   /* draw month area */
508   rcTemp.top=infoPtr->wdays.top;
509   rcTemp.left=infoPtr->wdays.left;
510   rcTemp.bottom=infoPtr->todayrect.bottom;
511   rcTemp.right =infoPtr->todayrect.right;
512   if(IntersectRect(&rcTemp, &(ps->rcPaint), &rcTemp))
513   {
514     hbr =  CreateSolidBrush(infoPtr->monthbk);
515     FillRect(hdc, &rcTemp, hbr);
516     DeleteObject(hbr);
517   }
518
519 /* draw line under day abbreviatons */
520
521   MoveToEx(hdc, infoPtr->days.left + 3, title->bottom + textHeight + 1, NULL);
522   LineTo(hdc, infoPtr->days.right - 3, title->bottom + textHeight + 1);
523
524   prevMonth = infoPtr->currentMonth - 1;
525   if(prevMonth == 0) /* if currentMonth is january(1) prevMonth is */
526     prevMonth = 12;    /* december(12) of the previous year */
527
528   infoPtr->wdays.left   = infoPtr->days.left   = infoPtr->weeknums.right;
529 /* draw day abbreviations */
530
531   SelectObject(hdc, infoPtr->hFont);
532   SetBkColor(hdc, infoPtr->monthbk);
533   SetTextColor(hdc, infoPtr->trailingtxt);
534
535   /* copy this rect so we can change the values without changing */
536   /* the original version */
537   days->left = infoPtr->wdays.left;
538   days->right = days->left + infoPtr->width_increment;
539   days->top = infoPtr->wdays.top;
540   days->bottom = infoPtr->wdays.bottom;
541
542   i = infoPtr->firstDay;
543
544   for(j=0; j<7; j++) {
545     GetLocaleInfoW( LOCALE_USER_DEFAULT,LOCALE_SABBREVDAYNAME1 + (i+j+6)%7, buf, countof(buf));
546     DrawTextW(hdc, buf, strlenW(buf), days, DT_CENTER | DT_VCENTER | DT_SINGLELINE );
547     days->left+=infoPtr->width_increment;
548     days->right+=infoPtr->width_increment;
549   }
550
551 /* draw day numbers; first, the previous month */
552
553   firstDay = MONTHCAL_CalculateDayOfWeek(1, infoPtr->currentMonth, infoPtr->currentYear);
554
555   day = MONTHCAL_MonthLength(prevMonth, infoPtr->currentYear)  +
556     (infoPtr->firstDay + 7  - firstDay)%7 + 1;
557   if (day > MONTHCAL_MonthLength(prevMonth, infoPtr->currentYear))
558     day -=7;
559   startofprescal = day;
560   mask = 1<<(day-1);
561
562   i = 0;
563   m = 0;
564   while(day <= MONTHCAL_MonthLength(prevMonth, infoPtr->currentYear)) {
565     MONTHCAL_CalcDayRect(infoPtr, &rcDay, i, 0);
566     if(IntersectRect(&rcTemp, &(ps->rcPaint), &rcDay))
567     {
568       MONTHCAL_DrawDay(infoPtr, hdc, day, prevMonth, i, 0,
569           infoPtr->monthdayState[m] & mask);
570     }
571
572     mask<<=1;
573     day++;
574     i++;
575   }
576
577 /* draw `current' month  */
578
579   day = 1; /* start at the beginning of the current month */
580
581   infoPtr->firstDayplace = i;
582   SetTextColor(hdc, infoPtr->txt);
583   m++;
584   mask = 1;
585
586   /* draw the first week of the current month */
587   while(i<7) {
588     MONTHCAL_CalcDayRect(infoPtr, &rcDay, i, 0);
589     if(IntersectRect(&rcTemp, &(ps->rcPaint), &rcDay))
590     {
591
592       MONTHCAL_DrawDay(infoPtr, hdc, day, infoPtr->currentMonth, i, 0,
593         infoPtr->monthdayState[m] & mask);
594
595       if((infoPtr->currentMonth==infoPtr->todaysDate.wMonth) &&
596           (day==infoPtr->todaysDate.wDay) &&
597           (infoPtr->currentYear == infoPtr->todaysDate.wYear)) {
598         if(!(dwStyle & MCS_NOTODAYCIRCLE))
599           MONTHCAL_CircleDay(infoPtr, hdc, day, infoPtr->currentMonth);
600       }
601     }
602
603     mask<<=1;
604     day++;
605     i++;
606   }
607
608   j = 1; /* move to the 2nd week of the current month */
609   i = 0; /* move back to sunday */
610   while(day <= MONTHCAL_MonthLength(infoPtr->currentMonth, infoPtr->currentYear)) {
611     MONTHCAL_CalcDayRect(infoPtr, &rcDay, i, j);
612     if(IntersectRect(&rcTemp, &(ps->rcPaint), &rcDay))
613     {
614       MONTHCAL_DrawDay(infoPtr, hdc, day, infoPtr->currentMonth, i, j,
615           infoPtr->monthdayState[m] & mask);
616
617       if((infoPtr->currentMonth==infoPtr->todaysDate.wMonth) &&
618           (day==infoPtr->todaysDate.wDay) &&
619           (infoPtr->currentYear == infoPtr->todaysDate.wYear))
620         if(!(dwStyle & MCS_NOTODAYCIRCLE))
621           MONTHCAL_CircleDay(infoPtr, hdc, day, infoPtr->currentMonth);
622     }
623     mask<<=1;
624     day++;
625     i++;
626     if(i>6) { /* past saturday, goto the next weeks sunday */
627       i = 0;
628       j++;
629     }
630   }
631
632 /*  draw `next' month */
633
634   day = 1; /* start at the first day of the next month */
635   m++;
636   mask = 1;
637
638   SetTextColor(hdc, infoPtr->trailingtxt);
639   while((i<7) &&(j<6)) {
640     MONTHCAL_CalcDayRect(infoPtr, &rcDay, i, j);
641     if(IntersectRect(&rcTemp, &(ps->rcPaint), &rcDay))
642     {
643       MONTHCAL_DrawDay(infoPtr, hdc, day, infoPtr->currentMonth + 1, i, j,
644                 infoPtr->monthdayState[m] & mask);
645     }
646
647     mask<<=1;
648     day++;
649     i++;
650     if(i==7) { /* past saturday, go to next week's sunday */
651       i = 0;
652       j++;
653     }
654   }
655   SetTextColor(hdc, infoPtr->txt);
656
657
658 /* draw `today' date if style allows it, and draw a circle before today's
659  * date if necessary */
660
661   if(!(dwStyle & MCS_NOTODAY))  {
662     int offset = 0;
663     if(!(dwStyle & MCS_NOTODAYCIRCLE))  {
664       /*day is the number of days from nextmonth we put on the calendar */
665       MONTHCAL_CircleDay(infoPtr, hdc,
666                          day+MONTHCAL_MonthLength(infoPtr->currentMonth,infoPtr->currentYear),
667                          infoPtr->currentMonth);
668       offset+=textWidth;
669     }
670     if (!LoadStringW(COMCTL32_hModule,IDM_TODAY,buf1,countof(buf1)))
671       {
672         WARN("Can't load resource\n");
673         strcpyW(buf1, todayW);
674       }
675     MONTHCAL_CalcDayRect(infoPtr, &rtoday, 1, 6);
676     MONTHCAL_CopyTime(&infoPtr->todaysDate,&localtime);
677     GetDateFormatW(LOCALE_USER_DEFAULT,DATE_SHORTDATE,&localtime,NULL,buf2,countof(buf2));
678     wsprintfW(buf, fmt2W, buf1, buf2);
679     SelectObject(hdc, infoPtr->hBoldFont);
680
681     DrawTextW(hdc, buf, -1, &rtoday, DT_CALCRECT | DT_LEFT | DT_VCENTER | DT_SINGLELINE);
682     if(IntersectRect(&rcTemp, &(ps->rcPaint), &rtoday))
683     {
684       DrawTextW(hdc, buf, -1, &rtoday, DT_LEFT | DT_VCENTER | DT_SINGLELINE);
685     }
686     SelectObject(hdc, infoPtr->hFont);
687   }
688
689 /*eventually draw week numbers*/
690   if(dwStyle & MCS_WEEKNUMBERS)  {
691     /* display weeknumbers*/
692     int mindays;
693
694     /* Rules what week to call the first week of a new year:
695        LOCALE_IFIRSTWEEKOFYEAR == 0 (e.g US?):
696        The week containing Jan 1 is the first week of year
697        LOCALE_IFIRSTWEEKOFYEAR == 2 (e.g. Germany):
698        First week of year must contain 4 days of the new year
699        LOCALE_IFIRSTWEEKOFYEAR == 1  (what contries?)
700        The first week of the year must contain only days of the new year
701     */
702     GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_IFIRSTWEEKOFYEAR, buf, countof(buf));
703     weeknum = atoiW(buf);
704     switch (weeknum)
705       {
706       case 1: mindays = 6;
707         break;
708       case 2: mindays = 3;
709         break;
710       case 0:
711       default:
712         mindays = 0;
713       }
714     if (infoPtr->currentMonth < 2)
715       {
716         /* calculate all those exceptions for january */
717         weeknum1=MONTHCAL_CalculateDayOfWeek(1,1,infoPtr->currentYear);
718         if ((infoPtr->firstDay +7 - weeknum1)%7 > mindays)
719             weeknum =1;
720         else
721           {
722             weeknum = 0;
723             for(i=0; i<11; i++)
724               weeknum+=MONTHCAL_MonthLength(i+1, infoPtr->currentYear-1);
725             weeknum +=startofprescal+ 7;
726             weeknum /=7;
727             weeknum1=MONTHCAL_CalculateDayOfWeek(1,1,infoPtr->currentYear-1);
728             if ((infoPtr->firstDay + 7 - weeknum1)%7 > mindays)
729               weeknum++;
730           }
731       }
732     else
733       {
734         weeknum = 0;
735         for(i=0; i<prevMonth-1; i++)
736           weeknum+=MONTHCAL_MonthLength(i+1, infoPtr->currentYear);
737         weeknum +=startofprescal+ 7;
738         weeknum /=7;
739         weeknum1=MONTHCAL_CalculateDayOfWeek(1,1,infoPtr->currentYear);
740         if ((infoPtr->firstDay + 7 - weeknum1)%7 > mindays)
741           weeknum++;
742       }
743     days->left = infoPtr->weeknums.left;
744     days->right = infoPtr->weeknums.right;
745     days->top = infoPtr->weeknums.top;
746     days->bottom = days->top +infoPtr->height_increment;
747     for(i=0; i<6; i++) {
748       if((i==0)&&(weeknum>50))
749         {
750           wsprintfW(buf, fmt3W, weeknum);
751           weeknum=0;
752         }
753       else if((i==5)&&(weeknum>47))
754         {
755           wsprintfW(buf, fmt3W, 1);
756         }
757       else
758         wsprintfW(buf, fmt3W, weeknum + i);
759       DrawTextW(hdc, buf, -1, days, DT_CENTER | DT_VCENTER | DT_SINGLELINE );
760       days->top+=infoPtr->height_increment;
761       days->bottom+=infoPtr->height_increment;
762     }
763
764     MoveToEx(hdc, infoPtr->weeknums.right, infoPtr->weeknums.top + 3 , NULL);
765     LineTo(hdc,   infoPtr->weeknums.right, infoPtr->weeknums.bottom );
766
767   }
768   /* currentFont was font at entering Refresh */
769
770   SetBkColor(hdc, oldBkColor);
771   SelectObject(hdc, currentFont);
772   SetTextColor(hdc, oldTextColor);
773 }
774
775
776 static LRESULT
777 MONTHCAL_GetMinReqRect(const MONTHCAL_INFO *infoPtr, LPARAM lParam)
778 {
779   LPRECT lpRect = (LPRECT) lParam;
780
781   TRACE("rect %p\n", lpRect);
782
783   /* validate parameters */
784
785   if((infoPtr==NULL) ||(lpRect == NULL) ) return FALSE;
786
787   lpRect->left = infoPtr->title.left;
788   lpRect->top = infoPtr->title.top;
789   lpRect->right = infoPtr->title.right;
790   lpRect->bottom = infoPtr->todayrect.bottom;
791   AdjustWindowRect(lpRect, GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE), FALSE);
792
793   TRACE("%s\n", wine_dbgstr_rect(lpRect));
794
795   return TRUE;
796 }
797
798
799 static LRESULT
800 MONTHCAL_GetColor(const MONTHCAL_INFO *infoPtr, WPARAM wParam)
801 {
802   TRACE("\n");
803
804   switch((int)wParam) {
805     case MCSC_BACKGROUND:
806       return infoPtr->bk;
807     case MCSC_TEXT:
808       return infoPtr->txt;
809     case MCSC_TITLEBK:
810       return infoPtr->titlebk;
811     case MCSC_TITLETEXT:
812       return infoPtr->titletxt;
813     case MCSC_MONTHBK:
814       return infoPtr->monthbk;
815     case MCSC_TRAILINGTEXT:
816       return infoPtr->trailingtxt;
817   }
818
819   return -1;
820 }
821
822
823 static LRESULT
824 MONTHCAL_SetColor(MONTHCAL_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
825 {
826   int prev = -1;
827
828   TRACE("%ld: color %08lx\n", wParam, lParam);
829
830   switch((int)wParam) {
831     case MCSC_BACKGROUND:
832       prev = infoPtr->bk;
833       infoPtr->bk = (COLORREF)lParam;
834       break;
835     case MCSC_TEXT:
836       prev = infoPtr->txt;
837       infoPtr->txt = (COLORREF)lParam;
838       break;
839     case MCSC_TITLEBK:
840       prev = infoPtr->titlebk;
841       infoPtr->titlebk = (COLORREF)lParam;
842       break;
843     case MCSC_TITLETEXT:
844       prev=infoPtr->titletxt;
845       infoPtr->titletxt = (COLORREF)lParam;
846       break;
847     case MCSC_MONTHBK:
848       prev = infoPtr->monthbk;
849       infoPtr->monthbk = (COLORREF)lParam;
850       break;
851     case MCSC_TRAILINGTEXT:
852       prev = infoPtr->trailingtxt;
853       infoPtr->trailingtxt = (COLORREF)lParam;
854       break;
855   }
856
857   InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
858   return prev;
859 }
860
861
862 static LRESULT
863 MONTHCAL_GetMonthDelta(const MONTHCAL_INFO *infoPtr)
864 {
865   TRACE("\n");
866
867   if(infoPtr->delta)
868     return infoPtr->delta;
869   else
870     return infoPtr->visible;
871 }
872
873
874 static LRESULT
875 MONTHCAL_SetMonthDelta(MONTHCAL_INFO *infoPtr, WPARAM wParam)
876 {
877   int prev = infoPtr->delta;
878
879   TRACE("delta %ld\n", wParam);
880
881   infoPtr->delta = (int)wParam;
882   return prev;
883 }
884
885
886 static LRESULT
887 MONTHCAL_GetFirstDayOfWeek(const MONTHCAL_INFO *infoPtr)
888 {
889   return MAKELONG(infoPtr->firstDay, infoPtr->firstDayHighWord);
890 }
891
892
893 /* sets the first day of the week that will appear in the control */
894 /* 0 == Sunday, 6 == Saturday */
895 /* FIXME: this needs to be implemented properly in MONTHCAL_Refresh() */
896 /* FIXME: we need more error checking here */
897 static LRESULT
898 MONTHCAL_SetFirstDayOfWeek(MONTHCAL_INFO *infoPtr, LPARAM lParam)
899 {
900   int prev = MAKELONG(infoPtr->firstDay, infoPtr->firstDayHighWord);
901   int localFirstDay;
902   WCHAR buf[40];
903
904   TRACE("day %ld\n", lParam);
905
906   GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_IFIRSTDAYOFWEEK, buf, countof(buf));
907   TRACE("%s %d\n", debugstr_w(buf), strlenW(buf));
908
909   localFirstDay = atoiW(buf);
910
911   if(lParam == -1)
912   {
913     infoPtr->firstDay = localFirstDay;
914     infoPtr->firstDayHighWord = FALSE;
915   }
916   else if(lParam >= 7)
917   {
918     infoPtr->firstDay = localFirstDay;
919     infoPtr->firstDayHighWord = TRUE;
920   }
921   else
922   {
923     infoPtr->firstDay = lParam;
924     infoPtr->firstDayHighWord = TRUE;
925   }
926
927   return prev;
928 }
929
930
931 static LRESULT
932 MONTHCAL_GetMonthRange(const MONTHCAL_INFO *infoPtr)
933 {
934   TRACE("\n");
935
936   return infoPtr->monthRange;
937 }
938
939
940 static LRESULT
941 MONTHCAL_GetMaxTodayWidth(const MONTHCAL_INFO *infoPtr)
942 {
943   return(infoPtr->todayrect.right - infoPtr->todayrect.left);
944 }
945
946
947 static LRESULT
948 MONTHCAL_SetRange(MONTHCAL_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
949 {
950     SYSTEMTIME *lprgSysTimeArray=(SYSTEMTIME *)lParam;
951     FILETIME ft_min, ft_max;
952
953     TRACE("%lx %lx\n", wParam, lParam);
954
955     if ((wParam & GDTR_MIN && !MONTHCAL_ValidateTime(lprgSysTimeArray[0])) ||
956         (wParam & GDTR_MAX && !MONTHCAL_ValidateTime(lprgSysTimeArray[1])))
957         return FALSE;
958
959     if (wParam & GDTR_MIN)
960     {
961         MONTHCAL_CopyTime(&lprgSysTimeArray[0], &infoPtr->minDate);
962         infoPtr->rangeValid |= GDTR_MIN;
963     }
964     if (wParam & GDTR_MAX)
965     {
966         MONTHCAL_CopyTime(&lprgSysTimeArray[1], &infoPtr->maxDate);
967         infoPtr->rangeValid |= GDTR_MAX;
968     }
969
970     /* Only one limit set - we are done */
971     if ((infoPtr->rangeValid & (GDTR_MIN | GDTR_MAX)) != (GDTR_MIN | GDTR_MAX))
972         return TRUE;
973     
974     SystemTimeToFileTime(&infoPtr->maxDate, &ft_max);
975     SystemTimeToFileTime(&infoPtr->minDate, &ft_min);
976
977     if (CompareFileTime(&ft_min, &ft_max) > 0)
978     {
979         if ((wParam & (GDTR_MIN | GDTR_MAX)) == (GDTR_MIN | GDTR_MAX))
980         {
981             /* Native swaps limits only when both limits are being set. */
982             SYSTEMTIME st_tmp = infoPtr->minDate;
983             infoPtr->minDate  = infoPtr->maxDate;
984             infoPtr->maxDate  = st_tmp;
985         }
986         else
987         {
988             /* Reset the other limit. */
989             /* FIXME: native sets date&time to 0. Should we do this too? */
990             infoPtr->rangeValid &= wParam & GDTR_MIN ? ~GDTR_MAX : ~GDTR_MIN ;
991         }
992     }
993
994     return TRUE;
995 }
996
997
998 static LRESULT
999 MONTHCAL_GetRange(HWND hwnd, WPARAM wParam, LPARAM lParam)
1000 {
1001   MONTHCAL_INFO *infoPtr = MONTHCAL_GetInfoPtr(hwnd);
1002   SYSTEMTIME *lprgSysTimeArray = (SYSTEMTIME *)lParam;
1003
1004   /* validate parameters */
1005
1006   if((infoPtr==NULL) || (lprgSysTimeArray==NULL)) return FALSE;
1007
1008   MONTHCAL_CopyTime(&infoPtr->maxDate, &lprgSysTimeArray[1]);
1009   MONTHCAL_CopyTime(&infoPtr->minDate, &lprgSysTimeArray[0]);
1010
1011   return infoPtr->rangeValid;
1012 }
1013
1014
1015 static LRESULT
1016 MONTHCAL_SetDayState(const MONTHCAL_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
1017
1018 {
1019   int i, iMonths = (int)wParam;
1020   MONTHDAYSTATE *dayStates = (LPMONTHDAYSTATE)lParam;
1021
1022   TRACE("%lx %lx\n", wParam, lParam);
1023   if(iMonths!=infoPtr->monthRange) return 0;
1024
1025   for(i=0; i<iMonths; i++)
1026     infoPtr->monthdayState[i] = dayStates[i];
1027   return 1;
1028 }
1029
1030 static LRESULT
1031 MONTHCAL_GetCurSel(const MONTHCAL_INFO *infoPtr, LPARAM lParam)
1032 {
1033   SYSTEMTIME *lpSel = (SYSTEMTIME *) lParam;
1034
1035   TRACE("%lx\n", lParam);
1036   if((infoPtr==NULL) ||(lpSel==NULL)) return FALSE;
1037   if(GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE) & MCS_MULTISELECT) return FALSE;
1038
1039   MONTHCAL_CopyTime(&infoPtr->minSel, lpSel);
1040   TRACE("%d/%d/%d\n", lpSel->wYear, lpSel->wMonth, lpSel->wDay);
1041   return TRUE;
1042 }
1043
1044 /* FIXME: if the specified date is not visible, make it visible */
1045 /* FIXME: redraw? */
1046 static LRESULT
1047 MONTHCAL_SetCurSel(MONTHCAL_INFO *infoPtr, LPARAM lParam)
1048 {
1049   SYSTEMTIME *lpSel = (SYSTEMTIME *)lParam;
1050
1051   TRACE("%lx\n", lParam);
1052   if((infoPtr==NULL) ||(lpSel==NULL)) return FALSE;
1053   if(GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE) & MCS_MULTISELECT) return FALSE;
1054
1055   if(!MONTHCAL_ValidateTime(*lpSel)) return FALSE;
1056
1057   infoPtr->currentMonth=lpSel->wMonth;
1058   infoPtr->currentYear=lpSel->wYear;
1059
1060   MONTHCAL_CopyTime(lpSel, &infoPtr->minSel);
1061   MONTHCAL_CopyTime(lpSel, &infoPtr->maxSel);
1062
1063   InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1064
1065   return TRUE;
1066 }
1067
1068
1069 static LRESULT
1070 MONTHCAL_GetMaxSelCount(const MONTHCAL_INFO *infoPtr)
1071 {
1072   return infoPtr->maxSelCount;
1073 }
1074
1075
1076 static LRESULT
1077 MONTHCAL_SetMaxSelCount(MONTHCAL_INFO *infoPtr, WPARAM wParam)
1078 {
1079   TRACE("%lx\n", wParam);
1080
1081   if(GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE) & MCS_MULTISELECT)  {
1082     infoPtr->maxSelCount = wParam;
1083   }
1084
1085   return TRUE;
1086 }
1087
1088
1089 static LRESULT
1090 MONTHCAL_GetSelRange(const MONTHCAL_INFO *infoPtr, LPARAM lParam)
1091 {
1092   SYSTEMTIME *lprgSysTimeArray = (SYSTEMTIME *) lParam;
1093
1094   TRACE("%lx\n", lParam);
1095
1096   /* validate parameters */
1097
1098   if((infoPtr==NULL) ||(lprgSysTimeArray==NULL)) return FALSE;
1099
1100   if(GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE) & MCS_MULTISELECT)
1101   {
1102     MONTHCAL_CopyTime(&infoPtr->maxSel, &lprgSysTimeArray[1]);
1103     MONTHCAL_CopyTime(&infoPtr->minSel, &lprgSysTimeArray[0]);
1104     TRACE("[min,max]=[%d %d]\n", infoPtr->minSel.wDay, infoPtr->maxSel.wDay);
1105     return TRUE;
1106   }
1107
1108   return FALSE;
1109 }
1110
1111
1112 static LRESULT
1113 MONTHCAL_SetSelRange(MONTHCAL_INFO *infoPtr, LPARAM lParam)
1114 {
1115   SYSTEMTIME *lprgSysTimeArray = (SYSTEMTIME *) lParam;
1116
1117   TRACE("%lx\n", lParam);
1118
1119   /* validate parameters */
1120
1121   if((infoPtr==NULL) ||(lprgSysTimeArray==NULL)) return FALSE;
1122
1123   if(GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE) & MCS_MULTISELECT)
1124   {
1125     MONTHCAL_CopyTime(&lprgSysTimeArray[1], &infoPtr->maxSel);
1126     MONTHCAL_CopyTime(&lprgSysTimeArray[0], &infoPtr->minSel);
1127     TRACE("[min,max]=[%d %d]\n", infoPtr->minSel.wDay, infoPtr->maxSel.wDay);
1128     return TRUE;
1129   }
1130
1131   return FALSE;
1132 }
1133
1134
1135 static LRESULT
1136 MONTHCAL_GetToday(const MONTHCAL_INFO *infoPtr, LPARAM lParam)
1137 {
1138   SYSTEMTIME *lpToday = (SYSTEMTIME *) lParam;
1139
1140   TRACE("%lx\n", lParam);
1141
1142   /* validate parameters */
1143
1144   if((infoPtr==NULL) || (lpToday==NULL)) return FALSE;
1145   MONTHCAL_CopyTime(&infoPtr->todaysDate, lpToday);
1146   return TRUE;
1147 }
1148
1149
1150 static LRESULT
1151 MONTHCAL_SetToday(MONTHCAL_INFO *infoPtr, LPARAM lParam)
1152 {
1153   SYSTEMTIME *lpToday = (SYSTEMTIME *) lParam;
1154
1155   TRACE("%lx\n", lParam);
1156
1157   /* validate parameters */
1158
1159   if((infoPtr==NULL) ||(lpToday==NULL)) return FALSE;
1160   MONTHCAL_CopyTime(lpToday, &infoPtr->todaysDate);
1161   InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1162   return TRUE;
1163 }
1164
1165
1166 static LRESULT
1167 MONTHCAL_HitTest(const MONTHCAL_INFO *infoPtr, LPARAM lParam)
1168 {
1169   PMCHITTESTINFO lpht = (PMCHITTESTINFO)lParam;
1170   UINT x,y;
1171   DWORD retval;
1172   int day,wday,wnum;
1173
1174
1175   x = lpht->pt.x;
1176   y = lpht->pt.y;
1177   retval = MCHT_NOWHERE;
1178
1179   ZeroMemory(&lpht->st, sizeof(lpht->st));
1180
1181   /* Comment in for debugging...
1182   TRACE("%d %d wd[%d %d %d %d] d[%d %d %d %d] t[%d %d %d %d] wn[%d %d %d %d]\n", x, y,
1183         infoPtr->wdays.left, infoPtr->wdays.right,
1184         infoPtr->wdays.top, infoPtr->wdays.bottom,
1185         infoPtr->days.left, infoPtr->days.right,
1186         infoPtr->days.top, infoPtr->days.bottom,
1187         infoPtr->todayrect.left, infoPtr->todayrect.right,
1188         infoPtr->todayrect.top, infoPtr->todayrect.bottom,
1189         infoPtr->weeknums.left, infoPtr->weeknums.right,
1190         infoPtr->weeknums.top, infoPtr->weeknums.bottom);
1191   */
1192
1193   /* are we in the header? */
1194
1195   if(PtInRect(&infoPtr->title, lpht->pt)) {
1196     if(PtInRect(&infoPtr->titlebtnprev, lpht->pt)) {
1197       retval = MCHT_TITLEBTNPREV;
1198       goto done;
1199     }
1200     if(PtInRect(&infoPtr->titlebtnnext, lpht->pt)) {
1201       retval = MCHT_TITLEBTNNEXT;
1202       goto done;
1203     }
1204     if(PtInRect(&infoPtr->titlemonth, lpht->pt)) {
1205       retval = MCHT_TITLEMONTH;
1206       goto done;
1207     }
1208     if(PtInRect(&infoPtr->titleyear, lpht->pt)) {
1209       retval = MCHT_TITLEYEAR;
1210       goto done;
1211     }
1212
1213     retval = MCHT_TITLE;
1214     goto done;
1215   }
1216
1217   day = MONTHCAL_CalcDayFromPos(infoPtr,x,y,&wday,&wnum);
1218   if(PtInRect(&infoPtr->wdays, lpht->pt)) {
1219     retval = MCHT_CALENDARDAY;
1220     lpht->st.wYear  = infoPtr->currentYear;
1221     lpht->st.wMonth = (day < 1)? infoPtr->currentMonth -1 : infoPtr->currentMonth;
1222     lpht->st.wDay   = (day < 1)?
1223       MONTHCAL_MonthLength(infoPtr->currentMonth-1,infoPtr->currentYear) -day : day;
1224     goto done;
1225   }
1226   if(PtInRect(&infoPtr->weeknums, lpht->pt)) {
1227     retval = MCHT_CALENDARWEEKNUM;
1228     lpht->st.wYear  = infoPtr->currentYear;
1229     lpht->st.wMonth = (day < 1) ? infoPtr->currentMonth -1 :
1230       (day > MONTHCAL_MonthLength(infoPtr->currentMonth,infoPtr->currentYear)) ?
1231       infoPtr->currentMonth +1 :infoPtr->currentMonth;
1232     lpht->st.wDay   = (day < 1 ) ?
1233       MONTHCAL_MonthLength(infoPtr->currentMonth-1,infoPtr->currentYear) -day :
1234       (day > MONTHCAL_MonthLength(infoPtr->currentMonth,infoPtr->currentYear)) ?
1235       day - MONTHCAL_MonthLength(infoPtr->currentMonth,infoPtr->currentYear) : day;
1236     goto done;
1237   }
1238   if(PtInRect(&infoPtr->days, lpht->pt))
1239     {
1240       lpht->st.wYear  = infoPtr->currentYear;
1241       if ( day < 1)
1242         {
1243           retval = MCHT_CALENDARDATEPREV;
1244           lpht->st.wMonth = infoPtr->currentMonth - 1;
1245           if (lpht->st.wMonth <1)
1246             {
1247               lpht->st.wMonth = 12;
1248               lpht->st.wYear--;
1249             }
1250           lpht->st.wDay   = MONTHCAL_MonthLength(lpht->st.wMonth,lpht->st.wYear) -day;
1251         }
1252       else if (day > MONTHCAL_MonthLength(infoPtr->currentMonth,infoPtr->currentYear))
1253         {
1254           retval = MCHT_CALENDARDATENEXT;
1255           lpht->st.wMonth = infoPtr->currentMonth + 1;
1256           if (lpht->st.wMonth <12)
1257             {
1258               lpht->st.wMonth = 1;
1259               lpht->st.wYear++;
1260             }
1261           lpht->st.wDay   = day - MONTHCAL_MonthLength(infoPtr->currentMonth,infoPtr->currentYear) ;
1262         }
1263       else {
1264         retval = MCHT_CALENDARDATE;
1265         lpht->st.wMonth = infoPtr->currentMonth;
1266         lpht->st.wDay   = day;
1267         lpht->st.wDayOfWeek   = MONTHCAL_CalculateDayOfWeek(day,lpht->st.wMonth,lpht->st.wYear);
1268       }
1269       goto done;
1270     }
1271   if(PtInRect(&infoPtr->todayrect, lpht->pt)) {
1272     retval = MCHT_TODAYLINK;
1273     goto done;
1274   }
1275
1276
1277   /* Hit nothing special? What's left must be background :-) */
1278
1279   retval = MCHT_CALENDARBK;
1280  done:
1281   lpht->uHit = retval;
1282   return retval;
1283 }
1284
1285
1286 static void MONTHCAL_GoToNextMonth(MONTHCAL_INFO *infoPtr)
1287 {
1288   DWORD dwStyle = GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE);
1289
1290   TRACE("MONTHCAL_GoToNextMonth\n");
1291
1292   infoPtr->currentMonth++;
1293   if(infoPtr->currentMonth > 12) {
1294     infoPtr->currentYear++;
1295     infoPtr->currentMonth = 1;
1296   }
1297
1298   if(dwStyle & MCS_DAYSTATE) {
1299     NMDAYSTATE nmds;
1300     int i;
1301
1302     nmds.nmhdr.hwndFrom = infoPtr->hwndSelf;
1303     nmds.nmhdr.idFrom   = GetWindowLongPtrW(infoPtr->hwndSelf, GWLP_ID);
1304     nmds.nmhdr.code     = MCN_GETDAYSTATE;
1305     nmds.cDayState      = infoPtr->monthRange;
1306     nmds.prgDayState    = Alloc(infoPtr->monthRange * sizeof(MONTHDAYSTATE));
1307
1308     SendMessageW(infoPtr->hwndNotify, WM_NOTIFY,
1309     (WPARAM)nmds.nmhdr.idFrom, (LPARAM)&nmds);
1310     for(i=0; i<infoPtr->monthRange; i++)
1311       infoPtr->monthdayState[i] = nmds.prgDayState[i];
1312   }
1313 }
1314
1315
1316 static void MONTHCAL_GoToPrevMonth(MONTHCAL_INFO *infoPtr)
1317 {
1318   DWORD dwStyle = GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE);
1319
1320   TRACE("\n");
1321
1322   infoPtr->currentMonth--;
1323   if(infoPtr->currentMonth < 1) {
1324     infoPtr->currentYear--;
1325     infoPtr->currentMonth = 12;
1326   }
1327
1328   if(dwStyle & MCS_DAYSTATE) {
1329     NMDAYSTATE nmds;
1330     int i;
1331
1332     nmds.nmhdr.hwndFrom = infoPtr->hwndSelf;
1333     nmds.nmhdr.idFrom   = GetWindowLongPtrW(infoPtr->hwndSelf, GWLP_ID);
1334     nmds.nmhdr.code     = MCN_GETDAYSTATE;
1335     nmds.cDayState      = infoPtr->monthRange;
1336     nmds.prgDayState    = Alloc
1337                         (infoPtr->monthRange * sizeof(MONTHDAYSTATE));
1338
1339     SendMessageW(infoPtr->hwndNotify, WM_NOTIFY,
1340         (WPARAM)nmds.nmhdr.idFrom, (LPARAM)&nmds);
1341     for(i=0; i<infoPtr->monthRange; i++)
1342        infoPtr->monthdayState[i] = nmds.prgDayState[i];
1343   }
1344 }
1345
1346 static LRESULT
1347 MONTHCAL_RButtonDown(MONTHCAL_INFO *infoPtr, LPARAM lParam)
1348 {
1349   static const WCHAR todayW[] = { 'G','o',' ','t','o',' ','T','o','d','a','y',':',0 };
1350   HMENU hMenu;
1351   POINT menupoint;
1352   WCHAR buf[32];
1353
1354   hMenu = CreatePopupMenu();
1355   if (!LoadStringW(COMCTL32_hModule,IDM_GOTODAY,buf,countof(buf)))
1356     {
1357       WARN("Can't load resource\n");
1358       strcpyW(buf, todayW);
1359     }
1360   AppendMenuW(hMenu, MF_STRING|MF_ENABLED,1, buf);
1361   menupoint.x=(short)LOWORD(lParam);
1362   menupoint.y=(short)HIWORD(lParam);
1363   ClientToScreen(infoPtr->hwndSelf, &menupoint);
1364   if( TrackPopupMenu(hMenu,TPM_RIGHTBUTTON| TPM_NONOTIFY|TPM_RETURNCMD,
1365                      menupoint.x, menupoint.y, 0, infoPtr->hwndSelf, NULL))
1366     {
1367       infoPtr->currentMonth=infoPtr->todaysDate.wMonth;
1368       infoPtr->currentYear=infoPtr->todaysDate.wYear;
1369       InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1370     }
1371   return 0;
1372 }
1373
1374 static LRESULT
1375 MONTHCAL_LButtonDown(MONTHCAL_INFO *infoPtr, LPARAM lParam)
1376 {
1377   static const WCHAR EditW[] = { 'E','D','I','T',0 };
1378   MCHITTESTINFO ht;
1379   DWORD hit;
1380   HMENU hMenu;
1381   RECT rcDay; /* used in determining area to invalidate */
1382   WCHAR buf[32];
1383   int i;
1384   POINT menupoint;
1385
1386   TRACE("%lx\n", lParam);
1387
1388   if (infoPtr->hWndYearUpDown)
1389     {
1390       infoPtr->currentYear=SendMessageW( infoPtr->hWndYearUpDown, UDM_SETPOS,   (WPARAM) 0,(LPARAM)0);
1391       if(!DestroyWindow(infoPtr->hWndYearUpDown))
1392         {
1393           FIXME("Can't destroy Updown Control\n");
1394         }
1395       else
1396         infoPtr->hWndYearUpDown=0;
1397       if(!DestroyWindow(infoPtr->hWndYearEdit))
1398         {
1399           FIXME("Can't destroy Updown Control\n");
1400         }
1401       else
1402         infoPtr->hWndYearEdit=0;
1403       InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1404     }
1405
1406   ht.pt.x = (short)LOWORD(lParam);
1407   ht.pt.y = (short)HIWORD(lParam);
1408   hit = MONTHCAL_HitTest(infoPtr, (LPARAM)&ht);
1409
1410   /* FIXME: these flags should be checked by */
1411   /*((hit & MCHT_XXX) == MCHT_XXX) b/c some of the flags are */
1412   /* multi-bit */
1413   if(hit ==MCHT_TITLEBTNNEXT) {
1414     MONTHCAL_GoToNextMonth(infoPtr);
1415     infoPtr->status = MC_NEXTPRESSED;
1416     SetTimer(infoPtr->hwndSelf, MC_NEXTMONTHTIMER, MC_NEXTMONTHDELAY, 0);
1417     InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1418     return TRUE;
1419   }
1420   if(hit == MCHT_TITLEBTNPREV){
1421     MONTHCAL_GoToPrevMonth(infoPtr);
1422     infoPtr->status = MC_PREVPRESSED;
1423     SetTimer(infoPtr->hwndSelf, MC_PREVMONTHTIMER, MC_NEXTMONTHDELAY, 0);
1424     InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1425     return TRUE;
1426   }
1427
1428   if(hit == MCHT_TITLEMONTH) {
1429     hMenu = CreatePopupMenu();
1430
1431     for (i=0; i<12;i++)
1432       {
1433         GetLocaleInfoW(LOCALE_USER_DEFAULT,LOCALE_SMONTHNAME1+i, buf,countof(buf));
1434         AppendMenuW(hMenu, MF_STRING|MF_ENABLED,i+1, buf);
1435       }
1436     menupoint.x=infoPtr->titlemonth.right;
1437     menupoint.y=infoPtr->titlemonth.bottom;
1438     ClientToScreen(infoPtr->hwndSelf, &menupoint);
1439     i= TrackPopupMenu(hMenu,TPM_LEFTALIGN | TPM_NONOTIFY | TPM_RIGHTBUTTON | TPM_RETURNCMD,
1440                       menupoint.x, menupoint.y, 0, infoPtr->hwndSelf, NULL);
1441     if ((i>0) && (i<13))
1442       {
1443         infoPtr->currentMonth=i;
1444         InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1445       }
1446   }
1447   if(hit == MCHT_TITLEYEAR) {
1448     infoPtr->hWndYearEdit=CreateWindowExW(0,
1449                          EditW,
1450                            0,
1451                          WS_VISIBLE | WS_CHILD |UDS_SETBUDDYINT,
1452                          infoPtr->titleyear.left+3,infoPtr->titlebtnnext.top,
1453                          infoPtr->titleyear.right-infoPtr->titleyear.left+4,
1454                          infoPtr->textHeight,
1455                          infoPtr->hwndSelf,
1456                          NULL,
1457                          NULL,
1458                          NULL);
1459     SendMessageW( infoPtr->hWndYearEdit, WM_SETFONT, (WPARAM) infoPtr->hBoldFont, (LPARAM)TRUE);
1460     infoPtr->hWndYearUpDown=CreateWindowExW(0,
1461                          UPDOWN_CLASSW,
1462                            0,
1463                          WS_VISIBLE | WS_CHILD |UDS_SETBUDDYINT|UDS_NOTHOUSANDS|UDS_ARROWKEYS,
1464                          infoPtr->titleyear.right+7,infoPtr->titlebtnnext.top,
1465                          18,
1466                          infoPtr->textHeight,
1467                          infoPtr->hwndSelf,
1468                          NULL,
1469                          NULL,
1470                          NULL);
1471     SendMessageW( infoPtr->hWndYearUpDown, UDM_SETRANGE, (WPARAM) 0, MAKELONG (9999, 1753));
1472     SendMessageW( infoPtr->hWndYearUpDown, UDM_SETBUDDY, (WPARAM) infoPtr->hWndYearEdit, (LPARAM)0 );
1473     SendMessageW( infoPtr->hWndYearUpDown, UDM_SETPOS,   (WPARAM) 0,(LPARAM)infoPtr->currentYear );
1474     return TRUE;
1475
1476   }
1477   if(hit == MCHT_TODAYLINK) {
1478     infoPtr->currentMonth=infoPtr->todaysDate.wMonth;
1479     infoPtr->currentYear=infoPtr->todaysDate.wYear;
1480     InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1481     return TRUE;
1482   }
1483   if(hit == MCHT_CALENDARDATE) {
1484     SYSTEMTIME selArray[2];
1485     NMSELCHANGE nmsc;
1486
1487     MONTHCAL_CopyTime(&ht.st, &selArray[0]);
1488     MONTHCAL_CopyTime(&ht.st, &selArray[1]);
1489     MONTHCAL_SetSelRange(infoPtr, (LPARAM)&selArray);
1490     MONTHCAL_SetCurSel(infoPtr, (LPARAM)&selArray);
1491     TRACE("MCHT_CALENDARDATE\n");
1492     nmsc.nmhdr.hwndFrom = infoPtr->hwndSelf;
1493     nmsc.nmhdr.idFrom   = GetWindowLongPtrW(infoPtr->hwndSelf, GWLP_ID);
1494     nmsc.nmhdr.code     = MCN_SELCHANGE;
1495     MONTHCAL_CopyTime(&infoPtr->minSel,&nmsc.stSelStart);
1496     MONTHCAL_CopyTime(&infoPtr->maxSel,&nmsc.stSelEnd);
1497
1498     SendMessageW(infoPtr->hwndNotify, WM_NOTIFY,
1499            (WPARAM)nmsc.nmhdr.idFrom,(LPARAM)&nmsc);
1500
1501
1502     /* redraw both old and new days if the selected day changed */
1503     if(infoPtr->curSelDay != ht.st.wDay) {
1504       MONTHCAL_CalcPosFromDay(infoPtr, ht.st.wDay, ht.st.wMonth, &rcDay);
1505       InvalidateRect(infoPtr->hwndSelf, &rcDay, TRUE);
1506
1507       MONTHCAL_CalcPosFromDay(infoPtr, infoPtr->curSelDay, infoPtr->currentMonth, &rcDay);
1508       InvalidateRect(infoPtr->hwndSelf, &rcDay, TRUE);
1509     }
1510
1511     infoPtr->firstSelDay = ht.st.wDay;
1512     infoPtr->curSelDay = ht.st.wDay;
1513     infoPtr->status = MC_SEL_LBUTDOWN;
1514     return TRUE;
1515   }
1516
1517   return 0;
1518 }
1519
1520
1521 static LRESULT
1522 MONTHCAL_LButtonUp(MONTHCAL_INFO *infoPtr, LPARAM lParam)
1523 {
1524   NMSELCHANGE nmsc;
1525   NMHDR nmhdr;
1526   BOOL redraw = FALSE;
1527   MCHITTESTINFO ht;
1528   DWORD hit;
1529
1530   TRACE("\n");
1531
1532   if(infoPtr->status & MC_NEXTPRESSED) {
1533     KillTimer(infoPtr->hwndSelf, MC_NEXTMONTHTIMER);
1534     infoPtr->status &= ~MC_NEXTPRESSED;
1535     redraw = TRUE;
1536   }
1537   if(infoPtr->status & MC_PREVPRESSED) {
1538     KillTimer(infoPtr->hwndSelf, MC_PREVMONTHTIMER);
1539     infoPtr->status &= ~MC_PREVPRESSED;
1540     redraw = TRUE;
1541   }
1542
1543   ht.pt.x = (short)LOWORD(lParam);
1544   ht.pt.y = (short)HIWORD(lParam);
1545   hit = MONTHCAL_HitTest(infoPtr, (LPARAM)&ht);
1546
1547   infoPtr->status = MC_SEL_LBUTUP;
1548
1549   if(hit ==MCHT_CALENDARDATENEXT) {
1550     MONTHCAL_GoToNextMonth(infoPtr);
1551     InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1552     return TRUE;
1553   }
1554   if(hit == MCHT_CALENDARDATEPREV){
1555     MONTHCAL_GoToPrevMonth(infoPtr);
1556     InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1557     return TRUE;
1558   }
1559   nmhdr.hwndFrom = infoPtr->hwndSelf;
1560   nmhdr.idFrom   = GetWindowLongPtrW(infoPtr->hwndSelf, GWLP_ID);
1561   nmhdr.code     = NM_RELEASEDCAPTURE;
1562   TRACE("Sent notification from %p to %p\n", infoPtr->hwndSelf, infoPtr->hwndNotify);
1563
1564   SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, (WPARAM)nmhdr.idFrom, (LPARAM)&nmhdr);
1565   /* redraw if necessary */
1566   if(redraw)
1567     InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1568   /* only send MCN_SELECT if currently displayed month's day was selected */
1569   if(hit == MCHT_CALENDARDATE) {
1570     nmsc.nmhdr.hwndFrom = infoPtr->hwndSelf;
1571     nmsc.nmhdr.idFrom   = GetWindowLongPtrW(infoPtr->hwndSelf, GWLP_ID);
1572     nmsc.nmhdr.code     = MCN_SELECT;
1573     MONTHCAL_CopyTime(&infoPtr->minSel, &nmsc.stSelStart);
1574     MONTHCAL_CopyTime(&infoPtr->maxSel, &nmsc.stSelEnd);
1575
1576     SendMessageW(infoPtr->hwndNotify, WM_NOTIFY, (WPARAM)nmsc.nmhdr.idFrom, (LPARAM)&nmsc);
1577
1578   }
1579   return 0;
1580 }
1581
1582
1583 static LRESULT
1584 MONTHCAL_Timer(MONTHCAL_INFO *infoPtr, WPARAM wParam)
1585 {
1586   BOOL redraw = FALSE;
1587
1588   TRACE("%ld\n", wParam);
1589
1590   switch(wParam) {
1591   case MC_NEXTMONTHTIMER:
1592     redraw = TRUE;
1593     MONTHCAL_GoToNextMonth(infoPtr);
1594     break;
1595   case MC_PREVMONTHTIMER:
1596     redraw = TRUE;
1597     MONTHCAL_GoToPrevMonth(infoPtr);
1598     break;
1599   default:
1600     ERR("got unknown timer\n");
1601     break;
1602   }
1603
1604   /* redraw only if necessary */
1605   if(redraw)
1606     InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1607
1608   return 0;
1609 }
1610
1611
1612 static LRESULT
1613 MONTHCAL_MouseMove(MONTHCAL_INFO *infoPtr, WPARAM wParam, LPARAM lParam)
1614 {
1615   MCHITTESTINFO ht;
1616   int oldselday, selday, hit;
1617   RECT r;
1618
1619   if(!(infoPtr->status & MC_SEL_LBUTDOWN)) return 0;
1620
1621   ht.pt.x = (short)LOWORD(lParam);
1622   ht.pt.y = (short)HIWORD(lParam);
1623
1624   hit = MONTHCAL_HitTest(infoPtr, (LPARAM)&ht);
1625
1626   /* not on the calendar date numbers? bail out */
1627   TRACE("hit:%x\n",hit);
1628   if((hit & MCHT_CALENDARDATE) != MCHT_CALENDARDATE) return 0;
1629
1630   selday = ht.st.wDay;
1631   oldselday = infoPtr->curSelDay;
1632   infoPtr->curSelDay = selday;
1633   MONTHCAL_CalcPosFromDay(infoPtr, selday, ht.st. wMonth, &r);
1634
1635   if(GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE) & MCS_MULTISELECT)  {
1636     SYSTEMTIME selArray[2];
1637     int i;
1638
1639     MONTHCAL_GetSelRange(infoPtr, (LPARAM)&selArray);
1640     i = 0;
1641     if(infoPtr->firstSelDay==selArray[0].wDay) i=1;
1642     TRACE("oldRange:%d %d %d %d\n", infoPtr->firstSelDay, selArray[0].wDay, selArray[1].wDay, i);
1643     if(infoPtr->firstSelDay==selArray[1].wDay) {
1644       /* 1st time we get here: selArray[0]=selArray[1])  */
1645       /* if we're still at the first selected date, return */
1646       if(infoPtr->firstSelDay==selday) goto done;
1647       if(selday<infoPtr->firstSelDay) i = 0;
1648     }
1649
1650     if(abs(infoPtr->firstSelDay - selday) >= infoPtr->maxSelCount) {
1651       if(selday>infoPtr->firstSelDay)
1652         selday = infoPtr->firstSelDay + infoPtr->maxSelCount;
1653       else
1654         selday = infoPtr->firstSelDay - infoPtr->maxSelCount;
1655     }
1656
1657     if(selArray[i].wDay!=selday) {
1658       TRACE("newRange:%d %d %d %d\n", infoPtr->firstSelDay, selArray[0].wDay, selArray[1].wDay, i);
1659
1660       selArray[i].wDay = selday;
1661
1662       if(selArray[0].wDay>selArray[1].wDay) {
1663         DWORD tempday;
1664         tempday = selArray[1].wDay;
1665         selArray[1].wDay = selArray[0].wDay;
1666         selArray[0].wDay = tempday;
1667       }
1668
1669       MONTHCAL_SetSelRange(infoPtr, (LPARAM)&selArray);
1670     }
1671   }
1672
1673 done:
1674
1675   /* only redraw if the currently selected day changed */
1676   /* FIXME: this should specify a rectangle containing only the days that changed */
1677   /* using InvalidateRect */
1678   if(oldselday != infoPtr->curSelDay)
1679     InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1680
1681   return 0;
1682 }
1683
1684
1685 static LRESULT
1686 MONTHCAL_Paint(MONTHCAL_INFO *infoPtr, WPARAM wParam)
1687 {
1688   HDC hdc;
1689   PAINTSTRUCT ps;
1690
1691   if (wParam)
1692   {
1693     GetClientRect(infoPtr->hwndSelf, &ps.rcPaint);
1694     hdc = (HDC)wParam;
1695   }
1696   else
1697     hdc = BeginPaint(infoPtr->hwndSelf, &ps);
1698
1699   MONTHCAL_Refresh(infoPtr, hdc, &ps);
1700   if (!wParam) EndPaint(infoPtr->hwndSelf, &ps);
1701   return 0;
1702 }
1703
1704
1705 static LRESULT
1706 MONTHCAL_KillFocus(const MONTHCAL_INFO *infoPtr)
1707 {
1708   TRACE("\n");
1709
1710   InvalidateRect(infoPtr->hwndSelf, NULL, TRUE);
1711
1712   return 0;
1713 }
1714
1715
1716 static LRESULT
1717 MONTHCAL_SetFocus(const MONTHCAL_INFO *infoPtr)
1718 {
1719   TRACE("\n");
1720
1721   InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1722
1723   return 0;
1724 }
1725
1726 /* sets the size information */
1727 static void MONTHCAL_UpdateSize(MONTHCAL_INFO *infoPtr)
1728 {
1729   static const WCHAR SunW[] = { 'S','u','n',0 };
1730   static const WCHAR O0W[] = { '0','0',0 };
1731   HDC hdc = GetDC(infoPtr->hwndSelf);
1732   RECT *title=&infoPtr->title;
1733   RECT *prev=&infoPtr->titlebtnprev;
1734   RECT *next=&infoPtr->titlebtnnext;
1735   RECT *titlemonth=&infoPtr->titlemonth;
1736   RECT *titleyear=&infoPtr->titleyear;
1737   RECT *wdays=&infoPtr->wdays;
1738   RECT *weeknumrect=&infoPtr->weeknums;
1739   RECT *days=&infoPtr->days;
1740   RECT *todayrect=&infoPtr->todayrect;
1741   SIZE size;
1742   TEXTMETRICW tm;
1743   DWORD dwStyle = GetWindowLongW(infoPtr->hwndSelf, GWL_STYLE);
1744   HFONT currentFont;
1745   int xdiv, left_offset;
1746   RECT rcClient;
1747
1748   GetClientRect(infoPtr->hwndSelf, &rcClient);
1749
1750   currentFont = SelectObject(hdc, infoPtr->hFont);
1751
1752   /* get the height and width of each day's text */
1753   GetTextMetricsW(hdc, &tm);
1754   infoPtr->textHeight = tm.tmHeight + tm.tmExternalLeading + tm.tmInternalLeading;
1755   GetTextExtentPoint32W(hdc, SunW, 3, &size);
1756   infoPtr->textWidth = size.cx + 2;
1757
1758   /* recalculate the height and width increments and offsets */
1759   GetTextExtentPoint32W(hdc, O0W, 2, &size);
1760
1761   xdiv = (dwStyle & MCS_WEEKNUMBERS) ? 8 : 7;
1762
1763   infoPtr->width_increment = size.cx * 2 + 4;
1764   infoPtr->height_increment = infoPtr->textHeight;
1765   left_offset = (rcClient.right - rcClient.left) - (infoPtr->width_increment * xdiv);
1766
1767   /* calculate title area */
1768   title->top    = rcClient.top;
1769   title->bottom = title->top + 3 * infoPtr->height_increment / 2;
1770   title->left   = left_offset;
1771   title->right  = rcClient.right;
1772
1773   /* set the dimensions of the next and previous buttons and center */
1774   /* the month text vertically */
1775   prev->top    = next->top    = title->top + 4;
1776   prev->bottom = next->bottom = title->bottom - 4;
1777   prev->left   = title->left + 4;
1778   prev->right  = prev->left + (title->bottom - title->top) ;
1779   next->right  = title->right - 4;
1780   next->left   = next->right - (title->bottom - title->top);
1781
1782   /* titlemonth->left and right change based upon the current month */
1783   /* and are recalculated in refresh as the current month may change */
1784   /* without the control being resized */
1785   titlemonth->top    = titleyear->top    = title->top    + (infoPtr->height_increment)/2;
1786   titlemonth->bottom = titleyear->bottom = title->bottom - (infoPtr->height_increment)/2;
1787
1788   /* setup the dimensions of the rectangle we draw the names of the */
1789   /* days of the week in */
1790   weeknumrect->left = left_offset;
1791   if(dwStyle & MCS_WEEKNUMBERS)
1792     weeknumrect->right=prev->right;
1793   else
1794     weeknumrect->right=weeknumrect->left;
1795   wdays->left   = days->left   = weeknumrect->right;
1796   wdays->right  = days->right  = wdays->left + 7 * infoPtr->width_increment;
1797   wdays->top    = title->bottom ;
1798   wdays->bottom = wdays->top + infoPtr->height_increment;
1799
1800   days->top    = weeknumrect->top = wdays->bottom ;
1801   days->bottom = weeknumrect->bottom = days->top + 6 * infoPtr->height_increment;
1802
1803   todayrect->left   = rcClient.left;
1804   todayrect->right  = rcClient.right;
1805   todayrect->top    = days->bottom;
1806   todayrect->bottom = days->bottom + infoPtr->height_increment;
1807
1808   TRACE("dx=%d dy=%d client[%s] title[%s] wdays[%s] days[%s] today[%s]\n",
1809         infoPtr->width_increment,infoPtr->height_increment,
1810         wine_dbgstr_rect(&rcClient),
1811         wine_dbgstr_rect(title),
1812         wine_dbgstr_rect(wdays),
1813         wine_dbgstr_rect(days),
1814         wine_dbgstr_rect(todayrect));
1815
1816   /* restore the originally selected font */
1817   SelectObject(hdc, currentFont);
1818
1819   ReleaseDC(infoPtr->hwndSelf, hdc);
1820 }
1821
1822 static LRESULT MONTHCAL_Size(MONTHCAL_INFO *infoPtr, int Width, int Height)
1823 {
1824   TRACE("(width=%d, height=%d)\n", Width, Height);
1825
1826   MONTHCAL_UpdateSize(infoPtr);
1827
1828   /* invalidate client area and erase background */
1829   InvalidateRect(infoPtr->hwndSelf, NULL, TRUE);
1830
1831   return 0;
1832 }
1833
1834 static LRESULT MONTHCAL_GetFont(const MONTHCAL_INFO *infoPtr)
1835 {
1836     return (LRESULT)infoPtr->hFont;
1837 }
1838
1839 static LRESULT MONTHCAL_SetFont(MONTHCAL_INFO *infoPtr, HFONT hFont, BOOL redraw)
1840 {
1841     HFONT hOldFont;
1842     LOGFONTW lf;
1843
1844     if (!hFont) return 0;
1845
1846     hOldFont = infoPtr->hFont;
1847     infoPtr->hFont = hFont;
1848
1849     GetObjectW(infoPtr->hFont, sizeof(lf), &lf);
1850     lf.lfWeight = FW_BOLD;
1851     infoPtr->hBoldFont = CreateFontIndirectW(&lf);
1852
1853     if (redraw)
1854         InvalidateRect(infoPtr->hwndSelf, NULL, FALSE);
1855
1856     return (LRESULT)hOldFont;
1857 }
1858
1859 /* update theme after a WM_THEMECHANGED message */
1860 static LRESULT theme_changed (const MONTHCAL_INFO* infoPtr)
1861 {
1862     HTHEME theme = GetWindowTheme (infoPtr->hwndSelf);
1863     CloseThemeData (theme);
1864     theme = OpenThemeData (infoPtr->hwndSelf, themeClass);
1865     return 0;
1866 }
1867
1868 /* FIXME: check whether dateMin/dateMax need to be adjusted. */
1869 static LRESULT
1870 MONTHCAL_Create(HWND hwnd, WPARAM wParam, LPARAM lParam)
1871 {
1872   MONTHCAL_INFO *infoPtr;
1873
1874   /* allocate memory for info structure */
1875   infoPtr =(MONTHCAL_INFO*)Alloc(sizeof(MONTHCAL_INFO));
1876   SetWindowLongPtrW(hwnd, 0, (DWORD_PTR)infoPtr);
1877
1878   if(infoPtr == NULL) {
1879     ERR( "could not allocate info memory!\n");
1880     return 0;
1881   }
1882
1883   infoPtr->hwndSelf = hwnd;
1884   infoPtr->hwndNotify = ((LPCREATESTRUCTW)lParam)->hwndParent;
1885
1886   MONTHCAL_SetFont(infoPtr, GetStockObject(DEFAULT_GUI_FONT), FALSE);
1887
1888   /* initialize info structure */
1889   /* FIXME: calculate systemtime ->> localtime(substract timezoneinfo) */
1890
1891   GetLocalTime(&infoPtr->todaysDate);
1892   infoPtr->firstDayHighWord = FALSE;
1893   MONTHCAL_SetFirstDayOfWeek(infoPtr, (LPARAM)-1);
1894   infoPtr->currentMonth = infoPtr->todaysDate.wMonth;
1895   infoPtr->currentYear = infoPtr->todaysDate.wYear;
1896   MONTHCAL_CopyTime(&infoPtr->todaysDate, &infoPtr->minDate);
1897   MONTHCAL_CopyTime(&infoPtr->todaysDate, &infoPtr->maxDate);
1898   infoPtr->maxDate.wYear=2050;
1899   infoPtr->minDate.wYear=1950;
1900   infoPtr->maxSelCount  = 7;
1901   infoPtr->monthRange = 3;
1902   infoPtr->monthdayState = Alloc
1903                          (infoPtr->monthRange * sizeof(MONTHDAYSTATE));
1904   infoPtr->titlebk     = GetSysColor(COLOR_ACTIVECAPTION);
1905   infoPtr->titletxt    = GetSysColor(COLOR_WINDOW);
1906   infoPtr->monthbk     = GetSysColor(COLOR_WINDOW);
1907   infoPtr->trailingtxt = GetSysColor(COLOR_GRAYTEXT);
1908   infoPtr->bk          = GetSysColor(COLOR_WINDOW);
1909   infoPtr->txt         = GetSysColor(COLOR_WINDOWTEXT);
1910
1911   /* set the current day for highlighing */
1912   infoPtr->minSel.wDay = infoPtr->todaysDate.wDay;
1913   infoPtr->maxSel.wDay = infoPtr->todaysDate.wDay;
1914
1915   /* call MONTHCAL_UpdateSize to set all of the dimensions */
1916   /* of the control */
1917   MONTHCAL_UpdateSize(infoPtr);
1918   
1919   OpenThemeData (infoPtr->hwndSelf, themeClass);
1920
1921   return 0;
1922 }
1923
1924
1925 static LRESULT
1926 MONTHCAL_Destroy(MONTHCAL_INFO *infoPtr)
1927 {
1928   /* free month calendar info data */
1929   Free(infoPtr->monthdayState);
1930   SetWindowLongPtrW(infoPtr->hwndSelf, 0, 0);
1931
1932   CloseThemeData (GetWindowTheme (infoPtr->hwndSelf));
1933   
1934   Free(infoPtr);
1935   return 0;
1936 }
1937
1938
1939 static LRESULT WINAPI
1940 MONTHCAL_WindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
1941 {
1942   MONTHCAL_INFO *infoPtr;
1943
1944   TRACE("hwnd=%p msg=%x wparam=%lx lparam=%lx\n", hwnd, uMsg, wParam, lParam);
1945
1946   infoPtr = MONTHCAL_GetInfoPtr(hwnd);
1947   if (!infoPtr && (uMsg != WM_CREATE))
1948     return DefWindowProcW(hwnd, uMsg, wParam, lParam);
1949   switch(uMsg)
1950   {
1951   case MCM_GETCURSEL:
1952     return MONTHCAL_GetCurSel(infoPtr, lParam);
1953
1954   case MCM_SETCURSEL:
1955     return MONTHCAL_SetCurSel(infoPtr, lParam);
1956
1957   case MCM_GETMAXSELCOUNT:
1958     return MONTHCAL_GetMaxSelCount(infoPtr);
1959
1960   case MCM_SETMAXSELCOUNT:
1961     return MONTHCAL_SetMaxSelCount(infoPtr, wParam);
1962
1963   case MCM_GETSELRANGE:
1964     return MONTHCAL_GetSelRange(infoPtr, lParam);
1965
1966   case MCM_SETSELRANGE:
1967     return MONTHCAL_SetSelRange(infoPtr, lParam);
1968
1969   case MCM_GETMONTHRANGE:
1970     return MONTHCAL_GetMonthRange(infoPtr);
1971
1972   case MCM_SETDAYSTATE:
1973     return MONTHCAL_SetDayState(infoPtr, wParam, lParam);
1974
1975   case MCM_GETMINREQRECT:
1976     return MONTHCAL_GetMinReqRect(infoPtr, lParam);
1977
1978   case MCM_GETCOLOR:
1979     return MONTHCAL_GetColor(infoPtr, wParam);
1980
1981   case MCM_SETCOLOR:
1982     return MONTHCAL_SetColor(infoPtr, wParam, lParam);
1983
1984   case MCM_GETTODAY:
1985     return MONTHCAL_GetToday(infoPtr, lParam);
1986
1987   case MCM_SETTODAY:
1988     return MONTHCAL_SetToday(infoPtr, lParam);
1989
1990   case MCM_HITTEST:
1991     return MONTHCAL_HitTest(infoPtr, lParam);
1992
1993   case MCM_GETFIRSTDAYOFWEEK:
1994     return MONTHCAL_GetFirstDayOfWeek(infoPtr);
1995
1996   case MCM_SETFIRSTDAYOFWEEK:
1997     return MONTHCAL_SetFirstDayOfWeek(infoPtr, lParam);
1998
1999   case MCM_GETRANGE:
2000     return MONTHCAL_GetRange(hwnd, wParam, lParam);
2001
2002   case MCM_SETRANGE:
2003     return MONTHCAL_SetRange(infoPtr, wParam, lParam);
2004
2005   case MCM_GETMONTHDELTA:
2006     return MONTHCAL_GetMonthDelta(infoPtr);
2007
2008   case MCM_SETMONTHDELTA:
2009     return MONTHCAL_SetMonthDelta(infoPtr, wParam);
2010
2011   case MCM_GETMAXTODAYWIDTH:
2012     return MONTHCAL_GetMaxTodayWidth(infoPtr);
2013
2014   case WM_GETDLGCODE:
2015     return DLGC_WANTARROWS | DLGC_WANTCHARS;
2016
2017   case WM_KILLFOCUS:
2018     return MONTHCAL_KillFocus(infoPtr);
2019
2020   case WM_RBUTTONDOWN:
2021     return MONTHCAL_RButtonDown(infoPtr, lParam);
2022
2023   case WM_LBUTTONDOWN:
2024     return MONTHCAL_LButtonDown(infoPtr, lParam);
2025
2026   case WM_MOUSEMOVE:
2027     return MONTHCAL_MouseMove(infoPtr, wParam, lParam);
2028
2029   case WM_LBUTTONUP:
2030     return MONTHCAL_LButtonUp(infoPtr, lParam);
2031
2032   case WM_PRINTCLIENT:
2033   case WM_PAINT:
2034     return MONTHCAL_Paint(infoPtr, wParam);
2035
2036   case WM_SETFOCUS:
2037     return MONTHCAL_SetFocus(infoPtr);
2038
2039   case WM_SIZE:
2040     return MONTHCAL_Size(infoPtr, (SHORT)LOWORD(lParam), (SHORT)HIWORD(lParam));
2041
2042   case WM_CREATE:
2043     return MONTHCAL_Create(hwnd, wParam, lParam);
2044
2045   case WM_SETFONT:
2046     return MONTHCAL_SetFont(infoPtr, (HFONT)wParam, (BOOL)lParam);
2047
2048   case WM_GETFONT:
2049     return MONTHCAL_GetFont(infoPtr);
2050
2051   case WM_TIMER:
2052     return MONTHCAL_Timer(infoPtr, wParam);
2053     
2054   case WM_THEMECHANGED:
2055     return theme_changed (infoPtr);
2056
2057   case WM_DESTROY:
2058     return MONTHCAL_Destroy(infoPtr);
2059
2060   default:
2061     if ((uMsg >= WM_USER) && (uMsg < WM_APP))
2062       ERR( "unknown msg %04x wp=%08lx lp=%08lx\n", uMsg, wParam, lParam);
2063     return DefWindowProcW(hwnd, uMsg, wParam, lParam);
2064   }
2065 }
2066
2067
2068 void
2069 MONTHCAL_Register(void)
2070 {
2071   WNDCLASSW wndClass;
2072
2073   ZeroMemory(&wndClass, sizeof(WNDCLASSW));
2074   wndClass.style         = CS_GLOBALCLASS;
2075   wndClass.lpfnWndProc   = MONTHCAL_WindowProc;
2076   wndClass.cbClsExtra    = 0;
2077   wndClass.cbWndExtra    = sizeof(MONTHCAL_INFO *);
2078   wndClass.hCursor       = LoadCursorW(0, (LPWSTR)IDC_ARROW);
2079   wndClass.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);
2080   wndClass.lpszClassName = MONTHCAL_CLASSW;
2081
2082   RegisterClassW(&wndClass);
2083 }
2084
2085
2086 void
2087 MONTHCAL_Unregister(void)
2088 {
2089     UnregisterClassW(MONTHCAL_CLASSW, NULL);
2090 }