1、概述
Internet Explorer有實在太多沒有公布的東西。上一篇文章《Internet Explorer
編程簡述(六)自定義瀏覽器上下文菜單》提到的獲取“編碼”菜單的方法就是利用了瀏覽器的上層窗口“Shell DocObject
View”的未公布的命令I(lǐng)D。本文將要介紹的是如何用這個ID把“編碼”菜單放到我們自己的菜單中來(如工具條上的“編碼”按鈕的下拉菜單)。
#define SHDVID_GETMIMECSETMENU 27
......
CComPtr spCT;
hr = pcmdTarget->QueryInterface(IID_IOleCommandTarget, (void**)&spCT);
......
// Get the language submenu
hr = spCT->Exec(&CGID_ShellDocView, SHDVID_GETMIMECSETMENU, 0, NULL, &var);
2、原理
上面指向IOleCommandTarget接口的智能指針spCT是從IDocHostUIHandler::ShowContextMenu的參數(shù)pcmdTarget得到的,它其實也可以從HTML文檔接口得到,這就是實現(xiàn)的關(guān)鍵。
3、實現(xiàn)
下面的代碼演示了如何將“編碼”菜單放置到我們自己的編碼菜單上去。
void CMainFrame::OnDropDown( NMHDR* pNotifyStruct, LRESULT* pResult )
{
const UINT CmdID_GetMimeSubMenu = 27;
// Command ID for getting the Encoding submenu
NMTOOLBAR* pNMToolBar = ( NMTOOLBAR* )pNotifyStruct;
CMenu menu;
CMenu* pPopup = 0;
CMyHtmlView *pView = NULL;
m_bIsEncodMenuPopup = false;//標志變量,用以在WM_INITMENUPOPUP消息處理函數(shù)中檢查“編碼”菜單
switch ( pNMToolBar->iItem )
{
......
case ID_VIEW_ENCODE://按下“編碼”按鈕
{
m_bIsEncodMenuPopup = true;
VERIFY( menu.LoadMenu( IDR_ENCODE ) );//IDR_ENCODE是預(yù)置的“編碼”菜單資源,內(nèi)含任意一項占位用的菜單
CMyHtmlView = GetActiveMyHtmlView();//檢查當(dāng)前是否存在活動的瀏覽器視圖窗口
if ( pView != NULL )
{
LPDISPATCH lpDispatch =pView->GetHtmlDocument();//獲得文檔指針
if ( lpDispatch != NULL )
{
// Get an IDispatch pointer for the IOleCommandTarget interface.
IOleCommandTarget * pCmdTarget = NULL;
HRESULT hr = lpDispatch->QueryInterface(IID_IOleCommandTarget, (void**)&pCmdTarget);
if ( SUCCEEDED( hr ) )
{
VARIANT varEncSubMenu;
::VariantInit( &varEncSubMenu );
hr = pCmdTarget->Exec( &::CGID_ShellDocView,
CmdID_GetMimeSubMenu, OLECMDEXECOPT_DODEFAULT, NULL, &varEncSubMenu
);
if ( SUCCEEDED( hr ) )
{
// 添加“編碼”菜單
MENUITEMINFO miiEncoding;
::memset( &miiEncoding, 0, sizeof(MENUITEMINFO) );
miiEncoding.cbSize = sizeof(MENUITEMINFO);
miiEncoding.fMask = MIIM_SUBMENU;
miiEncoding.hSubMenu = reinterpret_cast< HMENU > (varEncSubMenu.byref);
menu.SetMenuItemInfo(0, &miiEncoding, TRUE);//丟掉設(shè)計時占位用的菜單,替換為“編碼”菜單
}
}
}
}
pPopup = menu.GetSubMenu( 0 );
break;
}
......
}
if ( pPopup != 0 )
{
CRect rc;
::SendMessage( pNMToolBar->hdr.hwndFrom, TB_GETRECT, pNMToolBar->iItem, ( LPARAM )&rc );
rc.top = rc.bottom;
::ClientToScreen( pNMToolBar->hdr.hwndFrom, &rc.TopLeft() );
long lResult = pPopup->TrackPopupMenu( TPM_LEFTALIGN | TPM_LEFTBUTTON | TPM_RETURNCMD, rc.left, rc.top, this );
m_bIsEncodMenuPopup = false;
if ( pNMToolBar->iItem == ID_VIEW_ENCODE )
{
//其余的事教給瀏覽器去做,參考《Internet Explorer 編程簡述(五)調(diào)用IE隱藏的命令(中文版)》
CFindIEWnd FindIEWnd( pView->m_wndBrowser.m_hWnd, "Internet Explorer_Server");
::SendMessage( FindIEWnd.m_hWnd, WM_COMMAND, MAKEWPARAM(LOWORD(lResult), 0x0), 0 );
}
else
{
SendMessage( WM_COMMAND, MAKEWPARAM(LOWORD(lResult), 0x0), 0 );
}
}
*pResult = TBDDRET_DEFAULT;
}
void CMainFrame::OnInitMenuPopup(CMenu* pPopupMenu, UINT nIndex, BOOL bSysMenu)
{
CMDIFrameWndEx::OnInitMenuPopup(pPopupMenu, nIndex, bSysMenu);
if ( m_bIsEncodMenuPopup )
{
//默認情況下“編碼”的所有菜單項都是Disabled的,在此修改其狀態(tài)為Enabled
for ( UINT i=0; i
GetMenuItemCount(); i++ )
{
pPopupMenu->EnableMenuItem( pPopupMenu->GetMenuItemID( i ), MF_ENABLED | MF_BYCOMMAND );
}
}
}
這樣一來,原本只在瀏覽器上下文菜單中出現(xiàn)的“編碼”菜單就出現(xiàn)在了我們自己的工具條按鈕下拉菜單上,無需更多的處理,菜單狀態(tài)的改變,編碼的設(shè)置等,一切都教給瀏覽器自己去完成了。