|
|
| version 1.17, 2002/02/18 12:13:42 | version 1.78, 2008/02/14 10:11:16 |
|---|---|
| Line 1 | Line 1 |
| /** @file | /** @file |
| Parser: @b date parser class. | Parser: @b date parser class. |
| Copyright (c) 2001, 2002 ArtLebedev Group (http://www.artlebedev.com) | Copyright (c) 2001-2005 ArtLebedev Group (http://www.artlebedev.com) |
| Author: Alexandr Petrosian <paf@design.ru> (http://paf.design.ru) | Author: Alexandr Petrosian <paf@design.ru> (http://paf.design.ru) |
| $Id$ | |
| */ | */ |
| static const char * const IDENT_DATE_C="$Date$"; | |
| #include "classes.h" | #include "classes.h" |
| #include "pa_vmethod_frame.h" | |
| #include "pa_request.h" | #include "pa_request.h" |
| #include "pa_vdouble.h" | #include "pa_vdouble.h" |
| #include "pa_vdate.h" | #include "pa_vdate.h" |
| #include "pa_vtable.h" | #include "pa_vtable.h" |
| // defines | |
| #define DATE_CLASS_NAME "date" | |
| // class | // class |
| class MDate : public Methoded { | class MDate: public Methoded { |
| public: // VStateless_class | public: // VStateless_class |
| Value *create_new_value(Pool& pool) { return new(pool) VDate(pool, 0); } | Value* create_new_value(Pool&, HashStringValue&) { return new VDate(0); } |
| public: | public: |
| MDate(Pool& pool); | MDate(); |
| public: // Methoded | public: // Methoded |
| bool used_directly() { return true; } | bool used_directly() { return true; } |
| }; | }; |
| // global variable | |
| DECLARE_CLASS_VAR(date, new MDate, 0); | |
| // helpers | |
| class Date_calendar_table_template_columns: public ArrayString { | |
| public: | |
| Date_calendar_table_template_columns(): ArrayString(6+2) { | |
| for(int i=0; i<=6; i++) { | |
| char *cname=new(PointerFreeGC) char[1/*strlen("6")*/+1/*terminating 0*/]; | |
| *this+=new String(cname, sprintf(cname, "%d", i)); // .i column name | |
| } | |
| *this+=new String("week"); | |
| *this+=new String("year"); | |
| } | |
| }; | |
| Table date_calendar_table_template(new Date_calendar_table_template_columns); | |
| // methods | // methods |
| static void _now(Request& r, const String& method_name, MethodParams *) { | static void _now(Request& r, MethodParams& params) { |
| Pool& pool=r.pool(); | VDate& vdate=GET_SELF(r, VDate); |
| VDate *vdate=static_cast<VDate *>(r.self); | |
| vdate->set_time(time(0)); | time_t t=time(0); |
| if(params.count()==1) // ^now(offset) | |
| t+=(time_t)round(params.as_double(0, "offset must be double", r)*SECS_PER_DAY); | |
| vdate.set_time(t); | |
| } | } |
| static void _create(Request& r, const String& method_name, MethodParams *params) { | /// shrinked range: 1970/1/1 to 2038/1/1 |
| Pool& pool=r.pool(); | static int to_year(int iyear) { |
| VDate *vdate=static_cast<VDate *>(r.self); | if(iyear<1970 || iyear>2038) |
| throw Exception(0, | |
| time_t time; | 0, |
| if(params->size()==1) // ^set(float days) | "year '%d' is out of valid range", iyear); |
| time=(time_t)(params->as_double(0, "float days must be double", r)*SECS_PER_DAY); | return iyear; |
| else if(params->size()>=3) { // ^set(y;m;d[;h[;m[;s]]]) | } |
| tm tmIn={0}; | |
| static int to_month(int imonth) { | |
| return max(1, min(imonth, 12)) -1; | |
| } | |
| static int to_tm_year(int iyear) { | |
| return to_year(iyear)-1900; | |
| } | |
| // 2002-04-25 18:14:00 | |
| // 18:14:00 | |
| // 2002:04:25 [+maybe time] | |
| /*not static, used in image.C*/ tm cstr_to_time_t(char *cstr) { | |
| if( !cstr || !*cstr ) | |
| throw Exception(0, | |
| 0, | |
| "empty string is not valid datetime"); | |
| char *cur=cstr; | |
| char date_delim=isdigit((unsigned char)cur[0])&&isdigit((unsigned char)cur[1])&&isdigit((unsigned char)cur[2])&&isdigit((unsigned char)cur[3])&&cur[4]==':'?':' | |
| :'-'; | |
| const char* year=lsplit(&cur, date_delim); | |
| const char* month=lsplit(&cur, date_delim); | |
| const char* mday=lsplit(&cur, ' '); | |
| if(!month) | |
| cur=cstr; | |
| const char* hour=lsplit(&cur, ':'); | |
| const char* min=lsplit(&cur, ':'); | |
| const char* sec=lsplit(&cur, '.'); | |
| const char* msec=cur; | |
| tm tmIn; memset(&tmIn, 0, sizeof(tmIn)); | |
| tmIn.tm_isdst=-1; | |
| if(!month) | |
| if(min) { | |
| year=mday=0; // HH:MM | |
| time_t t=time(0); | |
| tm *tmNow=localtime(&t); | |
| tmIn.tm_year=tmNow->tm_year; | |
| tmIn.tm_mon=tmNow->tm_mon; | |
| tmIn.tm_mday=tmNow->tm_mday; | |
| goto date_part_set; | |
| } else | |
| hour=min=sec=msec=0; // not YYYY- & not HH: = just YYYY | |
| tmIn.tm_year=to_tm_year(pa_atoi(year)); | |
| tmIn.tm_mon=month?pa_atoi(month)-1:0; | |
| tmIn.tm_mday=mday?pa_atoi(mday):1; | |
| date_part_set: | |
| tmIn.tm_hour=hour?pa_atoi(hour):0; | |
| tmIn.tm_min=min?pa_atoi(min):0; | |
| tmIn.tm_sec=sec?pa_atoi(sec):0; | |
| //tmIn.tm_[msec<<no such, waits reimplementation of the class]=f(msec); | |
| return tmIn; | |
| } | |
| static void _create(Request& r, MethodParams& params) { | |
| VDate& vdate=GET_SELF(r, VDate); | |
| if(params.count()==1) { | |
| // ^create[2002-04-25 18:14:00] | |
| // ^create[18:14:00] | |
| if(const String* sdate=params[0].get_string()) | |
| vdate.set_time(cstr_to_time_t(sdate->cstrm())); | |
| else { // ^create(float days) | |
| time_t t=(time_t)round(params.as_double(0, "float days must be double", r)*SECS_PER_DAY); | |
| if(t<0 || !localtime(&t)) | |
| throw Exception(0, | |
| 0, | |
| "invalid datetime"); | |
| vdate.set_time(t); | |
| } | |
| } else { // ^create(y;m;d[;h[;m[;s]]]) | |
| assert(params.count()<=6); | |
| tm tmIn; memset(&tmIn, 0, sizeof(tmIn)); | |
| tmIn.tm_isdst=-1; | tmIn.tm_isdst=-1; |
| int year=params->as_int(0, "year must be int", r); | tmIn.tm_year=to_tm_year(params.as_int(0, "year must be int", r)); |
| if(year<70) // 0..69 -> 100..169 [2000..2069] | tmIn.tm_mon=params.as_int(1, "month must be int", r)-1; |
| year+=100; | tmIn.tm_mday=params.count()>2?params.as_int(2, "mday must be int", r):1; |
| if(year>=1900) | int savedHour=0; |
| year-=1900; | if(params.count()>3) savedHour=tmIn.tm_hour=params.as_int(3, "hour must be int", r); |
| tmIn.tm_year=year; | if(params.count()>4) tmIn.tm_min=params.as_int(4, "minutes must be int", r); |
| tmIn.tm_mon=params->as_int(1, "month must be int", r)-1; | if(params.count()>5) tmIn.tm_sec=params.as_int(5, "seconds must be int", r); |
| tmIn.tm_mday=params->as_int(2, "mday must be int", r); | vdate.set_time(tmIn); |
| if(params->size()>3) tmIn.tm_hour=params->as_int(3, "hour must be int", r); | }; |
| if(params->size()>4) tmIn.tm_min=params->as_int(4, "minutes must be int", r); | |
| if(params->size()>5) tmIn.tm_sec=params->as_int(5, "seconds must be int", r); | |
| time=mktime(&tmIn); | |
| if(time<0) | |
| throw Exception(0, 0, | |
| &method_name, | |
| "invalid datetime"); | |
| } else | |
| throw Exception(0, 0, | |
| &method_name, | |
| "invalid params count, must be 1 or >=3"); | |
| vdate->set_time(time); | |
| } | } |
| static void _sql_string(Request& r, const String& method_name, MethodParams *) { | static void _sql_string(Request& r, MethodParams&) { |
| Pool& pool=r.pool(); | VDate& vdate=GET_SELF(r, VDate); |
| VDate *vdate=static_cast<VDate *>(r.self); | |
| int size=1+ 4+1+2+1+2 +1+ 2+1+2+1+2 +1 +1; | int size=1+ 4+1+2+1+2 +1+ 2+1+2+1+2 +1 +1; |
| char *buf=(char *)pool.malloc(size); | char *buf=new(PointerFreeGC) char[size]; |
| time_t time=vdate->get_time(); | size=strftime(buf, size, "%Y-%m-%d %H:%M:%S", &vdate.get_localtime()); |
| size=strftime(buf, size, "%Y-%m-%d %H:%M:%S", localtime(&time)); | |
| r.write_assign_lang(String(buf, size)); | |
| String& string=*new(pool) String(pool); | |
| string.APPEND_CLEAN(buf, size, | |
| method_name.origin().file, | |
| method_name.origin().line); | |
| Value& result=*new(pool) VString(string); | |
| r.write_assign_lang(result); | |
| } | } |
| static void _roll(Request& r, MethodParams& params) { | |
| VDate& vdate=GET_SELF(r, VDate); | |
| static void _roll(Request& r, const String& method_name, MethodParams *params) { | const String& what=params.as_string(0, "'what' must be string"); |
| Pool& pool=r.pool(); | int oyear=0; |
| VDate *vdate=static_cast<VDate *>(r.self); | int omonth=0; |
| int oday=0; | |
| const String& what=params->as_string(0, "'what' must be string"); | |
| int oyear=0; | |
| int omonth=0; | |
| int oday=0; | |
| int *offset; | int *offset; |
| if(what=="year") offset=&oyear; | if(what=="year") offset=&oyear; |
| else if(what=="month") offset=&omonth; | else if(what=="month") offset=&omonth; |
| else if(what=="day") offset=&oday; | else if(what=="day") offset=&oday; |
| else | else if(what=="TZ") { |
| throw Exception(0, 0, | const String& argument_tz=params.as_string(1, "'TZ' must be string"); |
| vdate.set_tz(&argument_tz); | |
| return; | |
| } else | |
| throw Exception(PARSER_RUNTIME, | |
| &what, | &what, |
| "must be year|month|day"); | "must be year|month|day|TZ"); |
| *offset=params->as_int(1, "offset must be int", r); | *offset=params.as_int(1, "offset must be int", r); |
| if(!(*offset==1 || *offset==-1)) | |
| throw Exception(0, 0, | |
| &method_name, | |
| "offset must be +/- 1"); | |
| time_t self_time=vdate->get_time(); | time_t self_time=vdate.get_time(); |
| tm tmIn=*localtime(&self_time); | tm tmIn=*localtime(&self_time); |
| // we will preserve daytime from day-light-saving shifts across roll | |
| tm tmSaved=tmIn; | tm tmSaved=tmIn; |
| tmIn.tm_hour=24/2; tmIn.tm_min=tmIn.tm_sec=0; | int adjust_day=0; |
| time_t t_changed_date; | |
| while(true) { | |
| tmIn.tm_year+=oyear; | |
| tmIn.tm_mon+=omonth; | |
| tmIn.tm_mday+=oday+adjust_day; | |
| tmIn.tm_hour=24/2; | |
| tmIn.tm_min=0; | |
| tmIn.tm_sec=0; | |
| int saved_mon=(tmIn.tm_mon+12*100)%12; // crossing year boundary backwards | |
| t_changed_date=mktime/*normalizetime*/(&tmIn); | |
| if(t_changed_date<0) | |
| throw Exception(0, | |
| 0, | |
| "bad resulting time (rolled out of valid date range)"); | |
| if(oday==0 && tmIn.tm_mon!=saved_mon/*but it changed*/) { | |
| if(adjust_day <= -3/*31->28 max, so never, but...*/) | |
| throw Exception(0, | |
| 0, | |
| "bad resulting time (day hole still with %d day adjustment)", adjust_day ); | |
| tmIn=tmSaved; // restoring | |
| --adjust_day; //retrying with prev day | |
| } else | |
| break; | |
| } | |
| tmIn.tm_year+=oyear; | tm *tmOut=localtime(&t_changed_date); |
| time_t t=mktime(&tmIn); | |
| t+=omonth*getMonthDays(tmIn.tm_year, (tmIn.tm_mon+(omonth<0?-1:0)+12)%12)*SECS_PER_DAY; | |
| t+=oday*SECS_PER_DAY; | |
| tm *tmOut=localtime(&t); | |
| if(!tmOut) | if(!tmOut) |
| throw Exception(0, 0, | throw Exception(0, |
| &method_name, | 0, |
| "bad resulting time (seconds from epoch=%ld)", t); | "bad resulting time (seconds from epoch=%d)", t_changed_date); |
| tmOut->tm_hour=tmSaved.tm_hour; | tmOut->tm_hour=tmSaved.tm_hour; |
| tmOut->tm_min=tmSaved.tm_min; | tmOut->tm_min=tmSaved.tm_min; |
| tmOut->tm_sec=tmSaved.tm_sec; | tmOut->tm_sec=tmSaved.tm_sec; |
| tmOut->tm_isdst=-1; | |
| { | { |
| time_t t=mktime(tmOut); | time_t t_changed_time=mktime/*normalizetime*/(tmOut); |
| if(t<0) | /*autofix: in msk timezone last sunday of march hour hole: [2am->3am) |
| throw Exception(0, 0, | if( |
| &method_name, | tmOut->tm_hour!=tmSaved.tm_hour |
| "bad resulting time (after reconstruction)"); | ||tmOut->tm_min!=tmSaved.tm_min) |
| throw Exception(0, | |
| 0, | |
| "bad resulting time (hour hole)"); | |
| */ | |
| if(t_changed_time<0) | |
| throw Exception(0, | |
| 0, | |
| "bad resulting time (after reconstruction)"); | |
| vdate->set_time(t); | vdate.set_time(t_changed_time); |
| } | } |
| } | } |
| static Table *fill_month_days(Request& r, | static Table& fill_month_days(Request& r, MethodParams& params, bool rus){ |
| const String& method_name, MethodParams *params, bool rus){ | Table::Action_options table_options; |
| Pool& pool=r.pool(); | Table& result=*new Table(date_calendar_table_template, table_options); |
| Table *result=new(pool) Table(pool, &method_name, 0/*&columns*/); | |
| int year=to_year(params.as_int(1, "year must be int", r)); | |
| int year=params->as_int(1, "year must be int", r); | int month=to_month(params.as_int(2, "month must be int", r)); |
| int month=max(1, min(params->as_int(2, "month must be int", r), 12)) -1; | |
| tm tmIn; | |
| memset(&tmIn, 0, sizeof(tmIn)); | |
| tmIn.tm_mday=1; | |
| tmIn.tm_mon=month; | |
| tmIn.tm_year=year-1900; | |
| tm tmIn={0, 0, 0, 1, month, year-1900}; | time_t t=mktime(&tmIn); |
| time_t t=mktime(&tmIn); | |
| if(t<0) | if(t<0) |
| throw Exception(0, 0, | throw Exception(0, |
| &method_name, | 0, |
| "invalid date"); | "invalid date"); |
| tm *tmOut=localtime(&t); | tm *tmOut=localtime(&t); |
| int weekDay1=tmOut->tm_wday; | int weekDay1=tmOut->tm_wday; |
| if(rus) | if(rus) |
| weekDay1=weekDay1?weekDay1-1:6; //sunday last | weekDay1=weekDay1?weekDay1-1:6; //sunday last |
| int monthDays=getMonthDays(year, month); | int monthDays=getMonthDays(year, month); |
| for(int _day=1-weekDay1; _day<=monthDays;) { | for(int _day=1-weekDay1; _day<=monthDays;) { |
| Array& row=*new(pool) Array(pool, 7); | Table::element_type row(new ArrayString(7)); |
| for(int wday=0; wday<7; wday++, _day++) { | // calculating year week no [1..54] |
| String *cell=new(pool) String(pool); | char *weekno_buf=0; // surely would be assigned to, but to calm down compiler |
| size_t weekno_size=0; // same | |
| int weekyear=0; // same | |
| // 0..6 week days-cells fill with month days | |
| for(int wday=0; wday<7; wday++, _day++) { | |
| String* cell=new String; | |
| if(_day>=1 && _day<=monthDays) { | if(_day>=1 && _day<=monthDays) { |
| char *buf=(char *)pool.malloc(2+1); | char *buf=new(PointerFreeGC) char[2+1]; |
| cell->APPEND_CLEAN(buf, sprintf(buf, "%02d", _day), | cell->append_know_length(buf, sprintf(buf, "%02d", _day), String::L_CLEAN); |
| method_name.origin().file, method_name.origin().line); | } |
| } | *row+=cell; |
| row+=cell; | |
| if(wday==(rus?3:4)/*thursday*/) { | |
| tm tms; | |
| memset(&tms, 0, sizeof(tmIn)); | |
| tms.tm_mday=_day; | |
| tms.tm_mon=month; | |
| tms.tm_year=year-1900; | |
| /*normalize*/mktime(&tms); | |
| weekyear=tms.tm_year+1900; | |
| const int weekno_buf_size=2+1/*for stupid snprintfs*/ +1; | |
| weekno_buf=new(PointerFreeGC) char[weekno_buf_size]; | |
| VDate::yw week = VDate::CalcWeek(tms); | |
| weekno_size=snprintf(weekno_buf, weekno_buf_size, "%02d", week.week); | |
| } | |
| } | } |
| *result+=&row; | // appending year week no |
| } | *row+=new String(weekno_buf, weekno_size); |
| // appending year week year | |
| return result; | { |
| char* buf=new(PointerFreeGC) char[4+1]; | |
| *row+=new String(buf, sprintf(buf, "%02d", weekyear)); | |
| } | |
| result+=row; | |
| } | |
| return result; | |
| } | } |
| static Table *fill_week_days(Request& r, | static Table& fill_week_days(Request& r, MethodParams& params, bool rus){ |
| const String& method_name, MethodParams *params, bool rus){ | Table::columns_type columns(new ArrayString(4)); |
| Pool& pool=r.pool(); | *columns+=new String("year"); |
| Array& columns=*new(pool) Array(pool, 4); | *columns+=new String("month"); |
| Table *result=new(pool) Table(pool, &method_name, &columns); | *columns+=new String("day"); |
| *columns+=new String("weekday"); | |
| int year=params->as_int(1, "year must be int", r); | Table& result=*new Table(columns); |
| int month=max(1, min(params->as_int(2, "month must be int", r), 12)) -1; | |
| int day=params->as_int(3, "day must be int", r); | int year=to_year(params.as_int(1, "year must be int", r)); |
| int month=to_month(params.as_int(2, "month must be int", r)); | |
| tm tmIn={0, 0, 18, day, month, year-1900}; | int day=params.as_int(3, "day must be int", r); |
| time_t t=mktime(&tmIn); | |
| tm tmIn; | |
| memset(&tmIn, 0, sizeof(tmIn)); | |
| tmIn.tm_hour=18; | |
| tmIn.tm_mday=day; | |
| tmIn.tm_mon=month; | |
| tmIn.tm_year=year-1900; | |
| time_t t=mktime(&tmIn); | |
| if(t<0) | if(t<0) |
| throw Exception(0, 0, | throw Exception(0, |
| &method_name, | 0, |
| "invalid date"); | "invalid date"); |
| tm *tmOut=localtime(&t); | tm *tmOut=localtime(&t); |
| int baseWeekDay=tmOut->tm_wday; | int baseWeekDay=tmOut->tm_wday; |
| if(rus) | if(rus) |
| baseWeekDay=baseWeekDay?baseWeekDay-1:6; //sunday last | baseWeekDay=baseWeekDay?baseWeekDay-1:6; //sunday last |
| t-=baseWeekDay*SECS_PER_DAY; | t-=baseWeekDay*SECS_PER_DAY; |
| for(int curWeekDay=0; curWeekDay<7; curWeekDay++, t+=SECS_PER_DAY) { | for(int curWeekDay=0; curWeekDay<7; curWeekDay++, t+=SECS_PER_DAY) { |
| tm *tmOut=localtime(&t); | tm *tmOut=localtime(&t); |
| Array& row=*new(pool) Array(pool, 4); | Table::element_type row(new ArrayString(4)); |
| #define WDFILL(size, value) { \ | #define WDFILL(size, value) { \ |
| char *buf=(char *)pool.malloc(size+1); \ | char *buf=new(PointerFreeGC) char[size+1]; \ |
| String *cell=new(pool) String(pool); \ | *row+=new String(buf, sprintf(buf, "%0"#size"d", value)); \ |
| cell->APPEND_CLEAN(buf, sprintf(buf, "%0"#size"d", value), \ | |
| method_name.origin().file, \ | |
| method_name.origin().line); \ | |
| row+=cell; \ | |
| } | } |
| WDFILL(4, 1900+tmOut->tm_year); | WDFILL(4, 1900+tmOut->tm_year); |
| WDFILL(2, 1+tmOut->tm_mon); | WDFILL(2, 1+tmOut->tm_mon); |
| WDFILL(2, tmOut->tm_mday); | WDFILL(2, tmOut->tm_mday); |
| WDFILL(2, tmOut->tm_wday); | WDFILL(2, tmOut->tm_wday); |
| *result+=&row; | result+=row; |
| } | } |
| return result; | return result; |
| } | } |
| static void _calendar(Request& r, const String& method_name, MethodParams *params) { | static void _calendar(Request& r, MethodParams& params) { |
| Pool& pool=r.pool(); | const String& what=params.as_string(0, "format must be strig"); |
| const String& what=params->as_string(0, "format must be strig"); | |
| bool rus=false; | bool rus=false; |
| if(what=="rus") | if(what=="rus") |
| rus=true; | rus=true; |
| else if(what=="eng") | else if(what=="eng") |
| rus=false; | rus=false; |
| else | else |
| throw Exception(0, 0, | throw Exception(PARSER_RUNTIME, |
| &what, | &what, |
| "must be rus|eng"); | "must be rus|eng"); |
| Table *table; | Table* table; |
| if(params->size()==1+2) | if(params.count()==1+2) |
| table=fill_month_days(r, method_name, params, rus); | table=&fill_month_days(r, params, rus); |
| else // 1+3 | else // 1+3 |
| table=fill_week_days(r, method_name, params, rus); | table=&fill_week_days(r, params, rus); |
| VTable& result=*new(pool) VTable(pool, table); | r.write_no_lang(*new VTable(table)); |
| result.set_name(method_name); | |
| r.write_no_lang(result); | |
| } | } |
| // constructor | static void _unix_timestamp(Request& r, MethodParams& params) { |
| VDate& vdate=GET_SELF(r, VDate); | |
| if(params.count()==0) { | |
| // ^date.unix-timestamp[] | |
| r.write_no_lang(*new VInt((int)vdate.get_time())); | |
| } else { | |
| if(vdate.get_time()) | |
| throw Exception(0, | |
| 0, | |
| "date object already constructed"); | |
| // ^unix-timestamp(time_t) | |
| time_t t=(time_t)params.as_int(0, "Unix timestamp must be integer", r); | |
| vdate.set_time(t); | |
| } | |
| } | |
| MDate::MDate(Pool& apool) : Methoded(apool) { | static void _lastday(Request& r, MethodParams& params) { |
| set_name(*NEW String(pool(), DATE_CLASS_NAME)); | int year; |
| int month; | |
| if(&r.get_self() == date_class) { | |
| // ^date:lastday(year;month) | |
| if(params.count() == 2) { | |
| year=to_year(params.as_int(0, "year must be int", r)); | |
| month=to_month(params.as_int(1, "month must be int", r)); | |
| } else | |
| throw Exception(PARSER_RUNTIME, | |
| 0, | |
| "year and month must be defined"); | |
| } else { | |
| // ^date.lastday[] | |
| tm &tmIn=GET_SELF(r, VDate).get_localtime(); | |
| year=tmIn.tm_year+1900; | |
| month=tmIn.tm_mon; | |
| } | |
| r.write_assign_lang(*new VInt(getMonthDays(year, month))); | |
| } | |
| // ^now[] | // constructor |
| add_native_method("now", Method::CT_DYNAMIC, _now, 0, 0); | |
| // ^set(float days) | MDate::MDate(): Methoded("date") { |
| // ^date::now[] | |
| add_native_method("now", Method::CT_DYNAMIC, _now, 0, 1); | |
| // ^date::create(float days) | |
| add_native_method("create", Method::CT_DYNAMIC, _create, 1, 6); | add_native_method("create", Method::CT_DYNAMIC, _create, 1, 6); |
| // old name for compatibility with <= v1.17 2002/2/18 12:13:42 paf | |
| add_native_method("set", Method::CT_DYNAMIC, _create, 1, 6); | add_native_method("set", Method::CT_DYNAMIC, _create, 1, 6); |
| // ^sql-string[] | // ^date.sql-string[] |
| add_native_method("sql-string", Method::CT_DYNAMIC, _sql_string, 0, 0); | add_native_method("sql-string", Method::CT_DYNAMIC, _sql_string, 0, 0); |
| // ^roll(year|month|day;+/- 1) | // ^date:lastday(year;month) |
| // ^date.lastday[] | |
| add_native_method("lastday", Method::CT_ANY, _lastday, 0, 2); | |
| // ^date.roll(year|month|day;+/- 1) | |
| add_native_method("roll", Method::CT_DYNAMIC, _roll, 2, 2); | add_native_method("roll", Method::CT_DYNAMIC, _roll, 2, 2); |
| // ^date:calendar[month|montheng;year;month] = table | // ^date:calendar[month|montheng;year;month] = table |
| // ^date:calendar[week|weekeng;year;month;day] = table | // ^date:calendar[week|weekeng;year;month;day] = table |
| add_native_method("calendar", Method::CT_STATIC, _calendar, 3, 4); | add_native_method("calendar", Method::CT_STATIC, _calendar, 3, 4); |
| } | |
| // global variable | |
| Methoded *date_class; | |
| // creator | |
| Methoded *MDate_create(Pool& pool) { | // ^date.unix-timestamp[] |
| return date_class=new(pool) MDate(pool); | // ^date::unix-timestamp[] |
| add_native_method("unix-timestamp", Method::CT_DYNAMIC, _unix_timestamp, 0, 1); | |
| } | } |