#include <cdk_int.h> /* * $Author: tom $ * $Date: 2016/11/20 20:06:42 $ * $Revision: 1.99 $ */ #define YEAR2INDEX(year) (((year) >= 1900) ? ((year) - 1900) : (year)) /* * Declare file local variables. */ static const char *monthsOfTheYear[] = { "NULL", "January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December" }; static int daysOfTheMonth[] = { -1, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }; /* * Declare file local prototypes. */ static int getMonthLength (int year, int month); static int getMonthStartWeekday (int year, int month); static time_t getCurrentTime (CDKCALENDAR *calendar); static void verifyCalendarDate (CDKCALENDAR *calendar); static void incrementCalendarDay (CDKCALENDAR *calendar, int adjust); static void decrementCalendarDay (CDKCALENDAR *calendar, int adjust); static void incrementCalendarMonth (CDKCALENDAR *calendar, int adjust); static void decrementCalendarMonth (CDKCALENDAR *calendar, int adjust); static void incrementCalendarYear (CDKCALENDAR *calendar, int adjust); static void decrementCalendarYear (CDKCALENDAR *calendar, int adjust); static void drawCDKCalendarField (CDKCALENDAR *calendar); DeclareCDKObjects (CALENDAR, Calendar, setCdk, Int); /* * This creates a calendar widget. */ CDKCALENDAR *newCDKCalendar (CDKSCREEN *cdkscreen, int xplace, int yplace, const char *title, int day, int month, int year, chtype dayAttrib, chtype monthAttrib, chtype yearAttrib, chtype highlight, boolean Box, boolean shadow) { /* *INDENT-EQLS* */ CDKCALENDAR *calendar = 0; int parentWidth = getmaxx (cdkscreen->window); int parentHeight = getmaxy (cdkscreen->window); int boxWidth = 24; int boxHeight = 11; int xpos = xplace; int ypos = yplace; int x; struct tm *dateInfo; time_t clck; const char *dayname = "Su Mo Tu We Th Fr Sa "; /* *INDENT-OFF* */ static const struct { int from; int to; } bindings[] = { { 'T', KEY_HOME }, { 't', KEY_HOME }, { 'n', KEY_NPAGE }, { CDK_FORCHAR, KEY_NPAGE }, { 'p', KEY_PPAGE }, { CDK_BACKCHAR, KEY_PPAGE }, }; /* *INDENT-ON* */ if ((calendar = newCDKObject (CDKCALENDAR, &my_funcs)) == 0) return (0); setCDKCalendarBox (calendar, Box); boxWidth = setCdkTitle (ObjOf (calendar), title, boxWidth); boxHeight += TitleLinesOf (calendar); /* * Make sure we didn't extend beyond the dimensions of the window. */ boxWidth = (boxWidth > parentWidth ? parentWidth : boxWidth); boxHeight = (boxHeight > parentHeight ? parentHeight : boxHeight); /* Rejustify the x and y positions if we need to. */ alignxy (cdkscreen->window, &xpos, &ypos, boxWidth, boxHeight); /* Create the calendar window. */ calendar->win = newwin (boxHeight, boxWidth, ypos, xpos); /* Is the window null? */ if (calendar->win == 0) { destroyCDKObject (calendar); return (0); } keypad (calendar->win, TRUE); /* Set some variables. */ calendar->xOffset = (boxWidth - 20) / 2; calendar->fieldWidth = boxWidth - 2 * (1 + BorderOf (calendar)); /* Set months and days names */ for (x = 0; x < MAX_MONTHS; x++) { calendar->MonthName[x] = copyChar (monthsOfTheYear[x]); } calendar->DayName = copyChar (dayname); /* *INDENT-EQLS* Set the rest of the widget values. */ ScreenOf (calendar) = cdkscreen; calendar->parent = cdkscreen->window; calendar->shadowWin = 0; calendar->xpos = xpos; calendar->ypos = ypos; calendar->boxWidth = boxWidth; calendar->boxHeight = boxHeight; calendar->day = day; calendar->month = month; calendar->year = year; calendar->dayAttrib = dayAttrib; calendar->monthAttrib = monthAttrib; calendar->yearAttrib = yearAttrib; calendar->highlight = highlight; calendar->width = boxWidth; initExitType (calendar); ObjOf (calendar)->acceptsFocus = TRUE; ObjOf (calendar)->inputWindow = calendar->win; calendar->shadow = shadow; calendar->labelWin = subwin (calendar->win, 1, calendar->fieldWidth, ypos + TitleLinesOf (calendar) + 1, xpos + 1 + BorderOf (calendar)); if (calendar->labelWin == 0) { destroyCDKObject (calendar); return (0); } calendar->fieldWin = subwin (calendar->win, 7, 20, ypos + TitleLinesOf (calendar) + 3, xpos + calendar->xOffset); if (calendar->fieldWin == 0) { destroyCDKObject (calendar); return (0); } setCDKCalendarBox (calendar, Box); calendar->marker = typeCallocN (chtype, CALENDAR_LIMIT); if (calendar->marker == 0) { destroyCDKObject (calendar); return (0); } /* If the day/month/year values were 0, then use today's date. */ if ((calendar->day == 0) && (calendar->month == 0) && (calendar->year == 0)) { time (&clck); dateInfo = gmtime (&clck); /* *INDENT-EQLS* */ calendar->day = dateInfo->tm_mday; calendar->month = dateInfo->tm_mon + 1; calendar->year = dateInfo->tm_year + 1900; } /* Verify the dates provided. */ verifyCalendarDate (calendar); /* Determine which day the month starts on. */ calendar->weekDay = getMonthStartWeekday (calendar->year, calendar->month); /* If a shadow was requested, then create the shadow window. */ if (shadow) { calendar->shadowWin = newwin (boxHeight, boxWidth, ypos + 1, xpos + 1); } /* Setup the key bindings. */ for (x = 0; x < (int)SIZEOF (bindings); ++x) bindCDKObject (vCALENDAR, calendar, (chtype)bindings[x].from, getcCDKBind, (void *)(long)bindings[x].to); registerCDKObject (cdkscreen, vCALENDAR, calendar); return (calendar); } /* * This function lets the user play with this widget. */ time_t activateCDKCalendar (CDKCALENDAR *calendar, chtype *actions) { /* *INDENT-EQLS* */ chtype input = 0; boolean functionKey; time_t ret = -1; /* Draw the widget. */ drawCDKCalendar (calendar, ObjOf (calendar)->box); if (actions == 0) { for (;;) { input = (chtype)getchCDKObject (ObjOf (calendar), &functionKey); /* Inject the character into the widget. */ ret = injectCDKCalendar (calendar, input); if (calendar->exitType != vEARLY_EXIT) { return ret; } } } else { int length = chlen (actions); int x = 0; /* Inject each character one at a time. */ for (x = 0; x < length; x++) { ret = injectCDKCalendar (calendar, actions[x]); if (calendar->exitType != vEARLY_EXIT) { return ret; } } } return ret; } /* * This injects a single character into the widget. */ static int _injectCDKCalendar (CDKOBJS *object, chtype input) { CDKCALENDAR *widget = (CDKCALENDAR *)object; /* Declare local variables. */ int ppReturn = 1; int ret = unknownInt; bool complete = FALSE; /* Set the exit type. */ setExitType (widget, 0); /* Refresh the widget field. */ drawCDKCalendarField (widget); /* Check if there is a pre-process function to be called. */ if (PreProcessFuncOf (widget) != 0) { /* Call the pre-process function. */ ppReturn = PreProcessFuncOf (widget) (vCALENDAR, widget, PreProcessDataOf (widget), input); } /* Should we continue? */ if (ppReturn != 0) { /* Check a predefined binding. */ if (checkCDKObjectBind (vCALENDAR, widget, input) != 0) { checkEarlyExit (widget); complete = TRUE; } else { switch (input) { case KEY_UP: decrementCalendarDay (widget, 7); break; case KEY_DOWN: incrementCalendarDay (widget, 7); break; case KEY_LEFT: decrementCalendarDay (widget, 1); break; case KEY_RIGHT: incrementCalendarDay (widget, 1); break; case KEY_NPAGE: incrementCalendarMonth (widget, 1); break; case 'N': incrementCalendarMonth (widget, 6); break; case KEY_PPAGE: decrementCalendarMonth (widget, 1); break; case 'P': decrementCalendarMonth (widget, 6); break; case '-': decrementCalendarYear (widget, 1); break; case '+': incrementCalendarYear (widget, 1); break; case KEY_HOME: setCDKCalendarDate (widget, -1, -1, -1); break; case KEY_ESC: setExitType (widget, input); complete = TRUE; break; case KEY_ERROR: setExitType (widget, input); complete = TRUE; break; case KEY_TAB: case KEY_ENTER: setExitType (widget, input); ret = (int)getCurrentTime (widget); complete = TRUE; break; case CDK_REFRESH: eraseCDKScreen (ScreenOf (widget)); refreshCDKScreen (ScreenOf (widget)); break; } } /* Should we do a post-process? */ if (!complete && (PostProcessFuncOf (widget) != 0)) { PostProcessFuncOf (widget) (vCALENDAR, widget, PostProcessDataOf (widget), input); } } if (!complete) { setExitType (widget, 0); } ResultOf (widget).valueInt = ret; return (ret != unknownInt); } /* * This moves the calendar field to the given location. */ static void _moveCDKCalendar (CDKOBJS *object, int xplace, int yplace, boolean relative, boolean refresh_flag) { CDKCALENDAR *calendar = (CDKCALENDAR *)object; /* *INDENT-EQLS* */ int currentX = getbegx (calendar->win); int currentY = getbegy (calendar->win); int xpos = xplace; int ypos = yplace; int xdiff = 0; int ydiff = 0; /* * If this is a relative move, then we will adjust where we want * to move to. */ if (relative) { xpos = getbegx (calendar->win) + xplace; ypos = getbegy (calendar->win) + yplace; } /* Adjust the window if we need to. */ alignxy (WindowOf (calendar), &xpos, &ypos, calendar->boxWidth, calendar->boxHeight); /* Get the difference. */ xdiff = currentX - xpos; ydiff = currentY - ypos; /* Move the window to the new location. */ moveCursesWindow (calendar->win, -xdiff, -ydiff); moveCursesWindow (calendar->fieldWin, -xdiff, -ydiff); moveCursesWindow (calendar->labelWin, -xdiff, -ydiff); moveCursesWindow (calendar->shadowWin, -xdiff, -ydiff); /* Touch the windows so they 'move'. */ refreshCDKWindow (WindowOf (calendar)); /* Redraw the window, if they asked for it. */ if (refresh_flag) { drawCDKCalendar (calendar, ObjOf (calendar)->box); } } /* * This draws the calendar widget. */ static void _drawCDKCalendar (CDKOBJS *object, boolean Box) { CDKCALENDAR *calendar = (CDKCALENDAR *)object; int headerLen = (int)strlen (calendar->DayName); int colLen = (6 + headerLen) / 7; int col; /* Is there a shadow? */ if (calendar->shadowWin != 0) { drawShadow (calendar->shadowWin); } /* Box the widget if asked. */ if (Box) { drawObjBox (calendar->win, ObjOf (calendar)); } drawCdkTitle (calendar->win, object); /* Draw in the day-of-the-week header. */ for (col = 0; col < 7; ++col) { int src = colLen * ((col + (calendar->weekBase % 7)) % 7); int dst = colLen * col; writeChar (calendar->win, calendar->xOffset + dst, TitleLinesOf (calendar) + 2, calendar->DayName + src, HORIZONTAL, 0, colLen); } wrefresh (calendar->win); drawCDKCalendarField (calendar); } /* * This draws the month field. */ static void drawCDKCalendarField (CDKCALENDAR *calendar) { /* *INDENT-EQLS* */ char *monthName = calendar->MonthName[calendar->month]; int monthLength = getMonthLength (calendar->year, calendar->month); int yearIndex = YEAR2INDEX (calendar->year); int day; int row, col; int save_y = -1; int save_x = -1; char temp[30]; day = (1 - calendar->weekDay + (calendar->weekBase % 7)); if (day > 0) day -= 7; for (row = 1; row <= 6; row++) { for (col = 0; col < 7; col++) { if (day >= 1 && day <= monthLength) { int xpos = col * 3; int ypos = row; chtype marker = calendar->dayAttrib; sprintf (temp, "%02d", day); if (calendar->day == day) { marker = calendar->highlight; save_y = (ypos + getbegy (calendar->fieldWin) - getbegy (InputWindowOf (calendar))); save_x = 1; } else { marker |= getCDKCalendarMarker (calendar, day, calendar->month, yearIndex); } writeCharAttrib (calendar->fieldWin, xpos, ypos, temp, marker, HORIZONTAL, 0, 2); } day++; } } wrefresh (calendar->fieldWin); /* Draw the month in. */ if (calendar->labelWin != 0) { int yearLen = 0; sprintf (temp, "%s %d,", monthName, calendar->day); writeChar (calendar->labelWin, 0, 0, temp, HORIZONTAL, 0, (int)strlen (temp)); wclrtoeol (calendar->labelWin); /* Draw the year in. */ sprintf (temp, "%d", calendar->year); yearLen = (int)strlen (temp); writeChar (calendar->labelWin, calendar->fieldWidth - yearLen, 0, temp, HORIZONTAL, 0, yearLen); wmove (calendar->labelWin, 0, 0); wrefresh (calendar->labelWin); } else if (save_y >= 0) { wmove (InputWindowOf (calendar), save_y, save_x); wrefresh (InputWindowOf (calendar)); } } /* * This sets multiple attributes of the widget. */ void setCDKCalendar (CDKCALENDAR *calendar, int day, int month, int year, chtype dayAttrib, chtype monthAttrib, chtype yearAttrib, chtype highlight, boolean Box) { setCDKCalendarDate (calendar, day, month, year); setCDKCalendarDayAttribute (calendar, dayAttrib); setCDKCalendarMonthAttribute (calendar, monthAttrib); setCDKCalendarYearAttribute (calendar, yearAttrib); setCDKCalendarHighlight (calendar, highlight); setCDKCalendarBox (calendar, Box); } /* * This sets the date and some attributes. */ void setCDKCalendarDate (CDKCALENDAR *calendar, int day, int month, int year) { /* Declare local variables. */ struct tm *dateInfo; time_t clck; /* * Get the current dates and set the default values for * the day/month/year values for the calendar. */ time (&clck); dateInfo = gmtime (&clck); /* Set the date elements if we need too. */ /* *INDENT-EQLS* */ calendar->day = (day == -1 ? dateInfo->tm_mday : day); calendar->month = (month == -1 ? dateInfo->tm_mon + 1 : month); calendar->year = (year == -1 ? dateInfo->tm_year + 1900 : year); /* Verify the date information. */ verifyCalendarDate (calendar); /* Get the start of the current month. */ calendar->weekDay = getMonthStartWeekday (calendar->year, calendar->month); } /* * This returns the current date on the calendar. */ void getCDKCalendarDate (CDKCALENDAR *calendar, int *day, int *month, int *year) { /* *INDENT-EQLS* */ (*day) = calendar->day; (*month) = calendar->month; (*year) = calendar->year; } /* * This sets the attribute of the days in the calendar. */ void setCDKCalendarDayAttribute (CDKCALENDAR *calendar, chtype attribute) { calendar->dayAttrib = attribute; } chtype getCDKCalendarDayAttribute (CDKCALENDAR *calendar) { return calendar->dayAttrib; } /* * This sets the attribute of the month names in the calendar. */ void setCDKCalendarMonthAttribute (CDKCALENDAR *calendar, chtype attribute) { calendar->monthAttrib = attribute; } chtype getCDKCalendarMonthAttribute (CDKCALENDAR *calendar) { return calendar->monthAttrib; } /* * This sets the attribute of the year in the calendar. */ void setCDKCalendarYearAttribute (CDKCALENDAR *calendar, chtype attribute) { calendar->yearAttrib = attribute; } chtype getCDKCalendarYearAttribute (CDKCALENDAR *calendar) { return calendar->yearAttrib; } /* * This sets the attribute of the highlight box. */ void setCDKCalendarHighlight (CDKCALENDAR *calendar, chtype highlight) { calendar->highlight = highlight; } chtype getCDKCalendarHighlight (CDKCALENDAR *calendar) { return calendar->highlight; } /* * This sets the box attibute of the widget. */ void setCDKCalendarBox (CDKCALENDAR *calendar, boolean Box) { ObjOf (calendar)->box = Box; ObjOf (calendar)->borderSize = Box ? 1 : 0; } boolean getCDKCalendarBox (CDKCALENDAR *calendar) { return ObjOf (calendar)->box; } /* * This sets the background attribute of the widget. */ static void _setBKattrCalendar (CDKOBJS *object, chtype attrib) { if (object != 0) { CDKCALENDAR *widget = (CDKCALENDAR *)object; wbkgd (widget->win, attrib); wbkgd (widget->fieldWin, attrib); if (widget->labelWin != 0) { wbkgd (widget->labelWin, attrib); } } } /* * This erases the calendar widget. */ static void _eraseCDKCalendar (CDKOBJS *object) { if (validCDKObject (object)) { CDKCALENDAR *calendar = (CDKCALENDAR *)object; eraseCursesWindow (calendar->labelWin); eraseCursesWindow (calendar->fieldWin); eraseCursesWindow (calendar->win); eraseCursesWindow (calendar->shadowWin); } } /* * This destroys the calendar object pointer. */ static void _destroyCDKCalendar (CDKOBJS *object) { if (object != 0) { CDKCALENDAR *calendar = (CDKCALENDAR *)object; int x; cleanCdkTitle (object); freeChar (calendar->DayName); for (x = 0; x < MAX_MONTHS; x++) { freeChar (calendar->MonthName[x]); } freeChecked (calendar->marker); /* Free up the window pointers. */ deleteCursesWindow (calendar->labelWin); deleteCursesWindow (calendar->fieldWin); deleteCursesWindow (calendar->shadowWin); deleteCursesWindow (calendar->win); /* Clean the key bindings. */ cleanCDKObjectBindings (vCALENDAR, calendar); /* Unregister the object. */ unregisterCDKObject (vCALENDAR, calendar); } } /* * This sets a marker on the calendar. */ void setCDKCalendarMarker (CDKCALENDAR *calendar, int day, int month, int year, chtype marker) { int yearIndex = YEAR2INDEX (year); chtype oldmarker = getCDKCalendarMarker (calendar, day, month, year); /* Check to see if a marker has not already been set. */ if (oldmarker != 0) { CALENDAR_CELL (calendar, day, month, yearIndex) = oldmarker | A_BLINK; } else { CALENDAR_CELL (calendar, day, month, yearIndex) = marker; } } chtype getCDKCalendarMarker (CDKCALENDAR *calendar, int day, int month, int year) { chtype result = 0; year = YEAR2INDEX (year); if (calendar->marker != 0) result = CALENDAR_CELL (calendar, day, month, year); return result; } /* * This sets a marker on the calendar. */ void removeCDKCalendarMarker (CDKCALENDAR *calendar, int day, int month, int year) { int yearIndex = YEAR2INDEX (year); CALENDAR_CELL (calendar, day, month, yearIndex) = 0; } /* * This function sets the month name. */ void setCDKCalendarMonthsNames (CDKCALENDAR *calendar, CDK_CSTRING2 months) { int x; for (x = 1; x < MAX_MONTHS; x++) { freeChar (calendar->MonthName[x]); calendar->MonthName[x] = copyChar (months[x]); } } /* * This function sets the day's name. */ void setCDKCalendarDaysNames (CDKCALENDAR *calendar, const char *days) { freeChar (calendar->DayName); calendar->DayName = copyChar (days); } /* * This makes sure that the dates provided exist. */ static void verifyCalendarDate (CDKCALENDAR *calendar) { int monthLength; /* Make sure the given year is not less than 1900. */ if (calendar->year < 1900) { calendar->year = 1900; } /* Make sure the month is within range. */ if (calendar->month > 12) { calendar->month = 12; } if (calendar->month < 1) { calendar->month = 1; } /* Make sure the day given is within range of the month. */ monthLength = getMonthLength (calendar->year, calendar->month); if (calendar->day < 1) { calendar->day = 1; } if (calendar->day > monthLength) { calendar->day = monthLength; } } /* * This returns what day of the week the month starts on. */ static int getMonthStartWeekday (int year, int month) { struct tm Date; /* *INDENT-EQLS* Set the tm structure correctly. */ Date.tm_sec = 0; Date.tm_min = 0; Date.tm_hour = 10; Date.tm_mday = 1; Date.tm_mon = month - 1; Date.tm_year = YEAR2INDEX (year); Date.tm_isdst = 1; /* Call the mktime function to fill in the holes. */ if (mktime (&Date) == (time_t) - 1) { return 0; } return Date.tm_wday; } /* * This function returns a 1 if it's a leap year and 0 if it's not. */ static int isLeapYear (int year) { int result = 0; if ((year % 4) == 0) { if ((year % 100) == 0) { if ((year % 400) == 0) { result = 1; } } else { result = 1; } } return result; } /* * This increments the current day by the given value. */ static void incrementCalendarDay (CDKCALENDAR *calendar, int adjust) { int monthLength = getMonthLength (calendar->year, calendar->month); /* Make sure we adjust the day correctly. */ if (adjust + calendar->day > monthLength) { /* Have to increment the month by one. */ calendar->day = calendar->day + adjust - monthLength; incrementCalendarMonth (calendar, 1); } else { calendar->day += adjust; drawCDKCalendarField (calendar); } } /* * This decrments the current day by the given value. */ static void decrementCalendarDay (CDKCALENDAR *calendar, int adjust) { /* Make sure we adjust the day correctly. */ if (calendar->day - adjust < 1) { int monthLength; /* Set the day according to the length of the month. */ if (calendar->month == 1) { /* Make sure we aren't going past the year limit. */ if (calendar->year == 1900) { const char *mesg[] = { "<C></U>Error", "Can not go past the year 1900" }; Beep (); popupLabel (ScreenOf (calendar), (CDK_CSTRING2)mesg, 2); return; } monthLength = getMonthLength (calendar->year - 1, 12); } else { monthLength = getMonthLength (calendar->year, calendar->month - 1); } calendar->day = monthLength - (adjust - calendar->day); /* Have to decrement the month by one. */ decrementCalendarMonth (calendar, 1); } else { calendar->day -= adjust; drawCDKCalendarField (calendar); } } /* * This increments the current month by the given value. */ static void incrementCalendarMonth (CDKCALENDAR *calendar, int adjust) { int monthLength; /* Are we at the end of the year. */ if (calendar->month + adjust > 12) { calendar->month = (calendar->month + adjust) - 12; calendar->year++; } else { calendar->month += adjust; } /* Get the length of the current month. */ monthLength = getMonthLength (calendar->year, calendar->month); if (calendar->day > monthLength) { calendar->day = monthLength; } /* Get the start of the current month. */ calendar->weekDay = getMonthStartWeekday (calendar->year, calendar->month); /* Redraw the calendar. */ eraseCDKCalendar (calendar); drawCDKCalendar (calendar, ObjOf (calendar)->box); } /* * This decrements the current month by the given value. */ static void decrementCalendarMonth (CDKCALENDAR *calendar, int adjust) { int monthLength; /* Are we at the end of the year. */ if (calendar->month <= adjust) { if (calendar->year == 1900) { const char *mesg[] = { "<C></U>Error", "Can not go past the year 1900" }; Beep (); popupLabel (ScreenOf (calendar), (CDK_CSTRING2)mesg, 2); return; } else { calendar->month = 13 - adjust; calendar->year--; } } else { calendar->month -= adjust; } /* Get the length of the current month. */ monthLength = getMonthLength (calendar->year, calendar->month); if (calendar->day > monthLength) { calendar->day = monthLength; } /* Get the start of the current month. */ calendar->weekDay = getMonthStartWeekday (calendar->year, calendar->month); /* Redraw the calendar. */ eraseCDKCalendar (calendar); drawCDKCalendar (calendar, ObjOf (calendar)->box); } /* * This increments the current year by the given value. */ static void incrementCalendarYear (CDKCALENDAR *calendar, int adjust) { /* Increment the year. */ calendar->year += adjust; /* If we are in Feb make sure we don't trip into voidness. */ if (calendar->month == 2) { int monthLength = getMonthLength (calendar->year, calendar->month); if (calendar->day > monthLength) { calendar->day = monthLength; } } /* Get the start of the current month. */ calendar->weekDay = getMonthStartWeekday (calendar->year, calendar->month); /* Redraw the calendar. */ eraseCDKCalendar (calendar); drawCDKCalendar (calendar, ObjOf (calendar)->box); } /* * This decrements the current year by the given value. */ static void decrementCalendarYear (CDKCALENDAR *calendar, int adjust) { /* Make sure we don't go out of bounds. */ if (calendar->year - adjust < 1900) { const char *mesg[] = { "<C></U>Error", "Can not go past the year 1900" }; Beep (); popupLabel (ScreenOf (calendar), (CDK_CSTRING2)mesg, 2); return; } /* Decrement the year. */ calendar->year -= adjust; /* If we are in Feb make sure we don't trip into voidness. */ if (calendar->month == 2) { int monthLength = getMonthLength (calendar->year, calendar->month); if (calendar->day > monthLength) { calendar->day = monthLength; } } /* Get the start of the current month. */ calendar->weekDay = getMonthStartWeekday (calendar->year, calendar->month); /* Redraw the calendar. */ eraseCDKCalendar (calendar); drawCDKCalendar (calendar, ObjOf (calendar)->box); } /* * This returns the length of the current month. */ static int getMonthLength (int year, int month) { int monthLength = daysOfTheMonth[month]; if (month == 2) { monthLength += isLeapYear (year); } return monthLength; } /* * This returns what day of the week the month starts on. */ static time_t getCurrentTime (CDKCALENDAR *calendar) { struct tm Date, *dateInfo; time_t clck; /* Determine the current time and determine if we are in DST. */ time (&clck); dateInfo = gmtime (&clck); /* *INDENT-EQLS* Set the tm structure correctly. */ Date.tm_sec = 0; Date.tm_min = 0; Date.tm_hour = 0; Date.tm_mday = calendar->day; Date.tm_mon = calendar->month - 1; Date.tm_year = YEAR2INDEX (calendar->year); Date.tm_isdst = dateInfo->tm_isdst; /* Call the mktime function to fill in the holes. */ return mktime (&Date); } static void _focusCDKCalendar (CDKOBJS *object) { CDKCALENDAR *widget = (CDKCALENDAR *)object; drawCDKFScale (widget, ObjOf (widget)->box); } static void _unfocusCDKCalendar (CDKOBJS *object) { CDKCALENDAR *widget = (CDKCALENDAR *)object; drawCDKFScale (widget, ObjOf (widget)->box); } dummyRefreshData (Calendar) dummySaveData (Calendar)