敲代碼時(shí),突然發(fā)現(xiàn)有一個(gè)背景圖片無法顯示,百思不得其解,最終發(fā)現(xiàn)是MoveWindow() SetWindowPos()這兩個(gè)函數(shù)的使用不當(dāng)造成的。
這里把這兩個(gè)函數(shù)的前世今生給分析一下。
先看MoveWindow()函數(shù),用于改變窗口的位置和尺寸,對(duì)于頂層窗口,按照屏幕對(duì)齊與左上角,對(duì)于子窗口,則對(duì)齊與父窗口的左上角。
BOOL MoveWindow( HWND hWnd,int x, int y, int nWidth, int nHeight,BOOL bRepaint = TRUE);
第一個(gè)表示窗口句柄;
參數(shù)x,y表示窗口的左上角起點(diǎn);
參數(shù)nwidth,nHeight 表示窗口高度和寬度;
最后一個(gè)bRepaint表示是否立即重繪。為true時(shí)系統(tǒng)會(huì)立即發(fā)送WM_PAINT到窗口程序(會(huì)調(diào)用UpdateWindow()函數(shù)),為false時(shí)不會(huì)發(fā)生任何類型的重繪操作。
MoveWindow會(huì)給窗口發(fā)送WM_WINDOWPOSCHANGING,WM_WINDOWPOSCHANGED,WM_MOVE,WM_SIZE和WM_NCCALCSIZE消息。
接下來看SetWindowPos(),這個(gè)函數(shù)同樣用來實(shí)現(xiàn)窗口移動(dòng)操作,只不過功能更多一點(diǎn)。
BOOL SetWindowPos(HWND hWnd,const CWnd* pWndInsertAfter, int x, int y,int cx, int cy, UINT nFlags);
第一個(gè)參數(shù)是窗口句柄;
第二個(gè)參數(shù)是窗口的Z order,就是有多個(gè)窗口時(shí),本窗口顯示的順序,最上面或者最下面,可選參數(shù)有HWND_BOTTOM,HWND_NOTOPMOST,HWND_TOP,HWND_TOPMOST.
HWND_BOTTOM:將窗口置于Z序的底部。如果參數(shù)hWnd標(biāo)識(shí)了一個(gè)頂層窗口,則窗口失去頂級(jí)位置,并且被置在其他窗口的底部。
HWND_NOTOPMOST:將窗口置于所有非頂層窗口之上(即在所有頂層窗口之后)。如果窗口已經(jīng)是非頂層窗口則該標(biāo)志不起作用。
HWND_TOP:將窗口置于Z序的頂部。
HWND_TOPMOST:將窗口置于所有非頂層窗口之上。即使窗口未被激活窗口也將保持頂級(jí)位置。
接下來四個(gè)參數(shù)是窗口的位置和尺寸。
最后一個(gè)參數(shù)nFlags是窗口顯示方式的標(biāo)志,這個(gè)參數(shù)很多,可以組合使用,詳細(xì)如下:
SWP_ASNCWINDOWPOS:如果調(diào)用進(jìn)程不擁有窗口,系統(tǒng)會(huì)向擁有窗口的線程發(fā)出需求。這就防止調(diào)用線程在其他線程處理需求的時(shí)候發(fā)生死鎖。
SWP_DEFERERASE:防止產(chǎn)生WM_SYNCPAINT消息。
SWP_DRAWFRAME:在窗口周圍畫一個(gè)邊框(定義在窗口類描述中)。
SWP_FRAMECHANGED:給窗口發(fā)送WM_NCCALCSIZE消息,即使窗口尺寸沒有改變也會(huì)發(fā)送該消息。如果未指定這個(gè)標(biāo)志,只有在改變了窗口尺寸時(shí)才發(fā)送WM_NCCALCSIZE。
SWP_HIDEWINDOW;隱藏窗口。
SWP_NOACTIVATE:不激活窗口。如果未設(shè)置標(biāo)志,則窗口被激活,并被設(shè)置到其他最高級(jí)窗口或非最高級(jí)組的頂部(根據(jù)參數(shù)hWndlnsertAfter設(shè)置)。
SWP_NOCOPYBITS:清除客戶區(qū)的所有內(nèi)容。如果未設(shè)置該標(biāo)志,客戶區(qū)的有效內(nèi)容被保存并且在窗口尺寸更新和重定位后拷貝回客戶區(qū)。
SWP_NOMOVE:維持當(dāng)前位置(忽略X和Y參數(shù))。
SWP_NOOWNERZORDER:不改變z序中的所有者窗口的位置。
SWP_NOREDRAW:不重畫改變的內(nèi)容。如果設(shè)置了這個(gè)標(biāo)志,則不發(fā)生任何重畫動(dòng)作。適用于客戶區(qū)和非客戶區(qū)(包括標(biāo)題欄和滾動(dòng)條)和任何由于窗回移動(dòng)而露出的父窗口的所有部分。如果設(shè)置了這個(gè)標(biāo)志,應(yīng)用程序必須明確地使窗口無效并重畫窗口的任何部分和父窗口需要重畫的部分。
SWP_NOREPOSITION;與SWP_NOOWNERZORDER標(biāo)志相同。
SWP_NOSENDCHANGING:防止窗口接收WM_WINDOWPOSCHANGING消息。
SWP_NOSIZE:維持當(dāng)前尺寸(忽略cx和Cy參數(shù))。
SWP_NOZORDER:維持當(dāng)前Z序(忽略hWndlnsertAfter參數(shù))。
SWP_SHOWWINDOW:顯示窗口。
注意事項(xiàng):
使用SetWindowPos()如果設(shè)置了SWP_SHOWWINDOWS或者SWP_HIDEWINDOW,那么窗口將不能被移動(dòng)和改變大小,我使用時(shí)就是設(shè)置了SWP_SHOWWINDOW,從而導(dǎo)致不能重繪背景。
兩者的比較:
兩者都能實(shí)現(xiàn)移動(dòng)和改變窗口大小。
MoveWindow()功能比較單一,只能移動(dòng)和改變窗口大小,而SetWindowPos()可以設(shè)置更多的參數(shù),實(shí)現(xiàn)Zorder及顯示方式標(biāo)志。
MoveWindow()發(fā)送WM_WINDOWPOSCHANGING,WM_WINDOWPOSCHANGED,WM_MOVE,WM_SIZE和WM_NCCALCSIZE等消息到窗口,SetWindowPos()只發(fā)送WM_WINDOWPOSCHANGED消息到窗口。所以說如果需要發(fā)送更多消息時(shí)還是使用MoveWindow()的好。
|
新聞熱點(diǎn)
疑難解答
圖片精選