在程序开发过程中,为了满足一些特殊的需求,有时需要屏蔽系统热键。这些系统热键包括Alt+Tab、Ctrl+Alt+Del、Alt+Esc、Ctrl+Esc、Win等。由于屏蔽系统热键涉及到系统的安全,所以一般不推荐这样做,但由于存在实际的应用需求,所以作为编程人员还是要对此加以了解。本文分三种情况来讨论屏蔽系统热键的方法。

在Windows 95/98操作系统下

  在Windows 95/98操作系统下屏蔽系统热键可以用一种非常简单的方法做到,就是让系统认为当前机器正处于屏幕保护状态:

SystemParametersInfo(SPI_SETSCREENSAVERRUNNING,true,&pOld,SPIF_UPDATEINIFILE);

微软公司的技术文档中不建议使用SPI_SETSCREENSAVERRUNNING参数,因为在该状态下,系统无法弹出任务列表,一旦某一进程被挂起,只能重新启动,并有可能造成数据丢失,所以要小心使用。

  取消该状态:

SystemParametersInfo(SPI_SETSCREENSAVERRUNNING,false,&pOld,SPIF_UPDATEINIFILE);

  其中pOld 为自定义的布尔变量。

在Windows NT 4.0 SP3或

Windows 2000下

  有人认为使用一个键盘钩子WH_KEYBOARD就可以解决问题,但实际上问题并不是那么简单。这是因为键盘钩子WH_KEYBOARD不能截取到系统键的输入。但笔者发现在Windows NT 4.0 SP3或 Windows 2000下系统提供了一个底层系统钩子(Low Level Hook)WH_KEYBOARD_LL。WH_KEYBOARD_LL的工作流程如图1所示。

 



图1

从图1我们可以看到,底层键盘钩子存在于用户敲击键盘和系统处理之间,而普通键盘钩子则存在于系统产生WM_KEY***消息之后。很清楚,普通键盘钩子只能截获WM_KEY***消息,而不能对系统键进行操作。但是底层键盘钩子有一个致命的弱点,就是如果调用它的进程或线程出现死循环,则系统将不能处理任何键盘操作。为了解决这个问题,微软在注册表中给出了一个底层键盘钩子处理的限制时间,如果超出了这个时间,系统将进入正常处理。这个时间键值存储在注册表的HKEY_CURRENT_USER/Control Panel/Desktop/ LowLevelHooksTimeout下。

  下面介绍底层键盘钩子的用法。首先需要安装钩子:

HHOOK SetWindowsHookEx(int iHookCode, 
HOOKPROC lpfn,HINSTANCE hModule,DWORD dwThreadId);

  其中,第一个参数是钩子的类型;第二个参数是钩子函数的地址;第三个参数是包含钩子函数的模块句柄;第四个参数指定监视的线程。如果指定确定的线程,即为线程专用钩子;如果指定为空,即为全局钩子。其中,全局钩子函数必须包含在DLL(动态链接库)中,而线程专用钩子还可以包含在可执行文件中。得到控制权的钩子函数在完成对消息的处理后,如果想要该消息继续传递,那么它必须调用另外一个SDK中的API函数CallNextHookEx来传递它。钩子函数也可以通过直接返回TRUE来丢弃该消息,并阻止该消息的进一步传递。

  下面是实现底层键盘钩子的部分源代码:

LRESULT CALLBACK LowLevelKeyboardProc(int nCode,WPARAM wParam, LPARAM lParam) 
{ 
  BOOL fEatKeystroke = FALSE; 
   If (nCode == HC_ACTION) { 
    switch (wParam) { 
    case WM_KEYDOWN: 
    case WM_SYSKEYDOWN: 
    case WM_KEYUP: 
    case WM_SYSKEYUP: 
     PKBDLLHOOKSTRUCT p = 
(PKBDLLHOOKSTRUCT) lParam; 
     fEatKeystroke =((p->vkCode == 
VK_TAB) && ((p->flags & LLKHF_ALTDOWN) 
!= 0)) ||((p->vkCode == VK_ESCAPE) && ((p->flags & LLKHF_ALTDOWN) != 0)) ||((p->vkCode == VK_ESCAPE) && 
((GetKeyState(VK_CONTROL) & 0x8000) != 0)); 
     break; 
} 
   } 
return(fEatKeystroke ? 1 : CallNextHookEx(NULL, nCode, wParam, lParam)); 
  } 
  int WINAPI WinMain(HINSTANCE hinstExe, HIN 
STANCE, PTSTR pszCmdLine, int) 
{ 
  //安装底层键盘钩子 
HHOOK hhkLowLevelKybd = SetWindow 
sHookEx(WH_KEYBOARD_LL, LowLevelKeyboardProc, 
hinstExe, 0); 
  MessageBox(NULL,TEXT(“Alt+Esc, Ctrl+Esc, and Alt+Tab are now disabled./n”),TEXT(“Click /“Ok/” to terminate this application and re-enable these keys.”),TEXT(“Disable Low-Level Keys”), MB_OK); 
  UnhookWindowsHookEx(hhkLowLevelKybd); 
  return(0); 
  }

在低于Windows NT 4.0 SP2的

NT操作系统下

  很遗憾,笔者在低于Windows NT 4.0 SP2或更低版本的NT操作系统下没有找到能屏蔽掉所有的系统热键的方法,本文介绍的方法仅仅可以屏蔽Alt+Tab和Alt+Esc。该方法是使用RegisterHotKey函数,在调用该函数后进程会在Alt+Tab按下时比系统优先一步得到通知。程序员需要处理的消息是WM_HOTKEY。

  RegisterHotKey函数原型如下:

BOOL RegisterHotKey(HWND hWnd,int id,UINT fsModifiers,UINT vk);

  其中,参数id为自定义的一个ID值,对一个线程来讲其值必需在0x0000~0xBFFF范围之内,对DLL来讲其值必需在0xC000~0xFFFF 范围之内,在同一进程内该值必须唯一;参数fsModifiers指明与热键联合使用的按键,可取值为:MOD_ALT、MOD_CONTROL、MOD_WIN和MOD_SHIFT;参数vk指明热键的虚拟键码。

// 初始化 
  CMainFrame::CMainFrame() 
  { 
   m_nHotKeyID = 100; 
  BOOL m_isKeyRegistered = RegisterHotKey 
(GetSafeHwnd(), m_nHotKeyID,MOD_ALT, VK_TAB); 
   ASSERT(m_isKeyRegistered != FALSE); 
  } 
  //取消 
  CMainFrame::~CMainFrame() 
  { 
   BOOL m_iskeyUnregistered = UnregisterHotKey(GetSafeHwnd(), m_nHotKeyID); 
   ASSERT(m_isKeyUnregistered != FALSE); 
  }

  最后需要在文件中定义ON_MESSAGE消息映射。在头文件中添加如下代码:

class CMainFrame : public XXXX 
  { 
   afx_msg LONG OnHotKey(WPARAM wP, 
LPARAM lP); 
  }

  在CPP文件中的MESSAGE_MAP处添加代码:

ON_MESSAGE(WM_HOTKEY,OnHotKey)

  则该热键在进程运行时一直有效,在进程结束后其状态会被系统自动恢复。