控件不响应消息的解决办法及PreTranslateMessage 消息截获

1.注意看控件的属性,在资源视图中发现有Notify这个属性,一定要设置成TRUE,比如Static控件。

2.控件是否获取到焦点,没有获取到可以使用SetFocus()函数得到焦点。

3.下断点调试。通过PreTranslateMessage( MSG* pMsg ),看是否有该消息。

PreTranslateMessage是消息在送给TranslateMessage函数之前被调用的,绝大多数本窗口的消息都要通过这里,比较常用,当你需要在MFC之前处理某些消息时,常常要在这里添加代码. MFC消息控制流最具特色的地方是CWnd类的虚拟函数PreTranslateMessage(),通过重载这个函数,我们可以改变MFC的消息控制流程,甚至可以作一个全新的控制流出来。只有穿过消息队列的消息才受PreTranslateMessage()影响,采用SendMessage()或其他类似的方式向窗口直接发送的而不经过消息队列的消息根本不会理睬PreTranslateMessage()的存在。

注意:PreTranslateMessage函数不能直接调用模态对话框。

消息循环的关键代码如下:

do 
{ 
   if ( !PreTranslateMessage(&msg)) 
    { 
      ::TranslateMessage(&msg); 
      ::DispatchMessage(&msg); 
    } 
}while(::PeekMessage(pMsg,NULL,NULL,NULL,PM_NOREMOVE));

从的代码中我们可以看到,如果PreTranslateMessage为真,那就进入不会进TranslateMessage,跟不用说是消息的分发与响应了,这也就是为什么可以通过重写PreTranslateMessage可以消息队列中的消息,甚至是重定向消息,PreTranslateMessage为真后,这个消息循环就空转,像一个死循环,为什么说像,而不是是,是因为windows会处理是不是idle的情况,当创建一个模态框后,焦点转到模态框,父窗口这是应该无法处理到idle的情况,而进入死循环,导致负荷一下飚升到50%(双核),修改PreTranslateMessage,出了要截获的消息,其他消息返回假,让消息循环继续处理消息。

在MFC中,PreTranslateMessage()是虚函数,我们可以重载它来处理键盘和鼠标消息。该函数表示在消息处理(TranslateMessage()和DispatchMessage()等)前所作的操作,如果函数返回值为TRUE,那么消息处理即终止,不会调用TranslateMessage()和DispatchMessage()来翻译和分发消息给相应的窗口;若返回值为FALSE,才会调用翻译和分发消息函数。

 

 

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
MFC自定义控件消息响应方式一般有以下几种: 1. 消息映射法:在控件所在的窗口类中添加响应函数,然后在消息映射表中添加对应的消息映射。例如: ``` BEGIN_MESSAGE_MAP(CMyWnd, CWnd) ON_WM_PAINT() ON_WM_LBUTTONDOWN() END_MESSAGE_MAP() void CMyWnd::OnPaint() { // TODO: 在此处添加消息处理程序代码 } void CMyWnd::OnLButtonDown(UINT nFlags, CPoint point) { // TODO: 在此处添加消息处理程序代码 } ``` 2. 消息钩子法:在控件类中重载虚函数PreTranslateMessage,然后在函数中进行消息处理。例如: ``` class CMyControl : public CWnd { public: virtual BOOL PreTranslateMessage(MSG* pMsg); }; BOOL CMyControl::PreTranslateMessage(MSG* pMsg) { if (pMsg->message == WM_LBUTTONDOWN) { // TODO: 在此处添加消息处理程序代码 return TRUE; } return CWnd::PreTranslateMessage(pMsg); } ``` 3. 消息分发法:在控件类中重载虚函数WindowProc,然后在函数中进行消息处理。例如: ``` class CMyControl : public CWnd { public: virtual LRESULT WindowProc(UINT message, WPARAM wParam, LPARAM lParam); }; LRESULT CMyControl::WindowProc(UINT message, WPARAM wParam, LPARAM lParam) { switch (message) { case WM_PAINT: // TODO: 在此处添加消息处理程序代码 break; case WM_LBUTTONDOWN: // TODO: 在此处添加消息处理程序代码 break; default: break; } return CWnd::WindowProc(message, wParam, lParam); } ``` 以上三种方法都可以用来响应自定义控件消息,具体应该根据实际情况选择。其中,消息映射法是MFC中最常用的方式,也是最简单易懂的方式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值