mirror of
https://github.com/pocoproject/poco.git
synced 2025-01-03 15:06:10 +01:00
388 lines
10 KiB
C++
388 lines
10 KiB
C++
|
//
|
||
|
// 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 <algorithm>
|
||
|
#include <math.h>
|
||
|
|
||
|
|
||
|
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
|