本文例子是一個(gè)典型的C++/MFC對(duì)話框程序,設(shè)置了 EX_WM_TOOLWINDOW 擴(kuò)展式樣,因此在標(biāo)題欄左上角看不到系統(tǒng)菜單圖標(biāo),但通過(guò) Ctrl+Space 或者在標(biāo)題欄單擊鼠標(biāo)右鍵可以調(diào)出系統(tǒng)菜單。例子程序?qū)ο到y(tǒng)菜單進(jìn)行了定制,在原有菜單基礎(chǔ)上添加了兩個(gè)菜單命令:一個(gè)是顯示“關(guān)于”對(duì)話框;另一個(gè)是“退出”。之所以要加一個(gè)“退出”菜單命令,是因?yàn)槔映绦蚋膶懥藢?duì)話框原來(lái)默認(rèn)的“關(guān)閉”菜單命令行為(Alt-F4),用來(lái)隱藏對(duì)話框。因此必須加一個(gè)程序的“退出”出口。此外,例子程序利用一個(gè)第三方的系統(tǒng)托盤處理類,利用系統(tǒng)托盤圖標(biāo)可以顯示/隱藏對(duì)話框。 下面我們就來(lái)看看用 C++/MFC 實(shí)現(xiàn)的細(xì)節(jié)。
添加菜單
首先在資源定義文件 resource.h 中定義菜單項(xiàng)標(biāo)示,也可以在標(biāo)準(zhǔn)頭文件中定義。菜單項(xiàng)標(biāo)示必須具有唯一性。其次,Windows 對(duì)系統(tǒng)菜單的處理與常規(guī)菜單的處理方法是不同的,不管是缺省的菜單還是定制的菜單,它們都沒(méi)有象常規(guī)菜單命令那樣的消息處理例程。假設(shè)我們要添加兩個(gè)定制的系統(tǒng)單:
#define IDM_ABOUT 16
#define IDM_EXIT 17
IDM_的意思是該定義為菜單項(xiàng)ID。添加菜單命令是在對(duì)話框的初始化例程以及窗口創(chuàng)建函數(shù)(OnInitDialog(), OnCreate())中進(jìn)行的。如: BOOL CBabelOnDlg::OnInitDialog()
{
CDialog::OnInitDialog();
// 在系統(tǒng)菜單中添加 "關(guān)于..." 和 "退出" 菜單項(xiàng)
// 解決 Windows 95 中的 bug
ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
// 命令 IDs 必須在預(yù)定義的系統(tǒng)菜單之后
ASSERT(IDM_ABOUTBOX < 0xF000);
// 解決 Windows 95 中的 bug
ASSERT((IDM_EXIT & 0xFFF0) == IDM_EXIT);
// 命令 IDs 必須在預(yù)定義的系統(tǒng)菜單之后
ASSERT(IDM_EXIT < 0xF000);
CMenu* pSysMenu = GetSystemMenu(FALSE);
if (pSysMenu != NULL)
{
pSysMenu->AppendMenu(MF_STRING,IDM_EXIT,"退出(&x)");
pSysMenu->AppendMenu(MF_SEPARATOR);
pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, "關(guān)于(&A)...");
......
}
......
//other initialization
}
這里在添加每個(gè)菜單前都有兩個(gè) ASSERT 語(yǔ)句,第一個(gè) ASSERT 的目的是修復(fù) Windows 95 中存在的 Bug,第二個(gè) ASSERT 保證定制的命令 IDs 是在預(yù)定義的系統(tǒng)菜單之后,以免發(fā)生沖突。查一下 MSDN 庫(kù)的 MFC 文檔關(guān)于系統(tǒng)菜單的描述,你會(huì)發(fā)現(xiàn)下面的內(nèi)容: “......所有預(yù)定義的控制菜單項(xiàng)(也就是系統(tǒng)菜單)的ID號(hào)必須大于 0xF000。如果某個(gè)應(yīng)用程序要添加系統(tǒng)菜單,
其系統(tǒng)菜單的 ID 號(hào)必須小于F000。”
接下來(lái),用 GetSystemMenu 函數(shù)獲取系統(tǒng)菜單指針。調(diào)用時(shí)使用參數(shù) FALSE 獲取指針。如果用 TRUE 作為參數(shù),那么該函數(shù)會(huì)將菜單重置回缺省狀態(tài)。
如果得到的指針有效,接著調(diào)用菜單添加命令在系統(tǒng)菜單后面添加菜單項(xiàng),傳遞菜單IDs以及菜單顯示時(shí)所用的字符串。
處理定制的菜單命令
為了讓這些系統(tǒng)菜單命令工作起來(lái),我們不能依賴常規(guī)的菜單消息處理機(jī)制----即便菜單項(xiàng)相同。通常系統(tǒng)菜單通過(guò) WM_SYSCOMMAND 消息處理: void CBabelOnDlg::OnSysCommand(UINT nID, LPARAM lParam)
{
//trap our own system menu messages
if ((nID & 0xFFF0) == IDM_ABOUTBOX)
{
CAboutDlg dlgAbout;
dlgAbout.DoModal();
} else if ((nID & 0xFFF0)==SC_CLOSE){
OnClose();
} else if ((nID & 0xFFF0)==IDM_EXIT) {
::PostQuitMessage(0);
}
else {
CDialog::OnSysCommand(nID, lParam);
}
}
通過(guò)比較傳入的菜單ID進(jìn)行相應(yīng)的處理。注意代碼中又有兩個(gè)“nID & 0xFFF0”,這主要也是解決 Windows 95 的 bug。如果選擇“退出”,那么會(huì)向應(yīng)用程序發(fā)送退出消息:::PostQuitMessage(0)。
注意第二個(gè)條件檢查:SC_CLOSE 是個(gè)預(yù)定義的菜單常量。一般它是由 Windows 處理的,因?yàn)樵诶映绦蛑形覀儗?duì)它進(jìn)行了定制,所以必須要自己處理它。本來(lái) SC_CLOSE 是退出程序,但例子程序我們把它的行為改寫成隱藏對(duì)話框,也就是將應(yīng)用變成一個(gè)托盤小圖標(biāo),處理例程見(jiàn) OnClose() 函數(shù)。如果傳入的菜單ID不等于任何定制的菜單項(xiàng),那么就讓 Windows 對(duì)它進(jìn)行默認(rèn)處理: CDialog::OnSysCommand(nID, lParam);
下面是幾個(gè)最常用的系統(tǒng)菜單命令:
菜單 說(shuō)明
SC_CLOSE 關(guān)閉 CWnd 對(duì)象
SC_MAXIMIZE 或者 SC_ZOOM 最大化 CWnd 對(duì)象
SC_MINIMIZE 或者 SC_ICON 最小化 CWnd 對(duì)象
SC_MOVE 移動(dòng) CWnd 對(duì)象
SC_RESTORE 恢復(fù)窗口的正常位置和大小
SC_SIZE 改變 CWnd 對(duì)象大小
其它的幾個(gè)系統(tǒng)菜單命令一般都是在特殊情況下才使用,有關(guān)細(xì)節(jié)請(qǐng)參考有關(guān) WM_SYSCOMMAND 的文檔。
修改現(xiàn)有的菜單命令
我們已經(jīng)看到,系統(tǒng)菜單本身默認(rèn)的處理行為是可以改變的,除此之外,系統(tǒng)菜單項(xiàng)的描述文本也是可以改變的,甚至還可以刪除它們。為了修改才單命令的描述文本,我們可以用 pSysMenu 指針調(diào)用 ModifyMenu() 函數(shù)。例如,如果想要把“關(guān)閉”菜單項(xiàng)改成“隱藏”,可以象下面這么做:
pSysMenu->ModifyMenu(SC_CLOSE, MF_BYCOMMAND,IDM_HIDE, "隱藏(&H)");
MF_BYCOMMAND 參數(shù)告訴該函數(shù)將 SC_CLOSE 解釋為命令 ID。IDM_HIDE 是新的菜單 ID。最后一個(gè)參數(shù)是菜單項(xiàng)的說(shuō)明文本。還有一種調(diào)用 ModifyMenu() 的方法是使用 菜單項(xiàng)索引作為參數(shù):
pSysMenu->ModifyMenu(0,MF_BYPOSITION,IDM_HIDE,"隱藏(&H)");
第一個(gè)參數(shù) 0 表示菜單項(xiàng)的索引,指第一個(gè)菜單。
刪除菜單命令
例子程序擬將去掉系統(tǒng)菜單中的窗口“關(guān)閉”命令,暫且不說(shuō)這樣做是否合適,但是我能做到這一點(diǎn):
pSysMenu->RemoveMenu(SC_CLOSE,MF_BYCOMMMAND);
pSysMenu->RemoveMenu(0,MF_BYPOSITION);
第一行代碼刪除了與 SC_CLOSE 關(guān)聯(lián)的菜單命令。而第二行代碼表示刪除系統(tǒng)菜單命令中的第一項(xiàng)。
用這種方式修改系統(tǒng)菜單盡管限定了應(yīng)用程序的某些行為,但對(duì)于小型應(yīng)用和實(shí)用程序來(lái)說(shuō)有時(shí)是很有用的,尤其是當(dāng)你想要從任務(wù)欄存取菜單命令時(shí)----也就是程序在后臺(tái)運(yùn)行或者以最小化方式運(yùn)行,右鍵單擊任務(wù)欄圖標(biāo)將彈出系統(tǒng)菜單