转载请说明原出处,谢谢~~
近些天在duilib群里常常有朋友问起,怎么让duilib的IE控件能够去边框。去滚动栏的问题,或者是怎样去控件IE控件的行为。为了避免反复的回答,我就写一篇博文,把处理方法说明一下。
duilib中有Webbrowser控件,是继承ActivexUI控件后针对IE进行的封装。使用IE控件的话就用他了。
这个控件留了一个接口名为SetWebBrowserEventHandler,这个函数用了指定一个事件处理器,来控制IE的行为。
而这个函数须要一个CWebBrowserEventHandler对象指针,这个CWebBrowserEventHandler类在duilib的Utils文件夹中已经写好了,是一个主要的事件处理器框架,仅仅要重写这个CWebBrowserEventHandler类的GetHostInfo函数就能够控制去掉IE浏览器的边框和滚动栏,同一时候还有其它功能,比方控制能否够显示IE右键菜单,NavigateComplete2来截获浏览器载入完成的事件等等。
所以我们假设要去控制浏览器,那么正确的做法就是写一个类,继承CWebBrowserEventHandler然后重写你须要的函数。
我简单写了一个名叫CCustomWebEventHandler,代码例如以下:
#ifndef _CCUSTOM_WEBBROWSER_EVENT_HANDLER_H_
#define _CCUSTOM_WEBBROWSER_EVENT_HANDLER_H_ #pragma once
class CCustomWebEventHandler:public CWebBrowserEventHandler
{
public:
CCustomWebEventHandler() {}
~CCustomWebEventHandler() {} virtual void BeforeNavigate2( IDispatch *pDisp,VARIANT *&url,VARIANT *&Flags,VARIANT *&TargetFrameName,VARIANT *&PostData,VARIANT *&Headers,VARIANT_BOOL *&Cancel ) {}
virtual void NavigateError(IDispatch *pDisp,VARIANT * &url,VARIANT *&TargetFrameName,VARIANT *&StatusCode,VARIANT_BOOL *&Cancel) {}
virtual void NavigateComplete2(IDispatch *pDisp,VARIANT *&url){ } virtual void ProgressChange(LONG nProgress, LONG nProgressMax){}
virtual void NewWindow3(IDispatch **pDisp, VARIANT_BOOL *&Cancel, DWORD dwFlags, BSTR bstrUrlContext, BSTR bstrUrl){}
virtual void CommandStateChange(long Command,VARIANT_BOOL Enable){} // interface IDocHostUIHandler
virtual HRESULT STDMETHODCALLTYPE ShowContextMenu(
/* [in] */ DWORD dwID,
/* [in] */ POINT __RPC_FAR *ppt,
/* [in] */ IUnknown __RPC_FAR *pcmdtReserved,
/* [in] */ IDispatch __RPC_FAR *pdispReserved)
{
return S_OK;
//return S_FALSE
} virtual HRESULT STDMETHODCALLTYPE GetHostInfo(
/* [out][in] */ DOCHOSTUIINFO __RPC_FAR *pInfo)
{
if (pInfo != NULL)
{
pInfo->cbSize = sizeof(DOCHOSTUIINFO);
pInfo->dwDoubleClick = DOCHOSTUIDBLCLK_DEFAULT;
pInfo->dwFlags |= DOCHOSTUIFLAG_NO3DBORDER | DOCHOSTUIFLAG_THEME |
DOCHOSTUIFLAG_NO3DOUTERBORDER | DOCHOSTUIFLAG_DIALOG |
DOCHOSTUIFLAG_DISABLE_HELP_MENU;//| DOCHOSTUIFLAG_SCROLL_NO;; //这里还能够加其它代码来控制网页
//LPWSTR m_pZoom = L"BODY{Zoom:100%;}";
//pInfo->pchHostCss = (LPWSTR)::CoTaskMemAlloc((lstrlenW(m_pZoom)+1)*2);
//lstrcpyW(pInfo->pchHostCss, m_pZoom);
} return S_OK; } virtual HRESULT STDMETHODCALLTYPE ShowUI(
/* [in] */ DWORD dwID,
/* [in] */ IOleInPlaceActiveObject __RPC_FAR *pActiveObject,
/* [in] */ IOleCommandTarget __RPC_FAR *pCommandTarget,
/* [in] */ IOleInPlaceFrame __RPC_FAR *pFrame,
/* [in] */ IOleInPlaceUIWindow __RPC_FAR *pDoc)
{
return S_FALSE;
} virtual HRESULT STDMETHODCALLTYPE HideUI( void)
{
return S_OK;
} virtual HRESULT STDMETHODCALLTYPE UpdateUI( void)
{
return S_OK;
} virtual HRESULT STDMETHODCALLTYPE EnableModeless(
/* [in] */ BOOL fEnable)
{
return S_OK;
} virtual HRESULT STDMETHODCALLTYPE OnDocWindowActivate(
/* [in] */ BOOL fActivate)
{
return S_OK;
} virtual HRESULT STDMETHODCALLTYPE OnFrameWindowActivate(
/* [in] */ BOOL fActivate)
{
return S_OK;
} virtual HRESULT STDMETHODCALLTYPE ResizeBorder(
/* [in] */ LPCRECT prcBorder,
/* [in] */ IOleInPlaceUIWindow __RPC_FAR *pUIWindow,
/* [in] */ BOOL fRameWindow)
{
return S_OK;
} virtual HRESULT STDMETHODCALLTYPE TranslateAccelerator(
/* [in] */ LPMSG lpMsg,
/* [in] */ const GUID __RPC_FAR *pguidCmdGroup,
/* [in] */ DWORD nCmdID)
{
return S_OK;
} virtual HRESULT STDMETHODCALLTYPE GetOptionKeyPath(
/* [out] */ LPOLESTR __RPC_FAR *pchKey,
/* [in] */ DWORD dw)
{
return S_OK;
} virtual HRESULT STDMETHODCALLTYPE GetDropTarget(
/* [in] */ IDropTarget __RPC_FAR *pDropTarget,
/* [out] */ IDropTarget __RPC_FAR *__RPC_FAR *ppDropTarget)
{
return S_OK;
} virtual HRESULT STDMETHODCALLTYPE GetExternal(
/* [out] */ IDispatch __RPC_FAR *__RPC_FAR *ppDispatch)
{
return S_OK;
} virtual HRESULT STDMETHODCALLTYPE TranslateUrl(
/* [in] */ DWORD dwTranslate,
/* [in] */ OLECHAR __RPC_FAR *pchURLIn,
/* [out] */ OLECHAR __RPC_FAR *__RPC_FAR *ppchURLOut)
{
return S_OK;
} virtual HRESULT STDMETHODCALLTYPE FilterDataObject(
/* [in] */ IDataObject __RPC_FAR *pDO,
/* [out] */ IDataObject __RPC_FAR *__RPC_FAR *ppDORet)
{
return S_OK;
} // virtual HRESULT STDMETHODCALLTYPE GetOverrideKeyPath(
// /* [annotation][out] */
// __deref_out LPOLESTR *pchKey,
// /* [in] */ DWORD dw)
// {
// return E_NOTIMPL;
// } // IDownloadManager
virtual HRESULT STDMETHODCALLTYPE Download(
/* [in] */ IMoniker *pmk,
/* [in] */ IBindCtx *pbc,
/* [in] */ DWORD dwBindVerb,
/* [in] */ LONG grfBINDF,
/* [in] */ BINDINFO *pBindInfo,
/* [in] */ LPCOLESTR pszHeaders,
/* [in] */ LPCOLESTR pszRedir,
/* [in] */ UINT uiCP)
{
return S_OK;
}
}; #endif //_CCUSTOM_WEBBROWSER_EVENT_HANDLER_H_
使用他的方法例如以下:
首先在xml中写入一个Webbrowser控件而且在c++代码中通过FindControl找到这个控件的指针,然后写类似的代码:
CWebBrowserUI* pActiveXUI = static_cast<CWebBrowserUI*>(m_PaintManager.FindControl(_T("ActiveXDemo1"))); if( pActiveXUI )
{
pActiveXUI->SetDelayCreate(false);
CCustomWebEventHandler *pWebHandle = new CCustomWebEventHandler;
pActiveXUI->SetWebBrowserEventHandler(pWebHandle);
pActiveXUI->Navigate2(L"about:blank"); //这行代码。假设凝视掉,就不会去掉边框,IE有bug,第二次载入网页才会让事件处理器有效
pActiveXUI->Navigate2(L"http://www.kugou.com/");
}
这样就能够过滤边框了,效果例如以下:
watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvemh1aG9uZ3NodQ==/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="">
我这里强调两点
第一、我这里仅仅是示范。为了方便。我使用的是alberl的demo。简单改了几行代码。能够看到我new了一个CCustomWebEventHandler。可是没有去delete。所以自己使用时千万要注意代码规范!!
第二、先载入了一个blank页面,再跳转到目标页面,第一次载入页面不会触发事件处理器,第二次才会。为了不影响效率我直接载入blank。关于这个bug的说明,在微软官网有。详细地址我忘了~~
整个demo的project源代码为:点击打开链接
如有问题。请在博客留言给我
Redrain 2014.10.23