1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
|
#include <windows.h>
#include "common.h"
/* Win32Error: Exception for Windows API errors. */
Win32Error::Win32Error() : dwErr(GetLastError()) {}
Win32Error::Win32Error(const DWORD dwErr) : dwErr(dwErr) {}
Win32Error::~Win32Error()
{
if (m_szMsg)
HeapFree(GetProcessHeap(), 0, m_szMsg);
if (m_wszMsg)
HeapFree(GetProcessHeap(), 0, m_wszMsg);
}
/* Library: Wrapper for loading and freeing dynamically linked libraries. */
Library::Library(const TCHAR* const tszLibrary)
{
m_hModule = LoadLibrary(tszLibrary);
if (!m_hModule)
throw Win32Error();
}
Library::~Library()
{
FreeLibrary(m_hModule);
}
/* Convert narrow unmanaged string to narrow/wide basic string. */
template <>
std::basic_string<char> BstrFromSz(const char* sz, int)
{
return std::string(sz);
}
template <>
std::basic_string<wchar_t> BstrFromSz(const char* sz, int iCp)
{
int cbMultiByte = strlen(sz)+1;
int cchWideChar = MultiByteToWideChar(iCp, 0, sz, cbMultiByte, NULL, 0);
std::wstring wstr(cchWideChar, 0);
if (!MultiByteToWideChar(iCp, 0, sz, cbMultiByte, wstr.data(), cchWideChar))
throw Win32Error(GetLastError());
return wstr;
}
/* Move message box to center of main window. */
static LRESULT CALLBACK CBTProc(const int nCode, const WPARAM wParam, const LPARAM lParam) noexcept
{
extern HWND g_hWnd;
if (!g_hWnd || nCode < 0 || nCode != HCBT_ACTIVATE)
return CallNextHookEx(0, nCode, wParam, lParam);
HWND hWnd = (HWND)wParam;
long lStyle = GetWindowLong(hWnd, GWL_STYLE);
if (!(lStyle & WS_POPUP)) return 0;
RECT rcMain, rcMsg;
GetWindowRect(g_hWnd, &rcMain);
GetWindowRect(hWnd, &rcMsg);
SetWindowPos(hWnd, NULL,
rcMain.left+(rcMain.right-rcMain.left)/2-(rcMsg.right-rcMsg.left)/2,
rcMain.top+(rcMain.bottom-rcMain.top)/2-(rcMsg.bottom-rcMsg.top)/2,
-1, -1,
SWP_NOZORDER|SWP_NOSIZE|SWP_NOACTIVATE);
return 0;
}
/* Show message box owned by and centered in the main window. */
int EBMessageBox(const TCHAR* const tszText, const TCHAR* const tszCaption, const unsigned uType)
{
extern HWND g_hWnd;
HHOOK hHook = require(SetWindowsHookEx(WH_CBT, CBTProc, (HINSTANCE)NULL, GetCurrentThreadId()));
int r = MessageBox(g_hWnd, tszText, tszCaption, uType);
require(UnhookWindowsHookEx(hHook));
return r;
}
|