/* -*- 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 "services.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include //_________________________________________________________________________________________________________________ // Defines //_________________________________________________________________________________________________________________ using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::frame; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::util; using namespace ::com::sun::star::container; using namespace ::com::sun::star::ui; static const char CONFIGURE_TOOLBARS_CMD[] = "ConfigureDialog"; static const char CONFIGURE_TOOLBARS[] = ".uno:ConfigureDialog"; static const char CMD_COLORBAR[] = ".uno:ColorControl"; static const char CMD_FORMULABAR[] = ".uno:InsertFormula"; static const char CMD_INPUTLINEBAR[] = ".uno:InputLineVisible"; static const char CMD_RESTOREVISIBILITY[] = ".cmd:RestoreVisibility"; static const char ITEM_DESCRIPTOR_RESOURCEURL[] = "ResourceURL"; static const char ITEM_DESCRIPTOR_UINAME[] = "UIName"; static const char STATIC_PRIVATE_TB_RESOURCE[] = "private:resource/toolbar/"; static const char STATIC_CMD_PART[] = ".uno:AvailableToolbars?Toolbar:string="; static const char STATIC_INTERNAL_CMD_PART[] = ".cmd:"; namespace framework { typedef boost::unordered_map< OUString, OUString, OUStringHash, ::std::equal_to< OUString > > ToolbarHashMap; struct ToolBarEntry { OUString aUIName; OUString aCommand; sal_Bool bVisible; sal_Bool bContextSensitive; const CollatorWrapper* pCollatorWrapper; }; sal_Bool CompareToolBarEntry( const ToolBarEntry& aOne, const ToolBarEntry& aTwo ) { sal_Int32 nComp = aOne.pCollatorWrapper->compareString( aOne.aUIName, aTwo.aUIName ); if ( nComp < 0 ) return sal_True; else return sal_False; } Reference< XLayoutManager > getLayoutManagerFromFrame( const Reference< XFrame >& rFrame ) { Reference< XPropertySet > xPropSet( rFrame, UNO_QUERY ); Reference< XLayoutManager > xLayoutManager; try { xPropSet->getPropertyValue("LayoutManager") >>= xLayoutManager; } catch ( const UnknownPropertyException& ) { } return xLayoutManager; } struct ToolBarInfo { OUString aToolBarResName; OUString aToolBarUIName; }; DEFINE_XSERVICEINFO_MULTISERVICE_2 ( ToolbarsMenuController , OWeakObject , SERVICENAME_POPUPMENUCONTROLLER , IMPLEMENTATIONNAME_TOOLBARSMENUCONTROLLER ) DEFINE_INIT_SERVICE ( ToolbarsMenuController, {} ) ToolbarsMenuController::ToolbarsMenuController( const ::com::sun::star::uno::Reference< ::com::sun::star::uno::XComponentContext >& xContext ) : svt::PopupMenuControllerBase( xContext ), m_xContext( xContext ), m_aPropUIName( "UIName" ), m_aPropResourceURL( "ResourceURL" ), m_bModuleIdentified( sal_False ), m_bResetActive( sal_False ), m_aIntlWrapper( xContext, Application::GetSettings().GetLanguageTag() ) { } ToolbarsMenuController::~ToolbarsMenuController() { } void ToolbarsMenuController::addCommand( Reference< css::awt::XPopupMenu >& rPopupMenu, const OUString& rCommandURL, const OUString& rLabel ) { sal_uInt16 nItemId = m_xPopupMenu->getItemCount()+1; OUString aLabel; if ( rLabel.isEmpty() ) aLabel = getUINameFromCommand( rCommandURL ); else aLabel = rLabel; rPopupMenu->insertItem( nItemId, aLabel, 0, nItemId ); rPopupMenu->setCommand( nItemId, rCommandURL ); bool bInternal = rCommandURL.startsWith( STATIC_INTERNAL_CMD_PART ); if ( !bInternal ) { if ( !getDispatchFromCommandURL( rCommandURL ).is() ) m_xPopupMenu->enableItem( nItemId, sal_False ); } SolarMutexGuard aSolarMutexGuard; Image aImage; const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings(); if ( rSettings.GetUseImagesInMenus() ) aImage = GetImageFromURL( m_xFrame, rCommandURL, false ); VCLXPopupMenu* pPopupMenu = (VCLXPopupMenu *)VCLXPopupMenu::GetImplementation( rPopupMenu ); if ( pPopupMenu ) { PopupMenu* pVCLPopupMenu = (PopupMenu *)pPopupMenu->GetMenu(); if ( !!aImage ) pVCLPopupMenu->SetItemImage( nItemId, aImage ); } m_aCommandVector.push_back( rCommandURL ); } Reference< XDispatch > ToolbarsMenuController::getDispatchFromCommandURL( const OUString& rCommandURL ) { URL aTargetURL; Reference< XURLTransformer > xURLTransformer; Reference< XFrame > xFrame; { SolarMutexGuard aSolarMutexGuard; xURLTransformer = m_xURLTransformer; xFrame = m_xFrame; } aTargetURL.Complete = rCommandURL; xURLTransformer->parseStrict( aTargetURL ); Reference< XDispatchProvider > xDispatchProvider( xFrame, UNO_QUERY ); if ( xDispatchProvider.is() ) return xDispatchProvider->queryDispatch( aTargetURL, OUString(), 0 ); else return Reference< XDispatch >(); } // private function OUString ToolbarsMenuController::getUINameFromCommand( const OUString& rCommandURL ) { OUString aLabel; if ( !m_bModuleIdentified ) { try { Reference< XModuleManager2 > xModuleManager = ModuleManager::create( m_xContext ); m_aModuleIdentifier = xModuleManager->identify( m_xFrame ); Reference< XNameAccess > xNameAccess = frame::UICommandDescription::create( m_xContext ); xNameAccess->getByName( m_aModuleIdentifier ) >>= m_xUICommandDescription; } catch ( const Exception& ) { } } if ( m_xUICommandDescription.is() ) { try { Sequence< PropertyValue > aPropSeq; OUString aStr; if ( m_xUICommandDescription->getByName( rCommandURL ) >>= aPropSeq ) { for ( sal_Int32 i = 0; i < aPropSeq.getLength(); i++ ) { if ( aPropSeq[i].Name == "Label" ) { aPropSeq[i].Value >>= aStr; break; } } } aLabel = aStr; } catch ( const Exception& ) { } } return aLabel; } static void fillHashMap( const Sequence< Sequence< ::com::sun::star::beans::PropertyValue > >& rSeqToolBars, ToolbarHashMap& rHashMap ) { for ( sal_Int32 i = 0; i < rSeqToolBars.getLength(); i++ ) { OUString aResourceURL; OUString aUIName; const PropertyValue* pProperties = rSeqToolBars[i].getConstArray(); for ( sal_Int32 j = 0; j < rSeqToolBars[i].getLength(); j++ ) { if ( pProperties[j].Name == ITEM_DESCRIPTOR_RESOURCEURL ) pProperties[j].Value >>= aResourceURL; else if ( pProperties[j].Name == ITEM_DESCRIPTOR_UINAME ) pProperties[j].Value >>= aUIName; } if ( !aResourceURL.isEmpty() && rHashMap.find( aResourceURL ) == rHashMap.end() ) rHashMap.insert( ToolbarHashMap::value_type( aResourceURL, aUIName )); } } // private function Sequence< Sequence< com::sun::star::beans::PropertyValue > > ToolbarsMenuController::getLayoutManagerToolbars( const Reference< ::com::sun::star::frame::XLayoutManager >& rLayoutManager ) { std::vector< ToolBarInfo > aToolBarArray; Sequence< Reference< XUIElement > > aUIElements = rLayoutManager->getElements(); for ( sal_Int32 i = 0; i < aUIElements.getLength(); i++ ) { Reference< XUIElement > xUIElement( aUIElements[i] ); Reference< XPropertySet > xPropSet( aUIElements[i], UNO_QUERY ); if ( xPropSet.is() && xUIElement.is() ) { try { OUString aResName; sal_Int16 nType( -1 ); xPropSet->getPropertyValue("Type") >>= nType; xPropSet->getPropertyValue("ResourceURL") >>= aResName; if (( nType == ::com::sun::star::ui::UIElementType::TOOLBAR ) && !aResName.isEmpty() ) { ToolBarInfo aToolBarInfo; aToolBarInfo.aToolBarResName = aResName; SolarMutexGuard aGuard; Reference< css::awt::XWindow > xWindow( xUIElement->getRealInterface(), UNO_QUERY ); Window* pWindow = VCLUnoHelper::GetWindow( xWindow ); if ( pWindow ) aToolBarInfo.aToolBarUIName = pWindow->GetText(); aToolBarArray.push_back( aToolBarInfo ); } } catch ( const Exception& ) { } } } Sequence< com::sun::star::beans::PropertyValue > aTbSeq( 2 ); aTbSeq[0].Name = m_aPropUIName; aTbSeq[1].Name = m_aPropResourceURL; Sequence< Sequence< com::sun::star::beans::PropertyValue > > aSeq( aToolBarArray.size() ); const sal_uInt32 nCount = aToolBarArray.size(); for ( sal_uInt32 i = 0; i < nCount; i++ ) { aTbSeq[0].Value <<= aToolBarArray[i].aToolBarUIName; aTbSeq[1].Value <<= aToolBarArray[i].aToolBarResName; aSeq[i] = aTbSeq; } return aSeq; } sal_Bool ToolbarsMenuController::isContextSensitiveToolbarNonVisible() { return m_bResetActive; } void ToolbarsMenuController::fillPopupMenu( Reference< css::awt::XPopupMenu >& rPopupMenu ) { if( SvtMiscOptions().DisableUICustomization() ) return; SolarMutexGuard aSolarMutexGuard; resetPopupMenu( rPopupMenu ); m_aCommandVector.clear(); // Retrieve layout manager for additional information OUString aEmptyString; Reference< XLayoutManager > xLayoutManager( getLayoutManagerFromFrame( m_xFrame )); m_bResetActive = sal_False; if ( xLayoutManager.is() ) { ToolbarHashMap aToolbarHashMap; if ( m_xDocCfgMgr.is() ) { Sequence< Sequence< com::sun::star::beans::PropertyValue > > aSeqDocToolBars = m_xDocCfgMgr->getUIElementsInfo( UIElementType::TOOLBAR ); fillHashMap( aSeqDocToolBars, aToolbarHashMap ); } if ( m_xModuleCfgMgr.is() ) { Sequence< Sequence< com::sun::star::beans::PropertyValue > > aSeqToolBars = m_xModuleCfgMgr->getUIElementsInfo( UIElementType::TOOLBAR ); fillHashMap( aSeqToolBars, aToolbarHashMap ); } std::vector< ToolBarEntry > aSortedTbs; OUString aStaticCmdPart( STATIC_CMD_PART ); Sequence< Sequence< com::sun::star::beans::PropertyValue > > aSeqFrameToolBars = getLayoutManagerToolbars( xLayoutManager ); fillHashMap( aSeqFrameToolBars, aToolbarHashMap ); ToolbarHashMap::const_iterator pIter = aToolbarHashMap.begin(); while ( pIter != aToolbarHashMap.end() ) { OUString aUIName = pIter->second; sal_Bool bHideFromMenu( sal_False ); sal_Bool bContextSensitive( sal_False ); sal_Bool bVisible( sal_False ); if ( aUIName.isEmpty() && m_xPersistentWindowState.is() ) { try { Sequence< PropertyValue > aWindowState; Any a( m_xPersistentWindowState->getByName( pIter->first )); if ( a >>= aWindowState ) { for ( sal_Int32 i = 0; i < aWindowState.getLength(); i++ ) { if ( aWindowState[i].Name == WINDOWSTATE_PROPERTY_UINAME ) aWindowState[i].Value >>= aUIName; else if ( aWindowState[i].Name == WINDOWSTATE_PROPERTY_HIDEFROMENU ) aWindowState[i].Value >>= bHideFromMenu; else if ( aWindowState[i].Name == WINDOWSTATE_PROPERTY_CONTEXT ) aWindowState[i].Value >>= bContextSensitive; else if ( aWindowState[i].Name == WINDOWSTATE_PROPERTY_VISIBLE ) aWindowState[i].Value >>= bVisible; } } } catch ( const Exception& ) { } // Check if we have to enable/disable "Reset" menu item if ( bContextSensitive && !bVisible ) m_bResetActive = sal_True; } if ( !aUIName.isEmpty() && !bHideFromMenu ) { ToolBarEntry aTbEntry; aTbEntry.aUIName = aUIName; aTbEntry.aCommand = pIter->first; aTbEntry.bVisible = xLayoutManager->isElementVisible( pIter->first ); aTbEntry.bContextSensitive = bContextSensitive; aTbEntry.pCollatorWrapper = m_aIntlWrapper.getCaseCollator(); aSortedTbs.push_back( aTbEntry ); } ++pIter; } // sort toolbars std::sort( aSortedTbs.begin(), aSortedTbs.end(), CompareToolBarEntry ); sal_Int16 nIndex( 1 ); const sal_uInt32 nCount = aSortedTbs.size(); for ( sal_uInt32 i = 0; i < nCount; i++ ) { sal_uInt16 nItemCount = m_xPopupMenu->getItemCount(); m_xPopupMenu->insertItem( nIndex, aSortedTbs[i].aUIName, css::awt::MenuItemStyle::CHECKABLE, nItemCount ); if ( aSortedTbs[i].bVisible ) m_xPopupMenu->checkItem( nIndex, sal_True ); { SolarMutexGuard aGuard; VCLXPopupMenu* pXPopupMenu = (VCLXPopupMenu *)VCLXMenu::GetImplementation( m_xPopupMenu ); PopupMenu* pVCLPopupMenu = (PopupMenu *)pXPopupMenu->GetMenu(); pVCLPopupMenu->SetUserValue( nIndex, sal_uIntPtr( aSortedTbs[i].bContextSensitive ? 1L : 0L )); } // use VCL popup menu pointer to set vital information that are not part of the awt implementation OUStringBuffer aStrBuf( aStaticCmdPart ); sal_Int32 n = aSortedTbs[i].aCommand.lastIndexOf( '/' ); if (( n > 0 ) && (( n+1 ) < aSortedTbs[i].aCommand.getLength() )) aStrBuf.append( aSortedTbs[i].aCommand.copy( n+1 )); OUString aCmd( aStrBuf.makeStringAndClear() ); // Store complete uno-command so it can also be dispatched. This is necessary to support // the test tool! rPopupMenu->setCommand( nIndex, aCmd ); ++nIndex; } // Create commands for non-toolbars if ( m_aModuleIdentifier == "com.sun.star.text.TextDocument" || m_aModuleIdentifier == "com.sun.star.text.WebDocument" || m_aModuleIdentifier == "com.sun.star.text.GlobalDocument" || m_aModuleIdentifier == "com.sun.star.drawing.DrawingDocument" || m_aModuleIdentifier == "com.sun.star.presentation.PresentationDocument" || m_aModuleIdentifier == "com.sun.star.sheet.SpreadsheetDocument" ) { if ( m_aModuleIdentifier == "com.sun.star.drawing.DrawingDocument" || m_aModuleIdentifier == "com.sun.star.presentation.PresentationDocument" ) addCommand( m_xPopupMenu, OUString( CMD_COLORBAR ), aEmptyString ); else if ( m_aModuleIdentifier == "com.sun.star.sheet.SpreadsheetDocument" ) addCommand( m_xPopupMenu, OUString( CMD_INPUTLINEBAR ), aEmptyString ); else addCommand( m_xPopupMenu, OUString( CMD_FORMULABAR ), aEmptyString ); } sal_Bool bAddCommand( sal_True ); SvtCommandOptions aCmdOptions; OUString aConfigureToolbar( CONFIGURE_TOOLBARS ); if ( aCmdOptions.HasEntries( SvtCommandOptions::CMDOPTION_DISABLED )) { if ( aCmdOptions.Lookup( SvtCommandOptions::CMDOPTION_DISABLED, OUString( CONFIGURE_TOOLBARS_CMD ))) bAddCommand = sal_False; } if ( bAddCommand ) { // Create command for configure if ( m_xPopupMenu->getItemCount() > 0 ) { sal_uInt16 nItemCount = m_xPopupMenu->getItemCount(); m_xPopupMenu->insertSeparator( nItemCount+1 ); } addCommand( m_xPopupMenu, aConfigureToolbar, aEmptyString ); } // Add separator if no configure has been added if ( !bAddCommand ) { // Create command for configure if ( m_xPopupMenu->getItemCount() > 0 ) { sal_uInt16 nItemCount = m_xPopupMenu->getItemCount(); m_xPopupMenu->insertSeparator( nItemCount+1 ); } } OUString aLabelStr(FWK_RESSTR(STR_RESTORE_TOOLBARS)); OUString aRestoreCmd( CMD_RESTOREVISIBILITY ); addCommand( m_xPopupMenu, aRestoreCmd, aLabelStr ); } } // XEventListener void SAL_CALL ToolbarsMenuController::disposing( const EventObject& ) throw ( RuntimeException ) { Reference< css::awt::XMenuListener > xHolder(( OWeakObject *)this, UNO_QUERY ); osl::MutexGuard aLock( m_aMutex ); m_xFrame.clear(); m_xDispatch.clear(); m_xDocCfgMgr.clear(); m_xModuleCfgMgr.clear(); m_xContext.clear(); if ( m_xPopupMenu.is() ) m_xPopupMenu->removeMenuListener( Reference< css::awt::XMenuListener >(( OWeakObject *)this, UNO_QUERY )); m_xPopupMenu.clear(); } // XStatusListener void SAL_CALL ToolbarsMenuController::statusChanged( const FeatureStateEvent& Event ) throw ( RuntimeException ) { OUString aFeatureURL( Event.FeatureURL.Complete ); // All other status events will be processed here sal_Bool bSetCheckmark = sal_False; sal_Bool bCheckmark = sal_False; osl::ClearableMutexGuard aLock( m_aMutex ); Reference< css::awt::XPopupMenu > xPopupMenu( m_xPopupMenu ); aLock.clear(); if ( xPopupMenu.is() ) { SolarMutexGuard aGuard; VCLXPopupMenu* pXPopupMenu = (VCLXPopupMenu *)VCLXMenu::GetImplementation( xPopupMenu ); PopupMenu* pVCLPopupMenu = (PopupMenu *)pXPopupMenu->GetMenu(); for ( sal_uInt16 i = 0; i < pVCLPopupMenu->GetItemCount(); i++ ) { sal_uInt16 nId = pVCLPopupMenu->GetItemId( i ); if ( nId == 0 ) continue; OUString aCmd = pVCLPopupMenu->GetItemCommand( nId ); if ( aCmd == aFeatureURL ) { // Enable/disable item pVCLPopupMenu->EnableItem( nId, Event.IsEnabled ); // Checkmark if ( Event.State >>= bCheckmark ) bSetCheckmark = sal_True; if ( bSetCheckmark ) pVCLPopupMenu->CheckItem( nId, bCheckmark ); else { OUString aItemText; if ( Event.State >>= aItemText ) pVCLPopupMenu->SetItemText( nId, aItemText ); } } } } } // XMenuListener void SAL_CALL ToolbarsMenuController::itemSelected( const css::awt::MenuEvent& rEvent ) throw (RuntimeException) { Reference< css::awt::XPopupMenu > xPopupMenu; Reference< XComponentContext > xContext; Reference< XURLTransformer > xURLTransformer; Reference< XFrame > xFrame; Reference< XNameAccess > xPersistentWindowState; osl::ClearableMutexGuard aLock( m_aMutex ); xPopupMenu = m_xPopupMenu; xContext = m_xContext; xURLTransformer = m_xURLTransformer; xFrame = m_xFrame; xPersistentWindowState = m_xPersistentWindowState; aLock.clear(); if ( xPopupMenu.is() ) { VCLXPopupMenu* pPopupMenu = (VCLXPopupMenu *)VCLXPopupMenu::GetImplementation( xPopupMenu ); if ( pPopupMenu ) { SolarMutexGuard aSolarMutexGuard; PopupMenu* pVCLPopupMenu = (PopupMenu *)pPopupMenu->GetMenu(); OUString aCmd( pVCLPopupMenu->GetItemCommand( rEvent.MenuId )); if ( aCmd.startsWith( STATIC_INTERNAL_CMD_PART ) ) { // Command to restore the visibility of all context sensitive toolbars Reference< XNameReplace > xNameReplace( xPersistentWindowState, UNO_QUERY ); if ( xPersistentWindowState.is() && xNameReplace.is() ) { try { Sequence< OUString > aElementNames = xPersistentWindowState->getElementNames(); sal_Int32 nCount = aElementNames.getLength(); bool bRefreshToolbars( false ); for ( sal_Int32 i = 0; i < nCount; i++ ) { try { OUString aElementName = aElementNames[i]; Sequence< PropertyValue > aWindowState; if ( xPersistentWindowState->getByName( aElementName ) >>= aWindowState ) { sal_Bool bVisible( sal_False ); sal_Bool bContextSensitive( sal_False ); sal_Int32 nVisibleIndex( -1 ); for ( sal_Int32 j = 0; j < aWindowState.getLength(); j++ ) { if ( aWindowState[j].Name == WINDOWSTATE_PROPERTY_VISIBLE ) { aWindowState[j].Value >>= bVisible; nVisibleIndex = j; } else if ( aWindowState[j].Name == WINDOWSTATE_PROPERTY_CONTEXT ) aWindowState[j].Value >>= bContextSensitive; } if ( !bVisible && bContextSensitive && nVisibleIndex >= 0 ) { // Default is: Every context sensitive toolbar is visible aWindowState[nVisibleIndex].Value <<= sal_True; xNameReplace->replaceByName( aElementName, makeAny( aWindowState )); bRefreshToolbars = true; } } } catch ( const NoSuchElementException& ) { } } if ( bRefreshToolbars ) { Reference< XLayoutManager > xLayoutManager( getLayoutManagerFromFrame( xFrame )); if ( xLayoutManager.is() ) { Reference< XPropertySet > xPropSet( xLayoutManager, UNO_QUERY ); if ( xPropSet.is() ) { try { xPropSet->setPropertyValue("RefreshContextToolbarVisibility", makeAny( sal_True )); } catch ( const RuntimeException& ) { } catch ( const Exception& ) { } } } RefreshToolbars( xFrame ); } } catch ( const RuntimeException& ) { throw; } catch ( const Exception& ) { } } } else if ( aCmd.indexOf( STATIC_CMD_PART ) < 0 ) { URL aTargetURL; Sequence aArgs; aTargetURL.Complete = aCmd; xURLTransformer->parseStrict( aTargetURL ); Reference< XDispatchProvider > xDispatchProvider( m_xFrame, UNO_QUERY ); if ( xDispatchProvider.is() ) { Reference< XDispatch > xDispatch = xDispatchProvider->queryDispatch( aTargetURL, OUString(), 0 ); ExecuteInfo* pExecuteInfo = new ExecuteInfo; pExecuteInfo->xDispatch = xDispatch; pExecuteInfo->aTargetURL = aTargetURL; pExecuteInfo->aArgs = aArgs; Application::PostUserEvent( STATIC_LINK(0, ToolbarsMenuController, ExecuteHdl_Impl), pExecuteInfo ); } } else { Reference< XLayoutManager > xLayoutManager( getLayoutManagerFromFrame( xFrame )); if ( xLayoutManager.is() ) { // Extract toolbar name from the combined uno-command. sal_Int32 nIndex = aCmd.indexOf( '=' ); if (( nIndex > 0 ) && (( nIndex+1 ) < aCmd.getLength() )) { OUStringBuffer aBuf( OUString( RTL_CONSTASCII_USTRINGPARAM( STATIC_PRIVATE_TB_RESOURCE) )); aBuf.append( aCmd.copy( nIndex+1 )); sal_Bool bShow( !pVCLPopupMenu->IsItemChecked( rEvent.MenuId )); OUString aToolBarResName( aBuf.makeStringAndClear() ); if ( bShow ) { xLayoutManager->createElement( aToolBarResName ); xLayoutManager->showElement( aToolBarResName ); } else { // closing means: // hide and destroy element xLayoutManager->hideElement( aToolBarResName ); xLayoutManager->destroyElement( aToolBarResName ); } } } } } } } void SAL_CALL ToolbarsMenuController::itemActivated( const css::awt::MenuEvent& ) throw (RuntimeException) { std::vector< OUString > aCmdVector; Reference< XDispatchProvider > xDispatchProvider( m_xFrame, UNO_QUERY ); Reference< XURLTransformer > xURLTransformer( m_xURLTransformer ); { osl::MutexGuard aLock( m_aMutex ); fillPopupMenu( m_xPopupMenu ); aCmdVector = m_aCommandVector; } // Update status for all commands inside our toolbars popup menu const sal_uInt32 nCount = aCmdVector.size(); for ( sal_uInt32 i = 0; i < nCount; i++ ) { bool bInternal = aCmdVector[i].startsWith( STATIC_INTERNAL_CMD_PART ); if ( !bInternal ) { URL aTargetURL; aTargetURL.Complete = aCmdVector[i]; xURLTransformer->parseStrict( aTargetURL ); Reference< XDispatch > xDispatch = xDispatchProvider->queryDispatch( aTargetURL, OUString(), 0 ); if ( xDispatch.is() ) { xDispatch->addStatusListener( (static_cast< XStatusListener* >(this)), aTargetURL ); xDispatch->removeStatusListener( (static_cast< XStatusListener* >(this)), aTargetURL ); } } else if ( aCmdVector[i] == CMD_RESTOREVISIBILITY ) { // Special code to determine the enable/disable state of this command FeatureStateEvent aFeatureStateEvent; aFeatureStateEvent.FeatureURL.Complete = aCmdVector[i]; aFeatureStateEvent.IsEnabled = isContextSensitiveToolbarNonVisible(); statusChanged( aFeatureStateEvent ); } } } // XPopupMenuController void SAL_CALL ToolbarsMenuController::setPopupMenu( const Reference< css::awt::XPopupMenu >& xPopupMenu ) throw ( RuntimeException ) { osl::MutexGuard aLock( m_aMutex ); throwIfDisposed(); if ( m_xFrame.is() && !m_xPopupMenu.is() ) { // Create popup menu on demand SolarMutexGuard aSolarMutexGuard; m_xPopupMenu = xPopupMenu; m_xPopupMenu->addMenuListener( Reference< css::awt::XMenuListener >( (OWeakObject*)this, UNO_QUERY )); fillPopupMenu( m_xPopupMenu ); } } // XInitialization void SAL_CALL ToolbarsMenuController::initialize( const Sequence< Any >& aArguments ) throw ( Exception, RuntimeException ) { osl::MutexGuard aLock( m_aMutex ); sal_Bool bInitalized( m_bInitialized ); if ( !bInitalized ) { svt::PopupMenuControllerBase::initialize(aArguments); if ( m_bInitialized ) { Reference< XModuleManager2 > xModuleManager = ModuleManager::create( m_xContext ); Reference< XNameAccess > xPersistentWindowStateSupplier = ::com::sun::star::ui::WindowStateConfiguration::create( m_xContext ); // Retrieve persistent window state reference for our module OUString aModuleIdentifier; try { aModuleIdentifier = xModuleManager->identify( m_xFrame ); xPersistentWindowStateSupplier->getByName( aModuleIdentifier ) >>= m_xPersistentWindowState; Reference< XModuleUIConfigurationManagerSupplier > xModuleCfgSupplier = ModuleUIConfigurationManagerSupplier::create( m_xContext ); m_xModuleCfgMgr = xModuleCfgSupplier->getUIConfigurationManager( aModuleIdentifier ); Reference< XController > xController = m_xFrame->getController(); Reference< XModel > xModel; if ( xController.is() ) xModel = xController->getModel(); if ( xModel.is() ) { Reference< XUIConfigurationManagerSupplier > xUIConfigurationManagerSupplier( xModel, UNO_QUERY ); if ( xUIConfigurationManagerSupplier.is() ) m_xDocCfgMgr = xUIConfigurationManagerSupplier->getUIConfigurationManager(); } m_aModuleIdentifier = aModuleIdentifier; } catch ( const Exception& ) { } } } } IMPL_STATIC_LINK_NOINSTANCE( ToolbarsMenuController, ExecuteHdl_Impl, ExecuteInfo*, pExecuteInfo ) { 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! if ( pExecuteInfo->xDispatch.is() ) { pExecuteInfo->xDispatch->dispatch( pExecuteInfo->aTargetURL, pExecuteInfo->aArgs ); } } catch ( const Exception& ) { } delete pExecuteInfo; return 0; } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */