2019-11-13 14:33:15 +08:00
|
|
|
|
// Copyright (c) 2016-2018 Kiwano - Nomango
|
2020-01-21 10:09:55 +08:00
|
|
|
|
//
|
2019-11-13 14:33:15 +08:00
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
|
|
// furnished to do so, subject to the following conditions:
|
2020-01-21 10:09:55 +08:00
|
|
|
|
//
|
2019-11-13 14:33:15 +08:00
|
|
|
|
// The above copyright notice and this permission notice shall be included in
|
|
|
|
|
|
// all copies or substantial portions of the Software.
|
2020-01-21 10:09:55 +08:00
|
|
|
|
//
|
2019-11-13 14:33:15 +08:00
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
|
|
// THE SOFTWARE.
|
|
|
|
|
|
|
2020-02-10 14:41:19 +08:00
|
|
|
|
#include <kiwano/core/Exception.h>
|
2020-01-21 10:09:55 +08:00
|
|
|
|
#include <kiwano/core/Logger.h>
|
|
|
|
|
|
#include <kiwano/core/Time.h>
|
2019-11-13 14:33:15 +08:00
|
|
|
|
#include <regex>
|
|
|
|
|
|
#include <unordered_map>
|
|
|
|
|
|
|
|
|
|
|
|
namespace kiwano
|
|
|
|
|
|
{
|
2020-01-21 10:09:55 +08:00
|
|
|
|
//-------------------------------------------------------
|
|
|
|
|
|
// Time
|
|
|
|
|
|
//-------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
Time::Time()
|
|
|
|
|
|
: dur_(0)
|
|
|
|
|
|
{
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Time::Time(long dur)
|
|
|
|
|
|
: dur_(dur)
|
|
|
|
|
|
{
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Time Time::operator+(const Duration& dur) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Time{ dur_ + dur.Milliseconds() };
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Time Time::operator-(const Duration& dur) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Time{ dur_ - dur.Milliseconds() };
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Time& Time::operator+=(const Duration& other)
|
|
|
|
|
|
{
|
|
|
|
|
|
dur_ += other.Milliseconds();
|
|
|
|
|
|
return (*this);
|
|
|
|
|
|
}
|
2019-11-13 14:33:15 +08:00
|
|
|
|
|
2020-01-21 10:09:55 +08:00
|
|
|
|
Time& Time::operator-=(const Duration& other)
|
|
|
|
|
|
{
|
|
|
|
|
|
dur_ -= other.Milliseconds();
|
|
|
|
|
|
return (*this);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration Time::operator-(const Time& other) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Duration(dur_ - other.dur_);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Time Time::Now() noexcept
|
|
|
|
|
|
{
|
|
|
|
|
|
static LARGE_INTEGER freq = {};
|
|
|
|
|
|
if (freq.QuadPart == 0LL)
|
|
|
|
|
|
{
|
|
|
|
|
|
// the Function will always succceed on systems that run Windows XP or later
|
|
|
|
|
|
QueryPerformanceFrequency(&freq);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
LARGE_INTEGER count;
|
|
|
|
|
|
QueryPerformanceCounter(&count);
|
|
|
|
|
|
|
|
|
|
|
|
const long long whole = (count.QuadPart / freq.QuadPart) * 1000LL;
|
|
|
|
|
|
const long long part = (count.QuadPart % freq.QuadPart) * 1000LL / freq.QuadPart;
|
|
|
|
|
|
return Time{ static_cast<long>(whole + part) };
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-------------------------------------------------------
|
|
|
|
|
|
// Duration
|
|
|
|
|
|
//-------------------------------------------------------
|
|
|
|
|
|
|
|
|
|
|
|
const Duration Duration::Ms = 1L;
|
|
|
|
|
|
const Duration Duration::Second = 1000 * Duration::Ms;
|
|
|
|
|
|
const Duration Duration::Minute = 60 * Duration::Second;
|
|
|
|
|
|
const Duration Duration::Hour = 60 * Duration::Minute;
|
|
|
|
|
|
|
|
|
|
|
|
namespace
|
|
|
|
|
|
{
|
2020-02-10 13:47:00 +08:00
|
|
|
|
const auto duration_regex = std::regex(R"(^[-+]?([0-9]*(\.[0-9]*)?(h|m|s|ms)+)+$)");
|
2020-01-21 10:09:55 +08:00
|
|
|
|
|
|
|
|
|
|
typedef std::unordered_map<String, Duration> UnitMap;
|
2020-02-10 13:47:00 +08:00
|
|
|
|
|
|
|
|
|
|
const auto unit_map =
|
|
|
|
|
|
UnitMap{ { "ms", Duration::Ms }, { "s", Duration::Second }, { "m", Duration::Minute }, { "h", Duration::Hour } };
|
|
|
|
|
|
|
2020-01-21 10:09:55 +08:00
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
|
|
Duration::Duration()
|
|
|
|
|
|
: milliseconds_(0)
|
|
|
|
|
|
{
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Duration::Duration(long milliseconds)
|
|
|
|
|
|
: milliseconds_(milliseconds)
|
|
|
|
|
|
{
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
float Duration::Seconds() const
|
|
|
|
|
|
{
|
|
|
|
|
|
long sec = milliseconds_ / Second.milliseconds_;
|
|
|
|
|
|
long ms = milliseconds_ % Second.milliseconds_;
|
|
|
|
|
|
return static_cast<float>(sec) + static_cast<float>(ms) / 1000.f;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
float Duration::Minutes() const
|
|
|
|
|
|
{
|
|
|
|
|
|
long min = milliseconds_ / Minute.milliseconds_;
|
|
|
|
|
|
long ms = milliseconds_ % Minute.milliseconds_;
|
|
|
|
|
|
return static_cast<float>(min) + static_cast<float>(ms) / (60 * 1000.f);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
float Duration::Hours() const
|
|
|
|
|
|
{
|
|
|
|
|
|
long hour = milliseconds_ / Hour.milliseconds_;
|
|
|
|
|
|
long ms = milliseconds_ % Hour.milliseconds_;
|
|
|
|
|
|
return static_cast<float>(hour) + static_cast<float>(ms) / (60 * 60 * 1000.f);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String Duration::ToString() const
|
|
|
|
|
|
{
|
|
|
|
|
|
if (IsZero())
|
|
|
|
|
|
{
|
2020-02-10 13:47:00 +08:00
|
|
|
|
return String("0s");
|
2020-01-21 10:09:55 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String result;
|
|
|
|
|
|
long total_ms = milliseconds_;
|
|
|
|
|
|
if (total_ms < 0)
|
|
|
|
|
|
{
|
2020-02-10 13:47:00 +08:00
|
|
|
|
result.append("-");
|
2020-01-21 10:09:55 +08:00
|
|
|
|
total_ms = -total_ms;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
long hour = total_ms / Hour.milliseconds_;
|
|
|
|
|
|
long min = total_ms / Minute.milliseconds_ - hour * 60;
|
|
|
|
|
|
long sec = total_ms / Second.milliseconds_ - (hour * 60 * 60 + min * 60);
|
|
|
|
|
|
long ms = total_ms % Second.milliseconds_;
|
|
|
|
|
|
|
|
|
|
|
|
if (hour)
|
|
|
|
|
|
{
|
2020-02-10 13:47:00 +08:00
|
|
|
|
result.append(String::parse(hour)).append("h");
|
|
|
|
|
|
result.append(String::parse(min)).append("m");
|
2020-01-21 10:09:55 +08:00
|
|
|
|
}
|
|
|
|
|
|
else if (min)
|
|
|
|
|
|
{
|
2020-02-10 13:47:00 +08:00
|
|
|
|
result.append(String::parse(min)).append("m");
|
2020-01-21 10:09:55 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (ms != 0)
|
|
|
|
|
|
{
|
2020-02-10 13:47:00 +08:00
|
|
|
|
result.append(String::parse(static_cast<float>(sec) + static_cast<float>(ms) / 1000.f)).append("s");
|
2020-01-21 10:09:55 +08:00
|
|
|
|
}
|
|
|
|
|
|
else if (sec != 0)
|
|
|
|
|
|
{
|
2020-02-10 13:47:00 +08:00
|
|
|
|
result.append(String::parse(sec)).append("s");
|
2020-01-21 10:09:55 +08:00
|
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Duration::operator==(const Duration& other) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return milliseconds_ == other.milliseconds_;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Duration::operator!=(const Duration& other) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return milliseconds_ != other.milliseconds_;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Duration::operator>(const Duration& other) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return milliseconds_ > other.milliseconds_;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Duration::operator>=(const Duration& other) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return milliseconds_ >= other.milliseconds_;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Duration::operator<(const Duration& other) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return milliseconds_ < other.milliseconds_;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Duration::operator<=(const Duration& other) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return milliseconds_ <= other.milliseconds_;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
float Duration::operator/(const Duration& other) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return static_cast<float>(milliseconds_) / other.milliseconds_;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration Duration::operator+(const Duration& other) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Duration(milliseconds_ + other.milliseconds_);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration Duration::operator-(const Duration& other) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Duration(milliseconds_ - other.milliseconds_);
|
2019-11-13 14:33:15 +08:00
|
|
|
|
}
|
2020-01-21 10:09:55 +08:00
|
|
|
|
|
|
|
|
|
|
const Duration Duration::operator-() const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Duration(-milliseconds_);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration Duration::operator*(int val) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Duration(milliseconds_ * val);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration Duration::operator*(unsigned long long val) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Duration(static_cast<long>(milliseconds_ * val));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration Duration::operator*(float val) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Duration(static_cast<long>(milliseconds_ * val));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration Duration::operator*(double val) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Duration(static_cast<long>(milliseconds_ * val));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration Duration::operator*(long double val) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Duration(static_cast<long>(milliseconds_ * val));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration Duration::operator/(int val) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Duration(milliseconds_ / val);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration Duration::operator/(float val) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Duration(static_cast<long>(milliseconds_ / val));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration Duration::operator/(double val) const
|
|
|
|
|
|
{
|
|
|
|
|
|
return Duration(static_cast<long>(milliseconds_ / val));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Duration& Duration::operator+=(const Duration& other)
|
|
|
|
|
|
{
|
|
|
|
|
|
milliseconds_ += other.milliseconds_;
|
|
|
|
|
|
return (*this);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Duration& Duration::operator-=(const Duration& other)
|
|
|
|
|
|
{
|
|
|
|
|
|
milliseconds_ -= other.milliseconds_;
|
|
|
|
|
|
return (*this);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Duration& Duration::operator*=(int val)
|
|
|
|
|
|
{
|
|
|
|
|
|
milliseconds_ *= val;
|
|
|
|
|
|
return (*this);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Duration& Duration::operator/=(int val)
|
|
|
|
|
|
{
|
|
|
|
|
|
milliseconds_ = static_cast<long>(milliseconds_ / val);
|
|
|
|
|
|
return (*this);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Duration& Duration::operator*=(float val)
|
|
|
|
|
|
{
|
|
|
|
|
|
milliseconds_ = static_cast<long>(milliseconds_ * val);
|
|
|
|
|
|
return (*this);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Duration& Duration::operator/=(float val)
|
|
|
|
|
|
{
|
|
|
|
|
|
milliseconds_ = static_cast<long>(milliseconds_ / val);
|
|
|
|
|
|
return (*this);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Duration& Duration::operator*=(double val)
|
|
|
|
|
|
{
|
|
|
|
|
|
milliseconds_ = static_cast<long>(milliseconds_ * val);
|
|
|
|
|
|
return (*this);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Duration& Duration::operator/=(double val)
|
|
|
|
|
|
{
|
|
|
|
|
|
milliseconds_ = static_cast<long>(milliseconds_ / val);
|
|
|
|
|
|
return (*this);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration operator*(int val, const Duration& dur)
|
|
|
|
|
|
{
|
|
|
|
|
|
return dur * val;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration operator/(int val, const Duration& dur)
|
|
|
|
|
|
{
|
|
|
|
|
|
return dur / val;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration operator*(float val, const Duration& dur)
|
|
|
|
|
|
{
|
|
|
|
|
|
return dur * val;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration operator/(float val, const Duration& dur)
|
|
|
|
|
|
{
|
|
|
|
|
|
return dur / val;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration operator*(double val, const Duration& dur)
|
|
|
|
|
|
{
|
|
|
|
|
|
return dur * val;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration operator/(double val, const Duration& dur)
|
|
|
|
|
|
{
|
|
|
|
|
|
return dur / val;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const Duration operator*(long double val, const Duration& dur)
|
|
|
|
|
|
{
|
|
|
|
|
|
return dur * val;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Duration Duration::Parse(const String& format)
|
|
|
|
|
|
{
|
|
|
|
|
|
bool negative = false;
|
|
|
|
|
|
size_t len = format.length();
|
|
|
|
|
|
size_t pos = 0;
|
|
|
|
|
|
Duration ret;
|
|
|
|
|
|
|
|
|
|
|
|
if (!std::regex_match(format.c_str(), duration_regex))
|
|
|
|
|
|
{
|
2020-02-10 14:41:19 +08:00
|
|
|
|
throw Exception("Duration::Parse failed, invalid duration");
|
2020-01-21 10:09:55 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2020-02-10 13:47:00 +08:00
|
|
|
|
if (format.empty() || format == "0")
|
2020-01-21 10:09:55 +08:00
|
|
|
|
{
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// <20><><EFBFBD><EFBFBD>λ
|
2020-02-10 13:47:00 +08:00
|
|
|
|
if (format[0] == '-' || format[0] == '+')
|
2020-01-21 10:09:55 +08:00
|
|
|
|
{
|
2020-02-10 13:47:00 +08:00
|
|
|
|
negative = (format[0] == '-');
|
2020-01-21 10:09:55 +08:00
|
|
|
|
pos++;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
while (pos < len)
|
|
|
|
|
|
{
|
|
|
|
|
|
// <20><>ֵ
|
|
|
|
|
|
size_t i = pos;
|
|
|
|
|
|
for (; i < len; ++i)
|
|
|
|
|
|
{
|
|
|
|
|
|
wchar_t ch = format[i];
|
2020-02-10 13:47:00 +08:00
|
|
|
|
if (!(ch == '.' || '0' <= ch && ch <= '9'))
|
2020-01-21 10:09:55 +08:00
|
|
|
|
{
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String num_str = format.substr(pos, i - pos);
|
|
|
|
|
|
|
2020-02-10 13:47:00 +08:00
|
|
|
|
pos = i;
|
|
|
|
|
|
|
|
|
|
|
|
if (num_str.empty() || num_str == ".")
|
2020-02-10 14:41:19 +08:00
|
|
|
|
throw Exception("Duration::Parse failed, invalid duration");
|
2020-01-21 10:09:55 +08:00
|
|
|
|
|
|
|
|
|
|
// <20><>λ
|
|
|
|
|
|
for (; i < len; ++i)
|
|
|
|
|
|
{
|
|
|
|
|
|
wchar_t ch = format[i];
|
2020-02-10 13:47:00 +08:00
|
|
|
|
if (ch == '.' || '0' <= ch && ch <= '9')
|
2020-01-21 10:09:55 +08:00
|
|
|
|
{
|
|
|
|
|
|
break;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String unit_str = format.substr(pos, i - pos);
|
2020-02-10 13:47:00 +08:00
|
|
|
|
|
|
|
|
|
|
pos = i;
|
2020-01-21 10:09:55 +08:00
|
|
|
|
|
|
|
|
|
|
if (unit_map.find(unit_str) == unit_map.end())
|
2020-02-10 14:41:19 +08:00
|
|
|
|
throw Exception("Duration::Parse failed, invalid duration");
|
2020-01-21 10:09:55 +08:00
|
|
|
|
|
2020-02-10 13:47:00 +08:00
|
|
|
|
double num = std::stod(num_str.c_str());
|
2020-01-21 10:09:55 +08:00
|
|
|
|
Duration unit = unit_map.at(unit_str);
|
|
|
|
|
|
ret += unit * num;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (negative)
|
|
|
|
|
|
{
|
|
|
|
|
|
ret = -ret;
|
|
|
|
|
|
}
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace kiwano
|