/* -*- 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 using namespace ::com::sun::star::awt; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::frame; using namespace ::com::sun::star::frame::status; using namespace ::com::sun::star::util; using namespace ::com::sun::star::container; namespace framework { static bool isEnumCommand( const OUString& rCommand ) { INetURLObject aURL( rCommand ); return ( aURL.GetProtocol() == INetProtocol::Uno ) && ( aURL.GetURLPath().indexOf( '.' ) != -1); } static OUString getEnumCommand( const OUString& rCommand ) { INetURLObject aURL( rCommand ); OUString aEnumCommand; OUString aURLPath = aURL.GetURLPath(); sal_Int32 nIndex = aURLPath.indexOf( '.' ); if (( nIndex > 0 ) && ( nIndex < aURLPath.getLength() )) aEnumCommand = aURLPath.copy( nIndex+1 ); return aEnumCommand; } static OUString getMasterCommand( const OUString& rCommand ) { OUString aMasterCommand( rCommand ); INetURLObject aURL( rCommand ); if ( aURL.GetProtocol() == INetProtocol::Uno ) { sal_Int32 nIndex = aURL.GetURLPath().indexOf( '.' ); if ( nIndex ) { aURL.SetURLPath( aURL.GetURLPath().copy( 0, nIndex ) ); aMasterCommand = aURL.GetMainURL( INetURLObject::DecodeMechanism::NONE ); } } return aMasterCommand; } GenericToolbarController::GenericToolbarController( const Reference< XComponentContext >& rxContext, const Reference< XFrame >& rFrame, ToolBox* pToolbar, sal_uInt16 nID, const OUString& aCommand ) : svt::ToolboxController( rxContext, rFrame, aCommand ) , m_xToolbar( pToolbar ) , m_nID( nID ) , m_bEnumCommand( isEnumCommand( aCommand )) , m_bMadeInvisible( false ) , m_aEnumCommand( getEnumCommand( aCommand )) { if ( m_bEnumCommand ) addStatusListener( getMasterCommand( aCommand ) ); } GenericToolbarController::~GenericToolbarController() { } void SAL_CALL GenericToolbarController::dispose() { SolarMutexGuard aSolarMutexGuard; svt::ToolboxController::dispose(); m_xToolbar.clear(); m_nID = 0; } void SAL_CALL GenericToolbarController::execute( sal_Int16 KeyModifier ) { Reference< XDispatch > xDispatch; OUString aCommandURL; { SolarMutexGuard aSolarMutexGuard; if ( m_bDisposed ) throw DisposedException(); if ( m_bInitialized && m_xFrame.is() && !m_aCommandURL.isEmpty() ) { aCommandURL = m_aCommandURL; URLToDispatchMap::iterator pIter = m_aListenerMap.find( m_aCommandURL ); if ( pIter != m_aListenerMap.end() ) xDispatch = pIter->second; } } if ( xDispatch.is() ) { css::util::URL aTargetURL; Sequence aArgs( 1 ); // Add key modifier to argument list aArgs[0].Name = "KeyModifier"; aArgs[0].Value <<= KeyModifier; aTargetURL.Complete = aCommandURL; if ( m_xUrlTransformer.is() ) m_xUrlTransformer->parseStrict( aTargetURL ); // Execute dispatch asynchronously ExecuteInfo* pExecuteInfo = new ExecuteInfo; pExecuteInfo->xDispatch = xDispatch; pExecuteInfo->aTargetURL = aTargetURL; pExecuteInfo->aArgs = aArgs; Application::PostUserEvent( LINK(nullptr, GenericToolbarController , ExecuteHdl_Impl), pExecuteInfo ); } } void GenericToolbarController::statusChanged( const FeatureStateEvent& Event ) { SolarMutexGuard aSolarMutexGuard; if ( m_bDisposed ) return; if ( m_xToolbar ) { m_xToolbar->EnableItem( m_nID, Event.IsEnabled ); ToolBoxItemBits nItemBits = m_xToolbar->GetItemBits( m_nID ); nItemBits &= ~ToolBoxItemBits::CHECKABLE; TriState eTri = TRISTATE_FALSE; bool bValue; OUString aStrValue; ItemStatus aItemState; Visibility aItemVisibility; ControlCommand aControlCommand; if (( Event.State >>= bValue ) && !m_bEnumCommand ) { // Boolean, treat it as checked/unchecked if ( m_bMadeInvisible ) m_xToolbar->ShowItem( m_nID ); m_xToolbar->CheckItem( m_nID, bValue ); if ( bValue ) eTri = TRISTATE_TRUE; nItemBits |= ToolBoxItemBits::CHECKABLE; } else if ( Event.State >>= aStrValue ) { if ( m_bEnumCommand ) { bValue = aStrValue == m_aEnumCommand; m_xToolbar->CheckItem( m_nID, bValue ); if ( bValue ) eTri = TRISTATE_TRUE; nItemBits |= ToolBoxItemBits::CHECKABLE; } else { // Replacement for place holders if ( aStrValue.startsWith("($1)") ) { aStrValue = FwkResId(STR_UPDATEDOC) + " " + aStrValue.copy( 4 ); } else if ( aStrValue.startsWith("($2)") ) { aStrValue = FwkResId(STR_CLOSEDOC_ANDRETURN) + aStrValue.copy( 4 ); } else if ( aStrValue.startsWith("($3)") ) { aStrValue = FwkResId(STR_SAVECOPYDOC) + aStrValue.copy( 4 ); } m_xToolbar->SetItemText( m_nID, aStrValue ); // tdf#124267 strip mnemonic from tooltip m_xToolbar->SetQuickHelpText(m_nID, aStrValue.replaceFirst("~", "")); } if ( m_bMadeInvisible ) m_xToolbar->ShowItem( m_nID ); } else if (( Event.State >>= aItemState ) && !m_bEnumCommand ) { eTri = TRISTATE_INDET; nItemBits |= ToolBoxItemBits::CHECKABLE; if ( m_bMadeInvisible ) m_xToolbar->ShowItem( m_nID ); } else if ( Event.State >>= aItemVisibility ) { m_xToolbar->ShowItem( m_nID, aItemVisibility.bVisible ); m_bMadeInvisible = !aItemVisibility.bVisible; } else if ( Event.State >>= aControlCommand ) { if (aControlCommand.Command == "SetQuickHelpText") { for (sal_Int32 i = 0; i < aControlCommand.Arguments.getLength(); i++) { if (aControlCommand.Arguments[i].Name == "HelpText") { OUString aHelpText; aControlCommand.Arguments[i].Value >>= aHelpText; m_xToolbar->SetQuickHelpText(m_nID, aHelpText); break; } } } if ( m_bMadeInvisible ) m_xToolbar->ShowItem( m_nID ); } else if ( m_bMadeInvisible ) m_xToolbar->ShowItem( m_nID ); m_xToolbar->SetItemState( m_nID, eTri ); m_xToolbar->SetItemBits( m_nID, nItemBits ); } } IMPL_STATIC_LINK( GenericToolbarController, ExecuteHdl_Impl, void*, p, void ) { ExecuteInfo* pExecuteInfo = static_cast(p); SolarMutexReleaser aReleaser; try { // Asynchronous execution as this can lead to our own destruction! // Framework can recycle our current frame and the layout manager disposes all user interface // elements if a component gets detached from its frame! pExecuteInfo->xDispatch->dispatch( pExecuteInfo->aTargetURL, pExecuteInfo->aArgs ); } catch ( const Exception& ) { } delete pExecuteInfo; } void MenuToolbarController::dispose() { try { if ( m_xMenuManager.is() ) m_xMenuManager->dispose(); } catch( const Exception& ) {} m_xMenuManager.clear(); m_xMenuDesc.clear(); pMenu.disposeAndClear(); } void MenuToolbarController::initialize( const css::uno::Sequence< css::uno::Any >& rArgs ) { ToolboxController::initialize( rArgs ); css::uno::Reference< css::container::XIndexAccess > xMenuContainer; try { css::uno::Reference< css::frame::XController > xController( m_xFrame->getController() ); css::uno::Reference< css::ui::XUIConfigurationManagerSupplier > xSupplier( xController->getModel(), css::uno::UNO_QUERY_THROW ); css::uno::Reference< css::ui::XUIConfigurationManager > xConfigManager( xSupplier->getUIConfigurationManager() ); xMenuContainer.set( xConfigManager->getSettings( m_aCommandURL, false ) ); } catch( const css::uno::Exception& ) {} if ( !xMenuContainer.is() ) { try { css::uno::Reference< css::ui::XModuleUIConfigurationManagerSupplier > xSupplier( css::ui::theModuleUIConfigurationManagerSupplier::get( m_xContext ) ); css::uno::Reference< css::ui::XUIConfigurationManager > xConfigManager( xSupplier->getUIConfigurationManager( m_sModuleName ) ); xMenuContainer.set( xConfigManager->getSettings( m_aCommandURL, false ) ); } catch( const css::uno::Exception& ) {} } if ( xMenuContainer.is() && xMenuContainer->getCount() ) { Sequence< PropertyValue > aProps; // drop down menu info is currently the first ( and only ) menu in the menusettings container xMenuContainer->getByIndex(0) >>= aProps; for ( const auto& aProp : std::as_const(aProps) ) { if ( aProp.Name == "ItemDescriptorContainer" ) { aProp.Value >>= m_xMenuDesc; break; } } ToolBox* pToolBox = nullptr; sal_uInt16 nId = 0; if ( getToolboxId( nId, &pToolBox ) ) pToolBox->SetItemBits( nId, pToolBox->GetItemBits( nId ) | ToolBoxItemBits::DROPDOWNONLY ); } } Reference< XWindow > SAL_CALL MenuToolbarController::createPopupWindow() { if ( !pMenu ) { pMenu = VclPtr::Create(); css::uno::Reference< css::frame::XDispatchProvider > xDispatchProvider( m_xFrame, css::uno::UNO_QUERY ); sal_uInt16 m_nMenuId = 1; MenuBarManager::FillMenu( m_nMenuId, pMenu, m_sModuleName, m_xMenuDesc, xDispatchProvider ); m_xMenuManager.set( new MenuBarManager( m_xContext, m_xFrame, m_xUrlTransformer, xDispatchProvider, m_sModuleName, pMenu, false, false ) ); } ToolBox* pToolBox = nullptr; sal_uInt16 nId = 0; if ( !getToolboxId( nId, &pToolBox ) ) return nullptr; pToolBox->SetItemDown( m_nToolBoxId, true ); pMenu->Execute( pToolBox, pToolBox->GetItemRect( nId ), PopupMenuFlags::ExecuteDown ); pToolBox->SetItemDown( m_nToolBoxId, false ); return nullptr; } } // namespace /* vim:set shiftwidth=4 softtabstop=4 expandtab: */