Этот вопрос очень тесно связан с Как получить событие "KeyPress" из Word 2010 Addin (разработанное на С#)? (и на самом деле включает образец код из ответа на этот вопрос), но это особенно касается разработки в Visual Studio (Professional) 2015 для Word 2016, работающей в Windows 10.
Я пытаюсь обнаружить, когда текст изменяется в документе Word из надстройки VSTO. Я понимаю из
- Как получить событие "KeyPress" из Word 2010 Addin (разработанное на С#)? (14 ноября 2011 г.)
- Захват события keydown MS Word с использованием С# (21 октября 2012 г.)
- Как поднять событие в MS Word Keypress (24 октября 2012 г.)
- Как заблокировать событие нажатия клавиш в MS Word с помощью VSTO? (5 ноября 2012 г.)
что для этого не существует способа, управляемого событиями. Word просто не отправляет события при изменении текста.
Я видел два обходных решения:
- Используйте WindowSelectionChange событие. К сожалению, это событие отправляется, когда выбор изменяется, нажимая клавиши со стрелками, используя мышь, выполняя отмену или повтор и, возможно, другие действия, но не при вводе или удалении.
- Используйте низкоуровневый крючок событий. Это обсуждалось в нескольких из этих вопросов StackOverflow и также называлось "широко распространенной техникой" в потоке на форуме Visual Studio в Февраль 2014 года.
Я пытаюсь использовать код в ответе Как получить событие "KeyPress" из Word 2010 Addin (разработанное на С#)?, и кажется наблюдать за каждым событием keydown, кроме тех, которые были отправлены в Word 2016.
Здесь используется код Im для упрощения ссылок.
using System;
using System.Diagnostics;
using System.Runtime.InteropServices;
using System.Threading;
using System.Windows.Forms;
namespace KeydownWordAddIn
{
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 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(() => {
Debug.WriteLine(pressedKey);
});
thread.Start();
}
return CallNextHookEx(hookId, nCode, wParam, lParam);
}
private void ThisAddIn_Startup(object sender, EventArgs e)
{
hookId = SetHook(procedure);
}
private void ThisAddIn_Shutdown(object sender, EventArgs e)
{
UnhookWindowsHookEx(hookId);
}
#region VSTO generated code
/// <summary>
/// Required method for Designer support.
/// </summary>
private void InternalStartup()
{
this.Startup += new System.EventHandler(ThisAddIn_Startup);
this.Shutdown += new System.EventHandler(ThisAddIn_Shutdown);
}
#endregion
}
}
Когда я запустил Word 2016 с этой надстройкой, я вижу события keydown, отправленные в браузер Edge и даже Visual Studio, но не в сам Word.
Являются ли блокировки клавиш как-то помешаны в Word 2016, или я делаю что-то неправильно?