2020-12-15 20:16:09 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#define WIN32_LEAN_AND_MEAN
|
|
|
|
#include <Windows.h>
|
|
|
|
#include <dwmapi.h>
|
|
|
|
|
|
|
|
#include <array>
|
|
|
|
#include <optional>
|
|
|
|
|
|
|
|
// Initializes and runs windows message loop
|
2022-08-27 07:17:20 +08:00
|
|
|
inline int run_message_loop(const bool until_idle = false, const std::optional<uint32_t> timeout_ms = {})
|
2020-12-15 20:16:09 +08:00
|
|
|
{
|
|
|
|
MSG msg{};
|
|
|
|
bool stop = false;
|
|
|
|
UINT_PTR timerId = 0;
|
2022-08-27 07:17:20 +08:00
|
|
|
if (timeout_ms.has_value())
|
2020-12-15 20:16:09 +08:00
|
|
|
{
|
2022-08-27 07:17:20 +08:00
|
|
|
timerId = SetTimer(nullptr, 0, *timeout_ms, nullptr);
|
2020-12-15 20:16:09 +08:00
|
|
|
}
|
|
|
|
|
2022-08-27 07:17:20 +08:00
|
|
|
while (!stop && (until_idle ? PeekMessageW(&msg, nullptr, 0, 0, PM_REMOVE) : GetMessageW(&msg, nullptr, 0, 0)))
|
2020-12-15 20:16:09 +08:00
|
|
|
{
|
|
|
|
TranslateMessage(&msg);
|
|
|
|
DispatchMessageW(&msg);
|
|
|
|
stop = until_idle && !PeekMessageW(&msg, nullptr, 0, 0, PM_NOREMOVE);
|
|
|
|
stop = stop || (msg.message == WM_TIMER && msg.wParam == timerId);
|
|
|
|
}
|
2022-08-27 07:17:20 +08:00
|
|
|
if (timeout_ms.has_value())
|
2020-12-15 20:16:09 +08:00
|
|
|
{
|
|
|
|
KillTimer(nullptr, timerId);
|
|
|
|
}
|
|
|
|
return static_cast<int>(msg.wParam);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if window is part of the shell or the taskbar.
|
|
|
|
inline bool is_system_window(HWND hwnd, const char* class_name)
|
|
|
|
{
|
|
|
|
// We compare the HWND against HWND of the desktop and shell windows,
|
|
|
|
// we also filter out some window class names know to belong to the taskbar.
|
|
|
|
constexpr std::array system_classes = { "SysListView32", "WorkerW", "Shell_TrayWnd", "Shell_SecondaryTrayWnd", "Progman" };
|
|
|
|
const std::array system_hwnds = { GetDesktopWindow(), GetShellWindow() };
|
|
|
|
for (auto system_hwnd : system_hwnds)
|
|
|
|
{
|
|
|
|
if (hwnd == system_hwnd)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (const auto system_class : system_classes)
|
|
|
|
{
|
|
|
|
if (!strcmp(system_class, class_name))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2022-08-27 07:17:20 +08:00
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline T GetWindowCreateParam(LPARAM lparam)
|
|
|
|
{
|
|
|
|
static_assert(sizeof(T) <= sizeof(void*));
|
2023-02-08 19:00:19 +08:00
|
|
|
T data{ static_cast <T>(reinterpret_cast<CREATESTRUCT*>(lparam)->lpCreateParams) };
|
2022-08-27 07:17:20 +08:00
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline void StoreWindowParam(HWND window, T data)
|
|
|
|
{
|
|
|
|
static_assert(sizeof(T) <= sizeof(void*));
|
|
|
|
SetWindowLongPtrW(window, GWLP_USERDATA, reinterpret_cast<LONG_PTR>(data));
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
inline T GetWindowParam(HWND window)
|
|
|
|
{
|
2023-02-08 19:00:19 +08:00
|
|
|
return reinterpret_cast <T>(GetWindowLongPtrW(window, GWLP_USERDATA));
|
2022-08-27 07:17:20 +08:00
|
|
|
}
|