原文地址:http://blog.csdn.net/clever101/article/details/4339088
做一个项目,我的模块完成了(我的模块是对话框程序),但是别人的还没完成,我还得配合别人测试,具体就是单击按钮给别人发任务。做得多了我觉得用鼠标比较繁琐,于是我想到添加快捷键。我想着到时我都可以把屏幕关了只按键盘就行了,我按小键盘的数字键1就发任务1,按数字键2就发任务2……
给按钮定义快捷键,常规的做法有以下几种:
方法一 给按钮的 Caption 中写入特殊字符
如: 要给“打开”按钮加 快捷建 “Atl+O”,在 Caption 中写入 “打开(&O)”
方法二 注册系统热键:
1 声明热键消息处理函数原型
在.h中消息映射声明处(AFX_mSG字样之后)加入如下语句:
LRESULT OnHotKey(WPARAM wParam,LPARAM lParam);
ON_MESSAGE(WM_HOTKEY,OnHotKey);3. 为方便以后的操作
4.向系统登记热键
在OnCreate()函数中加入如下代码以向系统登记热键,本例子的热键设为
Ctrl+Shift+A.
RegisterHotKey(m_hWnd,1001,MOD_CONTROL|MOD_SHIFT,'A'); RegisterHotKey(m_hWnd,1002,MOD_CONTROL|MOD_SHIFT,'a');
5.处理热键
在消息处理函数OnHotKey()中对热键进行处理,并可加入用户希望运行的程序代码
LRESULT C****::OnHotKey(WPARAM wParam,LPARAM lParam) if(wParam==1001||wParam==1002) CWnd::SetForegroundWindow();//使得被激活窗口出现在前景 MessageBox("Hello!");6.程序运行完毕后解除热键
UnRegisterHotKey(m_hWnd,1001); UnRegisterHotKey(m_hWnd,1002);
7.编译并运行程序
这两种方法都有共同的弊端,就是必须使用组合键,那就是我必须动用两个手指头。我决心实现只需一个手指头就够了。开始我以为只要响应WM_CHAR消息消息就行了,后来发现不行,因为当一个对话框中什么都没有的时候,ONCHAR 事件才能给窗体接收到,否则默认的消息传递都是给输入焦点的窗口。经过一番功夫,我找到了两种方法:
方法一 利用键盘钩子:
简单介绍一下键盘钩子需要用到的函数:
WINDOWS调用挂接的回调函数时首先会调用位于函数链首的函数,我们只要将自己的回调函数置于链首,该回调函数就会首先被调用。那么如何将我们自己的回调函数置于函数链的链首呢?函数SetWindowsHookEx()实现的就是该功能。我们首先来看一下SetWindowsHookEx函数的原型:
HHOOK SetWindowsHookEx( int idHook, HOOKPROC lpfn, HINSTANCE hMod, DWORD dwThreadId );
第一个参数:指定钩子的类型,有WH_MOUSE、WH_KEYBOARD等十多种(具体参见MSDN)
第二个参数:标识钩子函数的入口地址
第三个参数:钩子函数所在模块的句柄;
第四个参数:钩子相关函数的ID用以指定想让钩子去钩哪个线程,为0时则拦截整个系统的消息。
具体实现是这样的:
运行VS 2005建一个MFC对话框程序,然后开始添加代码:
1. 定义一个全局的钩子句柄:
static HHOOK hkb=NULL;
LRESULT CALLBACK KeyboardProc(int nCode,WPARAM wParam,LPARAM lParam) { if(((DWORD)lParam&0x40000000) && (HC_ACTION==nCode)) { switch(wParam) //键盘按键标识 { // 按下小键盘数字键1就给第一个按钮发送WM_COMMAND消息,以下同 case VK_NUMPAD1: { HWND hWnd = (theApp.m_pMainWnd)->GetSafeHwnd(); ::SendMessage(hWnd, WM_COMMAND, MAKEWPARAM(IDOK, BN_CLICKED), (LPARAM)((theApp.m_pMainWnd)->GetDlgItem(IDOK)->m_hWnd)); break; } case VK_NUMPAD2: { HWND hWnd = (theApp.m_pMainWnd)->GetSafeHwnd(); ::SendMessage(hWnd, WM_COMMAND, MAKEWPARAM(IDCANCEL, BN_CLICKED), (LPARAM)((theApp.m_pMainWnd)->GetDlgItem(IDCANCEL)->m_hWnd)); break; } case VK_NUMPAD3: { HWND hWnd = (theApp.m_pMainWnd)->GetSafeHwnd(); ::SendMessage(hWnd, WM_COMMAND, MAKEWPARAM(IDC_BUTTON1, BN_CLICKED), (LPARAM)((theApp.m_pMainWnd)->GetDlgItem(IDC_BUTTON1)->m_hWnd)); break; } case VK_NUMPAD4: { HWND hWnd = (theApp.m_pMainWnd)->GetSafeHwnd(); ::SendMessage(hWnd, WM_COMMAND, MAKEWPARAM(IDC_BUTTON2, BN_CLICKED), (LPARAM)((theApp.m_pMainWnd)->GetDlgItem(IDC_BUTTON2)->m_hWnd)); break; } default: break; } } LRESULT RetVal = CallNextHookEx( hkb, nCode, wParam, lParam ); return RetVal; }
hkb=SetWindowsHookEx(WH_KEYBOARD,(HOOKPROC)KeyboardProc,theApp.m_hInstance,0);
4. 对话框的析构函数里卸载钩子:
if(hkb) UnhookWindowsHookEx(hkb);
BOOL CTestCharDlg::PreTranslateMessage(MSG* pMsg) { // TODO: 在此添加专用代码和/或调用基类 // 假如截获键盘按下消息,就分析按下的哪个键,然后给相应的按钮发送消息 if(pMsg->message==WM_KEYDOWN) { UINT iKey=(UINT)pMsg->wParam; switch(iKey) { case VK_NUMPAD1: SendMessage(WM_COMMAND, MAKEWPARAM(IDC_BUTTON1, BN_CLICKED), (LPARAM)(GetDlgItem(IDC_BUTTON1)->m_hWnd)); ::SetFocus(GetDlgItem(IDC_BUTTON1)->m_hWnd); break; case VK_NUMPAD2: SendMessage(WM_COMMAND, MAKEWPARAM(IDC_BUTTON2, BN_CLICKED), (LPARAM)(GetDlgItem(IDC_BUTTON2)->m_hWnd)); ::SetFocus(GetDlgItem(IDC_BUTTON2)->m_hWnd); break; case VK_NUMPAD3: SendMessage(WM_COMMAND, MAKEWPARAM(IDC_BUTTON3, BN_CLICKED), (LPARAM)(GetDlgItem(IDC_BUTTON3)->m_hWnd)); ::SetFocus(GetDlgItem(IDC_BUTTON3)->m_hWnd); break; case VK_NUMPAD4: SendMessage(WM_COMMAND, MAKEWPARAM(IDC_BUTTON4, BN_CLICKED), (LPARAM)(GetDlgItem(IDC_BUTTON4)->m_hWnd)); ::SetFocus(GetDlgItem(IDC_BUTTON4)->m_hWnd); break; default: break; } } return CDialog::PreTranslateMessage(pMsg); }
参考文献:
<1>利用键盘钩子开发按键发音程序,作者:GDGF,http://www.vckbase.com/document/viewdoc/?id=271
<2>为什么对话框不出理WM_CHAR消息, http://topic.csdn.net/t/20030622/19/1944358.html
<3> 钩子函数初步掌握篇,http://www.qqgb.com/Program/VC/VCZH/Program_54891.html