gambas-source-code/main/lib/vb/vbdate.c

210 lines
6.1 KiB
C
Raw Normal View History

/***************************************************************************
vbdate.c
Visual Basic compatibility Component for date manipulation
(c) 2000-2003 Nigel Gerrard <nigel@gerrard1123.freeserve.co.uk>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 1, or (at your option)
any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
***************************************************************************/
#define __VBDATE_C
#include "vbdate.h"
#include <stdio.h>
#include <string.h>
#define DATE_YEAR_MIN -4801
#define DATE_YEAR_MAX 9999
/* Helper functions */
const char days_in_months[2][13] =
{ /* error, jan feb mar apr may jun jul aug sep oct nov dec */
{ 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 },
{ 0, 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 } /* leap year */
};
const short days_in_year[2][14] =
{ /* 0, jan feb mar apr may jun jul aug sep oct nov dec */
{ 0, 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365 },
{ 0, 0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335, 366 } /* leap year */
};
/* Returns 1 for a leap year, 0 else */
int date_is_leap_year(short year)
{
if (year < 0)
year += 8001;
if ((((year % 4) == 0) && ((year % 100) != 0)) || (year % 400) == 0)
return 1;
else
return 0;
}
int date_is_valid(GB_DATE_SERIAL *date)
{
return ((date->month >= 1) && (date->month <= 12) &&
(date->year >= DATE_YEAR_MIN) && (date->year <= DATE_YEAR_MAX) && (date->year != 0) &&
(date->day >= 1) && (date->day <= days_in_months[date_is_leap_year(date->year)][(short)date->month]) &&
(date->hour >= 0) && (date->hour <= 23) && (date->min >= 0) && (date->min <= 59) &&
(date->sec >= 0) && (date->sec <= 59));
}
/** Modulo function that works correctly for negative divisors */
#define modulo(x,y) ((x)%(y)<0?(x)%(y)+(y):(x)%(y))
void DATE_adjust( GB_DATE *vdate, int period, int interval) /* Adjust the date by the interval period */
{
GB_DATE_SERIAL *date;
int year, month, day;
date = GB.SplitDate(vdate);
switch(period){
case 0: /* Calendar Month */
year = ((date->year * 12) + (date->month - 1) + interval)/12;
month = modulo((date->month - 1)+interval, 12) + 1;
day = date->day > days_in_months[date_is_leap_year(year)][month] ? days_in_months[date_is_leap_year(year)][month] : date->day;
date->day = day;
date->month = month;
date->year = year;
GB.MakeDate(date, vdate);
break;
case 1: /* days */
vdate->value.date += interval;
break;
case 2: /* Time */
vdate->value.time += interval;
break;
case 3: /* weekdays - in this case weekdays are Mon - Fri */
vdate->value.date += ( 7 * ( interval / 5));
date->weekday += ( interval % 5 );
if (date->weekday > 5){
date->weekday -= 5;
vdate->value.date += 2;
}
if (date->weekday < 1){
date->weekday += 5;
vdate->value.date -= 2;
}
vdate->value.date += ( interval % 5);
break;
case 4: /* Add year */
while ( interval != 0){
if ( interval < 0 ){
vdate->value.date -= days_in_year[date_is_leap_year(date->year)][13];
date->year--;
interval++;
}
else {
vdate->value.date += days_in_year[date_is_leap_year(date->year)][13];
date->year++;
interval--;
}
}
break;
}
/* Now if time takes it into another day */
while (vdate->value.time >= 86400000){
vdate->value.date++;
vdate->value.time -= 86400000;
}
/* Or time is negative so we need to take off a day */
while (vdate->value.time < 0){
vdate->value.date--;
vdate->value.time += 86400000;
}
CLEAR(&date);
date = GB.SplitDate(vdate);
if (!date_is_valid(date)){
/*printf("Invalid date : year [%i] month [%i] day [%i] hour [%i] min [%i] sec [%i] weekday [%i] msec [%i]\n",
date->year, date->month, date->day, date->hour, date->min, date->sec, date->weekday, date->msec);*/
GB.Error("Invalid Date Returned");
}
CLEAR(&date);
}
******** Merged /branches/64bits r918:1003 into /trunk [CONFIGURATION] * NEW: 64 bits port. [EXAMPLES] * BUG: Fixed the AnalogWatch example. [WIKI CGI SCRIPT] * NEW: Some little cosmetic changes. [INTERPRETER] * NEW: The extern function implementation has been redesigned and is now based on libffi, so that it works on 64 bits system. Because of a flaw in the compiler design, projects that use the Pointer datatype must be recompiled to be used on a 64 bits system. This flaw will be fixed in Gambas 3. * OPT: Put some tables into read-only memory. About 1000 bytes are saved for each running interpreter, except the first one. * BUG: Does not crash anymore if a component cannot be loaded. * NEW: Spanish translation updated. * NEW: A new interpreter API for returning a pointer. [COMPILER] * BUG: Correctly compiles LONG constants inside code. [GB.DEBUG] * BUG: Compiles and links the gb.debug components with the thread libraries. [GB.DB.SQLITE3] * BUG: Getting the primary index of a table without primary index is safe now. [GB.GTK] * BUG: Modified the GLib priority of watched descriptors, as the main loop could enter in a loop in which user interface events were not managed. * BUG: Message boxes use application title without crashing now. [GB.OPENGL] * BUG: Disable dead code. [GB.QT.EXT] * BUG: TextEdit.TextWidth and TextEdit.TextHeight were not declared as read-only properties. [GB.XML.XSLT] * BUG: XSLT class is now declared as being not creatable. git-svn-id: svn://localhost/gambas/trunk@1006 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2008-01-17 22:39:26 +01:00
int DATE_diff( GB_DATE *vdate1, GB_DATE *vdate2, int period)
{
GB_DATE_SERIAL *date1, *date2;
******** Merged /branches/64bits r918:1003 into /trunk [CONFIGURATION] * NEW: 64 bits port. [EXAMPLES] * BUG: Fixed the AnalogWatch example. [WIKI CGI SCRIPT] * NEW: Some little cosmetic changes. [INTERPRETER] * NEW: The extern function implementation has been redesigned and is now based on libffi, so that it works on 64 bits system. Because of a flaw in the compiler design, projects that use the Pointer datatype must be recompiled to be used on a 64 bits system. This flaw will be fixed in Gambas 3. * OPT: Put some tables into read-only memory. About 1000 bytes are saved for each running interpreter, except the first one. * BUG: Does not crash anymore if a component cannot be loaded. * NEW: Spanish translation updated. * NEW: A new interpreter API for returning a pointer. [COMPILER] * BUG: Correctly compiles LONG constants inside code. [GB.DEBUG] * BUG: Compiles and links the gb.debug components with the thread libraries. [GB.DB.SQLITE3] * BUG: Getting the primary index of a table without primary index is safe now. [GB.GTK] * BUG: Modified the GLib priority of watched descriptors, as the main loop could enter in a loop in which user interface events were not managed. * BUG: Message boxes use application title without crashing now. [GB.OPENGL] * BUG: Disable dead code. [GB.QT.EXT] * BUG: TextEdit.TextWidth and TextEdit.TextHeight were not declared as read-only properties. [GB.XML.XSLT] * BUG: XSLT class is now declared as being not creatable. git-svn-id: svn://localhost/gambas/trunk@1006 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2008-01-17 22:39:26 +01:00
int diff = 0;
int year1, year2, month1, month2, weekday1, weekday2;
date1 = GB.SplitDate(vdate1);
weekday1 = date1->weekday;
year1 = date1->year;
month1 = date1->month;
date2 = GB.SplitDate(vdate2);
weekday2 = date2->weekday;
year2 = date2->year;
month2 = date2->month;
switch(period){
case 0: /* Calendar Month */
diff = ((year1 * 12) + month1) - ((year2 * 12) + month2);
break;
case 1: /* days */
diff = vdate1->value.date - vdate2->value.date;
break;
case 2: /* Time */
diff = ((vdate1->value.date - vdate2->value.date) * 86400000) + (vdate1->value.time - vdate2->value.time);
break;
case 3: /* weekdays */
/* If invalid weekday then find closest valid day:
* eg. 7 -1, 6 -5*/
weekday1 == 7 ? weekday1 = 1 : weekday1 == 6 ? weekday1 = 5 : weekday1;
weekday2 == 7 ? weekday2 = 1 : weekday2 == 6 ? weekday2 = 5 : weekday2;
diff = (((vdate1->value.date - vdate2->value.date) / 7) * 5)
+ (weekday1 - weekday2);
/* Needs validating */
break;
case 4: /* year */
diff = year1 - year2;
break;
case 5: /* Weeks */
diff = (vdate1->value.date - vdate2->value.date)/7;
break;
}
CLEAR(&date1);
CLEAR(&date2);
return diff;
}
/* End Helper functions */
/*
GB_DESC CVbDateDesc[] =
{
GB_DECLARE("Vb", 0), GB_NOT_CREATABLE(),
GB_STATIC_METHOD("DateAdd", "d", CVB_DateAdd, "(period)s(interval)i(Date)v"),
GB_STATIC_METHOD("DateDiff", "i", CVB_DateDiff, "(period)s(Date1)v(Date2)v"),
GB_END_DECLARE
};*/