Toolbar制作菜單條過程詳解
發表于:2007-07-14來源:作者:點擊數:
標簽:
現在許多用戶界面都使用工具欄制作菜單條,小弟最近對此感興趣,便從網上求助,可是得到的幫助大多是BCGControlBar的源代碼或者是SizableRebar的源代碼,對于只希望是自己的界面具有該功能的朋友來說,這也許是不錯的選擇,只要看一下demo,然后直接調用別人
現在許多用戶界面都使用工具欄制作菜單條,小弟最近對此感興趣,便從網上求助,可是得到的幫助大多是BCGControlBar的源代碼或者是SizableRebar的源代碼,對于只希望是自己的界面具有該功能的朋友來說,這也許是不錯的選擇,只要看一下demo,然后直接調用別人的類庫就可以了,但對于我等對此話題感興趣,希望弄懂其來龍去脈的讀者來說,直接看這些沒有詳細解釋的源代碼,要從中弄出個所以然來,實不是件容易的是,至少對于像我這樣的菜鳥來說是這樣的,本文出于此種原因,希望對還在尋求此幫助的讀者能提供一些幫助。
下面我們邊看邊侃:
在接收到toolbarbutton按下消息時,我們一般使用TrackPopupMenuEx彈出菜單,問題的關鍵是,在菜單未關閉時,TrackPopupMenuEx并不返回,并攔截鼠標和鍵盤消息,使用spy可以看到,此時的工具欄收不到任何消息,當然無從改變熱點,這就需要我們自己探測鼠標位置并在鼠標移動到下一個熱點時關閉上一個菜單并顯示下一個菜單。這里我們使用鉤子函數Set
WindowsHookEx在調用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);
(經驗與建議:不要試圖在此處調用TrackPopup,我曾試圖取消該函數內的while循環,直接在此調用該函數,結果是在TrackPopupMenuEx未返回之前,該函數已被調用)
}
else
{
iNextPop = -1;
}
}
}
這里,僅僅處理了鼠標移動消息,真正的菜單還應處理鍵盤導航消息,詳細的代碼可以參考
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/shell
clearcase/" target="_blank" >cc/platform/commctls/faq/iemenubar.asp
http://www.microsoft.com/msj/0199/c/c0199.aspx
原文轉自:http://www.kjueaiud.com