2020-09-21 17:17:34 +00:00
|
|
|
// SPDX-License-Identifier: MIT
|
2020-06-23 18:07:00 +00:00
|
|
|
//
|
2020-09-21 17:17:34 +00:00
|
|
|
// EmulationStation Desktop Edition
|
2020-06-23 18:07:00 +00:00
|
|
|
// TimeUtil.cpp
|
|
|
|
//
|
|
|
|
// Low-level date and time functions.
|
|
|
|
// Set and get time, format to string formats, count days and months etc.
|
|
|
|
//
|
|
|
|
|
2017-11-22 21:01:12 +00:00
|
|
|
#include "utils/TimeUtil.h"
|
|
|
|
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
namespace Utils
|
|
|
|
{
|
2020-06-23 18:07:00 +00:00
|
|
|
namespace Time
|
|
|
|
{
|
|
|
|
DateTime::DateTime()
|
|
|
|
{
|
|
|
|
mTime = 0;
|
|
|
|
mTimeStruct = { 0, 0, 0, 1, 0, 0, 0, 0, -1 };
|
|
|
|
mIsoString = "00000000T000000";
|
|
|
|
}
|
|
|
|
|
|
|
|
DateTime::DateTime(const time_t& _time)
|
|
|
|
{
|
|
|
|
setTime(_time);
|
|
|
|
}
|
|
|
|
|
|
|
|
DateTime::DateTime(const tm& _timeStruct)
|
|
|
|
{
|
|
|
|
setTimeStruct(_timeStruct);
|
|
|
|
}
|
|
|
|
|
|
|
|
DateTime::DateTime(const std::string& _isoString)
|
|
|
|
{
|
|
|
|
setIsoString(_isoString);
|
|
|
|
}
|
|
|
|
|
|
|
|
DateTime::~DateTime()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void DateTime::setTime(const time_t& _time)
|
|
|
|
{
|
|
|
|
mTime = (_time < 0) ? 0 : _time;
|
2020-12-29 12:44:13 +00:00
|
|
|
#if defined(_WIN64)
|
|
|
|
localtime_s(&mTimeStruct, &mTime);
|
|
|
|
#else
|
|
|
|
localtime_r(&mTime, &mTimeStruct);
|
|
|
|
#endif
|
2020-06-23 18:07:00 +00:00
|
|
|
mIsoString = timeToString(mTime);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DateTime::setTimeStruct(const tm& _timeStruct)
|
|
|
|
{
|
|
|
|
setTime(mktime((tm*)&_timeStruct));
|
|
|
|
}
|
|
|
|
|
|
|
|
void DateTime::setIsoString(const std::string& _isoString)
|
|
|
|
{
|
|
|
|
setTime(stringToTime(_isoString));
|
|
|
|
}
|
|
|
|
|
|
|
|
Duration::Duration(const time_t& _time)
|
|
|
|
{
|
2020-11-17 22:06:54 +00:00
|
|
|
mTotalSeconds = static_cast<unsigned int>(_time);
|
2020-06-23 18:07:00 +00:00
|
|
|
mDays = (mTotalSeconds - (mTotalSeconds % (60*60*24))) / (60*60*24);
|
|
|
|
mHours = ((mTotalSeconds % (60*60*24)) - (mTotalSeconds % (60*60))) / (60*60);
|
|
|
|
mMinutes = ((mTotalSeconds % (60*60)) - (mTotalSeconds % (60))) / 60;
|
|
|
|
mSeconds = mTotalSeconds % 60;
|
|
|
|
}
|
|
|
|
|
|
|
|
Duration::~Duration()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
time_t now()
|
|
|
|
{
|
|
|
|
time_t time;
|
|
|
|
::time(&time);
|
|
|
|
return time;
|
|
|
|
}
|
|
|
|
|
|
|
|
time_t stringToTime(const std::string& _string, const std::string& _format)
|
|
|
|
{
|
|
|
|
const char* s = _string.c_str();
|
|
|
|
const char* f = _format.c_str();
|
|
|
|
tm timeStruct = { 0, 0, 0, 1, 0, 0, 0, 0, -1 };
|
|
|
|
size_t parsedChars = 0;
|
|
|
|
|
2020-08-02 13:56:32 +00:00
|
|
|
if (_string == "19700101T010000")
|
2020-06-23 18:07:00 +00:00
|
|
|
return mktime(&timeStruct);
|
|
|
|
|
|
|
|
while (*f && (parsedChars < _string.length())) {
|
|
|
|
if (*f == '%') {
|
2021-01-25 17:07:11 +00:00
|
|
|
f++;
|
2020-06-23 18:07:00 +00:00
|
|
|
|
|
|
|
switch (*f++) {
|
|
|
|
// The year [1970,xxxx]
|
|
|
|
case 'Y': {
|
|
|
|
if ((parsedChars + 4) <= _string.length()) {
|
|
|
|
timeStruct.tm_year = (*s++ - '0') * 1000;
|
|
|
|
timeStruct.tm_year += (*s++ - '0') * 100;
|
|
|
|
timeStruct.tm_year += (*s++ - '0') * 10;
|
|
|
|
timeStruct.tm_year += (*s++ - '0');
|
|
|
|
if (timeStruct.tm_year >= 1900)
|
|
|
|
timeStruct.tm_year -= 1900;
|
|
|
|
}
|
|
|
|
parsedChars += 4;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// The month number [01,12]
|
|
|
|
case 'm': {
|
|
|
|
if ((parsedChars + 2) <= _string.length()) {
|
|
|
|
timeStruct.tm_mon = (*s++ - '0') * 10;
|
|
|
|
timeStruct.tm_mon += (*s++ - '0');
|
|
|
|
if (timeStruct.tm_mon >= 1)
|
|
|
|
timeStruct.tm_mon -= 1;
|
|
|
|
}
|
|
|
|
parsedChars += 2;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// The day of the month [01,31]
|
|
|
|
case 'd': {
|
|
|
|
if ((parsedChars + 2) <= _string.length()) {
|
|
|
|
timeStruct.tm_mday = (*s++ - '0') * 10;
|
|
|
|
timeStruct.tm_mday += (*s++ - '0');
|
|
|
|
}
|
|
|
|
parsedChars += 2;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// The hour (24-hour clock) [00,23]
|
|
|
|
case 'H': {
|
|
|
|
if ((parsedChars + 2) <= _string.length()) {
|
|
|
|
timeStruct.tm_hour = (*s++ - '0') * 10;
|
|
|
|
timeStruct.tm_hour += (*s++ - '0');
|
|
|
|
}
|
|
|
|
parsedChars += 2;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// The minute [00,59]
|
|
|
|
case 'M': {
|
|
|
|
if ((parsedChars + 2) <= _string.length()) {
|
|
|
|
timeStruct.tm_min = (*s++ - '0') * 10;
|
|
|
|
timeStruct.tm_min += (*s++ - '0');
|
|
|
|
}
|
|
|
|
parsedChars += 2;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// The second [00,59]
|
|
|
|
case 'S': {
|
|
|
|
if ((parsedChars + 2) <= _string.length()) {
|
|
|
|
timeStruct.tm_sec = (*s++ - '0') * 10;
|
|
|
|
timeStruct.tm_sec += (*s++ - '0');
|
|
|
|
}
|
|
|
|
parsedChars += 2;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2021-01-25 17:07:11 +00:00
|
|
|
s++;
|
|
|
|
f++;
|
2020-06-23 18:07:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return mktime(&timeStruct);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string timeToString(const time_t& _time, const std::string& _format)
|
|
|
|
{
|
|
|
|
const char* f = _format.c_str();
|
2020-12-29 12:44:13 +00:00
|
|
|
tm timeStruct;
|
|
|
|
#if defined(_WIN64)
|
|
|
|
localtime_s(&timeStruct, &_time);
|
|
|
|
#else
|
|
|
|
localtime_r(&_time, &timeStruct);
|
|
|
|
#endif
|
2020-06-23 18:07:00 +00:00
|
|
|
char buf[256] = { '\0' };
|
|
|
|
char* s = buf;
|
|
|
|
|
|
|
|
while (*f) {
|
|
|
|
if (*f == '%') {
|
2021-01-25 17:07:11 +00:00
|
|
|
f++;
|
2020-06-23 18:07:00 +00:00
|
|
|
|
|
|
|
switch (*f++) {
|
|
|
|
// The year, including the century (1900)
|
|
|
|
case 'Y': {
|
|
|
|
const int year = timeStruct.tm_year + 1900;
|
2020-11-17 22:06:54 +00:00
|
|
|
*s++ = static_cast<char>((year - (year % 1000)) / 1000) + '0';
|
|
|
|
*s++ = static_cast<char>(((year % 1000) - (year % 100)) / 100) + '0';
|
|
|
|
*s++ = static_cast<char>(((year % 100) - (year % 10)) / 10) + '0';
|
|
|
|
*s++ = static_cast<char>(year % 10) + '0';
|
2020-06-23 18:07:00 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// The month number [00,11]
|
|
|
|
case 'm': {
|
|
|
|
const int mon = timeStruct.tm_mon + 1;
|
2020-11-17 22:06:54 +00:00
|
|
|
*s++ = static_cast<char>(mon / 10) + '0';
|
|
|
|
*s++ = static_cast<char>(mon % 10) + '0';
|
2020-06-23 18:07:00 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// The day of the month [01,31]
|
|
|
|
case 'd': {
|
2020-11-17 22:06:54 +00:00
|
|
|
*s++ = static_cast<char>(timeStruct.tm_mday / 10) + '0';
|
|
|
|
*s++ = static_cast<char>(timeStruct.tm_mday % 10) + '0';
|
2020-06-23 18:07:00 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// The hour (24-hour clock) [00,23]
|
|
|
|
case 'H': {
|
2020-11-17 22:06:54 +00:00
|
|
|
*s++ = static_cast<char>(timeStruct.tm_hour / 10) + '0';
|
|
|
|
*s++ = static_cast<char>(timeStruct.tm_hour % 10) + '0';
|
2020-06-23 18:07:00 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// The minute [00,59]
|
|
|
|
case 'M': {
|
2020-11-17 22:06:54 +00:00
|
|
|
*s++ = static_cast<char>(timeStruct.tm_min / 10) + '0';
|
|
|
|
*s++ = static_cast<char>(timeStruct.tm_min % 10) + '0';
|
2020-06-23 18:07:00 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// The second [00,59]
|
|
|
|
case 'S': {
|
2020-11-17 22:06:54 +00:00
|
|
|
*s++ = static_cast<char>(timeStruct.tm_sec / 10) + '0';
|
|
|
|
*s++ = static_cast<char>(timeStruct.tm_sec % 10) + '0';
|
2020-06-23 18:07:00 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*s++ = *f++;
|
|
|
|
}
|
|
|
|
*s = '\0';
|
|
|
|
}
|
|
|
|
return std::string(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
int daysInMonth(const int _year, const int _month)
|
|
|
|
{
|
|
|
|
tm timeStruct = { 0, 0, 0, 0, _month, _year - 1900, 0, 0, -1 };
|
|
|
|
mktime(&timeStruct);
|
|
|
|
|
|
|
|
return timeStruct.tm_mday;
|
|
|
|
}
|
|
|
|
|
|
|
|
int daysInYear(const int _year)
|
|
|
|
{
|
|
|
|
tm timeStruct = { 0, 0, 0, 0, 0, _year - 1900 + 1, 0, 0, -1 };
|
|
|
|
mktime(&timeStruct);
|
|
|
|
|
|
|
|
return timeStruct.tm_yday + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // Time::
|
2017-11-22 21:01:12 +00:00
|
|
|
|
|
|
|
} // Utils::
|