Подтвердить что ты не робот

Как получить событие "KeyPress" из Word 2010 Addin (разработанное на С#)?

Как я могу "поймать" событие KeyPress из Word 2010 Addin, разработанного на С#?

Примечание. Я не ищу "сложные" решения, такие как подключение, но для приятных и аккуратных .NET даже из объектной модели.

Объект приложения, который у меня есть "в моих руках":

Microsoft.Office.Interop.Word.Application

С наилучшими пожеланиями

4b9b3361

Ответ 1

К сожалению, нет ничего встроенного в Word API или VSTO, который может получить ключевые штрихи, более подробную информацию об этом можно найти здесь

Я искал возможное решение в течение некоторого времени, но лучшее, что я мог придумать, это обработать его через 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);
        }
    }

Ответ 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