1
0
Fork 0
libreoffice/framework/source/uielement/controlmenucontroller.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.8 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 <sal/config.h>
#include <com/sun/star/frame/XDispatch.hpp>
#include <com/sun/star/frame/XDispatchProvider.hpp>
#include <com/sun/star/frame/XFrame.hpp>
#include <com/sun/star/frame/XStatusListener.hpp>
#include <com/sun/star/util/XURLTransformer.hpp>
#include <cppuhelper/supportsservice.hxx>
#include <vcl/graph.hxx>
#include <vcl/svapp.hxx>
#include <vcl/settings.hxx>
#include <vcl/image.hxx>
#include <svtools/popupmenucontrollerbase.hxx>
#include <toolkit/awt/vclxmenu.hxx>
#include <osl/mutex.hxx>
#include <unordered_map>
#include <classes/fwkresid.hxx>
#include <bitmaps.hlst>
#include <strings.hrc>
constexpr OUString aCommands[]
{
u".uno:ConvertToEdit"_ustr,
u".uno:ConvertToButton"_ustr,
u".uno:ConvertToFixed"_ustr,
u".uno:ConvertToList"_ustr,
u".uno:ConvertToCheckBox"_ustr,
u".uno:ConvertToRadio"_ustr,
u".uno:ConvertToGroup"_ustr,
u".uno:ConvertToCombo"_ustr,
u".uno:ConvertToImageBtn"_ustr,
u".uno:ConvertToFileControl"_ustr,
u".uno:ConvertToDate"_ustr,
u".uno:ConvertToTime"_ustr,
u".uno:ConvertToNumeric"_ustr,
u".uno:ConvertToCurrency"_ustr,
u".uno:ConvertToPattern"_ustr,
u".uno:ConvertToImageControl"_ustr,
u".uno:ConvertToFormatted"_ustr,
u".uno:ConvertToScrollBar"_ustr,
u".uno:ConvertToSpinButton"_ustr,
u".uno:ConvertToNavigationBar"_ustr
};
static TranslateId aLabels[] =
{
RID_STR_PROPTITLE_EDIT,
RID_STR_PROPTITLE_PUSHBUTTON,
RID_STR_PROPTITLE_FIXEDTEXT,
RID_STR_PROPTITLE_LISTBOX,
RID_STR_PROPTITLE_CHECKBOX,
RID_STR_PROPTITLE_RADIOBUTTON,
RID_STR_PROPTITLE_GROUPBOX,
RID_STR_PROPTITLE_COMBOBOX,
RID_STR_PROPTITLE_IMAGEBUTTON,
RID_STR_PROPTITLE_FILECONTROL,
RID_STR_PROPTITLE_DATEFIELD,
RID_STR_PROPTITLE_TIMEFIELD,
RID_STR_PROPTITLE_NUMERICFIELD,
RID_STR_PROPTITLE_CURRENCYFIELD,
RID_STR_PROPTITLE_PATTERNFIELD,
RID_STR_PROPTITLE_IMAGECONTROL,
RID_STR_PROPTITLE_FORMATTED,
RID_STR_PROPTITLE_SCROLLBAR,
RID_STR_PROPTITLE_SPINBUTTON,
RID_STR_PROPTITLE_NAVBAR
};
constexpr OUString aImgIds[]
{
RID_SVXBMP_EDITBOX,
RID_SVXBMP_BUTTON,
RID_SVXBMP_FIXEDTEXT,
RID_SVXBMP_LISTBOX,
RID_SVXBMP_CHECKBOX,
RID_SVXBMP_RADIOBUTTON,
RID_SVXBMP_GROUPBOX,
RID_SVXBMP_COMBOBOX,
RID_SVXBMP_IMAGEBUTTON,
RID_SVXBMP_FILECONTROL,
RID_SVXBMP_DATEFIELD,
RID_SVXBMP_TIMEFIELD,
RID_SVXBMP_NUMERICFIELD,
RID_SVXBMP_CURRENCYFIELD,
RID_SVXBMP_PATTERNFIELD,
RID_SVXBMP_IMAGECONTROL,
RID_SVXBMP_FORMATTEDFIELD,
RID_SVXBMP_SCROLLBAR,
RID_SVXBMP_SPINBUTTON,
RID_SVXBMP_NAVIGATIONBAR
};
using namespace css;
using namespace css::uno;
using namespace css::lang;
using namespace css::frame;
namespace {
class ControlMenuController : public svt::PopupMenuControllerBase
{
using svt::PopupMenuControllerBase::disposing;
public:
explicit ControlMenuController( const uno::Reference< uno::XComponentContext >& xContext );
// XServiceInfo
virtual OUString SAL_CALL getImplementationName() override
{
return u"com.sun.star.comp.framework.ControlMenuController"_ustr;
}
virtual sal_Bool SAL_CALL supportsService(OUString const & ServiceName) override
{
return cppu::supportsService(this, ServiceName);
}
virtual css::uno::Sequence<OUString> SAL_CALL getSupportedServiceNames() override
{
return {u"com.sun.star.frame.PopupMenuController"_ustr};
}
// XPopupMenuController
virtual void SAL_CALL updatePopupMenu() override;
// XStatusListener
virtual void SAL_CALL statusChanged( const frame::FeatureStateEvent& Event ) override;
// XMenuListener
virtual void SAL_CALL itemActivated( const awt::MenuEvent& rEvent ) override;
// XEventListener
virtual void SAL_CALL disposing( const lang::EventObject& Source ) override;
private:
// XInitialization
virtual void initializeImpl( std::unique_lock<std::mutex>& rGuard, const uno::Sequence< uno::Any >& aArguments ) override;
class UrlToDispatchMap : public std::unordered_map< OUString,
uno::Reference< frame::XDispatch > >
{
public:
void free()
{
UrlToDispatchMap().swap( *this );// get rid of reserved capacity
}
};
void updateImagesPopupMenu(Reference<awt::XPopupMenu> const& rPopupMenu);
void fillPopupMenu(uno::Reference<awt::XPopupMenu> const& rPopupMenu);
bool m_bShowMenuImages : 1;
UrlToDispatchMap m_aURLToDispatchMap;
};
ControlMenuController::ControlMenuController(const css::uno::Reference< css::uno::XComponentContext >& xContext)
: svt::PopupMenuControllerBase(xContext)
{
const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings();
m_bShowMenuImages = rSettings.GetUseImagesInMenus();
}
// private function
void ControlMenuController::updateImagesPopupMenu(Reference<awt::XPopupMenu> const& rPopupMenu)
{
if (!rPopupMenu)
return;
for (size_t i=0; i < std::size(aCommands); ++i)
{
sal_Int16 nItemId = i + 1;
if (m_bShowMenuImages)
{
Image aImage(StockImage::Yes, aImgIds[i]);
Graphic aGraphic(aImage);
rPopupMenu->setItemImage(nItemId, aGraphic.GetXGraphic(), false);
}
else
rPopupMenu->setItemImage(nItemId, nullptr, false);
}
}
// private function
void ControlMenuController::fillPopupMenu( Reference< css::awt::XPopupMenu > const & rPopupMenu )
{
resetPopupMenu( rPopupMenu );
for (size_t i=0; i < SAL_N_ELEMENTS(aCommands); ++i)
{
sal_Int16 nItemId = i + 1;
OUString sCommand(aCommands[i]);
rPopupMenu->insertItem(nItemId, FwkResId(aLabels[i]), 0, i);
rPopupMenu->setCommand(nItemId, sCommand);
rPopupMenu->enableItem(nItemId, false);
}
updateImagesPopupMenu(rPopupMenu);
rPopupMenu->hideDisabledEntries(true);
}
}
// XEventListener
void SAL_CALL ControlMenuController::disposing( const EventObject& )
{
Reference< css::awt::XMenuListener > xHolder(this);
std::unique_lock aLock( m_aMutex );
m_xFrame.clear();
m_xDispatch.clear();
if ( m_xPopupMenu.is() )
m_xPopupMenu->removeMenuListener( Reference< css::awt::XMenuListener >(this) );
m_xPopupMenu.clear();
}
// XStatusListener
void SAL_CALL ControlMenuController::statusChanged( const FeatureStateEvent& Event )
{
std::unique_lock aLock( m_aMutex );
if (!m_xPopupMenu)
return;
sal_Int16 nItemId = 0;
for (size_t i=0; i < SAL_N_ELEMENTS(aCommands); ++i)
{
if ( Event.FeatureURL.Complete == aCommands[i] )
{
nItemId = i + 1;
break;
}
}
if (!nItemId)
return;
m_xPopupMenu->enableItem(nItemId, Event.IsEnabled);
}
// XMenuListener
void SAL_CALL ControlMenuController::itemActivated( const css::awt::MenuEvent& )
{
std::unique_lock aLock( m_aMutex );
SolarMutexGuard aSolarMutexGuard;
// Check if some modes have changed so we have to update our menu images
const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings();
bool bShowMenuImages = rSettings.GetUseImagesInMenus();
if (bShowMenuImages != m_bShowMenuImages)
{
m_bShowMenuImages = bShowMenuImages;
updateImagesPopupMenu(m_xPopupMenu);
}
}
// XPopupMenuController
void SAL_CALL ControlMenuController::updatePopupMenu()
{
std::unique_lock aLock( m_aMutex );
throwIfDisposed(aLock);
if ( !(m_xFrame.is() && m_xPopupMenu.is()) )
return;
css::util::URL aTargetURL;
Reference< XDispatchProvider > xDispatchProvider( m_xFrame, UNO_QUERY );
fillPopupMenu( m_xPopupMenu );
m_aURLToDispatchMap.free();
for (const OUString& aCommand : aCommands)
{
aTargetURL.Complete = aCommand;
m_xURLTransformer->parseStrict( aTargetURL );
Reference< XDispatch > xDispatch = xDispatchProvider->queryDispatch( aTargetURL, OUString(), 0 );
if ( xDispatch.is() )
{
aLock.unlock(); // the addStatusListener will call back into ::statusChanged
xDispatch->addStatusListener( static_cast< XStatusListener* >(this), aTargetURL );
xDispatch->removeStatusListener( static_cast< XStatusListener* >(this), aTargetURL );
aLock.lock();
m_aURLToDispatchMap.emplace( aTargetURL.Complete, xDispatch );
}
}
}
// XInitialization
void ControlMenuController::initializeImpl( std::unique_lock<std::mutex>& rGuard, const Sequence< Any >& aArguments )
{
svt::PopupMenuControllerBase::initializeImpl(rGuard, aArguments);
m_aBaseURL.clear();
}
extern "C" SAL_DLLPUBLIC_EXPORT css::uno::XInterface *
com_sun_star_comp_framework_ControlMenuController_get_implementation(
css::uno::XComponentContext *context,
css::uno::Sequence<css::uno::Any> const &)
{
return cppu::acquire(new ControlMenuController(context));
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */