Internet Explorer有實在太多沒有公布的東西。上一篇文章《Internet Explorer 編程簡述(六)自訂瀏覽器操作功能表》提到的擷取“編碼”菜單的方法就是利用了瀏覽器的上層視窗“Shell DocObject View”的未公布的命令ID。本文將要介紹的是如何用這個ID把“編碼”菜單放到我們自己的菜單中來(如工具條上的“編碼”按鈕的下拉式功能表)。
上面指向IOleCommandTarget介面的智能指標spCT是從IDocHostUIHandler::ShowContextMenu的參數pcmdTarget得到的,它其實也可以從HTML文檔介面得到,這就是實現的關鍵。
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訊息處理函數中檢查“編碼”菜單
switch ( pNMToolBar->iItem )
{
......
case ID_VIEW_ENCODE://按下“編碼”按鈕
{
m_bIsEncodMenuPopup = true;
VERIFY( menu.LoadMenu( IDR_ENCODE ) );//IDR_ENCODE是預置的“編碼”菜單資源,內含任意一項佔位用的菜單
CMyHtmlView = GetActiveMyHtmlView();//檢查當前是否存在活動的瀏覽器視圖視窗
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);//丟掉設計時佔位用的菜單,替換為“編碼”菜單
}
}
}
}
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 編程簡述(五)調用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的,在此修改其狀態為Enabled
for ( UINT i=0; i
GetMenuItemCount(); i++ )
{
pPopupMenu->EnableMenuItem( pPopupMenu->GetMenuItemID( i ), MF_ENABLED | MF_BYCOMMAND );
}
}
}
這樣一來,原本只在瀏覽器操作功能表中出現的“編碼”菜單就出現在了我們自己的工具條按鈕下拉式功能表上,無需更多的處理,菜單狀態的改變,編碼的設定等,一切都教給瀏覽器自己去完成了。