1
0
Fork 0
libreoffice/vcl/win/window/salmenu.cxx
Daniel Baumann 8e63e14cf6
Adding upstream version 4:25.2.3.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
2025-06-22 16:20:04 +02:00

328 lines
9.5 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*
* This file is part of the LibreOffice project.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*
* This file incorporates work covered by the following license notice:
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed
* with this work for additional information regarding copyright
* ownership. The ASF licenses this file to you under the Apache
* License, Version 2.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.apache.org/licenses/LICENSE-2.0 .
*/
#include <svsys.h>
#include <vcl/menu.hxx>
#include <vcl/sysdata.hxx>
#include <o3tl/char16_t2wchar_t.hxx>
#include <o3tl/safeint.hxx>
#include <win/wincomp.hxx>
#include <win/saldata.hxx>
#include <win/salinst.h>
#include <win/salframe.h>
#include <win/salmenu.h>
#include <salgdi.hxx>
static DWORD myerr=0;
bool SalData::IsKnownMenuHandle( HMENU hMenu )
{
if( mhMenuSet.find( hMenu ) == mhMenuSet.end() )
return false;
else
return true;
}
// WinSalInst factory methods
std::unique_ptr<SalMenu> WinSalInstance::CreateMenu( bool bMenuBar, Menu* )
{
WinSalMenu *pSalMenu = new WinSalMenu();
pSalMenu->mbMenuBar = bMenuBar;
pSalMenu->mhWnd = nullptr;
if( bMenuBar )
pSalMenu->mhMenu = ::CreateMenu();
else
pSalMenu->mhMenu = ::CreatePopupMenu();
if( pSalMenu->mhMenu )
GetSalData()->mhMenuSet.insert( pSalMenu->mhMenu );
return std::unique_ptr<SalMenu>(pSalMenu);
}
std::unique_ptr<SalMenuItem> WinSalInstance::CreateMenuItem( const SalItemParams & rItemData )
{
WinSalMenuItem *pSalMenuItem = new WinSalMenuItem();
memset( &pSalMenuItem->mInfo, 0, sizeof( MENUITEMINFOW ) );
pSalMenuItem->mInfo.cbSize = sizeof( MENUITEMINFOW );
if( rItemData.eType == MenuItemType::SEPARATOR )
{
// separator
pSalMenuItem->mInfo.fMask = MIIM_TYPE;
pSalMenuItem->mInfo.fType = MFT_SEPARATOR;
}
else
{
// item
pSalMenuItem->mText = rItemData.aText;
pSalMenuItem->mpMenu = rItemData.pMenu;
pSalMenuItem->maBitmap= !!rItemData.aImage ? rItemData.aImage.GetBitmapEx().GetBitmap() : Bitmap();
pSalMenuItem->mnId = rItemData.nId;
// 'translate' mnemonics
pSalMenuItem->mText = pSalMenuItem->mText.replaceAll( "~", "&" );
pSalMenuItem->mInfo.fMask = MIIM_TYPE | MIIM_STATE | MIIM_ID | MIIM_DATA;
pSalMenuItem->mInfo.fType = MFT_STRING;
pSalMenuItem->mInfo.dwTypeData = o3tl::toW(const_cast<sal_Unicode *>(pSalMenuItem->mText.getStr()));
pSalMenuItem->mInfo.cch = pSalMenuItem->mText.getLength();
pSalMenuItem->mInfo.wID = rItemData.nId;
pSalMenuItem->mInfo.dwItemData = reinterpret_cast<ULONG_PTR>(pSalMenuItem); // user data
}
return std::unique_ptr<SalMenuItem>(pSalMenuItem);
}
/*
* WinSalMenu
*/
WinSalMenu::WinSalMenu()
{
mhMenu = nullptr;
mbMenuBar = false;
mhWnd = nullptr;
mpParentMenu = nullptr;
}
WinSalMenu::~WinSalMenu()
{
// only required if not associated to a window...
GetSalData()->mhMenuSet.erase( mhMenu );
::DestroyMenu( mhMenu );
}
bool WinSalMenu::VisibleMenuBar()
{
// The Win32 implementation never shows a native
// menubar. Thus, native menus are only visible
// when the menu is merged with an OLE container.
// The reason are missing tooltips, ownerdraw
// issues and accessibility which are better supported
// by VCL menus.
// Nevertheless, the native menus are always created
// and the application will properly react to all native
// menu messages.
return false;
}
void WinSalMenu::SetFrame( const SalFrame *pFrame )
{
if( pFrame )
mhWnd = static_cast<const WinSalFrame*>(pFrame)->mhWnd;
else
mhWnd = nullptr;
}
void WinSalMenu::InsertItem( SalMenuItem* pSalMenuItem, unsigned nPos )
{
if( pSalMenuItem )
{
WinSalMenuItem* pWItem = static_cast<WinSalMenuItem*>(pSalMenuItem);
if( nPos == MENU_APPEND )
{
nPos = ::GetMenuItemCount( mhMenu );
if( nPos == static_cast<unsigned>( -1 ) )
return;
}
if(!::InsertMenuItemW( mhMenu, nPos, TRUE, &pWItem->mInfo ))
myerr = GetLastError();
else
pWItem->mpSalMenu = this;
}
}
void WinSalMenu::RemoveItem( unsigned nPos )
{
int num = ::GetMenuItemCount( mhMenu );
if( num != -1 && nPos < o3tl::make_unsigned(num) )
{
WinSalMenuItem *pSalMenuItem = nullptr;
MENUITEMINFOW mi = {};
mi.cbSize = sizeof( mi );
mi.fMask = MIIM_DATA;
if( !GetMenuItemInfoW( mhMenu, nPos, TRUE, &mi) )
myerr = GetLastError();
else
pSalMenuItem = reinterpret_cast<WinSalMenuItem *>(mi.dwItemData);
if( !::RemoveMenu( mhMenu, nPos, MF_BYPOSITION ) )
myerr = GetLastError();
else
{
if( pSalMenuItem )
pSalMenuItem->mpSalMenu = nullptr;
}
}
}
static void ImplRemoveItemById( WinSalMenu *pSalMenu, unsigned nItemId )
{
if( !pSalMenu )
return;
WinSalMenuItem *pSalMenuItem = nullptr;
MENUITEMINFOW mi = {};
mi.cbSize = sizeof( mi );
mi.fMask = MIIM_DATA;
if( !GetMenuItemInfoW( pSalMenu->mhMenu, nItemId, FALSE, &mi) )
myerr = GetLastError();
else
pSalMenuItem = reinterpret_cast<WinSalMenuItem *>(mi.dwItemData);
if( !::RemoveMenu( pSalMenu->mhMenu, nItemId, MF_BYCOMMAND ) )
myerr = GetLastError();
else
{
if( pSalMenuItem )
pSalMenuItem->mpSalMenu = nullptr;
}
}
void WinSalMenu::SetSubMenu( SalMenuItem* pSalMenuItem, SalMenu* pSubMenu, unsigned nPos )
{
if( pSalMenuItem )
{
WinSalMenuItem* pWMenuItem = static_cast<WinSalMenuItem*>(pSalMenuItem);
WinSalMenu* pWSubMenu = static_cast<WinSalMenu*>(pSubMenu);
if( pWMenuItem->mInfo.hSubMenu )
{
GetSalData()->mhMenuSet.erase( pWMenuItem->mInfo.hSubMenu );
::DestroyMenu( pWMenuItem->mInfo.hSubMenu );
}
pWMenuItem->mInfo.fMask |= MIIM_SUBMENU;
if( !pSubMenu )
pWMenuItem->mInfo.hSubMenu = nullptr;
else
{
pWMenuItem->mInfo.hSubMenu = pWSubMenu->mhMenu;
pWSubMenu->mpParentMenu = this;
}
if(!::SetMenuItemInfoW( mhMenu, nPos, TRUE, &pWMenuItem->mInfo ) )
myerr = GetLastError();
}
}
void WinSalMenu::CheckItem( unsigned nPos, bool bCheck )
{
::CheckMenuItem(mhMenu, nPos, MF_BYPOSITION|(bCheck ? MF_CHECKED : MF_UNCHECKED));
}
void WinSalMenu::EnableItem( unsigned nPos, bool bEnable )
{
::EnableMenuItem(mhMenu, nPos, MF_BYPOSITION|(bEnable ? MF_ENABLED : (MF_DISABLED|MF_GRAYED)));
}
void WinSalMenu::SetItemImage( unsigned /*nPos*/, SalMenuItem* pSalMenuItem, const Image& rImage )
{
if( pSalMenuItem )
{
WinSalMenuItem* pWItem = static_cast<WinSalMenuItem*>(pSalMenuItem);
if( !!rImage )
pWItem->maBitmap = rImage.GetBitmapEx().GetBitmap();
else
pWItem->maBitmap = Bitmap();
}
}
void WinSalMenu::SetItemText( unsigned nPos, SalMenuItem* pSalMenuItem, const OUString& rText )
{
if( pSalMenuItem )
{
WinSalMenuItem* pWItem = static_cast<WinSalMenuItem*>(pSalMenuItem);
pWItem->mText = rText;
// 'translate' mnemonics
pWItem->mText = pWItem->mText.replaceAll( "~", "&" );
pWItem->mInfo.fMask = MIIM_TYPE | MIIM_DATA;
pWItem->mInfo.fType = MFT_STRING;
// combine text and accelerator text
OUString aStr( pWItem->mText );
if( pWItem->mAccelText.getLength() )
{
aStr += "\t" + pWItem->mAccelText;
}
pWItem->mInfo.dwTypeData = o3tl::toW(const_cast<sal_Unicode *>(aStr.getStr()));
pWItem->mInfo.cch = aStr.getLength();
if(!::SetMenuItemInfoW( mhMenu, nPos, TRUE, &pWItem->mInfo ))
myerr = GetLastError();
}
}
void WinSalMenu::SetAccelerator( unsigned nPos, SalMenuItem* pSalMenuItem, const vcl::KeyCode&, const OUString& rKeyName )
{
if( pSalMenuItem )
{
WinSalMenuItem* pWItem = static_cast<WinSalMenuItem*>(pSalMenuItem);
pWItem->mAccelText = rKeyName;
pWItem->mInfo.fMask = MIIM_TYPE | MIIM_DATA;
pWItem->mInfo.fType = MFT_STRING;
// combine text and accelerator text
OUString aStr( pWItem->mText );
if( pWItem->mAccelText.getLength() )
{
aStr += "\t" + pWItem->mAccelText;
}
pWItem->mInfo.dwTypeData = o3tl::toW(const_cast<sal_Unicode *>(aStr.getStr()));
pWItem->mInfo.cch = aStr.getLength();
if(!::SetMenuItemInfoW( mhMenu, nPos, TRUE, &pWItem->mInfo ))
myerr = GetLastError();
}
}
void WinSalMenu::GetSystemMenuData(SystemMenuData& rData)
{
rData.hMenu = mhMenu;
}
/*
* SalMenuItem
*/
WinSalMenuItem::WinSalMenuItem()
{
memset( &mInfo, 0, sizeof( MENUITEMINFOW ) );
mpMenu = nullptr;
mnId = 0xFFFF;
mpSalMenu = nullptr;
}
WinSalMenuItem::~WinSalMenuItem()
{
if( mpSalMenu )
ImplRemoveItemById( mpSalMenu, mnId );
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */