C# 背景按键侦听器

C# 背景按键侦听器,c#,winforms,focus,background-process,sendkeys,C#,Winforms,Focus,Background Process,Sendkeys,我有一个简单的窗口窗体应用程序,当我按下空格键时,它会打开capslock,如果我按下一个字母,它就会关闭 问题是,我必须把焦点放在窗口上,它才能工作(最上面的部分也不工作,最上面的部分也不聚焦,只是在所有其他未聚焦的窗口之上显示窗口) 任何人都知道即使我在记事本上写东西,我怎么能让它工作吗?键记录可以用来处理顽皮的东西,像那样操纵大写锁定似乎很奇怪,但由于信息已经公开,而且你比我更了解你的用户故事,我发布了一个解决方案 下面是一个基于MSDN论坛中发布的代码片段的示例 using System

我有一个简单的窗口窗体应用程序,当我按下空格键时,它会打开capslock,如果我按下一个字母,它就会关闭

问题是,我必须把焦点放在窗口上,它才能工作(最上面的部分也不工作,最上面的部分也不聚焦,只是在所有其他未聚焦的窗口之上显示窗口)


任何人都知道即使我在记事本上写东西,我怎么能让它工作吗?

键记录可以用来处理顽皮的东西,像那样操纵大写锁定似乎很奇怪,但由于信息已经公开,而且你比我更了解你的用户故事,我发布了一个解决方案

下面是一个基于MSDN论坛中发布的代码片段的示例

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

class Program
{
    private const int WH_KEYBOARD_LL = 13;
    private const int WM_KEYDOWN = 0x0100;
    private static LowLevelKeyboardProc _proc = HookCallback;
    private static IntPtr _hookID = IntPtr.Zero;
    private static bool lastKeyWasLetter = false;

    [DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
    private static extern IntPtr SetWindowsHookEx(int idHook, LowLevelKeyboardProc 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, IntPtr lParam);

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

    [DllImport("user32.dll")]
    static extern void keybd_event(byte bVk, byte bScan, uint dwFlags, UIntPtr dwExtraInfo);

    private delegate IntPtr LowLevelKeyboardProc(int nCode, IntPtr wParam, IntPtr lParam);

    [STAThread]
    static void Main(string[] args)
    {
        Application.EnableVisualStyles();
        Application.SetCompatibleTextRenderingDefault(false);

        _hookID = SetHook(_proc);
        Application.Run();

        UnhookWindowsHookEx(_hookID);
    }

    private static IntPtr SetHook(LowLevelKeyboardProc proc)
    {
        using (Process curProcess = Process.GetCurrentProcess())
        using (ProcessModule curModule = curProcess.MainModule)
        {
            return SetWindowsHookEx(WH_KEYBOARD_LL, proc, GetModuleHandle(curModule.ModuleName), 0);
        }
    }

    private static void ToggleCapsLock()
    {
        const int KEYEVENTF_EXTENDEDKEY = 0x1;
        const int KEYEVENTF_KEYUP = 0x2;

        UnhookWindowsHookEx(_hookID);
        keybd_event(0x14, 0x45, KEYEVENTF_EXTENDEDKEY, (UIntPtr)0);
        keybd_event(0x14, 0x45, KEYEVENTF_EXTENDEDKEY | KEYEVENTF_KEYUP, (UIntPtr)0);
        _hookID = SetHook(_proc);
    }

    private static IntPtr HookCallback(int nCode, IntPtr wParam, IntPtr lParam)
    {
        if (nCode >= 0 && wParam == (IntPtr)WM_KEYDOWN)
        {
            if (lastKeyWasLetter)
            {
                if (Control.IsKeyLocked(System.Windows.Forms.Keys.CapsLock))
                {
                    ToggleCapsLock();
                }
                lastKeyWasLetter = false;
            }
            Keys key = (Keys)Marshal.ReadInt32(lParam);            
            if (key == Keys.Space)
            {
                if (!Control.IsKeyLocked(System.Windows.Forms.Keys.CapsLock))
                {
                    ToggleCapsLock();
                }
            }
            else if (key >= Keys.A && key <= Keys.Z)
            {
                lastKeyWasLetter = true;
            }
        }
        return CallNextHookEx(_hookID, nCode, wParam, lParam);
    }
}
使用系统;
使用系统诊断;
使用System.Runtime.InteropServices;
使用System.Windows.Forms;
班级计划
{
专用常量int WH_键盘LL=13;
私有常量int WM_KEYDOWN=0x0100;
私有静态低层keyboardproc\u proc=HookCallback;
私有静态IntPtr _hookID=IntPtr.Zero;
私有静态bool lastKeyWasLetter=false;
[DllImport(“user32.dll”,CharSet=CharSet.Auto,SetLastError=true)]
私有静态外部IntPtr SETWINDOWSHOOKX(int idHook、低层键盘PROC lpfn、IntPtr hMod、uint dwThreadId);
[DllImport(“user32.dll”,CharSet=CharSet.Auto,SetLastError=true)]
[返回:Marshallas(UnmanagedType.Bool)]
私有静态外部bool unhookwindowshookx(IntPtr hhk);
[DllImport(“user32.dll”,CharSet=CharSet.Auto,SetLastError=true)]
私有静态外部IntPtr CallNextHookEx(IntPtr hhk、intncode、IntPtr wParam、IntPtr lParam);
[DllImport(“kernel32.dll”,CharSet=CharSet.Auto,SetLastError=true)]
私有静态外部IntPtr GetModuleHandle(字符串lpModuleName);
[DllImport(“user32.dll”)]
静态外部无效keybd_事件(字节bVk、字节bScan、uint dwFlags、UIntPtr dwExtraInfo);
私有委托IntPtr低级键盘PROC(int nCode、IntPtr wParam、IntPtr lParam);
[状态线程]
静态void Main(字符串[]参数)
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
_hookID=SetHook(_proc);
Application.Run();
unhookwindowshookx(_hookID);
}
专用静态IntPtr SetHook(低级键盘程序)
{
使用(Process curProcess=Process.GetCurrentProcess())
使用(ProcessModule curModule=curProcess.MainModule)
{
返回SetWindowsHookEx(WH\u KEYBOARD\u LL,proc,GetModuleHandle(curModule.ModuleName),0);
}
}
私有静态void ToggleCapsLock()
{
const int KEYEVENTF_EXTENDEDKEY=0x1;
const int keyevent fu KEYUP=0x2;
unhookwindowshookx(_hookID);
keybd_事件(0x14,0x45,KEYEVENTF_扩展键,(UIntPtr)0);
keybd_事件(0x14,0x45,KEYEVENTF_EXTENDEDKEY | KEYEVENTF_KEYUP,(UIntPtr)0);
_hookID=SetHook(_proc);
}
专用静态IntPtr钩子回调(int nCode、IntPtr wParam、IntPtr lParam)
{
如果(nCode>=0&&wParam==(IntPtr)WM\u KEYDOWN)
{
如果(lastKeyWasLetter)
{
if(Control.IsKeyLocked(System.Windows.Forms.Keys.CapsLock))
{
切换capslock();
}
lastKeyWasLetter=假;
}
Keys key=(Keys)Marshal.ReadInt32(lParam);
if(key==Keys.Space)
{
如果(!Control.IsKeyLocked(System.Windows.Forms.Keys.CapsLock))
{
切换capslock();
}
}

如果(key>=Keys.A&&key)我已经研究了代码项目一(他们做了比我想做的更复杂的事情)。我将查看第一个链接OK。我已经更新了我的答案,以展示如何让MSDN示例为您的目的工作。