/**************************************************************************** ** ** Copyright (C) 2008 Nokia Corporation and/or its subsidiary(-ies). ** Contact: Qt Software Information (qt-info@nokia.com) ** ** This file is part of the $MODULE$ of the Qt Toolkit. ** ** $TROLLTECH_DUAL_LICENSE$ ** ****************************************************************************/ #include "qscriptdate_p.h" #ifndef QT_NO_SCRIPT #include #include #include #ifndef Q_WS_WIN # include # include #else # include #endif QT_BEGIN_NAMESPACE namespace QScript { qsreal ToInteger(qsreal n); static const qsreal HoursPerDay = 24.0; static const qsreal MinutesPerHour = 60.0; static const qsreal SecondsPerMinute = 60.0; static const qsreal msPerSecond = 1000.0; static const qsreal msPerMinute = 60000.0; static const qsreal msPerHour = 3600000.0; static const qsreal msPerDay = 86400000.0; static qsreal LocalTZA = 0.0; // initialized at startup static inline qsreal TimeWithinDay(qsreal t) { qsreal r = ::fmod(t, msPerDay); return (r >= 0) ? r : r + msPerDay; } static inline int HourFromTime(qsreal t) { int r = int(::fmod(::floor(t / msPerHour), HoursPerDay)); return (r >= 0) ? r : r + int(HoursPerDay); } static inline int MinFromTime(qsreal t) { int r = int(::fmod(::floor(t / msPerMinute), MinutesPerHour)); return (r >= 0) ? r : r + int(MinutesPerHour); } static inline int SecFromTime(qsreal t) { int r = int(::fmod(::floor(t / msPerSecond), SecondsPerMinute)); return (r >= 0) ? r : r + int(SecondsPerMinute); } static inline int msFromTime(qsreal t) { int r = int(::fmod(t, msPerSecond)); return (r >= 0) ? r : r + int(msPerSecond); } static inline qsreal Day(qsreal t) { return ::floor(t / msPerDay); } static inline qsreal DaysInYear(qsreal y) { if (::fmod(y, 4)) return 365; else if (::fmod(y, 100)) return 366; else if (::fmod(y, 400)) return 365; return 366; } static inline qsreal DayFromYear(qsreal y) { return 365 * (y - 1970) + ::floor((y - 1969) / 4) - ::floor((y - 1901) / 100) + ::floor((y - 1601) / 400); } static inline qsreal TimeFromYear(qsreal y) { return msPerDay * DayFromYear(y); } static inline qsreal YearFromTime(qsreal t) { int y = 1970; y += (int) ::floor(t / (msPerDay * 365.2425)); qsreal t2 = TimeFromYear(y); return (t2 > t) ? y - 1 : ((t2 + msPerDay * DaysInYear(y)) <= t) ? y + 1 : y; } static inline bool InLeapYear(qsreal t) { qsreal x = DaysInYear(YearFromTime(t)); if (x == 365) return 0; Q_ASSERT (x == 366); return 1; } static inline qsreal DayWithinYear(qsreal t) { return Day(t) - DayFromYear(YearFromTime(t)); } static inline qsreal MonthFromTime(qsreal t) { qsreal d = DayWithinYear(t); qsreal l = InLeapYear(t); if (d < 31.0) return 0; else if (d < 59.0 + l) return 1; else if (d < 90.0 + l) return 2; else if (d < 120.0 + l) return 3; else if (d < 151.0 + l) return 4; else if (d < 181.0 + l) return 5; else if (d < 212.0 + l) return 6; else if (d < 243.0 + l) return 7; else if (d < 273.0 + l) return 8; else if (d < 304.0 + l) return 9; else if (d < 334.0 + l) return 10; else if (d < 365.0 + l) return 11; return qSNaN(); // ### assert? } static inline qsreal DateFromTime(qsreal t) { int m = (int) ToInteger(MonthFromTime(t)); qsreal d = DayWithinYear(t); qsreal l = InLeapYear(t); switch (m) { case 0: return d + 1.0; case 1: return d - 30.0; case 2: return d - 58.0 - l; case 3: return d - 89.0 - l; case 4: return d - 119.0 - l; case 5: return d - 150.0 - l; case 6: return d - 180.0 - l; case 7: return d - 211.0 - l; case 8: return d - 242.0 - l; case 9: return d - 272.0 - l; case 10: return d - 303.0 - l; case 11: return d - 333.0 - l; } return qSNaN(); // ### assert } static inline qsreal WeekDay(qsreal t) { qsreal r = ::fmod (Day(t) + 4.0, 7.0); return (r >= 0) ? r : r + 7.0; } static inline qsreal MakeTime(qsreal hour, qsreal min, qsreal sec, qsreal ms) { return ((hour * MinutesPerHour + min) * SecondsPerMinute + sec) * msPerSecond + ms; } static inline qsreal DayFromMonth(qsreal month, qsreal leap) { switch ((int) month) { case 0: return 0; case 1: return 31.0; case 2: return 59.0 + leap; case 3: return 90.0 + leap; case 4: return 120.0 + leap; case 5: return 151.0 + leap; case 6: return 181.0 + leap; case 7: return 212.0 + leap; case 8: return 243.0 + leap; case 9: return 273.0 + leap; case 10: return 304.0 + leap; case 11: return 334.0 + leap; } return qSNaN(); // ### assert? } static qsreal MakeDay(qsreal year, qsreal month, qsreal day) { year += ::floor(month / 12.0); month = ::fmod(month, 12.0); if (month < 0) month += 12.0; qsreal t = TimeFromYear(year); qsreal leap = InLeapYear(t); day += ::floor(t / msPerDay); day += DayFromMonth(month, leap); return day - 1; } static inline qsreal MakeDate(qsreal day, qsreal time) { return day * msPerDay + time; } static inline qsreal DaylightSavingTA(double t) { #ifndef Q_WS_WIN long int tt = (long int)(t / msPerSecond); struct tm *tmtm = localtime((const time_t*)&tt); if (! tmtm) return 0; return (tmtm->tm_isdst > 0) ? msPerHour : 0; #else Q_UNUSED(t); /// ### implement me return 0; #endif } static inline qsreal LocalTime(qsreal t) { return t + LocalTZA + DaylightSavingTA(t); } static inline qsreal UTC(qsreal t) { return t - LocalTZA - DaylightSavingTA(t - LocalTZA); } static inline qsreal currentTime() { #ifndef Q_WS_WIN struct timeval tv; gettimeofday(&tv, 0); return ::floor(tv.tv_sec * msPerSecond + (tv.tv_usec / 1000.0)); #else SYSTEMTIME st; GetSystemTime(&st); FILETIME ft; SystemTimeToFileTime(&st, &ft); LARGE_INTEGER li; li.LowPart = ft.dwLowDateTime; li.HighPart = ft.dwHighDateTime; return double(li.QuadPart - Q_INT64_C(116444736000000000)) / 10000.0; #endif } static inline qsreal TimeClip(qsreal t) { if (! qIsFinite(t) || fabs(t) > 8.64e15) return qSNaN(); return ToInteger(t); } static inline qsreal ParseString(const QString &s) { QDateTime dt = QDateTime::fromString(s, Qt::TextDate); if (!dt.isValid()) dt = QDateTime::fromString(s, Qt::ISODate); if (!dt.isValid()) { QStringList formats; formats << QLatin1String("M/d/yyyy") << QLatin1String("M/d/yyyy hh:mm") << QLatin1String("M/d/yyyy hh:mm A") << QLatin1String("M/d/yyyy, hh:mm") << QLatin1String("M/d/yyyy, hh:mm A") << QLatin1String("MMM d yyyy") << QLatin1String("MMM d yyyy hh:mm") << QLatin1String("MMM d yyyy hh:mm:ss") << QLatin1String("MMM d yyyy, hh:mm") << QLatin1String("MMM d yyyy, hh:mm:ss") << QLatin1String("MMMM d yyyy") << QLatin1String("MMMM d yyyy hh:mm") << QLatin1String("MMMM d yyyy hh:mm:ss") << QLatin1String("MMMM d yyyy, hh:mm") << QLatin1String("MMMM d yyyy, hh:mm:ss") << QLatin1String("MMM d, yyyy") << QLatin1String("MMM d, yyyy hh:mm") << QLatin1String("MMM d, yyyy hh:mm:ss") << QLatin1String("MMMM d, yyyy") << QLatin1String("MMMM d, yyyy hh:mm") << QLatin1String("MMMM d, yyyy hh:mm:ss") << QLatin1String("d MMM yyyy") << QLatin1String("d MMM yyyy hh:mm") << QLatin1String("d MMM yyyy hh:mm:ss") << QLatin1String("d MMM yyyy, hh:mm") << QLatin1String("d MMM yyyy, hh:mm:ss") << QLatin1String("d MMMM yyyy") << QLatin1String("d MMMM yyyy hh:mm") << QLatin1String("d MMMM yyyy hh:mm:ss") << QLatin1String("d MMMM yyyy, hh:mm") << QLatin1String("d MMMM yyyy, hh:mm:ss") << QLatin1String("d MMM, yyyy") << QLatin1String("d MMM, yyyy hh:mm") << QLatin1String("d MMM, yyyy hh:mm:ss") << QLatin1String("d MMMM, yyyy") << QLatin1String("d MMMM, yyyy hh:mm") << QLatin1String("d MMMM, yyyy hh:mm:ss"); for (int i = 0; i < formats.size(); ++i) { dt = QDateTime::fromString(s, formats.at(i)); if (dt.isValid()) break; } } return FromDateTime(dt); } static inline QString ToString(qsreal t) { if (qIsNaN(t)) return QLatin1String("Invalid Date"); QString str = ToDateTime(t, Qt::LocalTime).toString() + QLatin1String(" GMT"); qsreal tzoffset = LocalTZA + DaylightSavingTA(t); if (tzoffset) { int hours = tzoffset / 1000 / 60 / 60; int mins = int(tzoffset / 1000 / 60) % 60; str.append(QLatin1Char((tzoffset > 0) ? '+' : '-')); if (hours < 10) str.append(QLatin1Char('0')); str.append(QString::number(hours)); if (mins < 10) str.append(QLatin1Char('0')); str.append(QString::number(mins)); } return str; } static inline QString ToUTCString(qsreal t) { if (qIsNaN(t)) return QLatin1String("Invalid Date"); return ToDateTime(t, Qt::UTC).toString() + QLatin1String(" GMT"); } static inline QString ToDateString(qsreal t) { return ToDateTime(t, Qt::LocalTime).date().toString(); } static inline QString ToTimeString(qsreal t) { return ToDateTime(t, Qt::LocalTime).time().toString(); } static inline QString ToLocaleString(qsreal t) { return ToDateTime(t, Qt::LocalTime).toString(Qt::LocaleDate); } static inline QString ToLocaleDateString(qsreal t) { return ToDateTime(t, Qt::LocalTime).date().toString(Qt::LocaleDate); } static inline QString ToLocaleTimeString(qsreal t) { return ToDateTime(t, Qt::LocalTime).time().toString(Qt::LocaleDate); } static qsreal getLocalTZA() { #ifndef Q_WS_WIN struct tm* t; time_t curr; time(&curr); t = localtime(&curr); time_t locl = mktime(t); t = gmtime(&curr); time_t globl = mktime(t); return double(locl - globl) * 1000.0; #else TIME_ZONE_INFORMATION tzInfo; GetTimeZoneInformation(&tzInfo); return -tzInfo.Bias * 60.0 * 1000.0; #endif } /*! \internal Converts the QDateTime \dt to an ECMA Date value (in UTC form). */ qsreal FromDateTime(const QDateTime &dt) { if (!dt.isValid()) return qSNaN(); if (!LocalTZA) // ### move LocalTZA = getLocalTZA(); QDate date = dt.date(); QTime taim = dt.time(); int year = date.year(); int month = date.month() - 1; int day = date.day(); int hours = taim.hour(); int mins = taim.minute(); int secs = taim.second(); int ms = taim.msec(); double t = MakeDate(MakeDay(year, month, day), MakeTime(hours, mins, secs, ms)); if (dt.timeSpec() == Qt::LocalTime) t = UTC(t); return TimeClip(t); } /*! \internal Converts the ECMA Date value \tt (in UTC form) to QDateTime according to \a spec. */ QDateTime ToDateTime(qsreal t, Qt::TimeSpec spec) { if (qIsNaN(t)) return QDateTime(); if (!LocalTZA) // ### move LocalTZA = getLocalTZA(); if (spec == Qt::LocalTime) t = LocalTime(t); int year = int(YearFromTime(t)); int month = int(MonthFromTime(t) + 1); int day = int(DateFromTime(t)); int hours = HourFromTime(t); int mins = MinFromTime(t); int secs = SecFromTime(t); int ms = msFromTime(t); return QDateTime(QDate(year, month, day), QTime(hours, mins, secs, ms), spec); } } // namespace QScript QT_END_NAMESPACE #endif // QT_NO_SCRIPT