3

看来我没有小问题。

我将自己的窗口挂接到 IE 主窗口。我的窗口源自 WTL 的 CWindowImpl 并承载 IWebBrowers2 控件,该控件显示了一些内容。

IWebBrowser2 显示带有编辑框的 html <input type='text'/>,我需要支持所有键,这是编辑和操作文本所必需的(Ctrl + C、Ctrl + V、Ctrl + X 等...+ Esc、Delete、向上和向下箭头)。

我还需要禁止一些快捷方式,例如 Ctrl + P、Ctrl + S,因为它们会调用特定于我不需要的网页的对话框。


似乎是常见问题,我需要为我的 IWebBrowser2 对象调用 TranslateAccelerator。

网上也有类似的问题——Tab key doesn't work in IWebbrowser2

解决方案——热键不起作用!!

IWebBrowser2 控件中的 Tab 键支持

这是我所需要的非常有趣的线程——在托管的 Web 浏览器控件中处理 Control-C


所以,首先我需要为我的 IWebBrowser2 对象调用 TranslateAccelerator。但我必须先从键盘获取消息。

那么,让我们看看它的外观。

在此处输入图像描述

我的挂钩窗口没有收到任何键盘输入消息。一个带有“Internet Explorer_Server”类的窗口接收所有这些(实际上,它是 IWebBrowser2 内的 IE 的一个 hwnd)。

所以我需要挂钩这个hwnd的window proc。

m_ieOldProc = (PROC)::SetWindowLongPtr ( hIEWnd, GWLP_WNDPROC, (LONG_PTR)_IEWndProc );

在 hooked wnd proc 中,我执行以下操作:

// ...
switch ( uMsg )
{
    case WM_KEYUP: 
    case WM_KEYDOWN: 
    {
        // ...
        IOleInPlaceActiveObject* pAccelerator;
        CComPtr< IWebBrowser2 > pWebBrowser =  CWebWindow::GetWebBrowser( hwnd );
        if( pWebBrowser )
        {               
            hr = pWebBrowser->QueryInterface( IID_IOleInPlaceActiveObject,(void**)&pAccelerator );
            if ( SUCCEEDED(hr) )
            {
                pAccelerator->TranslateAccelerator(&msg);
                pAccelerator->Release();
            }           
        }
        // ...
    }
}
// ...

是的!有些事情成功了!Esc、Delete、向上和向下箭头键现在可以使用。

但不是所有的。快捷方式有问题。

  1. Ctrl + C,Ctrl + A 不起作用。

Ctrl + X,Ctrl + V - 工作。

这是 Spy++ 中用于 hwnd 的消息日志,其类为“Internet Explorer_Server”。

在 Ctrl + CI 上不接收 WM_COMMAND。

在此处输入图像描述

在 Ctrl + X 上:

在此处输入图像描述

我不知道为什么。

2. “坏”的捷径确实有效。打印对话框在 Ctrl + P 上调用,网页保存对话框在 Ctrl + S 上调用,等等。在这种情况下我无能为力。无论我在挂钩窗口 proc 中返回什么,它们仍在显示。所以我需要在发送到窗口之前处理它们。


解决方案?

你可以提到,在这个问题的所有解决方案中都有:

  1. 预翻译消息
  2. 实际上 TranslateAccelerator

直到此刻,我在挂钩消息过程中只有 TranslateAccelerator。

预翻译必须做这样的事情:

BOOL PreTranslateMessage(MSG* pMsg)
{
    if((pMsg->message < WM_KEYFIRST || pMsg->message > WM_KEYLAST) &&
        (pMsg->message < WM_MOUSEFIRST || pMsg->message > WM_MOUSELAST))
        return FALSE;

    // give HTML page a chance to translate this message
    return (BOOL)SendMessage(WM_FORWARDMSG, 0, (LPARAM)pMsg);
}

此外,在 PreTranslateMessage 中,我可以过滤“坏”快捷方式而不将它们发送到窗口。

另一个人建议也使用 PretranslateMessage (来自上面线程的对话框):

– 从 CMessageFilter 派生您的窗口,使用 CMessageLoop::AddMessageFilter 安装它,如示例所示实现 PreTranslateMessage。

– 我想我现在已经按照样本的建议做了。但问题是我的“父”窗口没有得到击键 - 它们都转到 IE AX 控件。

– 这就是您需要 CMessageFilter 的原因。在消息被分派到它们的目标窗口之前,它被连接到消息泵中。

好的,但是托管 IWebBrowser2 的我的根窗口没有收到任何键盘消息。此外,我的窗口中没有 PreTranslateMessage,只有窗口 proc 调用(由父级,CWindowImpl)。

如上所述,我可以从 CMessageFilter 派生,实现 PreTranslateMessage,但我无法使用 CMessageLoop::AddMessageFilter 订阅事件,因为我没有创建主窗口并且无权访问它的 CMessageLoop。


那么我现在应该做些什么来完成所有这些工作呢?我应该使用 PreTranslateMessage 以及如何使用?

4

2 回答 2

6

最近我也遇到了复制/剪切命令不起作用的问题。事实证明这不是 TranslateAccelerator 或类似的键盘问题。问题是 COM 库是使用 CoInitialize/CoInitializeEx 初始化的。但是需要调用 OleInitialize 才能使剪贴板工作。MSDN“WebBrowser Customization”页面上有解释:

您的应用程序应该使用 OleInitialize 而不是 CoInitialize 来启动 COM。OleInitialize 支持剪贴板、拖放操作、OLE 和就地激活。当您的应用程序关闭时,使用 OleUninitialize 关闭 COM 库。

http://msdn.microsoft.com/en-us/library/aa770041(v=vs.85).aspx

如果还是不行,看看我在 PHP 桌面项目中 IWebBrowser2 控件的实现(C API):

https://code.google.com/p/phpdesktop/source/browse/phpdesktop-msie/msie/

于 2014-01-24T02:02:36.350 回答
0

听起来您已经实现了一个工具栏。在这种情况下,您应该实现IInputObject,并且浏览器框架将在适当的时候将调用转发给您,以便您可以适当地翻译加速器(和否决加速器)。

于 2013-12-18T18:17:18.423 回答