/* -*- 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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include static const char* aCommands[] = { ".uno:ConvertToEdit", ".uno:ConvertToButton", ".uno:ConvertToFixed", ".uno:ConvertToList", ".uno:ConvertToCheckBox", ".uno:ConvertToRadio", ".uno:ConvertToGroup", ".uno:ConvertToCombo", ".uno:ConvertToImageBtn", ".uno:ConvertToFileControl", ".uno:ConvertToDate", ".uno:ConvertToTime", ".uno:ConvertToNumeric", ".uno:ConvertToCurrency", ".uno:ConvertToPattern", ".uno:ConvertToImageControl", ".uno:ConvertToFormatted", ".uno:ConvertToScrollBar", ".uno:ConvertToSpinButton", ".uno:ConvertToNavigationBar" }; 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; using namespace css::beans; 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 "com.sun.star.comp.framework.ControlMenuController"; } virtual sal_Bool SAL_CALL supportsService(OUString const & ServiceName) override { return cppu::supportsService(this, ServiceName); } virtual css::uno::Sequence SAL_CALL getSupportedServiceNames() override { return {"com.sun.star.frame.PopupMenuController"}; } // 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& 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 const& rPopupMenu); void fillPopupMenu(uno::Reference 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 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(OUString::createFromAscii(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.equalsAscii( 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 char* aCommand : aCommands) { aTargetURL.Complete = OUString::createFromAscii( 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& 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 const &) { return cppu::acquire(new ControlMenuController(context)); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */