怎么样在对话框中显示网页并屏蔽IE的弹出式菜单?
在对话框中显示网页并不是个新鲜的话题,但要将IE的那个讨的厌的弹出式菜单去掉,可不是件容易的事,可能很多的程序员都作过尝试,显然地,捕获WM_RBUTTONDOWN,WM_RBUTTONUP,WM_CONTEXTMENU,消息以及重载PreTranslateMessage等方法,都不能取得很好的结果。
这一切都与IE的扩展接口IDocHostUIHandler有关,在此并不想对此接口作详细讲述,只针此我们的需求--去掉弹出式菜单,用一个例子作为注解。 第一、实现CHtmlCtrl。 CHtmlCtrl这个类是对话框中使用CHtmlView显示网页的必备。其实很简单,只要注意以下几点就行: 1.从CHtmlView派生。 2.构造函数必需public,因为CHtmlView的构造函数是protect的。 3.捕获WM_MOUSEACTIVATE消息,如下: int CHtmlCtrl::OnMouseActivate(CWnd* pDesktopWnd, UINT nHitTest, UINT message) { return CWnd::OnMouseActivate(pDesktopWnd, nHitTest, message); ^^^^ 注意:不是CHtmlView } 4.重载PostNcDestroy,并且什么也不做。 OK,没问题了,你现在可以在对话框中声明CHtmlCtrl变量,并且Create,Navigate,现在网页显示出来了,下一步。 第二、写一个奇怪的,有点像COM类对象的一个C++类(CMyUi)。 #include <mshtmhst.h> class CMyUi:public IDocHostUIHandler { public: BOOL m_vbEnableCtxMenus; CMyUi():m_vbEnableCtxMenus(FALSE){}// FALSE 表示不显示弹出式菜单 virtual ~CMyUi(){}STDMETHOD(ShowContextMenu)(DWORD dwID, POINT FAR* ppt, IUnknown FAR* pcmdtReserved,
IDispatch FAR* pdispReserved) { if (m_vbEnableCtxMenus == TRUE) // Show Context Menu return S_FALSE; else return S_OK; } STDMETHOD(GetHostInfo)(DOCHOSTUIINFO FAR *pInfo) { return E_NOTIMPL; } STDMETHOD(ShowUI)(DWORD dwID, IOleInPlaceActiveObject FAR* pActiveObject, IOleCommandTarget FAR* pCommandTarget, IOleInPlaceFrame FAR* pFrame, IOleInPlaceUIWindow FAR* pDoc) { return E_NOTIMPL; } STDMETHOD(HideUI)(void) { return E_NOTIMPL; } STDMETHOD(UpdateUI)(void) { return E_NOTIMPL; } STDMETHOD(EnableModeless)(BOOL fEnable) { return E_NOTIMPL; } STDMETHOD(OnDocWindowActivate)(BOOL fActivate) { return E_NOTIMPL; } STDMETHOD(OnFrameWindowActivate)(BOOL fActivate) { return E_NOTIMPL; } STDMETHOD(ResizeBorder)(LPCRECT prcBorder, IOleInPlaceUIWindow FAR* pUIWindow, BOOL fRameWindow) { return E_NOTIMPL; } STDMETHOD(TranslateAccelerator)(LPMSG lpMsg, const GUID FAR* pguidCmdGroup, DWORD nCmdID) {return E_NOTIMPL;
} STDMETHOD(GetOptionKeyPath)(LPOLESTR FAR* pchKey, DWORD dw) { return E_NOTIMPL; } STDMETHOD(GetDropTarget)(IDropTarget* pDropTarget, IDropTarget** ppDropTarget) { return E_NOTIMPL; } STDMETHOD(GetExternal)(IDispatch** ppDispatch) { return E_NOTIMPL; } STDMETHOD(TranslateUrl)(DWORD dwTranslate, OLECHAR* pchURLIn, OLECHAR** ppchURLOut) { return E_NOTIMPL; } STDMETHOD(FilterDataObject)(IDataObject* pDO, IDataObject** ppDORet) { return E_NOTIMPL; } STDMETHODIMP QueryInterface(REFIID iid, LPVOID* ppvObj){ if (iid == IID_IUnknown || iid == IID_IDocHostUIHandler) { *ppvObj = this; return S_OK; } return S_FALSE; } STDMETHODIMP_(ULONG) AddRef(){ return 1; } STDMETHODIMP_(ULONG) Release(){ return 1; } }; 这个类主要实现了一个IDocHostUIHandler,由于并不是一个真正的COM类对象,所以不用实现COM的其它内容。这一步的工作也很简单,不是吗?当然,要知道为什么,你得去看看MSDN上关于IDocHostUIHandler的内容。 第三、拼在一起。 1.在上面实现的CHtmlCtrl加入公有成员IDocHostUIHandler* m_ui。 2.在CHtmlCtrl的构造函数中加入 CMyUi* t = new CMyUi; t->QueryInterface(IID_IDocHostUIHandler, (void**)&m_ui); 3.重载OnNavigateComplete2,加入以下代码: { CHtmlView::OnNavigateComplete2(strURL); IDispatch* pDoc = GetHtmlDocument(); if (pDoc == NULL) { pDoc->Release(); return; } ICustomDoc* pDoc2=NULL; HRESULT hresult = pDoc->QueryInterface(IID_ICustomDoc, (void**)&pDoc2); if (FAILED(hresult)) { pDoc->Release(); return; } pDoc->Release(); pDoc2->SetUIHandler(m_ui); pDoc2->Release(); } 大功告成,要是不出什么意外的话。 要是有BUG,或是出不来网页,没太大问题,用你的智慧与一点时间解决它。