我正在编写一个程序,用于显示窗口发送8个不同键盘消息的窗口过程的所有信息。这是代码
/*--------------------------------------------------------
KEYVIEW1.C -- Displays Keyboard and Character Messages
(c) Charles Petzold, 1998
--------------------------------------------------------*/
#include <windows.h>
LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM) ;
int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance,
PSTR szCmdLine, int iCmdShow)
{
static TCHAR szAppName[] = TEXT ("KeyView1") ;
HWND hwnd ;
MSG msg ;
WNDCLASS wndclass ;
wndclass.style = CS_HREDRAW | CS_VREDRAW ;
wndclass.lpfnWndProc = WndProc ;
wndclass.cbClsExtra = 0 ;
wndclass.cbWndExtra = 0 ;
wndclass.hInstance = hInstance ;
wndclass.hIcon = LoadIcon (NULL, IDI_APPLICATION) ;
wndclass.hCursor = LoadCursor (NULL, IDC_ARROW) ;
wndclass.hbrBackground = (HBRUSH) GetStockObject (WHITE_BRUSH) ;
wndclass.lpszMenuName = NULL ;
wndclass.lpszClassName = szAppName ;
if (!RegisterClass (&wndclass))
{
MessageBox (NULL, TEXT ("This program requires Windows NT!"),
szAppName, MB_ICONERROR) ;
return 0 ;
}
hwnd = CreateWindow (szAppName, TEXT ("Keyboard Message Viewer #1"),
WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, CW_USEDEFAULT,
CW_USEDEFAULT, CW_USEDEFAULT,
NULL, NULL, hInstance, NULL) ;
ShowWindow (hwnd, iCmdShow) ;
UpdateWindow (hwnd) ;
while (GetMessage (&msg, NULL, 0, 0))
{
TranslateMessage (&msg) ;
DispatchMessage (&msg) ;
}
return msg.wParam ;
}
LRESULT CALLBACK WndProc (HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
static int cxClientMax, cyClientMax, cxClient, cyClient, cxChar, cyChar ;
static int cLinesMax, cLines ;
static PMSG pmsg ;
static RECT rectScroll ;
static TCHAR szTop[] = TEXT ("Message Key Char ")
TEXT ("Repeat Scan Ext ALT Prev Tran") ;
static TCHAR szUnd[] = TEXT ("_______ ___ ____ ")
TEXT ("______ ____ ___ ___ ____ ____") ;
static TCHAR * szFormat[2] = {
TEXT ("%-13s %3d %-15s%c%6u %4d %3s %3s %4s %4s"),
TEXT ("%-13s 0x%04X%1s%c %6u %4d %3s %3s %4s %4s") } ;
static TCHAR * szYes = TEXT ("Yes") ;
static TCHAR * szNo = TEXT ("No") ;
static TCHAR * szDown = TEXT ("Down") ;
static TCHAR * szUp = TEXT ("Up") ;
static TCHAR * szMessage [] = {
TEXT ("WM_KEYDOWN"), TEXT ("WM_KEYUP"),
TEXT ("WM_CHAR"), TEXT ("WM_DEADCHAR"),
TEXT ("WM_SYSKEYDOWN"), TEXT ("WM_SYSKEYUP"),
TEXT ("WM_SYSCHAR"), TEXT ("WM_SYSDEADCHAR") } ;
HDC hdc ;
int i, iType ;
PAINTSTRUCT ps ;
TCHAR szBuffer[128], szKeyName [32] ;
TEXTMETRIC tm ;
switch (message)
{
case WM_CREATE:
case WM_DISPLAYCHANGE:
// Get maximum size of client area
cxClientMax = GetSystemMetrics (SM_CXMAXIMIZED) ;
cyClientMax = GetSystemMetrics (SM_CYMAXIMIZED) ;
// Get character size for fixed-pitch font
hdc = GetDC (hwnd) ;
SelectObject (hdc, GetStockObject (SYSTEM_FIXED_FONT)) ;
GetTextMetrics (hdc, &tm) ;
cxChar = tm.tmAveCharWidth ;
cyChar = tm.tmHeight ;
ReleaseDC (hwnd, hdc) ;
// Allocate memory for display lines
if (pmsg)
free (pmsg) ;
cLinesMax = cyClientMax / cyChar ;
pmsg = (PMSG)malloc (cLinesMax * sizeof (MSG)) ;
cLines = 0 ;
// fall through
case WM_SIZE:
if (message == WM_SIZE)
{
cxClient = LOWORD (lParam) ;
cyClient = HIWORD (lParam) ;
}
// Calculate scrolling rectangle
rectScroll.left = 0 ;
rectScroll.right = cxClient ;
rectScroll.top = cyChar ;
rectScroll.bottom = cyChar * (cyClient / cyChar) ;
InvalidateRect (hwnd, NULL, TRUE) ;
return 0 ;
case WM_KEYDOWN:
case WM_KEYUP:
case WM_CHAR:
case WM_DEADCHAR:
case WM_SYSKEYDOWN:
case WM_SYSKEYUP:
case WM_SYSCHAR:
case WM_SYSDEADCHAR:
// Rearrange storage array
for (i = cLinesMax - 1 ; i > 0 ; i--)
{
pmsg[i] = pmsg[i - 1] ;
}
// Store new message
pmsg[0].hwnd = hwnd ;
pmsg[0].message = message ;
pmsg[0].wParam = wParam ;
pmsg[0].lParam = lParam ;
cLines = min (cLines + 1, cLinesMax) ;
// Scroll up the display
ScrollWindow (hwnd, 0, -cyChar, &rectScroll, &rectScroll) ;
break ; // ie, call DefWindowProc so Sys messages work
case WM_PAINT:
hdc = BeginPaint (hwnd, &ps) ;
SelectObject (hdc, GetStockObject (SYSTEM_FIXED_FONT)) ;
SetBkMode (hdc, TRANSPARENT) ;
TextOut (hdc, 0, 0, szTop, lstrlen (szTop)) ;
TextOut (hdc, 0, 0, szUnd, lstrlen (szUnd)) ;
for (i = 1 ; i < max (cLines, cyClient / cyChar - 1) ; i++)
{
iType = pmsg[i].message == WM_CHAR ||
pmsg[i].message == WM_SYSCHAR ||
pmsg[i].message == WM_DEADCHAR ||
pmsg[i].message == WM_SYSDEADCHAR ;
GetKeyNameText (pmsg[i].lParam, szKeyName,
sizeof (szKeyName) / sizeof (TCHAR)) ;
TextOut (hdc, 0, (cyClient / cyChar - i) * cyChar, szBuffer,
wsprintf (szBuffer, szFormat [iType],
szMessage [pmsg[i].message - WM_KEYFIRST],
pmsg[i].wParam,
(PTSTR) (iType!=0 ? TEXT (" ") : szKeyName),
(TCHAR) (iType ? pmsg[i].wParam : ' '),
LOWORD (pmsg[i].lParam),
HIWORD (pmsg[i].lParam) & 0xFF,
0x01000000 & pmsg[i].lParam ? szYes : szNo,,// for extended key flag
0x20000000 & pmsg[i].lParam ? szYes : szNo,
0x40000000 & pmsg[i].lParam ? szDown : szUp,//Previous key state
0x80000000 & pmsg[i].lParam ? szUp : szDown)) ;//transition state
}
EndPaint (hwnd, &ps) ;
return 0 ;
case WM_DESTROY:
PostQuitMessage (0) ;
return 0 ;
}
return DefWindowProc (hwnd, message, wParam, lParam) ;
}
我的问题是我无法理解代码。我尽力了。
在第
行静态TCHAR * szFormat [2] = {
TEXT ("%-13s %3d %-15s%c%6u %4d %3s %3s %4s %4s"), TEXT ("%-13s 0x%04X%1s%c %6u %4d %3s %3s %4s %4s") } ;
发生了什么?我理解第一行而不是第二行。
TextOut函数中的
szMessage [pmsg[i].message - WM_KEYFIRST],
是什么
意味着
以及使用的代码
0x01000000
以及为什么我们特别使用&amp; 0XFF和HIWORD (pmsg[i].lParam) & 0xFF,
我知道这些问题很多,但书中没有解释过这些问题。任何人都可以回答至少其中一个问题。 任何帮助将不胜感激
答案 0 :(得分:5)
在第
行static TCHAR * szFormat[2] = { TEXT ("%-13s %3d %-15s%c%6u %4d %3s %3s %4s %4s"), TEXT ("%-13s 0x%04X%1s%c %6u %4d %3s %3s %4s %4s") } ;
发生了什么事?我理解第一行而不是第二行。
这不是特定于Windows的。它只是一个C风格的字符串数组,它们本身是指向以空字符结尾的字符数组的指针。
它被声明为static
,因此它只在首次调用函数时初始化一次,并在后续调用中保持其值。
TCHAR
是char
或wchar_t
的Windows typedef,具体取决于您是否正在构建Unicode应用程序。因此,TCHAR *
相当于char *
或wchar_t *
。
名称szFormat
是systems hungarian notation,在Win32编程中很常见,并且被Petzold广泛使用。 sz
表示它是一个nul( z ero) - 终止 s tring。
[2]
部分只是使它成为一个包含两个元素的数组。一个本身包含数组的数组。
TEXT
是一个类似函数的宏,用于根据构建配置适当地声明字符串文字。在Unicode构建中,L
附加到字符串文字的开头以使其成为宽字符串。在非Unicode构建中,不会附加任何内容。
字符串本身只是格式字符串,就像您可能在标准C中使用printf
function一样。除了Petzold的代码与wsprintf
一起使用时,此C函数的Windows版本。基本上,所有%
部分都是占位符,用您指定的变量值填充。它们具有特殊的装饰,可以更广泛地控制输出字符串的格式。
case WM_SIZE: if (message == WM_SIZE) { cxClient = LOWORD (lParam) ; cyClient = HIWORD (lParam) ; }
message == WM_SIZE
是什么意思?
我不知道。实际上,这看起来像是一个错误。在WM_SIZE
案例中,您已经知道 message
是WM_SIZE
。因此再次检查是没有意义的。只是假装它不存在。
szMessage [pmsg[i].message - WM_KEYFIRST],
是什么意思?
从顶部的名称和变量声明中,您知道szMessage
是一个C风格字符串数组,就像我们在szFormat
中看到的那样。它包含他感兴趣的窗口消息的名称,例如WM_KEYDOWN
和WM_KEYUP
。
括号中的东西是数组的索引器。就好像他已经编写了szMessage[1]
来访问数组中的第二个元素,除了他正在计算要使用表达式动态访问的项的索引。
构建表达式,使其与数组中的消息名称匹配,并在运行时使用已知的消息ID。您从pmsg[i].message
获取消息ID,然后从WM_KEYFIRST
中删除它,因为这是第一个键盘消息的ID。这使得消息ID与字符串数组中消息名称的顺序同步。
以及使用的代码
0x01000000
以及为什么我们特别使用&amp; 0XFF和HIWORD (pmsg[i].lParam) & 0xFF,
那些只是位掩码。使用&
运算符(按位AND)将它们与位值组合,可以屏蔽指定的位。
至于它们的含义,这些都包含在特定窗口消息的文档中。例如,在WM_KEYDOWN
docs中,它会告诉您lParam
值中特定位的含义。通过屏蔽这些位,您可以隔离它们以获取各个数据。
Petzold正在利用他正在监控的所有与键盘相关的消息(在我们刚才谈到的szMessage
数组中列出)都使用lParam
值的相同格式。通过查看每个消息的相应文档,可以验证这一点。例如,WM_CHAR
与我们刚才看到的WM_KEYDOWN
相同。这样可以大大简化代码。
当然,其目的是在每次绘制窗口时向用户显示有关键盘事件的信息。