/* * COPYRIGHT: See COPYING in the top level directory * PROJECT: ReactOS kernel * PURPOSE: Menus * FILE: win32ss/user/ntuser/menu.c * PROGRAMER: Thomas Weidenmueller (w3seek@users.sourceforge.net) */ #include DBG_DEFAULT_CHANNEL(UserMenu); /* INTERNAL ******************************************************************/ BOOL FASTCALL UITOOLS95_DrawFrameMenu(HDC dc, LPRECT r, UINT uFlags); /* draw.c */ HFONT ghMenuFont = NULL; HFONT ghMenuFontBold = NULL; static SIZE MenuCharSize; /* Use global popup window because there's no way 2 menus can * be tracked at the same time. */ static HWND top_popup = NULL; static HMENU top_popup_hmenu = NULL; BOOL fInsideMenuLoop = FALSE; BOOL fInEndMenu = FALSE; /* internal popup menu window messages */ #define MM_SETMENUHANDLE (WM_USER + 0) #define MM_GETMENUHANDLE (WM_USER + 1) /* internal flags for menu tracking */ #define TF_ENDMENU 0x10000 #define TF_SUSPENDPOPUP 0x20000 #define TF_SKIPREMOVE 0x40000 /* maximum allowed depth of any branch in the menu tree. * This value is slightly larger than in windows (25) to * stay on the safe side. */ #define MAXMENUDEPTH 30 #define MNS_STYLE_MASK (MNS_NOCHECK|MNS_MODELESS|MNS_DRAGDROP|MNS_AUTODISMISS|MNS_NOTIFYBYPOS|MNS_CHECKORBMP) #define MENUITEMINFO_TYPE_MASK \ (MFT_STRING | MFT_BITMAP | MFT_OWNERDRAW | MFT_SEPARATOR | \ MFT_MENUBARBREAK | MFT_MENUBREAK | MFT_RADIOCHECK | \ MFT_RIGHTORDER | MFT_RIGHTJUSTIFY /* same as MF_HELP */ ) #define TYPE_MASK (MENUITEMINFO_TYPE_MASK | MF_POPUP | MF_SYSMENU) #define STATE_MASK (~TYPE_MASK) #define MENUITEMINFO_STATE_MASK (STATE_MASK & ~(MF_BYPOSITION | MF_MOUSESELECT)) #define MII_STATE_MASK (MFS_GRAYED|MFS_CHECKED|MFS_HILITE|MFS_DEFAULT) #define IS_SYSTEM_MENU(MenuInfo) \ (!!((MenuInfo)->fFlags & MNF_SYSMENU)) #define IS_MAGIC_BITMAP(id) ((id) && ((INT_PTR)(id) < 12) && ((INT_PTR)(id) >= -1)) #define IS_STRING_ITEM(flags) (MF_STRING == MENU_ITEM_TYPE(flags)) /* Maximum number of menu items a menu can contain */ #define MAX_MENU_ITEMS (0x4000) #define MAX_GOINTOSUBMENU (0x10) /* Space between 2 columns */ #define MENU_COL_SPACE 4 #define MENU_ITEM_HBMP_SPACE (5) #define MENU_BAR_ITEMS_SPACE (12) #define SEPARATOR_HEIGHT (5) #define MENU_TAB_SPACE (8) typedef struct { UINT TrackFlags; PMENU CurrentMenu; /* current submenu (can be equal to hTopMenu)*/ PMENU TopMenu; /* initial menu */ PWND OwnerWnd; /* where notifications are sent */ POINT Pt; } MTRACKER; /* Internal MenuTrackMenu() flags */ #define TPM_INTERNAL 0xF0000000 #define TPM_BUTTONDOWN 0x40000000 /* menu was clicked before tracking */ #define TPM_POPUPMENU 0x20000000 /* menu is a popup menu */ #define ITEM_PREV -1 #define ITEM_NEXT 1 #define UpdateMenuItemState(state, change) \ {\ if((change) & MF_GRAYED) { \ (state) |= MF_GRAYED; \ } else { \ (state) &= ~MF_GRAYED; \ } /* Separate the two for test_menu_resource_layout.*/ \ if((change) & MF_DISABLED) { \ (state) |= MF_DISABLED; \ } else { \ (state) &= ~MF_DISABLED; \ } \ if((change) & MFS_CHECKED) { \ (state) |= MFS_CHECKED; \ } else { \ (state) &= ~MFS_CHECKED; \ } \ if((change) & MFS_HILITE) { \ (state) |= MFS_HILITE; \ } else { \ (state) &= ~MFS_HILITE; \ } \ if((change) & MFS_DEFAULT) { \ (state) |= MFS_DEFAULT; \ } else { \ (state) &= ~MFS_DEFAULT; \ } \ if((change) & MF_MOUSESELECT) { \ (state) |= MF_MOUSESELECT; \ } else { \ (state) &= ~MF_MOUSESELECT; \ } \ } #if 0 void FASTCALL DumpMenuItemList(PMENU Menu, PITEM MenuItem) { UINT cnt = 0, i = Menu->cItems; while(i) { if(MenuItem->lpstr.Length) DbgPrint(" %d. %wZ\n", ++cnt, &MenuItem->lpstr); else DbgPrint(" %d. NO TEXT dwTypeData==%d\n", ++cnt, (DWORD)MenuItem->lpstr.Buffer); DbgPrint(" fType="); if(MFT_BITMAP & MenuItem->fType) DbgPrint("MFT_BITMAP "); if(MFT_MENUBARBREAK & MenuItem->fType) DbgPrint("MFT_MENUBARBREAK "); if(MFT_MENUBREAK & MenuItem->fType) DbgPrint("MFT_MENUBREAK "); if(MFT_OWNERDRAW & MenuItem->fType) DbgPrint("MFT_OWNERDRAW "); if(MFT_RADIOCHECK & MenuItem->fType) DbgPrint("MFT_RADIOCHECK "); if(MFT_RIGHTJUSTIFY & MenuItem->fType) DbgPrint("MFT_RIGHTJUSTIFY "); if(MFT_SEPARATOR & MenuItem->fType) DbgPrint("MFT_SEPARATOR "); if(MFT_STRING & MenuItem->fType) DbgPrint("MFT_STRING "); DbgPrint("\n fState="); if(MFS_DISABLED & MenuItem->fState) DbgPrint("MFS_DISABLED "); else DbgPrint("MFS_ENABLED "); if(MFS_CHECKED & MenuItem->fState) DbgPrint("MFS_CHECKED "); else DbgPrint("MFS_UNCHECKED "); if(MFS_HILITE & MenuItem->fState) DbgPrint("MFS_HILITE "); else DbgPrint("MFS_UNHILITE "); if(MFS_DEFAULT & MenuItem->fState) DbgPrint("MFS_DEFAULT "); if(MFS_GRAYED & MenuItem->fState) DbgPrint("MFS_GRAYED "); DbgPrint("\n wId=%d\n", MenuItem->wID); MenuItem++; i--; } DbgPrint("Entries: %d\n", cnt); return; } #endif #define FreeMenuText(Menu,MenuItem) \ { \ if((MENU_ITEM_TYPE((MenuItem)->fType) == MF_STRING) && \ (MenuItem)->lpstr.Length) { \ DesktopHeapFree(((PMENU)Menu)->head.rpdesk, (MenuItem)->lpstr.Buffer); \ } \ } PMENU FASTCALL IntGetMenuObject(HMENU hMenu) { PMENU Menu = UserGetMenuObject(hMenu); if (Menu) Menu->head.cLockObj++; return Menu; } PMENU FASTCALL VerifyMenu(PMENU pMenu) { HMENU hMenu; PITEM pItem; ULONG Error; UINT i; if (!pMenu) return NULL; Error = EngGetLastError(); _SEH2_TRY { hMenu = UserHMGetHandle(pMenu); pItem = pMenu->rgItems; if (pItem) { i = pItem[0].wID; pItem[0].wID = i; } } _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) { ERR("Run away LOOP!\n"); EngSetLastError(Error); _SEH2_YIELD(return NULL); } _SEH2_END if ( UserObjectInDestroy(hMenu)) { ERR("Menu is marked for destruction!\n"); pMenu = NULL; } EngSetLastError(Error); return pMenu; } BOOL FASTCALL IntIsMenu(HMENU Menu) { if (UserGetMenuObject(Menu)) return TRUE; return FALSE; } PMENU WINAPI IntGetMenu(HWND hWnd) { PWND Wnd = ValidateHwndNoErr(hWnd); if (!Wnd) return NULL; return UserGetMenuObject(UlongToHandle(Wnd->IDMenu)); } PMENU get_win_sys_menu( HWND hwnd ) { PMENU ret = 0; WND *win = ValidateHwndNoErr( hwnd ); if (win) { ret = UserGetMenuObject(win->SystemMenu); } return ret; } BOOL IntDestroyMenu( PMENU pMenu, BOOL bRecurse) { PMENU SubMenu; ASSERT(UserIsEnteredExclusive()); if (pMenu->rgItems) /* recursively destroy submenus */ { int i; ITEM *item = pMenu->rgItems; for (i = pMenu->cItems; i > 0; i--, item++) { SubMenu = item->spSubMenu; item->spSubMenu = NULL; /* Remove Item Text */ FreeMenuText(pMenu,item); /* Remove Item Bitmap and set it for this process */ if (item->hbmp && !(item->fState & MFS_HBMMENUBMP)) { GreSetObjectOwner(item->hbmp, GDI_OBJ_HMGR_POWNED); item->hbmp = NULL; } /* Remove Item submenu */ if (bRecurse && SubMenu)//VerifyMenu(SubMenu)) { /* Release submenu since it was referenced when inserted */ IntReleaseMenuObject(SubMenu); IntDestroyMenuObject(SubMenu, bRecurse); } } /* Free the Item */ DesktopHeapFree(pMenu->head.rpdesk, pMenu->rgItems ); pMenu->rgItems = NULL; pMenu->cItems = 0; } return TRUE; } /* Callback for the object manager */ BOOLEAN UserDestroyMenuObject(PVOID Object) { return IntDestroyMenuObject(Object, TRUE); } BOOL FASTCALL IntDestroyMenuObject(PMENU Menu, BOOL bRecurse) { ASSERT(UserIsEnteredExclusive()); if (Menu) { PWND Window; if (PsGetCurrentProcessSessionId() == Menu->head.rpdesk->rpwinstaParent->dwSessionId) { BOOL ret; if (Menu->hWnd) { Window = ValidateHwndNoErr(Menu->hWnd); if (Window) { //Window->IDMenu = 0; Only in Win9x!! wine win test_SetMenu test... /* DestroyMenu should not destroy system menu popup owner */ if ((Menu->fFlags & (MNF_POPUP | MNF_SYSSUBMENU)) == MNF_POPUP) { // Should we check it to see if it has Class? ERR("FIXME Pop up menu window thing'ie\n"); //co_UserDestroyWindow( Window ); //Menu->hWnd = 0; } } } if (!UserMarkObjectDestroy(Menu)) return TRUE; /* Remove all menu items */ IntDestroyMenu( Menu, bRecurse); ret = UserDeleteObject(UserHMGetHandle(Menu), TYPE_MENU); TRACE("IntDestroyMenuObject %d\n",ret); return ret; } } return FALSE; } BOOL MenuInit(VOID) { NONCLIENTMETRICSW ncm; /* get the menu font */ if (!ghMenuFont || !ghMenuFontBold) { ncm.cbSize = sizeof(ncm); if(!UserSystemParametersInfo(SPI_GETNONCLIENTMETRICS, sizeof(ncm), &ncm, 0)) { ERR("MenuInit(): SystemParametersInfo(SPI_GETNONCLIENTMETRICS) failed!\n"); return FALSE; } ghMenuFont = GreCreateFontIndirectW(&ncm.lfMenuFont); if (ghMenuFont == NULL) { ERR("MenuInit(): CreateFontIndirectW(hMenuFont) failed!\n"); return FALSE; } ncm.lfMenuFont.lfWeight = min(ncm.lfMenuFont.lfWeight + (FW_BOLD - FW_NORMAL), FW_HEAVY); ghMenuFontBold = GreCreateFontIndirectW(&ncm.lfMenuFont); if (ghMenuFontBold == NULL) { ERR("MenuInit(): CreateFontIndirectW(hMenuFontBold) failed!\n"); GreDeleteObject(ghMenuFont); ghMenuFont = NULL; return FALSE; } GreSetObjectOwner(ghMenuFont, GDI_OBJ_HMGR_PUBLIC); GreSetObjectOwner(ghMenuFontBold, GDI_OBJ_HMGR_PUBLIC); co_IntSetupOBM(); } return TRUE; } /********************************************************************** * MENU_depth * * detect if there are loops in the menu tree (or the depth is too large) */ int FASTCALL MENU_depth( PMENU pmenu, int depth) { UINT i; ITEM *item; int subdepth; if (!pmenu) return depth; depth++; if( depth > MAXMENUDEPTH) return depth; item = pmenu->rgItems; subdepth = depth; for( i = 0; i < pmenu->cItems && subdepth <= MAXMENUDEPTH; i++, item++) { if( item->spSubMenu)//VerifyMenu(item->spSubMenu)) { int bdepth = MENU_depth( item->spSubMenu, depth); if( bdepth > subdepth) subdepth = bdepth; } if( subdepth > MAXMENUDEPTH) TRACE("<- hmenu %p\n", item->spSubMenu); } return subdepth; } /****************************************************************************** * * UINT MenuGetStartOfNextColumn( * PMENU Menu) * *****************************************************************************/ static UINT MENU_GetStartOfNextColumn( PMENU menu ) { PITEM pItem; UINT i; if(!menu) return NO_SELECTED_ITEM; i = menu->iItem + 1; if( i == NO_SELECTED_ITEM ) return i; pItem = menu->rgItems; if (!pItem) return NO_SELECTED_ITEM; for( ; i < menu->cItems; ++i ) { if (pItem[i].fType & (MF_MENUBREAK | MF_MENUBARBREAK)) return i; } return NO_SELECTED_ITEM; } /****************************************************************************** * * UINT MenuGetStartOfPrevColumn( * PMENU Menu) * *****************************************************************************/ static UINT MENU_GetStartOfPrevColumn( PMENU menu ) { UINT i; PITEM pItem; if( !menu ) return NO_SELECTED_ITEM; if( menu->iItem == 0 || menu->iItem == NO_SELECTED_ITEM ) return NO_SELECTED_ITEM; pItem = menu->rgItems; if (!pItem) return NO_SELECTED_ITEM; /* Find the start of the column */ for(i = menu->iItem; i != 0 && !(pItem[i].fType & (MF_MENUBREAK | MF_MENUBARBREAK)); --i); /* empty */ if(i == 0) return NO_SELECTED_ITEM; for(--i; i != 0; --i) { if (pItem[i].fType & (MF_MENUBREAK | MF_MENUBARBREAK)) break; } TRACE("ret %d.\n", i ); return i; } /*********************************************************************** * MENU_FindItem * * Find a menu item. Return a pointer on the item, and modifies *hmenu * in case the item was in a sub-menu. */ PITEM FASTCALL MENU_FindItem( PMENU *pmenu, UINT *nPos, UINT wFlags ) { MENU *menu = *pmenu; ITEM *fallback = NULL; UINT fallback_pos = 0; UINT i; if (!menu) return NULL; if (wFlags & MF_BYPOSITION) { if (!menu->cItems) return NULL; if (*nPos >= menu->cItems) return NULL; return &menu->rgItems[*nPos]; } else { PITEM item = menu->rgItems; for (i = 0; i < menu->cItems; i++, item++) { if (item->spSubMenu) { PMENU psubmenu = item->spSubMenu;//VerifyMenu(item->spSubMenu); PITEM subitem = MENU_FindItem( &psubmenu, nPos, wFlags ); if (subitem) { *pmenu = psubmenu; return subitem; } else if (item->wID == *nPos) { /* fallback to this item if nothing else found */ fallback_pos = i; fallback = item; } } else if (item->wID == *nPos) { *nPos = i; return item; } } } if (fallback) *nPos = fallback_pos; return fallback; } /*********************************************************************** * MenuFindSubMenu * * Find a Sub menu. Return the position of the submenu, and modifies * *hmenu in case it is found in another sub-menu. * If the submenu cannot be found, NO_SELECTED_ITEM is returned. */ static UINT FASTCALL MENU_FindSubMenu(PMENU *menu, PMENU SubTarget ) { UINT i; PITEM item; item = ((PMENU)*menu)->rgItems; for (i = 0; i < ((PMENU)*menu)->cItems; i++, item++) { if (!item->spSubMenu) continue; else { if (item->spSubMenu == SubTarget) { return i; } else { PMENU pSubMenu = item->spSubMenu; UINT pos = MENU_FindSubMenu( &pSubMenu, SubTarget ); if (pos != NO_SELECTED_ITEM) { *menu = pSubMenu; return pos; } } } } return NO_SELECTED_ITEM; } BOOL FASTCALL IntRemoveMenuItem( PMENU pMenu, UINT nPos, UINT wFlags, BOOL bRecurse ) { PITEM item; PITEM newItems; TRACE("(menu=%p pos=%04x flags=%04x)\n",pMenu, nPos, wFlags); if (!(item = MENU_FindItem( &pMenu, &nPos, wFlags ))) return FALSE; /* Remove item */ FreeMenuText(pMenu,item); if (bRecurse && item->spSubMenu) { IntDestroyMenuObject(item->spSubMenu, bRecurse); } ////// Use cAlloced with inc's of 8's.... if (--pMenu->cItems == 0) { DesktopHeapFree(pMenu->head.rpdesk, pMenu->rgItems ); pMenu->rgItems = NULL; } else { while (nPos < pMenu->cItems) { *item = *(item+1); item++; nPos++; } newItems = DesktopHeapReAlloc(pMenu->head.rpdesk, pMenu->rgItems, pMenu->cItems * sizeof(ITEM)); if (newItems) { pMenu->rgItems = newItems; } } return TRUE; } /********************************************************************** * MENU_InsertItem * * Insert (allocate) a new item into a menu. */ ITEM *MENU_InsertItem( PMENU menu, UINT pos, UINT flags, PMENU *submenu, UINT *npos ) { ITEM *newItems; /* Find where to insert new item */ if (flags & MF_BYPOSITION) { if (pos > menu->cItems) pos = menu->cItems; } else { if (!MENU_FindItem( &menu, &pos, flags )) { if (submenu) *submenu = menu; if (npos) *npos = pos; pos = menu->cItems; } } /* Make sure that MDI system buttons stay on the right side. * Note: XP treats only bitmap handles 1 - 6 as "magic" ones * regardless of their id. */ while ( pos > 0 && (INT_PTR)menu->rgItems[pos - 1].hbmp >= (INT_PTR)HBMMENU_SYSTEM && (INT_PTR)menu->rgItems[pos - 1].hbmp <= (INT_PTR)HBMMENU_MBAR_CLOSE_D) pos--; TRACE("inserting at %u flags %x\n", pos, flags); /* Create new items array */ newItems = DesktopHeapAlloc(menu->head.rpdesk, sizeof(ITEM) * (menu->cItems+1) ); if (!newItems) { WARN("allocation failed\n" ); return NULL; } if (menu->cItems > 0) { /* Copy the old array into the new one */ if (pos > 0) RtlCopyMemory( newItems, menu->rgItems, pos * sizeof(ITEM) ); if (pos < menu->cItems) RtlCopyMemory( &newItems[pos+1], &menu->rgItems[pos], (menu->cItems-pos)*sizeof(ITEM) ); DesktopHeapFree(menu->head.rpdesk, menu->rgItems ); } menu->rgItems = newItems; menu->cItems++; RtlZeroMemory( &newItems[pos], sizeof(*newItems) ); menu->cyMenu = 0; /* force size recalculate */ return &newItems[pos]; } BOOL FASTCALL IntInsertMenuItem( _In_ PMENU MenuObject, UINT uItem, BOOL fByPosition, PROSMENUITEMINFO ItemInfo, PUNICODE_STRING lpstr) { PITEM MenuItem; PMENU SubMenu = NULL; NT_ASSERT(MenuObject != NULL); if (MAX_MENU_ITEMS <= MenuObject->cItems) { EngSetLastError(ERROR_NOT_ENOUGH_MEMORY); return FALSE; } SubMenu = MenuObject; if(!(MenuItem = MENU_InsertItem( SubMenu, uItem, fByPosition ? MF_BYPOSITION : MF_BYCOMMAND, &SubMenu, &uItem ))) return FALSE; if(!IntSetMenuItemInfo(SubMenu, MenuItem, ItemInfo, lpstr)) { IntRemoveMenuItem(SubMenu, uItem, fByPosition ? MF_BYPOSITION : MF_BYCOMMAND, FALSE); return FALSE; } /* Force size recalculation! */ SubMenu->cyMenu = 0; MenuItem->hbmpChecked = MenuItem->hbmpUnchecked = 0; TRACE("IntInsertMenuItemToList = %u %i\n", uItem, (BOOL)((INT)uItem >= 0)); return TRUE; } PMENU FASTCALL IntCreateMenu( _Out_ PHANDLE Handle, _In_ BOOL IsMenuBar, _In_ PDESKTOP Desktop, _In_ PPROCESSINFO ppi) { PMENU Menu; Menu = (PMENU)UserCreateObject( gHandleTable, Desktop, ppi->ptiList, Handle, TYPE_MENU, sizeof(MENU)); if(!Menu) { *Handle = 0; return NULL; } Menu->cyMax = 0; /* Default */ Menu->hbrBack = NULL; /* No brush */ Menu->dwContextHelpId = 0; /* Default */ Menu->dwMenuData = 0; /* Default */ Menu->iItem = NO_SELECTED_ITEM; // Focused item Menu->fFlags = (IsMenuBar ? 0 : MNF_POPUP); Menu->spwndNotify = NULL; Menu->cyMenu = 0; // Height Menu->cxMenu = 0; // Width Menu->cItems = 0; // Item count Menu->iTop = 0; Menu->iMaxTop = 0; Menu->cxTextAlign = 0; Menu->rgItems = NULL; Menu->hWnd = NULL; Menu->TimeToHide = FALSE; return Menu; } BOOL FASTCALL IntCloneMenuItems(PMENU Destination, PMENU Source) { PITEM MenuItem, NewMenuItem = NULL; UINT i; if(!Source->cItems) return FALSE; NewMenuItem = DesktopHeapAlloc(Destination->head.rpdesk, Source->cItems * sizeof(ITEM)); if(!NewMenuItem) return FALSE; RtlZeroMemory(NewMenuItem, Source->cItems * sizeof(ITEM)); Destination->rgItems = NewMenuItem; MenuItem = Source->rgItems; for (i = 0; i < Source->cItems; i++, MenuItem++, NewMenuItem++) { NewMenuItem->fType = MenuItem->fType; NewMenuItem->fState = MenuItem->fState; NewMenuItem->wID = MenuItem->wID; NewMenuItem->spSubMenu = MenuItem->spSubMenu; NewMenuItem->hbmpChecked = MenuItem->hbmpChecked; NewMenuItem->hbmpUnchecked = MenuItem->hbmpUnchecked; NewMenuItem->dwItemData = MenuItem->dwItemData; if (MenuItem->lpstr.Length) { NewMenuItem->lpstr.Length = 0; NewMenuItem->lpstr.MaximumLength = MenuItem->lpstr.MaximumLength; NewMenuItem->lpstr.Buffer = DesktopHeapAlloc(Destination->head.rpdesk, MenuItem->lpstr.MaximumLength); if (!NewMenuItem->lpstr.Buffer) { DesktopHeapFree(Destination->head.rpdesk, NewMenuItem); break; } RtlCopyUnicodeString(&NewMenuItem->lpstr, &MenuItem->lpstr); NewMenuItem->lpstr.Buffer[MenuItem->lpstr.Length / sizeof(WCHAR)] = 0; NewMenuItem->Xlpstr = NewMenuItem->lpstr.Buffer; } else { NewMenuItem->lpstr.Buffer = MenuItem->lpstr.Buffer; NewMenuItem->Xlpstr = NewMenuItem->lpstr.Buffer; } NewMenuItem->hbmp = MenuItem->hbmp; Destination->cItems = i + 1; } return TRUE; } PMENU FASTCALL IntCloneMenu(PMENU Source) { HANDLE hMenu; PMENU Menu; if(!Source) return NULL; /* A menu is valid process wide. We can pass to the object manager any thread ptr */ Menu = (PMENU)UserCreateObject( gHandleTable, Source->head.rpdesk, ((PPROCESSINFO)Source->head.hTaskWow)->ptiList, &hMenu, TYPE_MENU, sizeof(MENU)); if(!Menu) return NULL; Menu->fFlags = Source->fFlags; Menu->cyMax = Source->cyMax; Menu->hbrBack = Source->hbrBack; Menu->dwContextHelpId = Source->dwContextHelpId; Menu->dwMenuData = Source->dwMenuData; Menu->iItem = NO_SELECTED_ITEM; Menu->spwndNotify = NULL; Menu->cyMenu = 0; Menu->cxMenu = 0; Menu->cItems = 0; Menu->iTop = 0; Menu->iMaxTop = 0; Menu->cxTextAlign = 0; Menu->rgItems = NULL; Menu->hWnd = NULL; Menu->TimeToHide = FALSE; IntCloneMenuItems(Menu, Source); return Menu; } BOOL FASTCALL IntSetMenuFlagRtoL(PMENU Menu) { ERR("SetMenuFlagRtoL\n"); Menu->fFlags |= MNF_RTOL; return TRUE; } BOOL FASTCALL IntSetMenuContextHelpId(PMENU Menu, DWORD dwContextHelpId) { Menu->dwContextHelpId = dwContextHelpId; return TRUE; } BOOL FASTCALL IntGetMenuInfo(PMENU Menu, PROSMENUINFO lpmi) { if(lpmi->fMask & MIM_BACKGROUND) lpmi->hbrBack = Menu->hbrBack; if(lpmi->fMask & MIM_HELPID) lpmi->dwContextHelpID = Menu->dwContextHelpId; if(lpmi->fMask & MIM_MAXHEIGHT) lpmi->cyMax = Menu->cyMax; if(lpmi->fMask & MIM_MENUDATA) lpmi->dwMenuData = Menu->dwMenuData; if(lpmi->fMask & MIM_STYLE) lpmi->dwStyle = Menu->fFlags & MNS_STYLE_MASK; if (sizeof(MENUINFO) < lpmi->cbSize) { lpmi->cItems = Menu->cItems; lpmi->iItem = Menu->iItem; lpmi->cxMenu = Menu->cxMenu; lpmi->cyMenu = Menu->cyMenu; lpmi->spwndNotify = Menu->spwndNotify; lpmi->cxTextAlign = Menu->cxTextAlign; lpmi->iTop = Menu->iTop; lpmi->iMaxTop = Menu->iMaxTop; lpmi->dwArrowsOn = Menu->dwArrowsOn; lpmi->fFlags = Menu->fFlags; lpmi->Self = UserHMGetHandle(Menu); lpmi->TimeToHide = Menu->TimeToHide; lpmi->Wnd = Menu->hWnd; } return TRUE; } BOOL FASTCALL IntSetMenuInfo(PMENU Menu, PROSMENUINFO lpmi) { if(lpmi->fMask & MIM_BACKGROUND) Menu->hbrBack = lpmi->hbrBack; if(lpmi->fMask & MIM_HELPID) Menu->dwContextHelpId = lpmi->dwContextHelpID; if(lpmi->fMask & MIM_MAXHEIGHT) Menu->cyMax = lpmi->cyMax; if(lpmi->fMask & MIM_MENUDATA) Menu->dwMenuData = lpmi->dwMenuData; if(lpmi->fMask & MIM_STYLE) Menu->fFlags ^= (Menu->fFlags ^ lpmi->dwStyle) & MNS_STYLE_MASK; if(lpmi->fMask & MIM_APPLYTOSUBMENUS) { int i; PITEM item = Menu->rgItems; for ( i = Menu->cItems; i; i--, item++) { if ( item->spSubMenu ) { IntSetMenuInfo( item->spSubMenu, lpmi); } } } if (sizeof(MENUINFO) < lpmi->cbSize) { Menu->iItem = lpmi->iItem; Menu->cyMenu = lpmi->cyMenu; Menu->cxMenu = lpmi->cxMenu; Menu->spwndNotify = lpmi->spwndNotify; Menu->cxTextAlign = lpmi->cxTextAlign; Menu->iTop = lpmi->iTop; Menu->iMaxTop = lpmi->iMaxTop; Menu->dwArrowsOn = lpmi->dwArrowsOn; Menu->TimeToHide = lpmi->TimeToHide; Menu->hWnd = lpmi->Wnd; } if ( lpmi->fMask & MIM_STYLE) { if (lpmi->dwStyle & MNS_AUTODISMISS) FIXME("MNS_AUTODISMISS unimplemented wine\n"); if (lpmi->dwStyle & MNS_DRAGDROP) FIXME("MNS_DRAGDROP unimplemented wine\n"); if (lpmi->dwStyle & MNS_MODELESS) FIXME("MNS_MODELESS unimplemented wine\n"); } return TRUE; } BOOL FASTCALL IntGetMenuItemInfo(PMENU Menu, /* UNUSED PARAM!! */ PITEM MenuItem, PROSMENUITEMINFO lpmii) { NTSTATUS Status; if(lpmii->fMask & (MIIM_FTYPE | MIIM_TYPE)) { lpmii->fType = MenuItem->fType; } if(lpmii->fMask & MIIM_BITMAP) { lpmii->hbmpItem = MenuItem->hbmp; } if(lpmii->fMask & MIIM_CHECKMARKS) { lpmii->hbmpChecked = MenuItem->hbmpChecked; lpmii->hbmpUnchecked = MenuItem->hbmpUnchecked; } if(lpmii->fMask & MIIM_DATA) { lpmii->dwItemData = MenuItem->dwItemData; } if(lpmii->fMask & MIIM_ID) { lpmii->wID = MenuItem->wID; } if(lpmii->fMask & MIIM_STATE) { lpmii->fState = MenuItem->fState; } if(lpmii->fMask & MIIM_SUBMENU) { lpmii->hSubMenu = (MenuItem->spSubMenu ? UserHMGetHandle(MenuItem->spSubMenu) : NULL); } if ((lpmii->fMask & MIIM_STRING) || ((lpmii->fMask & MIIM_TYPE) && (MENU_ITEM_TYPE(lpmii->fType) == MF_STRING))) { if (lpmii->dwTypeData == NULL) { lpmii->cch = MenuItem->lpstr.Length / sizeof(WCHAR); } else { //// lpmii->lpstr can be read in user mode!!!! Status = MmCopyToCaller(lpmii->dwTypeData, MenuItem->lpstr.Buffer, min(lpmii->cch * sizeof(WCHAR), MenuItem->lpstr.MaximumLength)); if (! NT_SUCCESS(Status)) { SetLastNtError(Status); return FALSE; } } } if (sizeof(ROSMENUITEMINFO) == lpmii->cbSize) { lpmii->Rect.left = MenuItem->xItem; lpmii->Rect.top = MenuItem->yItem; lpmii->Rect.right = MenuItem->cxItem; // Do this for now...... lpmii->Rect.bottom = MenuItem->cyItem; lpmii->dxTab = MenuItem->dxTab; lpmii->lpstr = MenuItem->lpstr.Buffer; lpmii->maxBmpSize.cx = MenuItem->cxBmp; lpmii->maxBmpSize.cy = MenuItem->cyBmp; } return TRUE; } BOOL FASTCALL IntSetMenuItemInfo(PMENU MenuObject, PITEM MenuItem, PROSMENUITEMINFO lpmii, PUNICODE_STRING lpstr) { PMENU SubMenuObject; BOOL circref = FALSE; if(!MenuItem || !MenuObject || !lpmii) { return FALSE; } if ( lpmii->fMask & MIIM_FTYPE ) { MenuItem->fType &= ~MENUITEMINFO_TYPE_MASK; MenuItem->fType |= lpmii->fType & MENUITEMINFO_TYPE_MASK; } if (lpmii->fMask & MIIM_TYPE) { #if 0 //// Done in User32. if (lpmii->fMask & ( MIIM_STRING | MIIM_FTYPE | MIIM_BITMAP)) { ERR("IntSetMenuItemInfo: Invalid combination of fMask bits used\n"); KeRosDumpStackFrames(NULL, 20); /* This does not happen on Win9x/ME */ SetLastNtError( ERROR_INVALID_PARAMETER); return FALSE; } #endif /* * Delete the menu item type when changing type from * MF_STRING. */ if (MenuItem->fType != lpmii->fType && MENU_ITEM_TYPE(MenuItem->fType) == MFT_STRING) { FreeMenuText(MenuObject,MenuItem); RtlInitUnicodeString(&MenuItem->lpstr, NULL); MenuItem->Xlpstr = NULL; } if(lpmii->fType & MFT_BITMAP) { if(lpmii->hbmpItem) MenuItem->hbmp = lpmii->hbmpItem; else { /* Win 9x/Me stuff */ MenuItem->hbmp = (HBITMAP)((ULONG_PTR)(LOWORD(lpmii->dwTypeData))); } lpmii->dwTypeData = 0; } } if(lpmii->fMask & MIIM_BITMAP) { MenuItem->hbmp = lpmii->hbmpItem; if (MenuItem->hbmp <= HBMMENU_POPUP_MINIMIZE && MenuItem->hbmp >= HBMMENU_CALLBACK) MenuItem->fState |= MFS_HBMMENUBMP; else MenuItem->fState &= ~MFS_HBMMENUBMP; } if(lpmii->fMask & MIIM_CHECKMARKS) { MenuItem->hbmpChecked = lpmii->hbmpChecked; MenuItem->hbmpUnchecked = lpmii->hbmpUnchecked; } if(lpmii->fMask & MIIM_DATA) { MenuItem->dwItemData = lpmii->dwItemData; } if(lpmii->fMask & MIIM_ID) { MenuItem->wID = lpmii->wID; } if(lpmii->fMask & MIIM_STATE) { /* Remove MFS_DEFAULT flag from all other menu items if this item has the MFS_DEFAULT state */ if(lpmii->fState & MFS_DEFAULT) UserSetMenuDefaultItem(MenuObject, -1, 0); /* Update the menu item state flags */ UpdateMenuItemState(MenuItem->fState, lpmii->fState); } if(lpmii->fMask & MIIM_SUBMENU) { if (lpmii->hSubMenu) { SubMenuObject = UserGetMenuObject(lpmii->hSubMenu); if ( SubMenuObject && !(UserObjectInDestroy(lpmii->hSubMenu)) ) { //// wine Bug 12171 : Adding Popup Menu to itself! Could create endless loops. //// CORE-7967. if (MenuObject == SubMenuObject) { HANDLE hMenu; ERR("Pop Up Menu Double Trouble!\n"); SubMenuObject = IntCreateMenu(&hMenu, FALSE, MenuObject->head.rpdesk, (PPROCESSINFO)MenuObject->head.hTaskWow); // It will be marked. if (!SubMenuObject) return FALSE; IntReleaseMenuObject(SubMenuObject); // This will be referenced again after insertion. circref = TRUE; } if ( MENU_depth( SubMenuObject, 0) > MAXMENUDEPTH ) { ERR( "Loop detected in menu hierarchy or maximum menu depth exceeded!\n"); if (circref) IntDestroyMenuObject(SubMenuObject, FALSE); return FALSE; } /* Make sure the submenu is marked as a popup menu */ SubMenuObject->fFlags |= MNF_POPUP; // Now fix the test_subpopup_locked_by_menu tests.... if (MenuItem->spSubMenu) IntReleaseMenuObject(MenuItem->spSubMenu); MenuItem->spSubMenu = SubMenuObject; UserReferenceObject(SubMenuObject); } else { EngSetLastError( ERROR_INVALID_PARAMETER); return FALSE; } } else { // If submenu just dereference it. if (MenuItem->spSubMenu) IntReleaseMenuObject(MenuItem->spSubMenu); MenuItem->spSubMenu = NULL; } } if ((lpmii->fMask & MIIM_STRING) || ((lpmii->fMask & MIIM_TYPE) && (MENU_ITEM_TYPE(lpmii->fType) == MF_STRING))) { /* free the string when used */ FreeMenuText(MenuObject,MenuItem); RtlInitUnicodeString(&MenuItem->lpstr, NULL); MenuItem->Xlpstr = NULL; if(lpmii->dwTypeData && lpmii->cch && lpstr && lpstr->Buffer) { UNICODE_STRING Source; if (!NT_VERIFY(lpmii->cch <= UNICODE_STRING_MAX_CHARS)) { return FALSE; } Source.Length = Source.MaximumLength = (USHORT)(lpmii->cch * sizeof(WCHAR)); Source.Buffer = lpmii->dwTypeData; MenuItem->lpstr.Buffer = DesktopHeapAlloc( MenuObject->head.rpdesk, Source.Length + sizeof(WCHAR)); if(MenuItem->lpstr.Buffer != NULL) { MenuItem->lpstr.Length = 0; MenuItem->lpstr.MaximumLength = Source.Length + sizeof(WCHAR); RtlCopyUnicodeString(&MenuItem->lpstr, &Source); MenuItem->lpstr.Buffer[MenuItem->lpstr.Length / sizeof(WCHAR)] = 0; MenuItem->cch = MenuItem->lpstr.Length / sizeof(WCHAR); MenuItem->Xlpstr = (USHORT*)MenuItem->lpstr.Buffer; } } } if( !(MenuObject->fFlags & MNF_SYSMENU) && !MenuItem->Xlpstr && !lpmii->dwTypeData && !(MenuItem->fType & MFT_OWNERDRAW) && !MenuItem->hbmp) MenuItem->fType |= MFT_SEPARATOR; if (sizeof(ROSMENUITEMINFO) == lpmii->cbSize) { MenuItem->xItem = lpmii->Rect.left; MenuItem->yItem = lpmii->Rect.top; MenuItem->cxItem = lpmii->Rect.right; // Do this for now...... MenuItem->cyItem = lpmii->Rect.bottom; MenuItem->dxTab = lpmii->dxTab; lpmii->lpstr = MenuItem->lpstr.Buffer; /* Send back new allocated string or zero */ MenuItem->cxBmp = lpmii->maxBmpSize.cx; MenuItem->cyBmp = lpmii->maxBmpSize.cy; } return TRUE; } UINT FASTCALL IntEnableMenuItem(PMENU MenuObject, UINT uIDEnableItem, UINT uEnable) { PITEM MenuItem; UINT res; if (!(MenuItem = MENU_FindItem( &MenuObject, &uIDEnableItem, uEnable ))) return (UINT)-1; res = MenuItem->fState & (MF_GRAYED | MF_DISABLED); MenuItem->fState ^= (res ^ uEnable) & (MF_GRAYED | MF_DISABLED); /* If the close item in the system menu change update the close button */ if (res != uEnable) { switch (MenuItem->wID) // More than just close. { case SC_CLOSE: case SC_MAXIMIZE: case SC_MINIMIZE: case SC_MOVE: case SC_RESTORE: case SC_SIZE: if (MenuObject->fFlags & MNF_SYSSUBMENU && MenuObject->spwndNotify != 0) { //RECTL rc = MenuObject->spwndNotify->rcWindow; /* Refresh the frame to reflect the change */ //IntMapWindowPoints(0, MenuObject->spwndNotify, (POINT *)&rc, 2); //rc.bottom = 0; //co_UserRedrawWindow(MenuObject->spwndNotify, &rc, 0, RDW_FRAME | RDW_INVALIDATE | RDW_NOCHILDREN); // Allow UxTheme! UserPaintCaption(MenuObject->spwndNotify, DC_BUTTONS); } default: break; } } return res; } DWORD FASTCALL IntCheckMenuItem(PMENU MenuObject, UINT uIDCheckItem, UINT uCheck) { PITEM MenuItem; DWORD res; if (!(MenuItem = MENU_FindItem( &MenuObject, &uIDCheckItem, uCheck ))) return -1; res = (DWORD)(MenuItem->fState & MF_CHECKED); MenuItem->fState ^= (res ^ uCheck) & MF_CHECKED; return res; } BOOL FASTCALL UserSetMenuDefaultItem(PMENU MenuObject, UINT uItem, UINT fByPos) { UINT i; PITEM MenuItem = MenuObject->rgItems; if (!MenuItem) return FALSE; /* reset all default-item flags */ for (i = 0; i < MenuObject->cItems; i++, MenuItem++) { MenuItem->fState &= ~MFS_DEFAULT; } /* no default item */ if(uItem == (UINT)-1) { return TRUE; } MenuItem = MenuObject->rgItems; if ( fByPos ) { if ( uItem >= MenuObject->cItems ) return FALSE; MenuItem[uItem].fState |= MFS_DEFAULT; return TRUE; } else { for (i = 0; i < MenuObject->cItems; i++, MenuItem++) { if (MenuItem->wID == uItem) { MenuItem->fState |= MFS_DEFAULT; return TRUE; } } } return FALSE; } UINT FASTCALL IntGetMenuDefaultItem(PMENU MenuObject, UINT fByPos, UINT gmdiFlags, DWORD *gismc) { UINT i = 0; PITEM MenuItem = MenuObject->rgItems; /* empty menu */ if (!MenuItem) return -1; while ( !( MenuItem->fState & MFS_DEFAULT ) ) { i++; MenuItem++; if (i >= MenuObject->cItems ) return -1; } /* default: don't return disabled items */ if ( (!(GMDI_USEDISABLED & gmdiFlags)) && (MenuItem->fState & MFS_DISABLED )) return -1; /* search rekursiv when needed */ if ( (gmdiFlags & GMDI_GOINTOPOPUPS) && MenuItem->spSubMenu ) { UINT ret; (*gismc)++; ret = IntGetMenuDefaultItem( MenuItem->spSubMenu, fByPos, gmdiFlags, gismc ); (*gismc)--; if ( -1 != ret ) return ret; /* when item not found in submenu, return the popup item */ } return ( fByPos ) ? i : MenuItem->wID; } PMENU FASTCALL co_IntGetSubMenu( PMENU pMenu, int nPos) { PITEM pItem; if (!(pItem = MENU_FindItem( &pMenu, (UINT*)&nPos, MF_BYPOSITION ))) return NULL; return pItem->spSubMenu; } /*********************************************************************** * MenuInitSysMenuPopup * * Grey the appropriate items in System menu. */ void FASTCALL MENU_InitSysMenuPopup(PMENU menu, DWORD style, DWORD clsStyle, LONG HitTest ) { BOOL gray; UINT DefItem; gray = !(style & WS_THICKFRAME) || (style & (WS_MAXIMIZE | WS_MINIMIZE)); IntEnableMenuItem( menu, SC_SIZE, (gray ? MF_GRAYED : MF_ENABLED) ); gray = ((style & (WS_MAXIMIZE | WS_MINIMIZE)) != 0); IntEnableMenuItem( menu, SC_MOVE, (gray ? MF_GRAYED : MF_ENABLED) ); gray = !(style & WS_MINIMIZEBOX) || (style & WS_MINIMIZE); IntEnableMenuItem( menu, SC_MINIMIZE, (gray ? MF_GRAYED : MF_ENABLED) ); gray = !(style & WS_MAXIMIZEBOX) || (style & WS_MAXIMIZE); IntEnableMenuItem( menu, SC_MAXIMIZE, (gray ? MF_GRAYED : MF_ENABLED) ); gray = !(style & (WS_MAXIMIZE | WS_MINIMIZE)); IntEnableMenuItem( menu, SC_RESTORE, (gray ? MF_GRAYED : MF_ENABLED) ); gray = (clsStyle & CS_NOCLOSE) != 0; /* The menu item must keep its state if it's disabled */ if(gray) IntEnableMenuItem( menu, SC_CLOSE, MF_GRAYED); /* Set default menu item */ if(style & WS_MINIMIZE) DefItem = SC_RESTORE; else if(HitTest == HTCAPTION) DefItem = ((style & (WS_MAXIMIZE | WS_MINIMIZE)) ? SC_RESTORE : SC_MAXIMIZE); else DefItem = SC_CLOSE; UserSetMenuDefaultItem(menu, DefItem, MF_BYCOMMAND); } /*********************************************************************** * MenuDrawPopupGlyph * * Draws popup magic glyphs (can be found in system menu). */ static void FASTCALL MENU_DrawPopupGlyph(HDC dc, LPRECT r, INT_PTR popupMagic, BOOL inactive, BOOL hilite) { LOGFONTW lf; HFONT hFont, hOldFont; COLORREF clrsave; INT bkmode; WCHAR symbol; switch (popupMagic) { case (INT_PTR) HBMMENU_POPUP_RESTORE: symbol = '2'; break; case (INT_PTR) HBMMENU_POPUP_MINIMIZE: symbol = '0'; break; case (INT_PTR) HBMMENU_POPUP_MAXIMIZE: symbol = '1'; break; case (INT_PTR) HBMMENU_POPUP_CLOSE: symbol = 'r'; break; default: ERR("Invalid popup magic bitmap %d\n", (int)popupMagic); return; } RtlZeroMemory(&lf, sizeof(LOGFONTW)); RECTL_vInflateRect(r, -2, -2); lf.lfHeight = r->bottom - r->top; lf.lfWidth = 0; lf.lfWeight = FW_NORMAL; lf.lfCharSet = DEFAULT_CHARSET; RtlCopyMemory(lf.lfFaceName, L"Marlett", sizeof(L"Marlett")); hFont = GreCreateFontIndirectW(&lf); /* save font and text color */ hOldFont = NtGdiSelectFont(dc, hFont); clrsave = GreGetTextColor(dc); bkmode = GreGetBkMode(dc); /* set color and drawing mode */ IntGdiSetBkMode(dc, TRANSPARENT); if (inactive) { /* draw shadow */ if (!hilite) { IntGdiSetTextColor(dc, IntGetSysColor(COLOR_HIGHLIGHTTEXT)); GreTextOutW(dc, r->left + 1, r->top + 1, &symbol, 1); } } IntGdiSetTextColor(dc, IntGetSysColor(inactive ? COLOR_GRAYTEXT : (hilite ? COLOR_HIGHLIGHTTEXT : COLOR_MENUTEXT))); /* draw selected symbol */ GreTextOutW(dc, r->left, r->top, &symbol, 1); /* restore previous settings */ IntGdiSetTextColor(dc, clrsave); NtGdiSelectFont(dc, hOldFont); IntGdiSetBkMode(dc, bkmode); GreDeleteObject(hFont); } /*********************************************************************** * MENU_AdjustMenuItemRect * * Adjust menu item rectangle according to scrolling state. */ VOID FASTCALL MENU_AdjustMenuItemRect(PMENU menu, PRECTL rect) { if (menu->dwArrowsOn) { UINT arrow_bitmap_height; arrow_bitmap_height = gpsi->oembmi[OBI_UPARROW].cy; ///// Menu up arrow! OBM_UPARROW rect->top += arrow_bitmap_height - menu->iTop; rect->bottom += arrow_bitmap_height - menu->iTop; } } /*********************************************************************** * MENU_FindItemByCoords * * Find the item at the specified coordinates (screen coords). Does * not work for child windows and therefore should not be called for * an arbitrary system menu. */ static ITEM *MENU_FindItemByCoords( MENU *menu, POINT pt, UINT *pos ) { ITEM *item; UINT i; INT cx, cy; RECT rect; PWND pWnd = ValidateHwndNoErr(menu->hWnd); if (!IntGetWindowRect(pWnd, &rect)) return NULL; cx = UserGetSystemMetrics(SM_CXDLGFRAME); cy = UserGetSystemMetrics(SM_CYDLGFRAME); RECTL_vInflateRect(&rect, -cx, -cy); if (pWnd->ExStyle & WS_EX_LAYOUTRTL) pt.x = rect.right - 1 - pt.x; else pt.x -= rect.left; pt.y -= rect.top; item = menu->rgItems; for (i = 0; i < menu->cItems; i++, item++) { //rect = item->rect; rect.left = item->xItem; rect.top = item->yItem; rect.right = item->cxItem; // Do this for now...... rect.bottom = item->cyItem; MENU_AdjustMenuItemRect(menu, &rect); if (RECTL_bPointInRect(&rect, pt.x, pt.y)) { if (pos) *pos = i; return item; } } return NULL; } INT FASTCALL IntMenuItemFromPoint(PWND pWnd, HMENU hMenu, POINT ptScreen) { MENU *menu = UserGetMenuObject(hMenu); UINT pos; /*FIXME: Do we have to handle hWnd here? */ if (!menu) return -1; if (!MENU_FindItemByCoords(menu, ptScreen, &pos)) return -1; return pos; } /*********************************************************************** * MenuFindItemByKey * * Find the menu item selected by a key press. * Return item id, -1 if none, -2 if we should close the menu. */ static UINT FASTCALL MENU_FindItemByKey(PWND WndOwner, PMENU menu, WCHAR Key, BOOL ForceMenuChar) { LRESULT MenuChar; WORD Flags = 0; TRACE("\tlooking for '%c' (0x%02x) in [%p]\n", (char)Key, Key, menu ); if (!menu || !VerifyMenu(menu)) menu = co_IntGetSubMenu( UserGetMenuObject(WndOwner->SystemMenu), 0 ); if (menu) { ITEM *item = menu->rgItems; if ( !ForceMenuChar ) { UINT i; BOOL cjk = UserGetSystemMetrics( SM_DBCSENABLED ); for (i = 0; i < menu->cItems; i++, item++) { LPWSTR text = item->Xlpstr; if( text) { const WCHAR *p = text - 2; do { const WCHAR *q = p + 2; p = wcschr (q, '&'); if (!p && cjk) p = wcschr (q, '\036'); /* Japanese Win16 */ } while (p != NULL && p [1] == '&'); if (p && (towupper(p[1]) == towupper(Key))) return i; } } } Flags |= menu->fFlags & MNF_POPUP ? MF_POPUP : 0; Flags |= menu->fFlags & MNF_SYSMENU ? MF_SYSMENU : 0; MenuChar = co_IntSendMessage( UserHMGetHandle(WndOwner), WM_MENUCHAR, MAKEWPARAM(Key, Flags), (LPARAM) UserHMGetHandle(menu)); if (HIWORD(MenuChar) == MNC_EXECUTE) return LOWORD(MenuChar); if (HIWORD(MenuChar) == MNC_CLOSE) return (UINT)(-2); } return (UINT)(-1); } /*********************************************************************** * MenuGetBitmapItemSize * * Get the size of a bitmap item. */ static void FASTCALL MENU_GetBitmapItemSize(PITEM lpitem, SIZE *size, PWND WndOwner) { BITMAP bm; HBITMAP bmp = lpitem->hbmp; size->cx = size->cy = 0; /* check if there is a magic menu item associated with this item */ if (IS_MAGIC_BITMAP(bmp)) { switch((INT_PTR) bmp) { case (INT_PTR)HBMMENU_CALLBACK: { MEASUREITEMSTRUCT measItem; measItem.CtlType = ODT_MENU; measItem.CtlID = 0; measItem.itemID = lpitem->wID; measItem.itemWidth = lpitem->cxItem - lpitem->xItem; //lpitem->Rect.right - lpitem->Rect.left; measItem.itemHeight = lpitem->cyItem - lpitem->yItem; //lpitem->Rect.bottom - lpitem->Rect.top; measItem.itemData = lpitem->dwItemData; co_IntSendMessage( UserHMGetHandle(WndOwner), WM_MEASUREITEM, 0, (LPARAM)&measItem); size->cx = measItem.itemWidth; size->cy = measItem.itemHeight; TRACE("HBMMENU_CALLBACK Height %d Width %d\n",measItem.itemHeight,measItem.itemWidth); return; } break; case (INT_PTR) HBMMENU_SYSTEM: if (lpitem->dwItemData) { bmp = (HBITMAP) lpitem->dwItemData; break; } /* fall through */ case (INT_PTR) HBMMENU_MBAR_RESTORE: case (INT_PTR) HBMMENU_MBAR_MINIMIZE: case (INT_PTR) HBMMENU_MBAR_CLOSE: case (INT_PTR) HBMMENU_MBAR_MINIMIZE_D: case (INT_PTR) HBMMENU_MBAR_CLOSE_D: case (INT_PTR) HBMMENU_POPUP_CLOSE: case (INT_PTR) HBMMENU_POPUP_RESTORE: case (INT_PTR) HBMMENU_POPUP_MAXIMIZE: case (INT_PTR) HBMMENU_POPUP_MINIMIZE: /* FIXME: Why we need to subtract these magic values? */ /* to make them smaller than the menu bar? */ size->cx = UserGetSystemMetrics(SM_CXSIZE) - 2; size->cy = UserGetSystemMetrics(SM_CYSIZE) - 4; return; } } if (GreGetObject(bmp, sizeof(BITMAP), &bm)) { size->cx = bm.bmWidth; size->cy = bm.bmHeight; } } /*********************************************************************** * MenuDrawBitmapItem * * Draw a bitmap item. */ static void FASTCALL MENU_DrawBitmapItem(HDC hdc, PITEM lpitem, const RECT *rect, PMENU Menu, PWND WndOwner, UINT odaction, BOOL MenuBar) { BITMAP bm; DWORD rop; HDC hdcMem; HBITMAP bmp; int w = rect->right - rect->left; int h = rect->bottom - rect->top; int bmp_xoffset = 0; int left, top; BOOL flat_menu; HBITMAP hbmToDraw = lpitem->hbmp; bmp = hbmToDraw; UserSystemParametersInfo(SPI_GETFLATMENU, 0, &flat_menu, 0); /* Check if there is a magic menu item associated with this item */ if (IS_MAGIC_BITMAP(hbmToDraw)) { UINT flags = 0; RECT r; r = *rect; switch ((INT_PTR)hbmToDraw) { case (INT_PTR)HBMMENU_SYSTEM: if (lpitem->dwItemData) { if (ValidateHwndNoErr((HWND)lpitem->dwItemData)) { ERR("Get Item Data from this Window!!!\n"); } ERR("Draw Bitmap\n"); bmp = (HBITMAP)lpitem->dwItemData; if (!GreGetObject( bmp, sizeof(bm), &bm )) return; } else { PCURICON_OBJECT pIcon = NULL; //if (!BmpSysMenu) BmpSysMenu = LoadBitmapW(0, MAKEINTRESOURCEW(OBM_CLOSE)); //bmp = BmpSysMenu; //if (! GreGetObject(bmp, sizeof(bm), &bm)) return; /* only use right half of the bitmap */ //bmp_xoffset = bm.bmWidth / 2; //bm.bmWidth -= bmp_xoffset; if (WndOwner) { pIcon = NC_IconForWindow(WndOwner); // FIXME: NC_IconForWindow should reference it for us */ if (pIcon) UserReferenceObject(pIcon); } ERR("Draw ICON\n"); if (pIcon) { LONG cx = UserGetSystemMetrics(SM_CXSMICON); LONG cy = UserGetSystemMetrics(SM_CYSMICON); LONG x = rect->left - cx/2 + 1 + (rect->bottom - rect->top)/2; // this is really what Window does LONG y = (rect->top + rect->bottom)/2 - cy/2; // center UserDrawIconEx(hdc, x, y, pIcon, cx, cy, 0, NULL, DI_NORMAL); UserDereferenceObject(pIcon); } return; } goto got_bitmap; case (INT_PTR)HBMMENU_MBAR_RESTORE: flags = DFCS_CAPTIONRESTORE; break; case (INT_PTR)HBMMENU_MBAR_MINIMIZE: r.right += 1; flags = DFCS_CAPTIONMIN; break; case (INT_PTR)HBMMENU_MBAR_MINIMIZE_D: r.right += 1; flags = DFCS_CAPTIONMIN | DFCS_INACTIVE; break; case (INT_PTR)HBMMENU_MBAR_CLOSE: flags = DFCS_CAPTIONCLOSE; break; case (INT_PTR)HBMMENU_MBAR_CLOSE_D: flags = DFCS_CAPTIONCLOSE | DFCS_INACTIVE; break; case (INT_PTR)HBMMENU_CALLBACK: { DRAWITEMSTRUCT drawItem; POINT origorg; drawItem.CtlType = ODT_MENU; drawItem.CtlID = 0; drawItem.itemID = lpitem->wID; drawItem.itemAction = odaction; drawItem.itemState = (lpitem->fState & MF_CHECKED)?ODS_CHECKED:0; drawItem.itemState |= (lpitem->fState & MF_DEFAULT)?ODS_DEFAULT:0; drawItem.itemState |= (lpitem->fState & MF_DISABLED)?ODS_DISABLED:0; drawItem.itemState |= (lpitem->fState & MF_GRAYED)?ODS_GRAYED|ODS_DISABLED:0; drawItem.itemState |= (lpitem->fState & MF_HILITE)?ODS_SELECTED:0; drawItem.itemState |= (!(Menu->fFlags & MNF_UNDERLINE))?ODS_NOACCEL:0; drawItem.itemState |= (Menu->fFlags & MNF_INACTIVE)?ODS_INACTIVE:0; drawItem.hwndItem = (HWND)UserHMGetHandle(Menu); drawItem.hDC = hdc; drawItem.rcItem = *rect; drawItem.itemData = lpitem->dwItemData; /* some applications make this assumption on the DC's origin */ GreSetViewportOrgEx( hdc, lpitem->xItem, lpitem->yItem, &origorg); RECTL_vOffsetRect(&drawItem.rcItem, -(LONG)lpitem->xItem, -(LONG)lpitem->yItem); co_IntSendMessage( UserHMGetHandle(WndOwner), WM_DRAWITEM, 0, (LPARAM)&drawItem); GreSetViewportOrgEx( hdc, origorg.x, origorg.y, NULL); return; } break; case (INT_PTR) HBMMENU_POPUP_CLOSE: case (INT_PTR) HBMMENU_POPUP_RESTORE: case (INT_PTR) HBMMENU_POPUP_MAXIMIZE: case (INT_PTR) HBMMENU_POPUP_MINIMIZE: MENU_DrawPopupGlyph(hdc, &r, (INT_PTR)hbmToDraw, lpitem->fState & MF_GRAYED, lpitem->fState & MF_HILITE); return; } RECTL_vInflateRect(&r, -1, -1); if (lpitem->fState & MF_HILITE) flags |= DFCS_PUSHED; DrawFrameControl(hdc, &r, DFC_CAPTION, flags); return; } if (!bmp || !GreGetObject( bmp, sizeof(bm), &bm )) return; got_bitmap: hdcMem = NtGdiCreateCompatibleDC( hdc ); NtGdiSelectBitmap( hdcMem, bmp ); /* handle fontsize > bitmap_height */ top = (h>bm.bmHeight) ? rect->top+(h-bm.bmHeight)/2 : rect->top; left=rect->left; rop=((lpitem->fState & MF_HILITE) && !IS_MAGIC_BITMAP(hbmToDraw)) ? NOTSRCCOPY : SRCCOPY; if ((lpitem->fState & MF_HILITE) && lpitem->hbmp) IntGdiSetBkColor(hdc, IntGetSysColor(COLOR_HIGHLIGHT)); if (MenuBar && !flat_menu && (lpitem->fState & (MF_HILITE | MF_GRAYED)) == MF_HILITE) { ++left; ++top; } NtGdiBitBlt( hdc, left, top, w, h, hdcMem, bmp_xoffset, 0, rop , 0, 0); IntGdiDeleteDC( hdcMem, FALSE ); } LONG IntGetDialogBaseUnits(VOID) { static DWORD units; if (!units) { HDC hdc; SIZE size; if ((hdc = UserGetDCEx(NULL, NULL, DCX_CACHE))) { size.cx = IntGetCharDimensions( hdc, NULL, (PDWORD)&size.cy ); if (size.cx) units = MAKELONG( size.cx, size.cy ); UserReleaseDC( 0, hdc, FALSE); } } return units; } /*********************************************************************** * MenuCalcItemSize * * Calculate the size of the menu item and store it in lpitem->rect. */ static void FASTCALL MENU_CalcItemSize( HDC hdc, PITEM lpitem, PMENU Menu, PWND pwndOwner, INT orgX, INT orgY, BOOL menuBar, BOOL textandbmp) { WCHAR *p; UINT check_bitmap_width = UserGetSystemMetrics( SM_CXMENUCHECK ); UINT arrow_bitmap_width; RECT Rect; INT itemheight = 0; TRACE("dc=%x owner=%x (%d,%d)\n", hdc, pwndOwner, orgX, orgY); arrow_bitmap_width = gpsi->oembmi[OBI_MNARROW].cx; MenuCharSize.cx = IntGetCharDimensions( hdc, NULL, (PDWORD)&MenuCharSize.cy ); RECTL_vSetRect( &Rect, orgX, orgY, orgX, orgY ); if (lpitem->fType & MF_OWNERDRAW) { MEASUREITEMSTRUCT mis; mis.CtlType = ODT_MENU; mis.CtlID = 0; mis.itemID = lpitem->wID; mis.itemData = lpitem->dwItemData; mis.itemHeight = HIWORD( IntGetDialogBaseUnits()); mis.itemWidth = 0; co_IntSendMessage( UserHMGetHandle(pwndOwner), WM_MEASUREITEM, 0, (LPARAM)&mis ); /* Tests reveal that Windows ( Win95 thru WinXP) adds twice the average * width of a menufont character to the width of an owner-drawn menu. */ Rect.right += mis.itemWidth + 2 * MenuCharSize.cx; if (menuBar) { /* under at least win95 you seem to be given a standard height for the menu and the height value is ignored */ Rect.bottom += UserGetSystemMetrics(SM_CYMENUSIZE); } else Rect.bottom += mis.itemHeight; // Or this, //lpitem->cxBmp = mis.itemWidth; //lpitem->cyBmp = mis.itemHeight; TRACE("MF_OWNERDRAW Height %d Width %d\n",mis.itemHeight,mis.itemWidth); TRACE("MF_OWNERDRAW id=%04lx size=%dx%d cx %d cy %d\n", lpitem->wID, Rect.right-Rect.left, Rect.bottom-Rect.top, MenuCharSize.cx, MenuCharSize.cy); lpitem->xItem = Rect.left; lpitem->yItem = Rect.top; lpitem->cxItem = Rect.right; lpitem->cyItem = Rect.bottom; return; } lpitem->xItem = orgX; lpitem->yItem = orgY; lpitem->cxItem = orgX; lpitem->cyItem = orgY; if (lpitem->fType & MF_SEPARATOR) { lpitem->cyItem += UserGetSystemMetrics( SM_CYMENUSIZE)/2;//SEPARATOR_HEIGHT; if( !menuBar) lpitem->cxItem += arrow_bitmap_width + MenuCharSize.cx; return; } lpitem->dxTab = 0; if (lpitem->hbmp) { SIZE size; if (!menuBar) { MENU_GetBitmapItemSize(lpitem, &size, pwndOwner ); /* Keep the size of the bitmap in callback mode to be able * to draw it correctly */ lpitem->cxBmp = size.cx; lpitem->cyBmp = size.cy; Menu->cxTextAlign = max(Menu->cxTextAlign, size.cx); lpitem->cxItem += size.cx + 2; itemheight = size.cy + 2; if( !((Menu->fFlags & MNS_STYLE_MASK) & MNS_NOCHECK)) lpitem->cxItem += 2 * check_bitmap_width; lpitem->cxItem += 4 + MenuCharSize.cx; lpitem->dxTab = lpitem->cxItem; lpitem->cxItem += arrow_bitmap_width; } else /* hbmpItem & MenuBar */ { MENU_GetBitmapItemSize(lpitem, &size, pwndOwner ); lpitem->cxItem += size.cx; if( lpitem->Xlpstr) lpitem->cxItem += 2; itemheight = size.cy; /* Special case: Minimize button doesn't have a space behind it. */ if (lpitem->hbmp == (HBITMAP)HBMMENU_MBAR_MINIMIZE || lpitem->hbmp == (HBITMAP)HBMMENU_MBAR_MINIMIZE_D) lpitem->cxItem -= 1; } } else if (!menuBar) { if( !((Menu->fFlags & MNS_STYLE_MASK) & MNS_NOCHECK)) lpitem->cxItem += check_bitmap_width; lpitem->cxItem += 4 + MenuCharSize.cx; lpitem->dxTab = lpitem->cxItem; lpitem->cxItem += arrow_bitmap_width; } /* it must be a text item - unless it's the system menu */ if (!(lpitem->fType & MF_SYSMENU) && lpitem->Xlpstr) { HFONT hfontOld = NULL; RECT rc;// = lpitem->Rect; LONG txtheight, txtwidth; rc.left = lpitem->xItem; rc.top = lpitem->yItem; rc.right = lpitem->cxItem; // Do this for now...... rc.bottom = lpitem->cyItem; if ( lpitem->fState & MFS_DEFAULT ) { hfontOld = NtGdiSelectFont( hdc, ghMenuFontBold ); } if (menuBar) { txtheight = DrawTextW( hdc, lpitem->Xlpstr, -1, &rc, DT_SINGLELINE|DT_CALCRECT); lpitem->cxItem += rc.right - rc.left; itemheight = max( max( itemheight, txtheight), UserGetSystemMetrics( SM_CYMENU) - 1); lpitem->cxItem += 2 * MenuCharSize.cx; } else { if ((p = wcschr( lpitem->Xlpstr, '\t' )) != NULL) { RECT tmprc = rc; LONG tmpheight; int n = (int)( p - lpitem->Xlpstr); /* Item contains a tab (only meaningful in popup menus) */ /* get text size before the tab */ txtheight = DrawTextW( hdc, lpitem->Xlpstr, n, &rc, DT_SINGLELINE|DT_CALCRECT); txtwidth = rc.right - rc.left; p += 1; /* advance past the Tab */ /* get text size after the tab */ tmpheight = DrawTextW( hdc, p, -1, &tmprc, DT_SINGLELINE|DT_CALCRECT); lpitem->dxTab += txtwidth; txtheight = max( txtheight, tmpheight); txtwidth += MenuCharSize.cx + /* space for the tab */ tmprc.right - tmprc.left; /* space for the short cut */ } else { txtheight = DrawTextW( hdc, lpitem->Xlpstr, -1, &rc, DT_SINGLELINE|DT_CALCRECT); txtwidth = rc.right - rc.left; lpitem->dxTab += txtwidth; } lpitem->cxItem += 2 + txtwidth; itemheight = max( itemheight, max( txtheight + 2, MenuCharSize.cy + 4)); } if (hfontOld) { NtGdiSelectFont (hdc, hfontOld); } } else if( menuBar) { itemheight = max( itemheight, UserGetSystemMetrics(SM_CYMENU)-1); } lpitem->cyItem += itemheight; TRACE("(%ld,%ld)-(%ld,%ld)\n", lpitem->xItem, lpitem->yItem, lpitem->cxItem, lpitem->cyItem); } /*********************************************************************** * MENU_GetMaxPopupHeight */ static UINT MENU_GetMaxPopupHeight(PMENU lppop) { if (lppop->cyMax) { //ERR("MGMaxPH cyMax %d\n",lppop->cyMax); return lppop->cyMax; } //ERR("MGMaxPH SyMax %d\n",UserGetSystemMetrics(SM_CYSCREEN) - UserGetSystemMetrics(SM_CYBORDER)); return UserGetSystemMetrics(SM_CYSCREEN) - UserGetSystemMetrics(SM_CYBORDER); } /*********************************************************************** * MenuPopupMenuCalcSize * * Calculate the size of a popup menu. */ static void FASTCALL MENU_PopupMenuCalcSize(PMENU Menu, PWND WndOwner) { PITEM lpitem; HDC hdc; int start, i; int orgX, orgY, maxX, maxTab, maxTabWidth, maxHeight; BOOL textandbmp = FALSE; Menu->cxMenu = Menu->cyMenu = 0; if (Menu->cItems == 0) return; hdc = UserGetDCEx(NULL, NULL, DCX_CACHE); NtGdiSelectFont( hdc, ghMenuFont ); start = 0; maxX = 0; Menu->cxTextAlign = 0; while (start < Menu->cItems) { lpitem = &Menu->rgItems[start]; orgX = maxX; if( lpitem->fType & (MF_MENUBREAK | MF_MENUBARBREAK)) orgX += MENU_COL_SPACE; orgY = 0; maxTab = maxTabWidth = 0; /* Parse items until column break or end of menu */ for (i = start; i < Menu->cItems; i++, lpitem++) { if (i != start && (lpitem->fType & (MF_MENUBREAK | MF_MENUBARBREAK))) break; MENU_CalcItemSize(hdc, lpitem, Menu, WndOwner, orgX, orgY, FALSE, textandbmp); maxX = max(maxX, lpitem->cxItem); orgY = lpitem->cyItem; if (IS_STRING_ITEM(lpitem->fType) && lpitem->dxTab ) { maxTab = max( maxTab, lpitem->dxTab ); maxTabWidth = max(maxTabWidth, lpitem->cxItem - lpitem->dxTab); } if( lpitem->Xlpstr && lpitem->hbmp) textandbmp = TRUE; } /* Finish the column (set all items to the largest width found) */ maxX = max( maxX, maxTab + maxTabWidth ); for (lpitem = &Menu->rgItems[start]; start < i; start++, lpitem++) { lpitem->cxItem = maxX; if (IS_STRING_ITEM(lpitem->fType) && lpitem->dxTab) lpitem->dxTab = maxTab; } Menu->cyMenu = max(Menu->cyMenu, orgY); } Menu->cxMenu = maxX; /* if none of the items have both text and bitmap then * the text and bitmaps are all aligned on the left. If there is at * least one item with both text and bitmap then bitmaps are * on the left and texts left aligned with the right hand side * of the bitmaps */ if( !textandbmp) Menu->cxTextAlign = 0; /* Adjust popup height if it exceeds maximum */ maxHeight = MENU_GetMaxPopupHeight(Menu); Menu->iMaxTop = Menu->cyMenu; if (Menu->cyMenu >= maxHeight) { Menu->cyMenu = maxHeight; Menu->dwArrowsOn = 1; } else { Menu->dwArrowsOn = 0; } UserReleaseDC( 0, hdc, FALSE ); } /*********************************************************************** * MENU_MenuBarCalcSize * * FIXME: Word 6 implements its own MDI and its own 'close window' bitmap * height is off by 1 pixel which causes lengthy window relocations when * active document window is maximized/restored. * * Calculate the size of the menu bar. */ static void MENU_MenuBarCalcSize( HDC hdc, LPRECT lprect, PMENU lppop, PWND pwndOwner ) { ITEM *lpitem; UINT start, i, helpPos; int orgX, orgY, maxY; if ((lprect == NULL) || (lppop == NULL)) return; if (lppop->cItems == 0) return; //TRACE("lprect %p %s\n", lprect, wine_dbgstr_rect( lprect)); lppop->cxMenu = lprect->right - lprect->left; lppop->cyMenu = 0; maxY = lprect->top; start = 0; helpPos = ~0U; lppop->cxTextAlign = 0; while (start < lppop->cItems) { lpitem = &lppop->rgItems[start]; orgX = lprect->left; orgY = maxY; /* Parse items until line break or end of menu */ for (i = start; i < lppop->cItems; i++, lpitem++) { if ((helpPos == ~0U) && (lpitem->fType & MF_RIGHTJUSTIFY)) helpPos = i; if ((i != start) && (lpitem->fType & (MF_MENUBREAK | MF_MENUBARBREAK))) break; TRACE("calling MENU_CalcItemSize org=(%d, %d)\n", orgX, orgY ); //debug_print_menuitem (" item: ", lpitem, ""); //MENU_CalcItemSize( hdc, lpitem, pwndOwner, orgX, orgY, TRUE, lppop ); MENU_CalcItemSize(hdc, lpitem, lppop, pwndOwner, orgX, orgY, TRUE, FALSE); if (lpitem->cxItem > lprect->right) { if (i != start) break; else lpitem->cxItem = lprect->right; } maxY = max( maxY, lpitem->cyItem ); orgX = lpitem->cxItem; } /* Finish the line (set all items to the largest height found) */ /* FIXME: Is this really needed? */ /*NO! it is not needed, why make the HBMMENU_MBAR_CLOSE, MINIMIZE & RESTORE, look the same size as the menu bar! */ #if 0 while (start < i) lppop->rgItems[start++].cyItem = maxY; #endif start = i; /* This works! */ } lprect->bottom = maxY + 1; lppop->cyMenu = lprect->bottom - lprect->top; /* Flush right all items between the MF_RIGHTJUSTIFY and */ /* the last item (if several lines, only move the last line) */ if (helpPos == ~0U) return; lpitem = &lppop->rgItems[lppop->cItems-1]; orgY = lpitem->yItem; orgX = lprect->right; for (i = lppop->cItems - 1; i >= helpPos; i--, lpitem--) { if (lpitem->yItem != orgY) break; /* Other line */ if (lpitem->cxItem >= orgX) break; /* Too far right already */ lpitem->xItem += orgX - lpitem->cxItem; lpitem->cxItem = orgX; orgX = lpitem->xItem; } } /*********************************************************************** * MENU_DrawScrollArrows * * Draw scroll arrows. */ static void MENU_DrawScrollArrows(PMENU lppop, HDC hdc) { UINT arrow_bitmap_height; RECT rect; UINT Flags = 0; arrow_bitmap_height = gpsi->oembmi[OBI_DNARROW].cy; rect.left = 0; rect.top = 0; rect.right = lppop->cxMenu; rect.bottom = arrow_bitmap_height; FillRect(hdc, &rect, IntGetSysColorBrush(COLOR_MENU)); UITOOLS95_DrawFrameMenu(hdc, &rect, (lppop->iTop ? 0 : DFCS_INACTIVE) | DFCS_MENUARROWUP); rect.top = lppop->cyMenu - arrow_bitmap_height; rect.bottom = lppop->cyMenu; FillRect(hdc, &rect, IntGetSysColorBrush(COLOR_MENU)); if (!(lppop->iTop < lppop->iMaxTop - (MENU_GetMaxPopupHeight(lppop) - 2 * arrow_bitmap_height))) Flags = DFCS_INACTIVE; UITOOLS95_DrawFrameMenu(hdc, &rect, Flags | DFCS_MENUARROWDOWN); } /*********************************************************************** * MenuDrawMenuItem * * Draw a single menu item. */ static void FASTCALL MENU_DrawMenuItem(PWND Wnd, PMENU Menu, PWND WndOwner, HDC hdc, PITEM lpitem, UINT Height, BOOL menuBar, UINT odaction) { RECT rect; PWCHAR Text; BOOL flat_menu = FALSE; int bkgnd; UINT arrow_bitmap_width = 0; //RECT bmprc; if (!menuBar) { arrow_bitmap_width = gpsi->oembmi[OBI_MNARROW].cx; } if (lpitem->fType & MF_SYSMENU) { if (!(Wnd->style & WS_MINIMIZE)) { NC_GetInsideRect(Wnd, &rect); UserDrawSysMenuButton(Wnd, hdc, &rect, lpitem->fState & (MF_HILITE | MF_MOUSESELECT)); } return; } UserSystemParametersInfo (SPI_GETFLATMENU, 0, &flat_menu, 0); bkgnd = (menuBar && flat_menu) ? COLOR_MENUBAR : COLOR_MENU; /* Setup colors */ if (lpitem->fState & MF_HILITE) { if(menuBar && !flat_menu) { IntGdiSetTextColor(hdc, IntGetSysColor(COLOR_MENUTEXT)); IntGdiSetBkColor(hdc, IntGetSysColor(COLOR_MENU)); } else { if (lpitem->fState & MF_GRAYED) IntGdiSetTextColor(hdc, IntGetSysColor(COLOR_GRAYTEXT)); else IntGdiSetTextColor(hdc, IntGetSysColor(COLOR_HIGHLIGHTTEXT)); IntGdiSetBkColor(hdc, IntGetSysColor(COLOR_HIGHLIGHT)); } } else { if (lpitem->fState & MF_GRAYED) IntGdiSetTextColor( hdc, IntGetSysColor( COLOR_GRAYTEXT ) ); else IntGdiSetTextColor( hdc, IntGetSysColor( COLOR_MENUTEXT ) ); IntGdiSetBkColor( hdc, IntGetSysColor( bkgnd ) ); } //TRACE("rect=%s\n", wine_dbgstr_rect( &lpitem->Rect)); //rect = lpitem->Rect; rect.left = lpitem->xItem; rect.top = lpitem->yItem; rect.right = lpitem->cxItem; // Do this for now...... rect.bottom = lpitem->cyItem; MENU_AdjustMenuItemRect(Menu, &rect); if (lpitem->fType & MF_OWNERDRAW) { /* ** Experimentation under Windows reveals that an owner-drawn ** menu is given the rectangle which includes the space it requested ** in its response to WM_MEASUREITEM _plus_ width for a checkmark ** and a popup-menu arrow. This is the value of lpitem->rect. ** Windows will leave all drawing to the application except for ** the popup-menu arrow. Windows always draws that itself, after ** the menu owner has finished drawing. */ DRAWITEMSTRUCT dis; COLORREF old_bk, old_text; dis.CtlType = ODT_MENU; dis.CtlID = 0; dis.itemID = lpitem->wID; dis.itemData = (DWORD)lpitem->dwItemData; dis.itemState = 0; if (lpitem->fState & MF_CHECKED) dis.itemState |= ODS_CHECKED; if (lpitem->fState & MF_DEFAULT) dis.itemState |= ODS_DEFAULT; if (lpitem->fState & MF_DISABLED) dis.itemState |= ODS_DISABLED; if (lpitem->fState & MF_GRAYED) dis.itemState |= ODS_GRAYED | ODS_DISABLED; if (lpitem->fState & MF_HILITE) dis.itemState |= ODS_SELECTED; if (!(Menu->fFlags & MNF_UNDERLINE)) dis.itemState |= ODS_NOACCEL; if (Menu->fFlags & MNF_INACTIVE) dis.itemState |= ODS_INACTIVE; dis.itemAction = odaction; /* ODA_DRAWENTIRE | ODA_SELECT | ODA_FOCUS; */ dis.hwndItem = (HWND) UserHMGetHandle(Menu); dis.hDC = hdc; dis.rcItem = rect; TRACE("Ownerdraw: owner=%p itemID=%d, itemState=%d, itemAction=%d, " "hwndItem=%p, hdc=%p, rcItem={%ld,%ld,%ld,%ld}\n", Wnd, dis.itemID, dis.itemState, dis.itemAction, dis.hwndItem, dis.hDC, dis.rcItem.left, dis.rcItem.top, dis.rcItem.right, dis.rcItem.bottom); TRACE("Ownerdraw: Width %d Height %d\n", dis.rcItem.right-dis.rcItem.left, dis.rcItem.bottom-dis.rcItem.top); old_bk = GreGetBkColor(hdc); old_text = GreGetTextColor(hdc); co_IntSendMessage(UserHMGetHandle(WndOwner), WM_DRAWITEM, 0, (LPARAM) &dis); IntGdiSetBkColor(hdc, old_bk); IntGdiSetTextColor(hdc, old_text); /* Draw the popup-menu arrow */ if (!menuBar && lpitem->spSubMenu) { RECT rectTemp; RtlCopyMemory(&rectTemp, &rect, sizeof(RECT)); rectTemp.left = rectTemp.right - UserGetSystemMetrics(SM_CXMENUCHECK); UITOOLS95_DrawFrameMenu(hdc, &rectTemp, DFCS_MENUARROW); } return; } if (menuBar && (lpitem->fType & MF_SEPARATOR)) return; if (lpitem->fState & MF_HILITE) { if (flat_menu) { RECTL_vInflateRect (&rect, -1, -1); FillRect(hdc, &rect, IntGetSysColorBrush(COLOR_MENUHILIGHT)); RECTL_vInflateRect (&rect, 1, 1); FrameRect(hdc, &rect, IntGetSysColorBrush(COLOR_HIGHLIGHT)); } else { if (menuBar) { FillRect(hdc, &rect, IntGetSysColorBrush(COLOR_MENU)); DrawEdge(hdc, &rect, BDR_SUNKENOUTER, BF_RECT); } else { FillRect(hdc, &rect, IntGetSysColorBrush(COLOR_HIGHLIGHT)); } } } else FillRect( hdc, &rect, IntGetSysColorBrush(bkgnd) ); IntGdiSetBkMode( hdc, TRANSPARENT ); /* vertical separator */ if (!menuBar && (lpitem->fType & MF_MENUBARBREAK)) { HPEN oldPen; RECT rc = rect; rc.left -= 3;//MENU_COL_SPACE / 2 + 1; == 3!! rc.top = 3; rc.bottom = Height - 3; if (flat_menu) { oldPen = NtGdiSelectPen( hdc, NtGdiGetStockObject(DC_PEN) ); IntSetDCPenColor(hdc, IntGetSysColor(COLOR_BTNSHADOW)); GreMoveTo( hdc, rc.left, rc.top, NULL ); NtGdiLineTo( hdc, rc.left, rc.bottom ); NtGdiSelectPen( hdc, oldPen ); } else DrawEdge (hdc, &rc, EDGE_ETCHED, BF_LEFT); } /* horizontal separator */ if (lpitem->fType & MF_SEPARATOR) { HPEN oldPen; RECT rc = rect; rc.left++; rc.right--; rc.top = (rc.top + rc.bottom) / 2 - 1; if (flat_menu) { oldPen = NtGdiSelectPen( hdc, NtGdiGetStockObject(DC_PEN) ); IntSetDCPenColor( hdc, IntGetSysColor(COLOR_BTNSHADOW)); GreMoveTo( hdc, rc.left, rc.top, NULL ); NtGdiLineTo( hdc, rc.right, rc.top ); NtGdiSelectPen( hdc, oldPen ); } else DrawEdge (hdc, &rc, EDGE_ETCHED, BF_TOP); return; } #if 0 /* helper lines for debugging */ /* This is a very good test tool when hacking menus! (JT) 07/16/2006 */ FrameRect(hdc, &rect, NtGdiGetStockObject(BLACK_BRUSH)); NtGdiSelectPen(hdc, NtGdiGetStockObject(DC_PEN)); IntSetDCPenColor(hdc, IntGetSysColor(COLOR_WINDOWFRAME)); GreMoveTo(hdc, rect.left, (rect.top + rect.bottom) / 2, NULL); NtGdiLineTo(hdc, rect.right, (rect.top + rect.bottom) / 2); #endif #if 0 // breaks mdi menu bar icons. if (lpitem->hbmp) { /* calculate the bitmap rectangle in coordinates relative * to the item rectangle */ if( menuBar) { if( lpitem->hbmp == HBMMENU_CALLBACK) bmprc.left = 3; else bmprc.left = lpitem->Xlpstr ? MenuCharSize.cx : 0; } else if ((Menu->fFlags & MNS_STYLE_MASK) & MNS_NOCHECK) bmprc.left = 4; else if ((Menu->fFlags & MNS_STYLE_MASK) & MNS_CHECKORBMP) bmprc.left = 2; else bmprc.left = 4 + UserGetSystemMetrics(SM_CXMENUCHECK); bmprc.right = bmprc.left + lpitem->cxBmp; if( menuBar && !(lpitem->hbmp == HBMMENU_CALLBACK)) bmprc.top = 0; else bmprc.top = (rect.bottom - rect.top - lpitem->cyBmp) / 2; bmprc.bottom = bmprc.top + lpitem->cyBmp; } #endif if (!menuBar) { HBITMAP bm; INT y = rect.top + rect.bottom; RECT rc = rect; BOOL checked = FALSE; UINT check_bitmap_width = UserGetSystemMetrics( SM_CXMENUCHECK ); UINT check_bitmap_height = UserGetSystemMetrics( SM_CYMENUCHECK ); /* Draw the check mark * * FIXME: * Custom checkmark bitmaps are monochrome but not always 1bpp. */ if( !((Menu->fFlags & MNS_STYLE_MASK) & MNS_NOCHECK)) { bm = (lpitem->fState & MF_CHECKED) ? lpitem->hbmpChecked : lpitem->hbmpUnchecked; if (bm) /* we have a custom bitmap */ { HDC hdcMem = NtGdiCreateCompatibleDC( hdc ); NtGdiSelectBitmap( hdcMem, bm ); NtGdiBitBlt( hdc, rc.left, (y - check_bitmap_height) / 2, check_bitmap_width, check_bitmap_height, hdcMem, 0, 0, SRCCOPY, 0,0); IntGdiDeleteDC( hdcMem, FALSE ); checked = TRUE; } else if (lpitem->fState & MF_CHECKED) /* standard bitmaps */ { RECT r; r = rect; r.right = r.left + check_bitmap_width; UITOOLS95_DrawFrameMenu(hdc, &r, (lpitem->fType & MFT_RADIOCHECK) ? DFCS_MENUBULLET : DFCS_MENUCHECK); checked = TRUE; } } if ( lpitem->hbmp )//&& !( checked && ((Menu->fFlags & MNS_STYLE_MASK) & MNS_CHECKORBMP))) { RECT bmpRect = rect; if (!((Menu->fFlags & MNS_STYLE_MASK) & MNS_CHECKORBMP) && !((Menu->fFlags & MNS_STYLE_MASK) & MNS_NOCHECK)) bmpRect.left += check_bitmap_width + 2; if (!(checked && ((Menu->fFlags & MNS_STYLE_MASK) & MNS_CHECKORBMP))) { bmpRect.right = bmpRect.left + lpitem->cxBmp; MENU_DrawBitmapItem(hdc, lpitem, &bmpRect, Menu, WndOwner, odaction, menuBar); } } /* Draw the popup-menu arrow */ if (lpitem->spSubMenu) { RECT rectTemp; RtlCopyMemory(&rectTemp, &rect, sizeof(RECT)); rectTemp.left = rectTemp.right - check_bitmap_width; UITOOLS95_DrawFrameMenu(hdc, &rectTemp, DFCS_MENUARROW); } rect.left += 4; if( !((Menu->fFlags & MNS_STYLE_MASK) & MNS_NOCHECK)) rect.left += check_bitmap_width; rect.right -= arrow_bitmap_width; } else if( lpitem->hbmp) { /* Draw the bitmap */ MENU_DrawBitmapItem(hdc, lpitem, &rect/*bmprc*/, Menu, WndOwner, odaction, menuBar); } /* process text if present */ if (lpitem->Xlpstr) { int i = 0; HFONT hfontOld = 0; UINT uFormat = menuBar ? DT_CENTER | DT_VCENTER | DT_SINGLELINE : DT_LEFT | DT_VCENTER | DT_SINGLELINE; if (((Menu->fFlags & MNS_STYLE_MASK) & MNS_CHECKORBMP)) rect.left += max(0, (int)(Menu->cxTextAlign - UserGetSystemMetrics(SM_CXMENUCHECK))); else rect.left += Menu->cxTextAlign; if ( lpitem->fState & MFS_DEFAULT ) { hfontOld = NtGdiSelectFont(hdc, ghMenuFontBold); } if (menuBar) { if( lpitem->hbmp) rect.left += lpitem->cxBmp; if( !(lpitem->hbmp == HBMMENU_CALLBACK)) rect.left += MenuCharSize.cx; rect.right -= MenuCharSize.cx; } Text = lpitem->Xlpstr; if(Text) { for (i = 0; Text[i]; i++) if (Text[i] == L'\t' || Text[i] == L'\b') break; } if (menuBar && !flat_menu && (lpitem->fState & (MF_HILITE | MF_GRAYED)) == MF_HILITE) { RECTL_vOffsetRect(&rect, +1, +1); } if (!menuBar) --rect.bottom; if(lpitem->fState & MF_GRAYED) { if (!(lpitem->fState & MF_HILITE) ) { ++rect.left; ++rect.top; ++rect.right; ++rect.bottom; IntGdiSetTextColor(hdc, IntGetSysColor(COLOR_BTNHIGHLIGHT)); DrawTextW( hdc, Text, i, &rect, uFormat ); --rect.left; --rect.top; --rect.right; --rect.bottom; } IntGdiSetTextColor(hdc, IntGetSysColor(COLOR_BTNSHADOW)); } DrawTextW( hdc, Text, i, &rect, uFormat); /* paint the shortcut text */ if (!menuBar && L'\0' != Text[i]) /* There's a tab or flush-right char */ { if (L'\t' == Text[i]) { rect.left = lpitem->dxTab; uFormat = DT_LEFT | DT_VCENTER | DT_SINGLELINE; } else { rect.right = lpitem->dxTab; uFormat = DT_RIGHT | DT_VCENTER | DT_SINGLELINE; } if (lpitem->fState & MF_GRAYED) { if (!(lpitem->fState & MF_HILITE) ) { ++rect.left; ++rect.top; ++rect.right; ++rect.bottom; IntGdiSetTextColor(hdc, IntGetSysColor(COLOR_BTNHIGHLIGHT)); DrawTextW( hdc, Text + i + 1, -1, &rect, uFormat); --rect.left; --rect.top; --rect.right; --rect.bottom; } IntGdiSetTextColor(hdc, IntGetSysColor(COLOR_BTNSHADOW)); } DrawTextW( hdc, Text + i + 1, -1, &rect, uFormat ); } if (!menuBar) ++rect.bottom; if (menuBar && !flat_menu && (lpitem->fState & (MF_HILITE | MF_GRAYED)) == MF_HILITE) { RECTL_vOffsetRect(&rect, -1, -1); } if (hfontOld) { NtGdiSelectFont (hdc, hfontOld); } } } /*********************************************************************** * MenuDrawPopupMenu * * Paint a popup menu. */ static void FASTCALL MENU_DrawPopupMenu(PWND wnd, HDC hdc, PMENU menu ) { HBRUSH hPrevBrush = 0, brush = IntGetSysColorBrush(COLOR_MENU); RECT rect; TRACE("DPM wnd=%p dc=%p menu=%p\n", wnd, hdc, menu); IntGetClientRect( wnd, &rect ); if (menu && menu->hbrBack) brush = menu->hbrBack; if((hPrevBrush = NtGdiSelectBrush( hdc, brush )) && (NtGdiSelectFont( hdc, ghMenuFont))) { HPEN hPrevPen; /* FIXME: Maybe we don't have to fill the background manually */ FillRect(hdc, &rect, brush); hPrevPen = NtGdiSelectPen( hdc, NtGdiGetStockObject( NULL_PEN ) ); if ( hPrevPen ) { TRACE("hmenu %p Style %08x\n", UserHMGetHandle(menu), (menu->fFlags & MNS_STYLE_MASK)); /* draw menu items */ if (menu && menu->cItems) { ITEM *item; UINT u; item = menu->rgItems; for( u = menu->cItems; u > 0; u--, item++) { MENU_DrawMenuItem(wnd, menu, menu->spwndNotify, hdc, item, menu->cyMenu, FALSE, ODA_DRAWENTIRE); } /* draw scroll arrows */ if (menu->dwArrowsOn) { MENU_DrawScrollArrows(menu, hdc); } } } else { NtGdiSelectBrush( hdc, hPrevBrush ); } } } /********************************************************************** * MENU_IsMenuActive */ PWND MENU_IsMenuActive(VOID) { return ValidateHwndNoErr(top_popup); } /********************************************************************** * MENU_EndMenu * * Calls EndMenu() if the hwnd parameter belongs to the menu owner * * Does the (menu stuff) of the default window handling of WM_CANCELMODE */ void MENU_EndMenu( PWND pwnd ) { PMENU menu = NULL; menu = UserGetMenuObject(top_popup_hmenu); if ( menu && ( UserHMGetHandle(pwnd) == menu->hWnd || pwnd == menu->spwndNotify ) ) { if (fInsideMenuLoop && top_popup) { fInsideMenuLoop = FALSE; if (fInEndMenu) { ERR("Already in End loop\n"); return; } fInEndMenu = TRUE; UserPostMessage( top_popup, WM_CANCELMODE, 0, 0); } } } DWORD WINAPI IntDrawMenuBarTemp(PWND pWnd, HDC hDC, LPRECT Rect, PMENU pMenu, HFONT Font) { UINT i; HFONT FontOld = NULL; BOOL flat_menu = FALSE; UserSystemParametersInfo(SPI_GETFLATMENU, 0, &flat_menu, 0); if (!pMenu) { pMenu = UserGetMenuObject(UlongToHandle(pWnd->IDMenu)); } if (!Font) { Font = ghMenuFont; } if (Rect == NULL || !pMenu) { return UserGetSystemMetrics(SM_CYMENU); } TRACE("(%x, %x, %p, %x, %x)\n", pWnd, hDC, Rect, pMenu, Font); FontOld = NtGdiSelectFont(hDC, Font); if (pMenu->cyMenu == 0) { MENU_MenuBarCalcSize(hDC, Rect, pMenu, pWnd); } Rect->bottom = Rect->top + pMenu->cyMenu; FillRect(hDC, Rect, IntGetSysColorBrush(flat_menu ? COLOR_MENUBAR : COLOR_MENU)); NtGdiSelectPen(hDC, NtGdiGetStockObject(DC_PEN)); IntSetDCPenColor(hDC, IntGetSysColor(COLOR_3DFACE)); GreMoveTo(hDC, Rect->left, Rect->bottom - 1, NULL); NtGdiLineTo(hDC, Rect->right, Rect->bottom - 1); if (pMenu->cItems == 0) { NtGdiSelectFont(hDC, FontOld); return UserGetSystemMetrics(SM_CYMENU); } for (i = 0; i < pMenu->cItems; i++) { MENU_DrawMenuItem(pWnd, pMenu, pWnd, hDC, &pMenu->rgItems[i], pMenu->cyMenu, TRUE, ODA_DRAWENTIRE); } NtGdiSelectFont(hDC, FontOld); return pMenu->cyMenu; } UINT MENU_DrawMenuBar( HDC hDC, LPRECT lprect, PWND pWnd, BOOL suppress_draw ) { HFONT hfontOld = 0; PMENU lppop = UserGetMenuObject(UlongToHandle(pWnd->IDMenu)); if (lppop == NULL) { // No menu. Do not reserve any space return 0; } if (lprect == NULL) { return UserGetSystemMetrics(SM_CYMENU); } if (suppress_draw) { hfontOld = NtGdiSelectFont(hDC, ghMenuFont); MENU_MenuBarCalcSize(hDC, lprect, lppop, pWnd); lprect->bottom = lprect->top + lppop->cyMenu; if (hfontOld) NtGdiSelectFont( hDC, hfontOld); return lppop->cyMenu; } else { return IntDrawMenuBarTemp(pWnd, hDC, lprect, lppop, NULL); } } /*********************************************************************** * MENU_InitPopup * * Popup menu initialization before WM_ENTERMENULOOP. */ static BOOL MENU_InitPopup( PWND pWndOwner, PMENU menu, UINT flags ) { PWND pWndCreated; PPOPUPMENU pPopupMenu; CREATESTRUCTW Cs; LARGE_STRING WindowName; UNICODE_STRING ClassName; DWORD ex_style = WS_EX_PALETTEWINDOW | WS_EX_DLGMODALFRAME; TRACE("owner=%p hmenu=%p\n", pWndOwner, menu); menu->spwndNotify = pWndOwner; if (flags & TPM_LAYOUTRTL || pWndOwner->ExStyle & WS_EX_LAYOUTRTL) ex_style |= WS_EX_LAYOUTRTL; ClassName.Buffer = WC_MENU; ClassName.Length = 0; RtlZeroMemory(&WindowName, sizeof(WindowName)); RtlZeroMemory(&Cs, sizeof(Cs)); Cs.style = WS_POPUP | WS_CLIPSIBLINGS | WS_BORDER; Cs.dwExStyle = ex_style; Cs.hInstance = hModClient; // hModuleWin; // Server side winproc! Cs.lpszName = (LPCWSTR) &WindowName; Cs.lpszClass = (LPCWSTR) &ClassName; Cs.lpCreateParams = UserHMGetHandle(menu); Cs.hwndParent = UserHMGetHandle(pWndOwner); /* NOTE: In Windows, top menu popup is not owned. */ pWndCreated = co_UserCreateWindowEx( &Cs, &ClassName, &WindowName, NULL, WINVER ); if( !pWndCreated ) return FALSE; // // Setup pop up menu structure. // menu->hWnd = UserHMGetHandle(pWndCreated); pPopupMenu = ((PMENUWND)pWndCreated)->ppopupmenu; pPopupMenu->spwndActivePopup = pWndCreated; // top_popup = MenuInfo.Wnd or menu->hWnd pPopupMenu->spwndNotify = pWndOwner; // Same as MenuInfo.spwndNotify(which could be wrong) or menu->hwndOwner //pPopupMenu->spmenu = menu; Should be set up already from WM_CREATE! pPopupMenu->fIsTrackPopup = !!(flags & TPM_POPUPMENU); pPopupMenu->fIsSysMenu = !!(flags & TPM_SYSTEM_MENU); pPopupMenu->fNoNotify = !!(flags & TPM_NONOTIFY); pPopupMenu->fRightButton = !!(flags & TPM_RIGHTBUTTON); pPopupMenu->fSynchronous = !!(flags & TPM_RETURNCMD); if (pPopupMenu->fRightButton) pPopupMenu->fFirstClick = !!(UserGetKeyState(VK_RBUTTON) & 0x8000); else pPopupMenu->fFirstClick = !!(UserGetKeyState(VK_LBUTTON) & 0x8000); if (gpsi->aiSysMet[SM_MENUDROPALIGNMENT] || menu->fFlags & MNF_RTOL) { pPopupMenu->fDroppedLeft = TRUE; } return TRUE; } #define SHOW_DEBUGRECT 0 #if SHOW_DEBUGRECT static void DebugRect(const RECT* rectl, COLORREF color) { HBRUSH brush; RECT rr; HDC hdc; if (!rectl) return; hdc = UserGetDCEx(NULL, 0, DCX_USESTYLE); brush = IntGdiCreateSolidBrush(color); rr = *rectl; RECTL_vInflateRect(&rr, 1, 1); FrameRect(hdc, rectl, brush); FrameRect(hdc, &rr, brush); NtGdiDeleteObjectApp(brush); UserReleaseDC(NULL, hdc, TRUE); } static void DebugPoint(INT x, INT y, COLORREF color) { RECT r1 = {x-10, y, x+10, y}; RECT r2 = {x, y-10, x, y+10}; DebugRect(&r1, color); DebugRect(&r2, color); } #endif static BOOL RECTL_Intersect(const RECT* pRect, INT x, INT y, UINT width, UINT height) { RECT other = {x, y, x + width, y + height}; RECT dum; return RECTL_bIntersectRect(&dum, pRect, &other); } static BOOL MENU_MoveRect(UINT flags, INT* x, INT* y, INT width, INT height, const RECT* pExclude, PMONITOR monitor) { /* Figure out if we should move vertical or horizontal */ if (flags & TPM_VERTICAL) { /* Move in the vertical direction: TPM_BOTTOMALIGN means drop it above, otherways drop it below */ if (flags & TPM_BOTTOMALIGN) { if (pExclude->top - height >= monitor->rcMonitor.top) { *y = pExclude->top - height; return TRUE; } } else { if (pExclude->bottom + height < monitor->rcMonitor.bottom) { *y = pExclude->bottom; return TRUE; } } } else { /* Move in the horizontal direction: TPM_RIGHTALIGN means drop it to the left, otherways go right */ if (flags & TPM_RIGHTALIGN) { if (pExclude->left - width >= monitor->rcMonitor.left) { *x = pExclude->left - width; return TRUE; } } else { if (pExclude->right + width < monitor->rcMonitor.right) { *x = pExclude->right; return TRUE; } } } return FALSE; } /*********************************************************************** * MenuShowPopup * * Display a popup menu. */ static BOOL FASTCALL MENU_ShowPopup(PWND pwndOwner, PMENU menu, UINT id, UINT flags, INT x, INT y, const RECT* pExclude) { INT width, height; POINT ptx; PMONITOR monitor; PWND pWnd; USER_REFERENCE_ENTRY Ref; BOOL bIsPopup = (flags & TPM_POPUPMENU) != 0; TRACE("owner=%p menu=%p id=0x%04x x=0x%04x y=0x%04x\n", pwndOwner, menu, id, x, y); if (menu->iItem != NO_SELECTED_ITEM) { menu->rgItems[menu->iItem].fState &= ~(MF_HILITE|MF_MOUSESELECT); menu->iItem = NO_SELECTED_ITEM; } #if SHOW_DEBUGRECT if (pExclude) DebugRect(pExclude, RGB(255, 0, 0)); #endif menu->dwArrowsOn = 0; MENU_PopupMenuCalcSize(menu, pwndOwner); /* adjust popup menu pos so that it fits within the desktop */ width = menu->cxMenu + UserGetSystemMetrics(SM_CXDLGFRAME) * 2; height = menu->cyMenu + UserGetSystemMetrics(SM_CYDLGFRAME) * 2; if (flags & TPM_LAYOUTRTL) flags ^= TPM_RIGHTALIGN; if (flags & TPM_RIGHTALIGN) x -= width; if (flags & TPM_CENTERALIGN) x -= width / 2; if (flags & TPM_BOTTOMALIGN) y -= height; if (flags & TPM_VCENTERALIGN) y -= height / 2; /* FIXME: should use item rect */ ptx.x = x; ptx.y = y; #if SHOW_DEBUGRECT DebugPoint(x, y, RGB(0, 0, 255)); #endif monitor = UserMonitorFromPoint( ptx, MONITOR_DEFAULTTONEAREST ); /* We are off the right side of the screen */ if (x + width > monitor->rcMonitor.right) { /* Position menu at right edge of the screen */ x = monitor->rcMonitor.right - width; } /* We are off the left side of the screen */ if (x < monitor->rcMonitor.left) { /* Position menu at left edge of the screen */ x = 0; if (x < monitor->rcMonitor.left || x >= monitor->rcMonitor.right || bIsPopup) x = monitor->rcMonitor.left; } /* Same here, but then the top */ if (y < monitor->rcMonitor.top) { y += height; if (y < monitor->rcMonitor.top || y >= monitor->rcMonitor.bottom || bIsPopup) y = monitor->rcMonitor.top; } /* And the bottom */ if (y + height > monitor->rcMonitor.bottom) { if ((y - height) < monitor->rcMonitor.top || y >= monitor->rcMonitor.bottom) y = monitor->rcMonitor.bottom - height; else { INT adjHgt = y + UserGetSystemMetrics(SM_CYMENUSIZE) + 2 * UserGetSystemMetrics(SM_CYDLGFRAME); if (adjHgt >= monitor->rcMonitor.bottom) y -= height; else y = adjHgt - height; } } if (pExclude) { RECT Cleaned; if (RECTL_bIntersectRect(&Cleaned, pExclude, &monitor->rcMonitor) && RECTL_Intersect(&Cleaned, x, y, width, height)) { UINT flag_mods[] = { 0, /* First try the 'normal' way */ TPM_BOTTOMALIGN | TPM_RIGHTALIGN, /* Then try the opposite side */ TPM_VERTICAL, /* Then swap horizontal / vertical */ TPM_BOTTOMALIGN | TPM_RIGHTALIGN | TPM_VERTICAL, /* Then the other side again (still swapped hor/ver) */ }; UINT n; for (n = 0; n < RTL_NUMBER_OF(flag_mods); ++n) { INT tx = x; INT ty = y; /* Try to move a bit around */ if (MENU_MoveRect(flags ^ flag_mods[n], &tx, &ty, width, height, &Cleaned, monitor) && !RECTL_Intersect(&Cleaned, tx, ty, width, height)) { x = tx; y = ty; break; } } /* If none worked, we go with the original x/y */ } } #if SHOW_DEBUGRECT { RECT rr = {x, y, x + width, y + height}; DebugRect(&rr, RGB(0, 255, 0)); } #endif pWnd = ValidateHwndNoErr( menu->hWnd ); if (!pWnd) { ERR("menu->hWnd bad hwnd %p\n",menu->hWnd); return FALSE; } if (!top_popup) { top_popup = menu->hWnd; top_popup_hmenu = UserHMGetHandle(menu); } /* Display the window */ UserRefObjectCo(pWnd, &Ref); co_WinPosSetWindowPos( pWnd, HWND_TOPMOST, x, y, width, height, SWP_SHOWWINDOW | SWP_NOACTIVATE); co_IntUpdateWindows(pWnd, RDW_ALLCHILDREN, FALSE); IntNotifyWinEvent(EVENT_SYSTEM_MENUPOPUPSTART, pWnd, OBJID_CLIENT, CHILDID_SELF, 0); UserDerefObjectCo(pWnd); return TRUE; } /*********************************************************************** * MENU_EnsureMenuItemVisible */ void MENU_EnsureMenuItemVisible(PMENU lppop, UINT wIndex, HDC hdc) { USER_REFERENCE_ENTRY Ref; if (lppop->dwArrowsOn) { ITEM *item = &lppop->rgItems[wIndex]; UINT nMaxHeight = MENU_GetMaxPopupHeight(lppop); UINT nOldPos = lppop->iTop; RECT rc; UINT arrow_bitmap_height; PWND pWnd = ValidateHwndNoErr(lppop->hWnd); IntGetClientRect(pWnd, &rc); arrow_bitmap_height = gpsi->oembmi[OBI_DNARROW].cy; rc.top += arrow_bitmap_height; rc.bottom -= arrow_bitmap_height; nMaxHeight -= UserGetSystemMetrics(SM_CYBORDER) + 2 * arrow_bitmap_height; UserRefObjectCo(pWnd, &Ref); if (item->cyItem > lppop->iTop + nMaxHeight) { lppop->iTop = item->cyItem - nMaxHeight; IntScrollWindow(pWnd, 0, nOldPos - lppop->iTop, &rc, &rc); MENU_DrawScrollArrows(lppop, hdc); //ERR("Scroll Down iTop %d iMaxTop %d nMaxHeight %d\n",lppop->iTop,lppop->iMaxTop,nMaxHeight); } else if (item->yItem < lppop->iTop) { lppop->iTop = item->yItem; IntScrollWindow(pWnd, 0, nOldPos - lppop->iTop, &rc, &rc); MENU_DrawScrollArrows(lppop, hdc); //ERR("Scroll Up iTop %d iMaxTop %d nMaxHeight %d\n",lppop->iTop,lppop->iMaxTop,nMaxHeight); } UserDerefObjectCo(pWnd); } } /*********************************************************************** * MenuSelectItem */ static void FASTCALL MENU_SelectItem(PWND pwndOwner, PMENU menu, UINT wIndex, BOOL sendMenuSelect, PMENU topmenu) { HDC hdc; PWND pWnd; TRACE("M_SI: owner=%p menu=%p index=0x%04x select=0x%04x\n", pwndOwner, menu, wIndex, sendMenuSelect); if (!menu || !menu->cItems) return; pWnd = ValidateHwndNoErr(menu->hWnd); if (!pWnd) return; if (menu->iItem == wIndex) return; if (menu->fFlags & MNF_POPUP) hdc = UserGetDCEx(pWnd, 0, DCX_USESTYLE); else hdc = UserGetDCEx(pWnd, 0, DCX_CACHE | DCX_WINDOW); if (!top_popup) { top_popup = menu->hWnd; //pPopupMenu->spwndActivePopup or //pPopupMenu->fIsTrackPopup set pPopupMenu->spwndPopupMenu; top_popup_hmenu = UserHMGetHandle(menu); //pPopupMenu->spmenu } NtGdiSelectFont( hdc, ghMenuFont ); /* Clear previous highlighted item */ if (menu->iItem != NO_SELECTED_ITEM) { menu->rgItems[menu->iItem].fState &= ~(MF_HILITE|MF_MOUSESELECT); MENU_DrawMenuItem(pWnd, menu, pwndOwner, hdc, &menu->rgItems[menu->iItem], menu->cyMenu, !(menu->fFlags & MNF_POPUP), ODA_SELECT); } /* Highlight new item (if any) */ menu->iItem = wIndex; if (menu->iItem != NO_SELECTED_ITEM) { if (!(menu->rgItems[wIndex].fType & MF_SEPARATOR)) { menu->rgItems[wIndex].fState |= MF_HILITE; MENU_EnsureMenuItemVisible(menu, wIndex, hdc); MENU_DrawMenuItem(pWnd, menu, pwndOwner, hdc, &menu->rgItems[wIndex], menu->cyMenu, !(menu->fFlags & MNF_POPUP), ODA_SELECT); } if (sendMenuSelect) { ITEM *ip = &menu->rgItems[menu->iItem]; WPARAM wParam = MAKEWPARAM( ip->spSubMenu ? wIndex : ip->wID, ip->fType | ip->fState | (ip->spSubMenu ? MF_POPUP : 0) | (menu->fFlags & MNF_SYSMENU ? MF_SYSMENU : 0 ) ); co_IntSendMessage(UserHMGetHandle(pwndOwner), WM_MENUSELECT, wParam, (LPARAM) UserHMGetHandle(menu)); } } else if (sendMenuSelect) { if (topmenu) { int pos; pos = MENU_FindSubMenu(&topmenu, menu); if (pos != NO_SELECTED_ITEM) { ITEM *ip = &topmenu->rgItems[pos]; WPARAM wParam = MAKEWPARAM( Pos, ip->fType | ip->fState | (ip->spSubMenu ? MF_POPUP : 0) | (topmenu->fFlags & MNF_SYSMENU ? MF_SYSMENU : 0 ) ); co_IntSendMessage(UserHMGetHandle(pwndOwner), WM_MENUSELECT, wParam, (LPARAM) UserHMGetHandle(topmenu)); } } } UserReleaseDC(pWnd, hdc, FALSE); } /*********************************************************************** * MenuMoveSelection * * Moves currently selected item according to the Offset parameter. * If there is no selection then it should select the last item if * Offset is ITEM_PREV or the first item if Offset is ITEM_NEXT. */ static void FASTCALL MENU_MoveSelection(PWND pwndOwner, PMENU menu, INT offset) { INT i; TRACE("pwnd=%x menu=%x off=0x%04x\n", pwndOwner, menu, offset); if ((!menu) || (!menu->rgItems)) return; if ( menu->iItem != NO_SELECTED_ITEM ) { if ( menu->cItems == 1 ) return; else for (i = menu->iItem + offset ; i >= 0 && i < menu->cItems ; i += offset) if (!(menu->rgItems[i].fType & MF_SEPARATOR)) { MENU_SelectItem( pwndOwner, menu, i, TRUE, 0 ); return; } } for ( i = (offset > 0) ? 0 : menu->cItems - 1; i >= 0 && i < menu->cItems ; i += offset) if (!(menu->rgItems[i].fType & MF_SEPARATOR)) { MENU_SelectItem( pwndOwner, menu, i, TRUE, 0 ); return; } } /*********************************************************************** * MenuHideSubPopups * * Hide the sub-popup menus of this menu. */ static void FASTCALL MENU_HideSubPopups(PWND pWndOwner, PMENU Menu, BOOL SendMenuSelect, UINT wFlags) { TRACE("owner=%x menu=%x 0x%04x\n", pWndOwner, Menu, SendMenuSelect); if ( Menu && top_popup ) { PITEM Item; if (Menu->iItem != NO_SELECTED_ITEM) { Item = &Menu->rgItems[Menu->iItem]; if (!(Item->spSubMenu) || !(Item->fState & MF_MOUSESELECT)) return; Item->fState &= ~MF_MOUSESELECT; } else return; if (Item->spSubMenu) { PWND pWnd; if (!VerifyMenu(Item->spSubMenu)) return; MENU_HideSubPopups(pWndOwner, Item->spSubMenu, FALSE, wFlags); MENU_SelectItem(pWndOwner, Item->spSubMenu, NO_SELECTED_ITEM, SendMenuSelect, NULL); TRACE("M_HSP top p hm %p pWndOwner IDMenu %p\n",top_popup_hmenu,pWndOwner->IDMenu); pWnd = ValidateHwndNoErr(Item->spSubMenu->hWnd); if (pWnd != NULL) { co_UserDestroyWindow(pWnd); } /* Native returns handle to destroyed window */ if (!(wFlags & TPM_NONOTIFY)) { co_IntSendMessage( UserHMGetHandle(pWndOwner), WM_UNINITMENUPOPUP, (WPARAM)UserHMGetHandle(Item->spSubMenu), MAKELPARAM(0, IS_SYSTEM_MENU(Item->spSubMenu) ? MF_SYSMENU : 0)); } //// // Call WM_UNINITMENUPOPUP FIRST before destroy!! // Fixes todo_wine User32 test menu.c line 2239 GetMenuBarInfo callback.... // Item->spSubMenu->hWnd = NULL; //// } } } /*********************************************************************** * MenuShowSubPopup * * Display the sub-menu of the selected item of this menu. * Return the handle of the submenu, or menu if no submenu to display. */ static PMENU FASTCALL MENU_ShowSubPopup(PWND WndOwner, PMENU Menu, BOOL SelectFirst, UINT Flags) { RECT Rect, ParentRect; ITEM *Item; HDC Dc; PWND pWnd; TRACE("owner=%x menu=%p 0x%04x\n", WndOwner, Menu, SelectFirst); if (!Menu) return Menu; if (Menu->iItem == NO_SELECTED_ITEM) return Menu; Item = &Menu->rgItems[Menu->iItem]; if (!(Item->spSubMenu) || (Item->fState & (MF_GRAYED | MF_DISABLED))) return Menu; /* message must be sent before using item, because nearly everything may be changed by the application ! */ /* Send WM_INITMENUPOPUP message only if TPM_NONOTIFY flag is not specified */ if (!(Flags & TPM_NONOTIFY)) { co_IntSendMessage(UserHMGetHandle(WndOwner), WM_INITMENUPOPUP, (WPARAM) UserHMGetHandle(Item->spSubMenu), MAKELPARAM(Menu->iItem, IS_SYSTEM_MENU(Menu))); } Item = &Menu->rgItems[Menu->iItem]; //Rect = ItemInfo.Rect; Rect.left = Item->xItem; Rect.top = Item->yItem; Rect.right = Item->cxItem; // Do this for now...... Rect.bottom = Item->cyItem; pWnd = ValidateHwndNoErr(Menu->hWnd); /* Grab the rect of our (entire) parent menu, so we can try to not overlap it */ if (Menu->fFlags & MNF_POPUP) { if (!IntGetWindowRect(pWnd, &ParentRect)) { ERR("No pWnd\n"); ParentRect = Rect; } /* Ensure we can slightly overlap our parent */ RECTL_vInflateRect(&ParentRect, -UserGetSystemMetrics(SM_CXEDGE) * 2, 0); } else { /* Inside the menu bar, we do not want to grab the entire window... */ ParentRect = Rect; if (pWnd) RECTL_vOffsetRect(&ParentRect, pWnd->rcWindow.left, pWnd->rcWindow.top); } /* correct item if modified as a reaction to WM_INITMENUPOPUP message */ if (!(Item->fState & MF_HILITE)) { if (Menu->fFlags & MNF_POPUP) Dc = UserGetDCEx(pWnd, NULL, DCX_USESTYLE); else Dc = UserGetDCEx(pWnd, 0, DCX_CACHE | DCX_WINDOW); NtGdiSelectFont(Dc, ghMenuFont); Item->fState |= MF_HILITE; MENU_DrawMenuItem(pWnd, Menu, WndOwner, Dc, Item, Menu->cyMenu, !(Menu->fFlags & MNF_POPUP), ODA_DRAWENTIRE); UserReleaseDC(pWnd, Dc, FALSE); } if (!Item->yItem && !Item->xItem && !Item->cyItem && !Item->cxItem) { Item->xItem = Rect.left; Item->yItem = Rect.top; Item->cxItem = Rect.right; // Do this for now...... Item->cyItem = Rect.bottom; } Item->fState |= MF_MOUSESELECT; if (IS_SYSTEM_MENU(Menu) && !(Menu->fFlags & MNF_POPUP)) { MENU_InitSysMenuPopup(Item->spSubMenu, pWnd->style, pWnd->pcls->style, HTSYSMENU); NC_GetSysPopupPos(pWnd, &Rect); /* Ensure we do not overlap this */ ParentRect = Rect; if (Flags & TPM_LAYOUTRTL) Rect.left = Rect.right; Rect.top = Rect.bottom; Rect.right = UserGetSystemMetrics(SM_CXSIZE); Rect.bottom = UserGetSystemMetrics(SM_CYSIZE); } else { IntGetWindowRect(pWnd, &Rect); if (Menu->fFlags & MNF_POPUP) { RECT rc; rc.left = Item->xItem; rc.top = Item->yItem; rc.right = Item->cxItem; rc.bottom = Item->cyItem; MENU_AdjustMenuItemRect(Menu, &rc); /* The first item in the popup menu has to be at the same y position as the focused menu item */ if(Flags & TPM_LAYOUTRTL) Rect.left += UserGetSystemMetrics(SM_CXDLGFRAME); else Rect.left += rc.right - UserGetSystemMetrics(SM_CXDLGFRAME); Rect.top += rc.top; } else { if(Flags & TPM_LAYOUTRTL) Rect.left += Rect.right - Item->xItem; //ItemInfo.Rect.left; else Rect.left += Item->xItem; //ItemInfo.Rect.left; Rect.top += Item->cyItem; //ItemInfo.Rect.bottom; Rect.right = Item->cxItem - Item->xItem; //ItemInfo.Rect.right - ItemInfo.Rect.left; Rect.bottom = Item->cyItem - Item->yItem; //ItemInfo.Rect.bottom - ItemInfo.Rect.top; } } /* Next menu does not need to be shown vertical anymore */ if (Menu->fFlags & MNF_POPUP) Flags &= (~TPM_VERTICAL); /* use default alignment for submenus */ Flags &= ~(TPM_CENTERALIGN | TPM_RIGHTALIGN | TPM_VCENTERALIGN | TPM_BOTTOMALIGN); MENU_InitPopup( WndOwner, Item->spSubMenu, Flags ); MENU_ShowPopup( WndOwner, Item->spSubMenu, Menu->iItem, Flags, Rect.left, Rect.top, &ParentRect); if (SelectFirst) { MENU_MoveSelection(WndOwner, Item->spSubMenu, ITEM_NEXT); } return Item->spSubMenu; } /*********************************************************************** * MenuExecFocusedItem * * Execute a menu item (for instance when user pressed Enter). * Return the wID of the executed item. Otherwise, -1 indicating * that no menu item was executed, -2 if a popup is shown; * Have to receive the flags for the TrackPopupMenu options to avoid * sending unwanted message. * */ static INT FASTCALL MENU_ExecFocusedItem(MTRACKER *pmt, PMENU Menu, UINT Flags) { PITEM Item; TRACE("%p menu=%p\n", pmt, Menu); if (!Menu || !Menu->cItems || Menu->iItem == NO_SELECTED_ITEM) { return -1; } Item = &Menu->rgItems[Menu->iItem]; TRACE("%p %08x %p\n", Menu, Item->wID, Item->spSubMenu); if (!(Item->spSubMenu)) { if (!(Item->fState & (MF_GRAYED | MF_DISABLED)) && !(Item->fType & MF_SEPARATOR)) { /* If TPM_RETURNCMD is set you return the id, but do not send a message to the owner */ if (!(Flags & TPM_RETURNCMD)) { if (Menu->fFlags & MNF_SYSMENU) { UserPostMessage(UserHMGetHandle(pmt->OwnerWnd), WM_SYSCOMMAND, Item->wID, MAKELPARAM((SHORT) pmt->Pt.x, (SHORT) pmt->Pt.y)); } else { DWORD dwStyle = ((Menu->fFlags & MNS_STYLE_MASK) | ( pmt->TopMenu ? (pmt->TopMenu->fFlags & MNS_STYLE_MASK) : 0) ); if (dwStyle & MNS_NOTIFYBYPOS) UserPostMessage(UserHMGetHandle(pmt->OwnerWnd), WM_MENUCOMMAND, Menu->iItem, (LPARAM)UserHMGetHandle(Menu)); else UserPostMessage(UserHMGetHandle(pmt->OwnerWnd), WM_COMMAND, Item->wID, 0); } } return Item->wID; } } else { pmt->CurrentMenu = MENU_ShowSubPopup(pmt->OwnerWnd, Menu, TRUE, Flags); return -2; } return -1; } /*********************************************************************** * MenuSwitchTracking * * Helper function for menu navigation routines. */ static void FASTCALL MENU_SwitchTracking(MTRACKER* pmt, PMENU PtMenu, UINT Index, UINT wFlags) { TRACE("%x menu=%x 0x%04x\n", pmt, PtMenu, Index); if ( pmt->TopMenu != PtMenu && !((PtMenu->fFlags | pmt->TopMenu->fFlags) & MNF_POPUP) ) { /* both are top level menus (system and menu-bar) */ MENU_HideSubPopups(pmt->OwnerWnd, pmt->TopMenu, FALSE, wFlags); MENU_SelectItem(pmt->OwnerWnd, pmt->TopMenu, NO_SELECTED_ITEM, FALSE, NULL); pmt->TopMenu = PtMenu; } else { MENU_HideSubPopups(pmt->OwnerWnd, PtMenu, FALSE, wFlags); } MENU_SelectItem(pmt->OwnerWnd, PtMenu, Index, TRUE, NULL); } /*********************************************************************** * MenuButtonDown * * Return TRUE if we can go on with menu tracking. */ static BOOL FASTCALL MENU_ButtonDown(MTRACKER* pmt, PMENU PtMenu, UINT Flags) { TRACE("%x PtMenu=%p\n", pmt, PtMenu); if (PtMenu) { UINT id = 0; PITEM item; // Special check for the icon system menu if (IS_SYSTEM_MENU(PtMenu) && !(PtMenu->fFlags & MNF_POPUP)) { item = PtMenu->rgItems; } else { item = MENU_FindItemByCoords( PtMenu, pmt->Pt, &id ); } if (item) { if (PtMenu->iItem != id) MENU_SwitchTracking(pmt, PtMenu, id, Flags); /* If the popup menu is not already "popped" */ if (!(item->fState & MF_MOUSESELECT)) { pmt->CurrentMenu = MENU_ShowSubPopup(pmt->OwnerWnd, PtMenu, FALSE, Flags); } return TRUE; } /* Else the click was on the menu bar, finish the tracking */ } return FALSE; } /*********************************************************************** * MenuButtonUp * * Return the value of MenuExecFocusedItem if * the selected item was not a popup. Else open the popup. * A -1 return value indicates that we go on with menu tracking. * */ static INT FASTCALL MENU_ButtonUp(MTRACKER *pmt, PMENU PtMenu, UINT Flags) { TRACE("%p pmenu=%x\n", pmt, PtMenu); if (PtMenu) { UINT Id = 0; ITEM *item; // Special check for the icon system menu if (IS_SYSTEM_MENU(PtMenu) && !(PtMenu->fFlags & MNF_POPUP)) { item = PtMenu->rgItems; } else { item = MENU_FindItemByCoords( PtMenu, pmt->Pt, &Id ); } if (item && ( PtMenu->iItem == Id)) { if (!(item->spSubMenu)) { INT ExecutedMenuId = MENU_ExecFocusedItem( pmt, PtMenu, Flags); if (ExecutedMenuId == -1 || ExecutedMenuId == -2) return -1; return ExecutedMenuId; } /* If we are dealing with the menu bar */ /* and this is a click on an already "popped" item: */ /* Stop the menu tracking and close the opened submenus */ if (pmt->TopMenu == PtMenu && PtMenu->TimeToHide) { return 0; } } if ( IntGetMenu(PtMenu->hWnd) == PtMenu ) { PtMenu->TimeToHide = TRUE; } } return -1; } /*********************************************************************** * MenuPtMenu * * Walks menu chain trying to find a menu pt maps to. */ static PMENU FASTCALL MENU_PtMenu(PMENU menu, POINT pt) { PITEM pItem; PMENU ret = NULL; if (!menu) return NULL; /* try subpopup first (if any) */ if (menu->iItem != NO_SELECTED_ITEM) { pItem = menu->rgItems; if ( pItem ) pItem = &pItem[menu->iItem]; if ( pItem && pItem->spSubMenu && pItem->fState & MF_MOUSESELECT) { ret = MENU_PtMenu( pItem->spSubMenu, pt); } } /* check the current window (avoiding WM_HITTEST) */ if (!ret) { PWND pWnd = ValidateHwndNoErr(menu->hWnd); INT ht = GetNCHitEx(pWnd, pt); if ( menu->fFlags & MNF_POPUP ) { if (ht != HTNOWHERE && ht != HTERROR) ret = menu; } else if (ht == HTSYSMENU) ret = get_win_sys_menu(menu->hWnd); else if (ht == HTMENU) ret = IntGetMenu( menu->hWnd ); } return ret; } /*********************************************************************** * MenuMouseMove * * Return TRUE if we can go on with menu tracking. */ static BOOL FASTCALL MENU_MouseMove(MTRACKER *pmt, PMENU PtMenu, UINT Flags) { UINT Index = NO_SELECTED_ITEM; if ( PtMenu ) MENU_FindItemByCoords( PtMenu, pmt->Pt, &Index ); if (Index == NO_SELECTED_ITEM) { MENU_SelectItem(pmt->OwnerWnd, pmt->CurrentMenu, NO_SELECTED_ITEM, TRUE, pmt->TopMenu); } else if (PtMenu->iItem != Index) { MENU_SwitchTracking(pmt, PtMenu, Index, Flags); pmt->CurrentMenu = MENU_ShowSubPopup(pmt->OwnerWnd, PtMenu, FALSE, Flags); } return TRUE; } /*********************************************************************** * MenuGetSubPopup * * Return the handle of the selected sub-popup menu (if any). */ static PMENU MENU_GetSubPopup( PMENU menu ) { ITEM *item; if ((!menu) || (menu->iItem == NO_SELECTED_ITEM)) return 0; item = &menu->rgItems[menu->iItem]; if (item && (item->spSubMenu) && (item->fState & MF_MOUSESELECT)) { return item->spSubMenu; } return 0; } /*********************************************************************** * MenuDoNextMenu * * NOTE: WM_NEXTMENU documented in Win32 is a bit different. */ static LRESULT FASTCALL MENU_DoNextMenu(MTRACKER* pmt, UINT Vk, UINT wFlags) { BOOL atEnd = FALSE; /* When skipping left, we need to do something special after the first menu. */ if (Vk == VK_LEFT && pmt->TopMenu->iItem == 0) { atEnd = TRUE; } /* When skipping right, for the non-system menu, we need to handle the last non-special menu item (ie skip any window icons such as MDI maximize, restore or close) */ else if ((Vk == VK_RIGHT) && !IS_SYSTEM_MENU(pmt->TopMenu)) { UINT i = pmt->TopMenu->iItem + 1; while (i < pmt->TopMenu->cItems) { if ((pmt->TopMenu->rgItems[i].wID >= SC_SIZE && pmt->TopMenu->rgItems[i].wID <= SC_RESTORE)) { i++; } else break; } if (i == pmt->TopMenu->cItems) { atEnd = TRUE; } } /* When skipping right, we need to cater for the system menu */ else if ((Vk == VK_RIGHT) && IS_SYSTEM_MENU(pmt->TopMenu)) { if (pmt->TopMenu->iItem == (pmt->TopMenu->cItems - 1)) { atEnd = TRUE; } } if ( atEnd ) { MDINEXTMENU NextMenu; PMENU MenuTmp; PWND pwndTemp; HMENU hNewMenu; HWND hNewWnd; UINT Id = 0; MenuTmp = (IS_SYSTEM_MENU(pmt->TopMenu)) ? co_IntGetSubMenu(pmt->TopMenu, 0) : pmt->TopMenu; NextMenu.hmenuIn = UserHMGetHandle(MenuTmp); NextMenu.hmenuNext = NULL; NextMenu.hwndNext = NULL; co_IntSendMessage(UserHMGetHandle(pmt->OwnerWnd), WM_NEXTMENU, Vk, (LPARAM) &NextMenu); TRACE("%p [%p] -> %p [%p]\n", pmt->CurrentMenu, pmt->OwnerWnd, NextMenu.hmenuNext, NextMenu.hwndNext ); if (NULL == NextMenu.hmenuNext || NULL == NextMenu.hwndNext) { hNewWnd = UserHMGetHandle(pmt->OwnerWnd); if (IS_SYSTEM_MENU(pmt->TopMenu)) { /* switch to the menu bar */ if (pmt->OwnerWnd->style & WS_CHILD || !(MenuTmp = IntGetMenu(hNewWnd))) return FALSE; if (Vk == VK_LEFT) { Id = MenuTmp->cItems - 1; /* Skip backwards over any system predefined icons, eg. MDI close, restore etc icons */ while ((Id > 0) && (MenuTmp->rgItems[Id].wID >= SC_SIZE && MenuTmp->rgItems[Id].wID <= SC_RESTORE)) Id--; } hNewMenu = UserHMGetHandle(MenuTmp); } else if (pmt->OwnerWnd->style & WS_SYSMENU) { /* switch to the system menu */ MenuTmp = get_win_sys_menu(hNewWnd); if (MenuTmp) hNewMenu = UserHMGetHandle(MenuTmp); else hNewMenu = NULL; } else return FALSE; } else /* application returned a new menu to switch to */ { hNewMenu = NextMenu.hmenuNext; hNewWnd = NextMenu.hwndNext; if ((MenuTmp = UserGetMenuObject(hNewMenu)) && (pwndTemp = ValidateHwndNoErr(hNewWnd))) { if ( pwndTemp->style & WS_SYSMENU && (get_win_sys_menu(hNewWnd) == MenuTmp) ) { /* get the real system menu */ MenuTmp = get_win_sys_menu(hNewWnd); hNewMenu = UserHMGetHandle(MenuTmp); } else if (pwndTemp->style & WS_CHILD || IntGetMenu(hNewWnd) != MenuTmp) { /* FIXME: Not sure what to do here; * perhaps try to track NewMenu as a popup? */ WARN(" -- got confused.\n"); return FALSE; } } else return FALSE; } if (hNewMenu != UserHMGetHandle(pmt->TopMenu)) { MENU_SelectItem(pmt->OwnerWnd, pmt->TopMenu, NO_SELECTED_ITEM, FALSE, 0 ); if (pmt->CurrentMenu != pmt->TopMenu) MENU_HideSubPopups(pmt->OwnerWnd, pmt->TopMenu, FALSE, wFlags); } if (hNewWnd != UserHMGetHandle(pmt->OwnerWnd)) { PTHREADINFO pti = PsGetCurrentThreadWin32Thread(); pmt->OwnerWnd = ValidateHwndNoErr(hNewWnd); ///// Use thread pms!!!! MsqSetStateWindow(pti, MSQ_STATE_MENUOWNER, hNewWnd); pti->MessageQueue->QF_flags &= ~QF_CAPTURELOCKED; co_UserSetCapture(UserHMGetHandle(pmt->OwnerWnd)); pti->MessageQueue->QF_flags |= QF_CAPTURELOCKED; } pmt->TopMenu = pmt->CurrentMenu = UserGetMenuObject(hNewMenu); /* all subpopups are hidden */ MENU_SelectItem(pmt->OwnerWnd, pmt->TopMenu, Id, TRUE, 0); return TRUE; } return FALSE; } /*********************************************************************** * MenuSuspendPopup * * The idea is not to show the popup if the next input message is * going to hide it anyway. */ static BOOL FASTCALL MENU_SuspendPopup(MTRACKER* pmt, UINT uMsg) { MSG msg; msg.hwnd = UserHMGetHandle(pmt->OwnerWnd); ////// ? silly wine'isms? co_IntGetPeekMessage( &msg, 0, uMsg, uMsg, PM_NOYIELD | PM_REMOVE, FALSE); pmt->TrackFlags |= TF_SKIPREMOVE; switch( uMsg ) { case WM_KEYDOWN: co_IntGetPeekMessage( &msg, 0, 0, 0, PM_NOYIELD | PM_NOREMOVE, FALSE); if( msg.message == WM_KEYUP || msg.message == WM_PAINT ) { co_IntGetPeekMessage( &msg, 0, 0, 0, PM_NOYIELD | PM_REMOVE, FALSE); co_IntGetPeekMessage( &msg, 0, 0, 0, PM_NOYIELD | PM_NOREMOVE, FALSE); if( msg.message == WM_KEYDOWN && (msg.wParam == VK_LEFT || msg.wParam == VK_RIGHT)) { pmt->TrackFlags |= TF_SUSPENDPOPUP; return TRUE; } } break; } /* failures go through this */ pmt->TrackFlags &= ~TF_SUSPENDPOPUP; return FALSE; } /*********************************************************************** * MenuKeyEscape * * Handle a VK_ESCAPE key event in a menu. */ static BOOL FASTCALL MENU_KeyEscape(MTRACKER *pmt, UINT Flags) { BOOL EndMenu = TRUE; if (pmt->CurrentMenu != pmt->TopMenu) { if (pmt->CurrentMenu && (pmt->CurrentMenu->fFlags & MNF_POPUP)) { PMENU MenuPrev, MenuTmp; MenuPrev = MenuTmp = pmt->TopMenu; /* close topmost popup */ while (MenuTmp != pmt->CurrentMenu) { MenuPrev = MenuTmp; MenuTmp = MENU_GetSubPopup(MenuPrev); } MENU_HideSubPopups(pmt->OwnerWnd, MenuPrev, TRUE, Flags); pmt->CurrentMenu = MenuPrev; EndMenu = FALSE; } } return EndMenu; } /*********************************************************************** * MenuKeyLeft * * Handle a VK_LEFT key event in a menu. */ static void FASTCALL MENU_KeyLeft(MTRACKER* pmt, UINT Flags, UINT msg) { PMENU MenuTmp, MenuPrev; UINT PrevCol; MenuPrev = MenuTmp = pmt->TopMenu; /* Try to move 1 column left (if possible) */ if ( (PrevCol = MENU_GetStartOfPrevColumn(pmt->CurrentMenu)) != NO_SELECTED_ITEM) { MENU_SelectItem(pmt->OwnerWnd, pmt->CurrentMenu, PrevCol, TRUE, 0); return; } /* close topmost popup */ while (MenuTmp != pmt->CurrentMenu) { MenuPrev = MenuTmp; MenuTmp = MENU_GetSubPopup(MenuPrev); } MENU_HideSubPopups(pmt->OwnerWnd, MenuPrev, TRUE, Flags); pmt->CurrentMenu = MenuPrev; if ((MenuPrev == pmt->TopMenu) && !(pmt->TopMenu->fFlags & MNF_POPUP)) { /* move menu bar selection if no more popups are left */ if (!MENU_DoNextMenu(pmt, VK_LEFT, Flags)) MENU_MoveSelection(pmt->OwnerWnd, pmt->TopMenu, ITEM_PREV); if (MenuPrev != MenuTmp || pmt->TrackFlags & TF_SUSPENDPOPUP) { /* A sublevel menu was displayed - display the next one * unless there is another displacement coming up */ if (!MENU_SuspendPopup(pmt, msg)) pmt->CurrentMenu = MENU_ShowSubPopup(pmt->OwnerWnd, pmt->TopMenu, TRUE, Flags); } } } /*********************************************************************** * MenuKeyRight * * Handle a VK_RIGHT key event in a menu. */ static void FASTCALL MENU_KeyRight(MTRACKER *pmt, UINT Flags, UINT msg) { PMENU menutmp; UINT NextCol; TRACE("MenuKeyRight called, cur %p, top %p.\n", pmt->CurrentMenu, pmt->TopMenu); if ((pmt->TopMenu->fFlags & MNF_POPUP) || (pmt->CurrentMenu != pmt->TopMenu)) { /* If already displaying a popup, try to display sub-popup */ menutmp = pmt->CurrentMenu; pmt->CurrentMenu = MENU_ShowSubPopup(pmt->OwnerWnd, menutmp, TRUE, Flags); /* if subpopup was displayed then we are done */ if (menutmp != pmt->CurrentMenu) return; } /* Check to see if there's another column */ if ( (NextCol = MENU_GetStartOfNextColumn(pmt->CurrentMenu)) != NO_SELECTED_ITEM) { TRACE("Going to %d.\n", NextCol); MENU_SelectItem(pmt->OwnerWnd, pmt->CurrentMenu, NextCol, TRUE, 0); return; } if (!(pmt->TopMenu->fFlags & MNF_POPUP)) /* menu bar tracking */ { if (pmt->CurrentMenu != pmt->TopMenu) { MENU_HideSubPopups(pmt->OwnerWnd, pmt->TopMenu, FALSE, Flags); menutmp = pmt->CurrentMenu = pmt->TopMenu; } else { menutmp = NULL; } /* try to move to the next item */ if ( !MENU_DoNextMenu(pmt, VK_RIGHT, Flags)) MENU_MoveSelection(pmt->OwnerWnd, pmt->TopMenu, ITEM_NEXT); if ( menutmp || pmt->TrackFlags & TF_SUSPENDPOPUP ) { if ( !MENU_SuspendPopup(pmt, msg) ) pmt->CurrentMenu = MENU_ShowSubPopup(pmt->OwnerWnd, pmt->TopMenu, TRUE, Flags); } } } /*********************************************************************** * MenuTrackMenu * * Menu tracking code. */ static INT FASTCALL MENU_TrackMenu(PMENU pmenu, UINT wFlags, INT x, INT y, PWND pwnd) { MSG msg; BOOL fRemove; INT executedMenuId = -1; MTRACKER mt; HWND capture_win; PMENU pmMouse; BOOL enterIdleSent = FALSE; BOOL firstClick = TRUE; PWND pWnd; PTHREADINFO pti = PsGetCurrentThreadWin32Thread(); if (pti != pwnd->head.pti) { ERR("Not the same PTI!!!!\n"); } mt.TrackFlags = 0; mt.CurrentMenu = pmenu; mt.TopMenu = pmenu; mt.OwnerWnd = pwnd; mt.Pt.x = x; mt.Pt.y = y; TRACE("MTM : hmenu=%p flags=0x%08x (%d,%d) hwnd=%x\n", UserHMGetHandle(pmenu), wFlags, x, y, UserHMGetHandle(pwnd)); pti->MessageQueue->QF_flags &= ~QF_ACTIVATIONCHANGE; if (wFlags & TPM_BUTTONDOWN) { /* Get the result in order to start the tracking or not */ fRemove = MENU_ButtonDown( &mt, pmenu, wFlags ); fInsideMenuLoop = fRemove; } if (wFlags & TF_ENDMENU) fInsideMenuLoop = FALSE; if (wFlags & TPM_POPUPMENU && pmenu->cItems == 0) // Tracking empty popup menu... { MsqSetStateWindow(pti, MSQ_STATE_MENUOWNER, NULL); pti->MessageQueue->QF_flags &= ~QF_CAPTURELOCKED; co_UserSetCapture(NULL); /* release the capture */ return 0; } capture_win = IntGetCapture(); while (fInsideMenuLoop) { BOOL ErrorExit = FALSE; if (!VerifyMenu( mt.CurrentMenu )) /* sometimes happens if I do a window manager close */ break; /* we have to keep the message in the queue until it's * clear that menu loop is not over yet. */ for (;;) { if (co_IntGetPeekMessage( &msg, 0, 0, 0, PM_NOREMOVE, FALSE )) { if (!IntCallMsgFilter( &msg, MSGF_MENU )) break; /* remove the message from the queue */ co_IntGetPeekMessage( &msg, 0, msg.message, msg.message, PM_REMOVE, FALSE ); } else { /* ReactOS Checks */ if (!VerifyWnd(mt.OwnerWnd) || !ValidateHwndNoErr(mt.CurrentMenu->hWnd) || //pti->MessageQueue->QF_flags & QF_ACTIVATIONCHANGE || // See CORE-17338 capture_win != IntGetCapture() ) // Should not happen, but this is ReactOS... { ErrorExit = TRUE; // Do not wait on dead windows, now win test_capture_4 works. break; } if (!enterIdleSent) { HWND win = mt.CurrentMenu->fFlags & MNF_POPUP ? mt.CurrentMenu->hWnd : NULL; enterIdleSent = TRUE; co_IntSendMessage( UserHMGetHandle(mt.OwnerWnd), WM_ENTERIDLE, MSGF_MENU, (LPARAM) win); } co_IntWaitMessage(NULL, 0, 0); } } if (ErrorExit) break; // Gracefully dropout. /* check if EndMenu() tried to cancel us, by posting this message */ if (msg.message == WM_CANCELMODE) { /* we are now out of the loop */ fInsideMenuLoop = FALSE; /* remove the message from the queue */ co_IntGetPeekMessage( &msg, 0, msg.message, msg.message, PM_REMOVE, FALSE ); /* break out of internal loop, ala ESCAPE */ break; } mt.Pt = msg.pt; if ( (msg.hwnd == mt.CurrentMenu->hWnd) || ((msg.message!=WM_TIMER) && (msg.message!=WM_SYSTIMER)) ) enterIdleSent=FALSE; fRemove = FALSE; if ((msg.message >= WM_MOUSEFIRST) && (msg.message <= WM_MOUSELAST)) { /* * Use the mouse coordinates in lParam instead of those in the MSG * struct to properly handle synthetic messages. They are already * in screen coordinates. */ mt.Pt.x = (short)LOWORD(msg.lParam); mt.Pt.y = (short)HIWORD(msg.lParam); /* Find a menu for this mouse event */ pmMouse = MENU_PtMenu( mt.TopMenu, mt.Pt ); switch(msg.message) { /* no WM_NC... messages in captured state */ case WM_RBUTTONDBLCLK: case WM_RBUTTONDOWN: if (!(wFlags & TPM_RIGHTBUTTON)) { if ( msg.message == WM_RBUTTONDBLCLK ) fInsideMenuLoop = FALSE; // Must exit or loop forever! break; } /* fall through */ case WM_LBUTTONDBLCLK: case WM_LBUTTONDOWN: { /* If the message belongs to the menu, removes it from the queue */ /* Else, end menu tracking */ pWnd = ValidateHwndNoErr(mt.TopMenu->hWnd); /* Don't remove WM_LBUTTONDBLCLK to allow the closing of a window or program */ if (msg.message == WM_LBUTTONDBLCLK && GetNCHitEx(pWnd, mt.Pt) == HTSYSMENU) fRemove = FALSE; else fRemove = MENU_ButtonDown(&mt, pmMouse, wFlags); fInsideMenuLoop = fRemove; if (msg.message == WM_RBUTTONDBLCLK) fInsideMenuLoop = FALSE; // Must exit or loop forever break; } case WM_RBUTTONUP: if (!(wFlags & TPM_RIGHTBUTTON)) break; /* fall through */ case WM_LBUTTONUP: /* Check if a menu was selected by the mouse */ if (pmMouse) { pWnd = ValidateHwndNoErr(mt.TopMenu->hWnd); /* Exit system menu if system icon is clicked a second time */ if (!firstClick && GetNCHitEx(pWnd, mt.Pt) == HTSYSMENU) { fRemove = TRUE; fInsideMenuLoop = FALSE; } else { /* End the loop if executedMenuId is an item ID */ /* or if the job was done (executedMenuId = 0). */ executedMenuId = MENU_ButtonUp( &mt, pmMouse, wFlags); fRemove = (executedMenuId != -1); fInsideMenuLoop = !fRemove; firstClick = FALSE; } } /* No menu was selected by the mouse */ /* if the function was called by TrackPopupMenu, continue with the menu tracking. If not, stop it */ else fInsideMenuLoop = ((wFlags & TPM_POPUPMENU) ? TRUE : FALSE); break; case WM_MOUSEMOVE: /* the selected menu item must be changed every time */ /* the mouse moves. */ if (pmMouse) fInsideMenuLoop |= MENU_MouseMove( &mt, pmMouse, wFlags ); } /* switch(msg.message) - mouse */ } else if ((msg.message >= WM_KEYFIRST) && (msg.message <= WM_KEYLAST)) { fRemove = TRUE; /* Keyboard messages are always removed */ switch(msg.message) { case WM_KEYDOWN: case WM_SYSKEYDOWN: switch(msg.wParam) { case VK_MENU: case VK_F10: fInsideMenuLoop = FALSE; break; case VK_HOME: case VK_END: MENU_SelectItem(mt.OwnerWnd, mt.CurrentMenu, NO_SELECTED_ITEM, FALSE, 0 ); MENU_MoveSelection(mt.OwnerWnd, mt.CurrentMenu, VK_HOME == msg.wParam ? ITEM_NEXT : ITEM_PREV); break; case VK_UP: case VK_DOWN: /* If on menu bar, pull-down the menu */ if (!(mt.CurrentMenu->fFlags & MNF_POPUP)) mt.CurrentMenu = MENU_ShowSubPopup(mt.OwnerWnd, mt.TopMenu, TRUE, wFlags); else /* otherwise try to move selection */ MENU_MoveSelection(mt.OwnerWnd, mt.CurrentMenu, (msg.wParam == VK_UP)? ITEM_PREV : ITEM_NEXT ); break; case VK_LEFT: MENU_KeyLeft( &mt, wFlags, msg.message ); break; case VK_RIGHT: MENU_KeyRight( &mt, wFlags, msg.message ); break; case VK_ESCAPE: fInsideMenuLoop = !MENU_KeyEscape(&mt, wFlags); break; case VK_F1: { HELPINFO hi; hi.cbSize = sizeof(HELPINFO); hi.iContextType = HELPINFO_MENUITEM; if (mt.CurrentMenu->iItem == NO_SELECTED_ITEM) hi.iCtrlId = 0; else hi.iCtrlId = pmenu->rgItems[mt.CurrentMenu->iItem].wID; hi.hItemHandle = UserHMGetHandle(mt.CurrentMenu); hi.dwContextId = pmenu->dwContextHelpId; hi.MousePos = msg.pt; co_IntSendMessage( UserHMGetHandle(pwnd), WM_HELP, 0, (LPARAM)&hi); break; } default: IntTranslateKbdMessage(&msg, 0); break; } break; /* WM_KEYDOWN */ case WM_CHAR: case WM_SYSCHAR: { UINT pos; BOOL fEndMenu; if (msg.wParam == L'\r' || msg.wParam == L' ') { executedMenuId = MENU_ExecFocusedItem(&mt, mt.CurrentMenu, wFlags); fEndMenu = (executedMenuId != -2); fInsideMenuLoop = !fEndMenu; break; } /* Hack to avoid control chars. */ /* We will find a better way real soon... */ if (msg.wParam < 32) break; pos = MENU_FindItemByKey(mt.OwnerWnd, mt.CurrentMenu, LOWORD(msg.wParam), FALSE); if (pos == (UINT)-2) fInsideMenuLoop = FALSE; else if (pos == (UINT)-1) UserPostMessage(hwndSAS, WM_LOGONNOTIFY, LN_MESSAGE_BEEP, 0); //MessageBeep(0); else { MENU_SelectItem(mt.OwnerWnd, mt.CurrentMenu, pos, TRUE, 0); executedMenuId = MENU_ExecFocusedItem(&mt, mt.CurrentMenu, wFlags); fEndMenu = (executedMenuId != -2); fInsideMenuLoop = !fEndMenu; } } break; } /* switch(msg.message) - kbd */ } else { co_IntGetPeekMessage( &msg, 0, msg.message, msg.message, PM_REMOVE, FALSE ); IntDispatchMessage( &msg ); continue; } if (fInsideMenuLoop) fRemove = TRUE; /* finally remove message from the queue */ if (fRemove && !(mt.TrackFlags & TF_SKIPREMOVE) ) co_IntGetPeekMessage( &msg, 0, msg.message, msg.message, PM_REMOVE, FALSE ); else mt.TrackFlags &= ~TF_SKIPREMOVE; } MsqSetStateWindow(pti, MSQ_STATE_MENUOWNER, NULL); pti->MessageQueue->QF_flags &= ~QF_CAPTURELOCKED; co_UserSetCapture(NULL); /* release the capture */ /* If dropdown is still painted and the close box is clicked on then the menu will be destroyed as part of the DispatchMessage above. This will then invalidate the menu handle in mt.hTopMenu. We should check for this first. */ if ( VerifyMenu( mt.TopMenu ) ) { if (VerifyWnd(mt.OwnerWnd)) { MENU_HideSubPopups(mt.OwnerWnd, mt.TopMenu, FALSE, wFlags); if (mt.TopMenu->fFlags & MNF_POPUP) { PWND pwndTM = ValidateHwndNoErr(mt.TopMenu->hWnd); if (pwndTM) { IntNotifyWinEvent(EVENT_SYSTEM_MENUPOPUPEND, pwndTM, OBJID_CLIENT, CHILDID_SELF, 0); co_UserDestroyWindow(pwndTM); } mt.TopMenu->hWnd = NULL; if (!(wFlags & TPM_NONOTIFY)) { co_IntSendMessage( UserHMGetHandle(mt.OwnerWnd), WM_UNINITMENUPOPUP, (WPARAM)UserHMGetHandle(mt.TopMenu), MAKELPARAM(0, IS_SYSTEM_MENU(mt.TopMenu) ? MF_SYSMENU : 0)); } } MENU_SelectItem( mt.OwnerWnd, mt.TopMenu, NO_SELECTED_ITEM, FALSE, 0 ); co_IntSendMessage( UserHMGetHandle(mt.OwnerWnd), WM_MENUSELECT, MAKEWPARAM(0, 0xffff), 0 ); } /* Reset the variable for hiding menu */ mt.TopMenu->TimeToHide = FALSE; } EngSetLastError( ERROR_SUCCESS ); /* The return value is only used by TrackPopupMenu */ if (!(wFlags & TPM_RETURNCMD)) return TRUE; if (executedMenuId == -1) executedMenuId = 0; return executedMenuId; } /*********************************************************************** * MenuInitTracking */ static BOOL FASTCALL MENU_InitTracking(PWND pWnd, PMENU Menu, BOOL bPopup, UINT wFlags) { HWND capture_win; PTHREADINFO pti = PsGetCurrentThreadWin32Thread(); TRACE("hwnd=%p hmenu=%p\n", UserHMGetHandle(pWnd), UserHMGetHandle(Menu)); co_UserHideCaret(0); /* This makes the menus of applications built with Delphi work. * It also enables menus to be displayed in more than one window, * but there are some bugs left that need to be fixed in this case. */ if (!bPopup) { Menu->hWnd = UserHMGetHandle(pWnd); } if (!top_popup) { top_popup = Menu->hWnd; top_popup_hmenu = UserHMGetHandle(Menu); } fInsideMenuLoop = TRUE; fInEndMenu = FALSE; /* Send WM_ENTERMENULOOP and WM_INITMENU message only if TPM_NONOTIFY flag is not specified */ if (!(wFlags & TPM_NONOTIFY)) { co_IntSendMessage( UserHMGetHandle(pWnd), WM_ENTERMENULOOP, bPopup, 0 ); } // // Capture is set before calling WM_INITMENU and after WM_ENTERMENULOOP, see msg_menu. // capture_win = (wFlags & TPM_POPUPMENU) ? Menu->hWnd : UserHMGetHandle(pWnd); MsqSetStateWindow(pti, MSQ_STATE_MENUOWNER, capture_win); // 1 co_UserSetCapture(capture_win); // 2 pti->MessageQueue->QF_flags |= QF_CAPTURELOCKED; // Set the Q bits so noone can change this! co_IntSendMessage( UserHMGetHandle(pWnd), WM_SETCURSOR, (WPARAM)UserHMGetHandle(pWnd), HTCAPTION ); if (!(wFlags & TPM_NONOTIFY)) { co_IntSendMessage( UserHMGetHandle(pWnd), WM_INITMENU, (WPARAM)UserHMGetHandle(Menu), 0 ); /* If an app changed/recreated menu bar entries in WM_INITMENU * menu sizes will be recalculated once the menu created/shown. */ } IntNotifyWinEvent( EVENT_SYSTEM_MENUSTART, pWnd, Menu->fFlags & MNF_SYSMENU ? OBJID_SYSMENU : OBJID_MENU, CHILDID_SELF, 0); return TRUE; } /*********************************************************************** * MenuExitTracking */ static BOOL FASTCALL MENU_ExitTracking(PWND pWnd, BOOL bPopup, UINT wFlags) { TRACE("Exit Track hwnd=%p bPopup %d\n", UserHMGetHandle(pWnd), bPopup); IntNotifyWinEvent( EVENT_SYSTEM_MENUEND, pWnd, OBJID_WINDOW, CHILDID_SELF, 0); if (!(wFlags & TPM_NONOTIFY)) co_IntSendMessage( UserHMGetHandle(pWnd), WM_EXITMENULOOP, bPopup, 0 ); co_UserShowCaret(0); top_popup = 0; top_popup_hmenu = NULL; return TRUE; } /*********************************************************************** * MenuTrackMouseMenuBar * * Menu-bar tracking upon a mouse event. Called from NC_HandleSysCommand(). */ VOID MENU_TrackMouseMenuBar( PWND pWnd, ULONG ht, POINT pt) { PMENU pMenu = (ht == HTSYSMENU) ? IntGetSystemMenu(pWnd, FALSE) : IntGetMenu( UserHMGetHandle(pWnd) ); // See 74276 and CORE-12801 UINT wFlags = TPM_BUTTONDOWN | TPM_LEFTALIGN | TPM_LEFTBUTTON | TPM_VERTICAL; TRACE("wnd=%p ht=0x%04x (%ld,%ld)\n", pWnd, ht, pt.x, pt.y); if (pWnd->ExStyle & WS_EX_LAYOUTRTL) wFlags |= TPM_LAYOUTRTL; if (VerifyMenu(pMenu)) { /* map point to parent client coordinates */ PWND Parent = UserGetAncestor(pWnd, GA_PARENT ); if (Parent != UserGetDesktopWindow()) { IntScreenToClient(Parent, &pt); } MENU_InitTracking(pWnd, pMenu, FALSE, wFlags); /* fetch the window menu again, it may have changed */ pMenu = (ht == HTSYSMENU) ? get_win_sys_menu( UserHMGetHandle(pWnd) ) : IntGetMenu( UserHMGetHandle(pWnd) ); MENU_TrackMenu(pMenu, wFlags, pt.x, pt.y, pWnd); MENU_ExitTracking(pWnd, FALSE, wFlags); } } /*********************************************************************** * MenuTrackKbdMenuBar * * Menu-bar tracking upon a keyboard event. Called from NC_HandleSysCommand(). */ VOID MENU_TrackKbdMenuBar(PWND pwnd, UINT wParam, WCHAR wChar) { UINT uItem = NO_SELECTED_ITEM; PMENU TrackMenu; UINT wFlags = TPM_LEFTALIGN | TPM_LEFTBUTTON; TRACE("hwnd %p wParam 0x%04x wChar 0x%04x\n", UserHMGetHandle(pwnd), wParam, wChar); /* find window that has a menu */ while (!( (pwnd->style & (WS_CHILD | WS_POPUP)) != WS_CHILD ) ) if (!(pwnd = UserGetAncestor( pwnd, GA_PARENT ))) return; /* check if we have to track a system menu */ TrackMenu = IntGetMenu( UserHMGetHandle(pwnd) ); if (!TrackMenu || (pwnd->style & WS_MINIMIZE) != 0 || wChar == ' ' ) { if (!(pwnd->style & WS_SYSMENU)) return; TrackMenu = get_win_sys_menu( UserHMGetHandle(pwnd) ); uItem = 0; wParam |= HTSYSMENU; /* prevent item lookup */ } if (!VerifyMenu( TrackMenu )) return; MENU_InitTracking( pwnd, TrackMenu, FALSE, wFlags ); /* fetch the window menu again, it may have changed */ TrackMenu = (wParam & HTSYSMENU) ? get_win_sys_menu( UserHMGetHandle(pwnd) ) : IntGetMenu( UserHMGetHandle(pwnd) ); if( wChar && wChar != ' ' ) { uItem = MENU_FindItemByKey( pwnd, TrackMenu, wChar, (wParam & HTSYSMENU) ); if ( uItem >= (UINT)(-2) ) { if( uItem == (UINT)(-1) ) UserPostMessage(hwndSAS, WM_LOGONNOTIFY, LN_MESSAGE_BEEP, 0); //MessageBeep(0); /* schedule end of menu tracking */ wFlags |= TF_ENDMENU; goto track_menu; } } MENU_SelectItem( pwnd, TrackMenu, uItem, TRUE, 0 ); if (!(wParam & HTSYSMENU) || wChar == ' ') { if( uItem == NO_SELECTED_ITEM ) MENU_MoveSelection( pwnd, TrackMenu, ITEM_NEXT ); else UserPostMessage( UserHMGetHandle(pwnd), WM_KEYDOWN, VK_RETURN, 0 ); } track_menu: MENU_TrackMenu( TrackMenu, wFlags, 0, 0, pwnd ); MENU_ExitTracking( pwnd, FALSE, wFlags); } /********************************************************************** * TrackPopupMenuEx (USER32.@) */ BOOL WINAPI IntTrackPopupMenuEx( PMENU menu, UINT wFlags, int x, int y, PWND pWnd, LPTPMPARAMS lpTpm) { BOOL ret = FALSE; PTHREADINFO pti = PsGetCurrentThreadWin32Thread(); if (pti != pWnd->head.pti) { ERR("Must be the same pti!\n"); return ret; } TRACE("hmenu %p flags %04x (%d,%d) hwnd %p lpTpm %p \n", //rect %s\n", UserHMGetHandle(menu), wFlags, x, y, UserHMGetHandle(pWnd), lpTpm); //, //lpTpm ? wine_dbgstr_rect( &lpTpm->rcExclude) : "-" ); if (menu->hWnd && IntIsWindow(menu->hWnd)) { EngSetLastError( ERROR_POPUP_ALREADY_ACTIVE ); return FALSE; } if (MENU_InitPopup( pWnd, menu, wFlags )) { MENU_InitTracking(pWnd, menu, TRUE, wFlags); /* Send WM_INITMENUPOPUP message only if TPM_NONOTIFY flag is not specified */ if (!(wFlags & TPM_NONOTIFY)) { co_IntSendMessage( UserHMGetHandle(pWnd), WM_INITMENUPOPUP, (WPARAM) UserHMGetHandle(menu), MAKELPARAM(0, IS_SYSTEM_MENU(menu))); } if (menu->fFlags & MNF_SYSMENU) MENU_InitSysMenuPopup( menu, pWnd->style, pWnd->pcls->style, HTSYSMENU); if (MENU_ShowPopup(pWnd, menu, 0, wFlags | TPM_POPUPMENU, x, y, lpTpm ? &lpTpm->rcExclude : NULL)) ret = MENU_TrackMenu( menu, wFlags | TPM_POPUPMENU, 0, 0, pWnd); else { MsqSetStateWindow(pti, MSQ_STATE_MENUOWNER, NULL); pti->MessageQueue->QF_flags &= ~QF_CAPTURELOCKED; co_UserSetCapture(NULL); /* release the capture */ } MENU_ExitTracking(pWnd, TRUE, wFlags); if (menu->hWnd) { PWND pwndM = ValidateHwndNoErr( menu->hWnd ); if (pwndM) // wine hack around this with their destroy function. { if (!(pWnd->state & WNDS_DESTROYED)) co_UserDestroyWindow( pwndM ); // Fix wrong error return. } menu->hWnd = 0; if (!(wFlags & TPM_NONOTIFY)) { co_IntSendMessage( UserHMGetHandle(pWnd), WM_UNINITMENUPOPUP, (WPARAM)UserHMGetHandle(menu), MAKELPARAM(0, IS_SYSTEM_MENU(menu) ? MF_SYSMENU : 0)); } } } return ret; } // // Menu Class Proc. // BOOL WINAPI PopupMenuWndProc( PWND Wnd, UINT Message, WPARAM wParam, LPARAM lParam, LRESULT *lResult) { PPOPUPMENU pPopupMenu; *lResult = 0; TRACE("PMWP : pwnd=%x msg=%d wp=0x%04lx lp=0x%08lx\n", Wnd, Message, wParam, lParam); if (Wnd) { if (!Wnd->fnid) { if (Message != WM_NCCREATE) { *lResult = IntDefWindowProc(Wnd, Message, wParam, lParam, FALSE); return TRUE; } Wnd->fnid = FNID_MENU; pPopupMenu = DesktopHeapAlloc( Wnd->head.rpdesk, sizeof(POPUPMENU) ); if (pPopupMenu == NULL) { return TRUE; } pPopupMenu->posSelectedItem = NO_SELECTED_ITEM; pPopupMenu->spwndPopupMenu = Wnd; ((PMENUWND)Wnd)->ppopupmenu = pPopupMenu; TRACE("Pop Up Menu is Setup! Msg %d\n",Message); *lResult = 1; return TRUE; } else { if (Wnd->fnid != FNID_MENU) { ERR("Wrong window class for Menu! fnid %x\n",Wnd->fnid); return TRUE; } pPopupMenu = ((PMENUWND)Wnd)->ppopupmenu; } } switch(Message) { case WM_CREATE: { CREATESTRUCTW *cs = (CREATESTRUCTW *) lParam; pPopupMenu->spmenu = UserGetMenuObject(cs->lpCreateParams); if (pPopupMenu->spmenu) { UserReferenceObject(pPopupMenu->spmenu); } break; } case WM_MOUSEACTIVATE: /* We don't want to be activated */ *lResult = MA_NOACTIVATE; break; case WM_PAINT: { PAINTSTRUCT ps; IntBeginPaint(Wnd, &ps); MENU_DrawPopupMenu(Wnd, ps.hdc, pPopupMenu->spmenu); IntEndPaint(Wnd, &ps); break; } case WM_PRINTCLIENT: { MENU_DrawPopupMenu( Wnd, (HDC)wParam, pPopupMenu->spmenu); break; } case WM_ERASEBKGND: *lResult = 1; break; case WM_DESTROY: /* zero out global pointer in case resident popup window was destroyed. */ if (pPopupMenu) { if (UserHMGetHandle(Wnd) == top_popup) { top_popup = NULL; top_popup_hmenu = NULL; } } else { ERR("No Window Pop Up!\n"); } break; case WM_NCDESTROY: { if (pPopupMenu->spmenu) { IntReleaseMenuObject(pPopupMenu->spmenu); } DesktopHeapFree(Wnd->head.rpdesk, pPopupMenu ); ((PMENUWND)Wnd)->ppopupmenu = 0; Wnd->fnid = FNID_DESTROY; break; } case MM_SETMENUHANDLE: // wine'isms case MN_SETHMENU: { PMENU pmenu = UserGetMenuObject((HMENU)wParam); if (!pmenu) { ERR("Bad Menu Handle\n"); break; } UserReferenceObject(pmenu); if (pPopupMenu->spmenu) { IntReleaseMenuObject(pPopupMenu->spmenu); } pPopupMenu->spmenu = pmenu; break; } case MM_GETMENUHANDLE: // wine'isms case MN_GETHMENU: *lResult = (LRESULT)(pPopupMenu ? (pPopupMenu->spmenu ? UserHMGetHandle(pPopupMenu->spmenu) : NULL) : NULL); break; default: if (Message > MN_GETHMENU && Message < MN_GETHMENU+19) { ERR("Someone is passing unknown menu messages %d\n",Message); } TRACE("PMWP to IDWP %d\n",Message); *lResult = IntDefWindowProc(Wnd, Message, wParam, lParam, FALSE); break; } return TRUE; } BOOL FASTCALL IntHiliteMenuItem(PWND WindowObject, PMENU MenuObject, UINT uItemHilite, UINT uHilite) { PITEM MenuItem; UINT uItem = uItemHilite; if (!(MenuItem = MENU_FindItem( &MenuObject, &uItem, uHilite ))) return TRUE; if (uHilite & MF_HILITE) { MenuItem->fState |= MF_HILITE; } else { MenuItem->fState &= ~MF_HILITE; } if (MenuObject->iItem == uItemHilite) return TRUE; MENU_HideSubPopups( WindowObject, MenuObject, FALSE, 0 ); MENU_SelectItem( WindowObject, MenuObject, uItemHilite, TRUE, 0 ); return TRUE; // Always returns true!!!! } BOOLEAN APIENTRY intGetTitleBarInfo(PWND pWindowObject, PTITLEBARINFO bti) { DWORD dwStyle = 0; DWORD dwExStyle = 0; BOOLEAN retValue = TRUE; if (bti->cbSize == sizeof(TITLEBARINFO)) { RtlZeroMemory(&bti->rgstate[0],sizeof(DWORD)*(CCHILDREN_TITLEBAR+1)); bti->rgstate[0] = STATE_SYSTEM_FOCUSABLE; dwStyle = pWindowObject->style; dwExStyle = pWindowObject->ExStyle; bti->rcTitleBar.top = 0; bti->rcTitleBar.left = 0; bti->rcTitleBar.right = pWindowObject->rcWindow.right - pWindowObject->rcWindow.left; bti->rcTitleBar.bottom = pWindowObject->rcWindow.bottom - pWindowObject->rcWindow.top; /* Is it iconiced ? */ if ((dwStyle & WS_ICONIC)!=WS_ICONIC) { /* Remove frame from rectangle */ if (HAS_THICKFRAME( dwStyle, dwExStyle )) { /* FIXME: Note this value should exists in pWindowObject for UserGetSystemMetrics(SM_CXFRAME) and UserGetSystemMetrics(SM_CYFRAME) */ RECTL_vInflateRect( &bti->rcTitleBar, -UserGetSystemMetrics(SM_CXFRAME), -UserGetSystemMetrics(SM_CYFRAME) ); } else if (HAS_DLGFRAME( dwStyle, dwExStyle )) { /* FIXME: Note this value should exists in pWindowObject for UserGetSystemMetrics(SM_CXDLGFRAME) and UserGetSystemMetrics(SM_CYDLGFRAME) */ RECTL_vInflateRect( &bti->rcTitleBar, -UserGetSystemMetrics(SM_CXDLGFRAME), -UserGetSystemMetrics(SM_CYDLGFRAME)); } else if (HAS_THINFRAME( dwStyle, dwExStyle)) { /* FIXME: Note this value should exists in pWindowObject for UserGetSystemMetrics(SM_CXBORDER) and UserGetSystemMetrics(SM_CYBORDER) */ RECTL_vInflateRect( &bti->rcTitleBar, -UserGetSystemMetrics(SM_CXBORDER), -UserGetSystemMetrics(SM_CYBORDER) ); } /* We have additional border information if the window * is a child (but not an MDI child) */ if ( (dwStyle & WS_CHILD) && ((dwExStyle & WS_EX_MDICHILD) == 0 ) ) { if (dwExStyle & WS_EX_CLIENTEDGE) { /* FIXME: Note this value should exists in pWindowObject for UserGetSystemMetrics(SM_CXEDGE) and UserGetSystemMetrics(SM_CYEDGE) */ RECTL_vInflateRect (&bti->rcTitleBar, -UserGetSystemMetrics(SM_CXEDGE), -UserGetSystemMetrics(SM_CYEDGE)); } if (dwExStyle & WS_EX_STATICEDGE) { /* FIXME: Note this value should exists in pWindowObject for UserGetSystemMetrics(SM_CXBORDER) and UserGetSystemMetrics(SM_CYBORDER) */ RECTL_vInflateRect (&bti->rcTitleBar, -UserGetSystemMetrics(SM_CXBORDER), -UserGetSystemMetrics(SM_CYBORDER)); } } } bti->rcTitleBar.top += pWindowObject->rcWindow.top; bti->rcTitleBar.left += pWindowObject->rcWindow.left; bti->rcTitleBar.right += pWindowObject->rcWindow.left; bti->rcTitleBar.bottom = bti->rcTitleBar.top; if (dwExStyle & WS_EX_TOOLWINDOW) { /* FIXME: Note this value should exists in pWindowObject for UserGetSystemMetrics(SM_CYSMCAPTION) */ bti->rcTitleBar.bottom += UserGetSystemMetrics(SM_CYSMCAPTION); } else { /* FIXME: Note this value should exists in pWindowObject for UserGetSystemMetrics(SM_CYCAPTION) and UserGetSystemMetrics(SM_CXSIZE) */ bti->rcTitleBar.bottom += UserGetSystemMetrics(SM_CYCAPTION); bti->rcTitleBar.left += UserGetSystemMetrics(SM_CXSIZE); } if (dwStyle & WS_CAPTION) { bti->rgstate[1] = STATE_SYSTEM_INVISIBLE; if (dwStyle & WS_SYSMENU) { if (!(dwStyle & (WS_MINIMIZEBOX|WS_MAXIMIZEBOX))) { bti->rgstate[2] = STATE_SYSTEM_INVISIBLE; bti->rgstate[3] = STATE_SYSTEM_INVISIBLE; } else { if (!(dwStyle & WS_MINIMIZEBOX)) { bti->rgstate[2] = STATE_SYSTEM_UNAVAILABLE; } if (!(dwStyle & WS_MAXIMIZEBOX)) { bti->rgstate[3] = STATE_SYSTEM_UNAVAILABLE; } } if (!(dwExStyle & WS_EX_CONTEXTHELP)) { bti->rgstate[4] = STATE_SYSTEM_INVISIBLE; } if (pWindowObject->pcls->style & CS_NOCLOSE) { bti->rgstate[5] = STATE_SYSTEM_UNAVAILABLE; } } else { bti->rgstate[2] = STATE_SYSTEM_INVISIBLE; bti->rgstate[3] = STATE_SYSTEM_INVISIBLE; bti->rgstate[4] = STATE_SYSTEM_INVISIBLE; bti->rgstate[5] = STATE_SYSTEM_INVISIBLE; } } else { bti->rgstate[0] |= STATE_SYSTEM_INVISIBLE; } } else { EngSetLastError(ERROR_INVALID_PARAMETER); retValue = FALSE; } return retValue; } DWORD FASTCALL UserInsertMenuItem( PMENU Menu, UINT uItem, BOOL fByPosition, LPCMENUITEMINFOW UnsafeItemInfo, PUNICODE_STRING lpstr) { NTSTATUS Status; ROSMENUITEMINFO ItemInfo; /* Try to copy the whole MENUITEMINFOW structure */ Status = MmCopyFromCaller(&ItemInfo, UnsafeItemInfo, sizeof(MENUITEMINFOW)); if (NT_SUCCESS(Status)) { if (sizeof(MENUITEMINFOW) != ItemInfo.cbSize && FIELD_OFFSET(MENUITEMINFOW, hbmpItem) != ItemInfo.cbSize) { EngSetLastError(ERROR_INVALID_PARAMETER); return FALSE; } return IntInsertMenuItem(Menu, uItem, fByPosition, &ItemInfo, lpstr); } /* Try to copy without last field (not present in older versions) */ Status = MmCopyFromCaller(&ItemInfo, UnsafeItemInfo, FIELD_OFFSET(MENUITEMINFOW, hbmpItem)); if (NT_SUCCESS(Status)) { if (FIELD_OFFSET(MENUITEMINFOW, hbmpItem) != ItemInfo.cbSize) { EngSetLastError(ERROR_INVALID_PARAMETER); return FALSE; } ItemInfo.hbmpItem = (HBITMAP)0; return IntInsertMenuItem(Menu, uItem, fByPosition, &ItemInfo, lpstr); } SetLastNtError(Status); return FALSE; } UINT FASTCALL IntGetMenuState( HMENU hMenu, UINT uId, UINT uFlags) { PMENU MenuObject; PITEM pItem; if (!(MenuObject = UserGetMenuObject(hMenu))) { return (UINT)-1; } if (!(pItem = MENU_FindItem( &MenuObject, &uId, uFlags ))) return -1; if (pItem->spSubMenu) { return (pItem->spSubMenu->cItems << 8) | ((pItem->fState|pItem->fType|MF_POPUP) & 0xff); } else return (pItem->fType | pItem->fState); } HMENU FASTCALL IntGetSubMenu( HMENU hMenu, int nPos) { PMENU MenuObject; PITEM pItem; if (!(MenuObject = UserGetMenuObject(hMenu))) { return NULL; } if (!(pItem = MENU_FindItem( &MenuObject, (UINT*)&nPos, MF_BYPOSITION ))) return NULL; if (pItem->spSubMenu) { HMENU hsubmenu = UserHMGetHandle(pItem->spSubMenu); return hsubmenu; } return NULL; } UINT FASTCALL IntFindSubMenu(HMENU *hMenu, HMENU hSubTarget ) { PMENU menu, pSubTarget; UINT Pos; if (((*hMenu)==(HMENU)0xffff) ||(!(menu = UserGetMenuObject(*hMenu)))) return NO_SELECTED_ITEM; pSubTarget = UserGetMenuObject(hSubTarget); Pos = MENU_FindSubMenu(&menu, pSubTarget ); *hMenu = (menu ? UserHMGetHandle(menu) : NULL); return Pos; } HMENU FASTCALL UserCreateMenu(PDESKTOP Desktop, BOOL PopupMenu) { PWINSTATION_OBJECT WinStaObject; HANDLE Handle; PMENU Menu; NTSTATUS Status; PEPROCESS CurrentProcess = PsGetCurrentProcess(); if (gpepCSRSS != CurrentProcess) { /* * gpepCSRSS does not have a Win32WindowStation */ Status = IntValidateWindowStationHandle(CurrentProcess->Win32WindowStation, UserMode, 0, &WinStaObject, 0); if (!NT_SUCCESS(Status)) { ERR("Validation of window station handle (%p) failed\n", CurrentProcess->Win32WindowStation); SetLastNtError(Status); return (HMENU)0; } Menu = IntCreateMenu(&Handle, !PopupMenu, Desktop, GetW32ProcessInfo()); if (Menu && Menu->head.rpdesk->rpwinstaParent != WinStaObject) { ERR("Desktop Window Station does not match Process one!\n"); } ObDereferenceObject(WinStaObject); } else { Menu = IntCreateMenu(&Handle, !PopupMenu, GetW32ThreadInfo()->rpdesk, GetW32ProcessInfo()); } if (Menu) UserDereferenceObject(Menu); return (HMENU)Handle; } BOOL FASTCALL IntMenuItemInfo( PMENU Menu, UINT Item, BOOL ByPosition, PROSMENUITEMINFO ItemInfo, BOOL SetOrGet, PUNICODE_STRING lpstr) { PITEM MenuItem; BOOL Ret; if (!(MenuItem = MENU_FindItem( &Menu, &Item, (ByPosition ? MF_BYPOSITION : MF_BYCOMMAND) ))) { EngSetLastError(ERROR_MENU_ITEM_NOT_FOUND); return FALSE; } if (SetOrGet) { Ret = IntSetMenuItemInfo(Menu, MenuItem, ItemInfo, lpstr); } else { Ret = IntGetMenuItemInfo(Menu, MenuItem, ItemInfo); } return Ret; } BOOL FASTCALL UserMenuItemInfo( PMENU Menu, UINT Item, BOOL ByPosition, PROSMENUITEMINFO UnsafeItemInfo, BOOL SetOrGet, PUNICODE_STRING lpstr) { PITEM MenuItem; ROSMENUITEMINFO ItemInfo; NTSTATUS Status; UINT Size; BOOL Ret; Status = MmCopyFromCaller(&Size, &UnsafeItemInfo->cbSize, sizeof(UINT)); if (! NT_SUCCESS(Status)) { SetLastNtError(Status); return FALSE; } if ( Size != sizeof(MENUITEMINFOW) && Size != FIELD_OFFSET(MENUITEMINFOW, hbmpItem) && Size != sizeof(ROSMENUITEMINFO) ) { EngSetLastError(ERROR_INVALID_PARAMETER); return FALSE; } Status = MmCopyFromCaller(&ItemInfo, UnsafeItemInfo, Size); if (! NT_SUCCESS(Status)) { SetLastNtError(Status); return FALSE; } /* If this is a pre-0x0500 _WIN32_WINNT MENUITEMINFOW, you can't set/get hbmpItem */ if (FIELD_OFFSET(MENUITEMINFOW, hbmpItem) == Size && 0 != (ItemInfo.fMask & MIIM_BITMAP)) { EngSetLastError(ERROR_INVALID_PARAMETER); return FALSE; } if (!(MenuItem = MENU_FindItem( &Menu, &Item, (ByPosition ? MF_BYPOSITION : MF_BYCOMMAND) ))) { /* workaround for Word 95: pretend that SC_TASKLIST item exists. */ if ( SetOrGet && Item == SC_TASKLIST && !ByPosition ) return TRUE; EngSetLastError(ERROR_MENU_ITEM_NOT_FOUND); return FALSE; } if (SetOrGet) { Ret = IntSetMenuItemInfo(Menu, MenuItem, &ItemInfo, lpstr); } else { Ret = IntGetMenuItemInfo(Menu, MenuItem, &ItemInfo); if (Ret) { Status = MmCopyToCaller(UnsafeItemInfo, &ItemInfo, Size); if (! NT_SUCCESS(Status)) { SetLastNtError(Status); return FALSE; } } } return Ret; } BOOL FASTCALL UserMenuInfo( PMENU Menu, PROSMENUINFO UnsafeMenuInfo, BOOL SetOrGet) { BOOL Res; DWORD Size; NTSTATUS Status; ROSMENUINFO MenuInfo; Status = MmCopyFromCaller(&Size, &UnsafeMenuInfo->cbSize, sizeof(DWORD)); if (! NT_SUCCESS(Status)) { SetLastNtError(Status); return FALSE; } if ( Size < sizeof(MENUINFO) || Size > sizeof(ROSMENUINFO) ) { EngSetLastError(ERROR_INVALID_PARAMETER); return FALSE; } Status = MmCopyFromCaller(&MenuInfo, UnsafeMenuInfo, Size); if (! NT_SUCCESS(Status)) { SetLastNtError(Status); return FALSE; } if(SetOrGet) { /* Set MenuInfo */ Res = IntSetMenuInfo(Menu, &MenuInfo); } else { /* Get MenuInfo */ Res = IntGetMenuInfo(Menu, &MenuInfo); if (Res) { Status = MmCopyToCaller(UnsafeMenuInfo, &MenuInfo, Size); if (! NT_SUCCESS(Status)) { SetLastNtError(Status); return FALSE; } } } return Res; } BOOL FASTCALL IntGetMenuItemRect( PWND pWnd, PMENU Menu, UINT uItem, PRECTL Rect) { LONG XMove, YMove; PITEM MenuItem; UINT I = uItem; if ((MenuItem = MENU_FindItem (&Menu, &I, MF_BYPOSITION))) { Rect->left = MenuItem->xItem; Rect->top = MenuItem->yItem; Rect->right = MenuItem->cxItem; // Do this for now...... Rect->bottom = MenuItem->cyItem; } else { ERR("Failed Item Lookup! %u\n", uItem); return FALSE; } if (!pWnd) { HWND hWnd = Menu->hWnd; if (!(pWnd = UserGetWindowObject(hWnd))) return FALSE; } if (Menu->fFlags & MNF_POPUP) { XMove = pWnd->rcClient.left; YMove = pWnd->rcClient.top; } else { XMove = pWnd->rcWindow.left; YMove = pWnd->rcWindow.top; } Rect->left += XMove; Rect->top += YMove; Rect->right += XMove; Rect->bottom += YMove; return TRUE; } PMENU FASTCALL MENU_GetSystemMenu(PWND Window, PMENU Popup) { PMENU Menu, NewMenu = NULL, SysMenu = NULL; HMENU hSysMenu, hNewMenu = NULL; ROSMENUITEMINFO ItemInfoSet = {0}; ROSMENUITEMINFO ItemInfo = {0}; UNICODE_STRING MenuName; hSysMenu = UserCreateMenu(Window->head.rpdesk, FALSE); if (NULL == hSysMenu) { return NULL; } SysMenu = UserGetMenuObject(hSysMenu); if (NULL == SysMenu) { UserDestroyMenu(hSysMenu); return NULL; } SysMenu->fFlags |= MNF_SYSMENU; SysMenu->hWnd = UserHMGetHandle(Window); if (!Popup) { //hNewMenu = co_IntLoadSysMenuTemplate(); if ( Window->ExStyle & WS_EX_MDICHILD ) { RtlInitUnicodeString( &MenuName, L"SYSMENUMDI"); hNewMenu = co_IntCallLoadMenu( hModClient, &MenuName); } else { RtlInitUnicodeString( &MenuName, L"SYSMENU"); hNewMenu = co_IntCallLoadMenu( hModClient, &MenuName); //ERR("%wZ\n",&MenuName); } if (!hNewMenu) { ERR("No Menu!!\n"); IntDestroyMenuObject(SysMenu, FALSE); return NULL; } Menu = UserGetMenuObject(hNewMenu); if (!Menu) { IntDestroyMenuObject(SysMenu, FALSE); return NULL; } // Do the rest in here. Menu->fFlags |= MNS_CHECKORBMP | MNF_SYSMENU | MNF_POPUP; ItemInfoSet.cbSize = sizeof( MENUITEMINFOW); ItemInfoSet.fMask = MIIM_BITMAP; ItemInfoSet.hbmpItem = HBMMENU_POPUP_CLOSE; IntMenuItemInfo(Menu, SC_CLOSE, FALSE, &ItemInfoSet, TRUE, NULL); ItemInfoSet.hbmpItem = HBMMENU_POPUP_RESTORE; IntMenuItemInfo(Menu, SC_RESTORE, FALSE, &ItemInfoSet, TRUE, NULL); ItemInfoSet.hbmpItem = HBMMENU_POPUP_MAXIMIZE; IntMenuItemInfo(Menu, SC_MAXIMIZE, FALSE, &ItemInfoSet, TRUE, NULL); ItemInfoSet.hbmpItem = HBMMENU_POPUP_MINIMIZE; IntMenuItemInfo(Menu, SC_MINIMIZE, FALSE, &ItemInfoSet, TRUE, NULL); NewMenu = IntCloneMenu(Menu); if (NewMenu == NULL) { IntDestroyMenuObject(Menu, FALSE); IntDestroyMenuObject(SysMenu, FALSE); return NULL; } IntReleaseMenuObject(NewMenu); UserSetMenuDefaultItem(NewMenu, SC_CLOSE, FALSE); IntDestroyMenuObject(Menu, FALSE); } else { NewMenu = Popup; } if (NewMenu) { NewMenu->fFlags |= MNF_SYSMENU | MNF_POPUP; if (Window->pcls->style & CS_NOCLOSE) IntRemoveMenuItem(NewMenu, SC_CLOSE, MF_BYCOMMAND, TRUE); ItemInfo.cbSize = sizeof(MENUITEMINFOW); ItemInfo.fMask = MIIM_FTYPE | MIIM_STRING | MIIM_STATE | MIIM_SUBMENU; ItemInfo.fType = MF_POPUP; ItemInfo.fState = MFS_ENABLED; ItemInfo.dwTypeData = NULL; ItemInfo.cch = 0; ItemInfo.hSubMenu = UserHMGetHandle(NewMenu); IntInsertMenuItem(SysMenu, (UINT) -1, TRUE, &ItemInfo, NULL); return SysMenu; } ERR("failed to load system menu!\n"); return NULL; } PMENU FASTCALL IntGetSystemMenu(PWND Window, BOOL bRevert) { PMENU Menu; if (bRevert) { if (Window->SystemMenu) { Menu = UserGetMenuObject(Window->SystemMenu); if (Menu && !(Menu->fFlags & MNF_SYSDESKMN)) { IntDestroyMenuObject(Menu, TRUE); Window->SystemMenu = NULL; } } } else { Menu = Window->SystemMenu ? UserGetMenuObject(Window->SystemMenu) : NULL; if ((!Menu || Menu->fFlags & MNF_SYSDESKMN) && Window->style & WS_SYSMENU) { Menu = MENU_GetSystemMenu(Window, NULL); Window->SystemMenu = Menu ? UserHMGetHandle(Menu) : NULL; } } if (Window->SystemMenu) { HMENU hMenu = IntGetSubMenu( Window->SystemMenu, 0); /* Store the dummy sysmenu handle to facilitate the refresh */ /* of the close button if the SC_CLOSE item change */ Menu = UserGetMenuObject(hMenu); if (Menu) { Menu->spwndNotify = Window; Menu->fFlags |= MNF_SYSSUBMENU; } return Menu; } return NULL; } BOOL FASTCALL IntSetSystemMenu(PWND Window, PMENU Menu) { PMENU OldMenu; if (!(Window->style & WS_SYSMENU)) return FALSE; if (Window->SystemMenu) { OldMenu = UserGetMenuObject(Window->SystemMenu); if (OldMenu) { OldMenu->fFlags &= ~MNF_SYSMENU; IntDestroyMenuObject(OldMenu, TRUE); } } OldMenu = MENU_GetSystemMenu(Window, Menu); if (OldMenu) { // Use spmenuSys too! Window->SystemMenu = UserHMGetHandle(OldMenu); } else Window->SystemMenu = NULL; if (Menu && Window != Menu->spwndNotify) { Menu->spwndNotify = Window; } return TRUE; } BOOL FASTCALL IntSetMenu( PWND Wnd, HMENU Menu, BOOL *Changed) { PMENU OldMenu, NewMenu = NULL; if ((Wnd->style & (WS_CHILD | WS_POPUP)) == WS_CHILD) { ERR("SetMenu: Window is a Child 0x%p!\n",UserHMGetHandle(Wnd)); EngSetLastError(ERROR_INVALID_WINDOW_HANDLE); return FALSE; } *Changed = (UlongToHandle(Wnd->IDMenu) != Menu); if (! *Changed) { return TRUE; } if (Wnd->IDMenu) { OldMenu = IntGetMenuObject(UlongToHandle(Wnd->IDMenu)); ASSERT(NULL == OldMenu || OldMenu->hWnd == UserHMGetHandle(Wnd)); } else { OldMenu = NULL; } if (NULL != Menu) { NewMenu = IntGetMenuObject(Menu); if (NULL == NewMenu) { if (NULL != OldMenu) { IntReleaseMenuObject(OldMenu); } EngSetLastError(ERROR_INVALID_MENU_HANDLE); return FALSE; } if (NULL != NewMenu->hWnd) { /* Can't use the same menu for two windows */ if (NULL != OldMenu) { IntReleaseMenuObject(OldMenu); } EngSetLastError(ERROR_INVALID_MENU_HANDLE); return FALSE; } } Wnd->IDMenu = (UINT_PTR) Menu; if (NULL != NewMenu) { NewMenu->hWnd = UserHMGetHandle(Wnd); IntReleaseMenuObject(NewMenu); } if (NULL != OldMenu) { OldMenu->hWnd = NULL; IntReleaseMenuObject(OldMenu); } return TRUE; } /* FUNCTIONS *****************************************************************/ /* * @implemented */ /* http://www.cyber-ta.org/releases/malware-analysis/public/SOURCES/b47155634ccb2c30630da7e3666d3d07/b47155634ccb2c30630da7e3666d3d07.trace.html#NtUserGetIconSize */ DWORD APIENTRY NtUserCalcMenuBar( HWND hwnd, DWORD leftBorder, DWORD rightBorder, DWORD top, LPRECT prc ) { HDC hdc; PWND Window; RECT Rect; DWORD ret; UserEnterExclusive(); if(!(Window = UserGetWindowObject(hwnd))) { EngSetLastError(ERROR_INVALID_WINDOW_HANDLE); UserLeave(); return 0; } hdc = UserGetDCEx(NULL, NULL, DCX_CACHE); if (!hdc) { UserLeave(); return 0; } Rect.left = leftBorder; Rect.right = Window->rcWindow.right - Window->rcWindow.left - rightBorder; Rect.top = top; Rect.bottom = 0; ret = MENU_DrawMenuBar(hdc, &Rect, Window, TRUE); UserReleaseDC( 0, hdc, FALSE ); UserLeave(); return ret; } /* * @implemented */ DWORD APIENTRY NtUserCheckMenuItem( HMENU hMenu, UINT uIDCheckItem, UINT uCheck) { PMENU Menu; DWORD Ret = (DWORD)-1; TRACE("Enter NtUserCheckMenuItem\n"); UserEnterExclusive(); Menu = UserGetMenuObject(hMenu); if (Menu) { Ret = IntCheckMenuItem(Menu, uIDCheckItem, uCheck); } TRACE("Leave NtUserCheckMenuItem, ret=%lu\n", Ret); UserLeave(); return Ret; } /* * @implemented */ BOOL APIENTRY NtUserDeleteMenu( HMENU hMenu, UINT uPosition, UINT uFlags) { PMENU Menu; BOOL Ret = FALSE; TRACE("Enter NtUserDeleteMenu\n"); UserEnterExclusive(); Menu = UserGetMenuObject(hMenu); if (Menu) { Ret = IntRemoveMenuItem(Menu, uPosition, uFlags, TRUE); } TRACE("Leave NtUserDeleteMenu, ret=%i\n", Ret); UserLeave(); return Ret; } /* * NtUserGetSystemMenu * * The NtUserGetSystemMenu function allows the application to access the * window menu (also known as the system menu or the control menu) for * copying and modifying. * * Parameters * hWnd * Handle to the window that will own a copy of the window menu. * bRevert * Specifies the action to be taken. If this parameter is FALSE, * NtUserGetSystemMenu returns a handle to the copy of the window menu * currently in use. The copy is initially identical to the window menu * but it can be modified. * If this parameter is TRUE, GetSystemMenu resets the window menu back * to the default state. The previous window menu, if any, is destroyed. * * Return Value * If the bRevert parameter is FALSE, the return value is a handle to a * copy of the window menu. If the bRevert parameter is TRUE, the return * value is NULL. * * Status * @implemented */ HMENU APIENTRY NtUserGetSystemMenu(HWND hWnd, BOOL bRevert) { PWND Window; PMENU Menu; HMENU Ret = NULL; TRACE("Enter NtUserGetSystemMenu\n"); UserEnterExclusive(); if (!(Window = UserGetWindowObject(hWnd))) { goto Exit; // Return NULL } if (!(Menu = IntGetSystemMenu(Window, bRevert))) { goto Exit; // Return NULL } Ret = UserHMGetHandle(Menu); Exit: TRACE("Leave NtUserGetSystemMenu, ret=%p\n", Ret); UserLeave(); return Ret; } /* * NtUserSetSystemMenu * * Status * @implemented */ BOOL APIENTRY NtUserSetSystemMenu(HWND hWnd, HMENU hMenu) { BOOL Result = FALSE; PWND Window; PMENU Menu; TRACE("Enter NtUserSetSystemMenu\n"); UserEnterExclusive(); if (!(Window = UserGetWindowObject(hWnd))) { goto Exit; // Return FALSE } if (hMenu) { /* * Assign new menu handle and Up the Lock Count. */ if (!(Menu = IntGetMenuObject(hMenu))) { goto Exit; // Return FALSE } Result = IntSetSystemMenu(Window, Menu); } else EngSetLastError(ERROR_INVALID_MENU_HANDLE); Exit: TRACE("Leave NtUserSetSystemMenu, ret=%i\n", Result); UserLeave(); return Result; } /* * @implemented */ BOOLEAN APIENTRY NtUserGetTitleBarInfo( HWND hwnd, PTITLEBARINFO bti) { PWND WindowObject; TITLEBARINFO bartitleinfo; BOOLEAN retValue = TRUE; TRACE("Enter NtUserGetTitleBarInfo\n"); UserEnterExclusive(); /* Vaildate the windows handle */ if (!(WindowObject = UserGetWindowObject(hwnd))) { EngSetLastError(ERROR_INVALID_WINDOW_HANDLE); retValue = FALSE; } _SEH2_TRY { /* Copy our usermode buffer bti to local buffer bartitleinfo */ ProbeForRead(bti, sizeof(TITLEBARINFO), 1); RtlCopyMemory(&bartitleinfo, bti, sizeof(TITLEBARINFO)); } _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) { /* Fail copy the data */ EngSetLastError(ERROR_INVALID_PARAMETER); retValue = FALSE; } _SEH2_END /* Get the tile bar info */ if (retValue) { retValue = intGetTitleBarInfo(WindowObject, &bartitleinfo); if (retValue) { _SEH2_TRY { /* Copy our buffer to user mode buffer bti */ ProbeForWrite(bti, sizeof(TITLEBARINFO), 1); RtlCopyMemory(bti, &bartitleinfo, sizeof(TITLEBARINFO)); } _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) { /* Fail copy the data */ EngSetLastError(ERROR_INVALID_PARAMETER); retValue = FALSE; } _SEH2_END } } TRACE("Leave NtUserGetTitleBarInfo, ret=%u\n", retValue); UserLeave(); return retValue; } /* * @implemented */ BOOL FASTCALL UserDestroyMenu(HMENU hMenu) { PMENU Menu; PTHREADINFO pti = PsGetCurrentThreadWin32Thread(); if(!(Menu = UserGetMenuObject(hMenu))) { return FALSE; } if (Menu->head.rpdesk != pti->rpdesk) { EngSetLastError(ERROR_ACCESS_DENIED); return FALSE; } return IntDestroyMenuObject(Menu, FALSE); } /* * @implemented */ BOOL APIENTRY NtUserDestroyMenu( HMENU hMenu) { PMENU Menu; BOOL Ret = FALSE; TRACE("Enter NtUserDestroyMenu\n"); UserEnterExclusive(); if(!(Menu = UserGetMenuObject(hMenu))) { goto Exit; // Return FALSE } if (Menu->head.rpdesk != gptiCurrent->rpdesk) { EngSetLastError(ERROR_ACCESS_DENIED); goto Exit; // Return FALSE } Ret = IntDestroyMenuObject(Menu, TRUE); Exit: TRACE("Leave NtUserDestroyMenu, ret=%i\n", Ret); UserLeave(); return Ret; } /* * @implemented */ UINT APIENTRY NtUserEnableMenuItem( HMENU hMenu, UINT uIDEnableItem, UINT uEnable) { PMENU Menu; UINT Ret = (UINT)-1; TRACE("Enter NtUserEnableMenuItem\n"); UserEnterExclusive(); Menu = UserGetMenuObject(hMenu); if (Menu) { Ret = IntEnableMenuItem(Menu, uIDEnableItem, uEnable); } TRACE("Leave NtUserEnableMenuItem, ret=%u\n", Ret); UserLeave(); return Ret; } /* * @implemented */ BOOL APIENTRY NtUserEndMenu(VOID) { //PWND pWnd; TRACE("Enter NtUserEndMenu\n"); UserEnterExclusive(); /* if ( gptiCurrent->pMenuState && gptiCurrent->pMenuState->pGlobalPopupMenu ) { pWnd = IntGetMSWND(gptiCurrent->pMenuState); if (pWnd) { UserPostMessage( UserHMGetHandle(pWnd), WM_CANCELMODE, 0, 0); } else gptiCurrent->pMenuState->fInsideMenuLoop = FALSE; }*/ if (fInsideMenuLoop && top_popup) { fInsideMenuLoop = FALSE; UserPostMessage( top_popup, WM_CANCELMODE, 0, 0); } UserLeave(); TRACE("Leave NtUserEndMenu\n"); return TRUE; } /* * @implemented */ BOOL APIENTRY NtUserGetMenuBarInfo( HWND hwnd, LONG idObject, LONG idItem, PMENUBARINFO pmbi) { PWND pWnd; HMENU hMenu; MENUBARINFO kmbi; BOOL Ret = FALSE; PPOPUPMENU pPopupMenu; USER_REFERENCE_ENTRY Ref; NTSTATUS Status = STATUS_SUCCESS; PMENU Menu = NULL; TRACE("Enter NtUserGetMenuBarInfo\n"); UserEnterShared(); if (!(pWnd = UserGetWindowObject(hwnd))) { EngSetLastError(ERROR_INVALID_WINDOW_HANDLE); goto Cleanup; // Return FALSE } UserRefObjectCo(pWnd, &Ref); RECTL_vSetEmptyRect(&kmbi.rcBar); kmbi.hMenu = NULL; kmbi.hwndMenu = NULL; kmbi.fBarFocused = FALSE; kmbi.fFocused = FALSE; switch (idObject) { case OBJID_CLIENT: if (!pWnd->pcls->fnid) goto Cleanup; // Return FALSE if (pWnd->pcls->fnid != FNID_MENU) { WARN("called on invalid window: %u\n", pWnd->pcls->fnid); EngSetLastError(ERROR_INVALID_MENU_HANDLE); goto Cleanup; // Return FALSE } // Windows does this! Wine checks for Atom and uses GetWindowLongPtrW. hMenu = (HMENU)co_IntSendMessage(hwnd, MN_GETHMENU, 0, 0); pPopupMenu = ((PMENUWND)pWnd)->ppopupmenu; if (pPopupMenu && pPopupMenu->spmenu) { if (UserHMGetHandle(pPopupMenu->spmenu) != hMenu) { ERR("Window Pop Up hMenu %p not the same as Get hMenu %p!\n", UserHMGetHandle(pPopupMenu->spmenu), hMenu); } } break; case OBJID_MENU: if (pWnd->style & WS_CHILD) goto Cleanup; // Return FALSE hMenu = UlongToHandle(pWnd->IDMenu); TRACE("GMBI: OBJID_MENU hMenu %p\n",hMenu); break; case OBJID_SYSMENU: if (!(pWnd->style & WS_SYSMENU)) goto Cleanup; // Return FALSE Menu = IntGetSystemMenu(pWnd, FALSE); hMenu = UserHMGetHandle(Menu); break; default: goto Cleanup; // Return FALSE } if (!hMenu) goto Cleanup; // Return FALSE _SEH2_TRY { ProbeForRead(pmbi, sizeof(MENUBARINFO), 1); kmbi.cbSize = pmbi->cbSize; } _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) { kmbi.cbSize = 0; } _SEH2_END if (kmbi.cbSize != sizeof(MENUBARINFO)) { EngSetLastError(ERROR_INVALID_PARAMETER); goto Cleanup; // Return FALSE } if (!Menu) { Menu = UserGetMenuObject(hMenu); if (!Menu) goto Cleanup; // Return FALSE } if ((idItem < 0) || ((ULONG)idItem > Menu->cItems)) goto Cleanup; // Return FALSE if (idItem == 0) { Ret = IntGetMenuItemRect(pWnd, Menu, 0, &kmbi.rcBar); kmbi.rcBar.right = kmbi.rcBar.left + Menu->cxMenu; kmbi.rcBar.bottom = kmbi.rcBar.top + Menu->cyMenu; TRACE("idItem a 0 %d\n",Ret); } else { Ret = IntGetMenuItemRect(pWnd, Menu, idItem-1, &kmbi.rcBar); TRACE("idItem b %d %d\n", idItem-1, Ret); } kmbi.hMenu = hMenu; kmbi.fBarFocused = top_popup_hmenu == hMenu; TRACE("GMBI: top p hm %p hMenu %p\n",top_popup_hmenu, hMenu); if (idItem) { kmbi.fFocused = Menu->iItem == idItem-1; if (kmbi.fFocused && (Menu->rgItems[idItem - 1].spSubMenu)) { kmbi.hwndMenu = Menu->rgItems[idItem - 1].spSubMenu->hWnd; } } else { kmbi.fFocused = kmbi.fBarFocused; } _SEH2_TRY { ProbeForWrite(pmbi, sizeof(MENUBARINFO), 1); RtlCopyMemory(pmbi, &kmbi, sizeof(MENUBARINFO)); } _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) { Status = _SEH2_GetExceptionCode(); } _SEH2_END if (!NT_SUCCESS(Status)) { SetLastNtError(Status); Ret = FALSE; goto Cleanup; } Ret = TRUE; Cleanup: if (pWnd) UserDerefObjectCo(pWnd); TRACE("Leave NtUserGetMenuBarInfo, ret=%i\n", Ret); UserLeave(); return Ret; } /* * @implemented */ UINT APIENTRY NtUserGetMenuIndex( HMENU hMenu, HMENU hSubMenu) { PMENU Menu, SubMenu; PITEM MenuItem; UINT i; UINT Ret = 0xFFFFFFFF; TRACE("Enter NtUserGetMenuIndex\n"); UserEnterShared(); if ( !(Menu = UserGetMenuObject(hMenu)) || !(SubMenu = UserGetMenuObject(hSubMenu)) ) goto Exit; // Return 0xFFFFFFFF MenuItem = Menu->rgItems; for (i = 0; i < Menu->cItems; i++, MenuItem++) { if (MenuItem->spSubMenu == SubMenu) { Ret = MenuItem->wID; break; } } Exit: TRACE("Leave NtUserGetMenuIndex, ret=%u\n", Ret); UserLeave(); return Ret; } /* * @implemented */ BOOL APIENTRY NtUserGetMenuItemRect( HWND hWnd, HMENU hMenu, UINT uItem, PRECTL lprcItem) { PWND ReferenceWnd; LONG XMove, YMove; RECTL Rect; PMENU Menu; PITEM MenuItem; NTSTATUS Status = STATUS_SUCCESS; BOOL Ret = FALSE; TRACE("Enter NtUserGetMenuItemRect\n"); UserEnterShared(); if (!(Menu = UserGetMenuObject(hMenu))) { goto Exit; // Return FALSE } if ((MenuItem = MENU_FindItem (&Menu, &uItem, MF_BYPOSITION))) { Rect.left = MenuItem->xItem; Rect.top = MenuItem->yItem; Rect.right = MenuItem->cxItem; // Do this for now...... Rect.bottom = MenuItem->cyItem; } else goto Exit; // Return FALSE if(!hWnd) { hWnd = Menu->hWnd; } if (lprcItem == NULL) goto Exit; // Return FALSE if (!(ReferenceWnd = UserGetWindowObject(hWnd))) goto Exit; // Return FALSE if (Menu->fFlags & MNF_POPUP) { XMove = ReferenceWnd->rcClient.left; YMove = ReferenceWnd->rcClient.top; } else { XMove = ReferenceWnd->rcWindow.left; YMove = ReferenceWnd->rcWindow.top; } Rect.left += XMove; Rect.top += YMove; Rect.right += XMove; Rect.bottom += YMove; _SEH2_TRY { RtlCopyMemory(lprcItem, &Rect, sizeof(RECTL)); } _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) { Status = _SEH2_GetExceptionCode(); } _SEH2_END if (!NT_SUCCESS(Status)) { SetLastNtError(Status); goto Exit; // Return FALSE } Ret = TRUE; Exit: TRACE("Leave NtUserGetMenuItemRect, ret=%i\n", Ret); UserLeave(); return Ret; } /* * @implemented */ BOOL APIENTRY NtUserHiliteMenuItem( HWND hWnd, HMENU hMenu, UINT uItemHilite, UINT uHilite) { PMENU Menu; PWND Window; BOOL Ret = FALSE; TRACE("Enter NtUserHiliteMenuItem\n"); UserEnterExclusive(); if(!(Window = UserGetWindowObject(hWnd))) { EngSetLastError(ERROR_INVALID_WINDOW_HANDLE); goto Exit; // Return FALSE } if(!(Menu = UserGetMenuObject(hMenu))) { EngSetLastError(ERROR_INVALID_MENU_HANDLE); goto Exit; // Return FALSE } Ret = IntHiliteMenuItem(Window, Menu, uItemHilite, uHilite); Exit: TRACE("Leave NtUserHiliteMenuItem, ret=%i\n", Ret); UserLeave(); return Ret; } /* * @implemented */ DWORD APIENTRY NtUserDrawMenuBarTemp( HWND hWnd, HDC hDC, PRECT pRect, HMENU hMenu, HFONT hFont) { PMENU Menu; PWND Window; RECT Rect; NTSTATUS Status = STATUS_SUCCESS; DWORD Ret = 0; ERR("Enter NtUserDrawMenuBarTemp\n"); UserEnterExclusive(); if(!(Window = UserGetWindowObject(hWnd))) { EngSetLastError(ERROR_INVALID_WINDOW_HANDLE); goto Exit; // Return 0 } if(!(Menu = UserGetMenuObject(hMenu))) { EngSetLastError(ERROR_INVALID_MENU_HANDLE); goto Exit; // Return 0 } _SEH2_TRY { ProbeForRead(pRect, sizeof(RECT), sizeof(ULONG)); RtlCopyMemory(&Rect, pRect, sizeof(RECT)); } _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) { Status = _SEH2_GetExceptionCode(); } _SEH2_END; if (Status != STATUS_SUCCESS) { SetLastNtError(Status); goto Exit; // Return 0 } Ret = IntDrawMenuBarTemp(Window, hDC, &Rect, Menu, hFont); Exit: ERR("Leave NtUserDrawMenuBarTemp, ret=%lu\n", Ret); UserLeave(); return Ret; } /* * @implemented */ int APIENTRY NtUserMenuItemFromPoint( HWND hWnd, HMENU hMenu, DWORD X, DWORD Y) { PMENU Menu; PWND Window = NULL; PITEM mi; ULONG i; int Ret = -1; TRACE("Enter NtUserMenuItemFromPoint\n"); UserEnterExclusive(); if (!(Menu = UserGetMenuObject(hMenu))) { goto Exit; // Return -1 } if (!(Window = UserGetWindowObject(Menu->hWnd))) { goto Exit; // Return -1 } X -= Window->rcWindow.left; Y -= Window->rcWindow.top; mi = Menu->rgItems; for (i = 0; i < Menu->cItems; i++, mi++) { RECTL Rect; Rect.left = mi->xItem; Rect.top = mi->yItem; Rect.right = mi->cxItem; Rect.bottom = mi->cyItem; MENU_AdjustMenuItemRect(Menu, &Rect); if (RECTL_bPointInRect(&Rect, X, Y)) { break; } } Ret = (mi ? i : NO_SELECTED_ITEM); Exit: TRACE("Leave NtUserMenuItemFromPoint, ret=%i\n", Ret); UserLeave(); return Ret; } DWORD APIENTRY NtUserPaintMenuBar( HWND hWnd, HDC hDC, ULONG leftBorder, ULONG rightBorder, ULONG top, BOOL bActive) { PWND Window; RECT Rect; DWORD ret; UserEnterExclusive(); if(!(Window = UserGetWindowObject(hWnd))) { EngSetLastError(ERROR_INVALID_WINDOW_HANDLE); UserLeave(); return 0; } Rect.left = leftBorder; Rect.right = Window->rcWindow.right - Window->rcWindow.left - rightBorder; Rect.top = top; Rect.bottom = 0; ret = MENU_DrawMenuBar(hDC, &Rect, Window, FALSE); UserLeave(); return ret; } /* * @implemented */ BOOL APIENTRY NtUserRemoveMenu( HMENU hMenu, UINT uPosition, UINT uFlags) { PMENU Menu; BOOL Ret = FALSE; TRACE("Enter NtUserRemoveMenu\n"); UserEnterExclusive(); Menu = UserGetMenuObject(hMenu); if (Menu) { Ret = IntRemoveMenuItem(Menu, uPosition, uFlags, FALSE); } TRACE("Leave NtUserRemoveMenu, ret=%i\n", Ret); UserLeave(); return Ret; } /* * @implemented */ BOOL APIENTRY NtUserSetMenu( HWND hWnd, HMENU Menu, BOOL Repaint) { PWND Window; BOOL Changed; BOOL Ret = FALSE; TRACE("Enter NtUserSetMenu\n"); UserEnterExclusive(); if (!(Window = UserGetWindowObject(hWnd))) { goto Exit; // Return FALSE } if (!IntSetMenu(Window, Menu, &Changed)) { goto Exit; // Return FALSE } // Not minimized and please repaint!!! if (!(Window->style & WS_MINIMIZE) && (Repaint || Changed)) { USER_REFERENCE_ENTRY Ref; UserRefObjectCo(Window, &Ref); co_WinPosSetWindowPos(Window, 0, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE | SWP_NOACTIVATE | SWP_NOZORDER | SWP_FRAMECHANGED); UserDerefObjectCo(Window); } Ret = TRUE; Exit: TRACE("Leave NtUserSetMenu, ret=%i\n", Ret); UserLeave(); return Ret; } /* * @implemented */ BOOL APIENTRY NtUserSetMenuContextHelpId( HMENU hMenu, DWORD dwContextHelpId) { PMENU Menu; BOOL Ret = FALSE; TRACE("Enter NtUserSetMenuContextHelpId\n"); UserEnterExclusive(); Menu = UserGetMenuObject(hMenu); if (Menu) { Ret = IntSetMenuContextHelpId(Menu, dwContextHelpId); } TRACE("Leave NtUserSetMenuContextHelpId, ret=%i\n", Ret); UserLeave(); return Ret; } /* * @implemented */ BOOL APIENTRY NtUserSetMenuDefaultItem( HMENU hMenu, UINT uItem, UINT fByPos) { PMENU Menu; BOOL Ret = FALSE; TRACE("Enter NtUserSetMenuDefaultItem\n"); UserEnterExclusive(); Menu = UserGetMenuObject(hMenu); if (Menu) { Ret = UserSetMenuDefaultItem(Menu, uItem, fByPos); } TRACE("Leave NtUserSetMenuDefaultItem, ret=%i\n", Ret); UserLeave(); return Ret; } /* * @implemented */ BOOL APIENTRY NtUserSetMenuFlagRtoL( HMENU hMenu) { PMENU Menu; BOOL Ret = FALSE; TRACE("Enter NtUserSetMenuFlagRtoL\n"); UserEnterExclusive(); Menu = UserGetMenuObject(hMenu); if (Menu) { Ret = IntSetMenuFlagRtoL(Menu); } TRACE("Leave NtUserSetMenuFlagRtoL, ret=%i\n", Ret); UserLeave(); return Ret; } /* * @implemented */ BOOL APIENTRY NtUserThunkedMenuInfo( HMENU hMenu, LPCMENUINFO lpcmi) { PMENU Menu; BOOL Ret = FALSE; TRACE("Enter NtUserThunkedMenuInfo\n"); UserEnterExclusive(); Menu = UserGetMenuObject(hMenu); if (Menu) { Ret = UserMenuInfo(Menu, (PROSMENUINFO)lpcmi, TRUE); } TRACE("Leave NtUserThunkedMenuInfo, ret=%i\n", Ret); UserLeave(); return Ret; } /* * @implemented */ BOOL APIENTRY NtUserThunkedMenuItemInfo( HMENU hMenu, UINT uItem, BOOL fByPosition, BOOL bInsert, LPMENUITEMINFOW lpmii, PUNICODE_STRING lpszCaption) { PMENU Menu; NTSTATUS Status; UNICODE_STRING lstrCaption; BOOL Ret = FALSE; TRACE("Enter NtUserThunkedMenuItemInfo\n"); UserEnterExclusive(); /* lpszCaption may be NULL, check for it and call RtlInitUnicodeString() if bInsert == TRUE call UserInsertMenuItem() else UserSetMenuItemInfo() */ RtlInitEmptyUnicodeString(&lstrCaption, NULL, 0); if (!(Menu = UserGetMenuObject(hMenu))) { goto Cleanup; // Return FALSE } /* Check if we got a Caption */ if (lpszCaption && lpszCaption->Buffer) { /* Copy the string to kernel mode */ Status = ProbeAndCaptureUnicodeString( &lstrCaption, UserMode, lpszCaption); if (!NT_SUCCESS(Status)) { ERR("Failed to capture MenuItem Caption (status 0x%08x)\n",Status); SetLastNtError(Status); goto Cleanup; // Return FALSE } } if (bInsert) { Ret = UserInsertMenuItem(Menu, uItem, fByPosition, lpmii, &lstrCaption); goto Cleanup; } Ret = UserMenuItemInfo(Menu, uItem, fByPosition, (PROSMENUITEMINFO)lpmii, TRUE, &lstrCaption); Cleanup: if (lstrCaption.Buffer) { ReleaseCapturedUnicodeString(&lstrCaption, UserMode); } TRACE("Leave NtUserThunkedMenuItemInfo, ret=%i\n", Ret); UserLeave(); return Ret; } /* * @implemented */ BOOL APIENTRY NtUserTrackPopupMenuEx( HMENU hMenu, UINT fuFlags, int x, int y, HWND hWnd, LPTPMPARAMS lptpm) { PMENU menu; PWND pWnd; TPMPARAMS tpm; BOOL Ret = FALSE; USER_REFERENCE_ENTRY Ref; TRACE("Enter NtUserTrackPopupMenuEx\n"); UserEnterExclusive(); /* Parameter check */ if (!(menu = UserGetMenuObject( hMenu ))) { ERR("TPME : Invalid Menu handle.\n"); EngSetLastError( ERROR_INVALID_MENU_HANDLE ); goto Exit; } if (!(pWnd = UserGetWindowObject(hWnd))) { ERR("TPME : Invalid Window handle.\n"); goto Exit; } if (lptpm) { _SEH2_TRY { ProbeForRead(lptpm, sizeof(TPMPARAMS), sizeof(ULONG)); tpm = *lptpm; } _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER) { _SEH2_YIELD(goto Exit); } _SEH2_END } UserRefObjectCo(pWnd, &Ref); Ret = IntTrackPopupMenuEx(menu, fuFlags, x, y, pWnd, lptpm ? &tpm : NULL); UserDerefObjectCo(pWnd); Exit: TRACE("Leave NtUserTrackPopupMenuEx, ret=%i\n",Ret); UserLeave(); return Ret; } /* EOF */