mfc編程入門教程
1.MFC窗口類CWnd
在Windows系統(tǒng)里,一個窗口的屬性分兩個地方存放:一部分放在“窗口類”里頭,如上所述的在注冊窗口時指定;另一部分放在Windows Object本身,如:窗口的尺寸,窗口的位置(X,Y軸),窗口的Z軸順序,窗口的狀態(tài)(ACTIVE,MINIMIZED,MAXMIZED,RESTORED…),和其他窗口的關(guān)系(父窗口,子窗口…),窗口是否可以接收鍵盤或鼠標消息,等等。
為了表達所有這些窗口的共性,MFC設(shè)計了一個窗口基類CWnd。有一點非常重要,那就是CWnd提供了一個標準而通用的MFC窗口過程,MFC下所有的窗口都使用這個窗口過程。至于通用的窗口過程卻能為各個窗口實現(xiàn)不同的操作,那就是MFC消息映射機制的奧秘和作用了。這些,將在后面有關(guān)章節(jié)詳細論述。
CWnd提供了一系列成員函數(shù),或者是對Win32相關(guān)函數(shù)的封裝,或者是CWnd新設(shè)計的一些函數(shù)。這些函數(shù)大致如下。
方法/步驟
?。?)窗口創(chuàng)建函數(shù)
這里主要討論函數(shù)Create和CreateEx。它們封裝了Win32窗口創(chuàng)建函數(shù)::CreateWindowEx。Create的原型如下:
BOOL CWnd::Create(LPCTSTR lpszClassName,
LPCTSTR lpszWindowName, DWORD dwStyle,
const RECT& rect,
CWnd* pParentWnd, UINT nID,
CCreateContext* pContext)
Create是一個虛擬函數(shù),用來創(chuàng)建子窗口(不能創(chuàng)建桌面窗口和POP UP窗口)。CWnd的基類可以覆蓋該函數(shù),例如邊框窗口類等覆蓋了該函數(shù)以實現(xiàn)邊框窗口的創(chuàng)建,視類則使用它來創(chuàng)建視窗口。
Create調(diào)用了成員函數(shù)CreateEx。CWnd::CreateEx的原型如下:
BOOL CWnd::CreateEx(DWORD dwExStyle, LPCTSTR lpszClassName,
LPCTSTR lpszWindowName, DWORD dwStyle,
int x, int y, int nWidth, int nHeight,
HWND hWndParent, HMENU nIDorHMenu, LPVOID lpParam)
CreateEx有11個參數(shù),它將調(diào)用::CreateWindowEx完成窗口的創(chuàng)建,這11個參數(shù)對應(yīng)地傳遞給::CreateWindowEx。參數(shù)指定了窗口擴展風格、“窗口類”、窗口名、窗口大小和位置、父窗口句柄、窗口菜單和窗口創(chuàng)建參數(shù)。
CreateEx的處理流程將在后面4.4.1節(jié)討論窗口過程時分析。
窗口創(chuàng)建時發(fā)送WM_CREATE消息,消息參數(shù)lParam指向一個CreateStruct結(jié)構(gòu)的變量,該結(jié)構(gòu)有11個域,其描述見后面4.4.1節(jié)對窗口過程的分析,Windows使用和CreateEx參數(shù)一樣的內(nèi)容填充該變量
(2)窗口銷毀函數(shù)
例如:
DestroyWindow函數(shù) 銷毀窗口
PostNcDestroy( ),銷毀窗口后調(diào)用,虛擬函數(shù)
3)用于設(shè)定、獲取、改變窗口屬性的函數(shù),例如:
SetWindowText(CString tiltle) 設(shè)置窗口標題
GetWindowText() 得到窗口標題
SetIcon(HICON hIcon, BOOL bBigIcon);設(shè)置窗口像標
GetIcon( BOOL bBigIcon ) ;得到窗口像標
GetDlgItem( int nID);得到窗口類指定ID的控制子窗口
GetDC(); 得到窗口的設(shè)備上下文
SetMenu(CMenu *pMenu); 設(shè)置窗口菜單
GetMenu();得到窗口菜單
…
(4)用于完成窗口動作的函數(shù)
用于更新窗口,滾動窗口,等等。一部分成員函數(shù)設(shè)計成或可重載(Overloaded)函數(shù),或虛擬(Overridden)函數(shù),或MFC消息處理函數(shù)。這些函數(shù)或者實現(xiàn)了一部分功能,或者僅僅是一個空函數(shù)。如:
有關(guān)消息發(fā)送的函數(shù):
SendMessage( UINT message,WPARAM wParam = 0, LPARAM lParam = 0 );
給窗口發(fā)送發(fā)送消息,立即調(diào)用方式
PostMessage(( UINT message,WPARAM wParam = 0, LPARAM lParam = 0 );
給窗口發(fā)送消息,放進消息隊列
…
有關(guān)改變窗口狀態(tài)的函數(shù)
MoveWindow( LPCRECT lpRect, BOOL bRepaint = TRUE );
移動窗口到指定位置
ShowWindow(BOOL );顯示窗口,使之可見或不可見
…
實現(xiàn)MFC消息處理機制的函數(shù):
virtual LRESULT WindowProc( UINT message, WPARAM wParam, LPARAM lParam ); 窗口過程,虛擬函數(shù)
virtual BOOL OnCommand( WPARAM wParam, LPARAM lParam );處理命令消息
…
消息處理函數(shù):
OnCreate( LPCREATESTRUCT lpCreateStruct );MFC窗口消息處理函數(shù),窗口創(chuàng)建時由MFC框架調(diào)用
OnClose();MFC窗口消息處理函數(shù),窗口創(chuàng)建時由MFC框架調(diào)用
…
其他功能的函數(shù)
CWnd的導出類是類型更具體、功能更完善的窗口類,它們繼承了CWnd的屬性和方法,并提供了新的成員函數(shù)(消息處理函數(shù)、虛擬函數(shù)、等等)。
常用的窗口類及其層次關(guān)系見圖1-1。
2.在MFC下創(chuàng)建一個窗口對象
MFC下創(chuàng)建一個窗口對象分兩步,首先創(chuàng)建MFC窗口對象,然后創(chuàng)建對應(yīng)的Windows窗口。在內(nèi)存使用上,MFC窗口對象可以在?;蛘叨眩ㄊ褂胣ew創(chuàng)建)中創(chuàng)建。具體表述如下:
創(chuàng)建MFC窗口對象。通過定義一個CWnd或其派生類的實例變量或者動態(tài)創(chuàng)建一個MFC窗口的實例,前者在棧空間創(chuàng)建一個MFC窗口對象,后者在堆空間創(chuàng)建一個MFC窗口對象。
調(diào)用相應(yīng)的窗口創(chuàng)建函數(shù),創(chuàng)建Windows窗口對象。
例如:在前面提到的AppWizard產(chǎn)生的源碼中,有CMainFrame(派生于CMDIFrame(SDI)或者CMDIFrameWnd(MDI))類。它有兩個成員變量定義如下:
CToolBar m_wndToolBar;
CStatusBar m_wndStatusBar;
當創(chuàng)建CMainFrame類對象時,上面兩個MFC Object也被構(gòu)造。
CMainFrame還有一個成員函數(shù)
OnCreate(LPCREATESTRUCT lpCreateStruct),
它的實現(xiàn)包含如下一段代碼,調(diào)用CToolBar和CStatusBar的成員函數(shù)Create來創(chuàng)建上述兩個MFC對象對應(yīng)的工具欄HWND窗口和狀態(tài)欄HWND窗口:
int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
…
if (!m_wndToolBar.Create(this) ||
?。_wndToolBar.LoadToolBar(IDR_MAINFRAME))
{
TRACE0(“Failed to create toolbar\n”);
return -1; // fail to create
}
if (!m_wndStatusBar.Create(this) ||
??!m_wndStatusBar.SetIndicators(indicators,
sizeof(indicators)/sizeof(UINT)))
{
TRACE0(“Failed to create status bar\n”);
return -1; // fail to create
}
…
}
關(guān)于工具欄、狀態(tài)欄將在后續(xù)有關(guān)章節(jié)作詳細討論。
在MFC中,還提供了一種動態(tài)創(chuàng)建技術(shù)。動態(tài)創(chuàng)建的過程實際上也如上所述分兩步,只不過MFC使用這個技術(shù)是由框架自動地完成整個過程的。通常框架窗口、文檔框架窗口、視使用了動態(tài)創(chuàng)建。介于MFC的結(jié)構(gòu),CFrameWnd和CView及其派生類的實例即使不使用動態(tài)創(chuàng)建,也要用new在堆中分配。理由見窗口的銷毀(2.2.5節(jié))。
至于動態(tài)創(chuàng)建技術(shù),將在下一章具體討論。
在Windows窗口的創(chuàng)建過程中,將發(fā)送一些消息,如:
在創(chuàng)建了窗口的非客戶區(qū)(Nonclient area)之后,發(fā)送消息WM_NCCREATE;
在創(chuàng)建了窗口的客戶區(qū)(client area)之后,發(fā)送消息WM_CREATE;
窗口的窗口過程在窗口顯示之前收到這兩個消息。
如果是子窗口,在發(fā)送了上述兩個消息之后,還給父窗口發(fā)送WM_PARENATNOTIFY消息。其他類或風格的窗口可能發(fā)送更多的消息,具體參見SDK開發(fā)文檔。
3.MFC窗口的使用
MFC提供了大量的窗口類,其功能和用途各異。程序員應(yīng)該選擇哪些類來使用,以及怎么使用他們呢?
直接使用MFC提供的窗口類或者先從MFC窗口類派生一個新的C++類然后使用它,這些在通常情況下都不需要程序員提供窗口注冊的代碼。是否需要派生新的C++類,視MFC已有的窗口類是否能滿足使用要求而定。派生的C++類繼承了基類的特性并改變或擴展了它的功能,例如增加或者改變對消息、事件的特殊處理等。
主要使用或繼承以下一些MFC窗口類(其層次關(guān)系圖見圖1-1):
框架類CFrameWnd,CMdiFrameWnd;
文檔框架CMdiChildWnd;
視圖CView和CView派生的有特殊功能的視圖如:列表CListView,編輯CEditView,樹形列表CTreeView,支持RTF的CRichEditView,基于對話框的視CFormView等等。
對話框CDialog。
通常,都要從這些類派生應(yīng)用程序的框架窗口和視窗口或者對話框。
工具條CToolBar
狀態(tài)條CStatusBar
其他各類控制窗口,如列表框CList,編輯框CEdit,組合框CComboBox,按鈕Cbutton等。
通常,直接使用這些類。
4.在MFC下窗口的銷毀
窗口對象使用完畢,應(yīng)該銷毀。在MFC下,一個窗口對象的銷毀包括HWND窗口對象的銷毀和MFC窗口對象的銷毀。一般情況下,MFC編程框架自動地處理了這些。
?。?)對CFrameWnd和CView的派生類
這些窗口的關(guān)閉導致銷毀窗口的函數(shù)DestroyWindow被調(diào)用。銷毀Windows窗口時,MFC框架調(diào)用的最后一個成員函數(shù)是OnNcDestroy函數(shù),該函數(shù)負責Windows清理工作,并在最后調(diào)用虛擬成員函數(shù)PostNcDestroy。CFrameWnd和CView的PostNcDestroy調(diào)用delete this刪除自身這個MFC窗口對象。
所以,對這些窗口,如前所述,應(yīng)在堆(Heap)中分配,而且,不要對這些對象使用delete操作。
?。?)對Windows Control窗口
在它們的析構(gòu)函數(shù)中,將調(diào)用DestroyWidnow來銷毀窗口。如果在棧中分配這樣的窗口對象,則在超出作用范圍的時候,隨著析構(gòu)函數(shù)的調(diào)用,MFC窗口對象和它的Windows window對象都被銷毀。如果在堆(Heap)中分配,則顯式調(diào)用delete操作符,導致析構(gòu)函數(shù)的調(diào)用和窗口的銷毀。
所以,這種類型的窗口應(yīng)盡可能在棧中分配,避免用額外的代碼來銷毀窗口。如前所述的CMainFrame的成員變量m_wndStatusBar和m_wndToolBar就是這樣的例子。
?。?)對于程序員直接從CWnd派生的窗口
程序員可以在派生類中實現(xiàn)上述兩種機制之一,然后,在相應(yīng)的規(guī)范下使用。
評論