作者:朱金灿
来源:http://www.cnblogs.com/clever101
做一个项目,我的模块完成了(我的模块是对话框程序),但是别人的还没完成,我还得配合别人测试,具体就是单击按钮给别人发任务。做得多了我觉得用鼠标比较繁琐,于是我想到添加快捷键。我想着到时我都可以把屏幕关了只按键盘就行了,我按小键盘的数字键1就发任务1,按数字键2就发任务2……
给按钮定义快捷键,常规的做法有以下几种:
方法一 给按钮的 Caption 中写入特殊字符
如: 要给“打开”按钮加 快捷建 “Atl+O”,在 Caption 中写入 “打开(&O)”
方法二 注册系统热键:
1 声明热键消息处理函数原型
在.h中消息映射声明处(AFX_mSG字样之后)加入如下语句:
LRESULT OnHotKey(WPARAM wParam,LPARAM lParam);
2. 消息与相应处理函数相关联
在.Cpp中加入消息映射宏,使消息与相应处理函数发生关系,
ON_MESSAGE(WM_HOTKEY,OnHotKey);
3. 为方便以后的操作
预先在类中创建一个响应WM_CREATE和WM_DESTROY消息的函数
OnCreate()与OnDestroy()的框架
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()中对热键进行处理,并可加入用户希望运行的程序代码

Code
LRESULT C****::OnHotKey(WPARAM wParam,LPARAM lParam)
if(wParam==1001||wParam==1002)
CWnd::SetForegroundWindow();//使得被激活窗口出现在前景
MessageBox("Hello!");
//用户可在此添加代码
return 0;
6.程序运行完毕后解除热键
在OnDestroy()中通过UnRegisterHotKey()解除热键登记,释放系统资源.
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;
2. 定义钩子回调函数:

Code
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;
}
3.在对话框OnInitDialog()函数调用SetWindowsHookEx函数来设置键盘钩子:
hkb=SetWindowsHookEx(WH_KEYBOARD,(HOOKPROC)KeyboardProc,theApp.m_hInstance,0);
4. 对话框的析构函数里卸载钩子:
if(hkb)
UnhookWindowsHookEx(hkb);
方法二更为简单,就是重载PreTranslateMessage函数,具体代码如下:

Code
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
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· [AI/GPT/综述] AI Agent的设计模式综述