529 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			529 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
| // Copyright (c) 2016-2018 Easy2D - Nomango
 | ||
| // 
 | ||
| // 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:
 | ||
| // 
 | ||
| // The above copyright notice and this permission notice shall be included in
 | ||
| // all copies or substantial portions of the Software.
 | ||
| // 
 | ||
| // 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.
 | ||
| 
 | ||
| #include "time.h"
 | ||
| #include "logs.h"
 | ||
| #include <regex>
 | ||
| 
 | ||
| namespace easy2d
 | ||
| {
 | ||
| 	namespace time
 | ||
| 	{
 | ||
| 		//-------------------------------------------------------
 | ||
| 		// TimePoint
 | ||
| 		//-------------------------------------------------------
 | ||
| 
 | ||
| 		TimePoint::TimePoint()
 | ||
| 		{
 | ||
| 		}
 | ||
| 
 | ||
| 		TimePoint::TimePoint(const Duration& dur)
 | ||
| 			: dur(dur)
 | ||
| 		{
 | ||
| 		}
 | ||
| 
 | ||
| 		TimePoint::TimePoint(const TimePoint & other)
 | ||
| 			: dur(other.dur)
 | ||
| 		{
 | ||
| 		}
 | ||
| 
 | ||
| 		TimePoint::TimePoint(TimePoint && other)
 | ||
| 			: dur(std::move(other.dur))
 | ||
| 		{
 | ||
| 		}
 | ||
| 
 | ||
| 		const TimePoint TimePoint::operator+(const Duration & dur) const
 | ||
| 		{
 | ||
| 			return TimePoint(dur + dur);
 | ||
| 		}
 | ||
| 
 | ||
| 		const TimePoint TimePoint::operator-(const Duration & dur) const
 | ||
| 		{
 | ||
| 			return TimePoint(dur - dur);
 | ||
| 		}
 | ||
| 
 | ||
| 		TimePoint & TimePoint::operator+=(const Duration & other)
 | ||
| 		{
 | ||
| 			dur += other;
 | ||
| 			return (*this);
 | ||
| 		}
 | ||
| 
 | ||
| 		TimePoint & TimePoint::operator-=(const Duration &other)
 | ||
| 		{
 | ||
| 			dur -= other;
 | ||
| 			return (*this);
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration TimePoint::operator-(const TimePoint & other) const
 | ||
| 		{
 | ||
| 			return dur - other.dur;
 | ||
| 		}
 | ||
| 
 | ||
| 		TimePoint& TimePoint::operator=(const TimePoint & other) E2D_NOEXCEPT
 | ||
| 		{
 | ||
| 			if (this == &other)
 | ||
| 				return *this;
 | ||
| 
 | ||
| 			dur = other.dur;
 | ||
| 			return *this;
 | ||
| 		}
 | ||
| 
 | ||
| 		TimePoint& TimePoint::operator=(TimePoint && other) E2D_NOEXCEPT
 | ||
| 		{
 | ||
| 			if (this == &other)
 | ||
| 				return *this;
 | ||
| 
 | ||
| 			dur = std::move(other.dur);
 | ||
| 			return *this;
 | ||
| 		}
 | ||
| 
 | ||
| 
 | ||
| 		//-------------------------------------------------------
 | ||
| 		// Duration
 | ||
| 		//-------------------------------------------------------
 | ||
| 
 | ||
| 		const Duration Millisecond	= Duration(1);
 | ||
| 		const Duration Second		= 1000 * Millisecond;
 | ||
| 		const Duration Minute		= 60 * Second;
 | ||
| 		const Duration Hour			= 60 * Minute;
 | ||
| 
 | ||
| 		namespace
 | ||
| 		{
 | ||
| 			const auto duration_regex = std::wregex(LR"([-+]?([0-9]*(\.[0-9]*)?[a-z]+)+)");
 | ||
| 
 | ||
| 			typedef std::map<std::wstring, Duration> UnitMap;
 | ||
| 			const auto unit_map = UnitMap
 | ||
| 			{
 | ||
| 				{L"ms", Millisecond},
 | ||
| 				{L"s", Second},
 | ||
| 				{L"m", Minute},
 | ||
| 				{L"h", Hour}
 | ||
| 			};
 | ||
| 		}
 | ||
| 
 | ||
| 		Duration::Duration()
 | ||
| 			: milliseconds_(0)
 | ||
| 		{
 | ||
| 		}
 | ||
| 
 | ||
| 		Duration::Duration(int64_t milliseconds)
 | ||
| 			: milliseconds_(milliseconds)
 | ||
| 		{
 | ||
| 		}
 | ||
| 
 | ||
| 		float Duration::Seconds() const
 | ||
| 		{
 | ||
| 			int64_t sec = milliseconds_ / Second.milliseconds_;
 | ||
| 			int64_t ms = milliseconds_ % Second.milliseconds_;
 | ||
| 			return static_cast<float>(sec) + static_cast<float>(ms) / 1000.f;
 | ||
| 		}
 | ||
| 
 | ||
| 		float Duration::Minutes() const
 | ||
| 		{
 | ||
| 			int64_t min = milliseconds_ / Minute.milliseconds_;
 | ||
| 			int64_t ms = milliseconds_ % Minute.milliseconds_;
 | ||
| 			return static_cast<float>(min) + static_cast<float>(ms) / (60 * 1000.f);
 | ||
| 		}
 | ||
| 
 | ||
| 		float Duration::Hours() const
 | ||
| 		{
 | ||
| 			int64_t hour = milliseconds_ / Hour.milliseconds_;
 | ||
| 			int64_t ms = milliseconds_ % Hour.milliseconds_;
 | ||
| 			return static_cast<float>(hour) + static_cast<float>(ms) / (60 * 60 * 1000.f);
 | ||
| 		}
 | ||
| 
 | ||
| 		std::wstring easy2d::time::Duration::ToString() const
 | ||
| 		{
 | ||
| 			if (IsZero())
 | ||
| 			{
 | ||
| 				return std::wstring(L"0s");
 | ||
| 			}
 | ||
| 
 | ||
| 			std::wstring result;
 | ||
| 			int64_t hour = milliseconds_ / Hour.milliseconds_;
 | ||
| 			int64_t min = milliseconds_ / Minute.milliseconds_ - hour * 60;
 | ||
| 			int64_t sec = milliseconds_ / Second.milliseconds_ - (hour * 60 * 60 + min * 60);
 | ||
| 			int64_t ms = milliseconds_ % Second.milliseconds_;
 | ||
| 
 | ||
| 			if (milliseconds_ < 0)
 | ||
| 				result.append(L"-");
 | ||
| 
 | ||
| 			if (hour)
 | ||
| 			{
 | ||
| 				result.append(std::to_wstring(hour)).append(L"h");
 | ||
| 				result.append(std::to_wstring(min)).append(L"m");
 | ||
| 			}
 | ||
| 			else if(min)
 | ||
| 			{
 | ||
| 				result.append(std::to_wstring(min)).append(L"m");
 | ||
| 			}
 | ||
| 
 | ||
| 			if (ms != 0)
 | ||
| 			{
 | ||
| 				auto float_to_str = [](float val) -> std::wstring
 | ||
| 				{
 | ||
| 					wchar_t buf[10] = {};
 | ||
| 					::swprintf_s(buf, L"%.2f", val);
 | ||
| 					return std::wstring(buf);
 | ||
| 				};
 | ||
| 
 | ||
| 				result.append(float_to_str(static_cast<float>(sec) + static_cast<float>(ms) / 1000.f))
 | ||
| 					.append(L"s");
 | ||
| 			}
 | ||
| 			else if (sec != 0)
 | ||
| 			{
 | ||
| 				result.append(std::to_wstring(sec)).append(L"s");
 | ||
| 			}
 | ||
| 			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 easy2d::time::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_);
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration Duration::operator-() const
 | ||
| 		{
 | ||
| 			return Duration(-milliseconds_);
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration Duration::operator*(int val) const
 | ||
| 		{
 | ||
| 			return Duration(milliseconds_ * val);
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration Duration::operator/(int val) const
 | ||
| 		{
 | ||
| 			return Duration(milliseconds_ / val);
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration easy2d::time::Duration::operator*(unsigned long long val) const
 | ||
| 		{
 | ||
| 			return Duration(static_cast<int64_t>(milliseconds_ * val));
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration easy2d::time::Duration::operator/(unsigned long long val) const
 | ||
| 		{
 | ||
| 			return Duration(static_cast<int64_t>(milliseconds_ / val));
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration Duration::operator*(float val) const
 | ||
| 		{
 | ||
| 			return Duration(static_cast<int64_t>(milliseconds_ * val));
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration Duration::operator/(float val) const
 | ||
| 		{
 | ||
| 			return Duration(static_cast<int64_t>(milliseconds_ / val));
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration Duration::operator*(double val) const
 | ||
| 		{
 | ||
| 			return Duration(static_cast<int64_t>(milliseconds_ * val));
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration Duration::operator*(long double val) const
 | ||
| 		{
 | ||
| 			return Duration(static_cast<int64_t>(milliseconds_ * val));
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration Duration::operator/(double val) const
 | ||
| 		{
 | ||
| 			return Duration(static_cast<int64_t>(milliseconds_ / val));
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration Duration::operator/(long double val) const
 | ||
| 		{
 | ||
| 			return Duration(static_cast<int64_t>(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<int64_t>(milliseconds_ / val);
 | ||
| 			return (*this);
 | ||
| 		}
 | ||
| 
 | ||
| 		Duration & easy2d::time::Duration::operator*=(unsigned long long val)
 | ||
| 		{
 | ||
| 			milliseconds_ = static_cast<int64_t>(milliseconds_ * val);
 | ||
| 			return (*this);
 | ||
| 		}
 | ||
| 
 | ||
| 		Duration & easy2d::time::Duration::operator/=(unsigned long long val)
 | ||
| 		{
 | ||
| 			milliseconds_ = static_cast<int64_t>(milliseconds_ * val);
 | ||
| 			return (*this);
 | ||
| 		}
 | ||
| 
 | ||
| 		Duration & Duration::operator*=(float val)
 | ||
| 		{
 | ||
| 			milliseconds_ = static_cast<int64_t>(milliseconds_ * val);
 | ||
| 			return (*this);
 | ||
| 		}
 | ||
| 
 | ||
| 		Duration & Duration::operator/=(float val)
 | ||
| 		{
 | ||
| 			milliseconds_ = static_cast<int64_t>(milliseconds_ / val);
 | ||
| 			return (*this);
 | ||
| 		}
 | ||
| 
 | ||
| 		Duration & Duration::operator*=(double val)
 | ||
| 		{
 | ||
| 			milliseconds_ = static_cast<int64_t>(milliseconds_ * val);
 | ||
| 			return (*this);
 | ||
| 		}
 | ||
| 
 | ||
| 		Duration & Duration::operator*=(long double val)
 | ||
| 		{
 | ||
| 			milliseconds_ = static_cast<int64_t>(milliseconds_ * val);
 | ||
| 			return (*this);
 | ||
| 		}
 | ||
| 
 | ||
| 		Duration & Duration::operator/=(double val)
 | ||
| 		{
 | ||
| 			milliseconds_ = static_cast<int64_t>(milliseconds_ / val);
 | ||
| 			return (*this);
 | ||
| 		}
 | ||
| 
 | ||
| 		Duration & Duration::operator/=(long double val)
 | ||
| 		{
 | ||
| 			milliseconds_ = static_cast<int64_t>(milliseconds_ / val);
 | ||
| 			return (*this);
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration easy2d::time::operator*(int val, const Duration & dur)
 | ||
| 		{
 | ||
| 			return dur * val;
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration easy2d::time::operator*(unsigned long long val, const Duration & dur)
 | ||
| 		{
 | ||
| 			return dur / val;
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration easy2d::time::operator/(int val, const Duration & dur)
 | ||
| 		{
 | ||
| 			return dur / val;
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration easy2d::time::operator/(unsigned long long val, const Duration & dur)
 | ||
| 		{
 | ||
| 			return dur * val;
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration easy2d::time::operator*(float val, const Duration & dur)
 | ||
| 		{
 | ||
| 			return dur * val;
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration easy2d::time::operator/(float val, const Duration & dur)
 | ||
| 		{
 | ||
| 			return dur / val;
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration easy2d::time::operator*(double val, const Duration & dur)
 | ||
| 		{
 | ||
| 			return dur * val;
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration easy2d::time::operator/(double val, const Duration & dur)
 | ||
| 		{
 | ||
| 			return dur / val;
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration easy2d::time::operator*(long double val, const Duration & dur)
 | ||
| 		{
 | ||
| 			return dur * val;
 | ||
| 		}
 | ||
| 
 | ||
| 		const Duration easy2d::time::operator/(long double val, const Duration & dur)
 | ||
| 		{
 | ||
| 			return dur / val;
 | ||
| 		}
 | ||
| 
 | ||
| 		std::wostream & easy2d::time::operator<<(std::wostream & out, const Duration & dur)
 | ||
| 		{
 | ||
| 			return out << dur.ToString();
 | ||
| 		}
 | ||
| 
 | ||
| 		std::wistream & easy2d::time::operator>>(std::wistream & in, Duration & dur)
 | ||
| 		{
 | ||
| 			std::wstring str;
 | ||
| 			in >> str;
 | ||
| 			dur = time::ParseDuration(str);
 | ||
| 			return in;
 | ||
| 		}
 | ||
| 
 | ||
| 
 | ||
| 		//-------------------------------------------------------
 | ||
| 		// Functions
 | ||
| 		//-------------------------------------------------------
 | ||
| 
 | ||
| 		TimePoint easy2d::time::Now()
 | ||
| 		{
 | ||
| 			static LARGE_INTEGER freq = {};
 | ||
| 			if (freq.QuadPart == 0LL)
 | ||
| 			{
 | ||
| 				if (QueryPerformanceFrequency(&freq) == 0)
 | ||
| 					throw std::runtime_error("QueryPerformanceFrequency not supported: " + std::to_string(GetLastError()));
 | ||
| 			}
 | ||
| 
 | ||
| 			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 TimePoint{ Duration{ whole + part } };
 | ||
| 		}
 | ||
| 
 | ||
| 		Duration easy2d::time::ParseDuration(const std::wstring & str)
 | ||
| 		{
 | ||
| 			size_t len = str.length();
 | ||
| 			size_t pos = 0;
 | ||
| 			bool negative = false;
 | ||
| 			Duration d;
 | ||
| 
 | ||
| 			if (!std::regex_match(str, duration_regex))
 | ||
| 			{
 | ||
| 				logs::Errorln("time::ParseDuration failed, invalid duration");
 | ||
| 				return Duration();
 | ||
| 			}
 | ||
| 
 | ||
| 			if (str.empty() || str == L"0") { return d; }
 | ||
| 
 | ||
| 			// <20><><EFBFBD><EFBFBD>λ
 | ||
| 			if (str[0] == L'-' || str[0] == L'+')
 | ||
| 			{
 | ||
| 				negative = (str[0] == L'-');
 | ||
| 				pos++;
 | ||
| 			}
 | ||
| 
 | ||
| 			while (pos < len)
 | ||
| 			{
 | ||
| 				// <20><>ֵ
 | ||
| 				size_t i = pos;
 | ||
| 				for (; i < len; ++i)
 | ||
| 				{
 | ||
| 					wchar_t ch = str[i];
 | ||
| 					if (!(ch == L'.' || L'0' <= ch && ch <= L'9'))
 | ||
| 					{
 | ||
| 						break;
 | ||
| 					}
 | ||
| 				}
 | ||
| 
 | ||
| 				std::wstring num_str = str.substr(pos, i - pos);
 | ||
| 				pos = i;
 | ||
| 
 | ||
| 				if (num_str.empty() || num_str == L".")
 | ||
| 				{
 | ||
| 					logs::Errorln("time::ParseDuration failed, invalid duration");
 | ||
| 					return Duration();
 | ||
| 				}
 | ||
| 
 | ||
| 				// <20><>λ
 | ||
| 				for (; i < len; ++i)
 | ||
| 				{
 | ||
| 					wchar_t ch = str[i];
 | ||
| 					if (ch == L'.' || L'0' <= ch && ch <= L'9')
 | ||
| 					{
 | ||
| 						break;
 | ||
| 					}
 | ||
| 				}
 | ||
| 
 | ||
| 				std::wstring unit_str = str.substr(pos, i - pos);
 | ||
| 				pos = i;
 | ||
| 
 | ||
| 				if (unit_map.find(unit_str) == unit_map.end())
 | ||
| 				{
 | ||
| 					logs::Errorln("time::ParseDuration failed, invalid duration");
 | ||
| 					return Duration();
 | ||
| 				}
 | ||
| 
 | ||
| 				double num = std::stod(num_str);
 | ||
| 				Duration unit = unit_map.at(unit_str);
 | ||
| 				d += unit * num;
 | ||
| 			}
 | ||
| 
 | ||
| 			if (negative)
 | ||
| 			{
 | ||
| 				d = -d;
 | ||
| 			}
 | ||
| 			return d;
 | ||
| 		}
 | ||
| 	}
 | ||
| } |