Блокировка клавиш быстрого доступа с помощью C#
Я использую приведенный ниже код для отключения Alt+Tab, Alt+Esc, Ctrl+Esc и Windows Key, но почему-то это не работает. Пожалуйста, помогите мне исправить это.
namespace BlockShortcuts
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
}
private delegate int LowLevelKeyboardProcDelegate(int nCode, int
wParam, ref KBDLLHOOKSTRUCT lParam);
[DllImport("user32.dll", EntryPoint = "SetWindowsHookExA", CharSet = CharSet.Ansi)]
private static extern int SetWindowsHookEx(
int idHook,
LowLevelKeyboardProcDelegate lpfn,
int hMod,
int dwThreadId);
[DllImport("user32.dll")]
private static extern int UnhookWindowsHookEx(int hHook);
[DllImport("user32.dll", EntryPoint = "CallNextHookEx", CharSet = CharSet.Ansi)]
private static extern int CallNextHookEx(
int hHook, int nCode,
int wParam, ref KBDLLHOOKSTRUCT lParam);
const int WH_KEYBOARD_LL = 13;
private int intLLKey;
private KBDLLHOOKSTRUCT lParam;
private struct KBDLLHOOKSTRUCT
{
public int vkCode;
int scanCode;
public int flags;
int time;
int dwExtraInfo;
}
private int LowLevelKeyboardProc(
int nCode, int wParam,
ref KBDLLHOOKSTRUCT lParam)
{
bool blnEat = false;
switch (wParam)
{
case 256:
case 257:
case 260:
case 261:
//Alt+Tab, Alt+Esc, Ctrl+Esc, Windows Key
if (((lParam.vkCode == 9) && (lParam.flags == 32)) ||
((lParam.vkCode == 27) && (lParam.flags == 32)) || ((lParam.vkCode ==
27) && (lParam.flags == 0)) || ((lParam.vkCode == 91) && (lParam.flags
== 1)) || ((lParam.vkCode == 92) && (lParam.flags == 1)) || ((true) &&
(lParam.flags == 32)))
{
blnEat = true;
}
break;
}
if (blnEat)
return 1;
else return CallNextHookEx(0, nCode, wParam, ref lParam);
}
private void KeyboardHook(object sender, EventArgs e)
{
intLLKey = SetWindowsHookEx(WH_KEYBOARD_LL,new LowLevelKeyboardProcDelegate(LowLevelKeyboardProc),
System.Runtime.InteropServices.Marshal.GetHINSTANCE(
System.Reflection.Assembly.GetExecutingAssembly().GetModules()[0]).ToInt32(), 0);
}
private void ReleaseKeyboardHook()
{
intLLKey = UnhookWindowsHookEx(intLLKey);
}
private void checkBox1_CheckedChanged(object sender, EventArgs e)
{
if (checkBox1.Checked)
KeyboardHook(this, e);
else
ReleaseKeyboardHook();
}
}
}
2 ответа
Код в целом работает просто отлично. Вероятный эффект, который вы наблюдаете, связан с запуском в отладчике Visual Studio, что обычно подразумевает, что вы работаете в хостинг-процессе Visual Studio (vshost.exe).
Это означает, что вызов System.Reflection.Assembly.GetExecutingAssembly()
в вашем KeyboardHook()
Функция возвращает vshost.exe, а не ваш исполняемый файл, поэтому желаемый эффект от установки ловушки для вашего исполняемого файла не может быть достигнут.
Таким образом, чтобы увидеть ваш код в действии, вам нужно выполнить одно из следующих действий:
- запустить его за пределами Visual Studio
- запустить его в Visual Studio, но через меню "Отладка" -> "Пуск без отладки"
- отключить процесс размещения Visual Studio, см. ниже
Обратите внимание, что вы можете отключить процесс размещения Visual Studio, но учтите возможные побочные эффекты, приведите:
Когда процесс хостинга отключен, некоторые функции отладки недоступны или снижается производительность. Для получения дополнительной информации см. Отладка и Процесс хостинга.
Вы можете проверить мой ответ на связанный вопрос здесь. Обратите внимание на разницу в RegisterLowLevelHook
метод (ты назвал свой KeyboardHook
так ты знаешь что сравнивать). У меня не было никаких проблем с использованием этого, даже при отладке из VS. В основном, как утверждали другие, не используйте GetExecutingAssembly
метод, но вместо того, что я перечислил в другом ответе.
Вот выдержка из одной интересующей вас функции:
private IntPtr RegisterLowLevelHook(LowLevelKeyboardProc hook)
{
IntPtr handle = IntPtr.Zero;
using (Process currentProcess = Process.GetCurrentProcess())
using (ProcessModule currentModule = currentProcess.MainModule)
{
IntPtr module = Kernel32.GetModuleHandle(currentModule.ModuleName);
handle = User32.SetWindowsHookEx(HookType.KEYBOARD_LL, hook, module, 0);
}
return handle;
}