關於《MicroSoft C# Windows 程式設計》學習過程中關於OnPaint()的理解
WM_PAINT是視窗每次重繪都會產生的一個訊息。
OnPaint是對這個訊息的反應函式。
WM_PAINT
WM_PAINT是Windows視窗系統中一條重要的訊息,應用程式通過處理該訊息實現在視窗上的繪製工作。
一、WM_PAINT訊息
在系統繪製視窗時向程式發出WM_PAINT訊息。程式在接收到WM_PAINT訊息後呼叫BeginPaint函式獲取當前的Device Context進行繪圖操作,繪圖完畢後使用EndPaint釋放Device Context。
1. 系統何時傳送WM_PAINT訊息?
系統會在多個不同的時機發送WM_PAINT訊息:當第一次建立一個視窗時,當改變視窗的大小時,當把視窗從另一個視窗背後移出時,當最大化或最小化視窗時,等等,這些動作都是由 系統管理的,應用只是被動地接收該訊息,在訊息處理函式中進行繪製操作;大多數的時候應用也需要能夠主動引發視窗中的繪製操作,比如當視窗顯示的資料改變的時候,這一般是通過InvalidateRect和 InvalidateRgn函式來完成的。InvalidateRect和InvalidateRgn把指定的區域加到視窗的Update Region中,當應用的訊息佇列沒有其他訊息時,如果視窗的Update Region不為空時,系統就會自動產生WM_PAINT訊息。
系統為什麼不在呼叫Invalidate時傳送WM_PAINT訊息呢?又為什麼非要等應用訊息佇列為空時才傳送WM_PAINT訊息呢?這是因為系統把在視窗中的繪製操作當作一種低優先順序的操作,於是盡 可能地推後做。不過這樣也有利於提高繪製的效率:兩個WM_PAINT訊息之間通過InvalidateRect和InvaliateRgn使之失效的區域就會被累加起來,然後在一個WM_PAINT訊息中一次得到 更新,不僅能避免多次重複地更新同一區域,也優化了應用的更新操作。
像這種通過InvalidateRect和InvalidateRgn來使視窗區域無效,依賴於系統在合適的時機發送WM_PAINT訊息的機 制實際上是一種非同步工作方式,也就是說,在無效化視窗區域和傳送WM_PAINT訊息之間是有延遲的;
//有時候這種延遲並不是我們希望的,這時我們當然可以在無效化視窗區域後利用SendMessage 傳送一條WM_PAINT訊息來強制立即重畫。
註解:
SendMessage會block到被髮送的訊息被處理完才返回,但是WM_PAINT訊息的處理時間又是使用者不可控制的:“GetMessage returns the WM_PAINT message when there are no other messages in the application's message queue, and DispatchMessage sends the message to the appropriate window procedure. ”(MSDN原文),那麼也就是說,你呼叫SendMessage之後,這個方法需要等待多長時間才能返回是不可控制的。所以MSDN不推薦使用者直接傳送WM_PAINT訊息:“The WM_PAINT message is generated by the system and should not be sent by an application”
但不如使用Windows GDI為我們提供的更方便和強大的函式:
UpdateWindow和RedrawWindow。UpdateWindow會檢查視窗的Update Region,當其不為空時才傳送WM_PAINT訊息;
RedrawWindow則給我們更多的控制:是否重畫非客戶區和背景,是否總是傳送WM_PAINT訊息而不管Update Region是否為空等。
WM_PAINT觸發機制
如果WM_PAINT不是由InvalidateRect或InvalidateRgn產生時,先發WM_ERASEBKGND,再發WM_PAINT
如果WM_PAINT是由InvalidateRect或InvalidateRgn產生時,則先發WM_PAINT,
然後beginPaint()再根據Invalidate的bErase引數(重畫背景)來決定是否發WM_ERASEBKGND訊息
當WM_PAINT不是由InvalidateRect產生時,即由最大化,最小化等產生時,或者移動產生(移動有時只會產生WM_ERASEBKGND訊息)系統先發送WM_ERASEBKGND訊息,再發送WM_PAINT訊息.
當WM_PAINT由InvalidateRect()產生時,先發送WM_PAINT訊息(非同步),如果InvalidateRect的bErase為TRUE,BeginPaint檢查到更新區域需要刪除背景,向視窗傳送一個WM_ERASEBKGND訊息