/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the QtQml module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3.0 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU General Public License version 3.0 requirements will be ** met: http://www.gnu.org/copyleft/gpl.html. ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qv4dateobject_p.h" #include "qv4objectproto_p.h" #include "qv4scopedvalue_p.h" #include #include #include #include #include #include #include #include #include #include #include #include #ifdef Q_OS_WIN # include #else # ifndef Q_OS_VXWORKS # include # else # include "qplatformdefs.h" # endif #endif using namespace QV4; static const double HoursPerDay = 24.0; static const double MinutesPerHour = 60.0; static const double SecondsPerMinute = 60.0; static const double msPerSecond = 1000.0; static const double msPerMinute = 60000.0; static const double msPerHour = 3600000.0; static const double msPerDay = 86400000.0; static double LocalTZA = 0.0; // initialized at startup static inline double TimeWithinDay(double t) { double r = ::fmod(t, msPerDay); return (r >= 0) ? r : r + msPerDay; } static inline int HourFromTime(double t) { int r = int(::fmod(::floor(t / msPerHour), HoursPerDay)); return (r >= 0) ? r : r + int(HoursPerDay); } static inline int MinFromTime(double t) { int r = int(::fmod(::floor(t / msPerMinute), MinutesPerHour)); return (r >= 0) ? r : r + int(MinutesPerHour); } static inline int SecFromTime(double t) { int r = int(::fmod(::floor(t / msPerSecond), SecondsPerMinute)); return (r >= 0) ? r : r + int(SecondsPerMinute); } static inline int msFromTime(double t) { int r = int(::fmod(t, msPerSecond)); return (r >= 0) ? r : r + int(msPerSecond); } static inline double Day(double t) { return ::floor(t / msPerDay); } static inline double DaysInYear(double 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 double DayFromYear(double y) { return 365 * (y - 1970) + ::floor((y - 1969) / 4) - ::floor((y - 1901) / 100) + ::floor((y - 1601) / 400); } static inline double TimeFromYear(double y) { return msPerDay * DayFromYear(y); } static inline double YearFromTime(double t) { int y = 1970; y += (int) ::floor(t / (msPerDay * 365.2425)); double t2 = TimeFromYear(y); return (t2 > t) ? y - 1 : ((t2 + msPerDay * DaysInYear(y)) <= t) ? y + 1 : y; } static inline bool InLeapYear(double t) { double x = DaysInYear(YearFromTime(t)); if (x == 365) return 0; assert(x == 366); return 1; } static inline double DayWithinYear(double t) { return Day(t) - DayFromYear(YearFromTime(t)); } static inline double MonthFromTime(double t) { double d = DayWithinYear(t); double 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 double DateFromTime(double t) { int m = (int) Primitive::toInteger(MonthFromTime(t)); double d = DayWithinYear(t); double 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 double WeekDay(double t) { double r = ::fmod (Day(t) + 4.0, 7.0); return (r >= 0) ? r : r + 7.0; } static inline double MakeTime(double hour, double min, double sec, double ms) { return ((hour * MinutesPerHour + min) * SecondsPerMinute + sec) * msPerSecond + ms; } static inline double DayFromMonth(double month, double 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 double MakeDay(double year, double month, double day) { year += ::floor(month / 12.0); month = ::fmod(month, 12.0); if (month < 0) month += 12.0; double d = DayFromYear(year); bool leap = InLeapYear(d*msPerDay); d += DayFromMonth(month, leap); d += day - 1; return d; } static inline double MakeDate(double day, double time) { return day * msPerDay + time; } static inline double DaylightSavingTA(double t) { struct tm tmtm; #if defined(_MSC_VER) && _MSC_VER >= 1400 __time64_t tt = (__time64_t)(t / msPerSecond); // _localtime_64_s returns non-zero on failure if (_localtime64_s(&tmtm, &tt) != 0) #else long int tt = (long int)(t / msPerSecond); if (!localtime_r((const time_t*) &tt, &tmtm)) #endif return 0; return (tmtm.tm_isdst > 0) ? msPerHour : 0; } static inline double LocalTime(double t) { return t + LocalTZA + DaylightSavingTA(t); } static inline double UTC(double t) { return t - LocalTZA - DaylightSavingTA(t - LocalTZA); } static inline double currentTime() { #ifndef Q_OS_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 double TimeClip(double t) { if (! qIsFinite(t) || fabs(t) > 8.64e15) return qSNaN(); return Primitive::toInteger(t); } static inline double ParseString(const QString &s) { // first try the format defined in 15.9.1.15, only if that fails fall back to // QDateTime for parsing // the define string format is YYYY-MM-DDTHH:mm:ss.sssZ // It can be date or time only, and the second and later components // of both fields are optional // and extended syntax for negative and large positive years exists: +/-YYYYYY enum Format { Year, Month, Day, Hour, Minute, Second, MilliSecond, TimezoneHour, TimezoneMinute, Done }; const QChar *ch = s.constData(); const QChar *end = ch + s.length(); uint format = Year; int current = 0; int currentSize = 0; bool extendedYear = false; int yearSign = 1; int year = 0; int month = 0; int day = 1; int hour = 0; int minute = 0; int second = 0; int msec = 0; int offsetSign = 1; int offset = 0; bool error = false; if (*ch == '+' || *ch == '-') { extendedYear = true; if (*ch == '-') yearSign = -1; ++ch; } while (ch <= end) { if (*ch >= '0' && *ch <= '9') { current *= 10; current += ch->unicode() - '0'; ++currentSize; } else { // other char, delimits field switch (format) { case Year: year = current; if (extendedYear) error = (currentSize != 6); else error = (currentSize != 4); break; case Month: month = current - 1; error = (currentSize != 2) || month > 11; break; case Day: day = current; error = (currentSize != 2) || day > 31; break; case Hour: hour = current; error = (currentSize != 2) || hour > 24; break; case Minute: minute = current; error = (currentSize != 2) || minute > 60; break; case Second: second = current; error = (currentSize != 2) || second > 60; break; case MilliSecond: msec = current; error = (currentSize != 3); break; case TimezoneHour: offset = current*60; error = (currentSize != 2) || offset > 23*60; break; case TimezoneMinute: offset += current; error = (currentSize != 2) || current >= 60; break; } if (*ch == 'T') { if (format >= Hour) error = true; format = Hour; } else if (*ch == '-') { if (format < Day) ++format; else if (format < Minute) error = true; else if (format >= TimezoneHour) error = true; else { offsetSign = -1; format = TimezoneHour; } } else if (*ch == ':') { if (format != Hour && format != Minute && format != TimezoneHour) error = true; ++format; } else if (*ch == '.') { if (format != Second) error = true; ++format; } else if (*ch == '+') { if (format < Minute || format >= TimezoneHour) error = true; format = TimezoneHour; } else if (*ch == 'Z' || *ch == 0) { format = Done; } current = 0; currentSize = 0; } if (error || format == Done) break; ++ch; } if (!error) { double t = MakeDate(MakeDay(year * yearSign, month, day), MakeTime(hour, minute, second, msec)); t -= offset * offsetSign * 60 * 1000; return t; } QDateTime dt = QDateTime::fromString(s, Qt::TextDate); if (!dt.isValid()) dt = QDateTime::fromString(s, Qt::ISODate); if (!dt.isValid()) { QStringList formats; formats << QStringLiteral("M/d/yyyy") << QStringLiteral("M/d/yyyy hh:mm") << QStringLiteral("M/d/yyyy hh:mm A") << QStringLiteral("M/d/yyyy, hh:mm") << QStringLiteral("M/d/yyyy, hh:mm A") << QStringLiteral("MMM d yyyy") << QStringLiteral("MMM d yyyy hh:mm") << QStringLiteral("MMM d yyyy hh:mm:ss") << QStringLiteral("MMM d yyyy, hh:mm") << QStringLiteral("MMM d yyyy, hh:mm:ss") << QStringLiteral("MMMM d yyyy") << QStringLiteral("MMMM d yyyy hh:mm") << QStringLiteral("MMMM d yyyy hh:mm:ss") << QStringLiteral("MMMM d yyyy, hh:mm") << QStringLiteral("MMMM d yyyy, hh:mm:ss") << QStringLiteral("MMM d, yyyy") << QStringLiteral("MMM d, yyyy hh:mm") << QStringLiteral("MMM d, yyyy hh:mm:ss") << QStringLiteral("MMMM d, yyyy") << QStringLiteral("MMMM d, yyyy hh:mm") << QStringLiteral("MMMM d, yyyy hh:mm:ss") << QStringLiteral("d MMM yyyy") << QStringLiteral("d MMM yyyy hh:mm") << QStringLiteral("d MMM yyyy hh:mm:ss") << QStringLiteral("d MMM yyyy, hh:mm") << QStringLiteral("d MMM yyyy, hh:mm:ss") << QStringLiteral("d MMMM yyyy") << QStringLiteral("d MMMM yyyy hh:mm") << QStringLiteral("d MMMM yyyy hh:mm:ss") << QStringLiteral("d MMMM yyyy, hh:mm") << QStringLiteral("d MMMM yyyy, hh:mm:ss") << QStringLiteral("d MMM, yyyy") << QStringLiteral("d MMM, yyyy hh:mm") << QStringLiteral("d MMM, yyyy hh:mm:ss") << QStringLiteral("d MMMM, yyyy") << QStringLiteral("d MMMM, yyyy hh:mm") << QStringLiteral("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; } } if (!dt.isValid()) return qSNaN(); return dt.toMSecsSinceEpoch(); } /*! \internal Converts the ECMA Date value \tt (in UTC form) to QDateTime according to \a spec. */ static inline QDateTime ToDateTime(double t, Qt::TimeSpec spec) { if (std::isnan(t)) return QDateTime(); 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); } static inline QString ToString(double t) { if (std::isnan(t)) return QStringLiteral("Invalid Date"); QString str = ToDateTime(t, Qt::LocalTime).toString() + QStringLiteral(" GMT"); double tzoffset = LocalTZA + DaylightSavingTA(t); if (tzoffset) { int hours = static_cast(::fabs(tzoffset) / 1000 / 60 / 60); int mins = int(::fabs(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(double t) { if (std::isnan(t)) return QStringLiteral("Invalid Date"); return ToDateTime(t, Qt::UTC).toString() + QStringLiteral(" GMT"); } static inline QString ToDateString(double t) { return ToDateTime(t, Qt::LocalTime).date().toString(); } static inline QString ToTimeString(double t) { return ToDateTime(t, Qt::LocalTime).time().toString(); } static inline QString ToLocaleString(double t) { return ToDateTime(t, Qt::LocalTime).toString(Qt::LocaleDate); } static inline QString ToLocaleDateString(double t) { return ToDateTime(t, Qt::LocalTime).date().toString(Qt::LocaleDate); } static inline QString ToLocaleTimeString(double t) { return ToDateTime(t, Qt::LocalTime).time().toString(Qt::LocaleDate); } static double getLocalTZA() { #ifndef Q_OS_WIN struct tm t; time_t curr; time(&curr); localtime_r(&curr, &t); time_t locl = mktime(&t); gmtime_r(&curr, &t); 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 } DEFINE_MANAGED_VTABLE(DateObject); DateObject::DateObject(ExecutionEngine *engine, const QDateTime &date) : Object(engine->dateClass) { vtbl = &static_vtbl; type = Type_DateObject; value.setDouble(date.isValid() ? date.toMSecsSinceEpoch() : qSNaN()); } QDateTime DateObject::toQDateTime() const { return ToDateTime(value.asDouble(), Qt::LocalTime); } DEFINE_MANAGED_VTABLE(DateCtor); DateCtor::DateCtor(ExecutionContext *scope) : FunctionObject(scope, QStringLiteral("Date")) { vtbl = &static_vtbl; } ReturnedValue DateCtor::construct(Managed *m, CallData *callData) { double t = 0; if (callData->argc == 0) t = currentTime(); else if (callData->argc == 1) { Scope scope(m->engine()); ScopedValue arg(scope, callData->args[0]); if (DateObject *d = arg->asDateObject()) arg = d->value; else arg = __qmljs_to_primitive(arg, PREFERREDTYPE_HINT); if (arg->isString()) t = ParseString(arg->stringValue()->toQString()); else t = TimeClip(arg->toNumber()); } else { // d.argc > 1 double year = callData->args[0].toNumber(); double month = callData->args[1].toNumber(); double day = callData->argc >= 3 ? callData->args[2].toNumber() : 1; double hours = callData->argc >= 4 ? callData->args[3].toNumber() : 0; double mins = callData->argc >= 5 ? callData->args[4].toNumber() : 0; double secs = callData->argc >= 6 ? callData->args[5].toNumber() : 0; double ms = callData->argc >= 7 ? callData->args[6].toNumber() : 0; if (year >= 0 && year <= 99) year += 1900; t = MakeDate(MakeDay(year, month, day), MakeTime(hours, mins, secs, ms)); t = TimeClip(UTC(t)); } return Encode(m->engine()->newDateObject(Primitive::fromDouble(t))); } ReturnedValue DateCtor::call(Managed *m, CallData *) { double t = currentTime(); return m->engine()->newString(ToString(t))->asReturnedValue(); } void DatePrototype::init(ExecutionEngine *engine, ObjectRef ctor) { Scope scope(engine); ScopedObject o(scope); ctor->defineReadonlyProperty(engine->id_prototype, (o = this)); ctor->defineReadonlyProperty(engine->id_length, Primitive::fromInt32(7)); LocalTZA = getLocalTZA(); ctor->defineDefaultProperty(QStringLiteral("parse"), method_parse, 1); ctor->defineDefaultProperty(QStringLiteral("UTC"), method_UTC, 7); ctor->defineDefaultProperty(QStringLiteral("now"), method_now, 0); defineDefaultProperty(QStringLiteral("constructor"), (o = ctor)); defineDefaultProperty(engine->id_toString, method_toString, 0); defineDefaultProperty(QStringLiteral("toDateString"), method_toDateString, 0); defineDefaultProperty(QStringLiteral("toTimeString"), method_toTimeString, 0); defineDefaultProperty(QStringLiteral("toLocaleString"), method_toLocaleString, 0); defineDefaultProperty(QStringLiteral("toLocaleDateString"), method_toLocaleDateString, 0); defineDefaultProperty(QStringLiteral("toLocaleTimeString"), method_toLocaleTimeString, 0); defineDefaultProperty(engine->id_valueOf, method_valueOf, 0); defineDefaultProperty(QStringLiteral("getTime"), method_getTime, 0); defineDefaultProperty(QStringLiteral("getYear"), method_getYear, 0); defineDefaultProperty(QStringLiteral("getFullYear"), method_getFullYear, 0); defineDefaultProperty(QStringLiteral("getUTCFullYear"), method_getUTCFullYear, 0); defineDefaultProperty(QStringLiteral("getMonth"), method_getMonth, 0); defineDefaultProperty(QStringLiteral("getUTCMonth"), method_getUTCMonth, 0); defineDefaultProperty(QStringLiteral("getDate"), method_getDate, 0); defineDefaultProperty(QStringLiteral("getUTCDate"), method_getUTCDate, 0); defineDefaultProperty(QStringLiteral("getDay"), method_getDay, 0); defineDefaultProperty(QStringLiteral("getUTCDay"), method_getUTCDay, 0); defineDefaultProperty(QStringLiteral("getHours"), method_getHours, 0); defineDefaultProperty(QStringLiteral("getUTCHours"), method_getUTCHours, 0); defineDefaultProperty(QStringLiteral("getMinutes"), method_getMinutes, 0); defineDefaultProperty(QStringLiteral("getUTCMinutes"), method_getUTCMinutes, 0); defineDefaultProperty(QStringLiteral("getSeconds"), method_getSeconds, 0); defineDefaultProperty(QStringLiteral("getUTCSeconds"), method_getUTCSeconds, 0); defineDefaultProperty(QStringLiteral("getMilliseconds"), method_getMilliseconds, 0); defineDefaultProperty(QStringLiteral("getUTCMilliseconds"), method_getUTCMilliseconds, 0); defineDefaultProperty(QStringLiteral("getTimezoneOffset"), method_getTimezoneOffset, 0); defineDefaultProperty(QStringLiteral("setTime"), method_setTime, 1); defineDefaultProperty(QStringLiteral("setMilliseconds"), method_setMilliseconds, 1); defineDefaultProperty(QStringLiteral("setUTCMilliseconds"), method_setUTCMilliseconds, 1); defineDefaultProperty(QStringLiteral("setSeconds"), method_setSeconds, 2); defineDefaultProperty(QStringLiteral("setUTCSeconds"), method_setUTCSeconds, 2); defineDefaultProperty(QStringLiteral("setMinutes"), method_setMinutes, 3); defineDefaultProperty(QStringLiteral("setUTCMinutes"), method_setUTCMinutes, 3); defineDefaultProperty(QStringLiteral("setHours"), method_setHours, 4); defineDefaultProperty(QStringLiteral("setUTCHours"), method_setUTCHours, 4); defineDefaultProperty(QStringLiteral("setDate"), method_setDate, 1); defineDefaultProperty(QStringLiteral("setUTCDate"), method_setUTCDate, 1); defineDefaultProperty(QStringLiteral("setMonth"), method_setMonth, 2); defineDefaultProperty(QStringLiteral("setUTCMonth"), method_setUTCMonth, 2); defineDefaultProperty(QStringLiteral("setYear"), method_setYear, 1); defineDefaultProperty(QStringLiteral("setFullYear"), method_setFullYear, 3); defineDefaultProperty(QStringLiteral("setUTCFullYear"), method_setUTCFullYear, 3); defineDefaultProperty(QStringLiteral("toUTCString"), method_toUTCString, 0); defineDefaultProperty(QStringLiteral("toGMTString"), method_toUTCString, 0); defineDefaultProperty(QStringLiteral("toISOString"), method_toISOString, 0); defineDefaultProperty(QStringLiteral("toJSON"), method_toJSON, 1); } double DatePrototype::getThisDate(ExecutionContext *ctx) { if (DateObject *thisObject = ctx->callData->thisObject.asDateObject()) return thisObject->value.asDouble(); else { ctx->throwTypeError(); return 0; } } ReturnedValue DatePrototype::method_parse(SimpleCallContext *ctx) { if (!ctx->callData->argc) return Encode(qSNaN()); return Encode(ParseString(ctx->callData->args[0].toString(ctx)->toQString())); } ReturnedValue DatePrototype::method_UTC(SimpleCallContext *ctx) { const int numArgs = ctx->callData->argc; if (numArgs >= 2) { double year = ctx->callData->args[0].toNumber(); double month = ctx->callData->args[1].toNumber(); double day = numArgs >= 3 ? ctx->callData->args[2].toNumber() : 1; double hours = numArgs >= 4 ? ctx->callData->args[3].toNumber() : 0; double mins = numArgs >= 5 ? ctx->callData->args[4].toNumber() : 0; double secs = numArgs >= 6 ? ctx->callData->args[5].toNumber() : 0; double ms = numArgs >= 7 ? ctx->callData->args[6].toNumber() : 0; if (year >= 0 && year <= 99) year += 1900; double t = MakeDate(MakeDay(year, month, day), MakeTime(hours, mins, secs, ms)); return Encode(TimeClip(t)); } return Encode::undefined(); } ReturnedValue DatePrototype::method_now(SimpleCallContext *ctx) { Q_UNUSED(ctx); double t = currentTime(); return Encode(t); } ReturnedValue DatePrototype::method_toString(SimpleCallContext *ctx) { double t = getThisDate(ctx); return ctx->engine->newString(ToString(t))->asReturnedValue(); } ReturnedValue DatePrototype::method_toDateString(SimpleCallContext *ctx) { double t = getThisDate(ctx); return ctx->engine->newString(ToDateString(t))->asReturnedValue(); } ReturnedValue DatePrototype::method_toTimeString(SimpleCallContext *ctx) { double t = getThisDate(ctx); return ctx->engine->newString(ToTimeString(t))->asReturnedValue(); } ReturnedValue DatePrototype::method_toLocaleString(SimpleCallContext *ctx) { double t = getThisDate(ctx); return ctx->engine->newString(ToLocaleString(t))->asReturnedValue(); } ReturnedValue DatePrototype::method_toLocaleDateString(SimpleCallContext *ctx) { double t = getThisDate(ctx); return ctx->engine->newString(ToLocaleDateString(t))->asReturnedValue(); } ReturnedValue DatePrototype::method_toLocaleTimeString(SimpleCallContext *ctx) { double t = getThisDate(ctx); return ctx->engine->newString(ToLocaleTimeString(t))->asReturnedValue(); } ReturnedValue DatePrototype::method_valueOf(SimpleCallContext *ctx) { double t = getThisDate(ctx); return Encode(t); } ReturnedValue DatePrototype::method_getTime(SimpleCallContext *ctx) { double t = getThisDate(ctx); return Encode(t); } ReturnedValue DatePrototype::method_getYear(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = YearFromTime(LocalTime(t)) - 1900; return Encode(t); } ReturnedValue DatePrototype::method_getFullYear(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = YearFromTime(LocalTime(t)); return Encode(t); } ReturnedValue DatePrototype::method_getUTCFullYear(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = YearFromTime(t); return Encode(t); } ReturnedValue DatePrototype::method_getMonth(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = MonthFromTime(LocalTime(t)); return Encode(t); } ReturnedValue DatePrototype::method_getUTCMonth(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = MonthFromTime(t); return Encode(t); } ReturnedValue DatePrototype::method_getDate(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = DateFromTime(LocalTime(t)); return Encode(t); } ReturnedValue DatePrototype::method_getUTCDate(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = DateFromTime(t); return Encode(t); } ReturnedValue DatePrototype::method_getDay(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = WeekDay(LocalTime(t)); return Encode(t); } ReturnedValue DatePrototype::method_getUTCDay(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = WeekDay(t); return Encode(t); } ReturnedValue DatePrototype::method_getHours(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = HourFromTime(LocalTime(t)); return Encode(t); } ReturnedValue DatePrototype::method_getUTCHours(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = HourFromTime(t); return Encode(t); } ReturnedValue DatePrototype::method_getMinutes(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = MinFromTime(LocalTime(t)); return Encode(t); } ReturnedValue DatePrototype::method_getUTCMinutes(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = MinFromTime(t); return Encode(t); } ReturnedValue DatePrototype::method_getSeconds(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = SecFromTime(LocalTime(t)); return Encode(t); } ReturnedValue DatePrototype::method_getUTCSeconds(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = SecFromTime(t); return Encode(t); } ReturnedValue DatePrototype::method_getMilliseconds(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = msFromTime(LocalTime(t)); return Encode(t); } ReturnedValue DatePrototype::method_getUTCMilliseconds(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = msFromTime(t); return Encode(t); } ReturnedValue DatePrototype::method_getTimezoneOffset(SimpleCallContext *ctx) { double t = getThisDate(ctx); if (! std::isnan(t)) t = (t - LocalTime(t)) / msPerMinute; return Encode(t); } ReturnedValue DatePrototype::method_setTime(SimpleCallContext *ctx) { Scope scope(ctx); Scoped self(scope, ctx->callData->thisObject); if (!self) ctx->throwTypeError(); double t = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); self->value.setDouble(TimeClip(t)); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setMilliseconds(SimpleCallContext *ctx) { Scope scope(ctx); Scoped self(scope, ctx->callData->thisObject); if (!self) ctx->throwTypeError(); double t = LocalTime(self->value.asDouble()); double ms = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); self->value.setDouble(TimeClip(UTC(MakeDate(Day(t), MakeTime(HourFromTime(t), MinFromTime(t), SecFromTime(t), ms))))); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setUTCMilliseconds(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = self->value.asDouble(); double ms = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); self->value.setDouble(TimeClip(UTC(MakeDate(Day(t), MakeTime(HourFromTime(t), MinFromTime(t), SecFromTime(t), ms))))); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setSeconds(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = LocalTime(self->value.asDouble()); double sec = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); double ms = (ctx->callData->argc < 2) ? msFromTime(t) : ctx->callData->args[1].toNumber(); t = TimeClip(UTC(MakeDate(Day(t), MakeTime(HourFromTime(t), MinFromTime(t), sec, ms)))); self->value.setDouble(t); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setUTCSeconds(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = self->value.asDouble(); double sec = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); double ms = (ctx->callData->argc < 2) ? msFromTime(t) : ctx->callData->args[1].toNumber(); t = TimeClip(UTC(MakeDate(Day(t), MakeTime(HourFromTime(t), MinFromTime(t), sec, ms)))); self->value.setDouble(t); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setMinutes(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = LocalTime(self->value.asDouble()); double min = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); double sec = (ctx->callData->argc < 2) ? SecFromTime(t) : ctx->callData->args[1].toNumber(); double ms = (ctx->callData->argc < 3) ? msFromTime(t) : ctx->callData->args[2].toNumber(); t = TimeClip(UTC(MakeDate(Day(t), MakeTime(HourFromTime(t), min, sec, ms)))); self->value.setDouble(t); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setUTCMinutes(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = self->value.asDouble(); double min = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); double sec = (ctx->callData->argc < 2) ? SecFromTime(t) : ctx->callData->args[1].toNumber(); double ms = (ctx->callData->argc < 3) ? msFromTime(t) : ctx->callData->args[2].toNumber(); t = TimeClip(UTC(MakeDate(Day(t), MakeTime(HourFromTime(t), min, sec, ms)))); self->value.setDouble(t); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setHours(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = LocalTime(self->value.asDouble()); double hour = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); double min = (ctx->callData->argc < 2) ? MinFromTime(t) : ctx->callData->args[1].toNumber(); double sec = (ctx->callData->argc < 3) ? SecFromTime(t) : ctx->callData->args[2].toNumber(); double ms = (ctx->callData->argc < 4) ? msFromTime(t) : ctx->callData->args[3].toNumber(); t = TimeClip(UTC(MakeDate(Day(t), MakeTime(hour, min, sec, ms)))); self->value.setDouble(t); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setUTCHours(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = self->value.asDouble(); double hour = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); double min = (ctx->callData->argc < 2) ? MinFromTime(t) : ctx->callData->args[1].toNumber(); double sec = (ctx->callData->argc < 3) ? SecFromTime(t) : ctx->callData->args[2].toNumber(); double ms = (ctx->callData->argc < 4) ? msFromTime(t) : ctx->callData->args[3].toNumber(); t = TimeClip(UTC(MakeDate(Day(t), MakeTime(hour, min, sec, ms)))); self->value.setDouble(t); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setDate(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = LocalTime(self->value.asDouble()); double date = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); t = TimeClip(UTC(MakeDate(MakeDay(YearFromTime(t), MonthFromTime(t), date), TimeWithinDay(t)))); self->value.setDouble(t); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setUTCDate(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = self->value.asDouble(); double date = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); t = TimeClip(UTC(MakeDate(MakeDay(YearFromTime(t), MonthFromTime(t), date), TimeWithinDay(t)))); self->value.setDouble(t); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setMonth(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = LocalTime(self->value.asDouble()); double month = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); double date = (ctx->callData->argc < 2) ? DateFromTime(t) : ctx->callData->args[1].toNumber(); t = TimeClip(UTC(MakeDate(MakeDay(YearFromTime(t), month, date), TimeWithinDay(t)))); self->value.setDouble(t); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setUTCMonth(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = self->value.asDouble(); double month = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); double date = (ctx->callData->argc < 2) ? DateFromTime(t) : ctx->callData->args[1].toNumber(); t = TimeClip(UTC(MakeDate(MakeDay(YearFromTime(t), month, date), TimeWithinDay(t)))); self->value.setDouble(t); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setYear(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = self->value.asDouble(); if (std::isnan(t)) t = 0; else t = LocalTime(t); double year = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); double r; if (std::isnan(year)) { r = qSNaN(); } else { if ((Primitive::toInteger(year) >= 0) && (Primitive::toInteger(year) <= 99)) year += 1900; r = MakeDay(year, MonthFromTime(t), DateFromTime(t)); r = UTC(MakeDate(r, TimeWithinDay(t))); r = TimeClip(r); } self->value.setDouble(r); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setUTCFullYear(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = self->value.asDouble(); double year = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); double month = (ctx->callData->argc < 2) ? MonthFromTime(t) : ctx->callData->args[1].toNumber(); double date = (ctx->callData->argc < 3) ? DateFromTime(t) : ctx->callData->args[2].toNumber(); t = TimeClip(UTC(MakeDate(MakeDay(year, month, date), TimeWithinDay(t)))); self->value.setDouble(t); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_setFullYear(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = LocalTime(self->value.asDouble()); if (std::isnan(t)) t = 0; double year = ctx->callData->argc ? ctx->callData->args[0].toNumber() : qSNaN(); double month = (ctx->callData->argc < 2) ? MonthFromTime(t) : ctx->callData->args[1].toNumber(); double date = (ctx->callData->argc < 3) ? DateFromTime(t) : ctx->callData->args[2].toNumber(); t = TimeClip(UTC(MakeDate(MakeDay(year, month, date), TimeWithinDay(t)))); self->value.setDouble(t); return self->value.asReturnedValue(); } ReturnedValue DatePrototype::method_toUTCString(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = self->value.asDouble(); return ctx->engine->newString(ToUTCString(t))->asReturnedValue(); } static void addZeroPrefixedInt(QString &str, int num, int nDigits) { str.resize(str.size() + nDigits); QChar *c = str.data() + str.size() - 1; while (nDigits) { *c = QChar(num % 10 + '0'); num /= 10; --c; --nDigits; } } ReturnedValue DatePrototype::method_toISOString(SimpleCallContext *ctx) { DateObject *self = ctx->callData->thisObject.asDateObject(); if (!self) ctx->throwTypeError(); double t = self->value.asDouble(); if (!std::isfinite(t)) ctx->throwRangeError(ctx->callData->thisObject); QString result; int year = (int)YearFromTime(t); if (year < 0 || year > 9999) { if (qAbs(year) >= 1000000) return ctx->engine->newString(QStringLiteral("Invalid Date"))->asReturnedValue(); result += year < 0 ? '-' : '+'; year = qAbs(year); addZeroPrefixedInt(result, year, 6); } else { addZeroPrefixedInt(result, year, 4); } result += '-'; addZeroPrefixedInt(result, (int)MonthFromTime(t) + 1, 2); result += '-'; addZeroPrefixedInt(result, (int)DateFromTime(t), 2); result += 'T'; addZeroPrefixedInt(result, HourFromTime(t), 2); result += ':'; addZeroPrefixedInt(result, MinFromTime(t), 2); result += ':'; addZeroPrefixedInt(result, SecFromTime(t), 2); result += '.'; addZeroPrefixedInt(result, msFromTime(t), 3); result += 'Z'; return ctx->engine->newString(result)->asReturnedValue(); } ReturnedValue DatePrototype::method_toJSON(SimpleCallContext *ctx) { Scope scope(ctx); ScopedValue O(scope, __qmljs_to_object(ctx, ValueRef(&ctx->callData->thisObject))); ScopedValue tv(scope, __qmljs_to_primitive(O, NUMBER_HINT)); if (tv->isNumber() && !std::isfinite(tv->toNumber())) return Encode::null(); ScopedString s(scope, ctx->engine->newString(QStringLiteral("toISOString"))); ScopedValue v(scope, O->objectValue()->get(s)); FunctionObject *toIso = v->asFunctionObject(); if (!toIso) ctx->throwTypeError(); ScopedCallData callData(scope, 0); callData->thisObject = ctx->callData->thisObject; return toIso->call(callData); } void DatePrototype::timezoneUpdated() { LocalTZA = getLocalTZA(); }