PreTranslateMessage作用和使用方法


函數原型:virtual BOOL PreTranslateMessage( MSG *pMsg) 

1、在MFC中,PreTranslateMessage是虛函數,是用來截獲消息的。我們可以通過重載它來處理鍵盤和鼠標消息。在sdk中,這有所不同,我們必須在回調函
數 LRESULT   CALLBACK   WndProc(HWND   hWnd,   UINT   message,   WPARAM   wParam,   LPARAM   lParam)中處理消息。它和PreTranslateMessage起的作用是類似的,只是MFC封裝的更好而已。
2、重載該函數可以實現窗口消息在派發給窗口函數TranslateMessage()和DispatchMessae()之前的過濾,缺省的實現是完成加速鍵的翻譯。 

3、 該函數表示在消息處理(TranslateMessge()和DispatchMessage()等)前所作的操作,如果函數返回值為TRUE,那么消息 處理即終止,不會調用TranslateMessge()和DispatchMessage()來翻譯和分發消息給相應的窗口;若返回值為FALSE,才 會調用翻譯和分發消息函數。

4、在win32程序中,關於消息有兩種傳遞方式:
a. MFC消息,MFC會把所有的消息一條條放到一個AFX_MSGMAP_ENTRY結構中,形成一個數組,該數組存放了所有的消息和與它們相關的參數。也可以說消息是放到消息隊列里去了。 b. 采用SendMessage()或其他類似的方式向窗口直接發送的而不經過消息隊列的消息。 這兩種方式中只有第一種(穿過消息隊列的消息)才受PreTranslateMessage()影響,第二種消息並不會理睬PreTranslateMessage()的存在。
 
一、是否調用TranslateMessage()和DispatchMessage()是由PreTranslateMessage()函數的返回值決定的,如果該函數返回TRUE,則不會把該消息分發給窗口函數處理。
二、傳給PreTranslateMessage()的消息是未經翻譯過的消息,它沒有經過
TranslateMessage()處理。例如:可以在該函數中使用(pMsg->wParam == VK_RETURN)來攔截回車鍵,其中wParam中存放的是鍵盤上字符的虛擬碼。
三、在WindowProc里不能處理WM_Char消息。(WindowProc函數見MFC消息響應機制一文)
四、SetWindowText會發送WM_Char給窗口。

、PeekMessage和GetMessage的區別:
    GetMessage在沒有消息的時候等待消息,cpu當然低
    PeekMessage沒有消息的時候立刻返回,所以cpu占用率高。因為游戲不能靠windows消息驅動,所以要用PeekMessage(); 
另一篇文章:
    在一個WIN32程序中,WINDOWS會將消息傳遞給相應的窗口。但是消息不是立即就被傳遞給相應的窗口,而是會從整個程序最頂層的窗口傳遞到下一級 窗口,再傳遞到下一級窗口,直到傳遞給目標窗口。在整個過程中,有些消息,在某些特定的情況下,無法默認傳遞到目標窗口的。比如用戶在EDIT控件中按下 回車鍵,CANCEL鍵等,如果EDIT窗口之前有對話框窗口,對話框會默認處理回車消息(即響應ONOK函數,然后關閉對話框),然后退出消息傳遞。所以EDIT會收不到。要解決這個問題,可以在EDIT窗口之前所有的對話框中重載PreTranslateMessage函數,然后在函數內加上:
if (pMsg->message==WM_KEYDOWN && pMsg->wParam==VK_RETURN) //如果消息類型為
           WM_KEYDOWN                                                                        //並且用戶按下的是回車
   return FALSE;    //不翻譯消息,直接將消息傳遞下去。具體可查MSDN。 注意,這里返回值不能為TRUE,TRUE的意思是翻譯消息后退出消息傳遞,如此一來雖然也能避開對話框默認處理,但是會退出消息傳遞,這樣EDIT控件照樣得不到消息。 如此就可避開對話框默認處理,將消息傳遞下去。注意:只有對話框才會默認處理按下回車,CANCEL消息,其他控件窗口則不會,所以在其他窗口中不必重載PreTranslateMessage函數,當然如果重載了也不會錯。 附:關於PreTranslateMessage()函數的小程序示例:
BOOL CSearchuserDlg::PreTranslateMessage(MSG* pMsg) 

{
           if(pMsg->message==WM_KEYDOWN)  //判斷是否有按鍵按下     

           {
                  switch(pMsg->wParam)        

           {
                     case VK_DOWN:     //表示是方向鍵中的向下的鍵               //add handle code here           

                        break;
                    case VK_UP:      //表示是方向鍵中的向上的鍵                //add handle code here           

                        break;           

                 default:           

                      break;        

           }    

       } 

}
**********************************************************************
PeekMessage 一般是用來重載消息循環,避免程序停止響應。           舉例:        
  MSG   msg;   
  if(::PeekMessage(&msg,NULL,0,0,PM_REMOVE))

{          

        if(msg.message==WM_QUIT)          

          {   
            ::PostQuitMessage(-1);            

         }   

        if(!AfxGetApp()->PreTranslateMessage(&msg))            

       {   
            ::TranslateMessage(&msg);                

            ::DispatchMessage(&msg);            

        }      

}
******************************************************************************
界面上有很多個控件,如edit,如何知道當前的焦點是在哪個控件上面?
1. GetFocus()可以得到當前控件的句柄, 你再枚舉子窗口句柄判斷, 跟得到的句柄一致的即為哪個被點中。
2. CWnd* pCtrl = CWnd::GetFocus();  

int iCtrlID = pCtrl->GetDlgCtrlID();  

if(iCtrlID == IDC_EDIT1)  {  } 
else if(iCtrlID == IDC_EDIT2)  {  }  ... 
3. GetFocus() 功能:確定當前焦點位於哪個控件上。函數執行成功時返回當前得到焦點控件的窗口句柄,發生錯誤時返回無效引用。
用法: 應用程序利用IsValid()函數可以檢測GetFocus()是否返回有效的控件引用。同時,使用TypeOf()函數可以確定控件的類型。
 


 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM