3

私は、キーボード入力を受け取り、入力を消費し、入力された入力と引き換えに他の文字を返すソフトウェアを開発しようとしています。たとえば、「abcd」と入力し、交換ルールをロシア語のアルファベットとして定義すると、出力は「ский」と予想されます。

私が使用しているコードは次のとおりです。

namespace hook_form
{
    public partial class Form1 : Form
    {
        //Keyboard API constants

        private const int WH_KEYBOARD_LL = 13;
        private const int WM_KEYDOWN = 0x0100;
        private const int WM_KEYUP = 0x0101;
        private const int WM_SYSKEYUP = 0x0105;
        private const int WM_SYSKEYDOWN = 0x0104;
        private HookHandlerDelegate proc;
        private IntPtr hookID = IntPtr.Zero;
        private delegate IntPtr HookHandlerDelegate(int nCode, IntPtr wParam, ref KBDLLHOOKSTRUCT lParam);

        private struct KBDLLHOOKSTRUCT
        {
            public int vkCode;
            int scanCode;
            public int flags;
            int time;
            int dwExtraInfo;
        }

        private IntPtr HookCallback(int nCode, IntPtr wParam, ref KBDLLHOOKSTRUCT lParam)
        {
            bool AllowKey = false;

            switch (lParam.vkCode)
            {
                case (65|97|66|98): // key codes for "a/b/c/d" etc. goes here
                    //Alt+Tab, Alt+Esc, Ctrl+Esc, Windows Key
                    AllowKey = true;
                    SendKeys.Send("\u0997"); // code for russian letters here....
                    break;
            }

            MessageBox.Show(lParam.vkCode.ToString());

            if (AllowKey == false)
                return (System.IntPtr)1;

            return CallNextHookEx(hookID, nCode, wParam, ref lParam);
        }
        #region DllImports

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

        [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        [return: MarshalAs(UnmanagedType.Bool)]
        private static extern bool UnhookWindowsHookEx(IntPtr hhk);

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

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

        private void buttonStart_Click(object sender, EventArgs e)
        {
            proc = new HookHandlerDelegate(HookCallback);
            using (Process curProcess = Process.GetCurrentProcess())
            using (ProcessModule curModule = curProcess.MainModule)
            {
                hookID = SetWindowsHookEx(WH_KEYBOARD_LL, proc,
                GetModuleHandle(curModule.ModuleName), 0);
            }
        }

        private void btnEnd_Click(object sender, EventArgs e)
        {
            UnhookWindowsHookEx(hookID);
        }
    }
}

「aсbкcиdй」として出力されます。つまり、入力された文字(英語)は、を返してもコードによって消費されません(つまり、出力に表示されないはずです)(System.IntPtr)1

フックを外すのが問題だと思います。誰かが私がこの問題を解決するのを手伝ってくれる?

不可能な場合は、キーボードフックを使用するオープンソースソフトウェアを誰かに紹介してもらえますか?

4

2 に答える 2

1

CallNextHookEx飲み込みたいイベントを呼び出さないようにしてください。

編集:私が見ている問題の1つは、キーを押すと2つのメッセージWM_KEYDOWNとWM_KEYUPが生成されることです。両方の場合に新しいキーボードメッセージを挿入し、それを複製します。Tim Barrassのリンクの例は、これをより適切に行う方法を示しています。ただし、これはあなたが見ている振る舞いを完全には説明していません。

于 2010-12-01T14:53:05.607 に答える
0

必要な新しい文字を送信する前に、バックスペースキーを押して送信してはどうでしょうか。

AllowKey = true;
SendKeys.Send("{BS}");
SendKeys.Send("\u0997"); // code for russian letters here....
break;
于 2010-12-01T14:46:43.373 に答える