rainmeter-studio/Library/MeasureTime.cpp

283 lines
7.0 KiB
C++
Raw Normal View History

2009-02-10 18:37:48 +00:00
/*
Copyright (C) 2001 Kimmo Pekkola
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
2009-02-10 18:37:48 +00:00
*/
#include "StdAfx.h"
2009-02-10 18:37:48 +00:00
#include "MeasureTime.h"
#include "Rainmeter.h"
2009-02-10 18:37:48 +00:00
int GetYearDay(int year, int month, int day)
{
2012-08-29 12:02:00 -07:00
static const int dates[] = { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334 };
int yearDay = dates[month - 1] + day;
2009-02-10 18:37:48 +00:00
2012-08-29 12:02:00 -07:00
if (month > 2 && ((((year % 4) == 0) && ((year % 100) != 0)) || (year % 400) == 0))
2009-02-10 18:37:48 +00:00
{
2010-03-30 22:37:05 +00:00
++yearDay;
2009-02-10 18:37:48 +00:00
}
return yearDay - 1;
}
/*
** The constructor
**
*/
2011-02-15 16:26:54 +00:00
CMeasureTime::CMeasureTime(CMeterWindow* meterWindow, const WCHAR* name) : CMeasure(meterWindow, name),
m_DeltaTime(),
m_Time(),
m_TimeStamp(-1)
2009-02-10 18:37:48 +00:00
{
/* Set time zone from TZ environment variable. If TZ is not set,
2011-03-29 19:21:57 +00:00
* the operating system is queried to obtain the default value
* for the variable.
*/
2010-09-21 22:47:53 +00:00
_tzset();
2009-02-10 18:37:48 +00:00
}
/*
** The destructor
**
*/
CMeasureTime::~CMeasureTime()
{
}
/*
** Converts given time to string.
** This function is a wrapper function for wcsftime.
**
*/
void CMeasureTime::TimeToString(WCHAR* buf, size_t bufLen, const WCHAR* format, const struct tm* time)
{
if (bufLen > 0)
{
_invalid_parameter_handler oldHandler = _set_invalid_parameter_handler(RmNullCRTInvalidParameterHandler);
_CrtSetReportMode(_CRT_ASSERT, 0);
errno = 0;
wcsftime(buf, bufLen, format, time);
if (errno == EINVAL)
{
LogWithArgs(LOG_ERROR, L"Time: \"Format=%s\" invalid in [%s]", format, m_Name.c_str());
buf[0] = 0;
}
_set_invalid_parameter_handler(oldHandler);
}
}
void CMeasureTime::FillCurrentTime()
2009-02-10 18:37:48 +00:00
{
if (m_TimeStamp < 0.0)
{
FILETIME ftUTCTime;
GetSystemTimeAsFileTime(&ftUTCTime);
2009-02-10 18:37:48 +00:00
// Modify the ltime to match the current timezone
// This way we can use the value also for the clock
m_Time.HighPart = ftUTCTime.dwHighDateTime;
m_Time.LowPart = ftUTCTime.dwLowDateTime;
2009-02-10 18:37:48 +00:00
m_Time.QuadPart += m_DeltaTime.QuadPart;
}
else
{
m_Time.QuadPart = (LONGLONG)(m_TimeStamp * 10000000);
}
}
/*
** Updates the current time
**
*/
void CMeasureTime::UpdateValue()
{
FillCurrentTime();
2009-02-10 18:37:48 +00:00
if (!m_Format.empty())
2009-02-10 18:37:48 +00:00
{
// If there is some date format, parse the value from it instead
WCHAR* tmpSz = new WCHAR[MAX_LINE_LENGTH];
2009-02-10 18:37:48 +00:00
SYSTEMTIME sysToday;
FILETIME ftToday;
tmpSz[0] = 0;
2009-02-10 18:37:48 +00:00
ftToday.dwHighDateTime = m_Time.HighPart;
ftToday.dwLowDateTime = m_Time.LowPart;
FileTimeToSystemTime(&ftToday, &sysToday);
2011-11-16 16:47:20 +00:00
const WCHAR* format = m_Format.c_str();
if (_wcsicmp(L"locale-time", format) == 0)
2009-02-10 18:37:48 +00:00
{
GetTimeFormat(LOCALE_USER_DEFAULT, 0, &sysToday, NULL, tmpSz, MAX_LINE_LENGTH);
}
2011-11-16 16:47:20 +00:00
else if (_wcsicmp(L"locale-date", format) == 0)
2009-02-10 18:37:48 +00:00
{
GetDateFormat(LOCALE_USER_DEFAULT, 0, &sysToday, NULL, tmpSz, MAX_LINE_LENGTH);
}
else
{
struct tm today;
today.tm_isdst = 0;
today.tm_hour = sysToday.wHour;
today.tm_mday = sysToday.wDay;
today.tm_min = sysToday.wMinute;
today.tm_mon = sysToday.wMonth - 1;
today.tm_sec = sysToday.wSecond;
today.tm_wday = sysToday.wDayOfWeek;
today.tm_yday = GetYearDay(sysToday.wYear, sysToday.wMonth, sysToday.wDay);
today.tm_year = sysToday.wYear - 1900;
2011-11-16 16:47:20 +00:00
TimeToString(tmpSz, MAX_LINE_LENGTH, format, &today);
2009-02-10 18:37:48 +00:00
}
m_Value = wcstod(tmpSz, NULL);
delete [] tmpSz;
2009-02-10 18:37:48 +00:00
}
2011-01-24 10:15:05 +00:00
else
{
m_Value = (double)(m_Time.QuadPart / 10000000);
}
2009-02-10 18:37:48 +00:00
}
/*
** Returns the time as string.
**
*/
const WCHAR* CMeasureTime::GetStringValue()
2009-02-10 18:37:48 +00:00
{
static WCHAR tmpSz[MAX_LINE_LENGTH];
struct tm today;
2011-03-29 19:21:57 +00:00
tmpSz[0] = 0;
2009-02-10 18:37:48 +00:00
SYSTEMTIME sysToday;
FILETIME ftToday;
ftToday.dwHighDateTime = m_Time.HighPart;
ftToday.dwLowDateTime = m_Time.LowPart;
FileTimeToSystemTime(&ftToday, &sysToday);
today.tm_isdst = 0;
today.tm_hour = sysToday.wHour;
today.tm_mday = sysToday.wDay;
today.tm_min = sysToday.wMinute;
today.tm_mon = sysToday.wMonth - 1;
today.tm_sec = sysToday.wSecond;
today.tm_wday = sysToday.wDayOfWeek;
today.tm_yday = GetYearDay(sysToday.wYear, sysToday.wMonth, sysToday.wDay);
today.tm_year = sysToday.wYear - 1900;
// Create the string
if (!m_Format.empty())
2009-02-10 18:37:48 +00:00
{
2011-11-16 16:47:20 +00:00
const WCHAR* format = m_Format.c_str();
if (_wcsicmp(L"locale-time", format) == 0)
2009-02-10 18:37:48 +00:00
{
GetTimeFormat(LOCALE_USER_DEFAULT, 0, &sysToday, NULL, tmpSz, MAX_LINE_LENGTH);
}
2011-11-16 16:47:20 +00:00
else if (_wcsicmp(L"locale-date", format) == 0)
2009-02-10 18:37:48 +00:00
{
GetDateFormat(LOCALE_USER_DEFAULT, 0, &sysToday, NULL, tmpSz, MAX_LINE_LENGTH);
}
else
{
2011-11-16 16:47:20 +00:00
TimeToString(tmpSz, MAX_LINE_LENGTH, format, &today);
2009-02-10 18:37:48 +00:00
}
}
else
{
TimeToString(tmpSz, MAX_LINE_LENGTH, L"%H:%M:%S", &today);
2009-02-10 18:37:48 +00:00
}
return CheckSubstitute(tmpSz);
2009-02-10 18:37:48 +00:00
}
/*
** Read the options specified in the ini file.
2009-02-10 18:37:48 +00:00
**
*/
2012-05-30 21:53:44 +03:00
void CMeasureTime::ReadOptions(CConfigParser& parser, const WCHAR* section)
2009-02-10 18:37:48 +00:00
{
2012-05-30 21:53:44 +03:00
CMeasure::ReadOptions(parser, section);
2009-02-10 18:37:48 +00:00
m_Format = parser.ReadString(section, L"Format", L"");
m_TimeStamp = parser.ReadFloat(section, L"TimeStamp", -1);
2012-08-29 12:02:00 -07:00
if (m_TimeStamp < 0.0)
2009-02-10 18:37:48 +00:00
{
2012-08-29 12:02:00 -07:00
const WCHAR* timezone = parser.ReadString(section, L"TimeZone", L"local").c_str();
if (_wcsicmp(L"local", timezone) == 0)
{
SYSTEMTIME sysLocalTime, sysUTCTime;
GetLocalTime(&sysLocalTime);
GetSystemTime(&sysUTCTime);
2009-02-10 18:37:48 +00:00
2012-08-29 12:02:00 -07:00
FILETIME ftLocalTime, ftUTCTime;
SystemTimeToFileTime(&sysLocalTime, &ftLocalTime);
SystemTimeToFileTime(&sysUTCTime, &ftUTCTime);
2009-02-10 18:37:48 +00:00
2012-08-29 12:02:00 -07:00
LARGE_INTEGER largeInt1, largeInt2;
largeInt1.HighPart = ftLocalTime.dwHighDateTime;
largeInt1.LowPart = ftLocalTime.dwLowDateTime;
largeInt2.HighPart = ftUTCTime.dwHighDateTime;
largeInt2.LowPart = ftUTCTime.dwLowDateTime;
2009-02-10 18:37:48 +00:00
2012-08-29 12:02:00 -07:00
m_DeltaTime.QuadPart = largeInt1.QuadPart - largeInt2.QuadPart;
2009-02-10 18:37:48 +00:00
}
else
{
double zone = parser.ParseDouble(timezone, 0.0);
2012-08-29 12:02:00 -07:00
bool dst = 1 == parser.ReadInt(section, L"DaylightSavingTime", 1);
struct tm* today;
time_t now;
time(&now);
today = localtime(&now);
if (dst && today->tm_isdst)
{
// Add DST
TIME_ZONE_INFORMATION tzi;
GetTimeZoneInformation(&tzi);
m_DeltaTime.QuadPart = (LONGLONG)((zone * 3600) - tzi.DaylightBias * 60) * 10000000;
}
else
{
m_DeltaTime.QuadPart = (LONGLONG)(zone * 3600) * 10000000;
}
2009-02-10 18:37:48 +00:00
}
}
2012-08-29 12:02:00 -07:00
else
{
m_DeltaTime.QuadPart = 0;
}
if (!m_Initialized)
{
// Initialize m_Time to avoid causing EINVAL in TimeToString() until calling UpdateValue()
FillCurrentTime();
}
2009-02-10 18:37:48 +00:00
}