让WebBrowser控件响应键盘快捷键或者鼠标信息
通过SDK方式或者Atl方式添加的IE控件(WebBrowser控件)无法响应键盘的tab消息,这在输入表单的时候非常不方便,每次都需要鼠标点击输入框,我在网上看过一些文章,据说可以通过IE控件的IDocHostShowUI接口和IDocHostUIHandler2接口获取响应的消息,不过我试过这些方法,没成功~_~.于是采用了一种比较偏门的方法:挂钩键盘消息。
首先,设置键盘消息处理函数,该函数的功能是获取键盘的输入键信息,如果该键是tab键,并且是按下状态,则将该消息发送给IE控件的快捷键处理函数进行快捷键处理,同样的,为了能通过回车键提交html表单,判断输入键是回车键,并且是按键释放时,将该消息发送给IE控件的快捷键处理函数进行快捷键处理。
先声明一个全局的HHOOK句柄
HHOOK h_Keyboard;
消息处理函数如下:
LRESULT CALLBACK KeyboardProc(
int code, // hook code
WPARAM wParam, // virtual-key code
LPARAM lParam // keystroke-message information
)
{
if(wParam==VK_TAB && GetKeyState(wParam)< 0)
{
//按下tab键
//::MessageBox(NULL, _T("TAB"), _T(""), 0);
//以下是获取浏览器对象并像对象发送快捷键的代码,我这里的m_wndIE是AxWindow类型,如果直接用SDK创建IE控件的话,前面的代码有所不同,但后面的获取接口的方式是一致的
CWebWindow* pWebWindow = g_webWindow;
MSG msg;
msg.hwnd = pWebWindow->m_wndIE.m_hWnd;
msg.message = WM_KEYDOWN;
msg.lParam = lParam;
msg.wParam = wParam;
HRESULT hr = S_OK;
IWebBrowser2Ptr browser;
hr = pWebWindow->m_wndIE.QueryControl(__uuidof(IWebBrowser2), (void**)&browser);
ATLASSERT(SUCCEEDED(hr));
CComPtr<IDispatch> disp;
hr = browser->get_Document(&disp);
ATLASSERT(SUCCEEDED(hr));
//CComPtr<IHTMLDocument2> doc;
CComQIPtr<IOleInPlaceActiveObject> spInPlace;
hr = disp->QueryInterface(__uuidof(IOleInPlaceActiveObject), (void**)&spInPlace);
ATLASSERT(SUCCEEDED(hr));
if (spInPlace)
bool bRet = (spInPlace->TranslateAccelerator(&msg) == S_OK) ? TRUE : FALSE;
}
else if (wParam == VK_RETURN && GetKeyState(wParam)> 0)
{
///释放回车键
//::MessageBox(NULL, _T("ENTER"), _T(""), 0);
////以下是获取浏览器对象并像对象发送快捷键的代码,我这里的m_wndIE是AxWindow类型,如果直接用SDK创建IE控件的话,前面的代码有所不同,但后面的获取接口的方式是一致的
CWebWindow* pWebWindow = g_webWindow;
MSG msg;
msg.hwnd = pWebWindow->m_wndIE.m_hWnd;
msg.message = WM_KEYUP;
msg.lParam = lParam;
msg.wParam = wParam;
HRESULT hr = S_OK;
IWebBrowser2Ptr browser;
hr = pWebWindow->m_wndIE.QueryControl(__uuidof(IWebBrowser2), (void**)&browser);
ATLASSERT(SUCCEEDED(hr));
CComPtr<IDispatch> disp;
hr = browser->get_Document(&disp);
ATLASSERT(SUCCEEDED(hr));
//CComPtr<IHTMLDocument2> doc;
CComQIPtr<IOleInPlaceActiveObject> spInPlace;
hr = disp->QueryInterface(__uuidof(IOleInPlaceActiveObject), (void**)&spInPlace);
ATLASSERT(SUCCEEDED(hr));
if (spInPlace)
bool bRet = (spInPlace->TranslateAccelerator(&msg) == S_OK) ? TRUE : FALSE;
}
//返回非0值,表示我们已处理此键盘消息了
return CallNextHookEx(h_Keyboard,code,wParam,lParam);
}
然后,就是设置挂钩函数,这个放在主函数的初始化部分:
h_Keyboard=SetWindowsHookEx(WH_KEYBOARD,KeyboardProc,hInstance,0);
最后,释放挂钩,这个放在主函数的退出之前:
UnhookWindowsHookEx(h_Keyboard);
原文地址:http://www.cnblogs.com/moodlxs/archive/2012/11/01/2750142.html

浙公网安备 33010602011771号