9#if !defined(CPP_UTILITIES_CHRONO_BASED_EXACT_TIME) && (defined(PLATFORM_MAC) || !defined(PLATFORM_UNIX))
10#define CPP_UTILITIES_CHRONO_BASED_EXACT_TIME
12#ifdef CPP_UTILITIES_CHRONO_BASED_EXACT_TIME
20const int DateTime::m_daysPerYear = 365;
21const int DateTime::m_daysPer4Years = 1461;
22const int DateTime::m_daysPer100Years = 36524;
23const int DateTime::m_daysPer400Years = 146097;
24const int DateTime::m_daysTo1601 = 584388;
25const int DateTime::m_daysTo1899 = 693593;
26const int DateTime::m_daysTo10000 = 3652059;
27const int DateTime::m_daysToMonth365[13] = { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365 };
28const int DateTime::m_daysToMonth366[13] = { 0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335, 366 };
29const int DateTime::m_daysInMonth365[12] = { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
30const int DateTime::m_daysInMonth366[12] = { 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
32template <
typename num1,
typename num2,
typename num3>
constexpr bool inRangeInclMax(num1 val, num2
min, num3
max)
34 return (val) >= (
min) && (val) <= (
max);
37template <
typename num1,
typename num2,
typename num3>
constexpr bool inRangeExclMax(num1 val, num2
min, num3
max)
39 return (val) >= (
min) && (val) < (
max);
79 struct tm *
const timeinfo = localtime(&timeStamp);
80 return DateTime::fromDateAndTime(timeinfo->tm_year + 1900, timeinfo->tm_mon + 1, timeinfo->tm_mday, timeinfo->tm_hour, timeinfo->tm_min,
81 timeinfo->tm_sec < 60 ? timeinfo->tm_sec : 59, 0);
115 return std::make_pair(expr.value, expr.delta);
130 stringstream s(stringstream::in | stringstream::out);
134 constexpr auto dateDelimiter =
'-', timeDelimiter =
':';
136 const int *
const firstTimeComponent = components + 3;
137 const int *
const firstFractionalComponent = components + 6;
138 const int *
const lastComponent = components + 8;
139 const int *componentsEnd = noMilliseconds ? firstFractionalComponent : lastComponent + 1;
140 for (
const int *
i = componentsEnd - 1;
i > components; --
i) {
141 if (
i >= firstTimeComponent && *
i == 0) {
143 }
else if (
i < firstTimeComponent && *
i == 1) {
147 for (
const int *
i = components;
i != componentsEnd; ++
i) {
148 if (
i == firstTimeComponent) {
150 }
else if (
i == firstFractionalComponent) {
153 if (
i == components) {
155 }
else if (
i < firstFractionalComponent) {
156 if (
i < firstTimeComponent) {
158 }
else if (
i > firstTimeComponent) {
162 }
else if (
i < lastComponent) {
176 s << setw(4) <<
year() <<
'-' << setw(2) <<
month() <<
'-' << setw(2) <<
day();
182 s << setw(2) <<
hour() <<
':' << setw(2) <<
minute() <<
':' << setw(2) <<
second();
184 if (!noMilliseconds && ms > 0) {
185 s <<
'.' << setw(3) << ms;
197 stringstream s(stringstream::in | stringstream::out);
199 s << setw(4) <<
year() << dateDelimiter << setw(2) <<
month() << dateDelimiter << setw(2) <<
day() <<
'T' << setw(2) <<
hour() << timeDelimiter
200 << setw(2) <<
minute() << timeDelimiter << setw(2) <<
second();
204 if (milli || micro || nano) {
205 s <<
'.' << setw(3) << milli;
207 s << setw(3) << micro;
213 if (!timeZoneDelta.
isNull()) {
220 s << setw(2) << timeZoneDelta.
hours() << timeZoneDelimiter << setw(2) << timeZoneDelta.
minutes();
286#ifdef CPP_UTILITIES_CHRONO_BASED_EXACT_TIME
288 +
static_cast<std::uint64_t
>(
289 std::chrono::time_point_cast<std::chrono::nanoseconds>(std::chrono::high_resolution_clock::now()).time_since_epoch().count())
293 clock_gettime(CLOCK_REALTIME, &t);
295 +
static_cast<std::uint64_t
>(t.tv_nsec) / 100);
308 throw ConversionException(
"month is out of range");
310 const auto *
const daysToMonth =
reinterpret_cast<const int *
>(
isLeapYear(
year) ? m_daysToMonth366 : m_daysToMonth365);
311 const int passedMonth =
month - 1;
313 throw ConversionException(
"day is out of range");
315 const auto passedYears =
static_cast<unsigned int>(
year - 1);
316 const auto passedDays =
static_cast<unsigned int>(
day - 1);
317 return (passedYears * m_daysPerYear + passedYears / 4 - passedYears / 100 + passedYears / 400
318 +
static_cast<unsigned int>(daysToMonth[passedMonth]) + passedDays)
325DateTime::TickType DateTime::timeToTicks(
int hour,
int minute,
int second,
double millisecond)
328 throw ConversionException(
"hour is out of range");
331 throw ConversionException(
"minute is out of range");
334 throw ConversionException(
"second is out of range");
337 throw ConversionException(
"millisecond is out of range");
347int DateTime::getDatePart(
DatePart part)
const
350 const auto full400YearBlocks = fullDays / m_daysPer400Years;
351 const auto daysMinusFull400YearBlocks = fullDays - full400YearBlocks * m_daysPer400Years;
352 auto full100YearBlocks = daysMinusFull400YearBlocks / m_daysPer100Years;
353 if (full100YearBlocks == 4) {
354 full100YearBlocks = 3;
356 const auto daysMinusFull100YearBlocks = daysMinusFull400YearBlocks - full100YearBlocks * m_daysPer100Years;
357 const auto full4YearBlocks = daysMinusFull100YearBlocks / m_daysPer4Years;
358 const auto daysMinusFull4YearBlocks = daysMinusFull100YearBlocks - full4YearBlocks * m_daysPer4Years;
359 auto full1YearBlocks = daysMinusFull4YearBlocks / m_daysPerYear;
360 if (full1YearBlocks == 4) {
364 return full400YearBlocks * 400 + full100YearBlocks * 100 + full4YearBlocks * 4 + full1YearBlocks + 1;
366 const auto restDays = daysMinusFull4YearBlocks - full1YearBlocks * m_daysPerYear;
370 const auto *
const daysToMonth = (full1YearBlocks == 3 && (full4YearBlocks != 24 || full100YearBlocks == 3)) ? m_daysToMonth366 : m_daysToMonth365;
372 while (restDays >= daysToMonth[
month]) {
378 return restDays - daysToMonth[
month - 1] + 1;
384static DateTimeParts dateTimePartsFromParsingDistance(
const int *valueIndex,
const int *values)
386 return static_cast<DateTimeParts>((1 << (valueIndex - values + 1)) - 1);
402 int values[9] = { 0 };
403 int *
const yearIndex = values + 0;
404 int *
const monthIndex = values + 1;
405 int *
const dayIndex = values + 2;
406 int *
const hourIndex = values + 3;
407 int *
const secondsIndex = values + 5;
408 int *
const miliSecondsIndex = values + 6;
409 int *
const deltaHourIndex = values + 7;
410 int *
const valuesEnd = values + 9;
411 int *valueIndex = values;
412 unsigned int remainingDigits = 4;
413 bool deltaNegative =
false;
414 double millisecondsFact = 100.0, milliseconds = 0.0;
415 for (
const char *strIndex = str;; ++strIndex) {
416 const char c = *strIndex;
417 if (c <=
'9' && c >=
'0') {
418 if (valueIndex == miliSecondsIndex) {
419 milliseconds += (c -
'0') * millisecondsFact;
420 millisecondsFact /= 10;
422 if (!remainingDigits) {
423 if (++valueIndex == miliSecondsIndex || valueIndex >= valuesEnd) {
429 *valueIndex += c -
'0';
430 remainingDigits -= 1;
432 }
else if (c ==
'T') {
433 if (++valueIndex != hourIndex) {
437 }
else if (c ==
'-') {
438 if (valueIndex < dayIndex) {
440 }
else if (++valueIndex >= secondsIndex) {
441 valueIndex = deltaHourIndex;
442 deltaNegative =
true;
447 }
else if (c ==
'.') {
448 if (valueIndex != secondsIndex) {
453 }
else if (c ==
':') {
454 if (valueIndex < hourIndex) {
456 }
else if (valueIndex == secondsIndex) {
462 }
else if ((c ==
'+') && (++valueIndex >= secondsIndex)) {
463 valueIndex = deltaHourIndex;
464 deltaNegative =
false;
466 }
else if ((c ==
'Z') && (++valueIndex >= secondsIndex)) {
467 valueIndex = deltaHourIndex + 2;
469 }
else if (c ==
'\0') {
477 res.delta =
TimeSpan(-res.delta.totalTicks());
479 if (valueIndex < monthIndex && !*monthIndex) {
482 if (valueIndex < dayIndex && !*dayIndex) {
485 res.value =
DateTime::fromDateAndTime(*yearIndex, *monthIndex, *dayIndex, *hourIndex, values[4], *secondsIndex, milliseconds);
486 res.parts = dateTimePartsFromParsingDistance(valueIndex, values);
502 int values[7] = { 0 };
503 int *
const monthIndex = values + 1;
504 int *
const dayIndex = values + 2;
505 int *
const secondsIndex = values + 5;
506 int *valueIndex = values;
507 int *
const valuesEnd = values + 7;
508 double millisecondsFact = 100.0, milliseconds = 0.0;
509 for (
const char *strIndex = str;; ++strIndex) {
510 const char c = *strIndex;
511 if (c <=
'9' && c >=
'0') {
512 if (valueIndex > secondsIndex) {
513 milliseconds += (c -
'0') * millisecondsFact;
514 millisecondsFact /= 10;
516 Detail::raiseAndAdd(*valueIndex, 10, c);
518 }
else if ((c ==
'-' || c ==
':' || c ==
'/') || (c ==
'.' && (valueIndex == secondsIndex))
519 || ((c ==
' ' || c ==
'T') && (valueIndex == dayIndex))) {
520 if (++valueIndex == valuesEnd) {
523 }
else if (c ==
'\0') {
529 if (valueIndex < monthIndex && !*monthIndex) {
532 if (valueIndex < dayIndex && !*dayIndex) {
535 res.value =
DateTime::fromDateAndTime(values[0], values[1], *dayIndex, values[3], values[4], *secondsIndex, milliseconds);
536 res.parts = dateTimePartsFromParsingDistance(valueIndex, values);
546 auto s = std::stringstream(std::stringstream::in | std::stringstream::out);
549 s << setw(4) <<
value.year();
555 s << setw(2) <<
value.month();
561 s << setw(2) <<
value.day();
567 s << setw(2) <<
value.hour();
573 s << setw(2) <<
value.minute();
579 s << setw(2) <<
value.second();
582 const auto milli =
value.millisecond();
583 const auto micro =
value.microsecond();
584 const auto nano =
value.nanosecond();
585 s <<
'.' << setw(3) << milli;
587 s << setw(3) << micro;
595 if (d.isNegative()) {
602 s << setw(2) << d.hours();
606 s << timeZoneDelimiter;
608 s << setw(2) << d.minutes();
The ConversionException class is thrown by the various conversion functions of this library when a co...
std::string toString(DateTimeOutputFormat format=DateTimeOutputFormat::DateAndTime, bool noMilliseconds=false) const
Returns the string representation of the current instance using the specified format.
int day() const
Returns the day component of the date represented by this instance.
constexpr DayOfWeek dayOfWeek() const
Returns the day of the week represented by this instance.
std::string toIsoStringWithCustomDelimiters(TimeSpan timeZoneDelta=TimeSpan(), char dateDelimiter='-', char timeDelimiter=':', char timeZoneDelimiter=':') const
Returns the string representation of the current instance in the ISO format with custom delimiters,...
bool isLeapYear() const
Returns an indication whether the year represented by this instance is a leap year.
int month() const
Returns the month component of the date represented by this instance.
constexpr DateTime()
Constructs a DateTime.
std::string toIsoString(TimeSpan timeZoneDelta=TimeSpan()) const
Returns the string representation of the current instance in the ISO format, eg.
static constexpr DateTime unixEpochStart()
Returns the DateTime object for the "1970-01-01T00:00:00Z".
constexpr int microsecond() const
Returns the microsecond component of the date represented by this instance.
static std::pair< DateTime, TimeSpan > fromIsoString(const char *str)
Parses the specified ISO date time denotation provided as C-style string.
static DateTime exactGmtNow()
Returns a DateTime object that is set to the current date and time on this computer,...
constexpr int hour() const
Returns the hour component of the date represented by this instance.
static DateTime fromString(const std::string &str)
Parses the given std::string as DateTime.
constexpr int second() const
Returns the second component of the date represented by this instance.
static DateTime fromDateAndTime(int year=1, int month=1, int day=1, int hour=0, int minute=0, int second=0, double millisecond=0.0)
Constructs a DateTime to the specified year, month, day, hour, minute, second and millisecond.
constexpr TickType totalTicks() const
Returns the number of ticks which represent the value of the current instance.
static DateTime fromTimeStamp(std::time_t timeStamp)
Constructs a new DateTime object with the local time from the specified UNIX timeStamp.
constexpr int millisecond() const
Returns the millisecond component of the date represented by this instance.
static const char * printDayOfWeek(DayOfWeek dayOfWeek, bool abbreviation=false)
Returns the string representation as C-style string for the given day of week.
constexpr int nanosecond() const
Returns the nanosecond component of the date represented by this instance.
constexpr int minute() const
Returns the minute component of the date represented by this instance.
int year() const
Returns the year component of the date represented by this instance.
Represents a time interval.
constexpr bool isNull() const
Returns true if the time interval represented by the current TimeSpan class is null.
static constexpr TickType nanosecondsPerTick
static constexpr TickType ticksPerMillisecond
constexpr int minutes() const
Returns the minutes component of the time interval represented by the current TimeSpan class.
constexpr TickType totalTicks() const
Returns the number of ticks that represent the value of the current TimeSpan class.
static constexpr TickType ticksPerMinute
static constexpr TimeSpan fromMinutes(double minutes)
Constructs a new instance of the TimeSpan class with the specified number of minutes.
static constexpr TickType ticksPerSecond
static constexpr TickType ticksPerDay
static constexpr TickType ticksPerHour
constexpr bool isNegative() const
Returns true if the time interval represented by the current TimeSpan class is negative.
constexpr int hours() const
Returns the hours component of the time interval represented by the current TimeSpan class.
Contains all utilities provided by the c++utilities library.
DatePart
Specifies the date part.
constexpr bool inRangeExclMax(num1 val, num2 min, num3 max)
constexpr bool inRangeInclMax(num1 val, num2 min, num3 max)
StringType argsToString(Args &&...args)
constexpr T max(T first, T second)
Returns the greatest of the given items.
DateTimeParts
The DateTimeParts enum specifies which parts of a timestamp are present.
DateTimeOutputFormat
Specifies the output format.
@ DateTimeAndShortWeekday
@ IsoOmittingDefaultComponents
constexpr T min(T first, T second)
Returns the smallest of the given items.
DayOfWeek
Specifies the day of the week.
The DateTimeExpression struct holds information about a time expression (e.g.
static DateTimeExpression fromString(const char *str)
Parses the given C-style string.
std::string toIsoString(char dateDelimiter='-', char timeDelimiter=':', char timeZoneDelimiter=':') const
Returns the string representation of the current instance in the ISO format.
static DateTimeExpression fromIsoString(const char *str)
Parses the specified ISO date time denotation provided as C-style string.