在对话框中显示网页并不是个新鲜的话题,但要将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,或是出不来网页,没太大问题,用你的智慧与一点时间解决它。