2014-04-08 06:51:53 +00:00
|
|
|
#pragma once
|
2011-03-09 19:18:01 +00:00
|
|
|
|
2022-06-27 21:00:37 +00:00
|
|
|
#include <cstring>
|
2011-03-09 19:18:01 +00:00
|
|
|
#include <string>
|
2016-02-03 01:17:58 +00:00
|
|
|
#include <exception>
|
2021-12-21 13:41:53 +00:00
|
|
|
#include <Common/DateLUT.h>
|
2011-03-09 19:18:01 +00:00
|
|
|
|
|
|
|
|
2017-08-04 23:16:43 +00:00
|
|
|
/** Stores a calendar date in broken-down form (year, month, day-in-month).
|
|
|
|
* Could be initialized from date in text form, like '2011-01-01' or from time_t with rounding to date.
|
|
|
|
* Also could be initialized from date in text form like '20110101... (only first 8 symbols are used).
|
|
|
|
* Could be implicitly casted to time_t.
|
|
|
|
* NOTE: Transforming between time_t and LocalDate is done in local time zone!
|
2011-03-18 20:26:54 +00:00
|
|
|
*
|
2017-08-04 23:16:43 +00:00
|
|
|
* When local time was shifted backwards (due to daylight saving time or whatever reason)
|
|
|
|
* - then to resolve the ambiguity of transforming to time_t, lowest of two possible values is selected.
|
|
|
|
*
|
|
|
|
* packed - for memcmp to work naturally (but because m_year is 2 bytes, on little endian, comparison is correct only before year 2047)
|
2011-03-18 20:26:54 +00:00
|
|
|
*/
|
2018-03-23 16:05:14 +00:00
|
|
|
class LocalDate
|
2011-03-09 19:18:01 +00:00
|
|
|
{
|
|
|
|
private:
|
2022-03-11 21:47:28 +00:00
|
|
|
unsigned short m_year; /// NOLINT
|
2011-10-14 18:03:25 +00:00
|
|
|
unsigned char m_month;
|
|
|
|
unsigned char m_day;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2011-03-10 20:31:02 +00:00
|
|
|
void init(time_t time)
|
2011-03-09 19:18:01 +00:00
|
|
|
{
|
2015-07-07 23:11:30 +00:00
|
|
|
const auto & date_lut = DateLUT::instance();
|
2015-06-26 15:11:31 +00:00
|
|
|
const auto & values = date_lut.getValues(time);
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2011-10-14 18:03:25 +00:00
|
|
|
m_year = values.year;
|
|
|
|
m_month = values.month;
|
|
|
|
m_day = values.day_of_month;
|
2011-03-10 20:31:02 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2011-03-10 20:31:02 +00:00
|
|
|
void init(const char * s, size_t length)
|
|
|
|
{
|
2012-05-02 19:16:50 +00:00
|
|
|
if (length < 8)
|
2016-02-03 01:17:58 +00:00
|
|
|
throw std::runtime_error("Cannot parse LocalDate: " + std::string(s, length));
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2011-10-14 18:03:25 +00:00
|
|
|
m_year = (s[0] - '0') * 1000 + (s[1] - '0') * 100 + (s[2] - '0') * 10 + (s[3] - '0');
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2012-05-02 19:16:50 +00:00
|
|
|
if (s[4] == '-')
|
2011-08-03 14:24:28 +00:00
|
|
|
{
|
2012-05-02 19:16:50 +00:00
|
|
|
if (length < 10)
|
2016-02-03 01:17:58 +00:00
|
|
|
throw std::runtime_error("Cannot parse LocalDate: " + std::string(s, length));
|
2011-10-14 18:03:25 +00:00
|
|
|
m_month = (s[5] - '0') * 10 + (s[6] - '0');
|
|
|
|
m_day = (s[8] - '0') * 10 + (s[9] - '0');
|
2011-08-03 14:24:28 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-10-14 18:03:25 +00:00
|
|
|
m_month = (s[4] -'0') * 10 + (s[5] -'0');
|
|
|
|
m_day = (s[6] - '0')* 10 + (s[7] -'0');
|
2011-08-03 14:24:28 +00:00
|
|
|
}
|
2011-03-10 20:31:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
2016-02-03 01:17:58 +00:00
|
|
|
explicit LocalDate(time_t time)
|
2011-03-10 20:31:02 +00:00
|
|
|
{
|
|
|
|
init(time);
|
2011-03-09 19:18:01 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-03-11 21:47:28 +00:00
|
|
|
LocalDate(DayNum day_num) /// NOLINT
|
2012-05-02 19:16:50 +00:00
|
|
|
{
|
2015-06-26 15:11:31 +00:00
|
|
|
const auto & values = DateLUT::instance().getValues(day_num);
|
2017-11-16 02:07:19 +00:00
|
|
|
m_year = values.year;
|
2012-05-02 19:16:50 +00:00
|
|
|
m_month = values.month;
|
2017-11-16 02:07:19 +00:00
|
|
|
m_day = values.day_of_month;
|
2012-05-02 19:16:50 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-07-15 11:41:52 +00:00
|
|
|
explicit LocalDate(ExtendedDayNum day_num)
|
|
|
|
{
|
|
|
|
const auto & values = DateLUT::instance().getValues(day_num);
|
|
|
|
m_year = values.year;
|
|
|
|
m_month = values.month;
|
|
|
|
m_day = values.day_of_month;
|
|
|
|
}
|
|
|
|
|
2022-03-11 21:47:28 +00:00
|
|
|
LocalDate(unsigned short year_, unsigned char month_, unsigned char day_) /// NOLINT
|
2011-10-14 18:03:25 +00:00
|
|
|
: m_year(year_), m_month(month_), m_day(day_)
|
2011-03-09 19:18:01 +00:00
|
|
|
{
|
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2016-02-03 01:17:58 +00:00
|
|
|
explicit LocalDate(const std::string & s)
|
2011-03-09 19:18:01 +00:00
|
|
|
{
|
2011-03-10 20:31:02 +00:00
|
|
|
init(s.data(), s.size());
|
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2016-02-03 01:17:58 +00:00
|
|
|
LocalDate(const char * data, size_t length)
|
2011-03-10 20:31:02 +00:00
|
|
|
{
|
|
|
|
init(data, length);
|
2011-03-09 19:18:01 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2016-02-03 01:17:58 +00:00
|
|
|
LocalDate() : m_year(0), m_month(0), m_day(0)
|
2011-03-09 19:18:01 +00:00
|
|
|
{
|
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2017-10-17 11:58:08 +00:00
|
|
|
LocalDate(const LocalDate &) noexcept = default;
|
|
|
|
LocalDate & operator= (const LocalDate &) noexcept = default;
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2018-05-25 13:29:15 +00:00
|
|
|
DayNum getDayNum() const
|
2011-05-25 18:15:00 +00:00
|
|
|
{
|
2020-04-17 13:26:44 +00:00
|
|
|
const auto & lut = DateLUT::instance();
|
|
|
|
return DayNum(lut.makeDayNum(m_year, m_month, m_day).toUnderType());
|
2011-05-25 18:15:00 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2021-07-15 11:41:52 +00:00
|
|
|
ExtendedDayNum getExtenedDayNum() const
|
|
|
|
{
|
|
|
|
const auto & lut = DateLUT::instance();
|
|
|
|
return ExtendedDayNum (lut.makeDayNum(m_year, m_month, m_day).toUnderType());
|
|
|
|
}
|
|
|
|
|
2022-03-11 21:47:28 +00:00
|
|
|
operator DayNum() const /// NOLINT
|
2011-06-03 21:02:20 +00:00
|
|
|
{
|
|
|
|
return getDayNum();
|
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-03-11 21:47:28 +00:00
|
|
|
unsigned short year() const { return m_year; } /// NOLINT
|
2011-10-14 18:03:25 +00:00
|
|
|
unsigned char month() const { return m_month; }
|
|
|
|
unsigned char day() const { return m_day; }
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2022-03-11 21:47:28 +00:00
|
|
|
void year(unsigned short x) { m_year = x; } /// NOLINT
|
2011-10-14 18:03:25 +00:00
|
|
|
void month(unsigned char x) { m_month = x; }
|
|
|
|
void day(unsigned char x) { m_day = x; }
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2016-02-03 01:17:58 +00:00
|
|
|
bool operator< (const LocalDate & other) const
|
2011-03-09 19:18:01 +00:00
|
|
|
{
|
2011-10-14 18:03:25 +00:00
|
|
|
return 0 > memcmp(this, &other, sizeof(*this));
|
2011-03-09 19:18:01 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2016-02-03 01:17:58 +00:00
|
|
|
bool operator> (const LocalDate & other) const
|
2011-03-21 19:46:50 +00:00
|
|
|
{
|
2011-10-14 18:03:25 +00:00
|
|
|
return 0 < memcmp(this, &other, sizeof(*this));
|
2011-03-21 19:46:50 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2016-02-03 01:17:58 +00:00
|
|
|
bool operator<= (const LocalDate & other) const
|
2011-03-21 19:46:50 +00:00
|
|
|
{
|
2011-10-14 18:03:25 +00:00
|
|
|
return 0 >= memcmp(this, &other, sizeof(*this));
|
2011-03-21 19:46:50 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2016-02-03 01:17:58 +00:00
|
|
|
bool operator>= (const LocalDate & other) const
|
2011-03-21 19:46:50 +00:00
|
|
|
{
|
2011-10-14 18:03:25 +00:00
|
|
|
return 0 <= memcmp(this, &other, sizeof(*this));
|
2011-03-21 19:46:50 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2016-02-03 01:17:58 +00:00
|
|
|
bool operator== (const LocalDate & other) const
|
2011-03-09 19:18:01 +00:00
|
|
|
{
|
2011-10-14 18:03:25 +00:00
|
|
|
return 0 == memcmp(this, &other, sizeof(*this));
|
2011-03-09 19:18:01 +00:00
|
|
|
}
|
2017-04-01 07:20:54 +00:00
|
|
|
|
2016-02-03 01:17:58 +00:00
|
|
|
bool operator!= (const LocalDate & other) const
|
2011-03-09 19:18:01 +00:00
|
|
|
{
|
|
|
|
return !(*this == other);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-07-18 17:02:47 +00:00
|
|
|
static_assert(sizeof(LocalDate) == 4);
|