Use m_xUrlTransformer, and drop the now unneeded m_xContext.is() check. Here is the history of this: - commitef8f9cfb26
(CWS docking1) introduced this code in ToolboxController::execute. - commita2bbbc5130
(CWS docking3) copy-pasted this into GenericToolbarController::execute. - commit0fc90c8a06
(CWS insight02) changed the original code, but not the copy-pasted one. - commitd7afd8e128
(CWS toolbars2) That CWS didn't have the above change yet, and copy-pasted the old code to yet another place (GenericToolboxController::execute). Change-Id: I7f4e8dfb52b96ea5285ea7b2bb3083847dd84932
364 lines
12 KiB
C++
364 lines
12 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 "uielement/generictoolbarcontroller.hxx"
|
|
|
|
#include <com/sun/star/util/URLTransformer.hpp>
|
|
#include <com/sun/star/util/XURLTransformer.hpp>
|
|
#include <com/sun/star/frame/XDispatchProvider.hpp>
|
|
#include <com/sun/star/beans/PropertyValue.hpp>
|
|
#include <com/sun/star/lang/DisposedException.hpp>
|
|
#include <com/sun/star/frame/status/ItemStatus.hpp>
|
|
#include <com/sun/star/frame/status/ItemState.hpp>
|
|
#include <com/sun/star/frame/status/Visibility.hpp>
|
|
#include <com/sun/star/ui/XUIConfigurationManagerSupplier.hpp>
|
|
#include <com/sun/star/ui/theModuleUIConfigurationManagerSupplier.hpp>
|
|
|
|
#include <comphelper/processfactory.hxx>
|
|
#include <svtools/toolboxcontroller.hxx>
|
|
#include <osl/mutex.hxx>
|
|
#include <vcl/svapp.hxx>
|
|
#include <vcl/mnemonic.hxx>
|
|
#include <tools/urlobj.hxx>
|
|
#include <classes/resource.hrc>
|
|
#include <classes/fwkresid.hxx>
|
|
#include <framework/menuconfiguration.hxx>
|
|
#include <uielement/menubarmanager.hxx>
|
|
|
|
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_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()
|
|
{
|
|
SolarMutexGuard aSolarMutexGuard;
|
|
|
|
svt::ToolboxController::dispose();
|
|
|
|
m_pToolbar.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<PropertyValue> 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_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 )
|
|
{
|
|
bValue = aStrValue == m_aEnumCommand;
|
|
|
|
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 += " " + 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<ExecuteInfo*>(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 : 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<PopupMenu>::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: */
|