參考資訊:
https://board.flatassembler.net/
當視窗有事件需要被處理時,系統會透過呼叫註冊的Callback副程式來處理,而這個Callback副程式就是在註冊Class時,設定給lpfnWndProc的事件處理副程式,因此,當這個事件處理副程式被呼叫時,可以處理自己感興趣的事件,而其餘事件則交給系統處理(透過呼叫DefWindowProc()),需要注意的是,例如:WM_CLOSE,就必須由該視窗自己處理,因為系統無法知道還有哪些額外的資源需要被釋放掉
main.asm
include "head.asm"
section ".text" code readable executable
proc WndProc hWnd, uMsg, wParam, lParam
mov eax, [uMsg]
cmp eax, WM_CLOSE
je .handle_close
cmp eax, WM_DESTROY
je .handle_destroy
invoke DefWindowProc, [hWnd], [uMsg], [wParam], [lParam]
jmp .finish
.handle_close:
invoke DestroyWindow, [hWnd]
xor eax, eax
jmp .finish
.handle_destroy:
invoke PostQuitMessage, 0
xor eax, eax
jmp .finish
.finish:
ret
endp
proc WinMain hInst, hPrevInst, CmdLine, CmdShow
local msg:MSG
mov [pWndClass.lpfnWndProc], WndProc
mov [pWndClass.lpszClassName], szName
invoke RegisterClass, pWndClass
invoke CreateWindowEx, WS_EX_LEFT, szName, szName, \
WS_OVERLAPPEDWINDOW or WS_VISIBLE, 0, 0, 300, 300, NULL, NULL, NULL, NULL
@@:
lea eax, [msg]
invoke GetMessage, eax, NULL, 0, 0
cmp eax, 0
je @f
lea eax, [msg]
invoke DispatchMessage, eax
jmp @b
@@:
mov eax, [msg.wParam]
ret
endp
start:
invoke GetModuleHandle, NULL
mov [hInstance], eax
invoke GetCommandLine
mov [pCommand], eax
stdcall WinMain, [hInstance], NULL, [pCommand], SW_SHOWNORMAL
invoke ExitProcess, eax
Line 5~9:關閉視窗的順序為:主視窗收到WM_CLOSE事件時,呼叫DestroyWindow(),DestroyWindow()會自動將子視窗的相關資源也一併釋放掉(包含Resource描述的資源),系統接著會發送WM_DESTROY事件給主視窗,待主視窗收到WM_DESTROY事件時,呼叫PostQuitMessage()結束視窗,值得注意的地方是,SendMessage()是屬於Block方式呼叫(必須等待動作執行完畢才返回),而PostQuitMessage()則是Non-block方式呼叫
Line 10:其餘事件由系統代為處理
完成