aboutsummaryrefslogtreecommitdiff
path: root/c/common.h
blob: 2c0be13181963eb4d48034c6f10349ce67dd39d3 (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
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
#ifndef COMMON_H
#define COMMON_H

#include <memory>
#include <optional>
#include <windows.h>

#ifdef UNICODE
#define WA "W"
#else
#define WA "A"
#endif

int EBMessageBox(const TCHAR* tszText, const TCHAR* tszCaption, unsigned uType);

struct Win32Error : public std::exception
{
	Win32Error();
	Win32Error(DWORD dwErr);
	~Win32Error();
	template <typename T = char> const T* what() const noexcept;

private:
	DWORD m_dwErr;
	char* m_szMsg = NULL;
	wchar_t* m_wszMsg = NULL;
};

struct Library
{
	Library(const TCHAR* tszLibrary);
	~Library();
	template <class T> T* GetProcAddress(const char* szProc);
private:
	HMODULE m_hModule;
};

template <typename T>
T* Library::GetProcAddress(const char* const szProc)
{
	return (T*)(void*)::GetProcAddress(m_hModule, szProc);
}

/* Create and return an object of type C. If construction fails,
 * return nothing. The returned value must be checked before being
 * used, as dereferencing is undefined if the value is empty. */
template <typename T, typename... U>
std::optional<T> maybe_make(U... xs)
{
	try {
		return T(xs...);
	} catch (...) {
		return {};
	}
}

/* Call Windows API function, throwing error on NULL. */
template <typename T>
inline T require(const T x)
{
	if (!x) throw Win32Error();
	return x;
}

/* Call Windows API function, showing a warning on NULL. */
template <typename T>
inline T prefer(const T x)
{
	if (!x) {
		EBMessageBox(Win32Error().what<TCHAR>(),
		    TEXT("System Error"), MB_ICONWARNING);
		return (T)NULL;
	}
	return x;
}

/* Conditionally choose between two values. This template is necessary
 * because the ternary conditional operator chooses only between
 * values of the same type. */
template <bool B, typename X, typename Y>
constexpr std::enable_if_t<B == true, X> choose(X x, Y) { return x; }
template <bool B, typename X, typename Y>
constexpr std::enable_if_t<B == false, Y> choose(X, Y y) { return y; }

/* Conditionally choose between ANSI and wide string literal. */
#define AWTEXT(t, s) choose<std::is_same_v<t, char>>(s, L"" s)

/* Conditionally choose between ANSI and wide Windows API function. */
#define AWFUN(t, f) choose<std::is_same_v<t, char>>(f##A, f##W)

/* Return integer scaled for current DPI. */
inline int Dpi(const int i)
{
	extern int g_iDPI;
	return MulDiv(i, g_iDPI, 96);
}

inline int Cmp(const int a, const int b)
{
	if (a == b) return 0;
	if (a > b) return 1;
	return -1;
}

/* Get window rectangle relative to parent. */
inline BOOL GetRelativeRect(const HWND hWnd, RECT* const pRr)
{
	if (!GetClientRect(hWnd, pRr)) return 0;
	return MapWindowPoints(hWnd, GetParent(hWnd), (POINT*)pRr, 2);
}

inline BOOL SetWindowRect(const HWND hWnd,
    const long left, const long top, const long right, const long bottom)
{
	return MoveWindow(hWnd,
	    left, top,
	    right-left, bottom-top,
	    TRUE);
}

inline BOOL SetWindowRect(const HWND hWnd, const RECT r)
{
	return SetWindowRect(hWnd, r.left, r.top, r.right, r.bottom);
}

#endif