WebBrowser (http://support.microsoft.com/kb/324419)

BUG: Scroll Bars and Borders Appear in Framesets When You Navigate Again in BeforeNavigate

function loadTOCNode(){}
Article ID:324419
Last Review:May 12, 2003
Revision:2.0
This article was previously published under Q324419
var sectionFilter = "type != 'notice' && type != 'securedata' && type != 'querywords'"; var tocArrow = "/library/images/support/kbgraphics/public/en-us/downarrow.gif"; var depthLimit = 10; var depth3Limit = 10; var depth4Limit = 5; var depth5Limit = 3; var tocEntryMinimum = 1; <noscript></noscript>

SYMPTOMS

loadTOCNode(1, 'symptoms');
An empty scroll bar and sometimes a border appear if the following conditions are true:
You display a frameset in an application that hosts the WebBrowser control. -and-

You navigate to a frame in the frameset elsewhere in the BeforeNavigate or the BeforeNavigate2 event handler.
These items should not appear in the frame.

Back to the top

RESOLUTION

loadTOCNode(1, 'resolution');
To work around the problem with the scroll bar, use one of the following methods:
In the HTML source for the frame page, manually add the "auto" or the "no" value to the scroll attribute in the <body> tag.
Add the scroll attribute dynamically through Dynamic HTML (DHTML).
Delay the navigation by posting a user-defined message and by performing the navigation in the user-defined message handler.
To work around the problem with the border, use one of the following methods:
Post a user-defined message, and then perform the navigation in the user-defined message handler.
Implement the IDocHostUIHandler interface, and then return DOCHOSTUIFLAG_NO3DBORDER in the GetHostInfo method.
For more information, see the "More Information" section of this article.

Back to the top

STATUS

loadTOCNode(1, 'status');
Microsoft has confirmed that this is a bug in the Microsoft products that are listed at the beginning of this article.

Back to the top

MORE INFORMATION

loadTOCNode(1, 'moreinformation');

Steps to Reproduce the Behavior

loadTOCNode(2, 'moreinformation'); NOTE: These steps are written for Microsoft Visual C++ 6.0.
1.Use the following code to create the frameset page, and then name the page frameset.htm:
<HTML>
	<FRAMESET rows="100%" cols="33%,33%,34%">
		<FRAME src="framesetChild.htm" frameborder="0" scrolling="0">
		</FRAME>
		<FRAME src="framesetChild.htm" frameborder="0" scrolling="0">
		</FRAME>
		<FRAME src="framesetChild.htm" frameborder="0" scrolling="0">
		</FRAME>
	</FRAMESET>
</HTML>
					
2.Copy frameset.htm to the Web server.
3.Use the following code to create a frame page, and then name the page framesetChild.htm:
<HTML>
	<body>
		This is a frame<br>
	</body>
</HTML>
					
4.Copy framesetChild.htm to the Web server.
5.Create a default Microsoft Foundation Classes (MFC) dialog-based application.
6.Right-click the dialog, and then click Insert ActiveX Control. Click Microsoft Web Browser Control.
7.To add a control data member for the WebBrowser control, follow these steps:
a. Open the Class Wizard, and then click the Member Variables tab.
b. Make sure that the dialog class is selected in the Class name list.
c. Click IDC_EXPLORER1 (which is the default ID of the WebBrowser control), and then click Add Variable.
d. You receive a message that states that the control has not been inserted into the project. Click OK to add the control to the project. Click OK again to accept the defaults for the CWebBrowser2 class in the Confirm Class dialog box.
e. Name your member variable m_webBrowser, and then click OK.
f. Close the Class Wizard.
8.To add the BeforeNavigate2 event handler, follow these steps:
a. Open the Class Wizard, and then click the Message Maps tab.
b. Make sure that the dialog class is selected in the Class name list.
c. Click IDC_EXPLORER1 in the Object IDs list, and then click BeforeNavigate2 in the Messages list.
d. Click Add Function to add the handler.
9.Add the following code:
void CMFCReproDlg::OnBeforeNavigate2Explorer1(LPDISPATCH pDisp, VARIANT FAR* URL, VARIANT FAR* Flags, VARIANT FAR* TargetFrameName, VARIANT FAR* PostData, VARIANT FAR* Headers, BOOL FAR* Cancel) 
{
	static int nCount = 0;

	nCount++;
	if (nCount == 2) // this should be the navigate for the first frame in frameset
	{
		IWebBrowser* pWB = NULL;
		HRESULT hr = pDisp->QueryInterface(IID_IWebBrowser, (void**)&pWB);
		COleVariant ve((long)0);
		pWB->Navigate(::SysAllocString(L"http://myserver/mydirectory/framesetChild.htm"), &ve, &ve, &ve, &ve);
		*Cancel = VARIANT_TRUE;
	}
}
					
10.Add the following code to navigate to the frame page in the end of the OnInitDialog function.
BOOL CMFCReproDlg::OnInitDialog()
{
	...
	m_webBrowser.Navigate("http://myserver/mydirectory/frameset.htm", NULL, NULL, NULL, NULL);
					
11.Build the application, and then run it. Notice that the first frame has a scroll bar and a border on the right side.

Back to the top

Remove the Scroll Bar

loadTOCNode(2, 'moreinformation'); To remove the scroll bar, use one of following methods:
Add scroll attribute value of "auto" or "no" to the framesetChild.htm page as follows:
<HTML>
	<body scroll="auto">
		This is a frame<br>
	</body>
</HTML>
					
Dynamically add the scroll attribute value of "auto" or "no" in your code through DHTML as follows:
#include <mshtml.h>
// For brevity, this code adds the attribute to all documents.
void CMFCReproDlg::OnDocumentCompleteExplorer1(LPDISPATCH pDisp, VARIANT FAR* URL) 
{
	HRESULT hr = S_OK;
	IWebBrowser2* pWB = NULL;

	hr = pDisp->QueryInterface(IID_IWebBrowser2, reinterpret_cast<void**>(&pWB));
	
	IDispatch* pDocDisp = NULL;
	hr = pWB->get_Document(&pDocDisp);
	
	if (pDocDisp)
	{
		VARIANT v;
		VariantInit(&v);

		IHTMLDocument2* pDoc = NULL;
		hr = pDocDisp->QueryInterface(IID_IHTMLDocument2, reinterpret_cast<void **>(&pDoc));

		IHTMLElement* pElement = NULL;
		hr = pDoc->get_body(&pElement);

		IHTMLBodyElement* pBodyElement = NULL;
		hr = pElement->QueryInterface(IID_IHTMLBodyElement, (void**)&pBodyElement);

		if (pBodyElement)
		{
			pBodyElement->put_scroll(::SysAllocString(L"auto"));
			pBodyElement->Release();
		}
		pElement->Release();
		pDoc->Release();
		pDocDisp->Release();
	}
	pWB->Release();
}

						
NOTE: These first two options only remove the scroll bar. The border may still persist.
Post a user-defined message, and then perform the navigation in the user-defined message handler to delay the navigation.

Add the following code to the header file:
class CMFCReproDlg : public CDialog
{
...
	afx_msg LRESULT OnMyMessage(WPARAM wParam, LPARAM lParam);
};
						
Add the following code to the implementation file:
#define WM_MYMESSAGE (WM_USER + 1)
BEGIN_MESSAGE_MAP(CMFCReproDlg, CDialog)
	//{{AFX_MSG_MAP(CMFCReproDlg)
	ON_WM_SYSCOMMAND()
	ON_WM_PAINT()
	ON_WM_QUERYDRAGICON()
	//}}AFX_MSG_MAP
	ON_MESSAGE(WM_MYMESSAGE, OnMyMessage)
END_MESSAGE_MAP()

struct CMyData
{
	IWebBrowser2* m_pWB;
	BSTR m_pUrl;
};

void CMFCReproDlg::OnBeforeNavigate2Explorer1(LPDISPATCH pDisp, VARIANT FAR* URL, VARIANT FAR* Flags, VARIANT FAR* TargetFrameName, VARIANT FAR* PostData, VARIANT FAR* Headers, BOOL FAR* Cancel) 
{
	static int nCount = 0;

	nCount++;
	if (nCount == 2) // this should be the navigate for the first frame in frameset
	{
        *Cancel = VARIANT_TRUE;
        CMyData *data = new CMyData;
		HRESULT hr = pDisp->QueryInterface(IID_IWebBrowser2, (void**)(&data->m_pWB));
        data->m_pUrl = ::SysAllocString(L"http://myserver/mydirectory/framesetChild.htm");
		PostMessage(WM_MYMESSAGE, (WPARAM)data, 0);
	}
}
LRESULT CMFCReproDlg::OnMyMessage(WPARAM wParam, LPARAM lParam)
{
    CMyData *data = (CMyData*)wParam;
    COleVariant ve((long)0);
    data->m_pWB->Navigate(data->m_pUrl, &ve, &ve, &ve, &ve);
    delete data;
	return 1;
}
					

Back to the top

Remove the Border

loadTOCNode(2, 'moreinformation'); To remove the borders, use one of following methods:
Post a user-defined message, and then perform the navigation in the user-defined message handler.
Follow the steps in Microsoft Knowledge Base article Q196835 to provide the custom control site in which you can add the IDocHostUIHandler interface. For additional information, click the article number below to view the article in the Microsoft Knowledge Base:
196835 (http://support.microsoft.com/kb/196835/EN-US/) HOWTO: Override the MFC Default Control Containment
After you implement all the functions, you must add DOCHOSTUIFLAG_NO3DBORDER to the DOCHOSTUIINFO stucture in the dwFlags field for the GetHostInfo method. It is beyond the scope of this article to provide the steps to implement IDocHostUIHandler.
NOTE: The border problem does not appear in an Active Template Library (ATL) container because the ATL class, CAxHostWindow, already implements the IDocHostUIHandler interface. By default, CAxHostWindow enables this flag.
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值