【问题标题】:How to get the "KeyPress" event from a Word 2010 Addin (developed in C#)?如何从 Word 2010 插件(用 C# 开发)中获取“KeyPress”事件?
【发布时间】:2011-11-14 10:26:05
【问题描述】:

如何从用 C# 开发的 Word 2010 插件中“捕捉”KeyPress 事件?

注意:我不是在寻找诸如挂钩之类的“复杂”解决方案,而是为了美观整洁 .NET 甚至来自对象模型。

我“手中”的应用程序对象是:

Microsoft.Office.Interop.Word.Application

最好的问候

【问题讨论】:

标签: c# ms-word add-in


【解决方案1】:

不幸的是,Word API 或 VSTO 中没有内置任何东西可以拾取击键,更多信息可以在here找到。

一段时间以来,我一直在寻找可行的解决方案,但我能想到的最好办法是使用钩子通过 Windows API 处理它,您很可能会得出相同的结论,所以这里有一个示例:

您需要将using 指令添加到以下程序集中:

using System;
using System.Diagnostics;
using System.Runtime.InteropServices;
using System.Threading;
using System.Windows.Forms;

这就是钩子:

   public partial class ThisAddIn
    {
        private const int WH_KEYBOARD_LL = 13;
        private const int WM_KEYDOWN = 0x0100;

        private static IntPtr hookId = IntPtr.Zero;
        private delegate IntPtr HookProcedure(int nCode, IntPtr wParam, IntPtr lParam);
        private static HookProcedure procedure = HookCallback;

        [DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        private static extern IntPtr GetModuleHandle(string lpModuleName);

        [DllImport("user32.dll", SetLastError = true)]
        private static extern bool UnhookWindowsHookEx(IntPtr hhk);

        [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        private static extern IntPtr SetWindowsHookEx(int idHook, HookProcedure lpfn, IntPtr hMod, uint dwThreadId);

        [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        private static extern IntPtr CallNextHookEx(IntPtr hhk, int nCode, IntPtr wParam, IntPtr lParam);  

        private void ThisAddIn_Startup(object sender, System.EventArgs e)
        {
            hookId = SetHook(procedure);
        }

        private void ThisAddIn_Shutdown(object sender, System.EventArgs e)
        {
            UnhookWindowsHookEx(hookId);
        }

        private static IntPtr SetHook(HookProcedure procedure)
        {
            using (Process process = Process.GetCurrentProcess())
            using (ProcessModule module = process.MainModule)
                return SetWindowsHookEx(WH_KEYBOARD_LL, procedure, GetModuleHandle(module.ModuleName), 0);
        }

        private static IntPtr HookCallback(int nCode, IntPtr wParam, IntPtr lParam)
        {
            if (nCode >= 0 && wParam == (IntPtr)WM_KEYDOWN)
            {
                int pointerCode = Marshal.ReadInt32(lParam);
                string pressedKey = ((Keys)pointerCode).ToString();

                //Do some sort of processing on key press
                var thread = new Thread(() => { MessageBox.Show(pressedKey); });
                thread.Start();
            }
            return CallNextHookEx(hookId, nCode, wParam, lParam);
        }

        private void InternalStartup()
        {
            this.Startup += new System.EventHandler(ThisAddIn_Startup);
            this.Shutdown += new System.EventHandler(ThisAddIn_Shutdown);
        }
    }

【讨论】:

  • 为什么处理代码在线程内部?在我自己的测试中,如果处理不是线程化的,CallNextHookEx 似乎会失败,但我找不到关于 why 的任何文档。
  • 以上代码没有在 MS Word 中捕获字符类型,它只是在单词之外
【解决方案2】:

您可以尝试使用Excel WebBrowser Control 而不是 System.Windows.Forms WebBrowser;它处理特殊键转发,如 TAB、DEL、CTRL+V 等。

为此更改 WebBrowser 构造函数

new System.Windows.Forms.WebBrowser();

new Microsoft.Office.Tools.Excel.Controls.WebBrowser();  

您需要添加对项目的引用:Project/Add Reference/Extensions 选择 Microsoft.Tools.Outlook & Microsoft.Tools.Outlook.v4.0.Utilities

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2011-01-09
    • 2011-03-27
    • 1970-01-01
    • 2014-05-12
    • 2015-06-17
    • 1970-01-01
    • 2010-11-27
    相关资源
    最近更新 更多