windowsSDK加速键实例分析

来源:互联网 发布:游戏源码吧 编辑:程序博客网 时间:2024/06/10 14:40

今天在看windows程序设计菜单里面的加速键,看了好几遍才勉强看懂,下面来解释一下书本里面的代码:


 #include <windows.h>#include "resource.h"#define ID_EDIT     1LRESULT 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里面的低字位保存了第一个被选中字符的位置,高字位是紧随选中文本后面的第一个字符的位置

如果相等,那么文本没有被选中。

这里我不是很理解,如果文本没有被选中的话,那么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

http://blog.csdn.net/yingzheng1983/article/details/3135818嬴政
原创粉丝点击