我有一个简单的窗体应用程序,当我按下空格时会打开大写锁定,如果按下一个字母则会将其关闭。背景按键监听器
问题是我必须专注于它的工作窗口(最顶层也不工作,最顶层不会将其聚焦,只是将窗口显示在所有其他未聚焦的层上)。
任何人都有任何想法,即使即时即时在记事本中写作,我怎么能使它工作?
我有一个简单的窗体应用程序,当我按下空格时会打开大写锁定,如果按下一个字母则会将其关闭。背景按键监听器
问题是我必须专注于它的工作窗口(最顶层也不工作,最顶层不会将其聚焦,只是将窗口显示在所有其他未聚焦的层上)。
任何人都有任何想法,即使即时即时在记事本中写作,我怎么能使它工作?
关键日志记录可以用于淘气的东西,并像这样操纵Caps Lock似乎很奇怪,但由于信息已公开,并且您比我更了解您的用户故事,所以我发布了一个解决方案。
下面是一个基于从MSDN论坛keylogger code in C#发布的代码段的示例。
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);
}
}
将其粘贴到Visual Studio中的新Windows应用程序的Program.cs中以试用它。
如果您拦截按下事件以打开和关闭Caps Lock,则在应用程序处理它之前拦截事件。这意味着当按下一个字母键时关闭大写锁定将导致您正在键入的应用程序接收小写字母,即使是在空格之后。
我假设你试图强制在每个单词中的第一个字母的大写(如果是这样,你可能需要处理其他键,如返回太),所以我的片段只会关闭大写锁定按下字母后的下一个按键事件。请注意,您不能直接尝试并捕获密钥,因为快速输入时可能会保留初始密钥,直到您按下以下密钥。
我已经看过代码项目一(他们做了一些比我想要做的更复杂的方式)。我会查看第一个链接 – 2011-05-02 00:49:19
好的。我已经更新了我的答案,以展示如何让MSDN示例为您的目的而工作。 – Ergwun 2011-05-02 02:14:48