/* -*- 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 "uielement/generictoolbarcontroller.hxx" #include #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 ); if (( aURL.GetProtocol() == INetProtocol::Uno ) && ( aURL.GetURLPath().indexOf( '.' ) != -1)) return true; return false; } 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::NO_DECODE ); } } 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_pToolbar( 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() throw ( RuntimeException, std::exception ) { SolarMutexGuard aSolarMutexGuard; svt::ToolboxController::dispose(); m_pToolbar.clear(); m_nID = 0; } void SAL_CALL GenericToolbarController::execute( sal_Int16 KeyModifier ) throw ( RuntimeException, std::exception ) { Reference< XDispatch > xDispatch; Reference< XURLTransformer > xURLTransformer; OUString aCommandURL; { SolarMutexGuard aSolarMutexGuard; if ( m_bDisposed ) throw DisposedException(); if ( m_bInitialized && m_xFrame.is() && !m_aCommandURL.isEmpty() ) { xURLTransformer = URLTransformer::create(m_xContext); aCommandURL = m_aCommandURL; URLToDispatchMap::iterator pIter = m_aListenerMap.find( m_aCommandURL ); if ( pIter != m_aListenerMap.end() ) xDispatch = pIter->second; } } if ( xDispatch.is() && xURLTransformer.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; 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 ) throw ( RuntimeException, std::exception ) { SolarMutexGuard aSolarMutexGuard; if ( m_bDisposed ) return; if ( m_pToolbar ) { m_pToolbar->EnableItem( m_nID, Event.IsEnabled ); ToolBoxItemBits nItemBits = m_pToolbar->GetItemBits( m_nID ); nItemBits &= ~ToolBoxItemBits::CHECKABLE; TriState eTri = TRISTATE_FALSE; bool bValue; OUString aStrValue; ItemStatus aItemState; Visibility aItemVisibility; if (( Event.State >>= bValue ) && !m_bEnumCommand ) { // Boolean, treat it as checked/unchecked if ( m_bMadeInvisible ) m_pToolbar->ShowItem( m_nID ); m_pToolbar->CheckItem( m_nID, bValue ); if ( bValue ) eTri = TRISTATE_TRUE; nItemBits |= ToolBoxItemBits::CHECKABLE; } else if ( Event.State >>= aStrValue ) { if ( m_bEnumCommand ) { if ( aStrValue == m_aEnumCommand ) bValue = true; else bValue = false; m_pToolbar->CheckItem( m_nID, bValue ); if ( bValue ) eTri = TRISTATE_TRUE; nItemBits |= ToolBoxItemBits::CHECKABLE; } else { // Replacement for place holders if ( aStrValue.startsWith("($1)") ) { OUString aTmp(FwkResId(STR_UPDATEDOC)); aTmp += " "; aTmp += aStrValue.copy( 4 ); aStrValue = aTmp; } else if ( aStrValue.startsWith("($2)") ) { OUString aTmp(FWK_RESSTR(STR_CLOSEDOC_ANDRETURN)); aTmp += aStrValue.copy( 4 ); aStrValue = aTmp; } else if ( aStrValue.startsWith("($3)") ) { OUString aTmp(FWK_RESSTR(STR_SAVECOPYDOC)); aTmp += aStrValue.copy( 4 ); aStrValue = aTmp; } m_pToolbar->SetItemText( m_nID, aStrValue ); m_pToolbar->SetQuickHelpText( m_nID, aStrValue ); } if ( m_bMadeInvisible ) m_pToolbar->ShowItem( m_nID ); } else if (( Event.State >>= aItemState ) && !m_bEnumCommand ) { eTri = TRISTATE_INDET; nItemBits |= ToolBoxItemBits::CHECKABLE; if ( m_bMadeInvisible ) m_pToolbar->ShowItem( m_nID ); } else if ( Event.State >>= aItemVisibility ) { m_pToolbar->ShowItem( m_nID, aItemVisibility.bVisible ); m_bMadeInvisible = !aItemVisibility.bVisible; } else if ( m_bMadeInvisible ) m_pToolbar->ShowItem( m_nID ); m_pToolbar->SetItemState( m_nID, eTri ); m_pToolbar->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; } MenuToolbarController::MenuToolbarController( const Reference< XComponentContext >& rxContext, const Reference< XFrame >& rFrame, ToolBox* pToolBar, sal_uInt16 nID, const OUString& aCommand, const OUString& aModuleIdentifier, const Reference< XIndexAccess >& xMenuDesc ) : GenericToolbarController( rxContext, rFrame, pToolBar, nID, aCommand ), m_xMenuDesc( xMenuDesc ), pMenu( nullptr ), m_aModuleIdentifier( aModuleIdentifier ) { } MenuToolbarController::~MenuToolbarController() { try { if ( m_xMenuManager.is() ) m_xMenuManager->dispose(); } catch( const Exception& ) {} if ( pMenu ) { pMenu.disposeAndClear(); } } class Toolbarmenu : public ::PopupMenu { public: Toolbarmenu(); virtual ~Toolbarmenu() override; virtual void dispose() override; }; Toolbarmenu::Toolbarmenu() { SAL_INFO("fwk.uielement", "constructing Toolbarmenu " << this); } Toolbarmenu::~Toolbarmenu() { disposeOnce(); } void Toolbarmenu::dispose() { SAL_INFO("fwk.uielement", "destructing Toolbarmenu " << this); ::PopupMenu::dispose(); } void SAL_CALL MenuToolbarController::click() throw (RuntimeException, std::exception) { createPopupWindow(); } Reference< XWindow > SAL_CALL MenuToolbarController::createPopupWindow() throw (css::uno::RuntimeException, std::exception) { if ( !pMenu ) { Reference< XDispatchProvider > xDispatch; Reference< XURLTransformer > xURLTransformer = URLTransformer::create( m_xContext ); pMenu = VclPtr::Create(); m_xMenuManager.set( new MenuBarManager( m_xContext, m_xFrame, xURLTransformer, xDispatch, m_aModuleIdentifier, pMenu, true, true, false ) ); if (m_xMenuManager.is()) { MenuBarManager& rMgr = dynamic_cast(*m_xMenuManager.get()); rMgr.SetItemContainer(m_xMenuDesc); } } if ( !pMenu || !m_pToolbar ) return nullptr; OSL_ENSURE ( pMenu->GetItemCount(), "Empty PopupMenu!" ); ::Rectangle aRect( m_pToolbar->GetItemRect( m_nID ) ); pMenu->Execute( m_pToolbar, aRect, PopupMenuFlags::ExecuteDown ); return nullptr; } } // namespace /* vim:set shiftwidth=4 softtabstop=4 expandtab: */