原文鏈接: http://blog.sina.com.cn/s/blog_5f8817250100taab.html
本文大部分來自MSDN和網友的博客,我在實踐的基礎上再作了一些總結。
1,虛擬鍵(VK_*)
鍵盤上每一個鍵對應一個掃描碼,掃描碼是OEM廠商制定的,不同的廠商生產的鍵盤同樣一個按鍵的掃描碼都有可能出現不一致的情況,為了擺脫由於系統設備不一致的情況,通過鍵盤驅動程序將掃描碼映射為統一的虛擬鍵碼表示,從而達到所有的設備都有一個統一的虛擬鍵,比如回車鍵的虛擬鍵是VK_RETURN。
Windows定義的虛擬鍵都定義在WinUser.h這個頭文件里面,都是以VK_作為前綴。
2,激活/關閉窗口對鍵盤的消息
激活/關閉消息:WM_SETFOCUS/WM_KILLFOCUS
創建光標:CreateCaret(...)
設置光標位置:SetCaretPos(…)
在窗口中顯示光標:ShowCaret(…)
銷毀光標:DestroyCaret()
3,鍵盤消息
1)字符消息
系統字符消息
WM_SYSCHAR:系統字符
WM_SYSDEADCHAR:系統死字符
非系統按鍵消息
WM_CHAR:非系統字符
WM_DEADCHAR:非系統死字符
2)按鍵消息
系統按鍵消息:與ALT鍵相組合的組合鍵(無論用戶處理否,都需要最后調用DefWindowProc(hWnd,iMessage,wParam,lParam))
WM_SYSKEYDOWN
WM_SYSKEYUP
非系統按鍵消息:
WM_KEYDOWN
WM_KEYUP
注意:
a) 除Print鍵之外都有“按下”消息。
b) 所有鍵都存在“彈起”消息。
c) 根據MSDN說明,只有下面這些鍵才會產生字符消息:
- 任何字符鍵
- 回退鍵(BACKSPACE)
- 回車鍵(carriage return)
- ESC
- SHIFT + ENTER (linefeed 換行)
- TAB
我們是怎么收到WM_CHAR的呢?就是因為我們在消息循環時調用了TranslateMessage對鍵盤消息進行翻譯,
如果消息為WM_KEYDOWN或者WM_SYSKEYDOWN,並且按鍵與位移狀態相組合產生一個字符,則TranslateMessage把字符消息放入消息隊列中。此字符消息將是GetMessage從消息隊列中得到的按鍵消息之后的下一個消息。
在我們處理這個消息時,對應的wParam不是虛擬鍵,而是ANSI或Unicode字符代碼,一般情況下我們可以這樣用: (TCHAR)wParam;
4,消息順序
因為TranslateMessage函數從WM_KEYDOWN和WM_SYSKEYDOWN消息產生了字符消息,所以字符消息是夾在按鍵消息之間傳遞給窗口消息處理程序的。例如,如果Caps Lock未打開,而使用者按下再釋放A鍵,則窗口消息處理程序將接收到如表6-10所示的三個消息:
表6-10
消息 |
按鍵或者代碼 |
WM_KEYDOWN |
「A」的虛擬鍵碼(0x41) |
WM_CHAR |
「a」的字符代碼(0x61) |
WM_KEYUP |
「A」的虛擬鍵碼(0x41) |
如果您按下Shift鍵,再按下A鍵,然后釋放A鍵,再釋放Shift鍵,就會輸入大寫的A,而窗口消息處理程序會接收到五個消息,如表6-11所示:
表6-11
消息 |
按鍵或者代碼 |
WM_KEYDOWN |
虛擬鍵碼VK_SHIFT (0x10) |
WM_KEYDOWN |
「A」的虛擬鍵碼(0x41) |
WM_CHAR |
「A」的字符代碼(0x41) |
WM_KEYUP |
「A」的虛擬鍵碼(0x41) |
WM_KEYUP |
虛擬鍵碼VK_SHIFT(0x10) |
Shift鍵本身不產生字符消息。
如果使用者按住A鍵,以使自動重復產生一系列的按鍵,那么對每條WM_KEYDOWN消息,都會得到一條字符消息,如表6-12所示:
表6-12
消息 |
按鍵或者代碼 |
WM_KEYDOWN |
「A」的虛擬鍵碼(0x41) |
WM_CHAR |
「a」的字符代碼(0x61) |
WM_KEYDOWN |
「A」的虛擬鍵碼(0x41) |
WM_CHAR |
「a」的字符代碼(0x61) |
WM_KEYDOWN |
「A」的虛擬鍵碼(0x41) |
WM_CHAR |
「a」的字符代碼(0x61) |
WM_KEYDOWN |
「A」的虛擬鍵碼(0x41) |
WM_CHAR |
「a」的字符代碼(0x61) |
WM_KEYUP |
「A」的虛擬鍵碼(0x41) |
如果某些WM_KEYDOWN消息的重復計數大於1,那么相應的WM_CHAR消息將具有同樣的重復計數。
組合使用Ctrl鍵與字母鍵會產生從0x01(Ctrl-A)到0x1A(Ctrl-Z)的ASCII控制代碼,其中的某些控制代碼也可以由表6-13列出的鍵產生:
表6-13
按鍵 |
字符代碼 |
產生方法 |
ANSI C控制字符 |
Backspace |
0x08 |
Ctrl-H |
\b |
Tab |
0x09 |
Ctrl-I |
\t |
Ctrl-Enter |
0x0A |
Ctrl-J |
\n |
Enter |
0x0D |
Ctrl-M |
\r |
Esc |
0x1B |
Ctrl-[ |
|
最右列給出了在ANSI C中定義的控制字符,它們用於描述這些鍵的字符代碼。
我們一般可以這樣處理WM_CARH消息:
case WM_CHAR:
{
switch (wParam)
{
case 0x08:
// Process a backspace.
break;
case 0x0A:
// Process a linefeed.
break;
case 0x1B:
// Process an escape.
break;
case 0x09:
// Process a tab.
break;
case 0x0D:
// Process a carriage return.
break;
default:
// Process displayable characters.
break;
}
}
我們可以在WM_CHAR里面判斷當前是否有指定的鍵被按下:
BOOL bIsCtrl = (::GetAsyncKeyState(VK_CONTROL) & 0x8000); (MFC源碼 afxcolordialog.cpp 460行)
或
BOOL bIsCtrl = (::GetKeyState(VK_CONTROL) & 0x8000);
下面我解釋一下鍵盤消息的lParam參數,這個參數在MSDN上面都可以查到,只是英文,我這里作一些簡單的說明:(以WM_KEYDOWN為例)
WPARAM:虛擬鍵值,VT_*等值。
LPARAM:根據其不同的位數表示的含義不同可以分以下幾部分:
(1) 重復計數位(0 - 15 位):表示消息按鍵數據。一般情況下為1,當鍵一直按下,窗口過程就會連續收到W_KEYDOWN消息,但有可能窗口過程來不及處理這些按鍵消息,那么Windows就會把幾個按鍵消息組合成一個,並增加重復計數。比如你處理WM_KEYDOWN時Sleep(200),那么得到的這個數字就可能大於1,一般可以這樣來得到這個計數:
DWORD count = (((DWORD)lParam) & 0x0000FFFF);
(2) OEM掃描碼(16~23位):OEM掃描碼是鍵盤發送的碼值,由於此域是設備相關的,因而此值往往被忽略。
(3) 擴展鍵標志(24位):擴展鍵標志在有Alt鍵(或Ctrl鍵)按下時為1,否則為0。
(4) 保留位(25~28位):保留位是系統缺省保留的,一般不用。
(5) 關聯碼(29位):關聯碼用來記錄某鍵與Alt鍵的組合狀態,若按下Alt,當WM_SYSKEYDOWN消息送到某個激活的窗口時,其值為1,否則為0。
(6) 鍵的先前狀態(位30):鍵的先前狀態用於記錄先前某鍵的狀態,對於WM_SYSKEYUP消息,其值始終為1。
(7) 轉換狀態(31位):轉換狀態的消息是始終按着某鍵所產生的消息,若某鍵原來是按下的,則其先前狀態為0。轉換狀態指示鍵被按下還是被松開。當鍵被按下時,對應於者WM_SYSKEYDOWN消息,其值始終為0,當鍵被松開時,其轉換狀態為1,對應於WM_SYSKEYUP消息,其值始終為1。
5,死字符消息
Windows程序經常忽略WM_DEADCHAR和WM_SYSDEADCHAR消息,但您應該明確地知道死字符是什么,以及它們工作的方式。
在某些非U.S.英語鍵盤上,有些鍵用於給字母加上音調。因為它們本身不產生字符,所以稱之為「死鍵」。例如,使用德語鍵盤時,對於U.S.鍵盤上的+/=鍵,德語鍵盤的對應位置就是一個死鍵,未按下Shift鍵時它用於標識銳音,按下Shift鍵時則用於標識抑音。
當使用者按下這個死鍵時,窗口消息處理程序接收到一個wParam等於音調本身的ASCII或者Unicode代碼的WM_DEADCHAR消息。當使用者再按下可以帶有此音調的字母鍵(例如A鍵)時,窗口消息處理程序會接收到WM_CHAR消息,其中wParam等於帶有音調的字母「a」的ANSI代碼。
因此,使用者程序不需要處理WM_DEADCHAR消息,原因是WM_CHAR消息已含有程序所需要的所有信息。Windows的做法甚至還設計了內部錯誤處理。如果在死鍵之后跟有不能帶此音調符號的字母(例如「s」),那么窗口消息處理程序將在一行接收到兩條WM_CHAR消息-前一個消息的wParam等於音調符號本身的ASCII代碼(與傳遞到WM_DEADCHAR消息的wParam值相同),第二個消息的wParam等於字母s的ASCII代碼。
當然,要感受這種做法的運作方式,最好的方法就是實際操作。您必須加載使用死鍵的外語鍵盤,例如前面講過的德語鍵盤。您可以這樣設定:在「控制台」中選擇「鍵盤」,然后選擇「語系」頁面標簽。然后您需要一個應用程序,該程序可以顯示它接收的每一個鍵盤消息的詳細信息。下面的KEYVIEW1就是這樣的程序。