windowsSDK加速键实例分析

今天在看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里面的低字位保存了 第一个被选中字符的位置,高字位是 紧随选中文本后面的第一个字符的位置

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

这里我不是很理解,如果文本没有被选中的话,那么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嬴政
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值