2015-11-20 11:54:54 +08:00
|
|
|
#include <cmath>
|
|
|
|
#include <ctime>
|
|
|
|
|
|
|
|
#include <xlnt/utils/timedelta.hpp>
|
|
|
|
|
|
|
|
namespace xlnt {
|
|
|
|
|
2015-11-23 01:41:27 +08:00
|
|
|
timedelta::timedelta() : timedelta(0, 0, 0, 0, 0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
timedelta::timedelta(int days_, int hours_, int minutes_, int seconds_, int microseconds_)
|
|
|
|
: days(days_),
|
|
|
|
hours(hours_),
|
|
|
|
minutes(minutes_),
|
|
|
|
seconds(seconds_),
|
|
|
|
microseconds(microseconds_)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-11-23 02:02:37 +08:00
|
|
|
long double timedelta::to_number() const
|
2015-11-20 11:54:54 +08:00
|
|
|
{
|
2015-11-23 01:41:27 +08:00
|
|
|
std::uint64_t total_microseconds = static_cast<std::uint64_t>(microseconds);
|
|
|
|
total_microseconds += static_cast<std::uint64_t>(seconds * 1e6);
|
|
|
|
total_microseconds += static_cast<std::uint64_t>(minutes * 1e6 * 60);
|
|
|
|
auto microseconds_per_hour = static_cast<std::uint64_t>(1e6) * 60 * 60;
|
|
|
|
total_microseconds += static_cast<std::uint64_t>(hours) * microseconds_per_hour;
|
|
|
|
auto number = total_microseconds / (24.0L * microseconds_per_hour);
|
|
|
|
auto hundred_billion = static_cast<std::uint64_t>(1e9) * 100;
|
|
|
|
number = std::floor(number * hundred_billion + 0.5L) / hundred_billion;
|
|
|
|
number += days;
|
|
|
|
|
|
|
|
return number;
|
2015-11-20 11:54:54 +08:00
|
|
|
}
|
|
|
|
|
2015-11-23 01:41:27 +08:00
|
|
|
timedelta timedelta::from_number(long double raw_time)
|
2015-11-20 11:54:54 +08:00
|
|
|
{
|
2015-11-23 01:41:27 +08:00
|
|
|
timedelta result;
|
|
|
|
|
2015-11-23 02:02:37 +08:00
|
|
|
long double integer_part;
|
|
|
|
long double fractional_part = std::modf(raw_time, &integer_part);
|
2015-11-23 01:41:27 +08:00
|
|
|
|
2015-11-23 02:02:37 +08:00
|
|
|
result.days = static_cast<int>(integer_part);
|
2015-11-23 01:41:27 +08:00
|
|
|
|
|
|
|
fractional_part *= 24;
|
|
|
|
result.hours = (int)fractional_part;
|
|
|
|
fractional_part = 60 * (fractional_part - result.hours);
|
|
|
|
result.minutes = (int)fractional_part;
|
|
|
|
fractional_part = 60 * (fractional_part - result.minutes);
|
|
|
|
result.seconds = (int)fractional_part;
|
|
|
|
fractional_part = 1000000 * (fractional_part - result.seconds);
|
|
|
|
result.microseconds = (int)fractional_part;
|
|
|
|
|
|
|
|
if (result.microseconds == 999999 && fractional_part - result.microseconds > 0.5)
|
|
|
|
{
|
|
|
|
result.microseconds = 0;
|
|
|
|
result.seconds += 1;
|
|
|
|
|
|
|
|
if (result.seconds == 60)
|
|
|
|
{
|
|
|
|
result.seconds = 0;
|
|
|
|
result.minutes += 1;
|
|
|
|
|
|
|
|
//TODO: too much nesting
|
|
|
|
if (result.minutes == 60)
|
|
|
|
{
|
|
|
|
result.minutes = 0;
|
|
|
|
result.hours += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-11-20 11:54:54 +08:00
|
|
|
|
2015-11-23 01:41:27 +08:00
|
|
|
return result;
|
2015-11-20 11:54:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace xlnt
|