/* Copyright (C) 2002 Kimmo Pekkola + few lsapi developers 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. */ #include "StdAfx.h" #include "Litestep.h" #include "Rainmeter.h" #include "DialogAbout.h" #include "System.h" extern CRainmeter* Rainmeter; static CRITICAL_SECTION g_CsLog = {0}; static CRITICAL_SECTION g_CsLogDelay = {0}; void InitalizeLitestep() { InitializeCriticalSection(&g_CsLog); InitializeCriticalSection(&g_CsLogDelay); } void FinalizeLitestep() { DeleteCriticalSection(&g_CsLog); DeleteCriticalSection(&g_CsLogDelay); } UINT GetUniqueID() { static UINT id = 0; return id++; } void RunCommand(HWND Owner, LPCTSTR szCommand, int nShowCmd, bool asAdmin) { // The stub implementation (some of this code is taken from lsapi.cpp) if (szCommand == NULL || *szCommand == 0) return; std::wstring args; std::wstring command = szCommand; size_t notwhite = command.find_first_not_of(L" \t\r\n"); command.erase(0, notwhite); if (command.empty()) return; size_t quotePos = command.find(L'"'); if (quotePos == 0) { size_t quotePos2 = command.find(L'"', quotePos + 1); if (quotePos2 != std::wstring::npos) { args.assign(command, quotePos2 + 1, command.length() - (quotePos2 + 1)); command.assign(command, quotePos + 1, quotePos2 - quotePos - 1); } else { command.erase(0, 1); } } else { size_t spacePos = command.find(L' '); if (spacePos != std::wstring::npos) { args.assign(command, spacePos + 1, command.length() - (spacePos + 1)); command.erase(spacePos); } } if (!command.empty()) { LPCWSTR szVerb = asAdmin ? L"runas" : L"open"; DWORD type = GetFileAttributes(command.c_str()); if (type & FILE_ATTRIBUTE_DIRECTORY && type != 0xFFFFFFFF) { ShellExecute(Owner, szVerb, command.c_str(), NULL, NULL, nShowCmd ? nShowCmd : SW_SHOWNORMAL); return; } std::wstring dir = CRainmeter::ExtractPath(command); SHELLEXECUTEINFO si = {sizeof(SHELLEXECUTEINFO)}; si.hwnd = Owner; si.lpVerb = szVerb; si.lpFile = command.c_str(); si.lpParameters = args.c_str(); si.lpDirectory = dir.c_str(); si.nShow = nShowCmd ? nShowCmd : SW_SHOWNORMAL; si.fMask = SEE_MASK_DOENVSUBST | SEE_MASK_FLAG_NO_UI; ShellExecuteEx(&si); } } std::string ConvertToAscii(LPCTSTR str) { std::string szAscii; if (str && *str) { int strLen = (int)wcslen(str); int bufLen = WideCharToMultiByte(CP_ACP, 0, str, strLen, NULL, 0, NULL, NULL); if (bufLen > 0) { szAscii.resize(bufLen); WideCharToMultiByte(CP_ACP, 0, str, strLen, &szAscii[0], bufLen, NULL, NULL); } } return szAscii; } std::wstring ConvertToWide(LPCSTR str) { std::wstring szWide; if (str && *str) { int strLen = (int)strlen(str); int bufLen = MultiByteToWideChar(CP_ACP, 0, str, strLen, NULL, 0); if (bufLen > 0) { szWide.resize(bufLen); MultiByteToWideChar(CP_ACP, 0, str, strLen, &szWide[0], bufLen); } } return szWide; } std::string ConvertToUTF8(LPCWSTR str) { std::string szAscii; if (str && *str) { int strLen = (int)wcslen(str); int bufLen = WideCharToMultiByte(CP_UTF8, 0, str, strLen, NULL, 0, NULL, NULL); if (bufLen > 0) { szAscii.resize(bufLen); WideCharToMultiByte(CP_UTF8, 0, str, strLen, &szAscii[0], bufLen, NULL, NULL); } } return szAscii; } std::wstring ConvertUTF8ToWide(LPCSTR str) { std::wstring szWide; if (str && *str) { int strLen = (int)strlen(str); int bufLen = MultiByteToWideChar(CP_UTF8, 0, str, strLen, NULL, 0); if (bufLen > 0) { szWide.resize(bufLen); MultiByteToWideChar(CP_UTF8, 0, str, strLen, &szWide[0], bufLen); } } return szWide; } void LogInternal(int nLevel, ULONGLONG elapsed, LPCTSTR pszMessage) { // Add timestamp WCHAR buffer[128]; size_t len = _snwprintf_s(buffer, _TRUNCATE, L"%02llu:%02llu:%02llu.%03llu", elapsed / (1000 * 60 * 60), (elapsed / (1000 * 60)) % 60, (elapsed / 1000) % 60, elapsed % 1000); Rainmeter->AddAboutLogInfo(nLevel, buffer, pszMessage); #ifndef _DEBUG if (!Rainmeter->GetLogging()) return; #endif std::wstring message; switch (nLevel) { case LOG_ERROR: message = L"ERRO"; break; case LOG_WARNING: message = L"WARN"; break; case LOG_NOTICE: message = L"NOTE"; break; case LOG_DEBUG: message = L"DBUG"; break; } message += L" ("; message.append(buffer, len); message += L") "; message += pszMessage; message += L'\n'; #ifdef _DEBUG _RPT0(_CRT_WARN, ConvertToAscii(message.c_str()).c_str()); if (!Rainmeter->GetLogging()) return; #endif const WCHAR* logFile = Rainmeter->GetLogFile().c_str(); if (_waccess(logFile, 0) == -1) { // Disable logging if the file was deleted manually Rainmeter->StopLogging(); } else { FILE* file = _wfopen(logFile, L"a+, ccs=UTF-8"); if (file) { fputws(message.c_str(), file); fclose(file); } } } void Log(int nLevel, const WCHAR* message) { struct DELAYED_LOG_INFO { int level; ULONGLONG elapsed; std::wstring message; }; static std::list c_LogDelay; static ULONGLONG startTime = CSystem::GetTickCount64(); ULONGLONG elapsed = CSystem::GetTickCount64() - startTime; if (TryEnterCriticalSection(&g_CsLog)) { // Log the queued messages first EnterCriticalSection(&g_CsLogDelay); while (!c_LogDelay.empty()) { DELAYED_LOG_INFO& logInfo = c_LogDelay.front(); LogInternal(logInfo.level, logInfo.elapsed, logInfo.message.c_str()); c_LogDelay.erase(c_LogDelay.begin()); } LeaveCriticalSection(&g_CsLogDelay); // Log the message LogInternal(nLevel, elapsed, message); LeaveCriticalSection(&g_CsLog); } else { // Queue the message EnterCriticalSection(&g_CsLogDelay); DELAYED_LOG_INFO logInfo = {nLevel, elapsed, message}; c_LogDelay.push_back(logInfo); LeaveCriticalSection(&g_CsLogDelay); } } void LogWithArgs(int nLevel, const WCHAR* format, ...) { WCHAR* buffer = new WCHAR[4096]; va_list args; va_start( args, format ); _invalid_parameter_handler oldHandler = _set_invalid_parameter_handler(RmNullCRTInvalidParameterHandler); _CrtSetReportMode(_CRT_ASSERT, 0); errno = 0; _vsnwprintf_s(buffer, 4096, _TRUNCATE, format, args); if (errno != 0) { nLevel = LOG_ERROR; _snwprintf_s(buffer, 4096, _TRUNCATE, L"LogWithArgs internal error: %s", format); } _set_invalid_parameter_handler(oldHandler); Log(nLevel, buffer); va_end(args); delete [] buffer; } void LogError(CError& error) { Log(LOG_ERROR, error.GetString().c_str()); CDialogAbout::ShowAboutLog(); } WCHAR* GetString(UINT id) { LPWSTR pData; int len = LoadString(Rainmeter->GetResourceInstance(), id, (LPWSTR)&pData, 0); return len ? pData : L""; } std::wstring GetFormattedString(UINT id, ...) { LPWSTR pBuffer = NULL; va_list args = NULL; va_start(args, id); DWORD len = FormatMessage(FORMAT_MESSAGE_FROM_STRING | FORMAT_MESSAGE_ALLOCATE_BUFFER, GetString(id), 0, 0, (LPWSTR)&pBuffer, 0, &args); va_end(args); std::wstring tmpSz(len ? pBuffer : L""); if (pBuffer) LocalFree(pBuffer); return tmpSz; } void RmNullCRTInvalidParameterHandler(const wchar_t* expression, const wchar_t* function, const wchar_t* file, unsigned int line, uintptr_t pReserved) { // Do nothing. }