PMGuide - Menus: Difference between revisions
| No edit summary | |||
| (10 intermediate revisions by the same user not shown) | |||
| Line 28: | Line 28: | ||
| By using the PU_MOUSEBUTTONn flag, you can enable the user to display the pop-up menu, select an item, and dismiss the menu, all in one operation. For example, if your window procedure displays the pop-up window when the user double-clicks mouse button 2, specify the PU_MOUSEBUTTON2DOWN flag in the WinPopupMenu function. Then, the user can display the menu with mouse button 2; and, while holding the button down, select an item. When the user releases the button, the item is chosen and the menu dismissed.   | By using the PU_MOUSEBUTTONn flag, you can enable the user to display the pop-up menu, select an item, and dismiss the menu, all in one operation. For example, if your window procedure displays the pop-up window when the user double-clicks mouse button 2, specify the PU_MOUSEBUTTON2DOWN flag in the WinPopupMenu function. Then, the user can display the menu with mouse button 2; and, while holding the button down, select an item. When the user releases the button, the item is chosen and the menu dismissed.   | ||
| ===System Menu=== | ===System Menu=== | ||
| The system menu in the upper-left corner of a standard frame window is different from the menus defined by the application. The system menu is controlled and defined almost exclusively by the system; your only decision about it is whether to include it when creating a frame window. (It is unusual for a frame window not to include a system menu.) The system menu generates WM_SYSCOMMAND messages instead of WM_COMMAND messages. Most applications simply use the default behavior for WM_SYSCOMMAND messages, although applications can add, delete, and change system-menu entries.   | The system menu in the upper-left corner of a standard frame window is different from the menus defined by the application. The system menu is controlled and defined almost exclusively by the system; your only decision about it is whether to include it when creating a frame window. (It is unusual for a frame window not to include a system menu.) The system menu generates [[WM_SYSCOMMAND]] messages instead of [[WM_COMMAND]] messages. Most applications simply use the default behavior for WM_SYSCOMMAND messages, although applications can add, delete, and change system-menu entries.   | ||
| ==Menu Items== | ==Menu Items== | ||
| All menus can contain two main types of menu items: command items and submenu items. When the user chooses a command item, the menu immediately posts a message to the parent window. When the user selects a submenu item, the menu displays a submenu from which the user may choose another item. Since a submenu window also can contain a submenu item, submenus can originate from other submenus. | All menus can contain two main types of menu items: command items and submenu items. When the user chooses a command item, the menu immediately posts a message to the parent window. When the user selects a submenu item, the menu displays a submenu from which the user may choose another item. Since a submenu window also can contain a submenu item, submenus can originate from other submenus. | ||
| When the user chooses a command item from a menu, the menu system posts a WM_COMMAND, WM_SYSCOMMAND, or WM_HELP message to the owner window, depending on the style bits of the menu item. | When the user chooses a command item from a menu, the menu system posts a [[WM_COMMAND]], [[WM_SYSCOMMAND]], or [[WM_HELP]] message to the owner window, depending on the style bits of the menu item. | ||
| Applications can change the attributes, style, and contents of menu items, and insert and delete items at run time, to reflect changes in the command environment. An application also can add items to or delete items from the menu bar, a pop-up menu, or a submenu. For example, an application might maintain a menu of the fonts currently available in the system. This application would use graphics programming interface (GPI) calls to determine which fonts were available and, then, insert a menu item for each font into a submenu. Furthermore, the application might set the check-mark attribute of the menu item for the currently chosen font. When the user chose a new font, the application would remove the check-mark attribute from the previous choice and add it to the new choice. | Applications can change the attributes, style, and contents of menu items, and insert and delete items at run time, to reflect changes in the command environment. An application also can add items to or delete items from the menu bar, a pop-up menu, or a submenu. For example, an application might maintain a menu of the fonts currently available in the system. This application would use graphics programming interface (GPI) calls to determine which fonts were available and, then, insert a menu item for each font into a submenu. Furthermore, the application might set the check-mark attribute of the menu item for the currently chosen font. When the user chose a new font, the application would remove the check-mark attribute from the previous choice and add it to the new choice. | ||
| Line 100: | Line 100: | ||
| | <kbd>Shift</kbd>+<kbd>Esc</kbd> || Shows the system menu. | | <kbd>Shift</kbd>+<kbd>Esc</kbd> || Shows the system menu. | ||
| |- | |- | ||
| | <kbd>Right Arrow</kbd> || Cycles to the next top-level menu item. If the selected item is at the far-left side of the menu, the menu code sends a WM_NEXTMENU message to the frame window. The default processing by the frame window is to cycle between the application and system menus. (An application can modify this behavior by subclassing the frame window.) If the selected item is in a submenu, the next column in the submenu is selected, or the next top-level menu item is selected; this keystroke also can send or process a WM_NEXTMENU message. | | <kbd>Right Arrow</kbd> || Cycles to the next top-level menu item. If the selected item is at the far-left side of the menu, the menu code sends a [[WM_NEXTMENU]] message to the frame window. The default processing by the frame window is to cycle between the application and system menus. (An application can modify this behavior by subclassing the frame window.) If the selected item is in a submenu, the next column in the submenu is selected, or the next top-level menu item is selected; this keystroke also can send or process a WM_NEXTMENU message. | ||
| |- | |- | ||
| | <kbd>Left Arrow</kbd> || Works like the Right Arrow key, except in the opposite direction. In submenus, this keystroke backs up one column, except when the currently selected item is in the far-left column, in which case the previous submenu is selected. | | <kbd>Left Arrow</kbd> || Works like the Right Arrow key, except in the opposite direction. In submenus, this keystroke backs up one column, except when the currently selected item is in the far-left column, in which case the previous submenu is selected. | ||
| Line 124: | Line 124: | ||
| ==Using Menus== | ==Using Menus== | ||
| This section explains how to perform the following tasks: | |||
| * Define menu items in a resource file | |||
| * Include a menu bar in a standard window | |||
| * Create a pop-up menu | |||
| * Add a menu to a dialog window | |||
| * Access the system menu | |||
| * Respond to a the menu choice of a user | |||
| * Set and query menu-item attributes | |||
| * Add and delete menu items | |||
| * Create a custom menu item | |||
| ===Defining Menu Items in a Resource File=== | ===Defining Menu Items in a Resource File=== | ||
| Typically, a menu resource represents the menu bar or pop-up menu and all the related submenus. A menu-item definition is organized as shown in the following code: | |||
|     MENUITEM item text, item identifier, item style, item attributes | |||
| The menu resource-definition file specifies the text of each item in the menu, its unique identifier, its style and attributes, and whether it is a command item or a submenu item. A menu item that has no specification for style or attributes has the default style of MIS_TEXT and all attribute bits off, indicating that the item is enabled. The MIS_SEPARATOR style identifies nonselectable lines between menu items. The following figure is sample Resource Compiler source code that defines a menu resource. The code defines a menu with three submenu items in the menu bar (File, Edit, and Font) and a command item (Help). Each submenu has several command items, and the Font submenu has two other submenus within it. | |||
| <PRE> | |||
|     MENU ID_MENU_RESOURCE | |||
|     BEGIN | |||
|         SUBMENU "~File", IDM_FILE | |||
|             BEGIN | |||
|                 MENUITEM "~Open...",       IDM_FI_OPEN | |||
|                 MENUITEM "~Close\tF3",     IDM_FI_CLOSE, 0, MIA_DISABLED | |||
|                 MENUITEM "~Quit",          IDM_FI_QUIT | |||
|                 MENUITEM "",               IDM_FI_SEP1, MIS_SEPARATOR | |||
|                 MENUITEM "~About Sample",  IDM_FI_ABOUT | |||
|             END | |||
|         SUBMENU "~Edit", IDM_EDIT | |||
|             BEGIN | |||
|                 MENUITEM "~Undo",          IDM_ED_UNDO, 0, MIA_DISABLED | |||
|                 MENUITEM "",               IDM_ED_SEP1, MIS_SEPARATOR | |||
|                 MENUITEM "~Cut",           IDM_ED_CUT | |||
|                 MENUITEM "C~opy",          IDM_ED_COPY | |||
|                 MENUITEM "~Paste",         IDM_ED_PASTE | |||
|                 MENUITEM "C~lear",         IDM_ED_CLEAR | |||
|             END | |||
|         SUBMENU "Font", IDM_FONT | |||
|             BEGIN | |||
|                 SUBMENU "Style",           IDM_FONT_STYLE | |||
|                     BEGIN | |||
|                         MENUITEM "Plain",  IDM_FONT_STYLE_PLAIN | |||
|                         MENUITEM "Bold",   IDM_FONT_STYLE_BOLD | |||
|                         MENUITEM "Italic", IDM_FONT_STYLE_ITALIC | |||
|                     END | |||
|                 SUBMENU "Size",            IDM_FONT_SIZE | |||
|                     BEGIN | |||
|                         MENUITEM "10",     IDM_FONT_SIZE_10 | |||
|                         MENUITEM "12",     IDM_FONT_SIZE_12 | |||
|                         MENUITEM "14",     IDM_FONT_SIZE_14 | |||
|                     END | |||
|             END | |||
|         MENUITEM "F1=Help", 0x00, MIS_TEXT | MIS_BUTTONSEPARATOR | MIS_HELP | |||
|     END | |||
| </PRE> | |||
| To define a menu item with the MIS_BITMAP style, an application must use a tool such as Icon Editor to create a bit map, include the bit map in its resource-definition file, and define a menu in the file (as shown in the following figure). The text for the bit map menu items is an ASCII representation of the resource identifier of the bit map resource to be displayed for that item. | |||
| <PRE> | |||
|     /* Bring externally created bit maps into the resource file. */ | |||
|     BITMAP 101 button.bmp | |||
|     BITMAP 102 hirest.bmp | |||
|     BITMAP 103 hizoom.bmp | |||
|     BITMAP 104 hired.bmp | |||
|     /* Connect a menu item with a bit map.                       */ | |||
|     SUBMENU "~Bitmaps", IDM_BITMAP | |||
|         BEGIN | |||
|             MENUITEM "#101", IDM_BM_01, MIS_BITMAP | |||
|             MENUITEM "#102", IDM_BM_02, MIS_BITMAP | |||
|             MENUITEM "#103", IDM_BM_03, MIS_BITMAP | |||
|             MENUITEM "#104", IDM_BM_04, MIS_BITMAP | |||
|         END | |||
| </PRE> | |||
| ===Including a Menu Bar in a Standard Window=== | ===Including a Menu Bar in a Standard Window=== | ||
| If you have defined a menu resource in a resource-definition file, you can use the menu resource to create a menu bar in a standard window. You include the menu bar by using the FCF_MENU attribute flag and specifying the menu-resource identifier in a call to WinCreateStdWindow, as shown in the following code fragment: | |||
| <PRE> | |||
|     #define ID_MENU_RESOURCE 100 | |||
|     HWND hwndFrame; | |||
|     CHAR szClassName[]="MyClass"; | |||
|     CHAR szTitle[]="My Title"; | |||
|     ULONG flControlStyle = FCF_MENU     | FCF_SIZEBORDER | | |||
|                            FCF_TITLEBAR | FCF_ACCELTABLE; | |||
|     hwndFrame = WinCreateStdWindow(HWND_DESKTOP, | |||
|         WS_VISIBLE, | |||
|         &flControlStyle, | |||
|         szClassName, | |||
|         szTitle, | |||
|         0, (HMODULE) NULL, | |||
|         ID_MENU_RESOURCE, | |||
|         NULL); | |||
| </PRE> | |||
| After you make this call, the operating system automatically includes the menu in the window, drawing the menu bar across the top of the window. When the user chooses an item from the menu, the menu posts the message to the frame window. The frame window passes any WM_COMMAND messages to the client window. (The frame window does not pass WM_SYSCOMMAND messages to the client window.) WM_HELP messages are posted to the focus window. The WinDefWindowProc function passes WM_HELP messages to the parent window. If a WM_HELP message is passed to a frame window, the frame window calls the HK_HELP hook. Your client window procedure must process these messages to respond to the user's actions.  | |||
| ===Creating a Pop-up Menu=== | ===Creating a Pop-up Menu=== | ||
| The following code fragment shows how to make a pop-up menu appear when the user double-clicks mouse button 2 anywhere in the parent window. The menu is positioned with the mouse pointer located on the item having the IDM_OPEN identifier and is constrained horizontally and vertically. Then, the user can select an item from the pop-up menu using mouse button 2. | |||
| <PRE> | |||
|     #define ID_MENU_RESOURCE  110 | |||
|     #define IDM_OPEN          120 | |||
|     HWND hwndFrame; | |||
|     MRESULT ClientWndProc( | |||
|     HWND hwnd, | |||
|     ULONG msg, | |||
|     MPARAM mp1, | |||
|     MPARAM mp2) | |||
|     { | |||
|         HWND hwndMenu; | |||
|         BOOL fSuccess; | |||
|         switch (msg) { | |||
|             . | |||
|             .   /* Process other messages. */ | |||
|             . | |||
|             case WM_BUTTON2DBLCLK: | |||
|                 hwndMenu = WinLoadMenu(hwnd, (HMODULE) NULL, ID_MENU_RESOURCE); | |||
|                 fSuccess = WinPopupMenu(hwnd, | |||
|                                         hwndFrame, | |||
|                                         hwndMenu, | |||
|                                         20, | |||
|                                         50, | |||
|                                         IDM_OPEN, | |||
|                                         PU_POSITIONONITEM   | | |||
|                                         PU_HCONSTRAIN       | | |||
|                                         PU_VCONSTRAIN       | | |||
|                                         PU_MOUSEBUTTON2DOWN | | |||
|                                         PU_MOUSEBUTTON2); | |||
|                        . | |||
|                        . | |||
|                        . | |||
| </PRE> | |||
| ===Adding a Menu to a Dialog Window=== | ===Adding a Menu to a Dialog Window=== | ||
| You might want to use menus in windows that were not created using the WinCreateStdWindow function. For these windows, you can load a menu resource by using the WinLoadMenu function and specifying the parent window for the menu. WinLoadMenu assigns the specified menu resource to the parent. To see the menu in the window, you must send a WM_UPDATEFRAME message to the parent after loading the menu resource. This strategy is especially useful for adding menus to a window created as a dialog window, but it can be used no matter what type of window is specified as the parent. | |||
| ===Accessing the System Menu=== | ===Accessing the System Menu=== | ||
| Although most applications do not alter the system menu, you can obtain the handle of the system menu by calling WinWindowFromID with a frame-window handle (or dialog-window handle) and the identifier FID_SYSMENU. Once you have the handle of the system menu, you can access the individual menu items by using predefined constants. For example, the following code fragment shows how to disable the Close menu item in the system menu of a window: | |||
| <PRE> | |||
|     HWND hwndSysMenu; | |||
|     HWND hwndFrame; | |||
|     hwndSysMenu = WinWindowFromID(hwndFrame, FID_SYSMENU); | |||
|     WinSendMsg(hwndSysMenu, MM_SETITEMATTR, | |||
|         MPFROM2SHORT(SC_CLOSE, TRUE), | |||
|         MPFROM2SHORT(MIA_DISABLED, MIA_DISABLED)); | |||
| </PRE> | |||
| ===Responding to a User's Menu Choice=== | ===Responding to a User's Menu Choice=== | ||
|  When a user chooses a menu item, the client window procedure receives a WM_COMMAND message with SHORT1FROMMP(mp1) equal to the menu identifier of the chosen item. Your application must use the menu identifier to guide its response to the choice. Typically, the code in the client window procedure resembles the following code fragment: | |||
|        case WM_COMMAND: | |||
|            DoMenuCommand(hwnd, SHORT1FROMMP(mp1)); | |||
|            return 0; | |||
| The function that translates the menu identifier into an action typically resembles the following code fragment: | |||
| <PRE> | |||
|     VOID DoMenuCommand( | |||
|     HWND hwnd, | |||
|     USHORT usItemID) | |||
|     { | |||
|         /* Test the menu item. */ | |||
|         switch (usItemID) { | |||
|             case IDM_FI_NEW: | |||
|                 DoNew(hwnd); | |||
|                 break; | |||
|           . | |||
|           . /* etc. */ | |||
|           . | |||
|         } | |||
|     } | |||
| </PRE> | |||
| The menu window sends a WM_MENUSELECT message every time the menu selection changes. SHORT1FROMMP(mp1) contains the identifier of the item that is changing state, and SHORT2FROMMP(mp2) is a 16-bit Boolean value that describes whether or not the item is chosen; the mp2 parameter contains the handle of the menu. | |||
| If the Boolean value is FALSE, the item is selected but not chosen; for example, the user may have moved the cursor or mouse pointer over the item while the button was down. An application can use this message to display Help information at the bottom of the application window. The return value is ignored. | |||
| If the Boolean value is TRUE, the item is chosen-that is, the user pressed Enter or released the mouse button while an item was selected. If the application returns FALSE, the menu does not generate a WM_COMMAND, WM_SYSCOMMAND, or WM_HELP message, and the menu is not dismissed. | |||
| ===Setting and Querying Menu-Item Attributes=== | ===Setting and Querying Menu-Item Attributes=== | ||
| Menu-item attributes are represented in the fAttribute field of the MENUITEM data structure. Typically, attributes are set in the resource-definition file of the menu and are changed at run time as required. Applications can use the MM_SETITEMATTR and MM_QUERYITEMATTR messages to set and query attributes for a particular menu item. One of the most common uses of these messages is to check and uncheck menu items to let the user know what option is selected currently. For example, if you have a menu item that should toggle between checked and unchecked each time the user selects it, you can use the following figure to change the checked attribute. In this example, you send an MM_QUERYITEMATTR message to the menu item to obtain its current checked attribute; then, you use the exclusive OR operator to toggle the state; and finally, you send the new attribute state back to the item using an MM_SETITEMATTR message. | |||
| <PRE> | |||
|     usAttrib = SHORT1FROMMR( | |||
|         WinSendMsg(hwndMenu,           /* Submenu window        */ | |||
|         MM_QUERYITEMATTR,              /* Message               */ | |||
|         (MPARAM)itemID,                /* Item identifier       */ | |||
|         (MPARAM)MIA_CHECKED            /* Attribute mask        */ | |||
|         )); | |||
|     usAttrib = MIA_CHECKED;        /* XOR to toggle checked attribute */ | |||
|     WinSendMsg(hwndMenu,                      /* Submenu window        */ | |||
|         MM_SETITEMATTR,                       /* Message               */ | |||
|         (MPARAM)itemID,                       /* Item identifier       */ | |||
|         MPFROM2SHORT(MIA_CHECKED, usAttrib)); /* Attribute mask, value */ | |||
| </PRE> | |||
| ===Adding and Deleting Menu Items=== | ===Adding and Deleting Menu Items=== | ||
| An application can add and delete items from its menus dynamically by sending MM_INSERTITEM and MM_DELETEITEM messages to the menu window. Any item, including those in submenus, can be deleted by sending a message to the menu window. Messages to insert items in submenus must be sent to the submenu's window (rather than to the window of the top-level menu). You can retrieve the handle of a submenu of the menu bar by sending an MM_QUERYITEM message to the menu-bar and specifying the identifier of the submenu item for the submenu, as shown in the following code fragment: | |||
| <PRE> | |||
|     /* IDM_MYMENUID is the identifier of the submenu containing the item. */ | |||
|     MENUITEM mi; | |||
|     HWND hwndMenu, hwndSubMenu, hwndPullDown,hwndFrame; | |||
|     hwndMenu = WinWindowFromID(hwndFrame, FID_MENU); | |||
|     WinSendMsg(hwndMenu,                           /* Handle of menu bar  */ | |||
|         MM_QUERYITEM,                              /* Message             */ | |||
|         MPFROM2SHORT(IDM_MYMENUID, TRUE),          /* Submenu identifier  */ | |||
|         (MPARAM) &mi);                             /* Pointer to MENUITEM */ | |||
|     hwndPullDown = mi.hwndSubMenu;                 /* Handle to submenu   */ | |||
| </PRE> | |||
| Once the application has the handle of the submenu, it can insert an item by filling in a MENUITEM structure and sending an MM_INSERTITEM message to the submenu. For text-menu items, the application must send a pointer to the text string as well as to the MENUITEM structure, as shown in the following figure. | |||
| <PRE> | |||
|     PSZ  pszNewItemString; | |||
|     mi.iPosition = MIT_END; | |||
|     mi.afStyle = MIS_TEXT; | |||
|     mi.afAttribute = 0; | |||
|     mi.id = IDM_MYMENU_FIRST; | |||
|     mi.hwndSubMenu = NULL; | |||
|     mi.hItem = 0; | |||
|     WinSendMsg(hwndPullDown, MM_INSERTITEM, (MPARAM) &mi, | |||
|         (MPARAM) pszNewItemString); | |||
| </PRE> | |||
| To delete an item, the application sends an MM_DELETEITEM message to the menu bar, specifying the identifier of the item to delete. For example, to clear all the items following IDM_MYMENU_FIRST in a submenu in which the items are numbered sequentially, use the following code: | |||
| <PRE> | |||
|     USHORT usItemNum; | |||
|     /* Clear all the items in MYMENU.                 */ | |||
|     hwndMenu = WinWindowFromID(hwndFrame, FID_MENU); | |||
|     usItemNum = IDM_MYMENU_FIRST; | |||
|     while (WinSendMsg(hwndMenu, MM_DELETEITEM, | |||
|         MPFROM2SHORT(usItemNum++, TRUE), NULL) != 0); | |||
| </PRE> | |||
| Adding a complete submenu to the menu bar is a more complicated procedure than that shown in the previous examples. There are two strategies. The recommended technique is to define all possible submenus in your resource-definition file; and then, as your application runs, selectively remove and insert the submenus as needed. | |||
| For example, assume that your application has a submenu that you want to be displayed only when a particular application tool is in use. You must first define the submenu as part of the main menu resource in your resource-definition file, so that the system reads in the resource menu template and creates the submenu window along with the rest of the menu. You then can remove the submenu from the menu bar, saving the title of the submenu and the MENUITEM structure that defines the submenu, as shown in the following figure: | |||
| <PRE> | |||
|     HWND hwndMenu, hwndClient; | |||
|     MENUITEM mi; | |||
|     CHAR szMenuTitle[MAX_STRINGSIZE]; | |||
|     /* Remove a submenu so that you can replace it later.           */ | |||
|     /* Obtain the handle of a menu.                                 */ | |||
|     hwndMenu = WinWindowFromID(WinQueryWindow(hwndClient, QW_PARENT), | |||
|                                FID_MENU); | |||
|     /* Obtain information on the item to remove.                    */ | |||
|     WinSendMsg(hwndMenu, MM_QUERYITEM, | |||
|         MPFROM2SHORT(IDM_MENUID, TRUE),  /* TRUE to search submenus */ | |||
|         (MPARAM)&mi); | |||
|     /* Save the text for the submenu item.                          */ | |||
|     WinSendMsg(hwndMenu, MM_QUERYITEMTEXT, | |||
|         MPFROM2SHORT(IDM_FONT, MAX_STRINGSIZE), | |||
|         (MPARAM)szMenuTitle); | |||
|     /* Remove the item, but retain mi and szMenuTitle.              */ | |||
|     WinSendMsg(hwndMenu, MM_REMOVEITEM, | |||
|         MPFROM2SHORT(IDM_FONT, TRUE), NULL); | |||
| </PRE> | |||
| It is important to use the MM_REMOVEITEM message, rather than MM_DELETEITEM, to remove the item; deleting the item destroys the submenu window-removing it does not. The submenu should remain intact so that you can insert it later. | |||
| To reinsert the submenu, send an MM_INSERTITEM message to the menu bar, passing the MENUITEM structure and menu title that you saved when you removed the item. The following code fragment shows how to insert a submenu that was removed by using the previous code example. | |||
| <PRE> | |||
|     /* Put the submenu back in and obtain the handle of the menu bar. */ | |||
|     hwndMenu = WinWindowFromID( | |||
|                     WinQueryWindow(hwndClient, QW_PARENT), FID_MENU); | |||
|     /* Use the information that you saved when you removed the menu.  */ | |||
|     WinSendMsg(hwndMenu, MM_INSERTITEM, (MPARAM)&mi, | |||
|                (MPARAM)szMenuTitle); | |||
| </PRE> | |||
| The other technique that you can use to insert a submenu in the menu bar is to build up, in memory, a data structure as a menu template and use that template and WinCreateWindow to create a submenu. The resultant submenu window handle then is placed in the hwndSubMenu field of a MENUITEM structure, and the menu item is sent to the menu bar with an MM_INSERTITEM message. | |||
| You also can create an empty submenu window by using WinCreateWindow. Pass NULL for the pCtlData and pPresParams parameters, instead of building the menu template in memory. Then insert a new menu item in the menu bar by using the MM_INSERTITEM message, setting the MIS_SUBMENU style, and putting the window handle of the created menu into the hwndSubMenu field. Then use the MM_INSERTITEM message to insert the items in the new pull-down menu. | |||
| ===Creating a Custom Menu Item === | ===Creating a Custom Menu Item === | ||
| Applications can customize the appearance of an individual menu item by setting the MIS_OWNERDRAW style bit for the item. The operating system sends two different messages to an application that include owner-drawn menu items: [[WM_MEASUREITEM]] and [[WM_DRAWITEM]]. Both messages include a pointer to an [[OWNERITEM]] data structure. | |||
| [[WM_MEASUREITEM]] is sent only once for each owner-drawn item when the menu is initialized. The message is sent to the owner of the menu (typically, a frame window), which forwards the message to its client window. Typically, the client window procedure processes WM_MEASUREITEM by filling in the yTop and Right fields of the RECTL structure, specified by the rclItem field of this OWNERITEM structure; this specifies the size of the rectangle needed to enclose the item when it is drawn. The following code fragment responds to a WM_MEASUREITEM message. | |||
| <PRE> | |||
|     case WM_MEASUREITEM: | |||
|         ((POWNERITEM) mp2)->rclItem.xRight = 26; | |||
|         ((POWNERITEM) mp2)->rclItem.yTop = 10; | |||
|         return 0; | |||
| </PRE> | |||
| If a menu item has the MIS_OWNERDRAW style, the owner window receives a WM_DRAWITEM message every time the menu item needs to be drawn. You process this message by using the hps and rclItem fields of the OWNERITEM structure to draw the item. There are two situations in which the owner window receives a WM_DRAWITEM message: | |||
| [[ | * When the item must be redrawn completely | ||
| * When the item must be highlighted or have its highlight removed  | |||
| You can choose to handle one or both of these situations. Typically, you handle the drawing of the item. You may not want to handle the second situation, however, since the system-default behavior (inverting the bits in the item rectangle) often is acceptable. The two situations in which a WM_DRAWITEM message is received are detected by comparing the values of the fsState and fsStateOld fields of the OWNERITEM structure that is sent as part of the message. If the two fields are the same, draw the item. Before drawing the item, however, check its attributes to see whether it has the attributes MIA_CHECKED, MIA_FRAMED, or MIA_DISABLED. Then draw the item according to the attributes. | |||
| For example, when the checked attribute of an owner-drawn menu item changes, the system sends a WM_DRAWITEM message to the item so that it can redraw itself and either draw or remove the check mark. If you want the system-default check mark, simply draw the item and leave the fsAttribute and fsAttributeOld fields unchanged; the system draws the check mark if necessary. If you draw the check mark yourself, clear the MIA_CHECKED bit in both fsAttribute and fsAttributeOld so that the system does not attempt to draw a check mark. | |||
| In the same example, if fsAttribute and fsAttributeOld are not equal, the highlight showing that an item is selected needs to change. The MIA_HILITED bit of the fsAttribute field is set if the item needs to be highlighted and is not set if the highlight needs to be removed. If you do not want to provide your own highlighting, you should ignore any WM_DRAWITEM message in which fsAttribute and fsAttributeOld are not equal. If you do not alter these two fields, the system performs its default highlighting operation. If you want to provide your own visual cue that an item is selected, respond to a WM_DRAWITEM message in which the fsAttribute and fsAttributeOld fields are not equal by providing the cue and clearing the MIA_HILITED bit of both fields before returning from the message. | |||
| Likewise, the MIA_CHECKED and MIA_FRAMED bits of fsAttribute and fsAttributeOld either can be used to perform the corresponding action or passed on, unchanged, so that the system performs the action. The following code fragment shows how to respond to a WM_DRAWITEM message when you want to draw the item and also be responsible for its highlighted state. | |||
| <PRE> | |||
|     case WM_DRAWITEM: | |||
|         { | |||
|         POWNERITEM poi; | |||
|         RECTL      rcl; | |||
|         MPARAM     mp2; | |||
|         poi = (POWNERITEM) mp2; | |||
|         /* | |||
|          * If the new attribute equals the old attribute, | |||
|          * redraw the entire item. | |||
|          */ | |||
|         if (poi->fsAttribute == poi->fsAttributeOld) { | |||
|             /* | |||
|              * Draw the item in poi->hps and poi->rclItem, and check the | |||
|              * attributes for check marks. If you produce your own check marks, | |||
|              * use this line of code: | |||
|              * | |||
|              *     poi->fsAttributeOld = (poi->fsAttribute &= ~MIA_CHECKED; | |||
|              */ | |||
|         } | |||
|         /* Else highlight the item or remove its highlight. */ | |||
|         else if ((poi->fsAttribute & MIA_HILITED) != | |||
|                 (poi->fsAttributeOld & MIA_HILITED)) { | |||
|             /* | |||
|              * Set bits the same so that the menu window does not highlight | |||
|              * the item or remove its highlight. | |||
|              */ | |||
|             poi->fsAttributeOld = (poi->fsAttribute &= ~MIA_HILITED); | |||
|         } | |||
|         return TRUE; /* TRUE means the item is drawn. */ | |||
|         } /* endcase */ | |||
| </PRE> | |||
| Responding to [[WM_DRAWITEM]] Message. | |||
Latest revision as of 03:31, 7 May 2025
Reprint Courtesy of International Business Machines Corporation, © International Business Machines Corporation
About Menus
A menu always is owned by another window, usually a frame window. When a user makes a choice from a menu, the menu posts a message containing the unique identifier for the menu item to its owner by way of the owner window's window procedure.
An application typically defines its menus using Resource Compiler, and then associates the menus with a frame window when the frame window is created. Applications also can create menus by filling in menu-template data structures and creating windows with the WC_MENU class. Either way, applications can add, delete, or change menu items dynamically by issuing messages to menu windows.
Menu Bar and Pull-Down Menus
A typical application uses a menu bar and several pull-down submenus. The pull-down submenus ordinarily are hidden, but become visible when the user makes selections in the menu bar. Pull-down submenus always are attached to the menu bar.
The menu bar is a child of the frame window; the menu bar window handle is the key to communicating with the menu bar and its submenus. You can retrieve this handle by calling WinWindowFromID, with the handle of the parent window and the FID_MENU frame-control identifier. Most messages for the menu bar and its submenus can be issued to the menu-bar window. Flags in the messages tell the window whether to search submenus for requested menu items.
Pop-Up Menus
A pop-up menu is like a pull-down submenu, except that it is not attached to the menu bar; it can appear anywhere in its parent window. A pop-up menu usually is associated with a portion of a window, such as the client window, or it is associated with a specific object, such as an icon.
A pop-up menu remains hidden until the user selects it (either by moving the cursor to the appropriate location and pressing Enter or clicking on the location with the mouse). Typically, pop-up menus are displayed at the position of the cursor or mouse pointer; they provide a quick mechanism for selecting often-used menu items.
To include a pop-up menu in an application, you first must define a menu resource in a resource-definition file, then load the resource using the WinLoadMenu or WinCreateMenu functions. You must call WinPopupMenu to create the pop-up menu and display it in the parent window. Applications typically call WinPopupMenu in a window procedure in response to a user-generated message, such as WM_BUTTON2DBLCLK or WM_CHAR.
WinPopupMenu requires that you specify the pop-up menu's handle and also the handles of the parent and owner windows of the pop-up menu. WinLoadMenu and WinCreateMenu return the handle of the pop-up menu window, but you must obtain the handles of the parent and owner by using WinQueryWindow.
You determine the position of the pop-up menu in relation to its parent by specifying coordinates and style flags in WinPopupMenu. The x and y coordinates determine the position of the lower-left corner of the menu relative to the lower-left corner of the parent. The system may adjust this position, however, if you include the PU_HCONSTRAIN or PU_VCONSTRAIN style flags in the call to WinPopupMenu. If necessary, PU_HCONSTRAIN adjusts the horizontal position of the menu so that its left and right edges are within the borders of the desktop window. PU_VCONSTRAIN makes the same adjustments vertically. Without these flags, a desktop-level pop-up menu can lie partially off the screen, with some items not visible nor selectable.
The PU_POSITIONONITEM flag also can affect the position of the pop-up menu. This flag positions the pop-up menu so that, when the pop-up menu appears, the specified item lies directly under the mouse pointer. Also, PU_POSITIONONITEM automatically selects the item. PU_POSITIONONITEM is useful for placing the current menu selection under the pointer so that, if the user releases the mouse button without selecting a new item, the current selection remains unchanged.
The PU_SELECTITEM flag is similar to PU_POSITIONONITEM except that it just selects the specified item; it does not affect the position of the menu.
You can enable the user to choose an item from a pop-up menu by using the same mouse button that was used to display the menu. To do this, specify the PU_MOUSEBUTTONn flag, where n corresponds to the mouse button used to display the menu. This flag specifies the mouse buttons for the user to interact with a pop-up menu once it is displayed.
By using the PU_MOUSEBUTTONn flag, you can enable the user to display the pop-up menu, select an item, and dismiss the menu, all in one operation. For example, if your window procedure displays the pop-up window when the user double-clicks mouse button 2, specify the PU_MOUSEBUTTON2DOWN flag in the WinPopupMenu function. Then, the user can display the menu with mouse button 2; and, while holding the button down, select an item. When the user releases the button, the item is chosen and the menu dismissed.
System Menu
The system menu in the upper-left corner of a standard frame window is different from the menus defined by the application. The system menu is controlled and defined almost exclusively by the system; your only decision about it is whether to include it when creating a frame window. (It is unusual for a frame window not to include a system menu.) The system menu generates WM_SYSCOMMAND messages instead of WM_COMMAND messages. Most applications simply use the default behavior for WM_SYSCOMMAND messages, although applications can add, delete, and change system-menu entries.
Menu Items
All menus can contain two main types of menu items: command items and submenu items. When the user chooses a command item, the menu immediately posts a message to the parent window. When the user selects a submenu item, the menu displays a submenu from which the user may choose another item. Since a submenu window also can contain a submenu item, submenus can originate from other submenus.
When the user chooses a command item from a menu, the menu system posts a WM_COMMAND, WM_SYSCOMMAND, or WM_HELP message to the owner window, depending on the style bits of the menu item.
Applications can change the attributes, style, and contents of menu items, and insert and delete items at run time, to reflect changes in the command environment. An application also can add items to or delete items from the menu bar, a pop-up menu, or a submenu. For example, an application might maintain a menu of the fonts currently available in the system. This application would use graphics programming interface (GPI) calls to determine which fonts were available and, then, insert a menu item for each font into a submenu. Furthermore, the application might set the check-mark attribute of the menu item for the currently chosen font. When the user chose a new font, the application would remove the check-mark attribute from the previous choice and add it to the new choice.
The Help Item
To present a standard interface to the novice user, all applications must have a Help item in their menu bars. The Help item is defined with a particular style, attributes, and position in the menu. When the user chooses the Help item, the menu posts a WM_HELP message to the owner window, enabling the application to respond appropriately.
The item should read Help, have an identifier of 0, and have the MIS_BUTTONSEPARATOR or MIS_HELP item styles. The Help menu item should be the last item in the menu template, so that it is displayed as the rightmost item in the menu bar.
If an application uses the system default accelerator table, the user can select the Help item using either a mouse or the F1 key.
Menu-Item Styles
All menu items have a combination of style bits that determine what kind of data the item contains and what kind of message it generates when the user selects it. For example, a menu item can have the MIS_TEXT, MIS_BITMAP, or other styles that specify the visual representation of the menu item on the screen. Other styles determine what kinds of messages the item sends to its owner and whether the owner draws the item. Menu-item styles typically do not change during program execution, but you can query and set them dynamically by sending MM_QUERYITEM and MM_SETITEM messages with the menu-item identifier to the menu-bar window. For text menu items (MIS_TEXT), an MM_SETITEMTEXT message sets the text. The MM_QUERYITEMTEXT message queries the text of the item. For non-text menu items, the hItem field of the MENUITEM structure typically contains the handle of a display object, such as a bit-map handle for MIS_BITMAP menu items.
An application can draw a menu item by setting the style MIS_OWNERDRAW for the menu item. This usually is done by specifying the MIS_OWNERDRAW style for the menu item in the resource-definition file; but it also can be done at run time. When the application draws a menu item, it must respond to messages from the menu each time the item must be drawn.
Menu-Item Attributes
Menu items have attributes that determine how the items are displayed and whether or not the user can choose them. An application can set and query menu-item attributes by sending MM_SETITEMATTR and MM_QUERYITEMATTR messages, with the menu-item identifier, to the menu-bar window. If the specified item is in a submenu, there are two methods of determining its attributes. The first is to send MM_SETITEMATTR and MM_QUERYITEMATTR messages to the top-level menu, specifying the identifier of the item and setting a flag so that the message searches all submenus for the item. Then, you can retrieve the handle of the menu-bar by calling WinWindowFromID, with the handle of the frame window and the FID_MENU frame-control identifier.
The second method, which is more efficient if you want to either work with more than one submenu item or set the same item several times, involves two steps:
- Send an MM_QUERYITEM message to the menu, with the identifier of the submenu. The updated MENUITEM structure contains the window handle of the submenu.
- Send an MM_QUERYITEMATTR (or MM_SETITEMATTR) message to the submenu window, specifying the identifier of the item in the submenu.
Menu-Item Structure
A single menu item is defined by the MENUITEM data structure. This structure is used with the MM_INSERTITEM message to insert items in a menu or to query and set item characteristics with the MM_QUERYITEM and MM_SETITEM messages. The MENUITEM structure has the following form:
typedef struct _MENUITEM { /* mi */
    SHORT   iPosition;
    USHORT  afStyle;
    USHORT  afAttribute;
    USHORT  id;
    HWND    hwndSubMenu;
    ULONG   hItem;
} MENUITEM;
You can derive the values of most of the fields in this structure directly from the resource-definition file. However, the last field in the structure, hItem, depends on the style of the menu item.
The iPosition field specifies the ordinal position of the item within its menu window. If the item is part of the menu bar, iPosition specifies its relative left-to-right position, with 0 being the leftmost item. If the item is part of a submenu, iPosition specifies its relative top-to-bottom and left-to-right positions, with 0 being the upper-left item. An item with the MIS_BREAKSEPARATOR style in a pull-down menu causes a new column to begin.
The afStyle field contains the style bits of the item. The afAttribute field contains the attribute bits.
The id field contains the menu-item identifier. The identifier should be unique but does not have to be. Just remember that, when multiple items have the same identifier, they post the same command number in the WM_COMMAND, WM_SYSCOMMAND, and WM_HELP messages. Also, any message that specifies a menu item with a non-unique identifier will find the first item that has that identifier.
The hwndSubMenu field contains the window handle of a submenu window (if the item is a submenu item). The hwndSubMenu field is NULL for command items.
The hItem field contains a handle to the display object for the item, unless the item has the MIS_TEXT style, in which case, hItem is 0. For example, a menu item with the MIS_BITMAP style has an hItem field that is equal to its bit-map handle.
Menu Access
The OS/2 operating system is designed to work with or without a mouse or other pointing device. The system provides default behavior that enables a user to interact with menus without a mouse. Following are the keystrokes that produce this default behavior:
| Keystroke | Action | 
|---|---|
| Alt | Toggles in and out of menu-bar mode. | 
| Alt+Spacebar | Shows the system menu. | 
| F10 | Backs up one level. If a submenu is displayed, it is canceled. If no submenu is displayed, this keystroke exits the menu. | 
| Shift+Esc | Shows the system menu. | 
| Right Arrow | Cycles to the next top-level menu item. If the selected item is at the far-left side of the menu, the menu code sends a WM_NEXTMENU message to the frame window. The default processing by the frame window is to cycle between the application and system menus. (An application can modify this behavior by subclassing the frame window.) If the selected item is in a submenu, the next column in the submenu is selected, or the next top-level menu item is selected; this keystroke also can send or process a WM_NEXTMENU message. | 
| Left Arrow | Works like the Right Arrow key, except in the opposite direction. In submenus, this keystroke backs up one column, except when the currently selected item is in the far-left column, in which case the previous submenu is selected. | 
| Up Arrow or Down Arrow | When pressed in a top-level menu, activates a submenu. When pressed in a submenu, this keystroke selects the previous or next item, respectively. | 
| Enter | Activates a submenu, and highlights the first item if an item has a submenu associated with it; otherwise, this keystroke chooses the item as though the user released the mouse button while the item was selected. | 
| Alphabetic character | Selects the first menu item with the specified character as its mnemonic key. A mnemonic is defined for a menu item by placing a tilde (~) before the character in the menu text. If the selected item has a submenu associated with it, the menu is displayed, and the first item is highlighted; otherwise, the item is chosen. | 
An application does not support the default keyboard behavior with any unusual code; instead, the application receives a message when a menu item is chosen by the keyboard just as though it had been chosen by a mouse.
Mnemonics
Adding mnemonics to menu items is one way of providing the user with keyboard access to menus. You can indicate a mnemonic keystroke for a menu item by preceding a character in the item text with a tilde, as in ~File. Then, the user can choose that item by pressing the mnemonic key when the menu is active.
The menu bar is active when the user presses and releases the Alt key, and the first item in the menu bar is highlighted. A pop-up or pull-down menu is active when it is open.
Accelerators
In addition to mnemonics, a menu item can have an associated keyboard accelerator. Accelerators are different from mnemonics in that the menu need not be active for the accelerator key to work. If you have associated a menu item with a keyboard accelerator, display the accelerator to the right of the menu item. Do this in the resource-definition file by placing a tab character (\t) in the menu text before the characters that will be displayed on the right. For example, if the Close item had the F3 function key as its keyboard accelerator, the text for the item would be Close\tF3.
Using Menus
This section explains how to perform the following tasks:
- Define menu items in a resource file
- Include a menu bar in a standard window
- Create a pop-up menu
- Add a menu to a dialog window
- Access the system menu
- Respond to a the menu choice of a user
- Set and query menu-item attributes
- Add and delete menu items
- Create a custom menu item
Defining Menu Items in a Resource File
Typically, a menu resource represents the menu bar or pop-up menu and all the related submenus. A menu-item definition is organized as shown in the following code:
MENUITEM item text, item identifier, item style, item attributes
The menu resource-definition file specifies the text of each item in the menu, its unique identifier, its style and attributes, and whether it is a command item or a submenu item. A menu item that has no specification for style or attributes has the default style of MIS_TEXT and all attribute bits off, indicating that the item is enabled. The MIS_SEPARATOR style identifies nonselectable lines between menu items. The following figure is sample Resource Compiler source code that defines a menu resource. The code defines a menu with three submenu items in the menu bar (File, Edit, and Font) and a command item (Help). Each submenu has several command items, and the Font submenu has two other submenus within it.
    MENU ID_MENU_RESOURCE
    BEGIN
        SUBMENU "~File", IDM_FILE
            BEGIN
                MENUITEM "~Open...",       IDM_FI_OPEN
                MENUITEM "~Close\tF3",     IDM_FI_CLOSE, 0, MIA_DISABLED
                MENUITEM "~Quit",          IDM_FI_QUIT
                MENUITEM "",               IDM_FI_SEP1, MIS_SEPARATOR
                MENUITEM "~About Sample",  IDM_FI_ABOUT
            END
        SUBMENU "~Edit", IDM_EDIT
            BEGIN
                MENUITEM "~Undo",          IDM_ED_UNDO, 0, MIA_DISABLED
                MENUITEM "",               IDM_ED_SEP1, MIS_SEPARATOR
                MENUITEM "~Cut",           IDM_ED_CUT
                MENUITEM "C~opy",          IDM_ED_COPY
                MENUITEM "~Paste",         IDM_ED_PASTE
                MENUITEM "C~lear",         IDM_ED_CLEAR
            END
        SUBMENU "Font", IDM_FONT
            BEGIN
                SUBMENU "Style",           IDM_FONT_STYLE
                    BEGIN
                        MENUITEM "Plain",  IDM_FONT_STYLE_PLAIN
                        MENUITEM "Bold",   IDM_FONT_STYLE_BOLD
                        MENUITEM "Italic", IDM_FONT_STYLE_ITALIC
                    END
                SUBMENU "Size",            IDM_FONT_SIZE
                    BEGIN
                        MENUITEM "10",     IDM_FONT_SIZE_10
                        MENUITEM "12",     IDM_FONT_SIZE_12
                        MENUITEM "14",     IDM_FONT_SIZE_14
                    END
            END
        MENUITEM "F1=Help", 0x00, MIS_TEXT | MIS_BUTTONSEPARATOR | MIS_HELP
    END
To define a menu item with the MIS_BITMAP style, an application must use a tool such as Icon Editor to create a bit map, include the bit map in its resource-definition file, and define a menu in the file (as shown in the following figure). The text for the bit map menu items is an ASCII representation of the resource identifier of the bit map resource to be displayed for that item.
    /* Bring externally created bit maps into the resource file. */
    BITMAP 101 button.bmp
    BITMAP 102 hirest.bmp
    BITMAP 103 hizoom.bmp
    BITMAP 104 hired.bmp
    /* Connect a menu item with a bit map.                       */
    SUBMENU "~Bitmaps", IDM_BITMAP
        BEGIN
            MENUITEM "#101", IDM_BM_01, MIS_BITMAP
            MENUITEM "#102", IDM_BM_02, MIS_BITMAP
            MENUITEM "#103", IDM_BM_03, MIS_BITMAP
            MENUITEM "#104", IDM_BM_04, MIS_BITMAP
        END
Including a Menu Bar in a Standard Window
If you have defined a menu resource in a resource-definition file, you can use the menu resource to create a menu bar in a standard window. You include the menu bar by using the FCF_MENU attribute flag and specifying the menu-resource identifier in a call to WinCreateStdWindow, as shown in the following code fragment:
    #define ID_MENU_RESOURCE 100
    HWND hwndFrame;
    CHAR szClassName[]="MyClass";
    CHAR szTitle[]="My Title";
    ULONG flControlStyle = FCF_MENU     | FCF_SIZEBORDER |
                           FCF_TITLEBAR | FCF_ACCELTABLE;
    hwndFrame = WinCreateStdWindow(HWND_DESKTOP,
        WS_VISIBLE,
        &flControlStyle,
        szClassName,
        szTitle,
        0, (HMODULE) NULL,
        ID_MENU_RESOURCE,
        NULL);
After you make this call, the operating system automatically includes the menu in the window, drawing the menu bar across the top of the window. When the user chooses an item from the menu, the menu posts the message to the frame window. The frame window passes any WM_COMMAND messages to the client window. (The frame window does not pass WM_SYSCOMMAND messages to the client window.) WM_HELP messages are posted to the focus window. The WinDefWindowProc function passes WM_HELP messages to the parent window. If a WM_HELP message is passed to a frame window, the frame window calls the HK_HELP hook. Your client window procedure must process these messages to respond to the user's actions.
Creating a Pop-up Menu
The following code fragment shows how to make a pop-up menu appear when the user double-clicks mouse button 2 anywhere in the parent window. The menu is positioned with the mouse pointer located on the item having the IDM_OPEN identifier and is constrained horizontally and vertically. Then, the user can select an item from the pop-up menu using mouse button 2.
    #define ID_MENU_RESOURCE  110
    #define IDM_OPEN          120
    HWND hwndFrame;
    MRESULT ClientWndProc(
    HWND hwnd,
    ULONG msg,
    MPARAM mp1,
    MPARAM mp2)
    {
        HWND hwndMenu;
        BOOL fSuccess;
        switch (msg) {
            .
            .   /* Process other messages. */
            .
            case WM_BUTTON2DBLCLK:
                hwndMenu = WinLoadMenu(hwnd, (HMODULE) NULL, ID_MENU_RESOURCE);
                fSuccess = WinPopupMenu(hwnd,
                                        hwndFrame,
                                        hwndMenu,
                                        20,
                                        50,
                                        IDM_OPEN,
                                        PU_POSITIONONITEM   |
                                        PU_HCONSTRAIN       |
                                        PU_VCONSTRAIN       |
                                        PU_MOUSEBUTTON2DOWN |
                                        PU_MOUSEBUTTON2);
                       .
                       .
                       .
Adding a Menu to a Dialog Window
You might want to use menus in windows that were not created using the WinCreateStdWindow function. For these windows, you can load a menu resource by using the WinLoadMenu function and specifying the parent window for the menu. WinLoadMenu assigns the specified menu resource to the parent. To see the menu in the window, you must send a WM_UPDATEFRAME message to the parent after loading the menu resource. This strategy is especially useful for adding menus to a window created as a dialog window, but it can be used no matter what type of window is specified as the parent.
Accessing the System Menu
Although most applications do not alter the system menu, you can obtain the handle of the system menu by calling WinWindowFromID with a frame-window handle (or dialog-window handle) and the identifier FID_SYSMENU. Once you have the handle of the system menu, you can access the individual menu items by using predefined constants. For example, the following code fragment shows how to disable the Close menu item in the system menu of a window:
    HWND hwndSysMenu;
    HWND hwndFrame;
    hwndSysMenu = WinWindowFromID(hwndFrame, FID_SYSMENU);
    WinSendMsg(hwndSysMenu, MM_SETITEMATTR,
        MPFROM2SHORT(SC_CLOSE, TRUE),
        MPFROM2SHORT(MIA_DISABLED, MIA_DISABLED));
Responding to a User's Menu Choice
When a user chooses a menu item, the client window procedure receives a WM_COMMAND message with SHORT1FROMMP(mp1) equal to the menu identifier of the chosen item. Your application must use the menu identifier to guide its response to the choice. Typically, the code in the client window procedure resembles the following code fragment:
      case WM_COMMAND:
          DoMenuCommand(hwnd, SHORT1FROMMP(mp1));
          return 0;
The function that translates the menu identifier into an action typically resembles the following code fragment:
    VOID DoMenuCommand(
    HWND hwnd,
    USHORT usItemID)
    {
        /* Test the menu item. */
        switch (usItemID) {
            case IDM_FI_NEW:
                DoNew(hwnd);
                break;
          .
          . /* etc. */
          .
        }
    }
The menu window sends a WM_MENUSELECT message every time the menu selection changes. SHORT1FROMMP(mp1) contains the identifier of the item that is changing state, and SHORT2FROMMP(mp2) is a 16-bit Boolean value that describes whether or not the item is chosen; the mp2 parameter contains the handle of the menu.
If the Boolean value is FALSE, the item is selected but not chosen; for example, the user may have moved the cursor or mouse pointer over the item while the button was down. An application can use this message to display Help information at the bottom of the application window. The return value is ignored.
If the Boolean value is TRUE, the item is chosen-that is, the user pressed Enter or released the mouse button while an item was selected. If the application returns FALSE, the menu does not generate a WM_COMMAND, WM_SYSCOMMAND, or WM_HELP message, and the menu is not dismissed.
Setting and Querying Menu-Item Attributes
Menu-item attributes are represented in the fAttribute field of the MENUITEM data structure. Typically, attributes are set in the resource-definition file of the menu and are changed at run time as required. Applications can use the MM_SETITEMATTR and MM_QUERYITEMATTR messages to set and query attributes for a particular menu item. One of the most common uses of these messages is to check and uncheck menu items to let the user know what option is selected currently. For example, if you have a menu item that should toggle between checked and unchecked each time the user selects it, you can use the following figure to change the checked attribute. In this example, you send an MM_QUERYITEMATTR message to the menu item to obtain its current checked attribute; then, you use the exclusive OR operator to toggle the state; and finally, you send the new attribute state back to the item using an MM_SETITEMATTR message.
    usAttrib = SHORT1FROMMR(
        WinSendMsg(hwndMenu,           /* Submenu window        */
        MM_QUERYITEMATTR,              /* Message               */
        (MPARAM)itemID,                /* Item identifier       */
        (MPARAM)MIA_CHECKED            /* Attribute mask        */
        ));
    usAttrib = MIA_CHECKED;        /* XOR to toggle checked attribute */
    WinSendMsg(hwndMenu,                      /* Submenu window        */
        MM_SETITEMATTR,                       /* Message               */
        (MPARAM)itemID,                       /* Item identifier       */
        MPFROM2SHORT(MIA_CHECKED, usAttrib)); /* Attribute mask, value */
Adding and Deleting Menu Items
An application can add and delete items from its menus dynamically by sending MM_INSERTITEM and MM_DELETEITEM messages to the menu window. Any item, including those in submenus, can be deleted by sending a message to the menu window. Messages to insert items in submenus must be sent to the submenu's window (rather than to the window of the top-level menu). You can retrieve the handle of a submenu of the menu bar by sending an MM_QUERYITEM message to the menu-bar and specifying the identifier of the submenu item for the submenu, as shown in the following code fragment:
    /* IDM_MYMENUID is the identifier of the submenu containing the item. */
    MENUITEM mi;
    HWND hwndMenu, hwndSubMenu, hwndPullDown,hwndFrame;
    hwndMenu = WinWindowFromID(hwndFrame, FID_MENU);
    WinSendMsg(hwndMenu,                           /* Handle of menu bar  */
        MM_QUERYITEM,                              /* Message             */
        MPFROM2SHORT(IDM_MYMENUID, TRUE),          /* Submenu identifier  */
        (MPARAM) &mi);                             /* Pointer to MENUITEM */
    hwndPullDown = mi.hwndSubMenu;                 /* Handle to submenu   */
Once the application has the handle of the submenu, it can insert an item by filling in a MENUITEM structure and sending an MM_INSERTITEM message to the submenu. For text-menu items, the application must send a pointer to the text string as well as to the MENUITEM structure, as shown in the following figure.
    PSZ  pszNewItemString;
    mi.iPosition = MIT_END;
    mi.afStyle = MIS_TEXT;
    mi.afAttribute = 0;
    mi.id = IDM_MYMENU_FIRST;
    mi.hwndSubMenu = NULL;
    mi.hItem = 0;
    WinSendMsg(hwndPullDown, MM_INSERTITEM, (MPARAM) &mi,
        (MPARAM) pszNewItemString);
To delete an item, the application sends an MM_DELETEITEM message to the menu bar, specifying the identifier of the item to delete. For example, to clear all the items following IDM_MYMENU_FIRST in a submenu in which the items are numbered sequentially, use the following code:
    USHORT usItemNum;
    /* Clear all the items in MYMENU.                 */
    hwndMenu = WinWindowFromID(hwndFrame, FID_MENU);
    usItemNum = IDM_MYMENU_FIRST;
    while (WinSendMsg(hwndMenu, MM_DELETEITEM,
        MPFROM2SHORT(usItemNum++, TRUE), NULL) != 0);
Adding a complete submenu to the menu bar is a more complicated procedure than that shown in the previous examples. There are two strategies. The recommended technique is to define all possible submenus in your resource-definition file; and then, as your application runs, selectively remove and insert the submenus as needed.
For example, assume that your application has a submenu that you want to be displayed only when a particular application tool is in use. You must first define the submenu as part of the main menu resource in your resource-definition file, so that the system reads in the resource menu template and creates the submenu window along with the rest of the menu. You then can remove the submenu from the menu bar, saving the title of the submenu and the MENUITEM structure that defines the submenu, as shown in the following figure:
    HWND hwndMenu, hwndClient;
    MENUITEM mi;
    CHAR szMenuTitle[MAX_STRINGSIZE];
    /* Remove a submenu so that you can replace it later.           */
    /* Obtain the handle of a menu.                                 */
    hwndMenu = WinWindowFromID(WinQueryWindow(hwndClient, QW_PARENT),
                               FID_MENU);
    /* Obtain information on the item to remove.                    */
    WinSendMsg(hwndMenu, MM_QUERYITEM,
        MPFROM2SHORT(IDM_MENUID, TRUE),  /* TRUE to search submenus */
        (MPARAM)&mi);
    /* Save the text for the submenu item.                          */
    WinSendMsg(hwndMenu, MM_QUERYITEMTEXT,
        MPFROM2SHORT(IDM_FONT, MAX_STRINGSIZE),
        (MPARAM)szMenuTitle);
    /* Remove the item, but retain mi and szMenuTitle.              */
    WinSendMsg(hwndMenu, MM_REMOVEITEM,
        MPFROM2SHORT(IDM_FONT, TRUE), NULL);
It is important to use the MM_REMOVEITEM message, rather than MM_DELETEITEM, to remove the item; deleting the item destroys the submenu window-removing it does not. The submenu should remain intact so that you can insert it later.
To reinsert the submenu, send an MM_INSERTITEM message to the menu bar, passing the MENUITEM structure and menu title that you saved when you removed the item. The following code fragment shows how to insert a submenu that was removed by using the previous code example.
    /* Put the submenu back in and obtain the handle of the menu bar. */
    hwndMenu = WinWindowFromID(
                    WinQueryWindow(hwndClient, QW_PARENT), FID_MENU);
    /* Use the information that you saved when you removed the menu.  */
    WinSendMsg(hwndMenu, MM_INSERTITEM, (MPARAM)&mi,
               (MPARAM)szMenuTitle);
The other technique that you can use to insert a submenu in the menu bar is to build up, in memory, a data structure as a menu template and use that template and WinCreateWindow to create a submenu. The resultant submenu window handle then is placed in the hwndSubMenu field of a MENUITEM structure, and the menu item is sent to the menu bar with an MM_INSERTITEM message.
You also can create an empty submenu window by using WinCreateWindow. Pass NULL for the pCtlData and pPresParams parameters, instead of building the menu template in memory. Then insert a new menu item in the menu bar by using the MM_INSERTITEM message, setting the MIS_SUBMENU style, and putting the window handle of the created menu into the hwndSubMenu field. Then use the MM_INSERTITEM message to insert the items in the new pull-down menu.
Creating a Custom Menu Item
Applications can customize the appearance of an individual menu item by setting the MIS_OWNERDRAW style bit for the item. The operating system sends two different messages to an application that include owner-drawn menu items: WM_MEASUREITEM and WM_DRAWITEM. Both messages include a pointer to an OWNERITEM data structure.
WM_MEASUREITEM is sent only once for each owner-drawn item when the menu is initialized. The message is sent to the owner of the menu (typically, a frame window), which forwards the message to its client window. Typically, the client window procedure processes WM_MEASUREITEM by filling in the yTop and Right fields of the RECTL structure, specified by the rclItem field of this OWNERITEM structure; this specifies the size of the rectangle needed to enclose the item when it is drawn. The following code fragment responds to a WM_MEASUREITEM message.
    case WM_MEASUREITEM:
        ((POWNERITEM) mp2)->rclItem.xRight = 26;
        ((POWNERITEM) mp2)->rclItem.yTop = 10;
        return 0;
If a menu item has the MIS_OWNERDRAW style, the owner window receives a WM_DRAWITEM message every time the menu item needs to be drawn. You process this message by using the hps and rclItem fields of the OWNERITEM structure to draw the item. There are two situations in which the owner window receives a WM_DRAWITEM message:
- When the item must be redrawn completely
- When the item must be highlighted or have its highlight removed
You can choose to handle one or both of these situations. Typically, you handle the drawing of the item. You may not want to handle the second situation, however, since the system-default behavior (inverting the bits in the item rectangle) often is acceptable. The two situations in which a WM_DRAWITEM message is received are detected by comparing the values of the fsState and fsStateOld fields of the OWNERITEM structure that is sent as part of the message. If the two fields are the same, draw the item. Before drawing the item, however, check its attributes to see whether it has the attributes MIA_CHECKED, MIA_FRAMED, or MIA_DISABLED. Then draw the item according to the attributes.
For example, when the checked attribute of an owner-drawn menu item changes, the system sends a WM_DRAWITEM message to the item so that it can redraw itself and either draw or remove the check mark. If you want the system-default check mark, simply draw the item and leave the fsAttribute and fsAttributeOld fields unchanged; the system draws the check mark if necessary. If you draw the check mark yourself, clear the MIA_CHECKED bit in both fsAttribute and fsAttributeOld so that the system does not attempt to draw a check mark.
In the same example, if fsAttribute and fsAttributeOld are not equal, the highlight showing that an item is selected needs to change. The MIA_HILITED bit of the fsAttribute field is set if the item needs to be highlighted and is not set if the highlight needs to be removed. If you do not want to provide your own highlighting, you should ignore any WM_DRAWITEM message in which fsAttribute and fsAttributeOld are not equal. If you do not alter these two fields, the system performs its default highlighting operation. If you want to provide your own visual cue that an item is selected, respond to a WM_DRAWITEM message in which the fsAttribute and fsAttributeOld fields are not equal by providing the cue and clearing the MIA_HILITED bit of both fields before returning from the message.
Likewise, the MIA_CHECKED and MIA_FRAMED bits of fsAttribute and fsAttributeOld either can be used to perform the corresponding action or passed on, unchanged, so that the system performs the action. The following code fragment shows how to respond to a WM_DRAWITEM message when you want to draw the item and also be responsible for its highlighted state.
    case WM_DRAWITEM:
        {
        POWNERITEM poi;
        RECTL      rcl;
        MPARAM     mp2;
        poi = (POWNERITEM) mp2;
        /*
         * If the new attribute equals the old attribute,
         * redraw the entire item.
         */
        if (poi->fsAttribute == poi->fsAttributeOld) {
            /*
             * Draw the item in poi->hps and poi->rclItem, and check the
             * attributes for check marks. If you produce your own check marks,
             * use this line of code:
             *
             *     poi->fsAttributeOld = (poi->fsAttribute &= ~MIA_CHECKED;
             */
        }
        /* Else highlight the item or remove its highlight. */
        else if ((poi->fsAttribute & MIA_HILITED) !=
                (poi->fsAttributeOld & MIA_HILITED)) {
            /*
             * Set bits the same so that the menu window does not highlight
             * the item or remove its highlight.
             */
            poi->fsAttributeOld = (poi->fsAttribute &= ~MIA_HILITED);
        }
        return TRUE; /* TRUE means the item is drawn. */
        } /* endcase */
Responding to WM_DRAWITEM Message.