rainmeter-studio/Library/Application.cpp

357 lines
8.5 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#define _CRTDBG_MAP_ALLOC
#include "StdAfx.h"
2009-02-10 18:37:48 +00:00
#include "resource.h"
#include "Rainmeter.h"
#include "TrayWindow.h"
2009-02-10 18:37:48 +00:00
#if defined _M_IX86
#pragma comment(linker,"/manifestdependency:\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='x86' publicKeyToken='6595b64144ccf1df' language='*'\"")
#elif defined _M_IA64
#pragma comment(linker,"/manifestdependency:\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='ia64' publicKeyToken='6595b64144ccf1df' language='*'\"")
#elif defined _M_X64
#pragma comment(linker,"/manifestdependency:\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='amd64' publicKeyToken='6595b64144ccf1df' language='*'\"")
#else
#pragma comment(linker,"/manifestdependency:\"type='win32' name='Microsoft.Windows.Common-Controls' version='6.0.0.0' processorArchitecture='*' publicKeyToken='6595b64144ccf1df' language='*'\"")
#endif
extern CRainmeter* Rainmeter;
void Bang(const WCHAR* command);
2009-02-10 18:37:48 +00:00
BOOL InitApplication(HINSTANCE hInstance, const WCHAR* WinClass);
HWND InitInstance(HINSTANCE hInstance, const WCHAR* WinClass, const WCHAR* WinName);
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam);
BOOL IsRunning(HANDLE* hMutex);
2009-02-10 18:37:48 +00:00
const WCHAR* WinClass = L"DummyRainWClass";
const WCHAR* WinName = L"Rainmeter control window";
2009-02-10 18:37:48 +00:00
2010-12-11 16:30:49 +00:00
enum RetValue
{
RetSuccess = 0,
RetError = 1
};
2011-03-29 19:21:57 +00:00
/*
2009-02-10 18:37:48 +00:00
** WinMain
**
** The Main-function
**
*/
2009-03-21 09:35:03 +00:00
int APIENTRY wWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPWSTR lpCmdLine, int nCmdShow)
2009-02-10 18:37:48 +00:00
{
HANDLE hMutex = NULL;
2009-02-10 18:37:48 +00:00
MSG msg;
BOOL bRet;
2009-02-10 18:37:48 +00:00
HWND hWnd;
_CrtSetDbgFlag(_CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF);
// _CrtSetBreakAlloc(5055);
// Avoid loading a dll from current directory
2011-02-20 23:03:15 +00:00
SetDllDirectory(L"");
if (lpCmdLine && lpCmdLine[0] == L'!')
{
// It's a !bang
Bang(lpCmdLine);
2010-12-11 16:30:49 +00:00
return RetSuccess;
}
// Check whether Rainmeter.exe is already running
if (IsRunning(&hMutex))
{
//MessageBox(NULL, L"Rainmeter.exe is already running.", L"Rainmeter", MB_ICONWARNING | MB_TOPMOST | MB_OK);
2010-12-11 16:30:49 +00:00
return RetSuccess;
}
2011-03-29 19:21:57 +00:00
if (!hPrevInstance)
2009-02-10 18:37:48 +00:00
{
2010-12-11 16:30:49 +00:00
if (!InitApplication(hInstance, WinClass)) return RetError;
2009-02-10 18:37:48 +00:00
}
hWnd = InitInstance(hInstance, WinClass, WinName);
2011-03-29 19:21:57 +00:00
if (!hWnd) return RetError;
2009-02-10 18:37:48 +00:00
// Remove quotes from the commandline
2010-12-11 16:30:49 +00:00
WCHAR Path[MAX_PATH+1] = {0};
2011-03-29 19:21:57 +00:00
if (lpCmdLine)
2009-02-10 18:37:48 +00:00
{
2010-12-11 16:30:49 +00:00
size_t Pos = 0;
2011-03-29 19:21:57 +00:00
for (size_t i = 0, len = wcslen(lpCmdLine); i <= len && Pos < MAX_PATH; ++i)
2009-02-10 18:37:48 +00:00
{
2011-03-29 19:21:57 +00:00
if (lpCmdLine[i] != L'\"') Path[Pos++] = lpCmdLine[i];
2009-02-10 18:37:48 +00:00
}
}
int result = 1;
try
{
Rainmeter = new CRainmeter;
2009-02-10 18:37:48 +00:00
if (Rainmeter)
{
result = Rainmeter->Initialize(hWnd, hInstance, lpCmdLine);
}
}
catch (CError& error)
2009-02-10 18:37:48 +00:00
{
MessageBox(hWnd, error.GetString().c_str(), APPNAME, MB_OK | MB_TOPMOST | MB_ICONEXCLAMATION);
2009-02-10 18:37:48 +00:00
}
if (result == 1)
2010-12-11 16:30:49 +00:00
{
DestroyWindow(hWnd);
}
else
2009-02-10 18:37:48 +00:00
{
// Run the standard window message loop
while ((bRet = GetMessage(&msg, NULL, 0, 0)) != 0)
{
if (bRet == -1) // error
{
CRainmeter::Quit();
break;
}
else
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
2011-03-29 19:21:57 +00:00
}
2009-02-10 18:37:48 +00:00
if (hMutex) ReleaseMutex(hMutex);
2011-03-29 19:21:57 +00:00
return (int)msg.wParam;
}
2009-02-10 18:37:48 +00:00
2011-03-29 19:21:57 +00:00
/*
2009-02-10 18:37:48 +00:00
** InitApplication
**
** Creates the windowclass
**
*/
BOOL InitApplication(HINSTANCE hInstance, const WCHAR* WinClass)
{
WNDCLASS wc;
wc.style = 0;
wc.lpfnWndProc = (WNDPROC)WndProc;
2009-02-10 18:37:48 +00:00
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = hInstance;
wc.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_WINDOW));
2009-02-10 18:37:48 +00:00
wc.hCursor = LoadCursor(NULL, IDC_ARROW);
2011-03-29 19:21:57 +00:00
wc.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
2009-02-10 18:37:48 +00:00
wc.lpszMenuName = NULL;
wc.lpszClassName = WinClass;
return RegisterClass(&wc);
}
2011-03-29 19:21:57 +00:00
/*
2009-02-10 18:37:48 +00:00
** InitInstance
**
** Creates the window. This is just an invisible window. The real window
** is created by the DLL.
**
*/
HWND InitInstance(HINSTANCE hInstance, const WCHAR* WinClass, const WCHAR* WinName)
{
return CreateWindowEx(
WS_EX_TOOLWINDOW,
WinClass,
WinName,
2011-09-03 21:51:25 +00:00
WS_POPUP | WS_DISABLED,
2009-02-10 18:37:48 +00:00
CW_USEDEFAULT,
CW_USEDEFAULT,
CW_USEDEFAULT,
CW_USEDEFAULT,
NULL,
NULL,
hInstance,
NULL
);
}
2011-03-29 19:21:57 +00:00
/*
2009-02-10 18:37:48 +00:00
** Bang
**
** Sends bangs to the DLL
**
*/
void Bang(const WCHAR* command)
2009-02-10 18:37:48 +00:00
{
// Check if Rainmeter is running
HWND wnd = FindWindow(WinClass, WinName);
2009-02-10 18:37:48 +00:00
if (wnd != NULL)
{
COPYDATASTRUCT copyData;
copyData.dwData = 1;
copyData.cbData = (DWORD)((wcslen(command) + 1) * sizeof(WCHAR));
copyData.lpData = (void*)command;
// Send the bang to the Rainmeter window
SendMessage(wnd, WM_COPYDATA, (WPARAM)NULL, (LPARAM)&copyData);
2009-02-10 18:37:48 +00:00
}
else
{
2011-07-03 13:21:01 +00:00
if (_wcsicmp(L"!rainmeterquit", command) != 0 &&
2011-07-03 13:18:10 +00:00
_wcsicmp(L"!quit", command) != 0)
{
2010-09-21 11:09:36 +00:00
MessageBox(NULL, L"Rainmeter is not running.\nUnable to send the !bang to it.", L"Rainmeter", MB_OK | MB_TOPMOST | MB_ICONERROR);
}
2009-02-10 18:37:48 +00:00
}
}
2011-03-29 19:21:57 +00:00
/*
** RmLoadSystemLibrary
**
** Loads a system dll from system32 directory.
**
*/
HMODULE RmLoadSystemLibrary(LPCWSTR lpLibFileName)
{
WCHAR buffer[MAX_PATH];
std::wstring path;
if (GetSystemDirectory(buffer, MAX_PATH))
{
path = buffer;
path += L"\\";
path += lpLibFileName;
return LoadLibrary(path.c_str());
}
return NULL;
}
2011-03-29 19:21:57 +00:00
/*
** IsRunning
**
** Checks whether Rainmeter.exe is running.
**
*/
BOOL IsRunning(HANDLE* hMutex)
{
typedef struct
{
ULONG i[2];
ULONG buf[4];
unsigned char in[64];
unsigned char digest[16];
} MD5_CTX;
typedef void (WINAPI *FPMD5INIT)(MD5_CTX *context);
typedef void (WINAPI *FPMD5UPDATE)(MD5_CTX *context, const unsigned char *input, unsigned int inlen);
typedef void (WINAPI *FPMD5FINAL)(MD5_CTX *context);
// Create MD5 digest from command line
HMODULE hCryptDll = RmLoadSystemLibrary(L"cryptdll.dll");
if (!hCryptDll) // Unable to check the mutex
{
*hMutex = NULL;
return FALSE;
}
FPMD5INIT MD5Init = (FPMD5INIT)GetProcAddress(hCryptDll, "MD5Init");
FPMD5UPDATE MD5Update = (FPMD5UPDATE)GetProcAddress(hCryptDll, "MD5Update");
FPMD5FINAL MD5Final = (FPMD5FINAL)GetProcAddress(hCryptDll, "MD5Final");
if (!MD5Init || !MD5Update || !MD5Final) // Unable to check the mutex
{
FreeLibrary(hCryptDll);
*hMutex = NULL;
return FALSE;
}
std::wstring cmdLine = GetCommandLine();
std::transform(cmdLine.begin(), cmdLine.end(), cmdLine.begin(), ::towlower);
MD5_CTX ctx = {0};
MD5Init(&ctx);
MD5Update(&ctx, (LPBYTE)cmdLine.c_str(), cmdLine.length() * sizeof(WCHAR));
MD5Final(&ctx);
FreeLibrary(hCryptDll);
// Convert MD5 digest to mutex string (e.g. "Rainmeter@0123456789abcdef0123456789abcdef")
const WCHAR szHexTable[] = L"0123456789abcdef";
WCHAR szMutex[64] = {0};
wcscpy(szMutex, L"Rainmeter@");
WCHAR* pos = szMutex + wcslen(szMutex);
for (size_t i = 0; i < 16; ++i)
{
*(pos++) = *(szHexTable + ((ctx.digest[i] >> 4) & 0xF));
*(pos++) = *(szHexTable + (ctx.digest[i] & 0xF));
}
// Create mutex
HANDLE hMutexTemp = CreateMutex(NULL, FALSE, szMutex);
if (GetLastError() == ERROR_ALREADY_EXISTS)
{
// Rainmeter.exe is already running
*hMutex = NULL;
return TRUE;
}
// Rainmeter.exe is not running
*hMutex = hMutexTemp;
return FALSE;
}
2011-03-29 19:21:57 +00:00
/*
** WndProc
2009-02-10 18:37:48 +00:00
**
** The main window procedure
**
*/
LRESULT CALLBACK WndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
2009-02-10 18:37:48 +00:00
{
switch (uMsg)
{
2009-02-10 18:37:48 +00:00
case WM_DESTROY:
{
CRainmeter::Quit();
2009-02-10 18:37:48 +00:00
PostQuitMessage(0);
}
break;
case WM_COPYDATA:
{
COPYDATASTRUCT* cds = (COPYDATASTRUCT*)lParam;
if (Rainmeter && cds && (cds->dwData == 1) && (cds->cbData > 0) && cds->lpData)
{
ExecuteBang((LPCWSTR)cds->lpData);
}
}
break;
2009-02-10 18:37:48 +00:00
default:
return DefWindowProc(hWnd, uMsg, wParam, lParam);
2009-02-10 18:37:48 +00:00
}
return 0;
}