今天在看windows程序设计菜单里面的加速键,看了好几遍才勉强看懂,下面来解释一下书本里面的代码:
#include <windows.h> #include "resource.h" #define ID_EDIT 1 LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM); TCHAR szAppName[] = TEXT ("PopPad2") ; int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance, PSTR szCmdLine, int iCmdShow) { HACCEL hAccel ; //定义加速键表的句柄 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 (hInstance, szAppName) ; wndclass.hCursor = LoadCursor (NULL, IDC_ARROW) ; wndclass.hbrBackground = (HBRUSH) GetStockObject (WHITE_BRUSH) ; wndclass.lpszMenuName = szAppName ; wndclass.lpszClassName = szAppName ; if (!RegisterClass (&wndclass)) { MessageBox (NULL, TEXT ("This program requires Windows NT!"), szAppName, MB_ICONERROR) ; return 0 ; } hwnd = CreateWindow (szAppName, szAppName, WS_OVERLAPPEDWINDOW, GetSystemMetrics (SM_CXSCREEN) / 4, //创建窗口的大小设定 GetSystemMetrics (SM_CYSCREEN) / 4, GetSystemMetrics (SM_CXSCREEN) / 2, GetSystemMetrics (SM_CYSCREEN) / 2, NULL, NULL, hInstance, NULL) ; ShowWindow (hwnd, iCmdShow) ; UpdateWindow (hwnd) ; hAccel = LoadAccelerators (hInstance, szAppName) ; //加载到内存中并获得句柄 while (GetMessage (&msg, NULL, 0, 0)) { if (!TranslateAccelerator (hwnd, hAccel, &msg)) //如果msg中的消息是键盘消息,那么函数在加速建表中寻找句柄为hAccel的匹配值,调用句柄为hwnd的窗口过程 { TranslateMessage (&msg) ; DispatchMessage (&msg) ; } } return msg.wParam ; } AskConfirmation (HWND hwnd) { return MessageBox (hwnd, TEXT ("Really want to close PopPad2?"), szAppName, MB_YESNO | MB_ICONQUESTION) ; } LRESULT CALLBACK WndProc (HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam) { static HWND hwndEdit ; //子窗口控件编辑框的句柄 int iSelect, iEnable ; switch (message) { case WM_CREATE: hwndEdit = CreateWindow (TEXT ("edit"), NULL, WS_CHILD | WS_VISIBLE | WS_HSCROLL | WS_VSCROLL | WS_BORDER | ES_LEFT | ES_MULTILINE | ES_AUTOHSCROLL | ES_AUTOVSCROLL, 0, 0, 0, 0, hwnd, (HMENU) ID_EDIT, ((LPCREATESTRUCT) lParam)->hInstance, NULL) ; return 0 ; case WM_SETFOCUS: SetFocus (hwndEdit) ; //设置焦点给子窗口控件 return 0 ; case WM_SIZE: MoveWindow (hwndEdit, 0, 0, LOWORD (lParam), HIWORD (lParam), TRUE) ; return 0 ; case WM_INITMENUPOPUP: //关键部分 if (lParam == 1) { EnableMenuItem ((HMENU) wParam, IDM_EDIT_UNDO, SendMessage (hwndEdit, EM_CANUNDO, 0, 0) ? MF_ENABLED : MF_GRAYED) ; EnableMenuItem ((HMENU) wParam, IDM_EDIT_PASTE, IsClipboardFormatAvailable (CF_TEXT) ? MF_ENABLED : MF_GRAYED) ; iSelect = SendMessage (hwndEdit, EM_GETSEL, 0, 0) ; if (HIWORD (iSelect) == LOWORD (iSelect)) iEnable = MF_GRAYED ; else iEnable = MF_ENABLED ; EnableMenuItem ((HMENU) wParam, IDM_EDIT_CUT, iEnable) ; EnableMenuItem ((HMENU) wParam, IDM_EDIT_COPY, iEnable) ; EnableMenuItem ((HMENU) wParam, IDM_EDIT_CLEAR, iEnable) ; return 0 ; } break ; case WM_COMMAND: if (lParam) { if (LOWORD (lParam) == ID_EDIT && (HIWORD (wParam) == EN_ERRSPACE || HIWORD (wParam) == EN_MAXTEXT)) MessageBox (hwnd, TEXT ("Edit control out of space."), szAppName, MB_OK | MB_ICONSTOP) ; return 0 ; } else switch (LOWORD (wParam)) { case IDM_FILE_NEW: case IDM_FILE_OPEN: case IDM_FILE_SAVE: case IDM_FILE_SAVE_AS: case IDM_FILE_PRINT: MessageBeep (0) ; return 0 ; case IDM_APP_EXIT: SendMessage (hwnd, WM_CLOSE, 0, 0) ; return 0 ; case IDM_EDIT_UNDO: SendMessage (hwndEdit, WM_UNDO, 0, 0) ; return 0 ; case IDM_EDIT_CUT: SendMessage (hwndEdit, WM_CUT, 0, 0) ; return 0 ; case IDM_EDIT_COPY: SendMessage (hwndEdit, WM_COPY, 0, 0) ; return 0 ; case IDM_EDIT_PASTE: SendMessage (hwndEdit, WM_PASTE, 0, 0) ; return 0 ; case IDM_EDIT_CLEAR: SendMessage (hwndEdit, WM_CLEAR, 0, 0) ; return 0 ; case IDM_EDIT_SELECT_ALL: SendMessage (hwndEdit, EM_SETSEL, 0, -1) ; return 0 ; case IDM_HELP_HELP: MessageBox (hwnd, TEXT ("Help not yet implemented!"), szAppName, MB_OK | MB_ICONEXCLAMATION) ; return 0 ; case IDM_APP_ABOUT: MessageBox (hwnd, TEXT ("POPPAD2 (c) Charles Petzold, 1998"), szAppName, MB_OK | MB_ICONINFORMATION) ; return 0 ; } break ; case WM_CLOSE: if (IDYES == AskConfirmation (hwnd)) DestroyWindow (hwnd) ; return 0 ; case WM_QUERYENDSESSION: if (IDYES == AskConfirmation (hwnd)) return 1 ; else return 0 ; case WM_DESTROY: PostQuitMessage (0) ; return 0 ; } return DefWindowProc (hwnd, message, wParam, lParam) ; }这个程序实现了一个简单的记事本功能,在EDIT菜单下能实现撤销(UNDO),剪切(CUT),复制(COPY),粘贴(PASTE),删除(DELETE),全选(SELECT ALL)这些功能
关键部分是下面的代码:
case WM_INITMENUPOPUP: if (lParam == 1) { EnableMenuItem ((HMENU) wParam, IDM_EDIT_UNDO, SendMessage (hwndEdit, EM_CANUNDO, 0, 0) ? MF_ENABLED : MF_GRAYED) ; EnableMenuItem ((HMENU) wParam, IDM_EDIT_PASTE, IsClipboardFormatAvailable (CF_TEXT) ? MF_ENABLED : MF_GRAYED) ; iSelect = SendMessage (hwndEdit, EM_GETSEL, 0, 0) ; if (HIWORD (iSelect) == LOWORD (iSelect)) iEnable = MF_GRAYED ; else iEnable = MF_ENABLED ; EnableMenuItem ((HMENU) wParam, IDM_EDIT_CUT, iEnable) ; EnableMenuItem ((HMENU) wParam, IDM_EDIT_COPY, iEnable) ; EnableMenuItem ((HMENU) wParam, IDM_EDIT_CLEAR, iEnable) ; return 0 ; } break ;现在我们就来具体分析一下吧^_^
一开始我不太明白WM_INITMENUPOPUP消息的意思,所以我查了下字典,init是初始化的意思,popup上次说过了是弹出的意思。
然后参考了一些资料:
WM_INITMENUPOPUP消息在下拉菜单或子菜单将要被激活的时候发出.如果没有替换整个菜单,
允许这个应用程序在菜单显示之前进行修改,
hmenuPopup = (HMENU) wParam; //子菜单句柄
uPos = (UINT) LOWORD(lParam); // 子菜单项位置
fSystemMenu = (BOOL) HIWORD(lParam); // 窗体菜单(系统菜单)标记
参数
hmenuPopup
wParam值.下拉菜单或子菜单的句柄
uPos
lParam低次序字的值.指定一个打开的下拉菜单或子菜单在菜单项中基于0相关联的位置
fSystemMenu
lParam高次序字的值.指定是否下拉菜单是窗体菜单(同样大家知道的系统菜单或控制菜单),如果菜单是窗体菜单,
这个参数是TRUE,否则它是FALSE;
返回值:
如果一个应用程序处理这个消息,它将要返回0
有了资料以后,我们就能理解了,为什么要处理这个消息呢?其实这个消息的处理正是EDIT下面的选项能执行的关键所在,下面具体来看看吧
if (lParam == 1)指定当菜单项为第1项的时候触发,因为最前面的是第0项,所以EDIT就是第一项了。
EnableMenuItem ((HMENU) wParam, IDM_EDIT_UNDO, SendMessage (hwndEdit, EM_CANUNDO, 0, 0) ? MF_ENABLED : MF_GRAYED) ;接下来我们碰到了EnableMenuItem这个函数,同样不知道我查了下资料如下
允许或禁止指定的菜单条目 BOOL EnableMenuItem(HMENU hMenu,UINT uIDEnableItem, UINT uEnable);
返回值 : BOOL 判断是否成功
hMenu ,菜单句柄
uIDEnableItem ,欲允许或禁止的一个菜单条目的标识符。
uEnable ,参考ModifyMenu函数中的菜单常数标志定义表,其中列出了允许使用的所有常数。对于这个函数,只能指定下述常数:MF_BYCOMMAND,MF_BYPOSITION,MF_ENABLED,MF_DISABLED以及MF_GRAYED
MF_BYCOMMAND 指定参数给出已存在的菜单项的命令ID号。此为缺省值。 ·
MF_BYPOSITION 指定参数给出已存在菜单项的位置。第一项所在的位置是0。 ·
MF_DISABLED 使菜单项无效,以便它不能被选择,但不变灰。 ·
MF_ENABLED 使菜单项有效,以便它能够被选择,并可从变灰的状态中恢复出来。 ·
MF_GRAYED 使菜单项无效,以便它不能被选择并同时变灰。
有了资料以后我们再来理解一下上面的代码:
第1个参数自然就是菜单的句柄了,第二个参数就是确定ID了,第三个参数用了一个简化的if条件语句判断,发送EM_CANUNDO给编辑控件,如果可以执行撤销(UNDO)操作,那么SendMessage返回非0值,同时这个选项启用,否则变灰色。
EnableMenuItem ((HMENU) wParam, IDM_EDIT_PASTE, IsClipboardFormatAvailable (CF_TEXT) ? MF_ENABLED : MF_GRAYED) ;这几行代码与上面的几行类似,只不过条件语句里面换了个函数 IsClipboardFormatAvailable,这个函数我们中文翻译一下,clipboard是剪切板的意思,所以函数的意思就是“剪切板里面有可用得东西吗?”并且用CF_TEXT来检测
iSelect = SendMessage (hwndEdit, EM_GETSEL, 0, 0) ; if (HIWORD (iSelect) == LOWORD (iSelect)) iEnable = MF_GRAYED ; else iEnable = MF_ENABLED ;书里面说,发送EM_SEL消息后在iSelect里面的低字位保存了 第一个被选中字符的位置,高字位是 紧随选中文本后面的第一个字符的位置
如果相等,那么文本没有被选中。
接着如果选中了,那么iEnable 里面存放MF_ENABLED, 如果没有选中,那么存放的是MF_GRAYED。
EnableMenuItem ((HMENU) wParam, IDM_EDIT_CUT, iEnable) ; EnableMenuItem ((HMENU) wParam, IDM_EDIT_COPY, iEnable) ; EnableMenuItem ((HMENU) wParam, IDM_EDIT_CLEAR, iEnable) ; 最后iEnable用于三个菜单项目EnableMenuItem的第三个参数。
好吧,理解了关键部分的代码之后,这个程序应该就能看懂了吧,呵呵~~~~
参考文献
windows程序设计
百度百科http://baike.baidu.com/view/1294033.htm