1019 lines
28 KiB
C++
1019 lines
28 KiB
C++
|
/* ---------------------------------------------------------------------------
|
||
|
Nullsoft Database Engine
|
||
|
--------------------
|
||
|
codename: Near Death Experience
|
||
|
--------------------------------------------------------------------------- */
|
||
|
|
||
|
/* ---------------------------------------------------------------------------
|
||
|
|
||
|
IntegerField Class
|
||
|
|
||
|
--------------------------------------------------------------------------- */
|
||
|
|
||
|
#include "nde.h"
|
||
|
#include "Query.h"
|
||
|
#include <time.h>
|
||
|
#ifdef _WIN32
|
||
|
#include <malloc.h> // for alloca
|
||
|
#endif
|
||
|
//---------------------------------------------------------------------------
|
||
|
IntegerField::IntegerField(int Val)
|
||
|
{
|
||
|
InitField();
|
||
|
Type = FIELD_INTEGER;
|
||
|
Value = Val;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
void IntegerField::InitField(void)
|
||
|
{
|
||
|
Type = FIELD_INTEGER;
|
||
|
Value=0;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
IntegerField::IntegerField()
|
||
|
{
|
||
|
InitField();
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
IntegerField::~IntegerField()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
void IntegerField::ReadTypedData(const uint8_t *data, size_t len)
|
||
|
{
|
||
|
CHECK_INT(len);
|
||
|
Value = *((int *)data);
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
void IntegerField::WriteTypedData(uint8_t *data, size_t len)
|
||
|
{
|
||
|
CHECK_INT(len);
|
||
|
*((int *)data) = Value;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
int IntegerField::GetValue(void)
|
||
|
{
|
||
|
return Value;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
void IntegerField::SetValue(int Val)
|
||
|
{
|
||
|
Value = Val;
|
||
|
}
|
||
|
|
||
|
|
||
|
#include <limits.h>
|
||
|
//---------------------------------------------------------------------------
|
||
|
size_t IntegerField::GetDataSize(void)
|
||
|
{
|
||
|
return 4;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
int IntegerField::Compare(Field *Entry)
|
||
|
{
|
||
|
if (!Entry) return -1;
|
||
|
return GetValue() < ((IntegerField*)Entry)->GetValue() ? -1 : (GetValue() > ((IntegerField*)Entry)->GetValue() ? 1 : 0);
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
bool IntegerField::ApplyFilter(Field *Data, int op)
|
||
|
{
|
||
|
bool r;
|
||
|
switch (op)
|
||
|
{
|
||
|
case FILTER_EQUALS:
|
||
|
r = Value == ((IntegerField *)Data)->GetValue();
|
||
|
break;
|
||
|
case FILTER_NOTEQUALS:
|
||
|
r = Value != ((IntegerField *)Data)->GetValue();
|
||
|
break;
|
||
|
case FILTER_NOTCONTAINS:
|
||
|
r = (bool)!(Value & ((IntegerField *)Data)->GetValue());
|
||
|
break;
|
||
|
case FILTER_CONTAINS:
|
||
|
r = !!(Value & ((IntegerField *)Data)->GetValue());
|
||
|
break;
|
||
|
case FILTER_ABOVE:
|
||
|
r = (bool)(Value > ((IntegerField *)Data)->GetValue());
|
||
|
break;
|
||
|
case FILTER_BELOW:
|
||
|
r = (bool)(Value < ((IntegerField *)Data)->GetValue());
|
||
|
break;
|
||
|
case FILTER_BELOWOREQUAL:
|
||
|
r = (bool)(Value <= ((IntegerField *)Data)->GetValue());
|
||
|
break;
|
||
|
case FILTER_ABOVEOREQUAL:
|
||
|
r = (bool)(Value >= ((IntegerField *)Data)->GetValue());
|
||
|
break;
|
||
|
case FILTER_ISEMPTY:
|
||
|
r = (Value == 0 || Value == -1);
|
||
|
break;
|
||
|
case FILTER_ISNOTEMPTY:
|
||
|
r = !(Value == 0 || Value == -1);
|
||
|
break;
|
||
|
default:
|
||
|
r = true;
|
||
|
break;
|
||
|
}
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
typedef struct {
|
||
|
CFStringRef token;
|
||
|
int tid;
|
||
|
} tokenstruct;
|
||
|
|
||
|
enum {
|
||
|
TOKEN_AGO = 128,
|
||
|
TOKEN_NOW,
|
||
|
TOKEN_YESTERDAY,
|
||
|
TOKEN_TOMORROW,
|
||
|
TOKEN_TODAY,
|
||
|
TOKEN_OF,
|
||
|
TOKEN_THE,
|
||
|
TOKEN_DATE,
|
||
|
TOKEN_FROM,
|
||
|
TOKEN_BEFORE,
|
||
|
TOKEN_AFTER,
|
||
|
TOKEN_THIS,
|
||
|
TOKEN_SUNDAY,
|
||
|
TOKEN_MONDAY,
|
||
|
TOKEN_TUESDAY,
|
||
|
TOKEN_WEDNESDAY,
|
||
|
TOKEN_THURSDAY,
|
||
|
TOKEN_FRIDAY,
|
||
|
TOKEN_SATURDAY,
|
||
|
|
||
|
TOKEN_MIDNIGHT,
|
||
|
TOKEN_NOON,
|
||
|
|
||
|
TOKEN_AM,
|
||
|
TOKEN_PM,
|
||
|
|
||
|
TOKEN_JANUARY,
|
||
|
TOKEN_FEBRUARY,
|
||
|
TOKEN_MARCH,
|
||
|
TOKEN_APRIL,
|
||
|
TOKEN_MAY,
|
||
|
TOKEN_JUNE,
|
||
|
TOKEN_JULY,
|
||
|
TOKEN_AUGUST,
|
||
|
TOKEN_SEPTEMBER,
|
||
|
TOKEN_OCTOBER,
|
||
|
TOKEN_NOVEMBER,
|
||
|
TOKEN_DECEMBER,
|
||
|
|
||
|
TOKEN_TIME,
|
||
|
TOKEN_SECOND,
|
||
|
TOKEN_MINUTE,
|
||
|
TOKEN_HOUR,
|
||
|
TOKEN_DAY,
|
||
|
TOKEN_WEEK,
|
||
|
TOKEN_MONTH,
|
||
|
TOKEN_YEAR,
|
||
|
TOKEN_AT,
|
||
|
};
|
||
|
|
||
|
tokenstruct Int_Tokens[] = { // Feel free to add more...
|
||
|
{CFSTR("ago"), TOKEN_AGO},
|
||
|
{CFSTR("now"), TOKEN_NOW},
|
||
|
{CFSTR("am"), TOKEN_AM},
|
||
|
{CFSTR("pm"), TOKEN_PM},
|
||
|
{CFSTR("this"), TOKEN_THIS},
|
||
|
{CFSTR("date"), TOKEN_DATE},
|
||
|
{CFSTR("time"), TOKEN_TIME},
|
||
|
{CFSTR("of"), TOKEN_OF},
|
||
|
{CFSTR("at"), TOKEN_AT},
|
||
|
{CFSTR("the"), TOKEN_THE},
|
||
|
{CFSTR("yesterday"), TOKEN_YESTERDAY},
|
||
|
{CFSTR("tomorrow"), TOKEN_TOMORROW},
|
||
|
{CFSTR("today"), TOKEN_TODAY},
|
||
|
{CFSTR("from"), TOKEN_FROM},
|
||
|
{CFSTR("before"), TOKEN_BEFORE},
|
||
|
{CFSTR("after"), TOKEN_AFTER},
|
||
|
{CFSTR("past"), TOKEN_AFTER},
|
||
|
{CFSTR("monday"), TOKEN_MONDAY},
|
||
|
{CFSTR("mon"), TOKEN_MONDAY},
|
||
|
{CFSTR("tuesday"), TOKEN_TUESDAY},
|
||
|
{CFSTR("tue"), TOKEN_TUESDAY},
|
||
|
{CFSTR("wednesday"), TOKEN_WEDNESDAY},
|
||
|
{CFSTR("wed"), TOKEN_WEDNESDAY},
|
||
|
{CFSTR("thursday"), TOKEN_THURSDAY},
|
||
|
{CFSTR("thu"), TOKEN_THURSDAY},
|
||
|
{CFSTR("friday"), TOKEN_FRIDAY},
|
||
|
{CFSTR("fri"), TOKEN_FRIDAY},
|
||
|
{CFSTR("saturday"), TOKEN_SATURDAY},
|
||
|
{CFSTR("sat"), TOKEN_SATURDAY},
|
||
|
{CFSTR("sunday"), TOKEN_SUNDAY},
|
||
|
{CFSTR("sun"), TOKEN_SUNDAY},
|
||
|
{CFSTR("midnight"), TOKEN_MIDNIGHT},
|
||
|
{CFSTR("noon"), TOKEN_NOON},
|
||
|
{CFSTR("second"), TOKEN_SECOND},
|
||
|
{CFSTR("seconds"), TOKEN_SECOND},
|
||
|
{CFSTR("sec"), TOKEN_SECOND},
|
||
|
{CFSTR("s"), TOKEN_SECOND},
|
||
|
{CFSTR("minute"), TOKEN_MINUTE},
|
||
|
{CFSTR("minutes"), TOKEN_MINUTE},
|
||
|
{CFSTR("min"), TOKEN_MINUTE},
|
||
|
{CFSTR("mn"), TOKEN_MINUTE},
|
||
|
{CFSTR("m"), TOKEN_MINUTE},
|
||
|
{CFSTR("hour"), TOKEN_HOUR},
|
||
|
{CFSTR("hours"), TOKEN_HOUR},
|
||
|
{CFSTR("h"), TOKEN_HOUR},
|
||
|
{CFSTR("day"), TOKEN_DAY},
|
||
|
{CFSTR("days"), TOKEN_DAY},
|
||
|
{CFSTR("d"), TOKEN_DAY},
|
||
|
{CFSTR("week"), TOKEN_WEEK},
|
||
|
{CFSTR("weeks"), TOKEN_WEEK},
|
||
|
{CFSTR("w"), TOKEN_WEEK},
|
||
|
{CFSTR("month"), TOKEN_MONTH},
|
||
|
{CFSTR("months"), TOKEN_MONTH},
|
||
|
{CFSTR("year"), TOKEN_YEAR},
|
||
|
{CFSTR("years"), TOKEN_YEAR},
|
||
|
{CFSTR("y"), TOKEN_YEAR},
|
||
|
{CFSTR("january"), TOKEN_JANUARY},
|
||
|
{CFSTR("jan"), TOKEN_JANUARY},
|
||
|
{CFSTR("february"), TOKEN_FEBRUARY},
|
||
|
{CFSTR("feb"), TOKEN_FEBRUARY},
|
||
|
{CFSTR("march"), TOKEN_MARCH},
|
||
|
{CFSTR("mar"), TOKEN_MARCH},
|
||
|
{CFSTR("april"), TOKEN_APRIL},
|
||
|
{CFSTR("apr"), TOKEN_APRIL},
|
||
|
{CFSTR("may"), TOKEN_MAY},
|
||
|
{CFSTR("june"), TOKEN_JUNE},
|
||
|
{CFSTR("jun"), TOKEN_JUNE},
|
||
|
{CFSTR("july"), TOKEN_JULY},
|
||
|
{CFSTR("jul"), TOKEN_JULY},
|
||
|
{CFSTR("august"), TOKEN_AUGUST},
|
||
|
{CFSTR("aug"), TOKEN_AUGUST},
|
||
|
{CFSTR("september"), TOKEN_SEPTEMBER},
|
||
|
{CFSTR("sep"), TOKEN_SEPTEMBER},
|
||
|
{CFSTR("october"), TOKEN_OCTOBER},
|
||
|
{CFSTR("oct"), TOKEN_OCTOBER},
|
||
|
{CFSTR("november"), TOKEN_NOVEMBER},
|
||
|
{CFSTR("nov"), TOKEN_NOVEMBER},
|
||
|
{CFSTR("december"), TOKEN_DECEMBER},
|
||
|
{CFSTR("dec"), TOKEN_DECEMBER},
|
||
|
};
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
int IntegerField::LookupToken(CFStringRef t) {
|
||
|
for (int i=0;i<sizeof(Int_Tokens)/sizeof(tokenstruct);i++) {
|
||
|
if (CFStringCompare(Int_Tokens[i].token, t, kCFCompareCaseInsensitive) == kCFCompareEqualTo)
|
||
|
return Int_Tokens[i].tid;
|
||
|
}
|
||
|
return TOKEN_IDENTIFIER;
|
||
|
}
|
||
|
|
||
|
static int isallnum(CFStringRef p)
|
||
|
{
|
||
|
// TODO: ideally we should cache this or create it during initialization (but need to b be careful about thread safety)
|
||
|
CFCharacterSetRef set = CFCharacterSetCreateInvertedSet(NULL, CFCharacterSetGetPredefined(kCFCharacterSetDecimalDigit));
|
||
|
|
||
|
CFRange range;
|
||
|
Boolean ret = CFStringFindCharacterFromSet(p, set, CFRangeMake(0, CFStringGetLength(p)), 0, &range);
|
||
|
CFRelease(set);
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
static bool Ends(CFStringRef str1, CFStringRef str2)
|
||
|
{
|
||
|
|
||
|
CFRange findRange = CFStringFind(str1, str2, kCFCompareCaseInsensitive|kCFCompareAnchored|kCFCompareBackwards);
|
||
|
if (findRange.location == kCFNotFound)
|
||
|
return false;
|
||
|
else
|
||
|
return findRange.location != 0;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
int IntegerField::ApplyConversion(const char *format, TimeParse *tp) {
|
||
|
size_t size;
|
||
|
|
||
|
int value = GetValue();
|
||
|
CFStringRef token = NULL;
|
||
|
bool ago = false;
|
||
|
bool from = false;
|
||
|
bool kthis = false;
|
||
|
int what = TOKEN_MINUTE;
|
||
|
|
||
|
int lastnumber = value;
|
||
|
|
||
|
if (tp) {
|
||
|
tp->is_relative = 0;
|
||
|
tp->offset_value = 0;
|
||
|
tp->offset_whence = -1;
|
||
|
tp->offset_what = -1;
|
||
|
tp->offset_used = 0;
|
||
|
tp->relative_year = -1;
|
||
|
tp->relative_month = -1;
|
||
|
tp->relative_day = -1;
|
||
|
tp->relative_hour = -1;
|
||
|
tp->relative_min = -1;
|
||
|
tp->relative_sec = -1;
|
||
|
tp->relative_kwday = -1;
|
||
|
tp->absolute_hastime = 0;
|
||
|
tp->absolute_hasdate = 0;
|
||
|
}
|
||
|
|
||
|
time_t now;
|
||
|
time(&now);
|
||
|
|
||
|
struct tm *o = localtime(&now);
|
||
|
struct tm origin = *o;
|
||
|
struct tm origin_flags = {0,0,0,0,0,0,0,0,0};
|
||
|
|
||
|
struct tm onow = *o;
|
||
|
|
||
|
const char *p = format;
|
||
|
int t = -1;
|
||
|
int lastt = -1;
|
||
|
|
||
|
origin.tm_isdst = -1;
|
||
|
|
||
|
while (1) {
|
||
|
int save_lastt = lastt;
|
||
|
lastt = t;
|
||
|
t = Scanner::Query_GetNextToken(p, &size, &token, 1);
|
||
|
if (t == TOKEN_EOQ) break;
|
||
|
|
||
|
switch (t) {
|
||
|
case TOKEN_THIS:
|
||
|
kthis = true;
|
||
|
break;
|
||
|
case TOKEN_AGO:
|
||
|
case TOKEN_BEFORE: // before defaults to before now (= ago)
|
||
|
ago = true;
|
||
|
if (tp) {
|
||
|
tp->is_relative = 1;
|
||
|
tp->offset_whence = 1;
|
||
|
tp->offset_used = 1;
|
||
|
}
|
||
|
break;
|
||
|
case TOKEN_AFTER: // if after, ago is discarded, coz 5 mn ago after x has no meaning, so we get it as 5 mn after x
|
||
|
ago = false;
|
||
|
// no break
|
||
|
case TOKEN_FROM:
|
||
|
from = true;
|
||
|
if (tp) {
|
||
|
tp->is_relative = 1;
|
||
|
tp->offset_whence = 0;
|
||
|
tp->offset_used = 1;
|
||
|
}
|
||
|
break;
|
||
|
case TOKEN_DATE: {
|
||
|
if (!kthis) break;
|
||
|
kthis = false;
|
||
|
origin.tm_year = onow.tm_year;
|
||
|
origin_flags.tm_year = 1;
|
||
|
origin.tm_mon = onow.tm_mon;
|
||
|
origin_flags.tm_mon = 1;
|
||
|
origin.tm_mday = onow.tm_mday - onow.tm_wday;
|
||
|
origin_flags.tm_mday = 1;
|
||
|
if (!origin_flags.tm_hour)
|
||
|
origin.tm_hour = 0;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
if (tp) {
|
||
|
tp->relative_year = -1;
|
||
|
tp->relative_month = -1;
|
||
|
tp->relative_day = -1;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
case TOKEN_TIME: {
|
||
|
if (!kthis) break;
|
||
|
kthis = false;
|
||
|
origin.tm_hour = onow.tm_hour;
|
||
|
origin_flags.tm_hour = 1;
|
||
|
origin.tm_min = onow.tm_min;
|
||
|
origin_flags.tm_min = 1;
|
||
|
origin.tm_sec = onow.tm_sec;
|
||
|
origin_flags.tm_sec = 1;
|
||
|
if (tp) {
|
||
|
tp->relative_sec = -1;
|
||
|
tp->relative_min = -1;
|
||
|
tp->relative_hour = -1;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
case TOKEN_SECOND:
|
||
|
case TOKEN_MINUTE:
|
||
|
case TOKEN_HOUR:
|
||
|
case TOKEN_DAY:
|
||
|
case TOKEN_WEEK:
|
||
|
case TOKEN_MONTH:
|
||
|
case TOKEN_YEAR:
|
||
|
if (kthis) {
|
||
|
kthis = false;
|
||
|
switch (t) {
|
||
|
case TOKEN_SECOND:
|
||
|
origin.tm_sec = onow.tm_sec;
|
||
|
origin_flags.tm_sec = 1;
|
||
|
if (tp) tp->relative_sec = -1;
|
||
|
break;
|
||
|
case TOKEN_MINUTE:
|
||
|
origin.tm_min = onow.tm_min;
|
||
|
origin_flags.tm_min = 1;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
if (tp) tp->relative_min = -1;
|
||
|
break;
|
||
|
case TOKEN_HOUR:
|
||
|
origin.tm_hour = onow.tm_hour;
|
||
|
origin_flags.tm_hour = 1;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
if (tp) tp->relative_hour = -1;
|
||
|
break;
|
||
|
case TOKEN_DAY:
|
||
|
origin.tm_mday = onow.tm_mday;
|
||
|
origin_flags.tm_mday = 1;
|
||
|
if (!origin_flags.tm_hour)
|
||
|
origin.tm_hour = 0;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
if (tp) tp->relative_day = -1;
|
||
|
break;
|
||
|
case TOKEN_WEEK:
|
||
|
origin.tm_mday = onow.tm_mday - onow.tm_wday;
|
||
|
origin_flags.tm_mday = 1;
|
||
|
if (!origin_flags.tm_hour)
|
||
|
origin.tm_hour = 0;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
if (tp) tp->relative_day = -2;
|
||
|
break;
|
||
|
case TOKEN_MONTH:
|
||
|
origin.tm_mon = onow.tm_mon;
|
||
|
origin_flags.tm_mon = 1;
|
||
|
if (!origin_flags.tm_mday)
|
||
|
origin.tm_mday = 1;
|
||
|
if (!origin_flags.tm_hour)
|
||
|
origin.tm_hour = 0;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
if (tp) tp->relative_month = -1;
|
||
|
break;
|
||
|
case TOKEN_YEAR:
|
||
|
origin.tm_year = onow.tm_year;
|
||
|
origin_flags.tm_year = 1;
|
||
|
if (!origin_flags.tm_mon)
|
||
|
origin.tm_mon = 0;
|
||
|
if (!origin_flags.tm_mday)
|
||
|
origin.tm_mday = 1;
|
||
|
if (!origin_flags.tm_hour)
|
||
|
origin.tm_hour = 0;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
if (tp) tp->relative_year = -1;
|
||
|
break;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
if (lastnumber > 0) {
|
||
|
value = lastnumber;
|
||
|
lastnumber = 0;
|
||
|
if (tp) tp->offset_value = value;
|
||
|
}
|
||
|
what = t;
|
||
|
if (tp) {
|
||
|
switch (what) {
|
||
|
case TOKEN_SECOND:
|
||
|
tp->offset_what = 6; break;
|
||
|
case TOKEN_MINUTE:
|
||
|
tp->offset_what = 5; break;
|
||
|
case TOKEN_HOUR:
|
||
|
tp->offset_what = 4; break;
|
||
|
case TOKEN_DAY:
|
||
|
tp->offset_what = 3; break;
|
||
|
case TOKEN_WEEK:
|
||
|
tp->offset_what = 2; break;
|
||
|
case TOKEN_MONTH:
|
||
|
tp->offset_what = 1; break;
|
||
|
case TOKEN_YEAR:
|
||
|
tp->offset_what = 0; break;
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case TOKEN_SUNDAY:
|
||
|
case TOKEN_MONDAY:
|
||
|
case TOKEN_TUESDAY:
|
||
|
case TOKEN_WEDNESDAY:
|
||
|
case TOKEN_THURSDAY:
|
||
|
case TOKEN_FRIDAY:
|
||
|
case TOKEN_SATURDAY: {
|
||
|
kthis = false;
|
||
|
int dow = t-TOKEN_MONDAY;
|
||
|
if (dow > onow.tm_mday)
|
||
|
origin.tm_mday = 7 - (dow - onow.tm_mday);
|
||
|
else
|
||
|
origin.tm_mday = dow;
|
||
|
origin_flags.tm_mday = 1;
|
||
|
if (!origin_flags.tm_hour)
|
||
|
origin.tm_hour = 0;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
}
|
||
|
if (tp) tp->relative_kwday = t-TOKEN_SUNDAY;
|
||
|
break;
|
||
|
case TOKEN_MIDNIGHT:
|
||
|
kthis = false;
|
||
|
origin.tm_hour = 0;
|
||
|
origin_flags.tm_hour = 1;
|
||
|
if (!origin_flags.tm_min) {
|
||
|
if (tp) tp->relative_min = 0;
|
||
|
origin.tm_min = 0;
|
||
|
origin_flags.tm_min = 1;
|
||
|
}
|
||
|
if (!origin_flags.tm_sec) {
|
||
|
if (tp) tp->relative_sec = 0;
|
||
|
origin.tm_sec = 0;
|
||
|
origin_flags.tm_sec = 1;
|
||
|
}
|
||
|
if (tp) tp->relative_hour = 0;
|
||
|
break;
|
||
|
case TOKEN_NOON:
|
||
|
kthis = false;
|
||
|
origin.tm_hour = 12;
|
||
|
origin_flags.tm_hour = 1;
|
||
|
if (!origin_flags.tm_min) {
|
||
|
if (tp) tp->relative_min = 0;
|
||
|
origin.tm_min = 0;
|
||
|
origin_flags.tm_min = 1;
|
||
|
}
|
||
|
if (!origin_flags.tm_sec) {
|
||
|
if (tp) tp->relative_sec = 0;
|
||
|
origin.tm_sec = 0;
|
||
|
origin_flags.tm_sec = 1;
|
||
|
}
|
||
|
if (tp) tp->relative_hour = 12;
|
||
|
break;
|
||
|
case TOKEN_AM:
|
||
|
kthis = false;
|
||
|
if (lastnumber > 0) {
|
||
|
origin.tm_hour = lastnumber;
|
||
|
if (!origin_flags.tm_min) {
|
||
|
if (tp) tp->relative_min = 0;
|
||
|
origin.tm_min = 0;
|
||
|
origin_flags.tm_min = 1;
|
||
|
}
|
||
|
if (!origin_flags.tm_sec) {
|
||
|
if (tp) tp->relative_sec = 0;
|
||
|
origin.tm_sec = 0;
|
||
|
origin_flags.tm_sec = 1;
|
||
|
}
|
||
|
if (tp) tp->relative_hour = lastnumber;
|
||
|
lastnumber = 0;
|
||
|
} else {
|
||
|
if (origin.tm_hour > 12) origin.tm_hour -= 12;
|
||
|
if (tp) tp->relative_hour = origin.tm_hour;
|
||
|
}
|
||
|
origin_flags.tm_hour = 1;
|
||
|
break;
|
||
|
case TOKEN_PM:
|
||
|
kthis = false;
|
||
|
if (lastnumber > 0) {
|
||
|
origin.tm_hour = lastnumber > 12 ? lastnumber : lastnumber + 12;
|
||
|
if (!origin_flags.tm_min) {
|
||
|
if (tp) tp->relative_min = 0;
|
||
|
origin.tm_min = 0;
|
||
|
origin_flags.tm_min = 1;
|
||
|
}
|
||
|
if (!origin_flags.tm_sec) {
|
||
|
if (tp) tp->relative_sec = 0;
|
||
|
origin.tm_sec = 0;
|
||
|
origin_flags.tm_sec = 1;
|
||
|
}
|
||
|
if (tp) tp->relative_hour = lastnumber;
|
||
|
lastnumber = 0;
|
||
|
} else {
|
||
|
if (origin.tm_hour <= 12) origin.tm_hour += 12;
|
||
|
if (tp) tp->relative_hour = origin.tm_hour;
|
||
|
}
|
||
|
origin_flags.tm_hour = 1;
|
||
|
break;
|
||
|
case TOKEN_NOW:
|
||
|
kthis = false;
|
||
|
if (!origin_flags.tm_year) {
|
||
|
if (tp) tp->relative_year = -1;
|
||
|
origin.tm_year = onow.tm_year;
|
||
|
}
|
||
|
origin_flags.tm_year = 1;
|
||
|
if (!origin_flags.tm_mon) {
|
||
|
if (tp) tp->relative_month = -1;
|
||
|
origin.tm_mon = onow.tm_mon;
|
||
|
}
|
||
|
origin_flags.tm_mon = 1;
|
||
|
if (!origin_flags.tm_mday) {
|
||
|
if (tp) tp->relative_day = -1;
|
||
|
origin.tm_mday = onow.tm_mday;
|
||
|
}
|
||
|
origin_flags.tm_mday = 1;
|
||
|
if (!origin_flags.tm_hour) {
|
||
|
if (tp) tp->relative_hour = -1;
|
||
|
origin.tm_hour = onow.tm_hour;
|
||
|
}
|
||
|
origin_flags.tm_hour = 1;
|
||
|
if (!origin_flags.tm_min) {
|
||
|
if (tp) tp->relative_min = -1;
|
||
|
origin.tm_min = onow.tm_min;
|
||
|
}
|
||
|
origin_flags.tm_min = 1;
|
||
|
if (!origin_flags.tm_sec) {
|
||
|
if (tp) tp->relative_sec = -1;
|
||
|
origin.tm_sec = onow.tm_sec;
|
||
|
}
|
||
|
break;
|
||
|
case TOKEN_YESTERDAY:
|
||
|
kthis = false;
|
||
|
origin.tm_mday = onow.tm_mday - 1;
|
||
|
origin_flags.tm_mday = 1;
|
||
|
if (tp) tp->relative_kwday = 7;
|
||
|
break;
|
||
|
case TOKEN_TODAY:
|
||
|
origin.tm_mday = onow.tm_mday;
|
||
|
origin_flags.tm_mday = 1;
|
||
|
if (tp) tp->relative_kwday = 8;
|
||
|
break;
|
||
|
case TOKEN_TOMORROW:
|
||
|
kthis = false;
|
||
|
origin.tm_mday = onow.tm_mday + 1;
|
||
|
origin_flags.tm_mday = 1;
|
||
|
if (tp) tp->relative_kwday = 9;
|
||
|
break;
|
||
|
case TOKEN_JANUARY:
|
||
|
case TOKEN_FEBRUARY:
|
||
|
case TOKEN_MARCH:
|
||
|
case TOKEN_APRIL:
|
||
|
case TOKEN_MAY:
|
||
|
case TOKEN_JUNE:
|
||
|
case TOKEN_JULY:
|
||
|
case TOKEN_AUGUST:
|
||
|
case TOKEN_SEPTEMBER:
|
||
|
case TOKEN_OCTOBER:
|
||
|
case TOKEN_NOVEMBER:
|
||
|
case TOKEN_DECEMBER:
|
||
|
kthis = false;
|
||
|
if (lastnumber > 0) {
|
||
|
origin.tm_mday = lastnumber;
|
||
|
origin_flags.tm_mday = 1;
|
||
|
lastnumber = 0;
|
||
|
}
|
||
|
origin.tm_mon = t-TOKEN_JANUARY;
|
||
|
if (!origin_flags.tm_mday)
|
||
|
origin.tm_mday = 1;
|
||
|
if (!origin_flags.tm_hour)
|
||
|
origin.tm_hour = 0;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
origin_flags.tm_mon = 1;
|
||
|
if (tp) tp->relative_month = t-TOKEN_JANUARY;
|
||
|
break;
|
||
|
case TOKEN_IDENTIFIER:
|
||
|
{
|
||
|
kthis = false;
|
||
|
|
||
|
// check for a year value
|
||
|
int i = CFStringGetIntValue(token);
|
||
|
if (i > 1970 && i < 2038 && isallnum(token)) { // max time_t range
|
||
|
origin.tm_year = i-1900;
|
||
|
if (!origin_flags.tm_mday)
|
||
|
origin.tm_mday = 1;
|
||
|
if (!origin_flags.tm_mon)
|
||
|
origin.tm_mon = 0;
|
||
|
if (!origin_flags.tm_hour)
|
||
|
origin.tm_hour = 0;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
if (tp) tp->relative_year = i;
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
// check for 1st, 2nd, 3rd, 4th, etc.
|
||
|
if (Ends(token, CFSTR("st")) | Ends(token, CFSTR("nd")) | Ends(token, CFSTR("rd")) | Ends(token, CFSTR("th")))
|
||
|
{
|
||
|
int j = CFStringGetIntValue(token);
|
||
|
if (j >= 1 && j <= 31)
|
||
|
{
|
||
|
origin.tm_mday = j;
|
||
|
origin_flags.tm_mday = 1;
|
||
|
if (tp) tp->relative_day = j;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// check for a time string (##:##:##)
|
||
|
#ifdef _WIN32
|
||
|
z = wcschr(token, L':');
|
||
|
if (z)
|
||
|
{
|
||
|
if (tp) tp->absolute_hastime = 1;
|
||
|
wchar_t *zz = wcschr(z+1, L':');
|
||
|
int a, b, c=0;
|
||
|
a = myatoi(token, (int)(z-token));
|
||
|
if (zz && *(zz+1) == 0) zz = NULL;
|
||
|
if (zz && !isallnum(zz+1)) zz = NULL;
|
||
|
if (zz) { b = myatoi(z+1, (int)(zz-(z+1))); c = wcstol(zz+1,0,10); }
|
||
|
else b = wcstol(z+1,0,10);
|
||
|
origin.tm_hour = a;
|
||
|
origin.tm_min = b;
|
||
|
if (tp) {
|
||
|
tp->relative_hour = a;
|
||
|
tp->relative_min = b;
|
||
|
}
|
||
|
if (zz && !origin_flags.tm_sec) {
|
||
|
origin.tm_sec = c;
|
||
|
if (tp) tp->relative_sec = c;
|
||
|
} else if (!origin_flags.tm_sec) {
|
||
|
origin.tm_sec = 0;
|
||
|
}
|
||
|
origin_flags.tm_sec = 1;
|
||
|
origin_flags.tm_hour = 1;
|
||
|
origin_flags.tm_min = 1;
|
||
|
break;
|
||
|
}
|
||
|
#else
|
||
|
// TODO!!! maybe CFDateFormatterGetAbsoluteTimeFromString ?
|
||
|
#endif
|
||
|
|
||
|
// check for a date string in the format ##/##/##
|
||
|
#ifdef _WIN32
|
||
|
z = wcschr(token, L'/');
|
||
|
if (z) {
|
||
|
if (tp) tp->absolute_hasdate = 1;
|
||
|
wchar_t *zz = wcschr(z+1, L'/');
|
||
|
int a, b, c=onow.tm_year;
|
||
|
a = myatoi(token, (int)(z-token));
|
||
|
if (zz && !isallnum(zz+1)) zz = NULL;
|
||
|
if (zz && *(zz+1) == 0) zz = NULL;
|
||
|
if (zz) { b = myatoi(z+1, (int)(zz-(z+1))); c = wcstol(zz+1,0,10); }
|
||
|
else b = _wtoi(z+1);
|
||
|
if (b > 1969 && b < 2038) {
|
||
|
// mm/yyyy
|
||
|
origin.tm_year = b-1900;
|
||
|
origin_flags.tm_year = 1;
|
||
|
origin.tm_mon = a-1;
|
||
|
origin_flags.tm_mon = 1;
|
||
|
if (!origin_flags.tm_mday)
|
||
|
origin.tm_mday = 1;
|
||
|
if (!origin_flags.tm_hour)
|
||
|
origin.tm_hour = 0;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
if (tp) {
|
||
|
tp->relative_year = b;
|
||
|
tp->relative_month = a-1;
|
||
|
}
|
||
|
} else {
|
||
|
// mm/dd(/yy[yy])
|
||
|
if (c < 70) c += 100;
|
||
|
if (c > 138) c -= 1900;
|
||
|
origin.tm_year = c;
|
||
|
origin.tm_mon = a-1;
|
||
|
origin.tm_mday = b == 0 ? 1 : b;
|
||
|
origin_flags.tm_year = 1;
|
||
|
origin_flags.tm_mon = 1;
|
||
|
origin_flags.tm_mday = 1;
|
||
|
if (!origin_flags.tm_hour)
|
||
|
origin.tm_hour = 0;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
if (tp) {
|
||
|
tp->relative_year = c+1900;
|
||
|
tp->relative_month = a-1;
|
||
|
tp->relative_day = b;
|
||
|
}
|
||
|
}
|
||
|
origin_flags.tm_year = 1;
|
||
|
origin_flags.tm_mon = 1;
|
||
|
origin_flags.tm_mday = 1;
|
||
|
break;
|
||
|
}
|
||
|
#else
|
||
|
// TODO!!! maybe CFDateFormatterCreateDateFromString ?
|
||
|
#endif
|
||
|
|
||
|
if (isallnum(token))
|
||
|
{
|
||
|
lastnumber = i;
|
||
|
switch (lastt) {
|
||
|
case TOKEN_JANUARY:
|
||
|
case TOKEN_FEBRUARY:
|
||
|
case TOKEN_MARCH:
|
||
|
case TOKEN_APRIL:
|
||
|
case TOKEN_MAY:
|
||
|
case TOKEN_JUNE:
|
||
|
case TOKEN_JULY:
|
||
|
case TOKEN_AUGUST:
|
||
|
case TOKEN_SEPTEMBER:
|
||
|
case TOKEN_OCTOBER:
|
||
|
case TOKEN_NOVEMBER:
|
||
|
case TOKEN_DECEMBER:
|
||
|
origin.tm_mday = lastnumber;
|
||
|
origin_flags.tm_mday = 1;
|
||
|
lastnumber = 0;
|
||
|
if (!origin_flags.tm_hour)
|
||
|
origin.tm_hour = 0;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
if (tp) tp->relative_day = lastnumber;
|
||
|
break;
|
||
|
case TOKEN_AT: {
|
||
|
origin.tm_hour = lastnumber;
|
||
|
origin.tm_min = 0;
|
||
|
origin.tm_sec = 0;
|
||
|
origin_flags.tm_hour = 1;
|
||
|
origin_flags.tm_min = 1;
|
||
|
origin_flags.tm_sec = 1;
|
||
|
if (tp) {
|
||
|
tp->relative_hour = lastnumber;
|
||
|
tp->relative_min = 0;
|
||
|
tp->relative_sec = 0;
|
||
|
}
|
||
|
lastnumber = 0;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
break;
|
||
|
}
|
||
|
default:
|
||
|
lastt = save_lastt;
|
||
|
break;
|
||
|
}
|
||
|
p += size;
|
||
|
}
|
||
|
|
||
|
if (lastnumber) {
|
||
|
switch (lastt) {
|
||
|
case TOKEN_JANUARY:
|
||
|
case TOKEN_FEBRUARY:
|
||
|
case TOKEN_MARCH:
|
||
|
case TOKEN_APRIL:
|
||
|
case TOKEN_MAY:
|
||
|
case TOKEN_JUNE:
|
||
|
case TOKEN_JULY:
|
||
|
case TOKEN_AUGUST:
|
||
|
case TOKEN_SEPTEMBER:
|
||
|
case TOKEN_OCTOBER:
|
||
|
case TOKEN_NOVEMBER:
|
||
|
case TOKEN_DECEMBER:
|
||
|
origin.tm_mday = lastnumber;
|
||
|
lastnumber = 0;
|
||
|
if (!origin_flags.tm_hour)
|
||
|
origin.tm_hour = 0;
|
||
|
if (!origin_flags.tm_min)
|
||
|
origin.tm_min = 0;
|
||
|
if (!origin_flags.tm_sec)
|
||
|
origin.tm_sec = 0;
|
||
|
if (tp) tp->relative_day = lastnumber;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (ago) { // if ago (or before), from is optional since if it wasn't specified we use now
|
||
|
switch (what) {
|
||
|
case TOKEN_SECOND:
|
||
|
origin.tm_sec -= value;
|
||
|
break;
|
||
|
case TOKEN_MINUTE:
|
||
|
origin.tm_min -= value;
|
||
|
break;
|
||
|
case TOKEN_HOUR:
|
||
|
origin.tm_hour -= value;
|
||
|
break;
|
||
|
case TOKEN_DAY:
|
||
|
origin.tm_mday -= value;
|
||
|
break;
|
||
|
case TOKEN_WEEK:
|
||
|
origin.tm_mday -= value*7;
|
||
|
break;
|
||
|
case TOKEN_MONTH:
|
||
|
origin.tm_mon -= value;
|
||
|
break;
|
||
|
case TOKEN_YEAR:
|
||
|
origin.tm_year -= value;
|
||
|
break;
|
||
|
}
|
||
|
time_t o = mktime(&origin);
|
||
|
SetValue(o);
|
||
|
if (token) CFRelease(token);
|
||
|
|
||
|
if (tp) tp->absolute_datetime = GetValue();
|
||
|
return 1;
|
||
|
} else if (from) { // from (or after) was specified, but not ago, 5 mn from x is x + 5 mn
|
||
|
switch (what) {
|
||
|
case TOKEN_SECOND:
|
||
|
origin.tm_sec += value;
|
||
|
break;
|
||
|
case TOKEN_MINUTE:
|
||
|
origin.tm_min += value;
|
||
|
break;
|
||
|
case TOKEN_HOUR:
|
||
|
origin.tm_hour += value;
|
||
|
break;
|
||
|
case TOKEN_DAY:
|
||
|
origin.tm_mday += value;
|
||
|
break;
|
||
|
case TOKEN_WEEK:
|
||
|
origin.tm_mday += value*7;
|
||
|
break;
|
||
|
case TOKEN_MONTH:
|
||
|
origin.tm_mon += value;
|
||
|
break;
|
||
|
case TOKEN_YEAR:
|
||
|
origin.tm_year += value;
|
||
|
break;
|
||
|
}
|
||
|
time_t o = mktime(&origin);
|
||
|
SetValue(o);
|
||
|
|
||
|
if (token) CFRelease(token);
|
||
|
|
||
|
if (tp) tp->absolute_datetime = GetValue();
|
||
|
return 1;
|
||
|
} else { // none of ago/from/before/after were specified, just make a date/time with what we got and ignore our old value
|
||
|
time_t o = mktime(&origin);
|
||
|
SetValue(o);
|
||
|
|
||
|
if (token) CFRelease(token);
|
||
|
|
||
|
if (tp) tp->absolute_datetime = GetValue();
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
if (token) CFRelease(token);
|
||
|
if (tp) tp->absolute_datetime = GetValue();
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
DateTimeField::DateTimeField(int Val) : IntegerField(Val)
|
||
|
{
|
||
|
Type = FIELD_DATETIME;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
DateTimeField::DateTimeField()
|
||
|
{
|
||
|
Type = FIELD_DATETIME;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
DateTimeField::~DateTimeField()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
LengthField::LengthField(int Val) : IntegerField(Val)
|
||
|
{
|
||
|
Type = FIELD_LENGTH;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
LengthField::LengthField()
|
||
|
{
|
||
|
Type = FIELD_LENGTH;
|
||
|
}
|
||
|
|
||
|
//---------------------------------------------------------------------------
|
||
|
LengthField::~LengthField()
|
||
|
{
|
||
|
}
|