Last updated: 11. 1.1998, 0:11
<*/NOWARN:F*> (*------------------------------------------------- CHECKER2.C --- Mouse Hit-Test Demo Program No. 2 (c) Charles Petzold, 1996 Checker2.mod --- Translation to Stony Brook Modula-2 (c) Peter Stadler, 1997 -------------------------------------------------*) MODULE Checker2; IMPORT WINUSER; IMPORT WINX; IMPORT WINGDI; IMPORT WIN32; IMPORT SYSTEM; CONST szAppName = "Checker2"; DIVISIONS=5; VAR hwnd : WIN32.HWND; msg : WINUSER.MSG; wc : WINUSER.WNDCLASSEX; VAR fState : ARRAY[0..DIVISIONS-1],[0..DIVISIONS-1] OF BOOLEAN; cxBlock : INTEGER; cyBlock : INTEGER; (*++++*****************************************************************) PROCEDURE MaxInt (a,b : INTEGER) : INTEGER; (**********************************************************************) BEGIN IF(a>b) THEN RETURN a; ELSE RETURN b; END; END MaxInt; (*++++*****************************************************************) PROCEDURE MinInt (a,b : INTEGER) : INTEGER; (**********************************************************************) BEGIN IF(a>b) THEN RETURN b; ELSE RETURN a; END; END MinInt; <*/PUSH*> %IF WIN32 %THEN <*/CALLS:WIN32SYSTEM*> %ELSE <*/CALLS:WINSYSTEM*> %END <*/PUSH*> %IF WIN32 %THEN <*/CALLS:WIN32SYSTEM*> %ELSE <*/CALLS:WINSYSTEM*> %END (*++++**********************************************************************) PROCEDURE WndProc (hwnd : WIN32.HWND; iMsg : WIN32.UINT; wParam : WIN32.WPARAM; lParam : WIN32.LPARAM) : WIN32.LRESULT [EXPORT]; (***************************************************************************) VAR hdc : WIN32.HDC; ps : WINUSER.PAINTSTRUCT; rect : WIN32.RECT; x : INTEGER; y : INTEGER; point : WIN32.POINT; BEGIN CASE (iMsg) OF | WINUSER.WM_SIZE : cxBlock := VAL(INTEGER,WINUSER.LOWORD (lParam)) DIV DIVISIONS; cyBlock := VAL(INTEGER,WINUSER.HIWORD (lParam)) DIV DIVISIONS; RETURN 0; | WINUSER.WM_SETFOCUS : WINUSER.ShowCursor (TRUE); RETURN 0; | WINUSER.WM_KILLFOCUS : WINUSER.ShowCursor (FALSE); RETURN 0; | WINUSER.WM_KEYDOWN : WINUSER.GetCursorPos(point); WINUSER.ScreenToClient (hwnd, point); x := MaxInt (0, MinInt (DIVISIONS - 1, point.x DIV cxBlock)); y := MaxInt (0, MinInt (DIVISIONS - 1, point.y DIV cyBlock)); CASE (wParam) OF | WINUSER.VK_UP : DEC(y); | WINUSER.VK_DOWN : INC(y); | WINUSER.VK_LEFT : DEC(x); | WINUSER.VK_RIGHT : INC(x); | WINUSER.VK_HOME : x := 0; y := 0; | WINUSER.VK_END : x := DIVISIONS - 1; y := DIVISIONS - 1; | WINUSER.VK_RETURN : WINUSER.SendMessage (hwnd, WINUSER.WM_LBUTTONDOWN, WINUSER.MK_LBUTTON, WINUSER.MAKELONG (x * cxBlock, y * cyBlock)); | WINUSER.VK_SPACE : WINUSER.SendMessage (hwnd, WINUSER.WM_LBUTTONDOWN, WINUSER.MK_LBUTTON, WINUSER.MAKELONG (x * cxBlock, y * cyBlock)); ELSE END; x := (x + DIVISIONS) REM DIVISIONS; y := (y + DIVISIONS) REM DIVISIONS; point.x := x * cxBlock + cxBlock DIV 2; point.y := y * cyBlock + cyBlock DIV 2; WINUSER.ClientToScreen (hwnd, point); WINUSER.SetCursorPos (point.x, point.y); RETURN 0; | WINUSER.WM_LBUTTONDOWN : x := VAL(INTEGER,WINUSER.LOWORD (lParam)) DIV cxBlock; y := VAL(INTEGER,WINUSER.HIWORD (lParam)) DIV cyBlock; IF (x < DIVISIONS) AND (y < DIVISIONS) THEN fState[x][y] := NOT fState[x][y]; rect.left := x * cxBlock; rect.top := y * cyBlock; rect.right := (x + 1) * cxBlock; rect.bottom := (y + 1) * cyBlock; WINUSER.InvalidateRect (hwnd, rect, FALSE); ELSE WINUSER.MessageBeep (0); END; RETURN 0; | WINUSER.WM_PAINT : hdc := WINUSER.BeginPaint (hwnd, ps); FOR x := 0 TO DIVISIONS-1 DO FOR y := 0 TO DIVISIONS-1 DO WINGDI.Rectangle (hdc, x * cxBlock, y * cyBlock, (x + 1) * cxBlock, (y + 1) * cyBlock); IF (fState [x][y]) THEN WINGDI.MoveToEx (hdc, x * cxBlock, y * cyBlock,WINX.NIL_POINT); WINGDI.LineTo (hdc, (x+1) * cxBlock, (y+1) * cyBlock); WINGDI.MoveToEx (hdc, x * cxBlock, (y+1) * cyBlock,WINX.NIL_POINT); WINGDI.LineTo (hdc, (x+1) * cxBlock, y * cyBlock); END; END; END; WINUSER.EndPaint (hwnd, ps); RETURN 0; | WINUSER.WM_DESTROY : WINUSER.PostQuitMessage (0); RETURN 0; ELSE RETURN WINUSER.DefWindowProc (hwnd, iMsg, wParam, lParam); END; END WndProc; <*/POP*> (*+++***********************************************************************) PROCEDURE InitApplication () : BOOLEAN; (***************************************************************************) BEGIN wc.cbSize := SIZE(WINUSER.WNDCLASSEX); wc.style := WINUSER.CS_HREDRAW BOR WINUSER.CS_VREDRAW; wc.lpfnWndProc := WndProc; wc.cbClsExtra := 0; wc.cbWndExtra := 0; wc.hInstance := WINX.Instance; wc.hIcon := WINUSER.LoadIcon (NIL, WINUSER.IDI_APPLICATION^); wc.hCursor := WINUSER.LoadCursor (NIL, WINUSER.IDC_ARROW^); wc.hbrBackground := SYSTEM.CAST(WIN32.HBRUSH, WINGDI.GetStockObject (WINGDI.WHITE_BRUSH)); wc.lpszMenuName := NIL; wc.lpszClassName := SYSTEM.ADR(szAppName); wc.hIconSm := WINUSER.LoadIcon (NIL, WINUSER.IDI_APPLICATION^); RETURN WINUSER.RegisterClassEx(wc)#0; END InitApplication; (*+++***********************************************************************) PROCEDURE InitMainWindow () : BOOLEAN; (***************************************************************************) BEGIN hwnd := WINUSER.CreateWindow (szAppName, "Checker2 Mouse Hit-Test Demo: Translation to Stony Brook Modula-2", WINUSER.WS_OVERLAPPEDWINDOW, WINUSER.CW_USEDEFAULT, WINUSER.CW_USEDEFAULT, WINUSER.CW_USEDEFAULT, WINUSER.CW_USEDEFAULT, NIL, NIL, wc.hInstance, NIL); IF hwnd = NIL THEN RETURN FALSE; END; WINUSER.ShowWindow (hwnd, WINUSER.SW_SHOWDEFAULT); WINUSER.UpdateWindow (hwnd); RETURN TRUE; END InitMainWindow; BEGIN IF InitApplication() AND InitMainWindow() THEN WHILE (WINUSER.GetMessage (msg, NIL, 0, 0)) DO WINUSER.TranslateMessage (msg); WINUSER.DispatchMessage (msg); END; END; END Checker2.