408 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			C++
		
	
	
	
		
		
			
		
	
	
			408 lines
		
	
	
		
			9.1 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 <regex>
 | |||
|  | 
 | |||
|  | namespace easy2d | |||
|  | { | |||
|  | 	namespace time | |||
|  | 	{ | |||
|  | 		using namespace std::chrono; | |||
|  | 
 | |||
|  | 		//-------------------------------------------------------
 | |||
|  | 		// TimePoint
 | |||
|  | 		//-------------------------------------------------------
 | |||
|  | 
 | |||
|  | 		TimePoint::TimePoint() | |||
|  | 		{ | |||
|  | 		} | |||
|  | 
 | |||
|  | 		TimePoint::TimePoint(std::chrono::steady_clock::time_point time) | |||
|  | 			: time_(time) | |||
|  | 		{ | |||
|  | 		} | |||
|  | 
 | |||
|  | 		TimePoint::TimePoint(const TimePoint & other) | |||
|  | 			: time_(other.time_) | |||
|  | 		{ | |||
|  | 		} | |||
|  | 
 | |||
|  | 		TimePoint::TimePoint(TimePoint && other) | |||
|  | 			: time_(std::move(other.time_)) | |||
|  | 		{ | |||
|  | 		} | |||
|  | 
 | |||
|  | 		time_t TimePoint::GetTimeStamp() const | |||
|  | 		{ | |||
|  | 			auto& duration = time_point_cast<milliseconds>(time_).time_since_epoch(); | |||
|  | 			return static_cast<time_t>(duration.count()); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		bool TimePoint::IsZero() const | |||
|  | 		{ | |||
|  | 			return time_.time_since_epoch().count() == 0LL; | |||
|  | 		} | |||
|  | 
 | |||
|  | 		TimePoint TimePoint::operator+(const Duration & other) const | |||
|  | 		{ | |||
|  | 			return TimePoint(time_ + milliseconds(other.Milliseconds())); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		TimePoint TimePoint::operator-(const Duration & other) const | |||
|  | 		{ | |||
|  | 			return TimePoint(time_ - milliseconds(other.Milliseconds())); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		TimePoint & TimePoint::operator+=(const Duration & other) | |||
|  | 		{ | |||
|  | 			time_ += milliseconds(other.Milliseconds()); | |||
|  | 			return (*this); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		TimePoint & TimePoint::operator-=(const Duration &other) | |||
|  | 		{ | |||
|  | 			time_ -= milliseconds(other.Milliseconds()); | |||
|  | 			return (*this); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration TimePoint::operator-(const TimePoint & other) const | |||
|  | 		{ | |||
|  | 			auto ms = duration_cast<milliseconds>(time_ - other.time_).count(); | |||
|  | 			return Duration(static_cast<int>(ms)); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		TimePoint& TimePoint::operator=(const TimePoint & other) E2D_NOEXCEPT | |||
|  | 		{ | |||
|  | 			if (this == &other) | |||
|  | 				return *this; | |||
|  | 
 | |||
|  | 			time_ = other.time_; | |||
|  | 			return *this; | |||
|  | 		} | |||
|  | 
 | |||
|  | 		TimePoint& TimePoint::operator=(TimePoint && other) E2D_NOEXCEPT | |||
|  | 		{ | |||
|  | 			if (this == &other) | |||
|  | 				return *this; | |||
|  | 
 | |||
|  | 			time_ = std::move(other.time_); | |||
|  | 			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(L"[-+]?([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(int milliseconds) | |||
|  | 			: milliseconds_(milliseconds) | |||
|  | 		{ | |||
|  | 		} | |||
|  | 
 | |||
|  | 		int Duration::Milliseconds() const | |||
|  | 		{ | |||
|  | 			return 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); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		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_; | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration Duration::operator+(const Duration & other) const | |||
|  | 		{ | |||
|  | 			return Duration(milliseconds_ + other.milliseconds_); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration Duration::operator-(const Duration & other) const | |||
|  | 		{ | |||
|  | 			return Duration(milliseconds_ - other.milliseconds_); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration Duration::operator-() const | |||
|  | 		{ | |||
|  | 			return Duration(-milliseconds_); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration Duration::operator*(int value) const | |||
|  | 		{ | |||
|  | 			return Duration(milliseconds_ * value); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration Duration::operator/(int value) const | |||
|  | 		{ | |||
|  | 			return Duration(milliseconds_ / value); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration Duration::operator*(float value) const | |||
|  | 		{ | |||
|  | 			return Duration(static_cast<int>(milliseconds_ * value)); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration Duration::operator/(float value) const | |||
|  | 		{ | |||
|  | 			return Duration(static_cast<int>(milliseconds_ / value)); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration Duration::operator*(double value) const | |||
|  | 		{ | |||
|  | 			return Duration(static_cast<int>(milliseconds_ * value)); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration Duration::operator/(double value) const | |||
|  | 		{ | |||
|  | 			return Duration(static_cast<int>(milliseconds_ / value)); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		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 value) | |||
|  | 		{ | |||
|  | 			milliseconds_ *= value; | |||
|  | 			return (*this); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration & Duration::operator/=(int value) | |||
|  | 		{ | |||
|  | 			milliseconds_ /= value; | |||
|  | 			return (*this); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration & Duration::operator*=(float value) | |||
|  | 		{ | |||
|  | 			milliseconds_ = static_cast<int>(milliseconds_ * value); | |||
|  | 			return (*this); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration & Duration::operator/=(float value) | |||
|  | 		{ | |||
|  | 			milliseconds_ = static_cast<int>(milliseconds_ / value); | |||
|  | 			return (*this); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration & Duration::operator*=(double value) | |||
|  | 		{ | |||
|  | 			milliseconds_ = static_cast<int>(milliseconds_ * value); | |||
|  | 			return (*this); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration & Duration::operator/=(double value) | |||
|  | 		{ | |||
|  | 			milliseconds_ = static_cast<int>(milliseconds_ / value); | |||
|  | 			return (*this); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration operator*(int value, const Duration & dur) | |||
|  | 		{ | |||
|  | 			return dur * value; | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration operator/(int value, const Duration & dur) | |||
|  | 		{ | |||
|  | 			return dur / value; | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration operator*(float value, const Duration & dur) | |||
|  | 		{ | |||
|  | 			return dur * value; | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration operator/(float value, const Duration & dur) | |||
|  | 		{ | |||
|  | 			return dur / value; | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration operator*(double value, const Duration & dur) | |||
|  | 		{ | |||
|  | 			return dur * value; | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration operator/(double value, const Duration & dur) | |||
|  | 		{ | |||
|  | 			return dur / value; | |||
|  | 		} | |||
|  | 
 | |||
|  | 
 | |||
|  | 		//-------------------------------------------------------
 | |||
|  | 		// Functions
 | |||
|  | 		//-------------------------------------------------------
 | |||
|  | 
 | |||
|  | 		TimePoint Now() | |||
|  | 		{ | |||
|  | 			return TimePoint(steady_clock::now()); | |||
|  | 		} | |||
|  | 
 | |||
|  | 		Duration 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)) | |||
|  | 			{ | |||
|  | 				E2D_WARNING("Time::Duration::Parse: 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".") | |||
|  | 				{ | |||
|  | 					E2D_WARNING("Time::Duration::Parse: 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()) | |||
|  | 				{ | |||
|  | 					E2D_WARNING("Time::Duration::Parse: 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; | |||
|  | 		} | |||
|  | 	} | |||
|  | } |