aboutsummaryrefslogtreecommitdiff
path: root/c/win.cpp
blob: 435e739eac7f158ea561b3f3d3516fdcde0b4af1 (plain)
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
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
#include <utility>
#include <windows.h>

#include "win.h"

Win32Error::Win32Error() : code(GetLastError()) {}
Win32Error::Win32Error(const DWORD code) : code(code) {}

Win32Error::~Win32Error()
{
	if (m_szMsg)
		HeapFree(GetProcessHeap(), 0, m_szMsg);
	if (m_wszMsg)
		HeapFree(GetProcessHeap(), 0, m_wszMsg);
}

const char* Win32Error::what() const noexcept
{
	if (!m_szMsg)
		FormatMessageA(
		    FORMAT_MESSAGE_ALLOCATE_BUFFER|FORMAT_MESSAGE_FROM_SYSTEM|FORMAT_MESSAGE_IGNORE_INSERTS,
		    NULL,
		    code,
		    MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US),
		    (char*)&m_szMsg,
		    0, NULL);
	return m_szMsg;
}

const wchar_t* Win32Error::What() const noexcept
{
	if (!m_wszMsg)
		FormatMessage(
		    FORMAT_MESSAGE_ALLOCATE_BUFFER|FORMAT_MESSAGE_FROM_SYSTEM|FORMAT_MESSAGE_IGNORE_INSERTS,
		    NULL,
		    code,
		    MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US),
		    (wchar_t*)&m_wszMsg,
		    0, NULL);
	return m_wszMsg;
}

std::optional<Library> Library::Maybe(const wchar_t* const lib)
{
	HMODULE hModule = LoadLibrary(lib);
	if (!hModule) return {};
	return Library(hModule);
}

Library::Library(const HMODULE hModule) : m_hModule(hModule) {}

Library::Library(const wchar_t* const lib)
{
	m_hModule = LoadLibrary(lib);
	if (!m_hModule)
		throw Win32Error();
}

Library::~Library()
{
	FreeLibrary(m_hModule);
}

int EBMessageBox(const wchar_t* const wszText, const wchar_t* const wszCaption, const unsigned uType)
{
	extern HWND g_hWnd;

	auto proc = [](const int nCode, const WPARAM wParam, const LPARAM lParam) noexcept
		-> LRESULT CALLBACK
	{
		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;
	};

	HHOOK hHook = require(SetWindowsHookEx(WH_CBT, proc, (HINSTANCE)NULL, GetCurrentThreadId()));
	int r = MessageBox(g_hWnd, wszText, wszCaption, uType);
	require(UnhookWindowsHookEx(hHook));
	return r;
}

int GetRelativeCursorPos(HWND hWnd, POINT* pt)
{
	RECT rc;
	if (!GetClientRect(hWnd, &rc)) return 0;
	SetLastError(ERROR_SUCCESS);
	if (!MapWindowPoints(hWnd, NULL, (POINT*)&rc, 2)) return 0;

	POINT ptMouse;
	if (!GetCursorPos(&ptMouse)) return 0;
	pt->x = ptMouse.x-rc.left;
	pt->y = ptMouse.y-rc.top;
	return 1;
}