您的位置:首页 > 其它

实现自定义对话框程序快捷键的两种方法

2012-10-23 12:50 253 查看
原文地址:/article/2582319.html做一个项目,我的模块完成了(我的模块是对话框程序),但是别人的还没完成,我还得配合别人测试,具体就是单击按钮给别人发任务。做得多了我觉得用鼠标比较繁琐,于是我想到添加快捷键。我想着到时我都可以把屏幕关了只按键盘就行了,我按小键盘的数字键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()中对热键进行处理,并可加入用户希望运行的程序代码
LRESULT   C****::OnHotKey(WPARAM     wParam,LPARAM     lParam)           
    if(wParam==1001||wParam==1002)           
    CWnd::SetForegroundWindow();//使得被激活窗口出现在前景           
    MessageBox("Hello!");
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. 定义钩子回调函数
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函数,具体代码如下:
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>利用键盘钩子开发按键发音程序,作者:GDGFhttp://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
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: