// // DateTime.cpp // // $Id: //poco/1.1.0/Foundation/src/DateTime.cpp#2 $ // // Library: Foundation // Package: DateTime // Module: DateTime // // Copyright (c) 2004-2006, Applied Informatics Software Engineering GmbH. // and Contributors. // // Permission is hereby granted, free of charge, to any person or organization // obtaining a copy of the software and accompanying documentation covered by // this license (the "Software") to use, reproduce, display, distribute, // execute, and transmit the Software, and to prepare derivative works of the // Software, and to permit third-parties to whom the Software is furnished to // do so, all subject to the following: // // The copyright notices in the Software and this entire statement, including // the above license grant, this restriction and the following disclaimer, // must be included in all copies of the Software, in whole or in part, and // all derivative works of the Software, unless such copies or derivative // works are solely in the form of machine-executable object code generated by // a source language processor. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT // SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE // FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE, // ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. // #include "Foundation/DateTime.h" #include #include Foundation_BEGIN inline double DateTime::toJulianDay(Timestamp::UtcTimeVal utcTime) { double utcDays = double(utcTime)/864000000000.0; return utcDays + 2299160.5; // first day of Gregorian reform (Oct 15 1582) } inline Timestamp::UtcTimeVal DateTime::toUtcTime(double julianDay) { return Timestamp::UtcTimeVal((julianDay - 2299160.5)*864000000000.0); } DateTime::DateTime() { Timestamp now; _utcTime = now.utcTime(); _julianDay = toJulianDay(_utcTime); computeGregorian(); } DateTime::DateTime(const Timestamp& timestamp): _utcTime(timestamp.utcTime()), _julianDay(toJulianDay(_utcTime)) { computeGregorian(); } DateTime::DateTime(int year, int month, int day, int hour, int minute, int second, int millisecond, int microsecond): _year(year), _month(month), _day(day), _hour(hour), _minute(minute), _second(second), _millisecond(millisecond), _microsecond(microsecond) { poco_assert (year >= 0 && year <= 9999); poco_assert (month >= 1 && month <= 12); poco_assert (day >= 1 && day <= daysOfMonth(year, month)); poco_assert (hour >= 0 && hour <= 23); poco_assert (minute >= 0 && minute <= 59); poco_assert (second >= 0 && second <= 59); poco_assert (millisecond >= 0 && millisecond <= 999); poco_assert (microsecond >= 0 && microsecond <= 999); _julianDay = toJulianDay(year, month, day, hour, minute, second, millisecond, microsecond); _utcTime = toUtcTime(_julianDay); } DateTime::DateTime(double julianDay): _utcTime(toUtcTime(julianDay)), _julianDay(julianDay) { computeGregorian(); } DateTime::DateTime(Timestamp::UtcTimeVal utcTime, Timestamp::TimeDiff diff): _utcTime(utcTime + diff*10), _julianDay(toJulianDay(_utcTime)) { computeGregorian(); } DateTime::DateTime(const DateTime& dateTime): _utcTime(dateTime._utcTime), _julianDay(dateTime._julianDay), _year(dateTime._year), _month(dateTime._month), _day(dateTime._day), _hour(dateTime._hour), _minute(dateTime._minute), _second(dateTime._second), _millisecond(dateTime._millisecond), _microsecond(dateTime._microsecond) { } DateTime::~DateTime() { } DateTime& DateTime::operator = (const DateTime& dateTime) { if (&dateTime != this) { _utcTime = dateTime._utcTime; _julianDay = dateTime._julianDay; _year = dateTime._year; _month = dateTime._month; _day = dateTime._day; _hour = dateTime._hour; _minute = dateTime._minute; _second = dateTime._second; _millisecond = dateTime._millisecond; _microsecond = dateTime._microsecond; } return *this; } DateTime& DateTime::operator = (const Timestamp& timestamp) { _utcTime = timestamp.utcTime(); _julianDay = toJulianDay(_utcTime); computeGregorian(); return *this; } DateTime& DateTime::operator = (double julianDay) { _julianDay = julianDay; _utcTime = toUtcTime(julianDay); computeGregorian(); return *this; } DateTime& DateTime::assign(int year, int month, int day, int hour, int minute, int second, int millisecond, int microsecond) { poco_assert (year >= 0 && year <= 9999); poco_assert (month >= 1 && month <= 12); poco_assert (day >= 1 && day <= daysOfMonth(year, month)); poco_assert (hour >= 0 && hour <= 23); poco_assert (minute >= 0 && minute <= 59); poco_assert (second >= 0 && second <= 59); poco_assert (millisecond >= 0 && millisecond <= 999); poco_assert (microsecond >= 0 && microsecond <= 999); _julianDay = toJulianDay(year, month, day, hour, minute, second, millisecond, microsecond); _utcTime = toUtcTime(_julianDay); _year = year; _month = month; _day = day; _hour = hour; _minute = minute; _second = second; _millisecond = millisecond; _microsecond = microsecond; return *this; } void DateTime::swap(DateTime& dateTime) { std::swap(_utcTime, dateTime._utcTime); std::swap(_julianDay, dateTime._julianDay); std::swap(_year, dateTime._year); std::swap(_month, dateTime._month); std::swap(_day, dateTime._day); std::swap(_hour, dateTime._hour); std::swap(_minute, dateTime._minute); std::swap(_second, dateTime._second); std::swap(_millisecond, dateTime._millisecond); std::swap(_microsecond, dateTime._microsecond); } int DateTime::dayOfWeek() const { return int((floor(_julianDay + 1.5))) % 7; } int DateTime::dayOfYear() const { int doy = 0; for (int month = 1; month < _month; ++month) doy += daysOfMonth(_year, month); doy += _day; return doy; } int DateTime::daysOfMonth(int year, int month) { poco_assert (month >= 1 && month <= 12); static int daysOfMonthTable[] = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; if (month == 2 && isLeapYear(year)) return 29; else return daysOfMonthTable[month]; } int DateTime::week(int firstDayOfWeek) const { poco_assert (firstDayOfWeek >= 0 && firstDayOfWeek <= 6); /// find the first firstDayOfWeek. int baseDay = 1; while (DateTime(_year, 1, baseDay).dayOfWeek() != firstDayOfWeek) ++baseDay; int doy = dayOfYear(); int offs = baseDay <= 4 ? 0 : 1; if (doy < baseDay) return offs; else return (doy - baseDay)/7 + 1 + offs; } DateTime DateTime::operator + (const Timespan& span) const { return DateTime(_utcTime, span.totalMicroseconds()); } DateTime DateTime::operator - (const Timespan& span) const { return DateTime(_utcTime, -span.totalMicroseconds()); } Timespan DateTime::operator - (const DateTime& dateTime) const { return Timespan((_utcTime - dateTime._utcTime)/10); } DateTime& DateTime::operator += (const Timespan& span) { _utcTime += span.totalMicroseconds()*10; _julianDay = toJulianDay(_utcTime); computeGregorian(); return *this; } DateTime& DateTime::operator -= (const Timespan& span) { _utcTime -= span.totalMicroseconds()*10; _julianDay = toJulianDay(_utcTime); computeGregorian(); return *this; } double DateTime::toJulianDay(int year, int month, int day, int hour, int minute, int second, int millisecond, int microsecond) { // lookup table for (153*month - 457)/5 - note that 3 <= month <= 14. static int lookup[] = {-91, -60, -30, 0, 31, 61, 92, 122, 153, 184, 214, 245, 275, 306, 337}; // day to double double dday = double(day) + ((double((hour*60 + minute)*60 + second)*1000 + millisecond)*1000 + microsecond)/86400000000.0; if (month < 3) { month += 12; --year; } double dyear = double(year); return dday + lookup[month] + 365*year + floor(dyear/4) - floor(dyear/100) + floor(dyear/400) + 1721118.5; } void DateTime::checkLimit(short& lower, short& higher, short limit) { if (lower > limit) { higher += short(lower / limit); lower = short(lower % limit); } } void DateTime::normalize() { checkLimit(_microsecond, _millisecond, 999); checkLimit(_millisecond, _second, 999); checkLimit(_second, _minute, 59); checkLimit(_minute, _hour, 59); checkLimit(_hour, _day, 23); if (_day > daysOfMonth(_year, _month)) { _day -= daysOfMonth(_year, _month); if (++_month > 12) { ++_year; _month -= 12; } } } void DateTime::computeGregorian() { double z = floor(_julianDay - 1721118.5); double r = _julianDay - 1721118.5 - z; double g = z - 0.25; double a = floor(g / 36524.25); double b = a - floor(a/4); _year = short(floor((b + g)/365.25)); double c = b + z - floor(365.25*_year); _month = short(floor((5*c + 456)/153)); double dday = c - floor((153.0*_month - 457)/5) + r; _day = short(dday); if (_month > 12) { ++_year; _month -= 12; } r *= 24; _hour = short(floor(r)); r -= floor(r); r *= 60; _minute = short(floor(r)); r -= floor(r); r *= 60; _second = short(floor(r)); r -= floor(r); r *= 1000; _millisecond = short(floor(r)); r -= floor(r); r *= 1000; _microsecond = short(r + 0.5); normalize(); poco_assert_dbg (_month >= 1 && _month <= 12); poco_assert_dbg (_day >= 1 && _day <= daysOfMonth(_year, _month)); poco_assert_dbg (_hour >= 0 && _hour <= 23); poco_assert_dbg (_minute >= 0 && _minute <= 59); poco_assert_dbg (_second >= 0 && _second <= 59); poco_assert_dbg (_millisecond >= 0 && _millisecond <= 999); poco_assert_dbg (_microsecond >= 0 && _microsecond <= 999); } Foundation_END