#ifndef COMMON_H
#define COMMON_H

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

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

template <typename T> std::basic_string<T> BstrFromSz(const char* sz, int iCp = CP_UTF8);
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 {};
	}
}

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

/* Check result of Windows API call, 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 can choose 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, wchar_t>>(L"" s, s)

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

/* 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