销售人员卡通形象:MFC对话框中对键盘消息的处理

来源:百度文库 编辑:中财网 时间:2024/05/05 09:16:18


2010-10-20 15:43:28|  分类: 编程 |  标签: |字号大中小 订阅

    今天想做一个程序用来测试键盘上的按键是否有效,因为对于字符键来说,可以用文本输入的方法测试其是否正常,但对于控制键来说这个就不太容易,所以想设计一个程序来测试下,用VC6创建一个基于对话框的程序,结果发现对于不能直接响应键盘按键的消息。

    原来,在MFC中,对话框程序在完成程序的初始化后,就在程序主线程中,调用CWinThread::Run函数。在该函数中,首先调用API函数PeekMessage,而函数PeekMessage检查线程消息队列,如果消息存在,就将该消息放于指定的MSG结构中,以后的消息处理都将针对这一MSG结构对象。捕获消息后,该函数将捕获的消息进行预处理,然后再将消息传递给相应的窗口处理函数。

    键盘消息被拦截而得不到正常响应,其中的关键就是Run函数对消息的预处理。在Run函数中,调用了函数CWinThread::PumpMessage,就是利用这一函数,MFC实现了对消息的分流,使得消息沿着MFC对各种消息规定的路线流动,直到被正确响应。

    函数PumpMessage调用了函数CWinThread::PreTranslateMessage对消息进行处理,如果该函数不对消息进行处理,则调用API函数TranslateMessage函数将虚拟键消息转换为字符消息并调用DispatchMessage分发消息给窗口处理程序。在对话框中,程序用CWinThread::PreTranslateMessage函数处理了键盘消息,所以对话框程序是否要响应键盘消息,将完全由CWinThread::PreTranslateMessage函数来决定了。

    在CWnd及其派生类的成员函数PreTranslateMessage函数是一个虚函数,可以通过重载来改变其处理过程。在默认情况下,没有重载这一函数。

例子如下,在VC6的Class view中找到相应的对话框类单击右键,在右键菜单中选择Add Virtual Fuction...项,然后找到PreTranslateMessage虚函数进行加载。相应的代码中如下:

BOOL CKeyinTstDlg::PreTranslateMessage(MSG* pMsg)    //CKeyinTstDlg我为自己创建的对话框类
{
 // TODO: Add your specialized code here and/or call the base class
 if(pMsg->message == WM_KEYDOWN)
 {
  m_KeyCode=1;
  m_KeyName="dsfsdf";

MessageBox("有键被按下");
pdateData(FALSE);
 }

 return CDialog::PreTranslateMessage(pMsg);
}

 

对话框的MFC,为什么一点回车键,程序就退出(转贴)

(2009-05-20 22:03:46)转载 标签:

回车键

关闭

对话框

vc

mfc

分类: mfc一些界面设计技巧

对话框如何屏蔽回车键enter  
  当用户按下回车键时,Windows发送一个特殊的WM_GETDEFID消息来获得缺省的命令ID,Windows再将它作为WM_COMMAND发送。所以,你要做的就是重载WM_GETDEFID消息,在有关Windows的文档中是这样描述WM_GETDEFID返回值的:“如果有缺省得按钮,则返回值的高位字包含DC_HASDEFID,低位字包含控制的标识符。否则,返回值是零。”  
  BEGIN_MESSAGE_MAP(CMyDlg,   CDialog)  
          ON_MESSAGE(DM_GETDEFID,   OnGetDefID)  
          ...  
  END_MESSAGE_MAP()  
   
  LRESULT   CMyDlg::OnGetDefID(WPARAM   wp,   LPARAM   lp)    
  {  
          return   MAKELONG(0,DC_HASDEFID);    
  }  
     
   
    另外的一个比较笨的方法:添加一个按钮,IDOK,然后删除,  
  CDialog的OnOK虚函数控制了OK按钮,它负责触发数据交换以及从对话框中退出,然而,Enter键碰巧也具有同样的效果。也许这是我们所需要的,也可能我们不需要这一功能,就是说不希望用户在编辑框中按Enter键后出现关闭程序的情况。怎么解决这一问题?  
   
          一、原因  
   
          当用户按下Enter键时,Windows就会自动去找输入焦点落在了哪一个按钮上,当获得焦点的按钮的四周将被点线矩形包围。如果所有按钮都没有获得输入焦点,Windows就会自动去寻找程序或资源所指定的默认按钮(默认按钮边框较粗)。如果对话框没有默认按钮,那么即使对话框中没有OK按钮,OnOK函数也会自动被调用。  
   
          二、解决办法  
   
          为了使Enter键无效,最简单的办法就是将CExDlg的OnOK函数写成空函数,然后针对OK按钮写一个新的函数来响应。下面是具体步骤:  
   
          首先建立一个以Ex为名的基于对话框的程序。  
   
          1.用ClassWizard映射IDOK按钮到OnOK虚函数。  
   
          在ClassWizard中,从CExDlg的Object   Ids列表中选择IDOK,然后单击BN_CLICKED,这样即产生了OnOK函数的原形和框架。  
   
          2.利用对话框编辑器改变OK按钮的ID。  
   
          选择OK按钮,将它的ID由IDOK改为IDC_OK,并去掉Default   Button属性的复选标记。这样OnOK函数即被分离出来。  
   
          3.用ClassWizard创建一个OnClickedOK成员函数。  
   
          在ClassWizard中,从CExDlg的Object   Ids列表中选择IDC_OK,然后单击BN_CLICKED,函数名用OnClickedOK,产生了OnClickedOK函数的原形和框架。  
   
          4.编辑代码。  
   
          void   CExDlg::OnOK()  
          {  
                  //   TODO:   Add   extra   validation   here  
                  //   CDialog::OnOK();  
          }  
   
          void   CExDlg::OnClickedOk()  
          {  
                  //   TODO:   Add   your   control   notification   handler   code   here  
                  CDialog::OnOK();  
          }  
   
          5.编译并测试程序。  
   
          重新编译程序,然后测试一下Enter键,你会发现按下Enter键后将没有任何动作,而单击OK按钮则仍然会和以前一样。  
   
          三、   OnCancel的处理  
   
          正如Enter键可以触发对OnOK函数的调用一样,Esc键也可以触发对OnCancel函数的调用。它将导致控制从对话框中退出。不过读者可以像对OK按钮一样处理,通过设置一个无效的OnCancel函数来防止这一情况的出现  
   
  第三个方法  
  BOOL   CxxxDlg::PreTranslateMessage(MSG*   pMsg)  
  {  
          //   修改回车键的操作反应  
          if(pMsg->message==WM_KEYDOWN   &&   pMsg->wParam==VK_RETURN)  
          {  
                  UINT   nID   =   GetFocus()->GetDlgCtrlID();  
                  if(ctButton   !=   GetCtrlType(nID))  
                  {  
                          NextDlgCtrl();  
                          int   nFocusID   =   GetFocus()->GetDlgCtrlID();  
                          if(nFocusID   ==   1001)  
                                  ((CComboBox*)(GetFocus()->GetParent()))->ShowDropDown();  
                          else   if(ctComboBox   ==   GetCtrlType(nFocusID))  
                                  ((CComboBox*)GetFocus())->ShowDropDown();  
   
                          return   TRUE;  
                  }  
          }  
          return   CDialog::PreTranslateMessage(pMsg);