用MFC的消息映像實(shí)現(xiàn)動(dòng)態(tài)菜單
當(dāng)我們提到動(dòng)態(tài)菜單的實(shí)現(xiàn)時(shí),我們通常的做法是使用GetMenu() 函數(shù)獲取一個(gè)Cmenu 類指針,然后調(diào)用CMenu 類方法AppendMenu, InsertMenu, ModifyMenu, RemoveMenu 等。本文介紹一種更加簡(jiǎn)潔的方法,它利用MFC 的消息映像機(jī)制及CCmdUI 類方法來(lái)實(shí)現(xiàn)。
首先,我們簡(jiǎn)要說(shuō)說(shuō)VC 中MFC 的消息映像。每個(gè)Windows 程序員大概都對(duì)以前使用的窗口函數(shù)WindowProc 記憶猶新,當(dāng)我們面對(duì)各種消息時(shí),我們別無(wú)他方,只能使用龐大而機(jī)械的switch-case 語(yǔ)句來(lái)實(shí)現(xiàn)不同的分支選擇。在VC5.0 中使用V4.2 版的MFC 基本類庫(kù),你將告別switch-case 語(yǔ)句,代之以透明的消息映像。要在一個(gè)類中使用消息映像,在類聲明中,必須顯式的加入宏DECLARE_MESSAGE_MAP:
class CMyClass: public CBaseClass
{
DECLARE_MESSAGE_MAP()
}
在類實(shí)現(xiàn)中,必須使用兩個(gè)宏BEGIN_MESSAGE_MAP 和END_MESSAGE_MAP,BEGIN_MESSAGE_MAP 帶兩個(gè)參數(shù):當(dāng)前類和直接父類:
BEGIN_MESSAGE_MAP(CMyClass, CBaseClass)
// 消息映像項(xiàng)
ON_COMMAND(ID_APP_ABOUT, OnAppAbout)
// 消息映像項(xiàng)
END_MESSAGE_MAP()
消息映像項(xiàng)使用下列基本語(yǔ)法:
ON_MessageName(ID, ClassMethod)
MessageName 是需要處理的消息,ID 是發(fā)送消息的標(biāo)識(shí)符,而ClassMethod 為處理此消息的類方法名。MessageName 是MFC 預(yù)定義的,可分為以下三種:
·命令消息
·子窗口通知消息
·Windows 消息
共一百多個(gè),用戶不必記住它們,因?yàn)橄⒂诚窨梢院芎?jiǎn)單的利用ClassWizard 加入。處理一個(gè)消息的類方法ClassMethod 必須在類定義中聲明,且有實(shí)現(xiàn)代碼。其原型為:
Afx_msg return_type ClassMethod(paras table)
類CCmdUI 專門(mén)(且僅僅)與ON_UPDATE_COMMAND_UI 消息映像宏配套使用,用于管理菜單(還有工具欄按扭等)的實(shí)時(shí)狀態(tài),如是否變灰,是否加選中標(biāo)記等。
ON_UPDATE_COMMAND_UI 消息映像宏原型為:
ON_UPDATE_COMMAND_UI(Menu_Item_ID, Menu_Proc)
ON_UPDATE_COMMAND_UI 消息映像宏將一個(gè)菜單項(xiàng)(命令項(xiàng))和一個(gè)更新處理過(guò)程聯(lián)結(jié),從而在適當(dāng)?shù)臅r(shí)機(jī)自動(dòng)調(diào)用此更新處理過(guò)程來(lái)完成對(duì)菜單項(xiàng)狀態(tài)的更新。
Menu_Item_ID 為菜單項(xiàng)的ID 號(hào),Menu_Proc 為此菜單項(xiàng)的更新處理函數(shù),為:
afx_msg void Menu_Proc (CCmdUI* pCmdUI)
它帶有一個(gè)CCmdUI 類指針,使用它可調(diào)用CCmdUI 的類方法。與菜單有關(guān)的類方法有:
·Enable(BOOL) 使菜單項(xiàng)有效或無(wú)效
·SetText(LPCTSTR) 設(shè)置菜單項(xiàng)的文本
·SetCheck(int) 加上或去掉選中標(biāo)記“X”
·SetRadio(BOOL) 加上或去掉選中標(biāo)記“.”
MenuProc 被調(diào)用的時(shí)機(jī)有以下幾種情況:
·用鼠標(biāo)選中包含該菜單項(xiàng)的菜單條
·用熱鍵選中包含該菜單項(xiàng)的菜單條
·用快捷鍵選中與該菜單項(xiàng)在同一菜單條下的任一菜單項(xiàng)
我們以下面菜單結(jié)構(gòu)為例:
Test menu
Item One ID_ITEM_ONE Ctrl+1
Item Two ID_ITEM_TWO Ctrl+2
Popup Popup One ID_POPUP_ONE Ctrl+3
Popup Two ID_POPUP_TWO Ctrl+4
當(dāng)用鼠標(biāo)左鍵點(diǎn)按Test menu 菜單條或按Alt+t 或按Ctrl+1/2/3/4 時(shí),四個(gè)菜單項(xiàng)的更新處理過(guò)程MenuProc 都將被調(diào)用。
當(dāng)我們考察上面這個(gè)具有嵌套結(jié)構(gòu)的菜單時(shí),我們面臨這樣一個(gè)問(wèn)題:菜單項(xiàng)Item One/Item Two 的更新函數(shù)和Popup One/Popup Two 的更新函數(shù)形式上是否一致?當(dāng)Popup One 和Popup Two 都變灰時(shí)Popup 是否自動(dòng)變灰?
根據(jù)MFC 的內(nèi)部機(jī)制,僅僅彈出菜單的第一項(xiàng)應(yīng)附加一些代碼,其余項(xiàng)的形式基本是一致的。也就是說(shuō)在上例中,除菜單項(xiàng)Popup One 外,其他菜單項(xiàng)更新函數(shù)的代碼基本一致,即根據(jù)條件,簡(jiǎn)單調(diào)用CCmdUI 類方法即可。菜單項(xiàng)Popup One 由于是彈出式菜單Popup 的第一項(xiàng),它的更新函數(shù)在以下兩種情況下都會(huì)被調(diào)用:
·當(dāng)彈出式菜單(Popup)的菜單項(xiàng)(Popup One 和Popup Two)要被繪出時(shí)
·當(dāng)此彈出式菜單即Popup 本身要被繪出時(shí)
第一種情況很好理解,正如我們選中Test menu 而Item One 和Item Two 的更新函數(shù)會(huì)自動(dòng)執(zhí)行一樣。第二種情況其實(shí)也很自然,因?yàn)镻opup 和Item One/Item Two 不一樣,它沒(méi)有ID 號(hào),不能添加消息映像項(xiàng),那么它的狀態(tài)如何更新呢?于是它的第一項(xiàng)的更新函數(shù)被調(diào)用,為了區(qū)分是不同的調(diào)用,它將CCmdUI 的類成員變量m_pSubMenu 設(shè)置為不同的值。在第一種情況下,m_pSubMenu 等于NULL, 第二種情況下,m_pSubMenu 不等于NULL。
以下我們給出一個(gè)實(shí)際的編程范例。由于篇幅關(guān)系,我們僅僅給出一些關(guān)鍵的語(yǔ)句,其余的則一并略去。
在頭文件的類聲明中:
BOOL m_bItemOne, m_bItemTwo, m_bPopupOne, m_bPopupTwo;
//用于決定各個(gè)菜單項(xiàng)的狀態(tài)
protected:
afx_msg void OnUpdateMenuitemOne(CCmdUI* pCmdUI);
afx_msg void OnUpdateMenuitemTwo(CCmdUI* pCmdUI);
afx_msg void OnUpdatePopupOne(CCmdUI* pCmdUI);
afx_msg void OnUpdatePopupTwo(CCmdUI* pCmdUI);
//各菜單項(xiàng)的更新函數(shù)
DECLARE_MESSAGE_MAP()
在源文件中:
BEGIN_MESSAGE_MAP(CMyDoc, CDocument)
ON_UPDATE_COMMAND_UI (ID_ITEM_ONE,
OnUpdateMenuitemOne)
ON_UPDATE_COMMAND_UI (ID_ITEM_TWO,
OnUpdateMenuitemTwo)
ON_UPDATE_COMMAND_UI (ID_POPUP_ONE,
OnUpdatePopupOne)
ON_UPDATE_COMMAND_UI (ID_ POPUP_TWO,
OnUpdatePopupTwo)
END_MESSAGE_MAP()
void CMyApp::OnUpdatetMenuitemOne (CCmdUI* pCmdUI)
{
pCmdUI->Enable(m_bItemOne);
if(m_bItemOne) pCmdUI->SetText("Item One");
else pCmdUI->SetText("Item One is now disabled");
}
void CMyApp::OnUpdatetMenuitemTwo (CCmdUI* pCmdUI)
{
pCmdUI->Enable(m_bItemTwo);
if(m_bItemTwo) pCmdUI->SetText("Item Two");
else pCmdUI->SetText("Item Two is now disabled");
}
void CMyApp::OnUpdatePopupOne(CCmdUI* pCmdUI)
{
if (pCmdUI->m_pSubMenu != NULL)
{
BOOL b_Popup = m_bPopupOne || m_bPopupTwo;
pCmdUI->m_pMenu->EnableMenuItem(pCmdUI->m_nIndex,
MF_BYPOSITION |
(bEnable ? MF_ENABLED :
(MF_DISABLED | MF_GRAYED)));
return;
}
pCmdUI->Enable(m_bPopupOne);
}
void CMyApp::OnUpdatePopupTwo(CCmdUI* pCmdUI)
{
pCmdUI->Enable(m_bPopupTwo);
}
來(lái)源:langen0次