Environment: VC6 SP4, NT4 SP5, WIN 2000/98/95
Here is an easy solution for hiding/replacing the right-click menu on a Web browser window. An example would be my situation. I had to implement a news group reader and writer. The news reader/writer used IwebBrowser2 control as the message display/edit window. If you do a right-click on it, there is an option View Source,.. other menu items. If I had an option, I want to act like I implemented everything from scratch, but the right-click menu put a hole in my plan and revealed that I reused the IE control. Besides, anyway, we didn’t want our users to mess around with HTML source code, and so forth, or expose some of them to brain damage from seeing real weird stuff while all they asked for was some news group item. The Web browser captures the right-click and shows the default IE right click menu. So how do I remove it?
I trap the PreTranslateMessage function of the application class (CWinApp derived class). In case of a right click, I will see if it is a Web browser control which generated the message, and if so, I display our menu. Here is our example.
BOOL CWebBrowserTestApp::PreTranslateMessage(MSG* pMsg)
// We have a right click
if ( (WM_RBUTTONDOWN == pMsg->message) ||
(WM_RBUTTONDBLCLK == pMsg->message))
// Now verify it is from our Web browser
CWebBrowserTestDlg* pWBDlg =
// If so, show our menu!
// mark as handled
Now that I have trapped the right click, I will have to ask the window that owns the Web browser control to see if the window that got the right click is a Web browser window. It is a little twitchy here. I will have to specifically go case by case, meaning I will have to check, if not this Web browser, the other usage instance of webbrowser and other (meaning if a dialog box had three controls which were Web browsers, I might have a CWnd m_WndBrowser1, CWnd m_WndBrowser2, and CWnd m_WndBrowser3). In the attached example, I have a dialog with one webbrowser control in it. I have a member variable for the Web browser window as follows.
My function will have to do a little bit more than just matching the window handles. Why so? A nested frame inside IE is again an instance of IE. HTMLs having nested frames will return a different window handle for different frames inside the same parent IE window. So what will be my logic?! I will have to traverse via GetParent from a frame to the parent IE window, whose window handle I already know or till I don’t have a parent of type IE. How do we know if it’s an IE Window? The Window class name will be “Internet Explorer_Server.” That’s what the next function does, see if our window is an Internet Explorer, if so we know, the outer shell window will be the parent of doc-view which will be the parent of the IE class window.
// This fn checks if this is a IE type window and returns
// the outer hosting Shell window handle for it, if it is
// one! else NULL
HWND CWebBrowserTestDlg::getParentWebBrowserWnd(HWND hMyWnd)
HWND hRetWBWnd = NULL;
static TCHAR tszIEClass = “Internet Explorer_Server”;
// Compare if our window is of type IE class
if(0 == _tcscmp(tszClass, tszIEClass))
// Then get the Shell which hosts the IE doc view window
hRetWBWnd = ::GetParent(::GetParent(hMyWnd));
// Checks if message window is our WebBrowser control
// and if so returns TRUE
BOOL CWebBrowserTestDlg::isThisOurBrowser(HWND hMyWnd)
HWND hWndOurWebBrowser = (HWND)m_WndBrowser.GetSafeHwnd();
// Get the Parent Web browser window if any
HWND hParentWBWnd = getParentWebBrowserWnd(hMyWnd);
// Now, we can have nested frames ..
// meaning a frame inside a frame is yet another webbrowser
// So our m_WndBrowser guy can be way up there ..
// so we have to iterate up to him through parent frames
// till we have a match or no more WB type parent
while( (NULL != hParentWBWnd) &&
(hParentWBWnd != hWndOurWebBrowser))
// Get the Parent of the Web browser window
//I have, recursively
return (NULL != hParentWBWnd);
Now the next function is a normal function to show a popup menu from a resource and I am not going to explain it. People who need explanation, please read the help file for TrackPopupMenu.
// This is a trivial fn which gets a submenu from a resource
// and displays it at the current cursor position
//Make your custom menu here
// Do we have a menu loaded?
// this is my popup menu
CMenu* pMyPopup = ReplaceMenu.GetSubMenu(0);
if(NULL != pMyPopup)
// Show it
pMyPopup->TrackPopupMenu (TPM_LEFTALIGN | TPM_RIGHTBUTTON,
CurPoint.y, this );
// deleting me and my parent when I am done