reactos/modules/rostests/tests/multithrdwin/multithrdwin.c

238 lines
6.4 KiB
C

#include <windows.h>
#include <stdio.h>
ULONG DbgPrint(PCH Format,...);
typedef struct _THRDCREATEWIN
{
HANDLE hThread;
DWORD ThreadId;
LPSTR Caption;
HWND *Parent;
HWND Window;
DWORD Style;
POINT Position;
SIZE Size;
} THRDCREATEWIN, *PTHRDCREATEWIN;
static HINSTANCE hAppInstance;
static HANDLE WinCreatedEvent;
static THRDCREATEWIN wnds[3];
LRESULT WINAPI MultiWndProc(HWND, UINT, WPARAM, LPARAM);
DWORD WINAPI
WindowThreadProc(LPVOID lpParameter)
{
MSG msg;
char caption[64];
PTHRDCREATEWIN cw = (PTHRDCREATEWIN)lpParameter;
sprintf(caption, cw->Caption, GetCurrentThreadId());
cw->Window = CreateWindow("MultiClass",
caption,
cw->Style | WS_VISIBLE,
cw->Position.x,
cw->Position.y,
cw->Size.cx,
cw->Size.cy,
(cw->Parent ? *(cw->Parent) : 0),
NULL,
hAppInstance,
NULL);
SetEvent(WinCreatedEvent);
if(!cw->Window)
{
fprintf(stderr, "CreateWindow failed (last error 0x%lX)\n",
GetLastError());
return 1;
}
CreateWindow("BUTTON","Sleep",WS_CHILD|WS_VISIBLE|WS_TABSTOP|WS_GROUP|BS_PUSHBUTTON, 10, 10, 70, 23, cw->Window, (PVOID)1, hAppInstance, NULL);
CreateWindow("BUTTON","1",WS_CHILD|WS_VISIBLE|WS_TABSTOP|WS_GROUP|BS_PUSHBUTTON, 83, 10, 20, 23, cw->Window, (PVOID)2, hAppInstance, NULL);
CreateWindow("BUTTON","2",WS_CHILD|WS_VISIBLE|WS_TABSTOP|WS_GROUP|BS_PUSHBUTTON, 105, 10, 20, 23, cw->Window, (PVOID)3, hAppInstance, NULL);
CreateWindow("BUTTON","3",WS_CHILD|WS_VISIBLE|WS_TABSTOP|WS_GROUP|BS_PUSHBUTTON, 127, 10, 20, 23, cw->Window, (PVOID)4, hAppInstance, NULL);
while(GetMessage(&msg, NULL, 0, 0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
return 0;
}
int WINAPI
WinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPSTR lpszCmdLine,
int nCmdShow)
{
WNDCLASS wc;
int i;
HANDLE Threads[3];
hAppInstance = hInstance;
WinCreatedEvent = CreateEvent(NULL,
FALSE,
FALSE,
NULL);
if(!WinCreatedEvent)
{
fprintf(stderr, "Failed to create event (last error 0x%lX)\n",
GetLastError());
return 1;
}
wc.lpszClassName = "MultiClass";
wc.lpfnWndProc = MultiWndProc;
wc.style = CS_VREDRAW | CS_HREDRAW;
wc.hInstance = hInstance;
wc.hIcon = LoadIcon(NULL, (LPCTSTR) IDI_APPLICATION);
wc.hCursor = LoadCursor(NULL, (LPCTSTR) IDC_ARROW);
wc.hbrBackground = (HBRUSH) (COLOR_WINDOW + 1);
wc.lpszMenuName = NULL;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
if (RegisterClass(&wc) == 0)
{
fprintf(stderr, "RegisterClass failed (last error 0x%lX)\n",
GetLastError());
return(1);
}
wnds[0].Caption = "TopLevel1 (ThreadID: %d)";
wnds[0].Parent = NULL;
wnds[0].Style = WS_OVERLAPPEDWINDOW | WS_CLIPCHILDREN | WS_CLIPSIBLINGS;
wnds[0].Position.x = wnds[0].Position.y = 0;
wnds[0].Size.cx = 320;
wnds[0].Size.cy = 240;
wnds[1].Caption = "Child1 of TopLevel1 (ThreadID: %d)";
wnds[1].Parent = &wnds[0].Window;
wnds[1].Style = WS_CHILD | WS_BORDER | WS_CAPTION | WS_VISIBLE | WS_SYSMENU | WS_CLIPCHILDREN | WS_CLIPSIBLINGS;
wnds[1].Position.x = 20;
wnds[1].Position.y = 120;
wnds[1].Size.cx = wnds[1].Size.cy = 240;
wnds[2].Caption = "TopLevel2 (ThreadID: %d)";
wnds[2].Parent = NULL;
wnds[2].Style = WS_OVERLAPPEDWINDOW | WS_CLIPCHILDREN | WS_CLIPSIBLINGS;
wnds[2].Position.x = 400;
wnds[2].Position.y = 0;
wnds[2].Size.cx = 160;
wnds[2].Size.cy = 490;
for(i = 0; i < (sizeof(wnds) / sizeof(THRDCREATEWIN)); i++)
{
wnds[i].hThread = CreateThread(NULL,
0,
WindowThreadProc,
&wnds[i],
0,
&wnds[i].ThreadId);
Threads[i] = wnds[i].hThread;
if(!wnds[i].hThread)
{
fprintf(stderr, "CreateThread #%i failed (last error 0x%lX)\n",
i, GetLastError());
return 1;
}
WaitForSingleObject(WinCreatedEvent, INFINITE);
}
WaitForMultipleObjects(sizeof(Threads) / sizeof(HANDLE), &Threads[0], TRUE, INFINITE);
UnregisterClass("MultiClass", hInstance);
return 0;
}
LRESULT CALLBACK MultiWndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
{
PAINTSTRUCT ps;
HDC hDC;
RECT Client;
HBRUSH Brush;
DWORD_PTR Ret;
static COLORREF Colors[] =
{
RGB(0x00, 0x00, 0x00),
RGB(0x80, 0x00, 0x00),
RGB(0x00, 0x80, 0x00),
RGB(0x00, 0x00, 0x80),
RGB(0x80, 0x80, 0x00),
RGB(0x80, 0x00, 0x80),
RGB(0x00, 0x80, 0x80),
RGB(0x80, 0x80, 0x80),
RGB(0xff, 0x00, 0x00),
RGB(0x00, 0xff, 0x00),
RGB(0x00, 0x00, 0xff),
RGB(0xff, 0xff, 0x00),
RGB(0xff, 0x00, 0xff),
RGB(0x00, 0xff, 0xff),
RGB(0xff, 0xff, 0xff)
};
static unsigned CurrentColor = 0;
switch(msg)
{
case WM_PAINT:
hDC = BeginPaint(hWnd, &ps);
GetClientRect(hWnd, &Client);
Brush = CreateSolidBrush(Colors[CurrentColor]);
FillRect(hDC, &Client, Brush);
DeleteObject(Brush);
CurrentColor++;
if (sizeof(Colors) / sizeof(Colors[0]) <= CurrentColor)
{
CurrentColor = 0;
}
EndPaint(hWnd, &ps);
break;
case WM_COMMAND:
switch(LOWORD(wParam))
{
case 1:
Sleep(20000);
break;
case 2:
case 3:
case 4:
if(SendMessageTimeout(wnds[LOWORD(wParam) - 2].Window, WM_NULL, 0, 0, SMTO_ABORTIFHUNG, 1000, &Ret))
{
DbgPrint("SendMessageTimeout() OK");
MessageBox(hWnd, "SendMessageTimeout() OK", NULL, 0);
}
else
{
if(GetLastError() == ERROR_TIMEOUT)
{
DbgPrint("SendMessageTimeout() Timeout");
MessageBox(hWnd, "SendMessageTimeout() Timeout", NULL, 0);
}
else
{
DbgPrint("SendMessageTimeout() Failed");
MessageBox(hWnd, "SendMessageTimeout() Failed", NULL, 0);
}
}
break;
}
break;
case WM_DESTROY:
PostQuitMessage(0);
break;
default:
return DefWindowProc(hWnd, msg, wParam, lParam);
}
return 0;
}