civil time

group turbo_times_civil_time

Typedefs

using CivilSecond = time_internal::cctz::detail::civil_time<time_internal::second_tag>

Each of these civil-time types is a simple value type with the same interface for construction and the same six accessors for each of the civil time fields (year, month, day, hour, minute, and second, aka YMDHMS).

These classes differ only in their alignment, which is indicated by the type name and specifies the field on which arithmetic operates. CONSTRUCTION Each of the civil-time types can be constructed in two ways: by directly passing to the constructor up to six integers representing the YMDHMS fields, or by copying the YMDHMS fields from a differently aligned civil-time type. Omitted fields are assigned their minimum valid value. hours, minutes, and seconds will be set to 0, month and day will be set to 1. Since there is no minimum year, the default is 1970. Examples:

turbo::CivilDay default_value;               // 1970-01-01 00:00:00
turbo::CivilDay a(2015, 2, 3);               // 2015-02-03 00:00:00
turbo::CivilDay b(2015, 2, 3, 4, 5, 6);      // 2015-02-03 00:00:00
turbo::CivilDay c(2015);                     // 2015-01-01 00:00:00
turbo::CivilSecond ss(2015, 2, 3, 4, 5, 6);  // 2015-02-03 04:05:06
turbo::CivilMinute mm(ss);                   // 2015-02-03 04:05:00
turbo::CivilHour hh(mm);                     // 2015-02-03 04:00:00
turbo::CivilDay d(hh);                       // 2015-02-03 00:00:00
turbo::CivilMonth m(d);                      // 2015-02-01 00:00:00
turbo::CivilYear y(m);                       // 2015-01-01 00:00:00
m = turbo::CivilMonth(y);                    // 2015-01-01 00:00:00
d = turbo::CivilDay(m);                      // 2015-01-01 00:00:00
hh = turbo::CivilHour(d);                    // 2015-01-01 00:00:00
mm = turbo::CivilMinute(hh);                 // 2015-01-01 00:00:00
ss = turbo::CivilSecond(mm);                 // 2015-01-01 00:00:00
Each civil-time class is aligned to the civil-time field indicated in the class’s name after normalization. Alignment is performed by setting all the inferior fields to their minimum valid value (as described above). The following are examples of how each of the six types would align the fields representing November 22, 2015 at 12:34:56 in the afternoon. (Note: the string format used here is not important; it’s just a shorthand way of showing the six YMDHMS fields.)
turbo::CivilSecond   : 2015-11-22 12:34:56
turbo::CivilMinute   : 2015-11-22 12:34:00
turbo::CivilHour     : 2015-11-22 12:00:00
turbo::CivilDay      : 2015-11-22 00:00:00
turbo::CivilMonth    : 2015-11-01 00:00:00
turbo::CivilYear     : 2015-01-01 00:00:00

Each civil-time type performs arithmetic on the field to which it is aligned. This means that adding 1 to an turbo::CivilDay increments the day field (normalizing as necessary), and subtracting 7 from an turbo::CivilMonth operates on the month field (normalizing as necessary). All arithmetic produces a valid civil time. Difference requires two similarly aligned civil-time objects and returns the scalar answer in units of the objects’ alignment. For example, the difference between two turbo::CivilHour objects will give an answer in units of civil hours. ALIGNMENT CONVERSION The alignment of a civil-time object cannot change, but the object may be used to construct a new object with a different alignment. This is referred to as “realigning”. When realigning to a type with the same or more precision (e.g., turbo::CivilDay -> turbo::CivilSecond), the conversion may be performed implicitly since no information is lost. However, if information could be discarded (e.g., CivilSecond -> CivilDay), the conversion must be explicit at the call site. Examples:

void UseDay(turbo::CivilDay day);
turbo::CivilSecond cs;
UseDay(cs);                  // Won't compile because data may be discarded
UseDay(turbo::CivilDay(cs));  // OK: explicit conversion
turbo::CivilDay cd;
UseDay(cd);                  // OK: no conversion needed
turbo::CivilMonth cm;
UseDay(cm);                  // OK: implicit conversion to turbo::CivilDay
NORMALIZATION Normalization takes invalid values and adjusts them to produce valid values. Within the civil-time library, integer arguments passed to the Civil* constructors may be out-of-range, in which case they are normalized by carrying overflow into a field of courser granularity to produce valid civil-time objects. This normalization enables natural arithmetic on constructor arguments without worrying about the field’s range. Examples:
// Out-of-range; normalized to 2016-11-01
turbo::CivilDay d(2016, 10, 32);
// Out-of-range, negative: normalized to 2016-10-30T23
turbo::CivilHour h1(2016, 10, 31, -1);
// Normalization is cumulative: normalized to 2016-10-30T23
turbo::CivilHour h2(2016, 10, 32, -25);
Note: If normalization is undesired, you can signal an error by comparing the constructor arguments to the normalized values returned by the YMDHMS properties. COMPARISON Comparison between civil-time objects considers all six YMDHMS fields, regardless of the type’s alignment. Comparison between differently aligned civil-time types is allowed. Examples:
turbo::CivilDay feb_3(2015, 2, 3);  // 2015-02-03 00:00:00
turbo::CivilDay mar_4(2015, 3, 4);  // 2015-03-04 00:00:00
// feb_3 < mar_4
// turbo::CivilYear(feb_3) == turbo::CivilYear(mar_4)
turbo::CivilSecond feb_3_noon(2015, 2, 3, 12, 0, 0);  // 2015-02-03 12:00:00
// feb_3 < feb_3_noon
// feb_3 == turbo::CivilDay(feb_3_noon)
// Iterates all the days of February 2015.
for (turbo::CivilDay d(2015, 2, 1); d < turbo::CivilMonth(2015, 3); ++d) {
// ...
}
ARITHMETIC Civil-time types support natural arithmetic operators such as addition, subtraction, and difference. Arithmetic operates on the civil-time field indicated in the type’s name. Difference operators require arguments with the same alignment and return the answer in units of the alignment. Example:
turbo::CivilDay a(2015, 2, 3);
++a;                              // 2015-02-04 00:00:00
--a;                              // 2015-02-03 00:00:00
turbo::CivilDay b = a + 1;         // 2015-02-04 00:00:00
turbo::CivilDay c = 1 + b;         // 2015-02-05 00:00:00
int n = c - a;                    // n = 2 (civil days)
int m = c - turbo::CivilMonth(c);  // Won't compile: different types.
ACCESSORS Each civil-time type has accessors for all six of the civil-time fields: year, month, day, hour, minute, and second. civil_year_t year() int month() int day() int hour() int minute() int second() Recall that fields inferior to the type’s alignment will be set to their minimum valid value. Example:
turbo::CivilDay d(2015, 6, 28);
// d.year() == 2015
// d.month() == 6
// d.day() == 28
// d.hour() == 0
// d.minute() == 0
// d.second() == 0
CASE STUDY: Adding a month to January 31. One of the classic questions that arises when considering a civil time library (or a date library or a date/time library) is this: “What is the result of adding a month to January 31?” This is an interesting question because it is unclear what is meant by a “month”, and several different answers are possible, depending on context:
  1. March 3 (or 2 if a leap year), if “add a month” means to add a month to the current month, and adjust the date to overflow the extra days into March. In this case the result of “February 31” would be normalized as within the civil-time library.

  2. February 28 (or 29 if a leap year), if “add a month” means to add a month, and adjust the date while holding the resulting month constant. In this case, the result of “February 31” would be truncated to the last day in February.

  3. An error. The caller may get some error, an exception, an invalid date object, or perhaps return false

    . This may make sense because there is no single unambiguously correct answer to the question. Practically speaking, any answer that is not what the programmer intended is the wrong answer. The Turbo time library avoids this problem by making it impossible to ask ambiguous questions. All civil-time objects are aligned to a particular civil-field boundary (such as aligned to a year, month, day, hour, minute, or second), and arithmetic operates on the field to which the object is aligned. This means that in order to “add a month” the object must first be aligned to a month boundary, which is equivalent to the first day of that month. Of course, there are ways to compute an answer the question at hand using this Turbo time library, but they require the programmer to be explicit about the answer they expect. To illustrate, let’s see how to compute all three of the above possible answers to the question of “Jan 31 plus 1

    month”: Example:

    const turbo::CivilDay d(2015, 1, 31);
    // Answer 1:
    // Add 1 to the month field in the constructor, and rely on normalization.
    const auto normalized = turbo::CivilDay(d.year(), d.month() + 1, d.day());
    // normalized == 2015-03-03 (aka Feb 31)
    // Answer 2:
    // Add 1 to month field, capping to the end of next month.
    const auto next_month = turbo::CivilMonth(d) + 1;
    const auto last_day_of_next_month = turbo::CivilDay(next_month + 1) - 1;
    const auto capped = std::min(normalized, last_day_of_next_month);
    // capped == 2015-02-28
    // Answer 3:
    // Signal an error if the normalized answer is not in next month.
    if (turbo::CivilMonth(normalized) != next_month) {
    // error, month overflow
    }
    

    See also

    turbo::CivilSecond

    See also

    turbo::CivilMinute

    See also

    turbo::CivilHour

    See also

    turbo::CivilDay

    See also

    turbo::CivilMonth

    See also

    turbo::CivilYear

    See also

    turbo::civil_year_t

    See also

    turbo::civil_diff_t

    See also

    turbo::Weekday

    See also

    turbo::get_weekday

    See also

    turbo::next_weekday

    See also

    turbo::prev_weekday

    See also

    turbo::get_year_day

    See also

    turbo::GetMonthDay

    See also

    turbo::GetYearWeek

    See also

    turbo::GetMonthWeek

    See also

    turbo::GetYearWeekDay

    See also

    turbo::GetMonthWeekDay

    See also

    turbo::GetYearWeekDayOrdinal

    See also

    turbo::GetMonthWeekDayOrdinal

    See also

    turbo::GetYearWeekOrdinal

    See also

    turbo::GetMonthWeekOrdinal

    See also

    turbo::GetYearWeekOrdinalSunday

    See also

    turbo::GetMonthWeekOrdinalSunday

    See also

    turbo::GetYearWeekSunday

    See also

    turbo::GetMonthWeekSunday

    See also

    turbo::GetYearWeekSundayOrdinal

    See also

    turbo::GetMonthWeekSundayOrdinal

    See also

    turbo::GetYearWeekMonday

    See also

    turbo::GetMonthWeekMonday

    See also

    turbo::GetYearWeekMondayOrdinal

    See also

    turbo::GetMonthWeekMondayOrdinal

using civil_year_t = time_internal::cctz::year_t

Type alias of a civil-time year value.

This type is guaranteed to (at least) support any year value supported by time_t. Example:

turbo::CivilSecond cs = ...;
turbo::civil_year_t y = cs.year();
cs = turbo::CivilSecond(y, 1, 1, 0, 0, 0);  // CivilSecond(CivilYear(cs))

using civil_diff_t = time_internal::cctz::diff_t

Type alias of the difference between two civil-time values.

This type is used to indicate arguments that are not normalized (such as parameters to the civil-time constructors), the results of civil-time subtraction, or the operand to civil-time addition. Example:

turbo::civil_diff_t n_sec = cs1 - cs2;             // cs1 == cs2 + n_sec;

using Weekday = time_internal::cctz::weekday

The Weekday enum class represents the civil-time concept of a “weekday” with members for all days of the week.

Example:

turbo::Weekday wd = turbo::Weekday::thursday;

Functions

inline Weekday get_weekday(CivilSecond cs)

Returns the turbo::Weekday for the given (realigned) civil-time value.

Example:

turbo::CivilDay a(2015, 8, 13);
turbo::Weekday wd = turbo::get_weekday(a);  // wd == turbo::Weekday::thursday

inline CivilDay next_weekday(CivilDay cd, Weekday wd)

Returns the turbo::CivilDay that strictly follows a given turbo::CivilDay, and that falls on the given turbo::Weekday.

Example, given the following month:

August 2015
Su Mo Tu We Th Fr Sa
1
2  3  4  5  6  7  8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31
turbo::CivilDay a(2015, 8, 13);
// turbo::get_weekday(a) == turbo::Weekday::thursday
turbo::CivilDay b = turbo::next_weekday(a, turbo::Weekday::thursday);
// b = 2015-08-20

turbo::CivilDay d = ...
// Gets the following Thursday if d is not already Thursday
turbo::CivilDay thurs1 = turbo::next_weekday(d - 1, turbo::Weekday::thursday);
// Gets the previous Thursday if d is not already Thursday
turbo::CivilDay thurs2 = turbo::prev_weekday(d + 1, turbo::Weekday::thursday);

inline CivilDay prev_weekday(CivilDay cd, Weekday wd)

similar to next_weekday, but returns the previous weekday

See also

turbo::next_weekday

inline int get_year_day(CivilSecond cs)

Returns the day-of-year for the given (realigned) civil-time value.

Example:

turbo::CivilDay a(2015, 1, 1);
int yd_jan_1 = turbo::get_year_day(a);   // yd_jan_1 = 1
turbo::CivilDay b(2015, 12, 31);
int yd_dec_31 = turbo::get_year_day(b);  // yd_dec_31 = 365

std::string format_civil_time(CivilSecond c)

Formats the given civil-time value into a string value of the following format:

std::string format_civil_time(CivilMinute c)

overload for format_civil_time

std::string format_civil_time(CivilHour c)

overload for format_civil_time

std::string format_civil_time(CivilDay c)

overload for format_civil_time

std::string format_civil_time(CivilMonth c)

overload for format_civil_time

std::string format_civil_time(CivilYear c)

overload for format_civil_time

bool parse_civil_time(std::string_view s, CivilSecond *c)

Parses a civil-time value from the specified std::string_view into the passed output parameter.

Returns true upon successful parsing. The expected form of the input string is as follows:

bool parse_civil_time(std::string_view s, CivilMinute *c)

overload for parse_civil_time

bool parse_civil_time(std::string_view s, CivilHour *c)

overload for parse_civil_time

bool parse_civil_time(std::string_view s, CivilDay *c)

overload for parse_civil_time

bool parse_civil_time(std::string_view s, CivilMonth *c)

overload for parse_civil_time

bool parse_civil_time(std::string_view s, CivilYear *c)

overload for parse_civil_time

bool parse_lenient_Civil_time(std::string_view s, CivilSecond *c)

Parses any of the formats accepted by turbo::parse_civil_time(), but is more lenient if the format of the string does not exactly match the associated type.

Example:

turbo::CivilDay d;
bool ok = turbo::parse_lenient_Civil_time("1969-07-20", &d); // OK
ok = turbo::parse_lenient_Civil_time("1969-07-20T10", &d);   // OK: T10 floored
ok = turbo::parse_lenient_Civil_time("1969-07", &d);   // OK: day defaults to 1

Parameters:
  • s – The string to parse.

  • c – The output parameter to store the parsed civil-time value.

Returns:

true if the string was parsed successfully.

bool parse_lenient_Civil_time(std::string_view s, CivilMinute *c)

overload for parse_lenient_Civil_time

bool parse_lenient_Civil_time(std::string_view s, CivilHour *c)

overload for parse_lenient_Civil_time

bool parse_lenient_Civil_time(std::string_view s, CivilDay *c)

overload for parse_lenient_Civil_time

bool parse_lenient_Civil_time(std::string_view s, CivilMonth *c)

overload for parse_lenient_Civil_time

bool parse_lenient_Civil_time(std::string_view s, CivilYear *c)

overload for parse_lenient_Civil_time