179 lines
5.0 KiB
C
179 lines
5.0 KiB
C
// 17 july 2014
|
|
|
|
#include "winapi_windows.h"
|
|
#include "_cgo_export.h"
|
|
|
|
// note that this includes the terminating '\0'
|
|
// this also assumes WC_TABCONTROL is longer than areaWindowClass
|
|
#define NCLASSNAME (sizeof WC_TABCONTROL / sizeof WC_TABCONTROL[0])
|
|
|
|
void uimsgloop(void)
|
|
{
|
|
MSG msg;
|
|
int res;
|
|
HWND active, focus;
|
|
WCHAR classchk[NCLASSNAME];
|
|
BOOL dodlgmessage;
|
|
BOOL istab;
|
|
BOOL idm;
|
|
|
|
for (;;) {
|
|
SetLastError(0);
|
|
res = GetMessageW(&msg, NULL, 0, 0);
|
|
if (res < 0)
|
|
xpanic("error calling GetMessage()", GetLastError());
|
|
if (res == 0) // WM_QUIT
|
|
break;
|
|
active = GetActiveWindow();
|
|
if (active != NULL) {
|
|
// bit of logic involved here:
|
|
// we don't want dialog messages passed into Areas, so we don't call IsDialogMessageW() there
|
|
// as for Tabs, we can't have both WS_TABSTOP and WS_EX_CONTROLPARENT set at the same time, so we hotswap the two styles to get the behavior we want
|
|
// theoretically we could use the class atom to avoid a wcscmp()
|
|
// however, raymond chen advises against this - http://blogs.msdn.com/b/oldnewthing/archive/2004/10/11/240744.aspx (and we're not in control of the Tab class, before you say anything)
|
|
// we could also theoretically just send msgAreaDefocuses directly, but what DefWindowProc() does to a WM_APP message is undocumented
|
|
dodlgmessage = TRUE;
|
|
istab = FALSE;
|
|
focus = GetFocus();
|
|
if (focus != NULL) {
|
|
if (GetClassNameW(focus, classchk, NCLASSNAME) == 0)
|
|
xpanic("error getting name of focused window class for Area check", GetLastError());
|
|
if (wcscmp(classchk, areaWindowClass) == 0)
|
|
dodlgmessage = FALSE;
|
|
else if (wcscmp(classchk, WC_TABCONTROL) == 0)
|
|
// THIS BIT IS IMPORTANT
|
|
// if the current tab has no children, then there will be no children left in the dialog to tab to, and IsDialogMessageW() will loop forever
|
|
istab = (BOOL) SendMessageW(focus, msgTabCurrentTabHasChildren, 0, 0);
|
|
}
|
|
if (dodlgmessage) {
|
|
if (istab)
|
|
tabEnterChildren(focus);
|
|
idm = IsDialogMessageW(active, &msg);
|
|
if (istab)
|
|
tabLeaveChildren(focus);
|
|
if (idm != 0)
|
|
continue;
|
|
}
|
|
}
|
|
TranslateMessage(&msg);
|
|
DispatchMessageW(&msg);
|
|
}
|
|
}
|
|
|
|
void issue(void *request)
|
|
{
|
|
SetLastError(0);
|
|
if (PostMessageW(msgwin, msgRequest, 0, (LPARAM) request) == 0)
|
|
xpanic("error issuing request", GetLastError());
|
|
}
|
|
|
|
HWND msgwin;
|
|
|
|
#define msgwinclass L"gouimsgwin"
|
|
|
|
struct modalqueue {
|
|
BOOL inmodal;
|
|
void **modals;
|
|
size_t len;
|
|
size_t cap;
|
|
};
|
|
|
|
static BOOL CALLBACK beginEndModalAll(HWND hwnd, LPARAM lParam)
|
|
{
|
|
if (hwnd != msgwin)
|
|
SendMessageW(hwnd, (UINT) lParam, 0, 0);
|
|
return TRUE;
|
|
}
|
|
|
|
static LRESULT CALLBACK msgwinproc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
LRESULT shared;
|
|
struct modalqueue *mq;
|
|
size_t i;
|
|
|
|
mq = (struct modalqueue *) GetWindowLongPtrW(hwnd, GWLP_USERDATA);
|
|
if (mq == NULL) {
|
|
mq = (struct modalqueue *) malloc(sizeof (struct modalqueue));
|
|
if (mq == NULL)
|
|
xpanic("error allocating modal queue structure", GetLastError());
|
|
ZeroMemory(mq, sizeof (struct modalqueue));
|
|
mq->inmodal = FALSE;
|
|
mq->len = 0;
|
|
mq->cap = 128;
|
|
mq->modals = (void **) malloc(mq->cap * sizeof (void *));
|
|
if (mq->modals == NULL)
|
|
xpanic("error allocating modal quque", GetLastError());
|
|
SetWindowLongPtrW(hwnd, GWLP_USERDATA, (LONG_PTR) mq);
|
|
}
|
|
|
|
if (sharedWndProc(hwnd, uMsg, wParam, lParam, &shared))
|
|
return shared;
|
|
switch (uMsg) {
|
|
case WM_CREATE:
|
|
// initial
|
|
makeCheckboxImageList(hwnd);
|
|
return 0;
|
|
// TODO respond to WM_THEMECHANGED
|
|
case msgRequest:
|
|
// in modal?
|
|
if (mq->inmodal) {
|
|
mq->modals[mq->len] = (void *) lParam;
|
|
mq->len++;
|
|
if (mq->len >= mq->cap) {
|
|
mq->cap *= 2;
|
|
mq->modals = (void **) realloc(mq->modals, mq->cap * sizeof (void *));
|
|
if (mq->modals == NULL)
|
|
xpanic("error growing modal queue", GetLastError());
|
|
}
|
|
return;
|
|
}
|
|
// nope, we can run now
|
|
doissue((void *) lParam);
|
|
return 0;
|
|
case msgBeginModal:
|
|
mq->inmodal = TRUE;
|
|
EnumThreadWindows(GetCurrentThreadId(), beginEndModalAll, msgBeginModal);
|
|
return 0;
|
|
case msgEndModal:
|
|
mq->inmodal = FALSE;
|
|
for (i = 0; i < mq->len; i++)
|
|
doissue(mq->modals[i]);
|
|
mq->len = 0;
|
|
EnumThreadWindows(GetCurrentThreadId(), beginEndModalAll, msgEndModal);
|
|
return 0;
|
|
default:
|
|
return DefWindowProcW(hwnd, uMsg, wParam, lParam);
|
|
}
|
|
xmissedmsg("message-only", "msgwinproc()", uMsg);
|
|
return 0; // unreachable
|
|
}
|
|
|
|
DWORD makemsgwin(char **errmsg)
|
|
{
|
|
WNDCLASSW wc;
|
|
|
|
ZeroMemory(&wc, sizeof (WNDCLASSW));
|
|
wc.lpfnWndProc = msgwinproc;
|
|
wc.hInstance = hInstance;
|
|
wc.hIcon = hDefaultIcon;
|
|
wc.hCursor = hArrowCursor;
|
|
wc.hbrBackground = (HBRUSH) (COLOR_BTNFACE + 1);
|
|
wc.lpszClassName = msgwinclass;
|
|
if (RegisterClassW(&wc) == 0) {
|
|
*errmsg = "error registering message-only window classs";
|
|
return GetLastError();
|
|
}
|
|
msgwin = CreateWindowExW(
|
|
0,
|
|
msgwinclass, L"package ui message-only window",
|
|
0,
|
|
CW_USEDEFAULT, CW_USEDEFAULT,
|
|
CW_USEDEFAULT, CW_USEDEFAULT,
|
|
HWND_MESSAGE, NULL, hInstance, NULL);
|
|
if (msgwin == NULL) {
|
|
*errmsg = "error creating message-only window";
|
|
return GetLastError();
|
|
}
|
|
return 0;
|
|
}
|