回 帖 发 新 帖 刷新版面

主题:问一个很菜的问题.

有谁知道如何屏蔽系统热键的..要求能屏蔽WIN键、CTRL+ALT+DEL、CTRL+SHILT+ESC

回复列表 (共2个回复)

沙发

在win95/win98操作系统下:

在win95/win98操作系统下屏蔽系统热键可以用一种非常简单的方法做到,就是让系统认为当前处于屏保状态。让系统认为当前处于屏保状态的方法是:

             SystemParametersInfo(SPI_SETSCREENSAVERRUNNING,true,&pOld,SPIF_UPDATEINIFILE);
SPI_SETSCREENSAVERRUNNING 该参数在微软的文档中并不建议使用,是因为在该状态下,系统无法弹出任务列表,一旦某一进程被挂起,只能重新启动,有可能造成数据丢失,但可以达到所需效果,我们仍然使用它。
当程序要取消该状态:
      SystemParametersInfo(SPI_SETSCREENSAVERRUNNING,false,&pOld,SPIF_UPDATEINIFILE);

pOld 为我们定义的布尔变量。 

在WinNT 4.0 Service Pack 3或 Win2000下
 也许你认为使用一个键盘钩子WH_KEYBOARD不就解决问题了吗?问题并不是那么简单。这是因为键盘钩子WH_KEYBOARD并不能截取到系统键的输入!幸好在Winnt4.0 Service Pack3或Win2000下系统给我们提供了一个底层系统钩子(Low Level Hook)WH_KEYBOARD_LL。下面我们来讨论底层键盘钩子的用法。首先需要按装钩子,这需要API函数:

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

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

       下面是实现底层键盘钩子的完整源代码,在VC5.0,VC6.0下调试通过。
               #include <Windows.h>
 
               LRESULT CALLBACK LowLevelKeyboardProc(int nCode, 
            WPARAM wParam, LPARAM lParam) {
 
            BOOL fEatKeystroke = FALSE;
 
   I        f (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, HINSTANCE, PTSTR pszCmdLine, int) {
 
   //按装底层键盘钩子
   HHOOK hhkLowLevelKybd  = SetWindowsHookEx(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);


板凳

在低于WinNT 4.0 Service Pack 2 或更低版本的NT操作系统下
很遗憾,在低于WinNT 4.0 Service Pack 2 或更低版本的NT操作系统下我们所能做的并不能屏蔽掉所有的系统热键,而仅仅可以屏蔽Alt+Tab,Alt+Esc。
完成这一功能的方法是使用RegisterHotKey函数。在调用该函数后你的进程会在ALT+TAB按下时比系统先得到通知。你需要处理的消息是WM_HOTKEY,下面是相关代码讲解。 

RegisterHotKey函数原型及说明:
BOOL RegisterHotKey(
  HWND hWnd,         // window to receive hot-key notification
  int id,            // identifier of hot key
  UINT fsModifiers,  // key-modifier flags
  UINT vk            // virtual-key code);
参数 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);
}
 
WM_HOTKEY消息含义:
idHotKey = (int) wParam;              // identifier of hot key 
fuModifiers = (UINT) LOWORD(lParam);  // key-modifier flags 
uVirtKey = (UINT) HIWORD(lParam);     // virtual-key code 
这三个值分别和调用RegisterHotKey时的id,fsModifiers,vk对应。
最后你需要在文件中定义ON_MESSAGE消息映射。
在头文件中:
class CMainFrame : public XXXX
{
         afx_msg LONG OnHotKey(WPARAM wP,LPARAM lP);
}
在CPP文件中MESSAGE_MAP处添加:
ON_MESSAGE(WM_HOTKEY,OnHotKey)
该热键在你的进程运行时一直有效,在进程结束后其状态会被系统恢复。

我来回复

您尚未登录,请登录后再回复。点此登录或注册