文章來源:網路作者:ggg82/CSDN
現在許多使用者介面都使用工具列製作選單條,小弟最近對此感興趣,便從網上求助,可是得到的幫助大多是BCGControlBar的源代碼或者是SizableRebar的源代碼,對於只希望是自己的界面具有該功能的朋友來說,這也許是不錯的選擇,只要看一下demo,然後直接調用別人的類庫就可以了,但對於我等對此主題感興趣,希望弄懂其來龍去脈的讀者來說,直接看這些沒有詳細解釋的源代碼,要從中弄出個所以然來,實不是件容易的是,至少對於像我這樣的菜鳥來說是這樣的,本文出於此種原因,希望對還在尋求此幫助的讀者能提供一些幫助。
下面我們邊看邊侃:
在接收到toolbarbutton按下訊息時,我們一般使用TrackPopupMenuEx彈出選單,問題的關鍵是,在選單未關閉時,TrackPopupMenuEx並不返回,並攔截滑鼠和鍵盤訊息,使用spy可以看到,此時的工具列收不到任何訊息,當然無從改變熱點,這就需要我們自己探測滑鼠位置並在滑鼠移動到下一個熱點時關閉上一個選單並顯示下一個選單。這裡我們使用鉤子函數SetWindowsHookEx在呼叫TrackPupupMenuEx前安裝WH_MSGFILTER鉤子,程式碼如下:
m_hMsgHook = SetWindowsHookEx( WH_MSGFILTER, MessageProc, 0, GetCurrentThreadId() );
MssageProc是鉤子函數,程式碼如下:
LRESULT CALLBACK MessageProc(int code, WPARAM wParam, LPARAM lParam)
{
if (code == MSGF_MENU)
{
HookMessageProc(lParam);
}
return CallNextHookEx(m_hMsgHook, code, wParam, lParam);
}
函數檢查訊息,如果是來自選單,則將訊息傳遞給函數HookMessageProc處理,我們所要做的就是在該函數中檢測訊息WM_MOUSEMOVE,並測試滑鼠位置,如果滑鼠已經移動到另一個按鈕上,則關閉選單並顯示下一個選單,關閉選單使用訊息WM_CANCELMODE,當選單關閉後,我們要釋放鉤子,在下一個選單彈出時重新安裝鉤子,彈出選單範例程式碼如下:
void TrackPopup(HWND hWndToolBar, int iButton)
{
while (iButton >= 0)
{
SendMessage(hWndToolBar,TB_SETHOTITEM,iButton,0);
iPopup = iButton;
//安裝鉤子
g_hMsgHook = SetWindowsHookEx(WH_MSGFILTER, MessageProc, 0, GetCurrentThreadId());
//彈出式選單
TrackPopupMenuEx(…);
//卸載鉤子
UnhookWindowsHookEx(g_hMsgHook);
iButton = iNextPop; //下一個彈出項,若為負,則退出
}
SendMessage(hWndToolBar,TB_SETHOTITEM,-1,0);
}
(經驗與建議:如果button使用樣式TBSTYLE_DROPDOWN,請不要在訊息TBN_DROPDOWN中直接呼叫函數,應使用中間訊息,然後使用PostMessa個發送該訊息,以使TBN_DROPDOWN可以直接返回,否則消除第一個高亮熱點是很麻煩的事。
iPopup為目前彈出項,iNextPop為下一個彈出項,這些變數需要在函數HookMessageProc中處理,範例程式碼如下:
void HookMessageProc(MSG * pMsg)
{
if (pMsg->message == WM_MOUSEMOVE)
{
int iButton, iCount;
POINT pt = { LOWORD(pMsg->lParam), HIWORD(pMsg->lParam) };
ScreenToClient(hWndToolbar, &pt);
iButton = SendMessage(hWndToolbar, TB_HITTEST, 0, &pt);
iCount = SendMessage(hWndToolbar, TB_BUTTONCOUNT, 0, 0);
if (iPopup != iButton && iButton < iCount && iButton >= 0)
{
iNextPop = iButton;
SendMessage(hWndMain, WM_CANCELMODE, 0, 0);
}
else
{
iNextPop = -1;
}
}
}
(經驗與建議:不要嘗試在此處調用TrackPopup,我曾試圖取消該函數內的while循環,直接在此調用該函數,結果是在TrackPopupMenuEx未返回之前,該函數已被調用)
這裡,僅僅處理了滑鼠移動訊息,真正的選單還應處理鍵盤導航訊息,詳細的程式碼可以參考
BCGControlBar(http://www.vckbase.com/code/downcode.asp?id=1382)
或SizableRebar(http://www.codeproject.com/docking/sizablerebar/SizableRebar_demo.zip
)
有了這個底層框架,這些處理過程應該不再困難,文章所涉及的一些API函數可以參考msdn。
Msdn上相關資料:
http://msdn.microsoft.com/library/default.asp?url=/library/en-us/shellcc/platform/commctls/faq/iemenubar.asp
http://www.microsoft.com/msj/0199/c/c0199.aspx