// ------------------------------------------------------------------------------------------------ #include "Library/Chrono/Time.hpp" #include "Library/Chrono/Timestamp.hpp" #include "Core/Utility.hpp" // ------------------------------------------------------------------------------------------------ namespace SqMod { // ------------------------------------------------------------------------------------------------ SQMOD_DECL_TYPENAME(Typename, _SC("SqTime")) // ------------------------------------------------------------------------------------------------ SQChar Time::Delimiter = ':'; // ------------------------------------------------------------------------------------------------ Time::Time(int64_t ts) : Time() { const auto h = static_cast< uint8_t >(std::lround(std::floor(ts / 3.6e+9))); ts -= int64_t{h} * 3.6e+9; const auto m = static_cast< uint8_t >(std::lround(std::floor(ts / 6e+7))); ts -= int64_t{m} * 6e+7; const auto s = static_cast< uint8_t >(std::lround(std::floor(ts / 1e+6))); ts -= int64_t{s} * 1e+6; // Set the specified time Set(h, m, s, static_cast< uint16_t >(ts / 1000LL)); } // ------------------------------------------------------------------------------------------------ int32_t Time::Compare(const Time & o) const { if (m_Hour < o.m_Hour) { // NOLINT(bugprone-branch-clone) return -1; } else if (m_Hour > o.m_Hour) { // NOLINT(bugprone-branch-clone) return 1; } else if (m_Minute < o.m_Minute) { return -1; } else if (m_Minute > o.m_Minute) { return 1; } else if (m_Second < o.m_Second) { return -1; } else if (m_Second > o.m_Second) { return 1; } else if (m_Millisecond < o.m_Millisecond) { return -1; } else if (m_Millisecond == o.m_Millisecond) { return 0; } else { return 1; } } // ------------------------------------------------------------------------------------------------ Time Time::operator + (const Time & o) const { return Time(o); } // ------------------------------------------------------------------------------------------------ Time Time::operator - (const Time & o) const { return Time(o); } // ------------------------------------------------------------------------------------------------ Time Time::operator * (const Time & o) const { return Time(o); } // ------------------------------------------------------------------------------------------------ Time Time::operator / (const Time & o) const { return Time(o); } // ------------------------------------------------------------------------------------------------ String Time::ToString() const { return fmt::format("{:02}{}{:02}{}{:02}{}{}", m_Hour, m_Delimiter, m_Minute, m_Delimiter, m_Second, m_Delimiter, m_Millisecond); } // ------------------------------------------------------------------------------------------------ void Time::Set(uint8_t hour, uint8_t minute, uint8_t second, uint16_t millisecond) { // Is the specified hour within range? if (hour >= 24) { STHROWF("Hour value is out of range: {} >= 24", hour); } // Is the specified minute within range? else if (minute >= 60) { STHROWF("Minute value is out of range: {} >= 60", minute); } // Is the specified second within range? else if (second >= 60) { STHROWF("Second value is out of range: {} >= 60", second); } // Is the specified millisecond within range? else if (millisecond >= 1000) { STHROWF("Millisecond value is out of range: {} >= 1000", millisecond); } // Now it's safe to assign the values m_Hour = hour; m_Minute = minute; m_Second = second; m_Millisecond = millisecond; } // ------------------------------------------------------------------------------------------------ void Time::SetStr(const SQChar * str) { // The format specifications that will be used to scan the string static SQChar fs[] = _SC(" %u : %u : %u : %u "); // Is the specified string empty? if (!str || *str == '\0') { // Clear the values m_Hour = 0; m_Minute = 0; m_Second = 0; m_Millisecond = 0; // We're done here return; } // Assign the specified delimiter fs[4] = m_Delimiter; fs[9] = m_Delimiter; fs[14] = m_Delimiter; // The sscanf function requires at least 32 bit integers uint32_t hour = 0, minute = 0, second = 0, milli = 0; // Attempt to extract the component values from the specified string sscanf(str, fs, &hour, &minute, &second, &milli); // Clamp the extracted values to the boundaries of associated type and assign them Set(ClampL< uint32_t, uint8_t >(hour), ClampL< uint32_t, uint8_t >(minute), ClampL< uint32_t, uint8_t >(second), ClampL< uint32_t, uint16_t >(milli) ); } // ------------------------------------------------------------------------------------------------ void Time::SetHour(uint8_t hour) { // Is the specified hour within range? if (hour >= 24) { STHROWF("Hour value is out of range: {} >= 24", hour); } // Now it's safe to assign the value m_Hour = hour; } // ------------------------------------------------------------------------------------------------ void Time::SetMinute(uint8_t minute) { // Is the specified minute within range? if (minute >= 60) { STHROWF("Minute value is out of range: {} >= 60", minute); } // Now it's safe to assign the value m_Minute = minute; } // ------------------------------------------------------------------------------------------------ void Time::SetSecond(uint8_t second) { // Is the specified second within range? if (second >= 60) { STHROWF("Second value is out of range: {} >= 60", second); } // Now it's safe to assign the value m_Second = second; } // ------------------------------------------------------------------------------------------------ void Time::SetMillisecond(uint16_t millisecond) { // Is the specified millisecond within range? if (millisecond >= 1000) { STHROWF("Millisecond value is out of range: {} >= 1000", millisecond); } // Now it's safe to assign the value m_Millisecond = millisecond; } // ------------------------------------------------------------------------------------------------ Time & Time::AddHours(int32_t hours) { // Did we even add any hours? if (hours) { // Add the specified amount of hours m_Hour += (hours % 24); // Make sure the value is within range m_Hour %= 24; } // Allow chaining operations return *this; } // ------------------------------------------------------------------------------------------------ Time & Time::AddMinutes(int32_t minutes) { // Did we even add any minutes? if (minutes) { // Extract the number of hours auto hours = static_cast< int32_t >(minutes / 60); // Extract the number of minutes m_Minute += (minutes % 60); // Are the minutes overlapping with the next hour? if (m_Minute >= 60) { // Increase the hours ++hours; // Subtract one hour from minutes m_Minute %= 60; } // Should we add any hours? if (hours) { AddHours(hours); } } // Allow chaining operations return *this; } // ------------------------------------------------------------------------------------------------ Time & Time::AddSeconds(int32_t seconds) { // Did we even add any seconds? if (seconds) { // Extract the number of minutes auto minutes = static_cast< int32_t >(seconds / 60); // Extract the number of seconds m_Second += (seconds % 60); // Are the seconds overlapping with the next minute? if (m_Second >= 60) { // Increase the minutes ++minutes; // Subtract one minute from seconds m_Second %= 60; } // Should we add any minutes? if (minutes) { AddMinutes(minutes); } } // Allow chaining operations return *this; } // ------------------------------------------------------------------------------------------------ Time & Time::AddMilliseconds(int32_t milliseconds) { // Did we even add any milliseconds? if (milliseconds) { // Extract the number of seconds auto seconds = static_cast< int32_t >(milliseconds / 1000); // Extract the number of milliseconds m_Millisecond += (milliseconds / 1000); // Are the milliseconds overlapping with the next second? if (m_Millisecond >= 1000) { // Increase the seconds ++seconds; // Subtract one second from milliseconds m_Millisecond %= 1000; } // Should we add any seconds? if (seconds) { AddSeconds(seconds); } } // Allow chaining operations return *this; } // ------------------------------------------------------------------------------------------------ Time Time::AndHours(int32_t hours) { // Did we even add any hours? if (hours) { return Time(static_cast< uint8_t >((m_Hour + (hours % 24)) % 24), m_Minute, m_Second, m_Millisecond); } // Return the time as is return Time(*this); } // ------------------------------------------------------------------------------------------------ Time Time::AndMinutes(int32_t minutes) { // Did we even added any minutes? if (!minutes) { return Time(*this); // Return the time as is } // Extract the number of hours auto hours = static_cast< int32_t >(minutes / 60); // Extract the number of minutes minutes = m_Minute + (minutes % 60); // Are the minutes overlapping with the next hour? if (minutes >= 60) { ++hours; // Increase hours } // Replicate the current time Time t(*this); // Should we add any hours? if (hours) { t.AddHours(hours); } // Assign the resulted minutes t.m_Minute = static_cast< uint8_t >(minutes % 60); // Return the result return t; } // ------------------------------------------------------------------------------------------------ Time Time::AndSeconds(int32_t seconds) { // Did we even added any seconds? if (!seconds) { return Time(*this); // Return the time as is } // Extract the number of minutes auto minutes = static_cast< int32_t >(seconds / 60); // Extract the number of seconds seconds = m_Second + (seconds % 60); // Are the seconds overlapping with the next minute? if (seconds >= 60) { ++minutes; // Increase minutes } // Replicate the current time Time t(*this); // Should we add any minutes? if (minutes) { t.AddMinutes(minutes); } // Assign the resulted seconds t.m_Second = static_cast< uint8_t >(seconds % 60); // Return the result return t; } // ------------------------------------------------------------------------------------------------ Time Time::AndMilliseconds(int32_t milliseconds) { // Did we even added any milliseconds? if (!milliseconds) { return Time(*this); // Return the time as is } // Extract the number of seconds auto seconds = static_cast< int32_t >(milliseconds / 1000); // Extract the number of milliseconds milliseconds = m_Millisecond + (milliseconds % 1000); // Are the milliseconds overlapping with the next second? if (milliseconds >= 1000) { ++seconds; // Increase seconds } // Replicate the current time Time t(*this); // Should we add any seconds? if (seconds) { t.AddSeconds(seconds); } // Assign the resulted milliseconds t.m_Millisecond = static_cast< uint16_t >(milliseconds % 1000); // Return the result return t; } // ------------------------------------------------------------------------------------------------ Timestamp Time::GetTimestamp() const { // Calculate the microseconds in the current time auto ms = static_cast< int64_t >(m_Hour * 3600000000LL); ms += static_cast< int64_t >(m_Minute * 60000000L); ms += static_cast< int64_t >(m_Second * 1000000L); ms += static_cast< int64_t >(m_Millisecond * 1000L); // Return the resulted timestamp return Timestamp(ms); } // ------------------------------------------------------------------------------------------------ std::time_t Time::ToTimeT() const { return GetTimestamp().ToTimeT(); } // ================================================================================================ void Register_ChronoTime(HSQUIRRELVM vm, Table & /*cns*/) { RootTable(vm).Bind(Typename::Str, Class< Time >(vm, Typename::Str) // Constructors .Ctor() .Ctor< uint8_t >() .Ctor< uint8_t, uint8_t >() .Ctor< uint8_t, uint8_t, uint8_t >() .Ctor< uint8_t, uint8_t, uint8_t, uint16_t >() // Static Properties .SetStaticValue(_SC("GlobalDelimiter"), &Time::Delimiter) // Core Meta-methods .SquirrelFunc(_SC("_typename"), &Typename::Fn) .Func(_SC("_tostring"), &Time::ToString) .Func(_SC("cmp"), &Time::Cmp) // Meta-methods .Func< Time (Time::*)(const Time &) const >(_SC("_add"), &Time::operator +) .Func< Time (Time::*)(const Time &) const >(_SC("_sub"), &Time::operator -) .Func< Time (Time::*)(const Time &) const >(_SC("_mul"), &Time::operator *) .Func< Time (Time::*)(const Time &) const >(_SC("_div"), &Time::operator /) // Properties .Prop(_SC("Delimiter"), &Time::GetDelimiter, &Time::SetDelimiter) .Prop(_SC("Str"), &Time::GetStr, &Time::SetStr) .Prop(_SC("Hour"), &Time::GetHour, &Time::SetHour) .Prop(_SC("Minute"), &Time::GetMinute, &Time::SetMinute) .Prop(_SC("Second"), &Time::GetSecond, &Time::SetSecond) .Prop(_SC("Millisecond"), &Time::GetMillisecond, &Time::SetMillisecond) .Prop(_SC("Timestamp"), &Time::GetTimestamp) // Member Methods .Func(_SC("AddHours"), &Time::AddHours) .Func(_SC("AddMinutes"), &Time::AddMinutes) .Func(_SC("AddSeconds"), &Time::AddSeconds) .Func(_SC("AddMillis"), &Time::AddMilliseconds) .Func(_SC("AddMilliseconds"), &Time::AddMilliseconds) .Func(_SC("AndHours"), &Time::AndHours) .Func(_SC("AndMinutes"), &Time::AndMinutes) .Func(_SC("AndSeconds"), &Time::AndSeconds) .Func(_SC("AndMillis"), &Time::AndMilliseconds) .Func(_SC("AndMilliseconds"), &Time::AndMilliseconds) // Overloaded Methods .Overload< void (Time::*)(uint8_t) >(_SC("Set"), &Time::Set) .Overload< void (Time::*)(uint8_t, uint8_t) >(_SC("Set"), &Time::Set) .Overload< void (Time::*)(uint8_t, uint8_t, uint8_t) >(_SC("Set"), &Time::Set) .Overload< void (Time::*)(uint8_t, uint8_t, uint8_t, uint16_t) >(_SC("Set"), &Time::Set) ); } } // Namespace:: SqMod