4 * Copyright 1993 Martin Ayotte
5 * Copyright 1994 Alexandre Julliard
6 * Copyright 1997 Morten Welinder
7 * Copyright 2005 Maxime Bellengé
8 * Copyright 2006 Phil Krylov
10 * This library is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU Lesser General Public
12 * License as published by the Free Software Foundation; either
13 * version 2.1 of the License, or (at your option) any later version.
15 * This library is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18 * Lesser General Public License for more details.
20 * You should have received a copy of the GNU Lesser General Public
21 * License along with this library; if not, write to the Free Software
22 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
26 * Note: the style MF_MOUSESELECT is used to mark popup items that
27 * have been selected, i.e. their popup menu is currently displayed.
28 * This is probably not the meaning this style has in MS-Windows.
30 * Note 2: where there is a difference, these menu API's are according
31 * the behavior of Windows 2k and Windows XP. Known differences with
32 * Windows 9x/ME are documented in the comments, in case an application
33 * is found to depend on the old behavior.
43 #include "wine/port.h"
54 #include "wine/winbase16.h"
55 #include "wine/winuser16.h"
57 #include "wine/server.h"
58 #include "wine/unicode.h"
61 #include "user_private.h"
62 #include "wine/debug.h"
64 WINE_DEFAULT_DEBUG_CHANNEL(menu);
65 WINE_DECLARE_DEBUG_CHANNEL(accel);
67 /* internal popup menu window messages */
69 #define MM_SETMENUHANDLE (WM_USER + 0)
70 #define MM_GETMENUHANDLE (WM_USER + 1)
72 /* Menu item structure */
74 /* ----------- MENUITEMINFO Stuff ----------- */
75 UINT fType; /* Item type. */
76 UINT fState; /* Item state. */
77 UINT_PTR wID; /* Item id. */
78 HMENU hSubMenu; /* Pop-up menu. */
79 HBITMAP hCheckBit; /* Bitmap when checked. */
80 HBITMAP hUnCheckBit; /* Bitmap when unchecked. */
81 LPWSTR text; /* Item text. */
82 ULONG_PTR dwItemData; /* Application defined. */
83 LPWSTR dwTypeData; /* depends on fMask */
84 HBITMAP hbmpItem; /* bitmap */
85 /* ----------- Wine stuff ----------- */
86 RECT rect; /* Item area (relative to menu window) */
87 UINT xTab; /* X position of text after Tab */
88 SIZE bmpsize; /* size needed for the HBMMENU_CALLBACK
92 /* Popup menu structure */
94 WORD wFlags; /* Menu flags (MF_POPUP, MF_SYSMENU) */
95 WORD wMagic; /* Magic number */
96 WORD Width; /* Width of the whole menu */
97 WORD Height; /* Height of the whole menu */
98 UINT nItems; /* Number of items in the menu */
99 HWND hWnd; /* Window containing the menu */
100 MENUITEM *items; /* Array of menu items */
101 UINT FocusedItem; /* Currently focused item */
102 HWND hwndOwner; /* window receiving the messages for ownerdraw */
103 BOOL bTimeToHide; /* Request hiding when receiving a second click in the top-level menu item */
104 BOOL bScrolling; /* Scroll arrows are active */
105 UINT nScrollPos; /* Current scroll position */
106 UINT nTotalHeight; /* Total height of menu items inside menu */
107 /* ------------ MENUINFO members ------ */
108 DWORD dwStyle; /* Extended menu style */
109 UINT cyMax; /* max height of the whole menu, 0 is screen height */
110 HBRUSH hbrBack; /* brush for menu background */
111 DWORD dwContextHelpID;
112 DWORD dwMenuData; /* application defined value */
113 HMENU hSysMenuOwner; /* Handle to the dummy sys menu holder */
114 SIZE maxBmpSize; /* Maximum size of the bitmap items */
115 } POPUPMENU, *LPPOPUPMENU;
117 /* internal flags for menu tracking */
119 #define TF_ENDMENU 0x10000
120 #define TF_SUSPENDPOPUP 0x20000
121 #define TF_SKIPREMOVE 0x40000
126 HMENU hCurrentMenu; /* current submenu (can be equal to hTopMenu)*/
127 HMENU hTopMenu; /* initial menu */
128 HWND hOwnerWnd; /* where notifications are sent */
132 #define MENU_MAGIC 0x554d /* 'MU' */
137 /* Internal MENU_TrackMenu() flags */
138 #define TPM_INTERNAL 0xF0000000
139 #define TPM_ENTERIDLEEX 0x80000000 /* set owner window for WM_ENTERIDLE */
140 #define TPM_BUTTONDOWN 0x40000000 /* menu was clicked before tracking */
141 #define TPM_POPUPMENU 0x20000000 /* menu is a popup menu */
143 /* Space between 2 columns */
144 #define MENU_COL_SPACE 4
146 /* top and bottom margins for popup menus */
147 #define MENU_TOP_MARGIN 3
148 #define MENU_BOTTOM_MARGIN 2
150 /* (other menu->FocusedItem values give the position of the focused item) */
151 #define NO_SELECTED_ITEM 0xffff
153 #define MENU_ITEM_TYPE(flags) \
154 ((flags) & (MF_STRING | MF_BITMAP | MF_OWNERDRAW | MF_SEPARATOR))
156 /* macro to test that flags do not indicate bitmap, ownerdraw or separator */
157 #define IS_STRING_ITEM(flags) (MENU_ITEM_TYPE ((flags)) == MF_STRING)
158 #define IS_MAGIC_BITMAP(id) ((id) && ((INT_PTR)(id) < 12) && ((INT_PTR)(id) >= -1))
160 #define IS_SYSTEM_MENU(menu) \
161 (!((menu)->wFlags & MF_POPUP) && ((menu)->wFlags & MF_SYSMENU))
163 #define MENUITEMINFO_TYPE_MASK \
164 (MFT_STRING | MFT_BITMAP | MFT_OWNERDRAW | MFT_SEPARATOR | \
165 MFT_MENUBARBREAK | MFT_MENUBREAK | MFT_RADIOCHECK | \
166 MFT_RIGHTORDER | MFT_RIGHTJUSTIFY /* same as MF_HELP */ )
167 #define TYPE_MASK (MENUITEMINFO_TYPE_MASK | MF_POPUP | MF_SYSMENU)
168 #define STATE_MASK (~TYPE_MASK)
169 #define MENUITEMINFO_STATE_MASK (STATE_MASK & ~(MF_BYPOSITION | MF_MOUSESELECT))
171 #define WIN_ALLOWED_MENU(style) ((style & (WS_CHILD | WS_POPUP)) != WS_CHILD)
173 static SIZE menucharsize;
174 static UINT ODitemheight; /* default owner drawn item height */
176 /* Use global popup window because there's no way 2 menus can
177 * be tracked at the same time. */
178 static HWND top_popup;
180 /* Flag set by EndMenu() to force an exit from menu tracking */
181 static BOOL fEndMenu = FALSE;
183 static LRESULT WINAPI PopupMenuWndProc( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam );
185 DWORD WINAPI DrawMenuBarTemp(HWND hwnd, HDC hDC, LPRECT lprect, HMENU hMenu, HFONT hFont);
187 /*********************************************************************
188 * menu class descriptor
190 const struct builtin_class_descr MENU_builtin_class =
192 (LPCWSTR)POPUPMENU_CLASS_ATOM, /* name */
193 CS_DROPSHADOW | CS_SAVEBITS | CS_DBLCLKS, /* style */
194 NULL, /* procA (winproc is Unicode only) */
195 PopupMenuWndProc, /* procW */
196 sizeof(HMENU), /* extra */
197 IDC_ARROW, /* cursor */
198 (HBRUSH)(COLOR_MENU+1) /* brush */
202 /***********************************************************************
203 * debug_print_menuitem
205 * Print a menuitem in readable form.
208 #define debug_print_menuitem(pre, mp, post) \
209 do { if (TRACE_ON(menu)) do_debug_print_menuitem(pre, mp, post); } while (0)
211 #define MENUOUT(text) \
212 TRACE("%s%s", (count++ ? "," : ""), (text))
214 #define MENUFLAG(bit,text) \
216 if (flags & (bit)) { flags &= ~(bit); MENUOUT ((text)); } \
219 static void do_debug_print_menuitem(const char *prefix, const MENUITEM *mp,
222 static const char * const hbmmenus[] = { "HBMMENU_CALLBACK", "", "HBMMENU_SYSTEM",
223 "HBMMENU_MBAR_RESTORE", "HBMMENU_MBAR_MINIMIZE", "UNKNOWN BITMAP", "HBMMENU_MBAR_CLOSE",
224 "HBMMENU_MBAR_CLOSE_D", "HBMMENU_MBAR_MINIMIZE_D", "HBMMENU_POPUP_CLOSE",
225 "HBMMENU_POPUP_RESTORE", "HBMMENU_POPUP_MAXIMIZE", "HBMMENU_POPUP_MINIMIZE"};
226 TRACE("%s ", prefix);
228 UINT flags = mp->fType;
229 TRACE( "{ ID=0x%lx", mp->wID);
231 TRACE( ", Sub=%p", mp->hSubMenu);
235 MENUFLAG( MFT_SEPARATOR, "sep");
236 MENUFLAG( MFT_OWNERDRAW, "own");
237 MENUFLAG( MFT_BITMAP, "bit");
238 MENUFLAG(MF_POPUP, "pop");
239 MENUFLAG(MFT_MENUBARBREAK, "barbrk");
240 MENUFLAG(MFT_MENUBREAK, "brk");
241 MENUFLAG(MFT_RADIOCHECK, "radio");
242 MENUFLAG(MFT_RIGHTORDER, "rorder");
243 MENUFLAG(MF_SYSMENU, "sys");
244 MENUFLAG(MFT_RIGHTJUSTIFY, "right"); /* same as MF_HELP */
246 TRACE( "+0x%x", flags);
252 MENUFLAG(MFS_GRAYED, "grey");
253 MENUFLAG(MFS_DEFAULT, "default");
254 MENUFLAG(MFS_DISABLED, "dis");
255 MENUFLAG(MFS_CHECKED, "check");
256 MENUFLAG(MFS_HILITE, "hi");
257 MENUFLAG(MF_USECHECKBITMAPS, "usebit");
258 MENUFLAG(MF_MOUSESELECT, "mouse");
260 TRACE( "+0x%x", flags);
263 TRACE( ", Chk=%p", mp->hCheckBit);
265 TRACE( ", Unc=%p", mp->hUnCheckBit);
267 TRACE( ", Text=%s", debugstr_w(mp->text));
269 TRACE( ", ItemData=0x%08lx", mp->dwItemData);
272 if( IS_MAGIC_BITMAP(mp->hbmpItem))
273 TRACE( ", hbitmap=%s", hbmmenus[ (INT_PTR)mp->hbmpItem + 1]);
275 TRACE( ", hbitmap=%p", mp->hbmpItem);
280 TRACE(" %s\n", postfix);
287 /***********************************************************************
290 * Validate the given menu handle and returns the menu structure pointer.
292 static POPUPMENU *MENU_GetMenu(HMENU hMenu)
294 POPUPMENU *menu = USER_HEAP_LIN_ADDR(hMenu);
295 if (!menu || menu->wMagic != MENU_MAGIC)
297 WARN("invalid menu handle=%p, ptr=%p, magic=%x\n", hMenu, menu, menu? menu->wMagic:0);
303 /***********************************************************************
306 * Get the system menu of a window
308 static HMENU get_win_sys_menu( HWND hwnd )
311 WND *win = WIN_GetPtr( hwnd );
312 if (win && win != WND_OTHER_PROCESS && win != WND_DESKTOP)
315 WIN_ReleasePtr( win );
320 /***********************************************************************
323 static HFONT get_menu_font( BOOL bold )
325 static HFONT hMenuFont, hMenuFontBold;
327 HFONT ret = bold ? hMenuFontBold : hMenuFont;
331 NONCLIENTMETRICSW ncm;
334 ncm.cbSize = sizeof(NONCLIENTMETRICSW);
335 SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, sizeof(NONCLIENTMETRICSW), &ncm, 0);
339 ncm.lfMenuFont.lfWeight += 300;
340 if (ncm.lfMenuFont.lfWeight > 1000) ncm.lfMenuFont.lfWeight = 1000;
342 if (!(ret = CreateFontIndirectW( &ncm.lfMenuFont ))) return 0;
343 prev = InterlockedCompareExchangePointer( (void **)(bold ? &hMenuFontBold : &hMenuFont),
347 /* another thread beat us to it */
355 /***********************************************************************
358 static HBITMAP get_arrow_bitmap(void)
360 static HBITMAP arrow_bitmap;
362 if (!arrow_bitmap) arrow_bitmap = LoadBitmapW(0, MAKEINTRESOURCEW(OBM_MNARROW));
366 /***********************************************************************
367 * get_down_arrow_bitmap
369 static HBITMAP get_down_arrow_bitmap(void)
371 static HBITMAP arrow_bitmap;
373 if (!arrow_bitmap) arrow_bitmap = LoadBitmapW(0, MAKEINTRESOURCEW(OBM_DNARROW));
377 /***********************************************************************
378 * get_down_arrow_inactive_bitmap
380 static HBITMAP get_down_arrow_inactive_bitmap(void)
382 static HBITMAP arrow_bitmap;
384 if (!arrow_bitmap) arrow_bitmap = LoadBitmapW(0, MAKEINTRESOURCEW(OBM_DNARROWI));
388 /***********************************************************************
389 * get_up_arrow_bitmap
391 static HBITMAP get_up_arrow_bitmap(void)
393 static HBITMAP arrow_bitmap;
395 if (!arrow_bitmap) arrow_bitmap = LoadBitmapW(0, MAKEINTRESOURCEW(OBM_UPARROW));
399 /***********************************************************************
400 * get_up_arrow_inactive_bitmap
402 static HBITMAP get_up_arrow_inactive_bitmap(void)
404 static HBITMAP arrow_bitmap;
406 if (!arrow_bitmap) arrow_bitmap = LoadBitmapW(0, MAKEINTRESOURCEW(OBM_UPARROWI));
410 /***********************************************************************
413 * Return the default system menu.
415 static HMENU MENU_CopySysPopup(void)
417 static const WCHAR sysmenuW[] = {'S','Y','S','M','E','N','U',0};
418 HMENU hMenu = LoadMenuW(user32_module, sysmenuW);
421 POPUPMENU* menu = MENU_GetMenu(hMenu);
422 menu->wFlags |= MF_SYSMENU | MF_POPUP;
423 SetMenuDefaultItem(hMenu, SC_CLOSE, FALSE);
426 ERR("Unable to load default system menu\n" );
428 TRACE("returning %p.\n", hMenu );
434 /**********************************************************************
437 * Create a copy of the system menu. System menu in Windows is
438 * a special menu bar with the single entry - system menu popup.
439 * This popup is presented to the outside world as a "system menu".
440 * However, the real system menu handle is sometimes seen in the
441 * WM_MENUSELECT parameters (and Word 6 likes it this way).
443 static HMENU MENU_GetSysMenu( HWND hWnd, HMENU hPopupMenu )
447 TRACE("loading system menu, hWnd %p, hPopupMenu %p\n", hWnd, hPopupMenu);
448 if ((hMenu = CreateMenu()))
450 POPUPMENU *menu = MENU_GetMenu(hMenu);
451 menu->wFlags = MF_SYSMENU;
452 menu->hWnd = WIN_GetFullHandle( hWnd );
453 TRACE("hWnd %p (hMenu %p)\n", menu->hWnd, hMenu);
456 hPopupMenu = MENU_CopySysPopup();
460 if (GetClassLongW(hWnd, GCL_STYLE) & CS_NOCLOSE)
461 DeleteMenu(hPopupMenu, SC_CLOSE, MF_BYCOMMAND);
463 InsertMenuW( hMenu, -1, MF_SYSMENU | MF_POPUP | MF_BYPOSITION,
464 (UINT_PTR)hPopupMenu, NULL );
466 menu->items[0].fType = MF_SYSMENU | MF_POPUP;
467 menu->items[0].fState = 0;
468 if ((menu = MENU_GetMenu(hPopupMenu))) menu->wFlags |= MF_SYSMENU;
470 TRACE("hMenu=%p (hPopup %p)\n", hMenu, hPopupMenu );
473 DestroyMenu( hMenu );
475 ERR("failed to load system menu!\n");
480 /***********************************************************************
481 * MENU_InitSysMenuPopup
483 * Grey the appropriate items in System menu.
485 static void MENU_InitSysMenuPopup( HMENU hmenu, DWORD style, DWORD clsStyle )
489 gray = !(style & WS_THICKFRAME) || (style & (WS_MAXIMIZE | WS_MINIMIZE));
490 EnableMenuItem( hmenu, SC_SIZE, (gray ? MF_GRAYED : MF_ENABLED) );
491 gray = ((style & WS_MAXIMIZE) != 0);
492 EnableMenuItem( hmenu, SC_MOVE, (gray ? MF_GRAYED : MF_ENABLED) );
493 gray = !(style & WS_MINIMIZEBOX) || (style & WS_MINIMIZE);
494 EnableMenuItem( hmenu, SC_MINIMIZE, (gray ? MF_GRAYED : MF_ENABLED) );
495 gray = !(style & WS_MAXIMIZEBOX) || (style & WS_MAXIMIZE);
496 EnableMenuItem( hmenu, SC_MAXIMIZE, (gray ? MF_GRAYED : MF_ENABLED) );
497 gray = !(style & (WS_MAXIMIZE | WS_MINIMIZE));
498 EnableMenuItem( hmenu, SC_RESTORE, (gray ? MF_GRAYED : MF_ENABLED) );
499 gray = (clsStyle & CS_NOCLOSE) != 0;
501 /* The menu item must keep its state if it's disabled */
503 EnableMenuItem( hmenu, SC_CLOSE, MF_GRAYED);
507 /******************************************************************************
509 * UINT MENU_GetStartOfNextColumn(
512 *****************************************************************************/
514 static UINT MENU_GetStartOfNextColumn(
517 POPUPMENU *menu = MENU_GetMenu(hMenu);
521 return NO_SELECTED_ITEM;
523 i = menu->FocusedItem + 1;
524 if( i == NO_SELECTED_ITEM )
527 for( ; i < menu->nItems; ++i ) {
528 if (menu->items[i].fType & (MF_MENUBREAK | MF_MENUBARBREAK))
532 return NO_SELECTED_ITEM;
536 /******************************************************************************
538 * UINT MENU_GetStartOfPrevColumn(
541 *****************************************************************************/
543 static UINT MENU_GetStartOfPrevColumn(
546 POPUPMENU *menu = MENU_GetMenu(hMenu);
550 return NO_SELECTED_ITEM;
552 if( menu->FocusedItem == 0 || menu->FocusedItem == NO_SELECTED_ITEM )
553 return NO_SELECTED_ITEM;
555 /* Find the start of the column */
557 for(i = menu->FocusedItem; i != 0 &&
558 !(menu->items[i].fType & (MF_MENUBREAK | MF_MENUBARBREAK));
562 return NO_SELECTED_ITEM;
564 for(--i; i != 0; --i) {
565 if (menu->items[i].fType & (MF_MENUBREAK | MF_MENUBARBREAK))
569 TRACE("ret %d.\n", i );
576 /***********************************************************************
579 * Find a menu item. Return a pointer on the item, and modifies *hmenu
580 * in case the item was in a sub-menu.
582 static MENUITEM *MENU_FindItem( HMENU *hmenu, UINT *nPos, UINT wFlags )
585 MENUITEM *fallback = NULL;
586 UINT fallback_pos = 0;
589 if ((*hmenu == (HMENU)0xffff) || (!(menu = MENU_GetMenu(*hmenu)))) return NULL;
590 if (wFlags & MF_BYPOSITION)
592 if (*nPos >= menu->nItems) return NULL;
593 return &menu->items[*nPos];
597 MENUITEM *item = menu->items;
598 for (i = 0; i < menu->nItems; i++, item++)
600 if (item->fType & MF_POPUP)
602 HMENU hsubmenu = item->hSubMenu;
603 MENUITEM *subitem = MENU_FindItem( &hsubmenu, nPos, wFlags );
609 else if (item->wID == *nPos)
611 /* fallback to this item if nothing else found */
616 else if (item->wID == *nPos)
625 *nPos = fallback_pos;
630 /***********************************************************************
633 * Find a Sub menu. Return the position of the submenu, and modifies
634 * *hmenu in case it is found in another sub-menu.
635 * If the submenu cannot be found, NO_SELECTED_ITEM is returned.
637 UINT MENU_FindSubMenu( HMENU *hmenu, HMENU hSubTarget )
642 if (((*hmenu)==(HMENU)0xffff) ||
643 (!(menu = MENU_GetMenu(*hmenu))))
644 return NO_SELECTED_ITEM;
646 for (i = 0; i < menu->nItems; i++, item++) {
647 if(!(item->fType & MF_POPUP)) continue;
648 if (item->hSubMenu == hSubTarget) {
652 HMENU hsubmenu = item->hSubMenu;
653 UINT pos = MENU_FindSubMenu( &hsubmenu, hSubTarget );
654 if (pos != NO_SELECTED_ITEM) {
660 return NO_SELECTED_ITEM;
663 /***********************************************************************
666 static void MENU_FreeItemData( MENUITEM* item )
669 HeapFree( GetProcessHeap(), 0, item->text );
672 /***********************************************************************
673 * MENU_AdjustMenuItemRect
675 * Adjust menu item rectangle according to scrolling state.
678 MENU_AdjustMenuItemRect(const POPUPMENU *menu, LPRECT rect)
680 if (menu->bScrolling)
682 UINT arrow_bitmap_height;
685 GetObjectW(get_up_arrow_bitmap(), sizeof(bmp), &bmp);
686 arrow_bitmap_height = bmp.bmHeight;
687 rect->top += arrow_bitmap_height - menu->nScrollPos;
688 rect->bottom += arrow_bitmap_height - menu->nScrollPos;
693 /***********************************************************************
694 * MENU_FindItemByCoords
696 * Find the item at the specified coordinates (screen coords). Does
697 * not work for child windows and therefore should not be called for
698 * an arbitrary system menu.
700 static MENUITEM *MENU_FindItemByCoords( const POPUPMENU *menu,
701 POINT pt, UINT *pos )
707 if (!GetWindowRect(menu->hWnd, &rect)) return NULL;
711 for (i = 0; i < menu->nItems; i++, item++)
714 MENU_AdjustMenuItemRect(menu, &rect);
715 if (PtInRect(&rect, pt))
725 /***********************************************************************
728 * Find the menu item selected by a key press.
729 * Return item id, -1 if none, -2 if we should close the menu.
731 static UINT MENU_FindItemByKey( HWND hwndOwner, HMENU hmenu,
732 WCHAR key, BOOL forceMenuChar )
734 TRACE("\tlooking for '%c' (0x%02x) in [%p]\n", (char)key, key, hmenu );
736 if (!IsMenu( hmenu )) hmenu = GetSubMenu( get_win_sys_menu(hwndOwner), 0);
740 POPUPMENU *menu = MENU_GetMenu( hmenu );
741 MENUITEM *item = menu->items;
748 for (i = 0; i < menu->nItems; i++, item++)
752 WCHAR *p = item->text - 2;
755 p = strchrW (p + 2, '&');
757 while (p != NULL && p [1] == '&');
758 if (p && (toupperW(p[1]) == toupperW(key))) return i;
762 menuchar = SendMessageW( hwndOwner, WM_MENUCHAR,
763 MAKEWPARAM( key, menu->wFlags ), (LPARAM)hmenu );
764 if (HIWORD(menuchar) == 2) return LOWORD(menuchar);
765 if (HIWORD(menuchar) == 1) return (UINT)(-2);
771 /***********************************************************************
772 * MENU_GetBitmapItemSize
774 * Get the size of a bitmap item.
776 static void MENU_GetBitmapItemSize( MENUITEM *lpitem, SIZE *size,
780 HBITMAP bmp = lpitem->hbmpItem;
782 size->cx = size->cy = 0;
784 /* check if there is a magic menu item associated with this item */
785 switch( (INT_PTR) bmp )
787 case (INT_PTR)HBMMENU_CALLBACK:
789 MEASUREITEMSTRUCT measItem;
790 measItem.CtlType = ODT_MENU;
792 measItem.itemID = lpitem->wID;
793 measItem.itemWidth = lpitem->rect.right - lpitem->rect.left;
794 measItem.itemHeight = lpitem->rect.bottom - lpitem->rect.top;
795 measItem.itemData = lpitem->dwItemData;
796 SendMessageW( hwndOwner, WM_MEASUREITEM, lpitem->wID, (LPARAM)&measItem);
797 size->cx = measItem.itemWidth;
798 size->cy = measItem.itemHeight;
802 case (INT_PTR)HBMMENU_SYSTEM:
803 if (lpitem->dwItemData)
805 bmp = (HBITMAP)lpitem->dwItemData;
809 case (INT_PTR)HBMMENU_MBAR_RESTORE:
810 case (INT_PTR)HBMMENU_MBAR_MINIMIZE:
811 case (INT_PTR)HBMMENU_MBAR_MINIMIZE_D:
812 case (INT_PTR)HBMMENU_MBAR_CLOSE:
813 case (INT_PTR)HBMMENU_MBAR_CLOSE_D:
814 size->cx = GetSystemMetrics( SM_CYMENU ) - 4;
817 case (INT_PTR)HBMMENU_POPUP_CLOSE:
818 case (INT_PTR)HBMMENU_POPUP_RESTORE:
819 case (INT_PTR)HBMMENU_POPUP_MAXIMIZE:
820 case (INT_PTR)HBMMENU_POPUP_MINIMIZE:
821 FIXME("Magic %p not implemented\n", bmp );
824 if (GetObjectW(bmp, sizeof(bm), &bm ))
826 size->cx = bm.bmWidth;
827 size->cy = bm.bmHeight;
831 /***********************************************************************
832 * MENU_DrawBitmapItem
834 * Draw a bitmap item.
836 static void MENU_DrawBitmapItem( HDC hdc, MENUITEM *lpitem, const RECT *rect,
837 HMENU hmenu, HWND hwndOwner, UINT odaction, BOOL menuBar)
843 int w = rect->right - rect->left;
844 int h = rect->bottom - rect->top;
847 HBITMAP hbmToDraw = lpitem->hbmpItem;
850 /* Check if there is a magic menu item associated with this item */
851 if (IS_MAGIC_BITMAP(hbmToDraw))
856 switch((INT_PTR)hbmToDraw)
858 case (INT_PTR)HBMMENU_SYSTEM:
859 if (lpitem->dwItemData)
861 bmp = (HBITMAP)lpitem->dwItemData;
862 if (!GetObjectW( bmp, sizeof(bm), &bm )) return;
866 static HBITMAP hBmpSysMenu;
868 if (!hBmpSysMenu) hBmpSysMenu = LoadBitmapW(0, MAKEINTRESOURCEW(OBM_CLOSE));
870 if (!GetObjectW( bmp, sizeof(bm), &bm )) return;
871 /* only use right half of the bitmap */
872 bmp_xoffset = bm.bmWidth / 2;
873 bm.bmWidth -= bmp_xoffset;
876 case (INT_PTR)HBMMENU_MBAR_RESTORE:
877 flags = DFCS_CAPTIONRESTORE;
879 case (INT_PTR)HBMMENU_MBAR_MINIMIZE:
880 flags = DFCS_CAPTIONMIN;
882 case (INT_PTR)HBMMENU_MBAR_MINIMIZE_D:
883 flags = DFCS_CAPTIONMIN | DFCS_INACTIVE;
885 case (INT_PTR)HBMMENU_MBAR_CLOSE:
886 flags = DFCS_CAPTIONCLOSE;
888 case (INT_PTR)HBMMENU_MBAR_CLOSE_D:
889 flags = DFCS_CAPTIONCLOSE | DFCS_INACTIVE;
891 case (INT_PTR)HBMMENU_CALLBACK:
893 DRAWITEMSTRUCT drawItem;
894 drawItem.CtlType = ODT_MENU;
896 drawItem.itemID = lpitem->wID;
897 drawItem.itemAction = odaction;
898 drawItem.itemState = (lpitem->fState & MF_CHECKED)?ODS_CHECKED:0;
899 drawItem.itemState |= (lpitem->fState & MF_DEFAULT)?ODS_DEFAULT:0;
900 drawItem.itemState |= (lpitem->fState & MF_DISABLED)?ODS_DISABLED:0;
901 drawItem.itemState |= (lpitem->fState & MF_GRAYED)?ODS_GRAYED|ODS_DISABLED:0;
902 drawItem.itemState |= (lpitem->fState & MF_HILITE)?ODS_SELECTED:0;
903 drawItem.hwndItem = (HWND)hmenu;
905 drawItem.itemData = lpitem->dwItemData;
906 drawItem.rcItem = *rect;
907 SendMessageW( hwndOwner, WM_DRAWITEM, 0, (LPARAM)&drawItem);
911 case (INT_PTR)HBMMENU_POPUP_CLOSE:
912 case (INT_PTR)HBMMENU_POPUP_RESTORE:
913 case (INT_PTR)HBMMENU_POPUP_MAXIMIZE:
914 case (INT_PTR)HBMMENU_POPUP_MINIMIZE:
916 FIXME("Magic %p not implemented\n", hbmToDraw);
920 InflateRect( &r, -1, -1 );
921 if (lpitem->fState & MF_HILITE) flags |= DFCS_PUSHED;
922 DrawFrameControl( hdc, &r, DFC_CAPTION, flags );
926 if (!bmp || !GetObjectW( bmp, sizeof(bm), &bm )) return;
929 hdcMem = CreateCompatibleDC( hdc );
930 SelectObject( hdcMem, bmp );
932 /* handle fontsize > bitmap_height */
933 top = (h>bm.bmHeight) ? rect->top+(h-bm.bmHeight)/2 : rect->top;
935 rop=((lpitem->fState & MF_HILITE) && !IS_MAGIC_BITMAP(hbmToDraw)) ? NOTSRCCOPY : SRCCOPY;
936 if ((lpitem->fState & MF_HILITE) && lpitem->hbmpItem)
937 SetBkColor(hdc, GetSysColor(COLOR_HIGHLIGHT));
938 BitBlt( hdc, left, top, w, h, hdcMem, bmp_xoffset, 0, rop );
943 /***********************************************************************
946 * Calculate the size of the menu item and store it in lpitem->rect.
948 static void MENU_CalcItemSize( HDC hdc, MENUITEM *lpitem, HWND hwndOwner,
949 INT orgX, INT orgY, BOOL menuBar, POPUPMENU* lppop )
952 UINT check_bitmap_width = GetSystemMetrics( SM_CXMENUCHECK );
953 UINT arrow_bitmap_width;
957 TRACE("dc=%p owner=%p (%d,%d)\n", hdc, hwndOwner, orgX, orgY);
958 debug_print_menuitem("MENU_CalcItemSize: menuitem:", lpitem,
959 (menuBar ? " (MenuBar)" : ""));
961 GetObjectW( get_arrow_bitmap(), sizeof(bm), &bm );
962 arrow_bitmap_width = bm.bmWidth;
964 /* not done in Menu_Init: GetDialogBaseUnits() breaks there */
965 if( !menucharsize.cx ) {
966 menucharsize.cx = GdiGetCharDimensions( hdc, NULL, &menucharsize.cy );
967 /* Win95/98/ME will use menucharsize.cy here. Testing is possible
968 * but it is unlikely an application will depend on that */
969 ODitemheight = HIWORD( GetDialogBaseUnits());
972 SetRect( &lpitem->rect, orgX, orgY, orgX, orgY );
974 if (lpitem->fType & MF_OWNERDRAW)
976 MEASUREITEMSTRUCT mis;
977 mis.CtlType = ODT_MENU;
979 mis.itemID = lpitem->wID;
980 mis.itemData = lpitem->dwItemData;
981 mis.itemHeight = ODitemheight;
983 SendMessageW( hwndOwner, WM_MEASUREITEM, 0, (LPARAM)&mis );
984 /* Tests reveal that Windows ( Win95 thru WinXP) adds twice the average
985 * width of a menufont character to the width of an owner-drawn menu.
987 lpitem->rect.right += mis.itemWidth + 2 * menucharsize.cx;
989 /* under at least win95 you seem to be given a standard
990 height for the menu and the height value is ignored */
991 lpitem->rect.bottom += GetSystemMetrics(SM_CYMENUSIZE);
993 lpitem->rect.bottom += mis.itemHeight;
995 TRACE("id=%04lx size=%dx%d\n",
996 lpitem->wID, lpitem->rect.right-lpitem->rect.left,
997 lpitem->rect.bottom-lpitem->rect.top);
1001 if (lpitem->fType & MF_SEPARATOR)
1003 lpitem->rect.bottom += GetSystemMetrics( SM_CYMENUSIZE)/2;
1005 lpitem->rect.right += arrow_bitmap_width + menucharsize.cx;
1013 if (lpitem->hbmpItem) {
1016 MENU_GetBitmapItemSize(lpitem, &size, hwndOwner);
1017 /* Keep the size of the bitmap in callback mode to be able
1018 * to draw it correctly */
1019 lpitem->bmpsize = size;
1020 lppop->maxBmpSize.cx = max( lppop->maxBmpSize.cx, size.cx);
1021 lppop->maxBmpSize.cy = max( lppop->maxBmpSize.cy, size.cy);
1022 lpitem->rect.right += size.cx + 2;
1023 itemheight = size.cy + 2;
1025 if( !(lppop->dwStyle & MNS_NOCHECK))
1026 lpitem->rect.right += check_bitmap_width;
1027 lpitem->rect.right += 4 + menucharsize.cx;
1028 lpitem->xTab = lpitem->rect.right;
1029 lpitem->rect.right += arrow_bitmap_width;
1030 } else if (lpitem->hbmpItem) { /* menuBar */
1033 MENU_GetBitmapItemSize( lpitem, &size, hwndOwner );
1034 lpitem->bmpsize = size;
1035 lpitem->rect.right += size.cx;
1036 if( lpitem->text) lpitem->rect.right += 2;
1037 itemheight = size.cy;
1040 /* it must be a text item - unless it's the system menu */
1041 if (!(lpitem->fType & MF_SYSMENU) && lpitem->text) {
1042 HFONT hfontOld = NULL;
1043 RECT rc = lpitem->rect;
1044 LONG txtheight, txtwidth;
1046 if ( lpitem->fState & MFS_DEFAULT ) {
1047 hfontOld = SelectObject( hdc, get_menu_font(TRUE) );
1050 txtheight = DrawTextW( hdc, lpitem->text, -1, &rc,
1051 DT_SINGLELINE|DT_CALCRECT);
1052 lpitem->rect.right += rc.right - rc.left;
1053 itemheight = max( max( itemheight, txtheight),
1054 GetSystemMetrics( SM_CYMENU) - 1);
1055 lpitem->rect.right += 2 * menucharsize.cx;
1057 if ((p = strchrW( lpitem->text, '\t' )) != NULL) {
1060 int n = (int)( p - lpitem->text);
1061 /* Item contains a tab (only meaningful in popup menus) */
1062 /* get text size before the tab */
1063 txtheight = DrawTextW( hdc, lpitem->text, n, &rc,
1064 DT_SINGLELINE|DT_CALCRECT);
1065 txtwidth = rc.right - rc.left;
1066 p += 1; /* advance past the Tab */
1067 /* get text size after the tab */
1068 tmpheight = DrawTextW( hdc, p, -1, &tmprc,
1069 DT_SINGLELINE|DT_CALCRECT);
1070 lpitem->xTab += txtwidth;
1071 txtheight = max( txtheight, tmpheight);
1072 txtwidth += menucharsize.cx + /* space for the tab */
1073 tmprc.right - tmprc.left; /* space for the short cut */
1075 txtheight = DrawTextW( hdc, lpitem->text, -1, &rc,
1076 DT_SINGLELINE|DT_CALCRECT);
1077 txtwidth = rc.right - rc.left;
1078 lpitem->xTab += txtwidth;
1080 lpitem->rect.right += 2 + txtwidth;
1081 itemheight = max( itemheight,
1082 max( txtheight + 2, menucharsize.cy + 4));
1084 if (hfontOld) SelectObject (hdc, hfontOld);
1085 } else if( menuBar) {
1086 itemheight = max( itemheight, GetSystemMetrics(SM_CYMENU)-1);
1088 lpitem->rect.bottom += itemheight;
1089 TRACE("%s\n", wine_dbgstr_rect( &lpitem->rect));
1093 /***********************************************************************
1094 * MENU_GetMaxPopupHeight
1097 MENU_GetMaxPopupHeight(const POPUPMENU *lppop)
1100 return lppop->cyMax;
1101 return GetSystemMetrics(SM_CYSCREEN) - GetSystemMetrics(SM_CYBORDER);
1105 /***********************************************************************
1106 * MENU_PopupMenuCalcSize
1108 * Calculate the size of a popup menu.
1110 static void MENU_PopupMenuCalcSize( LPPOPUPMENU lppop )
1115 int orgX, orgY, maxX, maxTab, maxTabWidth, maxHeight;
1117 lppop->Width = lppop->Height = 0;
1118 if (lppop->nItems == 0) return;
1121 SelectObject( hdc, get_menu_font(FALSE));
1126 lppop->maxBmpSize.cx = 0;
1127 lppop->maxBmpSize.cy = 0;
1129 while (start < lppop->nItems)
1131 lpitem = &lppop->items[start];
1133 if( lpitem->fType & (MF_MENUBREAK | MF_MENUBARBREAK))
1134 orgX += MENU_COL_SPACE;
1135 orgY = MENU_TOP_MARGIN;
1137 maxTab = maxTabWidth = 0;
1138 /* Parse items until column break or end of menu */
1139 for (i = start; i < lppop->nItems; i++, lpitem++)
1142 (lpitem->fType & (MF_MENUBREAK | MF_MENUBARBREAK))) break;
1144 MENU_CalcItemSize( hdc, lpitem, lppop->hwndOwner, orgX, orgY, FALSE, lppop );
1145 maxX = max( maxX, lpitem->rect.right );
1146 orgY = lpitem->rect.bottom;
1147 if (IS_STRING_ITEM(lpitem->fType) && lpitem->xTab)
1149 maxTab = max( maxTab, lpitem->xTab );
1150 maxTabWidth = max(maxTabWidth,lpitem->rect.right-lpitem->xTab);
1154 /* Finish the column (set all items to the largest width found) */
1155 maxX = max( maxX, maxTab + maxTabWidth );
1156 for (lpitem = &lppop->items[start]; start < i; start++, lpitem++)
1158 lpitem->rect.right = maxX;
1159 if (IS_STRING_ITEM(lpitem->fType) && lpitem->xTab)
1160 lpitem->xTab = maxTab;
1163 lppop->Height = max( lppop->Height, orgY );
1166 lppop->Width = maxX;
1168 /* space for 3d border */
1169 lppop->Height += MENU_BOTTOM_MARGIN;
1172 /* Adjust popup height if it exceeds maximum */
1173 maxHeight = MENU_GetMaxPopupHeight(lppop);
1174 lppop->nTotalHeight = lppop->Height - MENU_TOP_MARGIN;
1175 if (lppop->Height >= maxHeight)
1177 lppop->Height = maxHeight;
1178 lppop->bScrolling = TRUE;
1182 lppop->bScrolling = FALSE;
1185 ReleaseDC( 0, hdc );
1189 /***********************************************************************
1190 * MENU_MenuBarCalcSize
1192 * FIXME: Word 6 implements its own MDI and its own 'close window' bitmap
1193 * height is off by 1 pixel which causes lengthy window relocations when
1194 * active document window is maximized/restored.
1196 * Calculate the size of the menu bar.
1198 static void MENU_MenuBarCalcSize( HDC hdc, LPRECT lprect,
1199 LPPOPUPMENU lppop, HWND hwndOwner )
1202 int start, i, orgX, orgY, maxY, helpPos;
1204 if ((lprect == NULL) || (lppop == NULL)) return;
1205 if (lppop->nItems == 0) return;
1206 TRACE("lprect %p %s\n", lprect, wine_dbgstr_rect( lprect));
1207 lppop->Width = lprect->right - lprect->left;
1209 maxY = lprect->top+1;
1212 lppop->maxBmpSize.cx = 0;
1213 lppop->maxBmpSize.cy = 0;
1214 while (start < lppop->nItems)
1216 lpitem = &lppop->items[start];
1217 orgX = lprect->left;
1220 /* Parse items until line break or end of menu */
1221 for (i = start; i < lppop->nItems; i++, lpitem++)
1223 if ((helpPos == -1) && (lpitem->fType & MF_RIGHTJUSTIFY)) helpPos = i;
1225 (lpitem->fType & (MF_MENUBREAK | MF_MENUBARBREAK))) break;
1227 TRACE("calling MENU_CalcItemSize org=(%d, %d)\n", orgX, orgY );
1228 debug_print_menuitem (" item: ", lpitem, "");
1229 MENU_CalcItemSize( hdc, lpitem, hwndOwner, orgX, orgY, TRUE, lppop );
1231 if (lpitem->rect.right > lprect->right)
1233 if (i != start) break;
1234 else lpitem->rect.right = lprect->right;
1236 maxY = max( maxY, lpitem->rect.bottom );
1237 orgX = lpitem->rect.right;
1240 /* Finish the line (set all items to the largest height found) */
1241 while (start < i) lppop->items[start++].rect.bottom = maxY;
1244 lprect->bottom = maxY;
1245 lppop->Height = lprect->bottom - lprect->top;
1247 /* Flush right all items between the MF_RIGHTJUSTIFY and */
1248 /* the last item (if several lines, only move the last line) */
1249 lpitem = &lppop->items[lppop->nItems-1];
1250 orgY = lpitem->rect.top;
1251 orgX = lprect->right;
1252 for (i = lppop->nItems - 1; i >= helpPos; i--, lpitem--) {
1253 if ( (helpPos==-1) || (helpPos>i) )
1255 if (lpitem->rect.top != orgY) break; /* Other line */
1256 if (lpitem->rect.right >= orgX) break; /* Too far right already */
1257 lpitem->rect.left += orgX - lpitem->rect.right;
1258 lpitem->rect.right = orgX;
1259 orgX = lpitem->rect.left;
1264 /***********************************************************************
1265 * MENU_DrawScrollArrows
1267 * Draw scroll arrows.
1270 MENU_DrawScrollArrows(const POPUPMENU *lppop, HDC hdc)
1272 HDC hdcMem = CreateCompatibleDC(hdc);
1273 HBITMAP hOrigBitmap;
1274 UINT arrow_bitmap_width, arrow_bitmap_height;
1278 GetObjectW(get_down_arrow_bitmap(), sizeof(bmp), &bmp);
1279 arrow_bitmap_width = bmp.bmWidth;
1280 arrow_bitmap_height = bmp.bmHeight;
1283 if (lppop->nScrollPos)
1284 hOrigBitmap = SelectObject(hdcMem, get_up_arrow_bitmap());
1286 hOrigBitmap = SelectObject(hdcMem, get_up_arrow_inactive_bitmap());
1289 rect.right = lppop->Width;
1290 rect.bottom = arrow_bitmap_height;
1291 FillRect(hdc, &rect, GetSysColorBrush(COLOR_MENU));
1292 BitBlt(hdc, (lppop->Width - arrow_bitmap_width) / 2, 0,
1293 arrow_bitmap_width, arrow_bitmap_height, hdcMem, 0, 0, SRCCOPY);
1294 rect.top = lppop->Height - arrow_bitmap_height;
1295 rect.bottom = lppop->Height;
1296 FillRect(hdc, &rect, GetSysColorBrush(COLOR_MENU));
1297 if (lppop->nScrollPos < lppop->nTotalHeight - (MENU_GetMaxPopupHeight(lppop) - 2 * arrow_bitmap_height))
1298 SelectObject(hdcMem, get_down_arrow_bitmap());
1300 SelectObject(hdcMem, get_down_arrow_inactive_bitmap());
1301 BitBlt(hdc, (lppop->Width - arrow_bitmap_width) / 2,
1302 lppop->Height - arrow_bitmap_height,
1303 arrow_bitmap_width, arrow_bitmap_height, hdcMem, 0, 0, SRCCOPY);
1304 SelectObject(hdcMem, hOrigBitmap);
1309 /***********************************************************************
1312 * Draws the popup-menu arrow.
1314 static void draw_popup_arrow( HDC hdc, RECT rect, UINT arrow_bitmap_width,
1315 UINT arrow_bitmap_height)
1317 HDC hdcMem = CreateCompatibleDC( hdc );
1318 HBITMAP hOrigBitmap;
1320 hOrigBitmap = SelectObject( hdcMem, get_arrow_bitmap() );
1321 BitBlt( hdc, rect.right - arrow_bitmap_width - 1,
1322 (rect.top + rect.bottom - arrow_bitmap_height) / 2,
1323 arrow_bitmap_width, arrow_bitmap_height,
1324 hdcMem, 0, 0, SRCCOPY );
1325 SelectObject( hdcMem, hOrigBitmap );
1328 /***********************************************************************
1331 * Draw a single menu item.
1333 static void MENU_DrawMenuItem( HWND hwnd, HMENU hmenu, HWND hwndOwner, HDC hdc, MENUITEM *lpitem,
1334 UINT height, BOOL menuBar, UINT odaction )
1337 BOOL flat_menu = FALSE;
1339 UINT arrow_bitmap_width = 0, arrow_bitmap_height = 0;
1340 POPUPMENU *menu = MENU_GetMenu(hmenu);
1343 debug_print_menuitem("MENU_DrawMenuItem: ", lpitem, "");
1347 GetObjectW( get_arrow_bitmap(), sizeof(bmp), &bmp );
1348 arrow_bitmap_width = bmp.bmWidth;
1349 arrow_bitmap_height = bmp.bmHeight;
1352 if (lpitem->fType & MF_SYSMENU)
1354 if( !IsIconic(hwnd) )
1355 NC_DrawSysButton( hwnd, hdc, lpitem->fState & (MF_HILITE | MF_MOUSESELECT) );
1359 SystemParametersInfoW (SPI_GETFLATMENU, 0, &flat_menu, 0);
1360 bkgnd = (menuBar && flat_menu) ? COLOR_MENUBAR : COLOR_MENU;
1364 if (lpitem->fState & MF_HILITE)
1366 if(menuBar && !flat_menu) {
1367 SetTextColor(hdc, GetSysColor(COLOR_MENUTEXT));
1368 SetBkColor(hdc, GetSysColor(COLOR_MENU));
1370 if(lpitem->fState & MF_GRAYED)
1371 SetTextColor(hdc, GetSysColor(COLOR_GRAYTEXT));
1373 SetTextColor(hdc, GetSysColor(COLOR_HIGHLIGHTTEXT));
1374 SetBkColor(hdc, GetSysColor(COLOR_HIGHLIGHT));
1379 if (lpitem->fState & MF_GRAYED)
1380 SetTextColor( hdc, GetSysColor( COLOR_GRAYTEXT ) );
1382 SetTextColor( hdc, GetSysColor( COLOR_MENUTEXT ) );
1383 SetBkColor( hdc, GetSysColor( bkgnd ) );
1386 TRACE("rect=%s\n", wine_dbgstr_rect( &lpitem->rect));
1387 rect = lpitem->rect;
1388 MENU_AdjustMenuItemRect(MENU_GetMenu(hmenu), &rect);
1390 if (lpitem->fType & MF_OWNERDRAW)
1393 ** Experimentation under Windows reveals that an owner-drawn
1394 ** menu is given the rectangle which includes the space it requested
1395 ** in its response to WM_MEASUREITEM _plus_ width for a checkmark
1396 ** and a popup-menu arrow. This is the value of lpitem->rect.
1397 ** Windows will leave all drawing to the application except for
1398 ** the popup-menu arrow. Windows always draws that itself, after
1399 ** the menu owner has finished drawing.
1403 dis.CtlType = ODT_MENU;
1405 dis.itemID = lpitem->wID;
1406 dis.itemData = lpitem->dwItemData;
1408 if (lpitem->fState & MF_CHECKED) dis.itemState |= ODS_CHECKED;
1409 if (lpitem->fState & MF_GRAYED) dis.itemState |= ODS_GRAYED|ODS_DISABLED;
1410 if (lpitem->fState & MF_HILITE) dis.itemState |= ODS_SELECTED;
1411 dis.itemAction = odaction; /* ODA_DRAWENTIRE | ODA_SELECT | ODA_FOCUS; */
1412 dis.hwndItem = (HWND)hmenu;
1415 TRACE("Ownerdraw: owner=%p itemID=%d, itemState=%d, itemAction=%d, "
1416 "hwndItem=%p, hdc=%p, rcItem=%s\n", hwndOwner,
1417 dis.itemID, dis.itemState, dis.itemAction, dis.hwndItem,
1418 dis.hDC, wine_dbgstr_rect( &dis.rcItem));
1419 SendMessageW( hwndOwner, WM_DRAWITEM, 0, (LPARAM)&dis );
1420 /* Draw the popup-menu arrow */
1421 if (lpitem->fType & MF_POPUP)
1422 draw_popup_arrow( hdc, rect, arrow_bitmap_width,
1423 arrow_bitmap_height);
1427 if (menuBar && (lpitem->fType & MF_SEPARATOR)) return;
1429 if (lpitem->fState & MF_HILITE)
1433 InflateRect (&rect, -1, -1);
1434 FillRect(hdc, &rect, GetSysColorBrush(COLOR_MENUHILIGHT));
1435 InflateRect (&rect, 1, 1);
1436 FrameRect(hdc, &rect, GetSysColorBrush(COLOR_HIGHLIGHT));
1441 DrawEdge(hdc, &rect, BDR_SUNKENOUTER, BF_RECT);
1443 FillRect(hdc, &rect, GetSysColorBrush(COLOR_HIGHLIGHT));
1447 FillRect( hdc, &rect, GetSysColorBrush(bkgnd) );
1449 SetBkMode( hdc, TRANSPARENT );
1451 /* vertical separator */
1452 if (!menuBar && (lpitem->fType & MF_MENUBARBREAK))
1457 rc.left -= MENU_COL_SPACE / 2 + 1;
1459 rc.bottom = height - 3;
1462 oldPen = SelectObject( hdc, SYSCOLOR_GetPen(COLOR_BTNSHADOW) );
1463 MoveToEx( hdc, rc.left, rc.top, NULL );
1464 LineTo( hdc, rc.left, rc.bottom );
1465 SelectObject( hdc, oldPen );
1468 DrawEdge (hdc, &rc, EDGE_ETCHED, BF_LEFT);
1471 /* horizontal separator */
1472 if (lpitem->fType & MF_SEPARATOR)
1479 rc.top = ( rc.top + rc.bottom) / 2;
1482 oldPen = SelectObject( hdc, SYSCOLOR_GetPen(COLOR_BTNSHADOW) );
1483 MoveToEx( hdc, rc.left, rc.top, NULL );
1484 LineTo( hdc, rc.right, rc.top );
1485 SelectObject( hdc, oldPen );
1488 DrawEdge (hdc, &rc, EDGE_ETCHED, BF_TOP);
1492 /* helper lines for debugging */
1493 /* FrameRect(hdc, &rect, GetStockObject(BLACK_BRUSH));
1494 SelectObject( hdc, SYSCOLOR_GetPen(COLOR_WINDOWFRAME) );
1495 MoveToEx( hdc, rect.left, (rect.top + rect.bottom)/2, NULL );
1496 LineTo( hdc, rect.right, (rect.top + rect.bottom)/2 );
1499 if (lpitem->hbmpItem) {
1500 /* calculate the bitmap rectangle in coordinates relative
1501 * to the item rectangle */
1503 if( lpitem->hbmpItem == HBMMENU_CALLBACK)
1506 bmprc.left = lpitem->text ? menucharsize.cx : 0;
1509 if( !(menu->dwStyle & ( MNS_CHECKORBMP | MNS_NOCHECK)))
1510 bmprc.left += GetSystemMetrics( SM_CXMENUCHECK);
1512 bmprc.right = bmprc.left + lpitem->bmpsize.cx;
1513 if( menuBar && !(lpitem->hbmpItem == HBMMENU_CALLBACK))
1516 bmprc.top = (rect.bottom - rect.top -
1517 lpitem->bmpsize.cy) / 2;
1518 bmprc.bottom = bmprc.top + lpitem->bmpsize.cy;
1524 INT y = rect.top + rect.bottom;
1526 int checked = FALSE;
1527 UINT check_bitmap_width = GetSystemMetrics( SM_CXMENUCHECK );
1528 UINT check_bitmap_height = GetSystemMetrics( SM_CYMENUCHECK );
1529 /* Draw the check mark
1532 * Custom checkmark bitmaps are monochrome but not always 1bpp.
1534 if( !(menu->dwStyle & MNS_NOCHECK)) {
1535 bm = (lpitem->fState & MF_CHECKED) ? lpitem->hCheckBit :
1536 lpitem->hUnCheckBit;
1537 if (bm) /* we have a custom bitmap */
1539 HDC hdcMem = CreateCompatibleDC( hdc );
1541 SelectObject( hdcMem, bm );
1542 BitBlt( hdc, rc.left, (y - check_bitmap_height) / 2,
1543 check_bitmap_width, check_bitmap_height,
1544 hdcMem, 0, 0, SRCCOPY );
1548 else if (lpitem->fState & MF_CHECKED) /* standard bitmaps */
1551 HBITMAP bm = CreateBitmap( check_bitmap_width,
1552 check_bitmap_height, 1, 1, NULL );
1553 HDC hdcMem = CreateCompatibleDC( hdc );
1555 SelectObject( hdcMem, bm );
1556 SetRect( &r, 0, 0, check_bitmap_width, check_bitmap_height);
1557 DrawFrameControl( hdcMem, &r, DFC_MENU,
1558 (lpitem->fType & MFT_RADIOCHECK) ?
1559 DFCS_MENUBULLET : DFCS_MENUCHECK );
1560 BitBlt( hdc, rc.left, (y - r.bottom) / 2, r.right, r.bottom,
1561 hdcMem, 0, 0, SRCCOPY );
1567 if( lpitem->hbmpItem &&
1568 !( checked && (menu->dwStyle & MNS_CHECKORBMP))) {
1570 /* some applications make this assumption on the DC's origin */
1571 SetViewportOrgEx( hdc, rect.left, rect.top, &origorg);
1572 MENU_DrawBitmapItem(hdc, lpitem, &bmprc, hmenu, hwndOwner,
1574 SetViewportOrgEx( hdc, origorg.x, origorg.y, NULL);
1576 /* Draw the popup-menu arrow */
1577 if (lpitem->fType & MF_POPUP)
1578 draw_popup_arrow( hdc, rect, arrow_bitmap_width,
1579 arrow_bitmap_height);
1581 if( !(menu->dwStyle & MNS_NOCHECK))
1582 rect.left += check_bitmap_width;
1583 rect.right -= arrow_bitmap_width;
1585 else if( lpitem->hbmpItem)
1586 { /* Draw the bitmap */
1589 SetViewportOrgEx( hdc, rect.left, rect.top, &origorg);
1590 MENU_DrawBitmapItem( hdc, lpitem, &bmprc, hmenu, hwndOwner,
1592 SetViewportOrgEx( hdc, origorg.x, origorg.y, NULL);
1594 /* process text if present */
1600 UINT uFormat = (menuBar) ?
1601 DT_CENTER | DT_VCENTER | DT_SINGLELINE :
1602 DT_LEFT | DT_VCENTER | DT_SINGLELINE;
1604 if( !(menu->dwStyle & MNS_CHECKORBMP))
1605 rect.left += menu->maxBmpSize.cx;
1607 if ( lpitem->fState & MFS_DEFAULT )
1609 hfontOld = SelectObject( hdc, get_menu_font(TRUE) );
1613 if( lpitem->hbmpItem)
1614 rect.left += lpitem->bmpsize.cx;
1615 if( !(lpitem->hbmpItem == HBMMENU_CALLBACK))
1616 rect.left += menucharsize.cx;
1617 rect.right -= menucharsize.cx;
1620 for (i = 0; lpitem->text[i]; i++)
1621 if ((lpitem->text[i] == '\t') || (lpitem->text[i] == '\b'))
1624 if(lpitem->fState & MF_GRAYED)
1626 if (!(lpitem->fState & MF_HILITE) )
1628 ++rect.left; ++rect.top; ++rect.right; ++rect.bottom;
1629 SetTextColor(hdc, RGB(0xff, 0xff, 0xff));
1630 DrawTextW( hdc, lpitem->text, i, &rect, uFormat );
1631 --rect.left; --rect.top; --rect.right; --rect.bottom;
1633 SetTextColor(hdc, RGB(0x80, 0x80, 0x80));
1636 DrawTextW( hdc, lpitem->text, i, &rect, uFormat);
1638 /* paint the shortcut text */
1639 if (!menuBar && lpitem->text[i]) /* There's a tab or flush-right char */
1641 if (lpitem->text[i] == '\t')
1643 rect.left = lpitem->xTab;
1644 uFormat = DT_LEFT | DT_VCENTER | DT_SINGLELINE;
1648 rect.right = lpitem->xTab;
1649 uFormat = DT_RIGHT | DT_VCENTER | DT_SINGLELINE;
1652 if(lpitem->fState & MF_GRAYED)
1654 if (!(lpitem->fState & MF_HILITE) )
1656 ++rect.left; ++rect.top; ++rect.right; ++rect.bottom;
1657 SetTextColor(hdc, RGB(0xff, 0xff, 0xff));
1658 DrawTextW( hdc, lpitem->text + i + 1, -1, &rect, uFormat );
1659 --rect.left; --rect.top; --rect.right; --rect.bottom;
1661 SetTextColor(hdc, RGB(0x80, 0x80, 0x80));
1663 DrawTextW( hdc, lpitem->text + i + 1, -1, &rect, uFormat );
1667 SelectObject (hdc, hfontOld);
1672 /***********************************************************************
1673 * MENU_DrawPopupMenu
1675 * Paint a popup menu.
1677 static void MENU_DrawPopupMenu( HWND hwnd, HDC hdc, HMENU hmenu )
1679 HBRUSH hPrevBrush = 0;
1682 TRACE("wnd=%p dc=%p menu=%p\n", hwnd, hdc, hmenu);
1684 GetClientRect( hwnd, &rect );
1686 if((hPrevBrush = SelectObject( hdc, GetSysColorBrush(COLOR_MENU) ))
1687 && (SelectObject( hdc, get_menu_font(FALSE))))
1691 Rectangle( hdc, rect.left, rect.top, rect.right, rect.bottom );
1693 hPrevPen = SelectObject( hdc, GetStockObject( NULL_PEN ) );
1697 BOOL flat_menu = FALSE;
1699 SystemParametersInfoW (SPI_GETFLATMENU, 0, &flat_menu, 0);
1701 FrameRect(hdc, &rect, GetSysColorBrush(COLOR_BTNSHADOW));
1703 DrawEdge (hdc, &rect, EDGE_RAISED, BF_RECT);
1705 if( (menu = MENU_GetMenu( hmenu )))
1707 /* draw menu items */
1714 for( u = menu->nItems; u > 0; u--, item++)
1715 MENU_DrawMenuItem( hwnd, hmenu, menu->hwndOwner, hdc,
1716 item, menu->Height, FALSE, ODA_DRAWENTIRE );
1718 /* draw scroll arrows */
1719 if (menu->bScrolling)
1720 MENU_DrawScrollArrows(menu, hdc);
1724 SelectObject( hdc, hPrevBrush );
1729 /***********************************************************************
1732 * Paint a menu bar. Returns the height of the menu bar.
1733 * called from [windows/nonclient.c]
1735 UINT MENU_DrawMenuBar( HDC hDC, LPRECT lprect, HWND hwnd,
1740 HMENU hMenu = GetMenu(hwnd);
1742 lppop = MENU_GetMenu( hMenu );
1743 if (lppop == NULL || lprect == NULL)
1745 return GetSystemMetrics(SM_CYMENU);
1750 hfontOld = SelectObject( hDC, get_menu_font(FALSE));
1752 if (lppop->Height == 0)
1753 MENU_MenuBarCalcSize(hDC, lprect, lppop, hwnd);
1755 lprect->bottom = lprect->top + lppop->Height;
1757 if (hfontOld) SelectObject( hDC, hfontOld);
1758 return lppop->Height;
1761 return DrawMenuBarTemp(hwnd, hDC, lprect, hMenu, NULL);
1765 /***********************************************************************
1768 * Display a popup menu.
1770 static BOOL MENU_ShowPopup( HWND hwndOwner, HMENU hmenu, UINT id,
1771 INT x, INT y, INT xanchor, INT yanchor )
1779 TRACE("owner=%p hmenu=%p id=0x%04x x=0x%04x y=0x%04x xa=0x%04x ya=0x%04x\n",
1780 hwndOwner, hmenu, id, x, y, xanchor, yanchor);
1782 if (!(menu = MENU_GetMenu( hmenu ))) return FALSE;
1783 if (menu->FocusedItem != NO_SELECTED_ITEM)
1785 menu->items[menu->FocusedItem].fState &= ~(MF_HILITE|MF_MOUSESELECT);
1786 menu->FocusedItem = NO_SELECTED_ITEM;
1789 /* store the owner for DrawItem */
1790 menu->hwndOwner = hwndOwner;
1792 menu->nScrollPos = 0;
1793 MENU_PopupMenuCalcSize( menu );
1795 /* adjust popup menu pos so that it fits within the desktop */
1797 width = menu->Width + GetSystemMetrics(SM_CXBORDER);
1798 height = menu->Height + GetSystemMetrics(SM_CYBORDER);
1800 /* FIXME: should use item rect */
1803 monitor = MonitorFromPoint( pt, MONITOR_DEFAULTTONEAREST );
1804 info.cbSize = sizeof(info);
1805 GetMonitorInfoW( monitor, &info );
1806 if( x + width > info.rcWork.right)
1808 if( xanchor && x >= width - xanchor )
1809 x -= width - xanchor;
1811 if( x + width > info.rcWork.right)
1812 x = info.rcWork.right - width;
1814 if( x < info.rcWork.left ) x = info.rcWork.left;
1816 if( y + height > info.rcWork.bottom)
1818 if( yanchor && y >= height + yanchor )
1819 y -= height + yanchor;
1821 if( y + height > info.rcWork.bottom)
1822 y = info.rcWork.bottom - height;
1824 if( y < info.rcWork.top ) y = info.rcWork.top;
1826 /* NOTE: In Windows, top menu popup is not owned. */
1827 menu->hWnd = CreateWindowExW( 0, (LPCWSTR)POPUPMENU_CLASS_ATOM, NULL,
1828 WS_POPUP, x, y, width, height,
1829 hwndOwner, 0, (HINSTANCE)GetWindowLongPtrW(hwndOwner, GWLP_HINSTANCE),
1831 if( !menu->hWnd ) return FALSE;
1832 if (!top_popup) top_popup = menu->hWnd;
1834 /* Display the window */
1836 SetWindowPos( menu->hWnd, HWND_TOPMOST, 0, 0, 0, 0,
1837 SWP_SHOWWINDOW | SWP_NOSIZE | SWP_NOMOVE | SWP_NOACTIVATE );
1838 UpdateWindow( menu->hWnd );
1843 /***********************************************************************
1844 * MENU_EnsureMenuItemVisible
1847 MENU_EnsureMenuItemVisible(LPPOPUPMENU lppop, UINT wIndex, HDC hdc)
1849 if (lppop->bScrolling)
1851 MENUITEM *item = &lppop->items[wIndex];
1852 UINT nMaxHeight = MENU_GetMaxPopupHeight(lppop);
1853 UINT nOldPos = lppop->nScrollPos;
1855 UINT arrow_bitmap_height;
1858 GetClientRect(lppop->hWnd, &rc);
1860 GetObjectW(get_down_arrow_bitmap(), sizeof(bmp), &bmp);
1861 arrow_bitmap_height = bmp.bmHeight;
1863 rc.top += arrow_bitmap_height;
1864 rc.bottom -= arrow_bitmap_height + MENU_BOTTOM_MARGIN;
1866 nMaxHeight -= GetSystemMetrics(SM_CYBORDER) + 2 * arrow_bitmap_height;
1867 if (item->rect.bottom > lppop->nScrollPos + nMaxHeight)
1870 lppop->nScrollPos = item->rect.bottom - nMaxHeight;
1871 ScrollWindow(lppop->hWnd, 0, nOldPos - lppop->nScrollPos, &rc, &rc);
1872 MENU_DrawScrollArrows(lppop, hdc);
1874 else if (item->rect.top - MENU_TOP_MARGIN < lppop->nScrollPos)
1876 lppop->nScrollPos = item->rect.top - MENU_TOP_MARGIN;
1877 ScrollWindow(lppop->hWnd, 0, nOldPos - lppop->nScrollPos, &rc, &rc);
1878 MENU_DrawScrollArrows(lppop, hdc);
1884 /***********************************************************************
1887 static void MENU_SelectItem( HWND hwndOwner, HMENU hmenu, UINT wIndex,
1888 BOOL sendMenuSelect, HMENU topmenu )
1893 TRACE("owner=%p menu=%p index=0x%04x select=0x%04x\n", hwndOwner, hmenu, wIndex, sendMenuSelect);
1895 lppop = MENU_GetMenu( hmenu );
1896 if ((!lppop) || (!lppop->nItems) || (!lppop->hWnd)) return;
1898 if (lppop->FocusedItem == wIndex) return;
1899 if (lppop->wFlags & MF_POPUP) hdc = GetDC( lppop->hWnd );
1900 else hdc = GetDCEx( lppop->hWnd, 0, DCX_CACHE | DCX_WINDOW);
1901 if (!top_popup) top_popup = lppop->hWnd;
1903 SelectObject( hdc, get_menu_font(FALSE));
1905 /* Clear previous highlighted item */
1906 if (lppop->FocusedItem != NO_SELECTED_ITEM)
1908 lppop->items[lppop->FocusedItem].fState &= ~(MF_HILITE|MF_MOUSESELECT);
1909 MENU_DrawMenuItem(lppop->hWnd, hmenu, hwndOwner, hdc,&lppop->items[lppop->FocusedItem],
1910 lppop->Height, !(lppop->wFlags & MF_POPUP),
1914 /* Highlight new item (if any) */
1915 lppop->FocusedItem = wIndex;
1916 if (lppop->FocusedItem != NO_SELECTED_ITEM)
1918 if(!(lppop->items[wIndex].fType & MF_SEPARATOR)) {
1919 lppop->items[wIndex].fState |= MF_HILITE;
1920 MENU_EnsureMenuItemVisible(lppop, wIndex, hdc);
1921 MENU_DrawMenuItem( lppop->hWnd, hmenu, hwndOwner, hdc,
1922 &lppop->items[wIndex], lppop->Height,
1923 !(lppop->wFlags & MF_POPUP), ODA_SELECT );
1927 MENUITEM *ip = &lppop->items[lppop->FocusedItem];
1928 SendMessageW( hwndOwner, WM_MENUSELECT,
1929 MAKELONG(ip->fType & MF_POPUP ? wIndex: ip->wID,
1930 ip->fType | ip->fState |
1931 (lppop->wFlags & MF_SYSMENU)), (LPARAM)hmenu);
1934 else if (sendMenuSelect) {
1937 if((pos=MENU_FindSubMenu(&topmenu, hmenu))!=NO_SELECTED_ITEM){
1938 POPUPMENU *ptm = MENU_GetMenu( topmenu );
1939 MENUITEM *ip = &ptm->items[pos];
1940 SendMessageW( hwndOwner, WM_MENUSELECT, MAKELONG(pos,
1941 ip->fType | ip->fState |
1942 (ptm->wFlags & MF_SYSMENU)), (LPARAM)topmenu);
1946 ReleaseDC( lppop->hWnd, hdc );
1950 /***********************************************************************
1951 * MENU_MoveSelection
1953 * Moves currently selected item according to the offset parameter.
1954 * If there is no selection then it should select the last item if
1955 * offset is ITEM_PREV or the first item if offset is ITEM_NEXT.
1957 static void MENU_MoveSelection( HWND hwndOwner, HMENU hmenu, INT offset )
1962 TRACE("hwnd=%p hmenu=%p off=0x%04x\n", hwndOwner, hmenu, offset);
1964 menu = MENU_GetMenu( hmenu );
1965 if ((!menu) || (!menu->items)) return;
1967 if ( menu->FocusedItem != NO_SELECTED_ITEM )
1969 if( menu->nItems == 1 ) return; else
1970 for (i = menu->FocusedItem + offset ; i >= 0 && i < menu->nItems
1972 if (!(menu->items[i].fType & MF_SEPARATOR))
1974 MENU_SelectItem( hwndOwner, hmenu, i, TRUE, 0 );
1979 for ( i = (offset > 0) ? 0 : menu->nItems - 1;
1980 i >= 0 && i < menu->nItems ; i += offset)
1981 if (!(menu->items[i].fType & MF_SEPARATOR))
1983 MENU_SelectItem( hwndOwner, hmenu, i, TRUE, 0 );
1989 /**********************************************************************
1992 * Set an item's flags, id and text ptr. Called by InsertMenu() and
1995 static BOOL MENU_SetItemData( MENUITEM *item, UINT flags, UINT_PTR id,
1998 debug_print_menuitem("MENU_SetItemData from: ", item, "");
1999 TRACE("flags=%x str=%p\n", flags, str);
2001 if (IS_STRING_ITEM(flags))
2003 LPWSTR prevText = item->text;
2006 flags |= MF_SEPARATOR;
2012 /* Item beginning with a backspace is a help item */
2018 if (!(text = HeapAlloc( GetProcessHeap(), 0, (strlenW(str)+1) * sizeof(WCHAR) )))
2020 strcpyW( text, str );
2023 item->hbmpItem = NULL;
2024 HeapFree( GetProcessHeap(), 0, prevText );
2026 else if(( flags & MFT_BITMAP)) {
2027 item->hbmpItem = HBITMAP_32(LOWORD(str));
2028 /* setting bitmap clears text */
2029 HeapFree( GetProcessHeap(), 0, item->text );
2033 if (flags & MF_SEPARATOR) flags |= MF_GRAYED | MF_DISABLED;
2035 if (flags & MF_OWNERDRAW)
2036 item->dwItemData = (DWORD_PTR)str;
2038 item->dwItemData = 0;
2040 if ((item->fType & MF_POPUP) && (flags & MF_POPUP) && (item->hSubMenu != (HMENU)id) )
2041 DestroyMenu( item->hSubMenu ); /* ModifyMenu() spec */
2043 if (flags & MF_POPUP)
2045 POPUPMENU *menu = MENU_GetMenu((HMENU)id);
2046 if (menu) menu->wFlags |= MF_POPUP;
2058 if (flags & MF_POPUP) item->hSubMenu = (HMENU)id;
2060 if ((item->fType & MF_POPUP) && !(flags & MF_POPUP) )
2061 flags |= MF_POPUP; /* keep popup */
2063 item->fType = flags & TYPE_MASK;
2064 /* MFS_DEFAULT is not accepted. MF_HILITE is not listed as a valid flag
2065 for ModifyMenu, but Windows accepts it */
2066 item->fState = flags & MENUITEMINFO_STATE_MASK & ~MFS_DEFAULT;
2068 /* Don't call SetRectEmpty here! */
2070 debug_print_menuitem("MENU_SetItemData to : ", item, "");
2075 /**********************************************************************
2078 * Insert (allocate) a new item into a menu.
2080 static MENUITEM *MENU_InsertItem( HMENU hMenu, UINT pos, UINT flags )
2085 if (!(menu = MENU_GetMenu(hMenu)))
2088 /* Find where to insert new item */
2090 if (flags & MF_BYPOSITION) {
2091 if (pos > menu->nItems)
2094 if (!MENU_FindItem( &hMenu, &pos, flags ))
2097 if (!(menu = MENU_GetMenu( hMenu )))
2102 /* Make sure that MDI system buttons stay on the right side.
2103 * Note: XP treats only bitmap handles 1 - 6 as "magic" ones
2104 * regardless of their id.
2106 while (pos > 0 && (menu->items[pos - 1].fType & MFT_BITMAP) &&
2107 (INT_PTR)menu->items[pos - 1].hbmpItem >= (INT_PTR)HBMMENU_SYSTEM &&
2108 (INT_PTR)menu->items[pos - 1].hbmpItem <= (INT_PTR)HBMMENU_MBAR_CLOSE_D)
2111 TRACE("inserting at %u by pos %u\n", pos, flags & MF_BYPOSITION);
2113 /* Create new items array */
2115 newItems = HeapAlloc( GetProcessHeap(), 0, sizeof(MENUITEM) * (menu->nItems+1) );
2118 WARN("allocation failed\n" );
2121 if (menu->nItems > 0)
2123 /* Copy the old array into the new one */
2124 if (pos > 0) memcpy( newItems, menu->items, pos * sizeof(MENUITEM) );
2125 if (pos < menu->nItems) memcpy( &newItems[pos+1], &menu->items[pos],
2126 (menu->nItems-pos)*sizeof(MENUITEM) );
2127 HeapFree( GetProcessHeap(), 0, menu->items );
2129 menu->items = newItems;
2131 memset( &newItems[pos], 0, sizeof(*newItems) );
2132 menu->Height = 0; /* force size recalculate */
2133 return &newItems[pos];
2137 /**********************************************************************
2138 * MENU_ParseResource
2140 * Parse a standard menu resource and add items to the menu.
2141 * Return a pointer to the end of the resource.
2143 * NOTE: flags is equivalent to the mtOption field
2145 static LPCSTR MENU_ParseResource( LPCSTR res, HMENU hMenu, BOOL unicode )
2153 flags = GET_WORD(res);
2154 end_flag = flags & MF_END;
2155 /* Remove MF_END because it has the same value as MF_HILITE */
2157 res += sizeof(WORD);
2158 if (!(flags & MF_POPUP))
2161 res += sizeof(WORD);
2164 if (!unicode) res += strlen(str) + 1;
2165 else res += (strlenW((LPCWSTR)str) + 1) * sizeof(WCHAR);
2166 if (flags & MF_POPUP)
2168 HMENU hSubMenu = CreatePopupMenu();
2169 if (!hSubMenu) return NULL;
2170 if (!(res = MENU_ParseResource( res, hSubMenu, unicode )))
2172 if (!unicode) AppendMenuA( hMenu, flags, (UINT_PTR)hSubMenu, str );
2173 else AppendMenuW( hMenu, flags, (UINT_PTR)hSubMenu, (LPCWSTR)str );
2175 else /* Not a popup */
2177 if (!unicode) AppendMenuA( hMenu, flags, id, *str ? str : NULL );
2178 else AppendMenuW( hMenu, flags, id,
2179 *(LPCWSTR)str ? (LPCWSTR)str : NULL );
2181 } while (!end_flag);
2186 /**********************************************************************
2187 * MENUEX_ParseResource
2189 * Parse an extended menu resource and add items to the menu.
2190 * Return a pointer to the end of the resource.
2192 static LPCSTR MENUEX_ParseResource( LPCSTR res, HMENU hMenu)
2198 mii.cbSize = sizeof(mii);
2199 mii.fMask = MIIM_STATE | MIIM_ID | MIIM_TYPE;
2200 mii.fType = GET_DWORD(res);
2201 res += sizeof(DWORD);
2202 mii.fState = GET_DWORD(res);
2203 res += sizeof(DWORD);
2204 mii.wID = GET_DWORD(res);
2205 res += sizeof(DWORD);
2206 resinfo = GET_WORD(res); /* FIXME: for 16-bit apps this is a byte. */
2207 res += sizeof(WORD);
2208 /* Align the text on a word boundary. */
2209 res += (~((UINT_PTR)res - 1)) & 1;
2210 mii.dwTypeData = (LPWSTR) res;
2211 res += (1 + strlenW(mii.dwTypeData)) * sizeof(WCHAR);
2212 /* Align the following fields on a dword boundary. */
2213 res += (~((UINT_PTR)res - 1)) & 3;
2215 TRACE("Menu item: [%08x,%08x,%04x,%04x,%s]\n",
2216 mii.fType, mii.fState, mii.wID, resinfo, debugstr_w(mii.dwTypeData));
2218 if (resinfo & 1) { /* Pop-up? */
2219 /* DWORD helpid = GET_DWORD(res); FIXME: use this. */
2220 res += sizeof(DWORD);
2221 mii.hSubMenu = CreatePopupMenu();
2224 if (!(res = MENUEX_ParseResource(res, mii.hSubMenu))) {
2225 DestroyMenu(mii.hSubMenu);
2228 mii.fMask |= MIIM_SUBMENU;
2229 mii.fType |= MF_POPUP;
2231 else if(!*mii.dwTypeData && !(mii.fType & MF_SEPARATOR))
2233 WARN("Converting NULL menu item %04x, type %04x to SEPARATOR\n",
2234 mii.wID, mii.fType);
2235 mii.fType |= MF_SEPARATOR;
2237 InsertMenuItemW(hMenu, -1, MF_BYPOSITION, &mii);
2238 } while (!(resinfo & MF_END));
2243 /***********************************************************************
2246 * Return the handle of the selected sub-popup menu (if any).
2248 static HMENU MENU_GetSubPopup( HMENU hmenu )
2253 menu = MENU_GetMenu( hmenu );
2255 if ((!menu) || (menu->FocusedItem == NO_SELECTED_ITEM)) return 0;
2257 item = &menu->items[menu->FocusedItem];
2258 if ((item->fType & MF_POPUP) && (item->fState & MF_MOUSESELECT))
2259 return item->hSubMenu;
2264 /***********************************************************************
2265 * MENU_HideSubPopups
2267 * Hide the sub-popup menus of this menu.
2269 static void MENU_HideSubPopups( HWND hwndOwner, HMENU hmenu,
2270 BOOL sendMenuSelect, UINT wFlags )
2272 POPUPMENU *menu = MENU_GetMenu( hmenu );
2274 TRACE("owner=%p hmenu=%p 0x%04x\n", hwndOwner, hmenu, sendMenuSelect);
2276 if (menu && top_popup)
2282 if (menu->FocusedItem != NO_SELECTED_ITEM)
2284 item = &menu->items[menu->FocusedItem];
2285 if (!(item->fType & MF_POPUP) ||
2286 !(item->fState & MF_MOUSESELECT)) return;
2287 item->fState &= ~MF_MOUSESELECT;
2288 hsubmenu = item->hSubMenu;
2291 submenu = MENU_GetMenu( hsubmenu );
2292 MENU_HideSubPopups( hwndOwner, hsubmenu, FALSE, wFlags );
2293 MENU_SelectItem( hwndOwner, hsubmenu, NO_SELECTED_ITEM, sendMenuSelect, 0 );
2294 DestroyWindow( submenu->hWnd );
2297 if (!(wFlags & TPM_NONOTIFY))
2298 SendMessageW( hwndOwner, WM_UNINITMENUPOPUP, (WPARAM)hsubmenu,
2299 MAKELPARAM(0, IS_SYSTEM_MENU(submenu)) );
2304 /***********************************************************************
2307 * Display the sub-menu of the selected item of this menu.
2308 * Return the handle of the submenu, or hmenu if no submenu to display.
2310 static HMENU MENU_ShowSubPopup( HWND hwndOwner, HMENU hmenu,
2311 BOOL selectFirst, UINT wFlags )
2318 TRACE("owner=%p hmenu=%p 0x%04x\n", hwndOwner, hmenu, selectFirst);
2320 if (!(menu = MENU_GetMenu( hmenu ))) return hmenu;
2322 if (menu->FocusedItem == NO_SELECTED_ITEM) return hmenu;
2324 item = &menu->items[menu->FocusedItem];
2325 if (!(item->fType & MF_POPUP) || (item->fState & (MF_GRAYED | MF_DISABLED)))
2328 /* message must be sent before using item,
2329 because nearly everything may be changed by the application ! */
2331 /* Send WM_INITMENUPOPUP message only if TPM_NONOTIFY flag is not specified */
2332 if (!(wFlags & TPM_NONOTIFY))
2333 SendMessageW( hwndOwner, WM_INITMENUPOPUP, (WPARAM)item->hSubMenu,
2334 MAKELONG( menu->FocusedItem, IS_SYSTEM_MENU(menu) ));
2336 item = &menu->items[menu->FocusedItem];
2339 /* correct item if modified as a reaction to WM_INITMENUPOPUP message */
2340 if (!(item->fState & MF_HILITE))
2342 if (menu->wFlags & MF_POPUP) hdc = GetDC( menu->hWnd );
2343 else hdc = GetDCEx( menu->hWnd, 0, DCX_CACHE | DCX_WINDOW);
2345 SelectObject( hdc, get_menu_font(FALSE));
2347 item->fState |= MF_HILITE;
2348 MENU_DrawMenuItem( menu->hWnd, hmenu, hwndOwner, hdc, item, menu->Height, !(menu->wFlags & MF_POPUP), ODA_DRAWENTIRE );
2349 ReleaseDC( menu->hWnd, hdc );
2351 if (!item->rect.top && !item->rect.left && !item->rect.bottom && !item->rect.right)
2354 item->fState |= MF_MOUSESELECT;
2356 if (IS_SYSTEM_MENU(menu))
2358 MENU_InitSysMenuPopup(item->hSubMenu,
2359 GetWindowLongW( menu->hWnd, GWL_STYLE ),
2360 GetClassLongW( menu->hWnd, GCL_STYLE));
2362 NC_GetSysPopupPos( menu->hWnd, &rect );
2363 rect.top = rect.bottom;
2364 rect.right = GetSystemMetrics(SM_CXSIZE);
2365 rect.bottom = GetSystemMetrics(SM_CYSIZE);
2369 GetWindowRect( menu->hWnd, &rect );
2370 if (menu->wFlags & MF_POPUP)
2372 RECT rc = item->rect;
2374 MENU_AdjustMenuItemRect(menu, &rc);
2376 /* The first item in the popup menu has to be at the
2377 same y position as the focused menu item */
2378 rect.left += rc.right - GetSystemMetrics(SM_CXBORDER);
2379 rect.top += rc.top - MENU_TOP_MARGIN;
2380 rect.right = rc.left - rc.right + GetSystemMetrics(SM_CXBORDER);
2381 rect.bottom = rc.top - rc.bottom - MENU_TOP_MARGIN
2382 - MENU_BOTTOM_MARGIN - GetSystemMetrics(SM_CYBORDER);
2386 rect.left += item->rect.left;
2387 rect.top += item->rect.bottom;
2388 rect.right = item->rect.right - item->rect.left;
2389 rect.bottom = item->rect.bottom - item->rect.top;
2393 MENU_ShowPopup( hwndOwner, item->hSubMenu, menu->FocusedItem,
2394 rect.left, rect.top, rect.right, rect.bottom );
2396 MENU_MoveSelection( hwndOwner, item->hSubMenu, ITEM_NEXT );
2397 return item->hSubMenu;
2402 /**********************************************************************
2405 HWND MENU_IsMenuActive(void)
2410 /***********************************************************************
2413 * Walks menu chain trying to find a menu pt maps to.
2415 static HMENU MENU_PtMenu( HMENU hMenu, POINT pt )
2417 POPUPMENU *menu = MENU_GetMenu( hMenu );
2418 UINT item = menu->FocusedItem;
2421 /* try subpopup first (if any) */
2422 ret = (item != NO_SELECTED_ITEM &&
2423 (menu->items[item].fType & MF_POPUP) &&
2424 (menu->items[item].fState & MF_MOUSESELECT))
2425 ? MENU_PtMenu(menu->items[item].hSubMenu, pt) : 0;
2427 if (!ret) /* check the current window (avoiding WM_HITTEST) */
2429 INT ht = NC_HandleNCHitTest( menu->hWnd, pt );
2430 if( menu->wFlags & MF_POPUP )
2432 if (ht != HTNOWHERE && ht != HTERROR) ret = hMenu;
2434 else if (ht == HTSYSMENU)
2435 ret = get_win_sys_menu( menu->hWnd );
2436 else if (ht == HTMENU)
2437 ret = GetMenu( menu->hWnd );
2442 /***********************************************************************
2443 * MENU_ExecFocusedItem
2445 * Execute a menu item (for instance when user pressed Enter).
2446 * Return the wID of the executed item. Otherwise, -1 indicating
2447 * that no menu item was executed, -2 if a popup is shown;
2448 * Have to receive the flags for the TrackPopupMenu options to avoid
2449 * sending unwanted message.
2452 static INT MENU_ExecFocusedItem( MTRACKER* pmt, HMENU hMenu, UINT wFlags )
2455 POPUPMENU *menu = MENU_GetMenu( hMenu );
2457 TRACE("%p hmenu=%p\n", pmt, hMenu);
2459 if (!menu || !menu->nItems ||
2460 (menu->FocusedItem == NO_SELECTED_ITEM)) return -1;
2462 item = &menu->items[menu->FocusedItem];
2464 TRACE("hMenu %p wID %08lx hSubMenu %p fType %04x\n", hMenu, item->wID, item->hSubMenu, item->fType);
2466 if (!(item->fType & MF_POPUP))
2468 if (!(item->fState & (MF_GRAYED | MF_DISABLED)) && !(item->fType & MF_SEPARATOR))
2470 /* If TPM_RETURNCMD is set you return the id, but
2471 do not send a message to the owner */
2472 if(!(wFlags & TPM_RETURNCMD))
2474 if( menu->wFlags & MF_SYSMENU )
2475 PostMessageW( pmt->hOwnerWnd, WM_SYSCOMMAND, item->wID,
2476 MAKELPARAM((INT16)pmt->pt.x, (INT16)pmt->pt.y) );
2479 POPUPMENU *topmenu = MENU_GetMenu( pmt->hTopMenu );
2480 DWORD dwStyle = menu->dwStyle | (topmenu ? topmenu->dwStyle : 0);
2482 if (dwStyle & MNS_NOTIFYBYPOS)
2483 PostMessageW( pmt->hOwnerWnd, WM_MENUCOMMAND, menu->FocusedItem,
2486 PostMessageW( pmt->hOwnerWnd, WM_COMMAND, item->wID, 0 );
2494 pmt->hCurrentMenu = MENU_ShowSubPopup(pmt->hOwnerWnd, hMenu, TRUE, wFlags);
2501 /***********************************************************************
2502 * MENU_SwitchTracking
2504 * Helper function for menu navigation routines.
2506 static void MENU_SwitchTracking( MTRACKER* pmt, HMENU hPtMenu, UINT id, UINT wFlags )
2508 POPUPMENU *ptmenu = MENU_GetMenu( hPtMenu );
2509 POPUPMENU *topmenu = MENU_GetMenu( pmt->hTopMenu );
2511 TRACE("%p hmenu=%p 0x%04x\n", pmt, hPtMenu, id);
2513 if( pmt->hTopMenu != hPtMenu &&
2514 !((ptmenu->wFlags | topmenu->wFlags) & MF_POPUP) )
2516 /* both are top level menus (system and menu-bar) */
2517 MENU_HideSubPopups( pmt->hOwnerWnd, pmt->hTopMenu, FALSE, wFlags );
2518 MENU_SelectItem( pmt->hOwnerWnd, pmt->hTopMenu, NO_SELECTED_ITEM, FALSE, 0 );
2519 pmt->hTopMenu = hPtMenu;
2521 else MENU_HideSubPopups( pmt->hOwnerWnd, hPtMenu, FALSE, wFlags );
2522 MENU_SelectItem( pmt->hOwnerWnd, hPtMenu, id, TRUE, 0 );
2526 /***********************************************************************
2529 * Return TRUE if we can go on with menu tracking.
2531 static BOOL MENU_ButtonDown( MTRACKER* pmt, HMENU hPtMenu, UINT wFlags )
2533 TRACE("%p hPtMenu=%p\n", pmt, hPtMenu);
2538 POPUPMENU *ptmenu = MENU_GetMenu( hPtMenu );
2541 if( IS_SYSTEM_MENU(ptmenu) )
2542 item = ptmenu->items;
2544 item = MENU_FindItemByCoords( ptmenu, pmt->pt, &id );
2548 if( ptmenu->FocusedItem != id )
2549 MENU_SwitchTracking( pmt, hPtMenu, id, wFlags );
2551 /* If the popup menu is not already "popped" */
2552 if(!(item->fState & MF_MOUSESELECT ))
2554 pmt->hCurrentMenu = MENU_ShowSubPopup( pmt->hOwnerWnd, hPtMenu, FALSE, wFlags );
2559 /* Else the click was on the menu bar, finish the tracking */
2564 /***********************************************************************
2567 * Return the value of MENU_ExecFocusedItem if
2568 * the selected item was not a popup. Else open the popup.
2569 * A -1 return value indicates that we go on with menu tracking.
2572 static INT MENU_ButtonUp( MTRACKER* pmt, HMENU hPtMenu, UINT wFlags)
2574 TRACE("%p hmenu=%p\n", pmt, hPtMenu);
2579 POPUPMENU *ptmenu = MENU_GetMenu( hPtMenu );
2582 if( IS_SYSTEM_MENU(ptmenu) )
2583 item = ptmenu->items;
2585 item = MENU_FindItemByCoords( ptmenu, pmt->pt, &id );
2587 if( item && (ptmenu->FocusedItem == id ))
2589 debug_print_menuitem ("FocusedItem: ", item, "");
2591 if( !(item->fType & MF_POPUP) )
2593 INT executedMenuId = MENU_ExecFocusedItem( pmt, hPtMenu, wFlags);
2594 if (executedMenuId == -1 || executedMenuId == -2) return -1;
2595 return executedMenuId;
2598 /* If we are dealing with the top-level menu */
2599 /* and this is a click on an already "popped" item: */
2600 /* Stop the menu tracking and close the opened submenus */
2601 if((pmt->hTopMenu == hPtMenu) && ptmenu->bTimeToHide)
2604 ptmenu->bTimeToHide = TRUE;
2610 /***********************************************************************
2613 * Return TRUE if we can go on with menu tracking.
2615 static BOOL MENU_MouseMove( MTRACKER* pmt, HMENU hPtMenu, UINT wFlags )
2617 UINT id = NO_SELECTED_ITEM;
2618 POPUPMENU *ptmenu = NULL;
2622 ptmenu = MENU_GetMenu( hPtMenu );
2623 if( IS_SYSTEM_MENU(ptmenu) )
2626 MENU_FindItemByCoords( ptmenu, pmt->pt, &id );
2629 if( id == NO_SELECTED_ITEM )
2631 MENU_SelectItem( pmt->hOwnerWnd, pmt->hCurrentMenu,
2632 NO_SELECTED_ITEM, TRUE, pmt->hTopMenu);
2635 else if( ptmenu->FocusedItem != id )
2637 MENU_SwitchTracking( pmt, hPtMenu, id, wFlags );
2638 pmt->hCurrentMenu = MENU_ShowSubPopup(pmt->hOwnerWnd, hPtMenu, FALSE, wFlags);
2644 /***********************************************************************
2647 * NOTE: WM_NEXTMENU documented in Win32 is a bit different.
2649 static LRESULT MENU_DoNextMenu( MTRACKER* pmt, UINT vk, UINT wFlags )
2651 POPUPMENU *menu = MENU_GetMenu( pmt->hTopMenu );
2654 /* When skipping left, we need to do something special after the
2656 if (vk == VK_LEFT && menu->FocusedItem == 0)
2660 /* When skipping right, for the non-system menu, we need to
2661 handle the last non-special menu item (ie skip any window
2662 icons such as MDI maximize, restore or close) */
2663 else if ((vk == VK_RIGHT) && !IS_SYSTEM_MENU(menu))
2665 int i = menu->FocusedItem + 1;
2666 while (i < menu->nItems) {
2667 if ((menu->items[i].wID >= SC_SIZE &&
2668 menu->items[i].wID <= SC_RESTORE)) {
2672 if (i == menu->nItems) {
2676 /* When skipping right, we need to cater for the system menu */
2677 else if ((vk == VK_RIGHT) && IS_SYSTEM_MENU(menu))
2679 if (menu->FocusedItem == (menu->nItems - 1)) {
2686 MDINEXTMENU next_menu;
2691 next_menu.hmenuIn = (IS_SYSTEM_MENU(menu)) ? GetSubMenu(pmt->hTopMenu,0) : pmt->hTopMenu;
2692 next_menu.hmenuNext = 0;
2693 next_menu.hwndNext = 0;
2694 SendMessageW( pmt->hOwnerWnd, WM_NEXTMENU, vk, (LPARAM)&next_menu );
2696 TRACE("%p [%p] -> %p [%p]\n",
2697 pmt->hCurrentMenu, pmt->hOwnerWnd, next_menu.hmenuNext, next_menu.hwndNext );
2699 if (!next_menu.hmenuNext || !next_menu.hwndNext)
2701 DWORD style = GetWindowLongW( pmt->hOwnerWnd, GWL_STYLE );
2702 hNewWnd = pmt->hOwnerWnd;
2703 if( IS_SYSTEM_MENU(menu) )
2705 /* switch to the menu bar */
2707 if(style & WS_CHILD || !(hNewMenu = GetMenu(hNewWnd))) return FALSE;
2711 menu = MENU_GetMenu( hNewMenu );
2712 id = menu->nItems - 1;
2714 /* Skip backwards over any system predefined icons,
2715 eg. MDI close, restore etc icons */
2717 (menu->items[id].wID >= SC_SIZE &&
2718 menu->items[id].wID <= SC_RESTORE)) id--;
2721 else if (style & WS_SYSMENU )
2723 /* switch to the system menu */
2724 hNewMenu = get_win_sys_menu( hNewWnd );
2728 else /* application returned a new menu to switch to */
2730 hNewMenu = next_menu.hmenuNext;
2731 hNewWnd = WIN_GetFullHandle( next_menu.hwndNext );
2733 if( IsMenu(hNewMenu) && IsWindow(hNewWnd) )
2735 DWORD style = GetWindowLongW( hNewWnd, GWL_STYLE );
2737 if (style & WS_SYSMENU &&
2738 GetSubMenu(get_win_sys_menu(hNewWnd), 0) == hNewMenu )
2740 /* get the real system menu */
2741 hNewMenu = get_win_sys_menu(hNewWnd);
2743 else if (style & WS_CHILD || GetMenu(hNewWnd) != hNewMenu )
2745 /* FIXME: Not sure what to do here;
2746 * perhaps try to track hNewMenu as a popup? */
2748 TRACE(" -- got confused.\n");
2755 if( hNewMenu != pmt->hTopMenu )
2757 MENU_SelectItem( pmt->hOwnerWnd, pmt->hTopMenu, NO_SELECTED_ITEM,
2759 if( pmt->hCurrentMenu != pmt->hTopMenu )
2760 MENU_HideSubPopups( pmt->hOwnerWnd, pmt->hTopMenu, FALSE, wFlags );
2763 if( hNewWnd != pmt->hOwnerWnd )
2765 pmt->hOwnerWnd = hNewWnd;
2766 set_capture_window( pmt->hOwnerWnd, GUI_INMENUMODE, NULL );
2769 pmt->hTopMenu = pmt->hCurrentMenu = hNewMenu; /* all subpopups are hidden */
2770 MENU_SelectItem( pmt->hOwnerWnd, pmt->hTopMenu, id, TRUE, 0 );
2777 /***********************************************************************
2780 * The idea is not to show the popup if the next input message is
2781 * going to hide it anyway.
2783 static BOOL MENU_SuspendPopup( MTRACKER* pmt, UINT16 uMsg )
2787 msg.hwnd = pmt->hOwnerWnd;
2789 PeekMessageW( &msg, 0, 0, 0, PM_NOYIELD | PM_REMOVE);
2790 pmt->trackFlags |= TF_SKIPREMOVE;
2795 PeekMessageW( &msg, 0, 0, 0, PM_NOYIELD | PM_NOREMOVE);
2796 if( msg.message == WM_KEYUP || msg.message == WM_PAINT )
2798 PeekMessageW( &msg, 0, 0, 0, PM_NOYIELD | PM_REMOVE);
2799 PeekMessageW( &msg, 0, 0, 0, PM_NOYIELD | PM_NOREMOVE);
2800 if( msg.message == WM_KEYDOWN &&
2801 (msg.wParam == VK_LEFT || msg.wParam == VK_RIGHT))
2803 pmt->trackFlags |= TF_SUSPENDPOPUP;
2810 /* failures go through this */
2811 pmt->trackFlags &= ~TF_SUSPENDPOPUP;
2815 /***********************************************************************
2818 * Handle a VK_ESCAPE key event in a menu.
2820 static BOOL MENU_KeyEscape(MTRACKER* pmt, UINT wFlags)
2822 BOOL bEndMenu = TRUE;
2824 if (pmt->hCurrentMenu != pmt->hTopMenu)
2826 POPUPMENU *menu = MENU_GetMenu(pmt->hCurrentMenu);
2828 if (menu->wFlags & MF_POPUP)
2830 HMENU hmenutmp, hmenuprev;
2832 hmenuprev = hmenutmp = pmt->hTopMenu;
2834 /* close topmost popup */
2835 while (hmenutmp != pmt->hCurrentMenu)
2837 hmenuprev = hmenutmp;
2838 hmenutmp = MENU_GetSubPopup( hmenuprev );
2841 MENU_HideSubPopups( pmt->hOwnerWnd, hmenuprev, TRUE, wFlags );
2842 pmt->hCurrentMenu = hmenuprev;
2850 /***********************************************************************
2853 * Handle a VK_LEFT key event in a menu.
2855 static void MENU_KeyLeft( MTRACKER* pmt, UINT wFlags )
2858 HMENU hmenutmp, hmenuprev;
2861 hmenuprev = hmenutmp = pmt->hTopMenu;
2862 menu = MENU_GetMenu( hmenutmp );
2864 /* Try to move 1 column left (if possible) */
2865 if( (prevcol = MENU_GetStartOfPrevColumn( pmt->hCurrentMenu )) !=
2866 NO_SELECTED_ITEM ) {
2868 MENU_SelectItem( pmt->hOwnerWnd, pmt->hCurrentMenu,
2873 /* close topmost popup */
2874 while (hmenutmp != pmt->hCurrentMenu)
2876 hmenuprev = hmenutmp;
2877 hmenutmp = MENU_GetSubPopup( hmenuprev );
2880 MENU_HideSubPopups( pmt->hOwnerWnd, hmenuprev, TRUE, wFlags );
2881 pmt->hCurrentMenu = hmenuprev;
2883 if ( (hmenuprev == pmt->hTopMenu) && !(menu->wFlags & MF_POPUP) )
2885 /* move menu bar selection if no more popups are left */
2887 if( !MENU_DoNextMenu( pmt, VK_LEFT, wFlags ) )
2888 MENU_MoveSelection( pmt->hOwnerWnd, pmt->hTopMenu, ITEM_PREV );
2890 if ( hmenuprev != hmenutmp || pmt->trackFlags & TF_SUSPENDPOPUP )
2892 /* A sublevel menu was displayed - display the next one
2893 * unless there is another displacement coming up */
2895 if( !MENU_SuspendPopup( pmt, WM_KEYDOWN ) )
2896 pmt->hCurrentMenu = MENU_ShowSubPopup(pmt->hOwnerWnd,
2897 pmt->hTopMenu, TRUE, wFlags);
2903 /***********************************************************************
2906 * Handle a VK_RIGHT key event in a menu.
2908 static void MENU_KeyRight( MTRACKER* pmt, UINT wFlags )
2911 POPUPMENU *menu = MENU_GetMenu( pmt->hTopMenu );
2914 TRACE("MENU_KeyRight called, cur %p (%s), top %p (%s).\n",
2916 debugstr_w((MENU_GetMenu(pmt->hCurrentMenu))->items[0].text),
2917 pmt->hTopMenu, debugstr_w(menu->items[0].text) );
2919 if ( (menu->wFlags & MF_POPUP) || (pmt->hCurrentMenu != pmt->hTopMenu))
2921 /* If already displaying a popup, try to display sub-popup */
2923 hmenutmp = pmt->hCurrentMenu;
2924 pmt->hCurrentMenu = MENU_ShowSubPopup(pmt->hOwnerWnd, hmenutmp, TRUE, wFlags);
2926 /* if subpopup was displayed then we are done */
2927 if (hmenutmp != pmt->hCurrentMenu) return;
2930 /* Check to see if there's another column */
2931 if( (nextcol = MENU_GetStartOfNextColumn( pmt->hCurrentMenu )) !=
2932 NO_SELECTED_ITEM ) {
2933 TRACE("Going to %d.\n", nextcol );
2934 MENU_SelectItem( pmt->hOwnerWnd, pmt->hCurrentMenu,
2939 if (!(menu->wFlags & MF_POPUP)) /* menu bar tracking */
2941 if( pmt->hCurrentMenu != pmt->hTopMenu )
2943 MENU_HideSubPopups( pmt->hOwnerWnd, pmt->hTopMenu, FALSE, wFlags );
2944 hmenutmp = pmt->hCurrentMenu = pmt->hTopMenu;
2945 } else hmenutmp = 0;
2947 /* try to move to the next item */
2948 if( !MENU_DoNextMenu( pmt, VK_RIGHT, wFlags ) )
2949 MENU_MoveSelection( pmt->hOwnerWnd, pmt->hTopMenu, ITEM_NEXT );
2951 if( hmenutmp || pmt->trackFlags & TF_SUSPENDPOPUP )
2952 if( !MENU_SuspendPopup(pmt, WM_KEYDOWN) )
2953 pmt->hCurrentMenu = MENU_ShowSubPopup(pmt->hOwnerWnd,
2954 pmt->hTopMenu, TRUE, wFlags);
2958 /***********************************************************************
2961 * Menu tracking code.
2963 static BOOL MENU_TrackMenu( HMENU hmenu, UINT wFlags, INT x, INT y,
2964 HWND hwnd, const RECT *lprect )
2969 INT executedMenuId = -1;
2971 BOOL enterIdleSent = FALSE;
2975 mt.hCurrentMenu = hmenu;
2976 mt.hTopMenu = hmenu;
2977 mt.hOwnerWnd = WIN_GetFullHandle( hwnd );
2981 TRACE("hmenu=%p flags=0x%08x (%d,%d) hwnd=%p %s\n",
2982 hmenu, wFlags, x, y, hwnd, wine_dbgstr_rect( lprect));
2985 if (!(menu = MENU_GetMenu( hmenu )))
2987 WARN("Invalid menu handle %p\n", hmenu);
2988 SetLastError(ERROR_INVALID_MENU_HANDLE);
2992 if (wFlags & TPM_BUTTONDOWN)
2994 /* Get the result in order to start the tracking or not */
2995 fRemove = MENU_ButtonDown( &mt, hmenu, wFlags );
2996 fEndMenu = !fRemove;
2999 if (wFlags & TF_ENDMENU) fEndMenu = TRUE;
3001 /* owner may not be visible when tracking a popup, so use the menu itself */
3002 capture_win = (wFlags & TPM_POPUPMENU) ? menu->hWnd : mt.hOwnerWnd;
3003 set_capture_window( capture_win, GUI_INMENUMODE, NULL );
3007 menu = MENU_GetMenu( mt.hCurrentMenu );
3008 if (!menu) /* sometimes happens if I do a window manager close */
3011 /* we have to keep the message in the queue until it's
3012 * clear that menu loop is not over yet. */
3016 if (PeekMessageW( &msg, 0, 0, 0, PM_NOREMOVE ))
3018 if (!CallMsgFilterW( &msg, MSGF_MENU )) break;
3019 /* remove the message from the queue */
3020 PeekMessageW( &msg, 0, msg.message, msg.message, PM_REMOVE );
3026 HWND win = (wFlags & TPM_ENTERIDLEEX && menu->wFlags & MF_POPUP) ? menu->hWnd : 0;
3027 enterIdleSent = TRUE;
3028 SendMessageW( mt.hOwnerWnd, WM_ENTERIDLE, MSGF_MENU, (LPARAM)win );
3034 /* check if EndMenu() tried to cancel us, by posting this message */
3035 if(msg.message == WM_CANCELMODE)
3037 /* we are now out of the loop */
3040 /* remove the message from the queue */
3041 PeekMessageW( &msg, 0, msg.message, msg.message, PM_REMOVE );
3043 /* break out of internal loop, ala ESCAPE */
3047 TranslateMessage( &msg );
3050 if ( (msg.hwnd==menu->hWnd) || (msg.message!=WM_TIMER) )
3051 enterIdleSent=FALSE;
3054 if ((msg.message >= WM_MOUSEFIRST) && (msg.message <= WM_MOUSELAST))
3057 * Use the mouse coordinates in lParam instead of those in the MSG
3058 * struct to properly handle synthetic messages. They are already
3059 * in screen coordinates.
3061 mt.pt.x = (short)LOWORD(msg.lParam);
3062 mt.pt.y = (short)HIWORD(msg.lParam);
3064 /* Find a menu for this mouse event */
3065 hmenu = MENU_PtMenu( mt.hTopMenu, mt.pt );
3069 /* no WM_NC... messages in captured state */
3071 case WM_RBUTTONDBLCLK:
3072 case WM_RBUTTONDOWN:
3073 if (!(wFlags & TPM_RIGHTBUTTON)) break;
3075 case WM_LBUTTONDBLCLK:
3076 case WM_LBUTTONDOWN:
3077 /* If the message belongs to the menu, removes it from the queue */
3078 /* Else, end menu tracking */
3079 fRemove = MENU_ButtonDown( &mt, hmenu, wFlags );
3080 fEndMenu = !fRemove;
3084 if (!(wFlags & TPM_RIGHTBUTTON)) break;
3087 /* Check if a menu was selected by the mouse */
3090 executedMenuId = MENU_ButtonUp( &mt, hmenu, wFlags);
3091 TRACE("executedMenuId %d\n", executedMenuId);
3093 /* End the loop if executedMenuId is an item ID */
3094 /* or if the job was done (executedMenuId = 0). */
3095 fEndMenu = fRemove = (executedMenuId != -1);
3097 /* No menu was selected by the mouse */
3098 /* if the function was called by TrackPopupMenu, continue
3099 with the menu tracking. If not, stop it */
3101 fEndMenu = ((wFlags & TPM_POPUPMENU) ? FALSE : TRUE);
3106 /* the selected menu item must be changed every time */
3107 /* the mouse moves. */
3110 fEndMenu |= !MENU_MouseMove( &mt, hmenu, wFlags );
3112 } /* switch(msg.message) - mouse */
3114 else if ((msg.message >= WM_KEYFIRST) && (msg.message <= WM_KEYLAST))
3116 fRemove = TRUE; /* Keyboard messages are always removed */
3129 MENU_SelectItem( mt.hOwnerWnd, mt.hCurrentMenu,
3130 NO_SELECTED_ITEM, FALSE, 0 );
3131 MENU_MoveSelection( mt.hOwnerWnd, mt.hCurrentMenu,
3132 (msg.wParam == VK_HOME)? ITEM_NEXT : ITEM_PREV );
3136 case VK_DOWN: /* If on menu bar, pull-down the menu */
3138 menu = MENU_GetMenu( mt.hCurrentMenu );
3139 if (!(menu->wFlags & MF_POPUP))
3140 mt.hCurrentMenu = MENU_ShowSubPopup(mt.hOwnerWnd, mt.hTopMenu, TRUE, wFlags);
3141 else /* otherwise try to move selection */
3142 MENU_MoveSelection( mt.hOwnerWnd, mt.hCurrentMenu,
3143 (msg.wParam == VK_UP)? ITEM_PREV : ITEM_NEXT );
3147 MENU_KeyLeft( &mt, wFlags );
3151 MENU_KeyRight( &mt, wFlags );
3155 fEndMenu = MENU_KeyEscape(&mt, wFlags);
3161 hi.cbSize = sizeof(HELPINFO);
3162 hi.iContextType = HELPINFO_MENUITEM;
3163 if (menu->FocusedItem == NO_SELECTED_ITEM)
3166 hi.iCtrlId = menu->items[menu->FocusedItem].wID;
3167 hi.hItemHandle = hmenu;
3168 hi.dwContextId = menu->dwContextHelpID;
3169 hi.MousePos = msg.pt;
3170 SendMessageW(hwnd, WM_HELP, 0, (LPARAM)&hi);
3177 break; /* WM_KEYDOWN */
3184 if (msg.wParam == '\r' || msg.wParam == ' ')
3186 executedMenuId = MENU_ExecFocusedItem(&mt,mt.hCurrentMenu, wFlags);
3187 fEndMenu = (executedMenuId != -2);
3192 /* Hack to avoid control chars. */
3193 /* We will find a better way real soon... */
3194 if (msg.wParam < 32) break;
3196 pos = MENU_FindItemByKey( mt.hOwnerWnd, mt.hCurrentMenu,
3197 LOWORD(msg.wParam), FALSE );
3198 if (pos == (UINT)-2) fEndMenu = TRUE;
3199 else if (pos == (UINT)-1) MessageBeep(0);
3202 MENU_SelectItem( mt.hOwnerWnd, mt.hCurrentMenu, pos,
3204 executedMenuId = MENU_ExecFocusedItem(&mt,mt.hCurrentMenu, wFlags);
3205 fEndMenu = (executedMenuId != -2);
3209 } /* switch(msg.message) - kbd */
3213 PeekMessageW( &msg, 0, msg.message, msg.message, PM_REMOVE );
3214 DispatchMessageW( &msg );
3218 if (!fEndMenu) fRemove = TRUE;
3220 /* finally remove message from the queue */
3222 if (fRemove && !(mt.trackFlags & TF_SKIPREMOVE) )
3223 PeekMessageW( &msg, 0, msg.message, msg.message, PM_REMOVE );
3224 else mt.trackFlags &= ~TF_SKIPREMOVE;
3227 set_capture_window( 0, GUI_INMENUMODE, NULL );
3229 /* If dropdown is still painted and the close box is clicked on
3230 then the menu will be destroyed as part of the DispatchMessage above.
3231 This will then invalidate the menu handle in mt.hTopMenu. We should
3232 check for this first. */
3233 if( IsMenu( mt.hTopMenu ) )
3235 menu = MENU_GetMenu( mt.hTopMenu );
3237 if( IsWindow( mt.hOwnerWnd ) )
3239 MENU_HideSubPopups( mt.hOwnerWnd, mt.hTopMenu, FALSE, wFlags );
3241 if (menu && (menu->wFlags & MF_POPUP))
3243 DestroyWindow( menu->hWnd );
3246 if (!(wFlags & TPM_NONOTIFY))
3247 SendMessageW( mt.hOwnerWnd, WM_UNINITMENUPOPUP, (WPARAM)mt.hTopMenu,
3248 MAKELPARAM(0, IS_SYSTEM_MENU(menu)) );
3250 MENU_SelectItem( mt.hOwnerWnd, mt.hTopMenu, NO_SELECTED_ITEM, FALSE, 0 );
3251 SendMessageW( mt.hOwnerWnd, WM_MENUSELECT, MAKELONG(0,0xffff), 0 );
3254 /* Reset the variable for hiding menu */
3255 if( menu ) menu->bTimeToHide = FALSE;
3258 /* The return value is only used by TrackPopupMenu */
3259 if (!(wFlags & TPM_RETURNCMD)) return TRUE;
3260 if (executedMenuId == -1) executedMenuId = 0;
3261 return executedMenuId;
3264 /***********************************************************************
3267 static BOOL MENU_InitTracking(HWND hWnd, HMENU hMenu, BOOL bPopup, UINT wFlags)
3271 TRACE("hwnd=%p hmenu=%p\n", hWnd, hMenu);
3275 /* Send WM_ENTERMENULOOP and WM_INITMENU message only if TPM_NONOTIFY flag is not specified */
3276 if (!(wFlags & TPM_NONOTIFY))
3277 SendMessageW( hWnd, WM_ENTERMENULOOP, bPopup, 0 );
3279 SendMessageW( hWnd, WM_SETCURSOR, (WPARAM)hWnd, HTCAPTION );
3281 if (!(wFlags & TPM_NONOTIFY))
3283 SendMessageW( hWnd, WM_INITMENU, (WPARAM)hMenu, 0 );
3284 /* If an app changed/recreated menu bar entries in WM_INITMENU
3285 * menu sizes will be recalculated once the menu created/shown.
3289 /* This makes the menus of applications built with Delphi work.
3290 * It also enables menus to be displayed in more than one window,
3291 * but there are some bugs left that need to be fixed in this case.
3293 if ((menu = MENU_GetMenu( hMenu ))) menu->hWnd = hWnd;
3297 /***********************************************************************
3300 static BOOL MENU_ExitTracking(HWND hWnd)
3302 TRACE("hwnd=%p\n", hWnd);
3304 SendMessageW( hWnd, WM_EXITMENULOOP, 0, 0 );
3310 /***********************************************************************
3311 * MENU_TrackMouseMenuBar
3313 * Menu-bar tracking upon a mouse event. Called from NC_HandleSysCommand().
3315 void MENU_TrackMouseMenuBar( HWND hWnd, INT ht, POINT pt )
3317 HMENU hMenu = (ht == HTSYSMENU) ? get_win_sys_menu( hWnd ) : GetMenu( hWnd );
3318 UINT wFlags = TPM_ENTERIDLEEX | TPM_BUTTONDOWN | TPM_LEFTALIGN | TPM_LEFTBUTTON;
3320 TRACE("wnd=%p ht=0x%04x %s\n", hWnd, ht, wine_dbgstr_point( &pt));
3324 MENU_InitTracking( hWnd, hMenu, FALSE, wFlags );
3325 MENU_TrackMenu( hMenu, wFlags, pt.x, pt.y, hWnd, NULL );
3326 MENU_ExitTracking(hWnd);
3331 /***********************************************************************
3332 * MENU_TrackKbdMenuBar
3334 * Menu-bar tracking upon a keyboard event. Called from NC_HandleSysCommand().
3336 void MENU_TrackKbdMenuBar( HWND hwnd, UINT wParam, WCHAR wChar)
3338 UINT uItem = NO_SELECTED_ITEM;
3340 UINT wFlags = TPM_ENTERIDLEEX | TPM_LEFTALIGN | TPM_LEFTBUTTON;
3342 TRACE("hwnd %p wParam 0x%04x wChar 0x%04x\n", hwnd, wParam, wChar);
3344 /* find window that has a menu */
3346 while (!WIN_ALLOWED_MENU(GetWindowLongW( hwnd, GWL_STYLE )))
3347 if (!(hwnd = GetAncestor( hwnd, GA_PARENT ))) return;
3349 /* check if we have to track a system menu */
3351 hTrackMenu = GetMenu( hwnd );
3352 if (!hTrackMenu || IsIconic(hwnd) || wChar == ' ' )
3354 if (!(GetWindowLongW( hwnd, GWL_STYLE ) & WS_SYSMENU)) return;
3355 hTrackMenu = get_win_sys_menu( hwnd );
3357 wParam |= HTSYSMENU; /* prevent item lookup */
3360 if (!IsMenu( hTrackMenu )) return;
3362 MENU_InitTracking( hwnd, hTrackMenu, FALSE, wFlags );
3364 if( wChar && wChar != ' ' )
3366 uItem = MENU_FindItemByKey( hwnd, hTrackMenu, wChar, (wParam & HTSYSMENU) );
3367 if ( uItem >= (UINT)(-2) )
3369 if( uItem == (UINT)(-1) ) MessageBeep(0);
3370 /* schedule end of menu tracking */
3371 wFlags |= TF_ENDMENU;
3376 MENU_SelectItem( hwnd, hTrackMenu, uItem, TRUE, 0 );
3378 if (!(wParam & HTSYSMENU) || wChar == ' ')
3380 if( uItem == NO_SELECTED_ITEM )
3381 MENU_MoveSelection( hwnd, hTrackMenu, ITEM_NEXT );
3383 PostMessageW( hwnd, WM_KEYDOWN, VK_DOWN, 0L );
3387 MENU_TrackMenu( hTrackMenu, wFlags, 0, 0, hwnd, NULL );
3388 MENU_ExitTracking( hwnd );
3392 /**********************************************************************
3393 * TrackPopupMenu (USER32.@)
3395 * Like the win32 API, the function return the command ID only if the
3396 * flag TPM_RETURNCMD is on.
3399 BOOL WINAPI TrackPopupMenu( HMENU hMenu, UINT wFlags, INT x, INT y,
3400 INT nReserved, HWND hWnd, const RECT *lpRect )
3404 TRACE("hmenu %p flags %04x (%d,%d) reserved %d hwnd %p rect %s\n",
3405 hMenu, wFlags, x, y, nReserved, hWnd, wine_dbgstr_rect(lpRect));
3407 MENU_InitTracking(hWnd, hMenu, TRUE, wFlags);
3409 /* Send WM_INITMENUPOPUP message only if TPM_NONOTIFY flag is not specified */
3410 if (!(wFlags & TPM_NONOTIFY))
3411 SendMessageW( hWnd, WM_INITMENUPOPUP, (WPARAM)hMenu, 0);
3413 if (MENU_ShowPopup( hWnd, hMenu, 0, x, y, 0, 0 ))
3414 ret = MENU_TrackMenu( hMenu, wFlags | TPM_POPUPMENU, 0, 0, hWnd, lpRect );
3415 MENU_ExitTracking(hWnd);
3420 /**********************************************************************
3421 * TrackPopupMenuEx (USER32.@)
3423 BOOL WINAPI TrackPopupMenuEx( HMENU hMenu, UINT wFlags, INT x, INT y,
3424 HWND hWnd, LPTPMPARAMS lpTpm )
3426 FIXME("not fully implemented\n" );
3427 return TrackPopupMenu( hMenu, wFlags, x, y, 0, hWnd,
3428 lpTpm ? &lpTpm->rcExclude : NULL );
3431 /***********************************************************************
3434 * NOTE: Windows has totally different (and undocumented) popup wndproc.
3436 static LRESULT WINAPI PopupMenuWndProc( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam )
3438 TRACE("hwnd=%p msg=0x%04x wp=0x%04lx lp=0x%08lx\n", hwnd, message, wParam, lParam);
3444 CREATESTRUCTW *cs = (CREATESTRUCTW*)lParam;
3445 SetWindowLongPtrW( hwnd, 0, (LONG_PTR)cs->lpCreateParams );
3449 case WM_MOUSEACTIVATE: /* We don't want to be activated */
3450 return MA_NOACTIVATE;
3455 BeginPaint( hwnd, &ps );
3456 MENU_DrawPopupMenu( hwnd, ps.hdc,
3457 (HMENU)GetWindowLongPtrW( hwnd, 0 ) );
3458 EndPaint( hwnd, &ps );
3465 /* zero out global pointer in case resident popup window was destroyed. */
3466 if (hwnd == top_popup) top_popup = 0;
3473 if (!GetWindowLongPtrW( hwnd, 0 )) ERR("no menu to display\n");
3476 SetWindowLongPtrW( hwnd, 0, 0 );
3479 case MM_SETMENUHANDLE:
3480 SetWindowLongPtrW( hwnd, 0, wParam );
3483 case MM_GETMENUHANDLE:
3484 return GetWindowLongPtrW( hwnd, 0 );
3487 return DefWindowProcW( hwnd, message, wParam, lParam );
3493 /***********************************************************************
3494 * MENU_GetMenuBarHeight
3496 * Compute the size of the menu bar height. Used by NC_HandleNCCalcSize().
3498 UINT MENU_GetMenuBarHeight( HWND hwnd, UINT menubarWidth,
3499 INT orgX, INT orgY )
3505 TRACE("HWND %p, width %d, at (%d, %d).\n", hwnd, menubarWidth, orgX, orgY );
3507 if (!(lppop = MENU_GetMenu( GetMenu(hwnd) ))) return 0;
3509 hdc = GetDCEx( hwnd, 0, DCX_CACHE | DCX_WINDOW );
3510 SelectObject( hdc, get_menu_font(FALSE));
3511 SetRect(&rectBar, orgX, orgY, orgX+menubarWidth, orgY+GetSystemMetrics(SM_CYMENU));
3512 MENU_MenuBarCalcSize( hdc, &rectBar, lppop, hwnd );
3513 ReleaseDC( hwnd, hdc );
3514 return lppop->Height;
3518 /*******************************************************************
3519 * ChangeMenuA (USER32.@)
3521 BOOL WINAPI ChangeMenuA( HMENU hMenu, UINT pos, LPCSTR data,
3522 UINT id, UINT flags )
3524 TRACE("menu=%p pos=%d data=%p id=%08x flags=%08x\n", hMenu, pos, data, id, flags );
3525 if (flags & MF_APPEND) return AppendMenuA( hMenu, flags & ~MF_APPEND,
3527 if (flags & MF_DELETE) return DeleteMenu(hMenu, pos, flags & ~MF_DELETE);
3528 if (flags & MF_CHANGE) return ModifyMenuA(hMenu, pos, flags & ~MF_CHANGE,
3530 if (flags & MF_REMOVE) return RemoveMenu( hMenu,
3531 flags & MF_BYPOSITION ? pos : id,
3532 flags & ~MF_REMOVE );
3533 /* Default: MF_INSERT */
3534 return InsertMenuA( hMenu, pos, flags, id, data );
3538 /*******************************************************************
3539 * ChangeMenuW (USER32.@)
3541 BOOL WINAPI ChangeMenuW( HMENU hMenu, UINT pos, LPCWSTR data,
3542 UINT id, UINT flags )
3544 TRACE("menu=%p pos=%d data=%p id=%08x flags=%08x\n", hMenu, pos, data, id, flags );
3545 if (flags & MF_APPEND) return AppendMenuW( hMenu, flags & ~MF_APPEND,
3547 if (flags & MF_DELETE) return DeleteMenu(hMenu, pos, flags & ~MF_DELETE);
3548 if (flags & MF_CHANGE) return ModifyMenuW(hMenu, pos, flags & ~MF_CHANGE,
3550 if (flags & MF_REMOVE) return RemoveMenu( hMenu,
3551 flags & MF_BYPOSITION ? pos : id,
3552 flags & ~MF_REMOVE );
3553 /* Default: MF_INSERT */
3554 return InsertMenuW( hMenu, pos, flags, id, data );
3558 /*******************************************************************
3559 * CheckMenuItem (USER32.@)
3561 DWORD WINAPI CheckMenuItem( HMENU hMenu, UINT id, UINT flags )
3566 if (!(item = MENU_FindItem( &hMenu, &id, flags ))) return -1;
3567 ret = item->fState & MF_CHECKED;
3568 if (flags & MF_CHECKED) item->fState |= MF_CHECKED;
3569 else item->fState &= ~MF_CHECKED;
3574 /**********************************************************************
3575 * EnableMenuItem (USER32.@)
3577 BOOL WINAPI EnableMenuItem( HMENU hMenu, UINT wItemID, UINT wFlags )
3583 TRACE("(%p, %04x, %04x) !\n", hMenu, wItemID, wFlags);
3585 /* Get the Popupmenu to access the owner menu */
3586 if (!(menu = MENU_GetMenu(hMenu)))
3589 if (!(item = MENU_FindItem( &hMenu, &wItemID, wFlags )))
3592 oldflags = item->fState & (MF_GRAYED | MF_DISABLED);
3593 item->fState ^= (oldflags ^ wFlags) & (MF_GRAYED | MF_DISABLED);
3595 /* If the close item in the system menu change update the close button */
3596 if((item->wID == SC_CLOSE) && (oldflags != wFlags))
3598 if (menu->hSysMenuOwner != 0)
3601 POPUPMENU* parentMenu;
3603 /* Get the parent menu to access*/
3604 if (!(parentMenu = MENU_GetMenu(menu->hSysMenuOwner)))
3607 /* Refresh the frame to reflect the change */
3608 GetWindowRect(parentMenu->hWnd, &rc);
3609 MapWindowPoints(0, parentMenu->hWnd, (POINT *)&rc, 2);
3611 RedrawWindow(parentMenu->hWnd, &rc, 0, RDW_FRAME | RDW_INVALIDATE | RDW_NOCHILDREN);
3619 /*******************************************************************
3620 * GetMenuStringA (USER32.@)
3622 INT WINAPI GetMenuStringA(
3623 HMENU hMenu, /* [in] menuhandle */
3624 UINT wItemID, /* [in] menu item (dep. on wFlags) */
3625 LPSTR str, /* [out] outbuffer. If NULL, func returns entry length*/
3626 INT nMaxSiz, /* [in] length of buffer. if 0, func returns entry len*/
3627 UINT wFlags /* [in] MF_ flags */
3631 TRACE("menu=%p item=%04x ptr=%p len=%d flags=%04x\n", hMenu, wItemID, str, nMaxSiz, wFlags );
3632 if (str && nMaxSiz) str[0] = '\0';
3633 if (!(item = MENU_FindItem( &hMenu, &wItemID, wFlags ))) {
3634 SetLastError( ERROR_MENU_ITEM_NOT_FOUND);
3637 if (!item->text) return 0;
3638 if (!str || !nMaxSiz) return strlenW(item->text);
3639 if (!WideCharToMultiByte( CP_ACP, 0, item->text, -1, str, nMaxSiz, NULL, NULL ))
3641 TRACE("returning %s\n", debugstr_a(str));
3646 /*******************************************************************
3647 * GetMenuStringW (USER32.@)
3649 INT WINAPI GetMenuStringW( HMENU hMenu, UINT wItemID,
3650 LPWSTR str, INT nMaxSiz, UINT wFlags )
3654 TRACE("menu=%p item=%04x ptr=%p len=%d flags=%04x\n", hMenu, wItemID, str, nMaxSiz, wFlags );
3655 if (str && nMaxSiz) str[0] = '\0';
3656 if (!(item = MENU_FindItem( &hMenu, &wItemID, wFlags ))) {
3657 SetLastError( ERROR_MENU_ITEM_NOT_FOUND);
3660 if (!str || !nMaxSiz) return item->text ? strlenW(item->text) : 0;
3661 if( !(item->text)) {
3665 lstrcpynW( str, item->text, nMaxSiz );
3666 TRACE("returning %s\n", debugstr_w(str));
3667 return strlenW(str);
3671 /**********************************************************************
3672 * HiliteMenuItem (USER32.@)
3674 BOOL WINAPI HiliteMenuItem( HWND hWnd, HMENU hMenu, UINT wItemID,
3678 TRACE("(%p, %p, %04x, %04x);\n", hWnd, hMenu, wItemID, wHilite);
3679 if (!MENU_FindItem( &hMenu, &wItemID, wHilite )) return FALSE;
3680 if (!(menu = MENU_GetMenu(hMenu))) return FALSE;
3681 if (menu->FocusedItem == wItemID) return TRUE;
3682 MENU_HideSubPopups( hWnd, hMenu, FALSE, 0 );
3683 MENU_SelectItem( hWnd, hMenu, wItemID, TRUE, 0 );
3688 /**********************************************************************
3689 * GetMenuState (USER32.@)
3691 UINT WINAPI GetMenuState( HMENU hMenu, UINT wItemID, UINT wFlags )
3694 TRACE("(menu=%p, id=%04x, flags=%04x);\n", hMenu, wItemID, wFlags);
3695 if (!(item = MENU_FindItem( &hMenu, &wItemID, wFlags ))) return -1;
3696 debug_print_menuitem (" item: ", item, "");
3697 if (item->fType & MF_POPUP)
3699 POPUPMENU *menu = MENU_GetMenu( item->hSubMenu );
3700 if (!menu) return -1;
3701 else return (menu->nItems << 8) | ((item->fState|item->fType) & 0xff);
3705 /* We used to (from way back then) mask the result to 0xff. */
3706 /* I don't know why and it seems wrong as the documented */
3707 /* return flag MF_SEPARATOR is outside that mask. */
3708 return (item->fType | item->fState);
3713 /**********************************************************************
3714 * GetMenuItemCount (USER32.@)
3716 INT WINAPI GetMenuItemCount( HMENU hMenu )
3718 LPPOPUPMENU menu = MENU_GetMenu(hMenu);
3719 if (!menu) return -1;
3720 TRACE("(%p) returning %d\n", hMenu, menu->nItems );
3721 return menu->nItems;
3725 /**********************************************************************
3726 * GetMenuItemID (USER32.@)
3728 UINT WINAPI GetMenuItemID( HMENU hMenu, INT nPos )
3732 if (!(lpmi = MENU_FindItem(&hMenu,(UINT*)&nPos,MF_BYPOSITION))) return -1;
3733 if (lpmi->fType & MF_POPUP) return -1;
3739 /*******************************************************************
3740 * InsertMenuW (USER32.@)
3742 BOOL WINAPI InsertMenuW( HMENU hMenu, UINT pos, UINT flags,
3743 UINT_PTR id, LPCWSTR str )
3747 if (IS_STRING_ITEM(flags) && str)
3748 TRACE("hMenu %p, pos %d, flags %08x, id %04lx, str %s\n",
3749 hMenu, pos, flags, id, debugstr_w(str) );
3750 else TRACE("hMenu %p, pos %d, flags %08x, id %04lx, str %p (not a string)\n",
3751 hMenu, pos, flags, id, str );
3753 if (!(item = MENU_InsertItem( hMenu, pos, flags ))) return FALSE;
3755 if (!(MENU_SetItemData( item, flags, id, str )))
3757 RemoveMenu( hMenu, pos, flags );
3761 item->hCheckBit = item->hUnCheckBit = 0;
3766 /*******************************************************************
3767 * InsertMenuA (USER32.@)
3769 BOOL WINAPI InsertMenuA( HMENU hMenu, UINT pos, UINT flags,
3770 UINT_PTR id, LPCSTR str )
3774 if (IS_STRING_ITEM(flags) && str)
3776 INT len = MultiByteToWideChar( CP_ACP, 0, str, -1, NULL, 0 );
3777 LPWSTR newstr = HeapAlloc( GetProcessHeap(), 0, len * sizeof(WCHAR) );
3780 MultiByteToWideChar( CP_ACP, 0, str, -1, newstr, len );
3781 ret = InsertMenuW( hMenu, pos, flags, id, newstr );
3782 HeapFree( GetProcessHeap(), 0, newstr );
3786 else return InsertMenuW( hMenu, pos, flags, id, (LPCWSTR)str );
3790 /*******************************************************************
3791 * AppendMenuA (USER32.@)
3793 BOOL WINAPI AppendMenuA( HMENU hMenu, UINT flags,
3794 UINT_PTR id, LPCSTR data )
3796 return InsertMenuA( hMenu, -1, flags | MF_BYPOSITION, id, data );
3800 /*******************************************************************
3801 * AppendMenuW (USER32.@)
3803 BOOL WINAPI AppendMenuW( HMENU hMenu, UINT flags,
3804 UINT_PTR id, LPCWSTR data )
3806 return InsertMenuW( hMenu, -1, flags | MF_BYPOSITION, id, data );
3810 /**********************************************************************
3811 * RemoveMenu (USER32.@)
3813 BOOL WINAPI RemoveMenu( HMENU hMenu, UINT nPos, UINT wFlags )
3818 TRACE("(menu=%p pos=%04x flags=%04x)\n",hMenu, nPos, wFlags);
3819 if (!(item = MENU_FindItem( &hMenu, &nPos, wFlags ))) return FALSE;
3820 if (!(menu = MENU_GetMenu(hMenu))) return FALSE;
3824 MENU_FreeItemData( item );
3826 if (--menu->nItems == 0)
3828 HeapFree( GetProcessHeap(), 0, menu->items );
3833 while(nPos < menu->nItems)
3839 menu->items = HeapReAlloc( GetProcessHeap(), 0, menu->items,
3840 menu->nItems * sizeof(MENUITEM) );
3846 /**********************************************************************
3847 * DeleteMenu (USER32.@)
3849 BOOL WINAPI DeleteMenu( HMENU hMenu, UINT nPos, UINT wFlags )
3851 MENUITEM *item = MENU_FindItem( &hMenu, &nPos, wFlags );
3852 if (!item) return FALSE;
3853 if (item->fType & MF_POPUP) DestroyMenu( item->hSubMenu );
3854 /* nPos is now the position of the item */
3855 RemoveMenu( hMenu, nPos, wFlags | MF_BYPOSITION );
3860 /*******************************************************************
3861 * ModifyMenuW (USER32.@)
3863 BOOL WINAPI ModifyMenuW( HMENU hMenu, UINT pos, UINT flags,
3864 UINT_PTR id, LPCWSTR str )
3868 if (IS_STRING_ITEM(flags))
3869 TRACE("%p %d %04x %04lx %s\n", hMenu, pos, flags, id, debugstr_w(str) );
3871 TRACE("%p %d %04x %04lx %p\n", hMenu, pos, flags, id, str );
3873 if (!(item = MENU_FindItem( &hMenu, &pos, flags ))) return FALSE;
3874 MENU_GetMenu(hMenu)->Height = 0; /* force size recalculate */
3875 return MENU_SetItemData( item, flags, id, str );
3879 /*******************************************************************
3880 * ModifyMenuA (USER32.@)
3882 BOOL WINAPI ModifyMenuA( HMENU hMenu, UINT pos, UINT flags,
3883 UINT_PTR id, LPCSTR str )
3887 if (IS_STRING_ITEM(flags) && str)
3889 INT len = MultiByteToWideChar( CP_ACP, 0, str, -1, NULL, 0 );
3890 LPWSTR newstr = HeapAlloc( GetProcessHeap(), 0, len * sizeof(WCHAR) );
3893 MultiByteToWideChar( CP_ACP, 0, str, -1, newstr, len );
3894 ret = ModifyMenuW( hMenu, pos, flags, id, newstr );
3895 HeapFree( GetProcessHeap(), 0, newstr );
3899 else return ModifyMenuW( hMenu, pos, flags, id, (LPCWSTR)str );
3903 /**********************************************************************
3904 * CreatePopupMenu (USER32.@)
3906 HMENU WINAPI CreatePopupMenu(void)
3911 if (!(hmenu = CreateMenu())) return 0;
3912 menu = MENU_GetMenu( hmenu );
3913 menu->wFlags |= MF_POPUP;
3914 menu->bTimeToHide = FALSE;
3919 /**********************************************************************
3920 * GetMenuCheckMarkDimensions (USER.417)
3921 * GetMenuCheckMarkDimensions (USER32.@)
3923 DWORD WINAPI GetMenuCheckMarkDimensions(void)
3925 return MAKELONG( GetSystemMetrics(SM_CXMENUCHECK), GetSystemMetrics(SM_CYMENUCHECK) );
3929 /**********************************************************************
3930 * SetMenuItemBitmaps (USER32.@)
3932 BOOL WINAPI SetMenuItemBitmaps( HMENU hMenu, UINT nPos, UINT wFlags,
3933 HBITMAP hNewUnCheck, HBITMAP hNewCheck)
3937 if (!(item = MENU_FindItem( &hMenu, &nPos, wFlags ))) return FALSE;
3939 if (!hNewCheck && !hNewUnCheck)
3941 item->fState &= ~MF_USECHECKBITMAPS;
3943 else /* Install new bitmaps */
3945 item->hCheckBit = hNewCheck;
3946 item->hUnCheckBit = hNewUnCheck;
3947 item->fState |= MF_USECHECKBITMAPS;
3953 /**********************************************************************
3954 * CreateMenu (USER32.@)
3956 HMENU WINAPI CreateMenu(void)
3960 if (!(hMenu = USER_HEAP_ALLOC( sizeof(POPUPMENU) ))) return 0;
3961 menu = (LPPOPUPMENU) USER_HEAP_LIN_ADDR(hMenu);
3963 ZeroMemory(menu, sizeof(POPUPMENU));
3964 menu->wMagic = MENU_MAGIC;
3965 menu->FocusedItem = NO_SELECTED_ITEM;
3966 menu->bTimeToHide = FALSE;
3968 TRACE("return %p\n", hMenu );
3974 /**********************************************************************
3975 * DestroyMenu (USER32.@)
3977 BOOL WINAPI DestroyMenu( HMENU hMenu )
3979 LPPOPUPMENU lppop = MENU_GetMenu(hMenu);
3981 TRACE("(%p)\n", hMenu);
3984 if (!lppop) return FALSE;
3986 lppop->wMagic = 0; /* Mark it as destroyed */
3988 /* DestroyMenu should not destroy system menu popup owner */
3989 if ((lppop->wFlags & (MF_POPUP | MF_SYSMENU)) == MF_POPUP && lppop->hWnd)
3991 DestroyWindow( lppop->hWnd );
3995 if (lppop->items) /* recursively destroy submenus */
3998 MENUITEM *item = lppop->items;
3999 for (i = lppop->nItems; i > 0; i--, item++)
4001 if (item->fType & MF_POPUP) DestroyMenu(item->hSubMenu);
4002 MENU_FreeItemData( item );
4004 HeapFree( GetProcessHeap(), 0, lppop->items );
4006 USER_HEAP_FREE( hMenu );
4011 /**********************************************************************
4012 * GetSystemMenu (USER32.@)
4014 HMENU WINAPI GetSystemMenu( HWND hWnd, BOOL bRevert )
4016 WND *wndPtr = WIN_GetPtr( hWnd );
4019 if (wndPtr == WND_DESKTOP) return 0;
4020 if (wndPtr == WND_OTHER_PROCESS)
4022 if (IsWindow( hWnd )) FIXME( "not supported on other process window %p\n", hWnd );
4026 if (wndPtr->hSysMenu && bRevert)
4028 DestroyMenu(wndPtr->hSysMenu);
4029 wndPtr->hSysMenu = 0;
4032 if(!wndPtr->hSysMenu && (wndPtr->dwStyle & WS_SYSMENU) )
4033 wndPtr->hSysMenu = MENU_GetSysMenu( hWnd, 0 );
4035 if( wndPtr->hSysMenu )
4038 retvalue = GetSubMenu(wndPtr->hSysMenu, 0);
4040 /* Store the dummy sysmenu handle to facilitate the refresh */
4041 /* of the close button if the SC_CLOSE item change */
4042 menu = MENU_GetMenu(retvalue);
4044 menu->hSysMenuOwner = wndPtr->hSysMenu;
4046 WIN_ReleasePtr( wndPtr );
4048 return bRevert ? 0 : retvalue;
4052 /*******************************************************************
4053 * SetSystemMenu (USER32.@)
4055 BOOL WINAPI SetSystemMenu( HWND hwnd, HMENU hMenu )
4057 WND *wndPtr = WIN_GetPtr( hwnd );
4059 if (wndPtr && wndPtr != WND_OTHER_PROCESS && wndPtr != WND_DESKTOP)
4061 if (wndPtr->hSysMenu) DestroyMenu( wndPtr->hSysMenu );
4062 wndPtr->hSysMenu = MENU_GetSysMenu( hwnd, hMenu );
4063 WIN_ReleasePtr( wndPtr );
4070 /**********************************************************************
4071 * GetMenu (USER32.@)
4073 HMENU WINAPI GetMenu( HWND hWnd )
4075 HMENU retvalue = (HMENU)GetWindowLongPtrW( hWnd, GWLP_ID );
4076 TRACE("for %p returning %p\n", hWnd, retvalue);
4080 /**********************************************************************
4081 * GetMenuBarInfo (USER32.@)
4083 BOOL WINAPI GetMenuBarInfo( HWND hwnd, LONG idObject, LONG idItem, PMENUBARINFO pmbi )
4085 FIXME( "(%p,0x%08x,0x%08x,%p)\n", hwnd, idObject, idItem, pmbi );
4089 /**********************************************************************
4092 * Helper for SetMenu. Also called by WIN_CreateWindowEx to avoid the
4093 * SetWindowPos call that would result if SetMenu were called directly.
4095 BOOL MENU_SetMenu( HWND hWnd, HMENU hMenu )
4097 TRACE("(%p, %p);\n", hWnd, hMenu);
4099 if (hMenu && !IsMenu(hMenu))
4101 WARN("hMenu %p is not a menu handle\n", hMenu);
4104 if (!WIN_ALLOWED_MENU(GetWindowLongW( hWnd, GWL_STYLE )))
4107 hWnd = WIN_GetFullHandle( hWnd );
4108 if (GetCapture() == hWnd)
4109 set_capture_window( 0, GUI_INMENUMODE, NULL ); /* release the capture */
4115 if (!(lpmenu = MENU_GetMenu(hMenu))) return FALSE;
4117 lpmenu->hWnd = hWnd;
4118 lpmenu->Height = 0; /* Make sure we recalculate the size */
4120 SetWindowLongPtrW( hWnd, GWLP_ID, (LONG_PTR)hMenu );
4125 /**********************************************************************
4126 * SetMenu (USER32.@)
4128 BOOL WINAPI SetMenu( HWND hWnd, HMENU hMenu )
4130 if(!MENU_SetMenu(hWnd, hMenu))
4133 SetWindowPos( hWnd, 0, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE |
4134 SWP_NOACTIVATE | SWP_NOZORDER | SWP_FRAMECHANGED );
4139 /**********************************************************************
4140 * GetSubMenu (USER32.@)
4142 HMENU WINAPI GetSubMenu( HMENU hMenu, INT nPos )
4146 if (!(lpmi = MENU_FindItem(&hMenu,(UINT*)&nPos,MF_BYPOSITION))) return 0;
4147 if (!(lpmi->fType & MF_POPUP)) return 0;
4148 return lpmi->hSubMenu;
4152 /**********************************************************************
4153 * DrawMenuBar (USER32.@)
4155 BOOL WINAPI DrawMenuBar( HWND hWnd )
4158 HMENU hMenu = GetMenu(hWnd);
4160 if (!WIN_ALLOWED_MENU(GetWindowLongW( hWnd, GWL_STYLE )))
4162 if (!hMenu || !(lppop = MENU_GetMenu( hMenu ))) return FALSE;
4164 lppop->Height = 0; /* Make sure we call MENU_MenuBarCalcSize */
4165 lppop->hwndOwner = hWnd;
4166 SetWindowPos( hWnd, 0, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE |
4167 SWP_NOACTIVATE | SWP_NOZORDER | SWP_FRAMECHANGED );
4171 /***********************************************************************
4172 * DrawMenuBarTemp (USER32.@)
4176 * called by W98SE desk.cpl Control Panel Applet
4178 * Not 100% sure about the param names, but close.
4180 DWORD WINAPI DrawMenuBarTemp(HWND hwnd, HDC hDC, LPRECT lprect, HMENU hMenu, HFONT hFont)
4185 BOOL flat_menu = FALSE;
4187 SystemParametersInfoW (SPI_GETFLATMENU, 0, &flat_menu, 0);
4190 hMenu = GetMenu(hwnd);
4193 hFont = get_menu_font(FALSE);
4195 lppop = MENU_GetMenu( hMenu );
4196 if (lppop == NULL || lprect == NULL)
4198 retvalue = GetSystemMetrics(SM_CYMENU);
4202 TRACE("(%p, %p, %p, %p, %p)\n", hwnd, hDC, lprect, hMenu, hFont);
4204 hfontOld = SelectObject( hDC, hFont);
4206 if (lppop->Height == 0)
4207 MENU_MenuBarCalcSize(hDC, lprect, lppop, hwnd);
4209 lprect->bottom = lprect->top + lppop->Height;
4211 FillRect(hDC, lprect, GetSysColorBrush(flat_menu ? COLOR_MENUBAR : COLOR_MENU) );
4213 SelectObject( hDC, SYSCOLOR_GetPen(COLOR_3DFACE));
4214 MoveToEx( hDC, lprect->left, lprect->bottom, NULL );
4215 LineTo( hDC, lprect->right, lprect->bottom );
4217 if (lppop->nItems == 0)
4219 retvalue = GetSystemMetrics(SM_CYMENU);
4223 for (i = 0; i < lppop->nItems; i++)
4225 MENU_DrawMenuItem( hwnd, hMenu, hwnd,
4226 hDC, &lppop->items[i], lppop->Height, TRUE, ODA_DRAWENTIRE );
4228 retvalue = lppop->Height;
4231 if (hfontOld) SelectObject (hDC, hfontOld);
4235 /***********************************************************************
4236 * EndMenu (USER.187)
4237 * EndMenu (USER32.@)
4239 BOOL WINAPI EndMenu(void)
4241 /* if we are in the menu code, and it is active */
4242 if (!fEndMenu && top_popup)
4244 /* terminate the menu handling code */
4247 /* needs to be posted to wakeup the internal menu handler */
4248 /* which will now terminate the menu, in the event that */
4249 /* the main window was minimized, or lost focus, so we */
4250 /* don't end up with an orphaned menu */
4251 PostMessageW( top_popup, WM_CANCELMODE, 0, 0);
4257 /***********************************************************************
4258 * LookupMenuHandle (USER.217)
4260 HMENU16 WINAPI LookupMenuHandle16( HMENU16 hmenu, INT16 id )
4262 HMENU hmenu32 = HMENU_32(hmenu);
4264 if (!MENU_FindItem( &hmenu32, &id32, MF_BYCOMMAND )) return 0;
4265 else return HMENU_16(hmenu32);
4269 /**********************************************************************
4270 * LoadMenu (USER.150)
4272 HMENU16 WINAPI LoadMenu16( HINSTANCE16 instance, LPCSTR name )
4278 if (HIWORD(name) && name[0] == '#') name = ULongToPtr(atoi( name + 1 ));
4279 if (!name) return 0;
4281 instance = GetExePtr( instance );
4282 if (!(hRsrc = FindResource16( instance, name, (LPSTR)RT_MENU ))) return 0;
4283 if (!(handle = LoadResource16( instance, hRsrc ))) return 0;
4284 hMenu = LoadMenuIndirect16(LockResource16(handle));
4285 FreeResource16( handle );
4290 /*****************************************************************
4291 * LoadMenuA (USER32.@)
4293 HMENU WINAPI LoadMenuA( HINSTANCE instance, LPCSTR name )
4295 HRSRC hrsrc = FindResourceA( instance, name, (LPSTR)RT_MENU );
4296 if (!hrsrc) return 0;
4297 return LoadMenuIndirectA( (LPCVOID)LoadResource( instance, hrsrc ));
4301 /*****************************************************************
4302 * LoadMenuW (USER32.@)
4304 HMENU WINAPI LoadMenuW( HINSTANCE instance, LPCWSTR name )
4306 HRSRC hrsrc = FindResourceW( instance, name, (LPWSTR)RT_MENU );
4307 if (!hrsrc) return 0;
4308 return LoadMenuIndirectW( (LPCVOID)LoadResource( instance, hrsrc ));
4312 /**********************************************************************
4313 * LoadMenuIndirect (USER.220)
4315 HMENU16 WINAPI LoadMenuIndirect16( LPCVOID template )
4318 WORD version, offset;
4319 LPCSTR p = (LPCSTR)template;
4321 TRACE("(%p)\n", template );
4322 version = GET_WORD(p);
4326 WARN("version must be 0 for Win16\n" );
4329 offset = GET_WORD(p);
4330 p += sizeof(WORD) + offset;
4331 if (!(hMenu = CreateMenu())) return 0;
4332 if (!MENU_ParseResource( p, hMenu, FALSE ))
4334 DestroyMenu( hMenu );
4337 return HMENU_16(hMenu);
4341 /**********************************************************************
4342 * LoadMenuIndirectW (USER32.@)
4344 HMENU WINAPI LoadMenuIndirectW( LPCVOID template )
4347 WORD version, offset;
4348 LPCSTR p = (LPCSTR)template;
4350 version = GET_WORD(p);
4352 TRACE("%p, ver %d\n", template, version );
4355 case 0: /* standard format is version of 0 */
4356 offset = GET_WORD(p);
4357 p += sizeof(WORD) + offset;
4358 if (!(hMenu = CreateMenu())) return 0;
4359 if (!MENU_ParseResource( p, hMenu, TRUE ))
4361 DestroyMenu( hMenu );
4365 case 1: /* extended format is version of 1 */
4366 offset = GET_WORD(p);
4367 p += sizeof(WORD) + offset;
4368 if (!(hMenu = CreateMenu())) return 0;
4369 if (!MENUEX_ParseResource( p, hMenu))
4371 DestroyMenu( hMenu );
4376 ERR("version %d not supported.\n", version);
4382 /**********************************************************************
4383 * LoadMenuIndirectA (USER32.@)
4385 HMENU WINAPI LoadMenuIndirectA( LPCVOID template )
4387 return LoadMenuIndirectW( template );
4391 /**********************************************************************
4394 BOOL WINAPI IsMenu(HMENU hmenu)
4396 LPPOPUPMENU menu = MENU_GetMenu(hmenu);
4400 SetLastError(ERROR_INVALID_MENU_HANDLE);
4406 /**********************************************************************
4407 * GetMenuItemInfo_common
4410 static BOOL GetMenuItemInfo_common ( HMENU hmenu, UINT item, BOOL bypos,
4411 LPMENUITEMINFOW lpmii, BOOL unicode)
4413 MENUITEM *menu = MENU_FindItem (&hmenu, &item, bypos ? MF_BYPOSITION : 0);
4415 debug_print_menuitem("GetMenuItemInfo_common: ", menu, "");
4420 if( lpmii->fMask & MIIM_TYPE) {
4421 if( lpmii->fMask & ( MIIM_STRING | MIIM_FTYPE | MIIM_BITMAP)) {
4422 WARN("invalid combination of fMask bits used\n");
4423 /* this does not happen on Win9x/ME */
4424 SetLastError( ERROR_INVALID_PARAMETER);
4427 lpmii->fType = menu->fType & MENUITEMINFO_TYPE_MASK;
4428 if( menu->hbmpItem) lpmii->fType |= MFT_BITMAP;
4429 lpmii->hbmpItem = menu->hbmpItem; /* not on Win9x/ME */
4430 if( lpmii->fType & MFT_BITMAP) {
4431 lpmii->dwTypeData = (LPWSTR) menu->hbmpItem;
4433 } else if( lpmii->fType & (MFT_OWNERDRAW | MFT_SEPARATOR)) {
4434 /* this does not happen on Win9x/ME */
4435 lpmii->dwTypeData = 0;
4440 /* copy the text string */
4441 if ((lpmii->fMask & (MIIM_TYPE|MIIM_STRING))) {
4443 if(lpmii->dwTypeData && lpmii->cch) {
4446 *((WCHAR *)lpmii->dwTypeData) = 0;
4448 *((CHAR *)lpmii->dwTypeData) = 0;
4454 len = strlenW(menu->text);
4455 if(lpmii->dwTypeData && lpmii->cch)
4456 lstrcpynW(lpmii->dwTypeData, menu->text, lpmii->cch);
4460 len = WideCharToMultiByte( CP_ACP, 0, menu->text, -1, NULL,
4461 0, NULL, NULL ) - 1;
4462 if(lpmii->dwTypeData && lpmii->cch)
4463 if (!WideCharToMultiByte( CP_ACP, 0, menu->text, -1,
4464 (LPSTR)lpmii->dwTypeData, lpmii->cch, NULL, NULL ))
4465 ((LPSTR)lpmii->dwTypeData)[lpmii->cch - 1] = 0;
4467 /* if we've copied a substring we return its length */
4468 if(lpmii->dwTypeData && lpmii->cch)
4469 if (lpmii->cch <= len + 1)
4474 /* return length of string */
4475 /* not on Win9x/ME if fType & MFT_BITMAP */
4481 if (lpmii->fMask & MIIM_FTYPE)
4482 lpmii->fType = menu->fType & MENUITEMINFO_TYPE_MASK;
4484 if (lpmii->fMask & MIIM_BITMAP)
4485 lpmii->hbmpItem = menu->hbmpItem;
4487 if (lpmii->fMask & MIIM_STATE)
4488 lpmii->fState = menu->fState & MENUITEMINFO_STATE_MASK;
4490 if (lpmii->fMask & MIIM_ID)
4491 lpmii->wID = menu->wID;
4493 if (lpmii->fMask & MIIM_SUBMENU)
4494 lpmii->hSubMenu = menu->hSubMenu;
4496 /* hSubMenu is always cleared
4497 * (not on Win9x/ME ) */
4498 lpmii->hSubMenu = 0;
4501 if (lpmii->fMask & MIIM_CHECKMARKS) {
4502 lpmii->hbmpChecked = menu->hCheckBit;
4503 lpmii->hbmpUnchecked = menu->hUnCheckBit;
4505 if (lpmii->fMask & MIIM_DATA)
4506 lpmii->dwItemData = menu->dwItemData;
4511 /**********************************************************************
4512 * GetMenuItemInfoA (USER32.@)
4514 BOOL WINAPI GetMenuItemInfoA( HMENU hmenu, UINT item, BOOL bypos,
4515 LPMENUITEMINFOA lpmii)
4519 if( lpmii->cbSize != sizeof( mii) &&
4520 lpmii->cbSize != sizeof( mii) - sizeof ( mii.hbmpItem)) {
4521 SetLastError( ERROR_INVALID_PARAMETER);
4524 memcpy( &mii, lpmii, lpmii->cbSize);
4525 mii.cbSize = sizeof( mii);
4526 ret = GetMenuItemInfo_common (hmenu, item, bypos,
4527 (LPMENUITEMINFOW)&mii, FALSE);
4528 mii.cbSize = lpmii->cbSize;
4529 memcpy( lpmii, &mii, mii.cbSize);
4533 /**********************************************************************
4534 * GetMenuItemInfoW (USER32.@)
4536 BOOL WINAPI GetMenuItemInfoW( HMENU hmenu, UINT item, BOOL bypos,
4537 LPMENUITEMINFOW lpmii)
4541 if( lpmii->cbSize != sizeof( mii) &&
4542 lpmii->cbSize != sizeof( mii) - sizeof ( mii.hbmpItem)) {
4543 SetLastError( ERROR_INVALID_PARAMETER);
4546 memcpy( &mii, lpmii, lpmii->cbSize);
4547 mii.cbSize = sizeof( mii);
4548 ret = GetMenuItemInfo_common (hmenu, item, bypos, &mii, TRUE);
4549 mii.cbSize = lpmii->cbSize;
4550 memcpy( lpmii, &mii, mii.cbSize);
4555 /* set a menu item text from a ASCII or Unicode string */
4556 static inline void set_menu_item_text( MENUITEM *menu, LPCWSTR text, BOOL unicode )
4562 if ((menu->text = HeapAlloc( GetProcessHeap(), 0, (strlenW(text)+1) * sizeof(WCHAR) )))
4563 strcpyW( menu->text, text );
4567 LPCSTR str = (LPCSTR)text;
4568 int len = MultiByteToWideChar( CP_ACP, 0, str, -1, NULL, 0 );
4569 if ((menu->text = HeapAlloc( GetProcessHeap(), 0, len * sizeof(WCHAR) )))
4570 MultiByteToWideChar( CP_ACP, 0, str, -1, menu->text, len );
4575 /**********************************************************************
4576 * SetMenuItemInfo_common
4579 static BOOL SetMenuItemInfo_common(MENUITEM * menu,
4580 const MENUITEMINFOW *lpmii,
4583 if (!menu) return FALSE;
4585 debug_print_menuitem("SetMenuItemInfo_common from: ", menu, "");
4587 if (lpmii->fMask & MIIM_TYPE ) {
4588 if( lpmii->fMask & ( MIIM_STRING | MIIM_FTYPE | MIIM_BITMAP)) {
4589 WARN("invalid combination of fMask bits used\n");
4590 /* this does not happen on Win9x/ME */
4591 SetLastError( ERROR_INVALID_PARAMETER);
4595 /* Remove the old type bits and replace them with the new ones */
4596 menu->fType &= ~MENUITEMINFO_TYPE_MASK;
4597 menu->fType |= lpmii->fType & MENUITEMINFO_TYPE_MASK;
4599 if (IS_STRING_ITEM(menu->fType)) {
4600 HeapFree(GetProcessHeap(), 0, menu->text);
4601 set_menu_item_text( menu, lpmii->dwTypeData, unicode );
4602 } else if( (menu->fType) & MFT_BITMAP)
4603 menu->hbmpItem = HBITMAP_32(LOWORD(lpmii->dwTypeData));
4606 if (lpmii->fMask & MIIM_FTYPE ) {
4607 if(( lpmii->fType & MFT_BITMAP)) {
4608 SetLastError( ERROR_INVALID_PARAMETER);
4611 menu->fType &= ~MENUITEMINFO_TYPE_MASK;
4612 menu->fType |= lpmii->fType & MENUITEMINFO_TYPE_MASK;
4614 if (lpmii->fMask & MIIM_STRING ) {
4615 /* free the string when used */
4616 HeapFree(GetProcessHeap(), 0, menu->text);
4617 set_menu_item_text( menu, lpmii->dwTypeData, unicode );
4620 if (lpmii->fMask & MIIM_STATE)
4622 /* Other menu items having MFS_DEFAULT are not converted
4624 menu->fState = lpmii->fState & MENUITEMINFO_STATE_MASK;
4627 if (lpmii->fMask & MIIM_ID)
4628 menu->wID = lpmii->wID;
4630 if (lpmii->fMask & MIIM_SUBMENU) {
4631 menu->hSubMenu = lpmii->hSubMenu;
4632 if (menu->hSubMenu) {
4633 POPUPMENU *subMenu = MENU_GetMenu(menu->hSubMenu);
4635 subMenu->wFlags |= MF_POPUP;
4636 menu->fType |= MF_POPUP;
4639 SetLastError( ERROR_INVALID_PARAMETER);
4644 menu->fType &= ~MF_POPUP;
4647 if (lpmii->fMask & MIIM_CHECKMARKS)
4649 menu->hCheckBit = lpmii->hbmpChecked;
4650 menu->hUnCheckBit = lpmii->hbmpUnchecked;
4652 if (lpmii->fMask & MIIM_DATA)
4653 menu->dwItemData = lpmii->dwItemData;
4655 if (lpmii->fMask & MIIM_BITMAP)
4656 menu->hbmpItem = lpmii->hbmpItem;
4658 if( !menu->text && !(menu->fType & MFT_OWNERDRAW) && !menu->hbmpItem)
4659 menu->fType |= MFT_SEPARATOR;
4661 debug_print_menuitem("SetMenuItemInfo_common to : ", menu, "");
4665 /**********************************************************************
4666 * SetMenuItemInfoA (USER32.@)
4668 BOOL WINAPI SetMenuItemInfoA(HMENU hmenu, UINT item, BOOL bypos,
4669 const MENUITEMINFOA *lpmii)
4673 TRACE("hmenu %p, item %u, by pos %d, info %p\n", hmenu, item, bypos, lpmii);
4675 if( lpmii->cbSize != sizeof( mii) &&
4676 lpmii->cbSize != sizeof( mii) - sizeof ( mii.hbmpItem)) {
4677 SetLastError( ERROR_INVALID_PARAMETER);
4680 memcpy( &mii, lpmii, lpmii->cbSize);
4681 if( lpmii->cbSize != sizeof( mii)) {
4682 mii.cbSize = sizeof( mii);
4683 mii.hbmpItem = NULL;
4685 return SetMenuItemInfo_common(MENU_FindItem(&hmenu, &item, bypos? MF_BYPOSITION : 0),
4686 (const MENUITEMINFOW *)&mii, FALSE);
4689 /**********************************************************************
4690 * SetMenuItemInfoW (USER32.@)
4692 BOOL WINAPI SetMenuItemInfoW(HMENU hmenu, UINT item, BOOL bypos,
4693 const MENUITEMINFOW *lpmii)
4697 TRACE("hmenu %p, item %u, by pos %d, info %p\n", hmenu, item, bypos, lpmii);
4699 if( lpmii->cbSize != sizeof( mii) &&
4700 lpmii->cbSize != sizeof( mii) - sizeof ( mii.hbmpItem)) {
4701 SetLastError( ERROR_INVALID_PARAMETER);
4704 memcpy( &mii, lpmii, lpmii->cbSize);
4705 if( lpmii->cbSize != sizeof( mii)) {
4706 mii.cbSize = sizeof( mii);
4707 mii.hbmpItem = NULL;
4709 return SetMenuItemInfo_common(MENU_FindItem(&hmenu,
4710 &item, bypos? MF_BYPOSITION : 0), &mii, TRUE);
4713 /**********************************************************************
4714 * SetMenuDefaultItem (USER32.@)
4717 BOOL WINAPI SetMenuDefaultItem(HMENU hmenu, UINT uItem, UINT bypos)
4723 TRACE("(%p,%d,%d)\n", hmenu, uItem, bypos);
4725 if (!(menu = MENU_GetMenu(hmenu))) return FALSE;
4727 /* reset all default-item flags */
4729 for (i = 0; i < menu->nItems; i++, item++)
4731 item->fState &= ~MFS_DEFAULT;
4734 /* no default item */
4743 if ( uItem >= menu->nItems ) return FALSE;
4744 item[uItem].fState |= MFS_DEFAULT;
4749 for (i = 0; i < menu->nItems; i++, item++)
4751 if (item->wID == uItem)
4753 item->fState |= MFS_DEFAULT;
4762 /**********************************************************************
4763 * GetMenuDefaultItem (USER32.@)
4765 UINT WINAPI GetMenuDefaultItem(HMENU hmenu, UINT bypos, UINT flags)
4771 TRACE("(%p,%d,%d)\n", hmenu, bypos, flags);
4773 if (!(menu = MENU_GetMenu(hmenu))) return -1;
4775 /* find default item */
4779 if (! item) return -1;
4781 while ( !( item->fState & MFS_DEFAULT ) )
4784 if (i >= menu->nItems ) return -1;
4787 /* default: don't return disabled items */
4788 if ( (!(GMDI_USEDISABLED & flags)) && (item->fState & MFS_DISABLED )) return -1;
4790 /* search rekursiv when needed */
4791 if ( (item->fType & MF_POPUP) && (flags & GMDI_GOINTOPOPUPS) )
4794 ret = GetMenuDefaultItem( item->hSubMenu, bypos, flags );
4795 if ( -1 != ret ) return ret;
4797 /* when item not found in submenu, return the popup item */
4799 return ( bypos ) ? i : item->wID;
4804 /**********************************************************************
4805 * InsertMenuItemA (USER32.@)
4807 BOOL WINAPI InsertMenuItemA(HMENU hMenu, UINT uItem, BOOL bypos,
4808 const MENUITEMINFOA *lpmii)
4813 TRACE("hmenu %p, item %04x, by pos %d, info %p\n", hMenu, uItem, bypos, lpmii);
4815 if( lpmii->cbSize != sizeof( mii) &&
4816 lpmii->cbSize != sizeof( mii) - sizeof ( mii.hbmpItem)) {
4817 SetLastError( ERROR_INVALID_PARAMETER);
4820 memcpy( &mii, lpmii, lpmii->cbSize);
4821 if( lpmii->cbSize != sizeof( mii)) {
4822 mii.cbSize = sizeof( mii);
4823 mii.hbmpItem = NULL;
4826 item = MENU_InsertItem(hMenu, uItem, bypos ? MF_BYPOSITION : 0 );
4827 return SetMenuItemInfo_common(item, (const MENUITEMINFOW *)&mii, FALSE);
4831 /**********************************************************************
4832 * InsertMenuItemW (USER32.@)
4834 BOOL WINAPI InsertMenuItemW(HMENU hMenu, UINT uItem, BOOL bypos,
4835 const MENUITEMINFOW *lpmii)
4840 TRACE("hmenu %p, item %04x, by pos %d, info %p\n", hMenu, uItem, bypos, lpmii);
4842 if( lpmii->cbSize != sizeof( mii) &&
4843 lpmii->cbSize != sizeof( mii) - sizeof ( mii.hbmpItem)) {
4844 SetLastError( ERROR_INVALID_PARAMETER);
4847 memcpy( &mii, lpmii, lpmii->cbSize);
4848 if( lpmii->cbSize != sizeof( mii)) {
4849 mii.cbSize = sizeof( mii);
4850 mii.hbmpItem = NULL;
4853 item = MENU_InsertItem(hMenu, uItem, bypos ? MF_BYPOSITION : 0 );
4854 return SetMenuItemInfo_common(item, &mii, TRUE);
4857 /**********************************************************************
4858 * CheckMenuRadioItem (USER32.@)
4861 BOOL WINAPI CheckMenuRadioItem(HMENU hMenu,
4862 UINT first, UINT last, UINT check,
4867 MENUITEM *mi_first = NULL, *mi_check;
4868 HMENU m_first, m_check;
4870 for (i = first; i <= last; i++)
4877 mi_first = MENU_FindItem(&m_first, &pos, bypos);
4878 if (!mi_first) continue;
4879 mi_check = mi_first;
4885 mi_check = MENU_FindItem(&m_check, &pos, bypos);
4886 if (!mi_check) continue;
4889 if (m_first != m_check) continue;
4890 if (mi_check->fType == MFT_SEPARATOR) continue;
4894 mi_check->fType |= MFT_RADIOCHECK;
4895 mi_check->fState |= MFS_CHECKED;
4900 /* MSDN is wrong, Windows does not remove MFT_RADIOCHECK */
4901 mi_check->fState &= ~MFS_CHECKED;
4909 /**********************************************************************
4910 * GetMenuItemRect (USER32.@)
4912 * ATTENTION: Here, the returned values in rect are the screen
4913 * coordinates of the item just like if the menu was
4914 * always on the upper left side of the application.
4917 BOOL WINAPI GetMenuItemRect (HWND hwnd, HMENU hMenu, UINT uItem,
4920 POPUPMENU *itemMenu;
4924 TRACE("(%p,%p,%d,%p)\n", hwnd, hMenu, uItem, rect);
4926 item = MENU_FindItem (&hMenu, &uItem, MF_BYPOSITION);
4927 referenceHwnd = hwnd;
4931 itemMenu = MENU_GetMenu(hMenu);
4932 if (itemMenu == NULL)
4935 if(itemMenu->hWnd == 0)
4937 referenceHwnd = itemMenu->hWnd;
4940 if ((rect == NULL) || (item == NULL))
4945 MapWindowPoints(referenceHwnd, 0, (LPPOINT)rect, 2);
4951 /**********************************************************************
4952 * SetMenuInfo (USER32.@)
4955 * MIM_APPLYTOSUBMENUS
4956 * actually use the items to draw the menu
4958 BOOL WINAPI SetMenuInfo (HMENU hMenu, LPCMENUINFO lpmi)
4962 TRACE("(%p %p)\n", hMenu, lpmi);
4964 if (lpmi && (lpmi->cbSize==sizeof(MENUINFO)) && (menu = MENU_GetMenu(hMenu)))
4967 if (lpmi->fMask & MIM_BACKGROUND)
4968 menu->hbrBack = lpmi->hbrBack;
4970 if (lpmi->fMask & MIM_HELPID)
4971 menu->dwContextHelpID = lpmi->dwContextHelpID;
4973 if (lpmi->fMask & MIM_MAXHEIGHT)
4974 menu->cyMax = lpmi->cyMax;
4976 if (lpmi->fMask & MIM_MENUDATA)
4977 menu->dwMenuData = lpmi->dwMenuData;
4979 if (lpmi->fMask & MIM_STYLE)
4981 menu->dwStyle = lpmi->dwStyle;
4982 if (menu->dwStyle & MNS_AUTODISMISS) FIXME("MNS_AUTODISMISS unimplemented\n");
4983 if (menu->dwStyle & MNS_DRAGDROP) FIXME("MNS_DRAGDROP unimplemented\n");
4984 if (menu->dwStyle & MNS_MODELESS) FIXME("MNS_MODELESS unimplemented\n");
4992 /**********************************************************************
4993 * GetMenuInfo (USER32.@)
4999 BOOL WINAPI GetMenuInfo (HMENU hMenu, LPMENUINFO lpmi)
5002 TRACE("(%p %p)\n", hMenu, lpmi);
5004 if (lpmi && (menu = MENU_GetMenu(hMenu)))
5007 if (lpmi->fMask & MIM_BACKGROUND)
5008 lpmi->hbrBack = menu->hbrBack;
5010 if (lpmi->fMask & MIM_HELPID)
5011 lpmi->dwContextHelpID = menu->dwContextHelpID;
5013 if (lpmi->fMask & MIM_MAXHEIGHT)
5014 lpmi->cyMax = menu->cyMax;
5016 if (lpmi->fMask & MIM_MENUDATA)
5017 lpmi->dwMenuData = menu->dwMenuData;
5019 if (lpmi->fMask & MIM_STYLE)
5020 lpmi->dwStyle = menu->dwStyle;
5028 /**********************************************************************
5029 * SetMenuContextHelpId (USER32.@)
5031 BOOL WINAPI SetMenuContextHelpId( HMENU hMenu, DWORD dwContextHelpID)
5035 TRACE("(%p 0x%08x)\n", hMenu, dwContextHelpID);
5037 if ((menu = MENU_GetMenu(hMenu)))
5039 menu->dwContextHelpID = dwContextHelpID;
5046 /**********************************************************************
5047 * GetMenuContextHelpId (USER32.@)
5049 DWORD WINAPI GetMenuContextHelpId( HMENU hMenu )
5053 TRACE("(%p)\n", hMenu);
5055 if ((menu = MENU_GetMenu(hMenu)))
5057 return menu->dwContextHelpID;
5062 /**********************************************************************
5063 * MenuItemFromPoint (USER32.@)
5065 INT WINAPI MenuItemFromPoint(HWND hWnd, HMENU hMenu, POINT ptScreen)
5067 POPUPMENU *menu = MENU_GetMenu(hMenu);
5070 /*FIXME: Do we have to handle hWnd here? */
5071 if (!menu) return -1;
5072 if (!MENU_FindItemByCoords(menu, ptScreen, &pos)) return -1;
5077 /**********************************************************************
5078 * translate_accelerator
5080 static BOOL translate_accelerator( HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam,
5081 BYTE fVirt, WORD key, WORD cmd )
5086 if (wParam != key) return FALSE;
5088 if (GetKeyState(VK_CONTROL) & 0x8000) mask |= FCONTROL;
5089 if (GetKeyState(VK_MENU) & 0x8000) mask |= FALT;
5090 if (GetKeyState(VK_SHIFT) & 0x8000) mask |= FSHIFT;
5092 if (message == WM_CHAR || message == WM_SYSCHAR)
5094 if ( !(fVirt & FVIRTKEY) && (mask & FALT) == (fVirt & FALT) )
5096 TRACE_(accel)("found accel for WM_CHAR: ('%c')\n", LOWORD(wParam) & 0xff);
5102 if(fVirt & FVIRTKEY)
5104 TRACE_(accel)("found accel for virt_key %04lx (scan %04x)\n",
5105 wParam, 0xff & HIWORD(lParam));
5107 if(mask == (fVirt & (FSHIFT | FCONTROL | FALT))) goto found;
5108 TRACE_(accel)(", but incorrect SHIFT/CTRL/ALT-state\n");
5112 if (!(lParam & 0x01000000)) /* no special_key */
5114 if ((fVirt & FALT) && (lParam & 0x20000000))
5115 { /* ^^ ALT pressed */
5116 TRACE_(accel)("found accel for Alt-%c\n", LOWORD(wParam) & 0xff);
5125 if (message == WM_KEYUP || message == WM_SYSKEYUP)
5129 HMENU hMenu, hSubMenu, hSysMenu;
5130 UINT uSysStat = (UINT)-1, uStat = (UINT)-1, nPos;
5132 hMenu = (GetWindowLongW( hWnd, GWL_STYLE ) & WS_CHILD) ? 0 : GetMenu(hWnd);
5133 hSysMenu = get_win_sys_menu( hWnd );
5135 /* find menu item and ask application to initialize it */
5136 /* 1. in the system menu */
5137 hSubMenu = hSysMenu;
5139 if(MENU_FindItem(&hSubMenu, &nPos, MF_BYCOMMAND))
5143 if (!IsWindowEnabled(hWnd))
5147 SendMessageW(hWnd, WM_INITMENU, (WPARAM)hSysMenu, 0L);
5148 if(hSubMenu != hSysMenu)
5150 nPos = MENU_FindSubMenu(&hSysMenu, hSubMenu);
5151 TRACE_(accel)("hSysMenu = %p, hSubMenu = %p, nPos = %d\n", hSysMenu, hSubMenu, nPos);
5152 SendMessageW(hWnd, WM_INITMENUPOPUP, (WPARAM)hSubMenu, MAKELPARAM(nPos, TRUE));
5154 uSysStat = GetMenuState(GetSubMenu(hSysMenu, 0), cmd, MF_BYCOMMAND);
5157 else /* 2. in the window's menu */
5161 if(MENU_FindItem(&hSubMenu, &nPos, MF_BYCOMMAND))
5165 if (!IsWindowEnabled(hWnd))
5169 SendMessageW(hWnd, WM_INITMENU, (WPARAM)hMenu, 0L);
5170 if(hSubMenu != hMenu)
5172 nPos = MENU_FindSubMenu(&hMenu, hSubMenu);
5173 TRACE_(accel)("hMenu = %p, hSubMenu = %p, nPos = %d\n", hMenu, hSubMenu, nPos);
5174 SendMessageW(hWnd, WM_INITMENUPOPUP, (WPARAM)hSubMenu, MAKELPARAM(nPos, FALSE));
5176 uStat = GetMenuState(hMenu, cmd, MF_BYCOMMAND);
5183 if (uSysStat != (UINT)-1)
5185 if (uSysStat & (MF_DISABLED|MF_GRAYED))
5192 if (uStat != (UINT)-1)
5198 if (uStat & (MF_DISABLED|MF_GRAYED))
5210 if( mesg==WM_COMMAND )
5212 TRACE_(accel)(", sending WM_COMMAND, wParam=%0x\n", 0x10000 | cmd);
5213 SendMessageW(hWnd, mesg, 0x10000 | cmd, 0L);
5215 else if( mesg==WM_SYSCOMMAND )
5217 TRACE_(accel)(", sending WM_SYSCOMMAND, wParam=%0x\n", cmd);
5218 SendMessageW(hWnd, mesg, cmd, 0x00010000L);
5222 /* some reasons for NOT sending the WM_{SYS}COMMAND message:
5223 * #0: unknown (please report!)
5224 * #1: for WM_KEYUP,WM_SYSKEYUP
5225 * #2: mouse is captured
5226 * #3: window is disabled
5227 * #4: it's a disabled system menu option
5228 * #5: it's a menu option, but window is iconic
5229 * #6: it's a menu option, but disabled
5231 TRACE_(accel)(", but won't send WM_{SYS}COMMAND, reason is #%d\n",mesg);
5233 ERR_(accel)(" unknown reason - please report!\n");
5238 /**********************************************************************
5239 * TranslateAcceleratorA (USER32.@)
5240 * TranslateAccelerator (USER32.@)
5242 INT WINAPI TranslateAcceleratorA( HWND hWnd, HACCEL hAccel, LPMSG msg )
5245 LPACCEL16 lpAccelTbl;
5249 if (!hWnd || !msg) return 0;
5251 if (!hAccel || !(lpAccelTbl = (LPACCEL16) LockResource16(HACCEL_16(hAccel))))
5253 WARN_(accel)("invalid accel handle=%p\n", hAccel);
5257 wParam = msg->wParam;
5259 switch (msg->message)
5268 char ch = LOWORD(wParam);
5270 MultiByteToWideChar(CP_ACP, 0, &ch, 1, &wch, 1);
5271 wParam = MAKEWPARAM(wch, HIWORD(wParam));
5279 TRACE_(accel)("hAccel %p, hWnd %p, msg->hwnd %p, msg->message %04x, wParam %08lx, lParam %08lx\n",
5280 hAccel,hWnd,msg->hwnd,msg->message,msg->wParam,msg->lParam);
5284 if (translate_accelerator( hWnd, msg->message, wParam, msg->lParam,
5285 lpAccelTbl[i].fVirt, lpAccelTbl[i].key, lpAccelTbl[i].cmd))
5287 } while ((lpAccelTbl[i++].fVirt & 0x80) == 0);
5292 /**********************************************************************
5293 * TranslateAcceleratorW (USER32.@)
5295 INT WINAPI TranslateAcceleratorW( HWND hWnd, HACCEL hAccel, LPMSG msg )
5298 LPACCEL16 lpAccelTbl;
5301 if (!hWnd || !msg) return 0;
5303 if (!hAccel || !(lpAccelTbl = (LPACCEL16) LockResource16(HACCEL_16(hAccel))))
5305 WARN_(accel)("invalid accel handle=%p\n", hAccel);
5309 switch (msg->message)
5321 TRACE_(accel)("hAccel %p, hWnd %p, msg->hwnd %p, msg->message %04x, wParam %08lx, lParam %08lx\n",
5322 hAccel,hWnd,msg->hwnd,msg->message,msg->wParam,msg->lParam);
5326 if (translate_accelerator( hWnd, msg->message, msg->wParam, msg->lParam,
5327 lpAccelTbl[i].fVirt, lpAccelTbl[i].key, lpAccelTbl[i].cmd))
5329 } while ((lpAccelTbl[i++].fVirt & 0x80) == 0);