在C#中保持原始命令的同时挂钩按键

时间:2013-10-22 09:29:43

标签: c# key hook keypress

我希望在应用程序在后台运行时检测到按键并相应地执行某些事件。例如,当我按下W时,我希望我的应用程序检测到这一点,写下“W pressed”in let say a listbox而不影响正在进行的任何其他进程。

我发现的以下代码几乎与我上面所说的完全相同。唯一的问题是它完全拦截了按键。例如,如果我打开谷歌并输入“W”,则该信件不会出现在谷歌搜索栏中,但会被应用程序挂钩。

namespace WindowsFormsApplication1
{
   public partial class Form1 : Form
   {


     // DLL libraries used to manage hotkeys
     [DllImport("user32.dll")]
     public static extern bool RegisterHotKey(IntPtr hWnd, int id, int fsModifiers, int vlc);
     [DllImport("user32.dll")]
     public static extern bool UnregisterHotKey(IntPtr hWnd, int id);

     const int MYACTION_HOTKEY_ID = 1;


     public Form1()
     {
         InitializeComponent();

         // Modifier keys codes: Alt = 1, Ctrl = 2, Shift = 4, Win = 8
         // Compute the addition of each combination of the keys you want to be pressed
         // ALT+CTRL = 1 + 2 = 3 , CTRL+SHIFT = 2 + 4 = 6...
         RegisterHotKey(this.Handle, MYACTION_HOTKEY_ID, 0, (int)Keys.W);
     }


     protected override void WndProc(ref Message m)
     {
         if (m.Msg == 0x0312 && m.WParam.ToInt32() == MYACTION_HOTKEY_ID)
         {
             // My hotkey has been typed

             // Do what you want here
             // ...
             lstLog.Items.Add("W pressed");
         }
         base.WndProc(ref m);
     }
  }
}

所以,简而言之。我正在寻找一种方法来挂钩按键,同时仍然保持原始命令。在上面的示例中,“W”应出现在Google搜索栏中。

我一直在寻找这个(可能很少)问题的解决方案,并找到了很多 有关公钥和挂钩的帖子,如上所述,但没有一个真正按照我希望的方式工作。所以我决定在这里发表我的第一篇文章。希望你们能帮助我。如果有任何不清楚的地方,请告诉我。

THX

- 编辑 -

其他一些信息: 我打算在玩英雄联盟,魔兽世界等游戏时使用这个程序。所以我希望它不会与这些游戏的任何过程互动,因为可能禁止使用第三方软件。如果有什么我应该知道的,那么请告诉我。我和我的未来未被禁止的帐户提前感谢你。

1 个答案:

答案 0 :(得分:0)

您无法使用RegisterHotKey,因为它不会通过密钥事件。

您必须使用WH_KEYBOARD_LL Hook。

这是一个功能齐全的样本,既快又脏,但功能齐全:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Windows.Forms;
using System.Runtime.InteropServices;
using System.Runtime.CompilerServices;
using System.Diagnostics;


namespace StackoverflowHooks
{
    public partial class Form1 : Form
    {
        [DllImport("user32.dll")]
        private static extern IntPtr SetWindowsHookEx(int idHook, HookProc lpfn, IntPtr hInstance, int threadId);
        [DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
        private static extern IntPtr GetModuleHandle(string lpModuleName);
        [DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)]
        private static extern IntPtr CallNextHookEx(IntPtr hHook, int nCode, IntPtr wParam, IntPtr lParam);


        [StructLayout(LayoutKind.Sequential)]
        public struct KBDLLHOOKSTRUCT
        {
            public uint vkCode;
            public uint scanCode;
            public uint flags;
            public uint time;
            IntPtr dwExtraInfo;
        }


        IntPtr hHook;
        private delegate IntPtr HookProc(int nCode, IntPtr wp, IntPtr lp);
        HookProc lpfn;

        private IntPtr KeyboardHookProc(int code, IntPtr wParam, IntPtr lParam)
        {
            KBDLLHOOKSTRUCT data = (KBDLLHOOKSTRUCT)Marshal.PtrToStructure(lParam, typeof(KBDLLHOOKSTRUCT));
            MessageBox.Show("Pressed key: " + (Keys)data.vkCode);

            return CallNextHookEx(hHook, code, wParam, lParam);
        }


        public Form1()
        {
            InitializeComponent();  
            SetHook();
        }

        private void SetHook()
        {
            int id_hook = 13; //WH_KEYBOARD_LL - HOOK
            lpfn = new HookProc(this.KeyboardHookProc);
            using (ProcessModule curModule = Process.GetCurrentProcess().MainModule)
            hHook = SetWindowsHookEx(id_hook, lpfn, GetModuleHandle(curModule.ModuleName), 0);
        }
    }
}