Some uses of untyped Link<> (returning sal_IntPtr) are hard to update to typed versions, but upcoming changes to loplugin:implicitboolconversion would flag these uses of sal_False/True, so just explicitly cast them to sal_IntPtr for now. Change-Id: I654e555e85faba0b30178c978e7d50fc7240b004
468 lines
15 KiB
C++
468 lines
15 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 .
|
|
*/
|
|
|
|
|
|
#ifdef SOLARIS
|
|
#include <ctime>
|
|
#endif
|
|
|
|
#include <string>
|
|
#include <com/sun/star/uno/Reference.h>
|
|
#include <com/sun/star/beans/PropertyValue.hpp>
|
|
#include <com/sun/star/frame/Desktop.hpp>
|
|
#include <com/sun/star/frame/XDispatch.hpp>
|
|
#include <com/sun/star/frame/XDispatchProvider.hpp>
|
|
#include <com/sun/star/frame/XFrame.hpp>
|
|
#include <com/sun/star/util/URL.hpp>
|
|
#include <com/sun/star/util/URLTransformer.hpp>
|
|
#include <com/sun/star/util/XURLTransformer.hpp>
|
|
#include <comphelper/processfactory.hxx>
|
|
#include <tools/urlobj.hxx>
|
|
#include <svl/eitem.hxx>
|
|
#include <svl/stritem.hxx>
|
|
#include <svtools/imagemgr.hxx>
|
|
#include <svtools/menuoptions.hxx>
|
|
#include <framework/menuconfiguration.hxx>
|
|
|
|
#include <sfx2/app.hxx>
|
|
#include <sfx2/sfx.hrc>
|
|
#include <sfx2/msgpool.hxx>
|
|
#include <sfx2/msg.hxx>
|
|
#include <sfx2/bindings.hxx>
|
|
#include <sfx2/dispatch.hxx>
|
|
#include "sfxtypes.hxx"
|
|
#include "virtmenu.hxx"
|
|
#include <ctrlfactoryimpl.hxx>
|
|
#include <sfx2/mnuitem.hxx>
|
|
#include <sfx2/tbxctrl.hxx>
|
|
#include <sfx2/module.hxx>
|
|
#include <sfx2/unoctitm.hxx>
|
|
#include <sfx2/viewfrm.hxx>
|
|
#include <sfx2/imgmgr.hxx>
|
|
#include <sfx2/imagemgr.hxx>
|
|
#include <sfx2/sfxresid.hxx>
|
|
#include "../doc/doc.hrc"
|
|
#include <vcl/settings.hxx>
|
|
|
|
using namespace ::com::sun::star::uno;
|
|
using namespace ::com::sun::star::frame;
|
|
using namespace ::com::sun::star::beans;
|
|
using namespace ::com::sun::star::util;
|
|
|
|
|
|
|
|
// binds the instance to the specified id and assignes the title
|
|
|
|
void SfxMenuControl::Bind(
|
|
SfxVirtualMenu* pOwn,
|
|
sal_uInt16 nSlotId,
|
|
const OUString& rTitle,
|
|
SfxBindings &rBindings )
|
|
{
|
|
aTitle = rTitle;
|
|
pOwnMenu = pOwn;
|
|
pSubMenu = 0;
|
|
if ( pOwn )
|
|
SfxControllerItem::Bind(nSlotId, &rBindings);
|
|
else
|
|
SetId( nSlotId );
|
|
|
|
#ifdef DBG_UTIL
|
|
CheckConfigure_Impl(SfxSlotMode::MENUCONFIG);
|
|
#endif
|
|
}
|
|
|
|
|
|
|
|
|
|
// binds the item to the specified menu and assignes the title
|
|
|
|
void SfxMenuControl::Bind(
|
|
SfxVirtualMenu* pOwn,
|
|
sal_uInt16 nSlotId,
|
|
SfxVirtualMenu& rMenu,
|
|
const OUString& rTitle,
|
|
SfxBindings &rBindings )
|
|
{
|
|
SetId( nSlotId );
|
|
SetBindings(rBindings);
|
|
pOwnMenu = pOwn;
|
|
pSubMenu = &rMenu;
|
|
aTitle = rTitle;
|
|
}
|
|
|
|
|
|
|
|
// Constructor for explicit registration
|
|
|
|
SfxMenuControl::SfxMenuControl( bool bShowStrings )
|
|
: pOwnMenu(0),
|
|
pSubMenu(0),
|
|
b_ShowStrings(bShowStrings)
|
|
{
|
|
}
|
|
|
|
|
|
|
|
// Constructor for array
|
|
SfxMenuControl::SfxMenuControl():
|
|
pOwnMenu(0),
|
|
pSubMenu(0),
|
|
b_ShowStrings(false)
|
|
{
|
|
}
|
|
|
|
|
|
|
|
SfxMenuControl::SfxMenuControl(sal_uInt16 nSlotId, SfxBindings& rBindings):
|
|
SfxControllerItem(nSlotId, rBindings),
|
|
pOwnMenu(0),
|
|
pSubMenu(0),
|
|
b_ShowStrings(false)
|
|
{
|
|
// This constructor should make it possible already during the design
|
|
// to fall back to the bindings, but can as always be bound later.
|
|
// The usefulness of this is for example if a StatusForwarder should
|
|
// be created in the constructor of a derived class.
|
|
UnBind();
|
|
}
|
|
|
|
|
|
|
|
|
|
SfxMenuControl::~SfxMenuControl()
|
|
{
|
|
delete pSubMenu;
|
|
}
|
|
|
|
|
|
|
|
// changes the state in the virtual menu
|
|
|
|
void SfxMenuControl::StateChanged
|
|
(
|
|
sal_uInt16 nSID,
|
|
SfxItemState eState,
|
|
const SfxPoolItem* pState
|
|
)
|
|
{
|
|
(void)nSID;
|
|
DBG_ASSERT( nSID == GetId(), "strange SID" );
|
|
DBG_ASSERT( pOwnMenu != 0, "setting state to dangling SfxMenuControl" );
|
|
|
|
bool bIsObjMenu =
|
|
GetId() >= SID_OBJECTMENU0 && GetId() < SID_OBJECTMENU_LAST;
|
|
|
|
// Fix inclusion of enabled/disabled-Flag
|
|
|
|
#ifdef UNIX
|
|
if (nSID == SID_PASTE)
|
|
pOwnMenu->EnableItem( GetId(), true );
|
|
else
|
|
#endif
|
|
pOwnMenu->EnableItem( GetId(), bIsObjMenu
|
|
? 0 != pOwnMenu->GetSVMenu()->GetPopupMenu( GetId() )
|
|
: eState != SfxItemState::DISABLED );
|
|
|
|
if ( eState != SfxItemState::DEFAULT )
|
|
{
|
|
// check only for non-Object Menus
|
|
if ( !bIsObjMenu )
|
|
pOwnMenu->CheckItem( GetId(), false );
|
|
|
|
if ( pOwnMenu->GetSVMenu()->GetItemText( GetId() ) != GetTitle() )
|
|
{
|
|
DBG_WARNING("Title of menu item changed - please check if this needs correction!");
|
|
}
|
|
return;
|
|
}
|
|
|
|
bool bCheck = false;
|
|
if ( pState->ISA(SfxBoolItem) )
|
|
{
|
|
// BoolItem for check
|
|
DBG_ASSERT( GetId() < SID_OBJECTMENU0 || GetId() > SID_OBJECTMENU_LAST,
|
|
"SfxBoolItem not allowed for SID_OBJECTMENUx" );
|
|
bCheck = static_cast<const SfxBoolItem*>(pState)->GetValue();
|
|
Menu* pMenu = pOwnMenu->GetSVMenu();
|
|
pMenu->SetItemBits( GetId() , pMenu->GetItemBits( GetId() ) | MenuItemBits::CHECKABLE);
|
|
}
|
|
else if ( pState->ISA(SfxEnumItemInterface) &&
|
|
static_cast<const SfxEnumItemInterface *>(pState)->HasBoolValue() )
|
|
{
|
|
// Treat EnumItem as Bool
|
|
DBG_ASSERT( GetId() < SID_OBJECTMENU0 || GetId() > SID_OBJECTMENU_LAST,
|
|
"SfxEnumItem not allowed for SID_OBJECTMENUx" );
|
|
bCheck = static_cast<const SfxEnumItemInterface *>(pState)->GetBoolValue();
|
|
Menu* pMenu = pOwnMenu->GetSVMenu();
|
|
pMenu->SetItemBits( GetId() , pMenu->GetItemBits( GetId() ) | MenuItemBits::CHECKABLE);
|
|
}
|
|
else if ( ( b_ShowStrings || bIsObjMenu ) && pState->ISA(SfxStringItem) )
|
|
{
|
|
// Get MenuText from SfxStringItem
|
|
OUString aStr( static_cast<const SfxStringItem*>(pState)->GetValue() );
|
|
if ( aStr.startsWith("($1)") )
|
|
{
|
|
OUString aEntry(SfxResId(STR_UPDATEDOC).toString());
|
|
aEntry += " ";
|
|
aEntry += aStr.copy(4);
|
|
aStr = aEntry;
|
|
}
|
|
else if ( aStr.startsWith("($2)") )
|
|
{
|
|
OUString aEntry(SfxResId(STR_CLOSEDOC_ANDRETURN).toString());
|
|
aEntry += aStr.copy(4);
|
|
aStr = aEntry;
|
|
}
|
|
|
|
pOwnMenu->SetItemText( GetId(), aStr );
|
|
}
|
|
|
|
pOwnMenu->CheckItem( GetId(), bCheck );
|
|
}
|
|
|
|
|
|
|
|
SfxMenuControl* SfxMenuControl::CreateImpl( sal_uInt16 /*nId*/, Menu& /*rMenu*/, SfxBindings& /*rBindings*/ )
|
|
{
|
|
return new SfxMenuControl( true );
|
|
}
|
|
|
|
void SfxMenuControl::RegisterControl( sal_uInt16 nSlotId, SfxModule *pMod )
|
|
{
|
|
RegisterMenuControl( pMod, new SfxMenuCtrlFactory(
|
|
SfxMenuControl::CreateImpl, TYPE(SfxStringItem), nSlotId ) );
|
|
}
|
|
|
|
|
|
void SfxMenuControl::RegisterMenuControl(SfxModule* pMod, SfxMenuCtrlFactory* pFact)
|
|
{
|
|
SfxGetpApp()->RegisterMenuControl_Impl( pMod, pFact );
|
|
}
|
|
|
|
SfxMenuControl* SfxMenuControl::CreateControl( sal_uInt16 nId, Menu &rMenu, SfxBindings &rBindings )
|
|
{
|
|
TypeId aSlotType = SFX_SLOTPOOL().GetSlotType(nId);
|
|
if ( aSlotType )
|
|
{
|
|
SfxApplication *pApp = SfxGetpApp();
|
|
SfxDispatcher *pDisp = rBindings.GetDispatcher_Impl();
|
|
SfxModule *pMod = pDisp ? SfxModule::GetActiveModule( pDisp->GetFrame() ) :0;
|
|
if ( pMod )
|
|
{
|
|
SfxMenuCtrlFactArr_Impl *pFactories = pMod->GetMenuCtrlFactories_Impl();
|
|
if ( pFactories )
|
|
{
|
|
SfxMenuCtrlFactArr_Impl &rFactories = *pFactories;
|
|
for ( sal_uInt16 nFactory = 0; nFactory < rFactories.size(); ++nFactory )
|
|
if ( rFactories[nFactory].nTypeId == aSlotType &&
|
|
( ( rFactories[nFactory].nSlotId == 0 ) ||
|
|
( rFactories[nFactory].nSlotId == nId) ) )
|
|
return rFactories[nFactory].pCtor( nId, rMenu, rBindings );
|
|
}
|
|
}
|
|
|
|
SfxMenuCtrlFactArr_Impl &rFactories = pApp->GetMenuCtrlFactories_Impl();
|
|
|
|
for ( sal_uInt16 nFactory = 0; nFactory < rFactories.size(); ++nFactory )
|
|
if ( rFactories[nFactory].nTypeId == aSlotType &&
|
|
( ( rFactories[nFactory].nSlotId == 0 ) ||
|
|
( rFactories[nFactory].nSlotId == nId) ) )
|
|
return rFactories[nFactory].pCtor( nId, rMenu, rBindings );
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
PopupMenu* SfxMenuControl::GetPopup () const
|
|
{
|
|
if (GetPopupMenu())
|
|
return static_cast<PopupMenu*>(GetPopupMenu()->GetSVMenu());
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
sal_IntPtr Select_Impl( void* pHdl, void* pVoid );
|
|
|
|
SfxAppMenuControl_Impl::SfxAppMenuControl_Impl(
|
|
sal_uInt16 nPos, Menu& rMenu, SfxBindings& rBindings )
|
|
: SfxMenuControl( nPos, rBindings ), pMenu(0)
|
|
{
|
|
// Determine the current background color setting for menus
|
|
const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings();
|
|
m_sIconTheme = rSettings.DetermineIconTheme();
|
|
m_bShowMenuImages = rSettings.GetUseImagesInMenus();
|
|
|
|
::framework::MenuConfiguration aConf( ::comphelper::getProcessComponentContext() );
|
|
Reference<com::sun::star::frame::XFrame> aXFrame( GetBindings().GetDispatcher_Impl()->GetFrame()->GetFrame().GetFrameInterface() );
|
|
pMenu = aConf.CreateBookmarkMenu( aXFrame, GetId() == SID_NEWDOCDIRECT ? OUString(BOOKMARK_NEWMENU) : OUString(BOOKMARK_WIZARDMENU) );
|
|
if( pMenu )
|
|
{
|
|
pMenu->SetSelectHdl( Link<>( &(this->GetBindings()), Select_Impl ) );
|
|
pMenu->SetActivateHdl( LINK(this, SfxAppMenuControl_Impl, Activate) );
|
|
rMenu.SetPopupMenu( nPos, pMenu );
|
|
}
|
|
}
|
|
|
|
SfxAppMenuControl_Impl::~SfxAppMenuControl_Impl()
|
|
{
|
|
delete pMenu;
|
|
}
|
|
|
|
IMPL_LINK_TYPED( SfxAppMenuControl_Impl, Activate, Menu *, pActMenu, bool )
|
|
{
|
|
if ( pActMenu )
|
|
{
|
|
const StyleSettings& rSettings = Application::GetSettings().GetStyleSettings();
|
|
OUString sIconTheme = rSettings.DetermineIconTheme();
|
|
bool bShowMenuImages = rSettings.GetUseImagesInMenus();
|
|
|
|
if (( sIconTheme != m_sIconTheme ) ||
|
|
( bShowMenuImages != m_bShowMenuImages ))
|
|
{
|
|
m_sIconTheme = sIconTheme;
|
|
m_bShowMenuImages = bShowMenuImages;
|
|
|
|
sal_uInt16 nCount = pActMenu->GetItemCount();
|
|
for ( sal_uInt16 nSVPos = 0; nSVPos < nCount; nSVPos++ )
|
|
{
|
|
sal_uInt16 nItemId = pActMenu->GetItemId( nSVPos );
|
|
if ( pActMenu->GetItemType( nSVPos ) != MenuItemType::SEPARATOR )
|
|
{
|
|
if ( bShowMenuImages )
|
|
{
|
|
bool bImageSet = false;
|
|
OUString aImageId;
|
|
::framework::MenuAttributes* pMenuAttributes =
|
|
reinterpret_cast< ::framework::MenuAttributes*>(pMenu->GetUserValue( nItemId ));
|
|
|
|
if ( pMenuAttributes )
|
|
aImageId = pMenuAttributes->aImageId; // Retrieve image id from menu attributes
|
|
|
|
if ( aImageId.getLength() > 0 )
|
|
{
|
|
Reference< ::com::sun::star::frame::XFrame > xFrame;
|
|
Image aImage = GetImage( xFrame, aImageId, false );
|
|
if ( !!aImage )
|
|
{
|
|
bImageSet = true;
|
|
pActMenu->SetItemImage( nItemId, aImage );
|
|
}
|
|
}
|
|
|
|
OUString aCmd( pActMenu->GetItemCommand( nItemId ) );
|
|
if ( !bImageSet && !aCmd.isEmpty() )
|
|
{
|
|
Image aImage = SvFileInformationManager::GetImage(
|
|
INetURLObject(aCmd), false );
|
|
if ( !!aImage )
|
|
pActMenu->SetItemImage( nItemId, aImage );
|
|
}
|
|
}
|
|
else
|
|
pActMenu->SetItemImage( nItemId, Image() );
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
SfxUnoMenuControl* SfxMenuControl::CreateControl( const OUString& rCmd,
|
|
sal_uInt16 nId, Menu& rMenu, const OUString& sItemText,
|
|
SfxBindings& rBindings, SfxVirtualMenu* pVirt)
|
|
{
|
|
return new SfxUnoMenuControl( rCmd, nId, rMenu, sItemText, rBindings, pVirt);
|
|
}
|
|
|
|
SfxUnoMenuControl::SfxUnoMenuControl(
|
|
const OUString& rCmd, sal_uInt16 nSlotId, Menu& /*rMenu*/,
|
|
const OUString& rItemText,
|
|
SfxBindings& rBindings, SfxVirtualMenu* pVirt)
|
|
: SfxMenuControl( nSlotId, rBindings )
|
|
{
|
|
Bind( pVirt, nSlotId, rItemText, rBindings);
|
|
UnBind();
|
|
pUnoCtrl = new SfxUnoControllerItem( this, rBindings, rCmd );
|
|
pUnoCtrl->acquire();
|
|
pUnoCtrl->GetNewDispatch();
|
|
}
|
|
|
|
SfxUnoMenuControl::~SfxUnoMenuControl()
|
|
{
|
|
pUnoCtrl->UnBind();
|
|
pUnoCtrl->release();
|
|
}
|
|
|
|
sal_IntPtr Select_Impl( void* /*pHdl*/, void* pVoid )
|
|
{
|
|
Menu* pMenu = static_cast<Menu*>(pVoid);
|
|
OUString aURL( pMenu->GetItemCommand( pMenu->GetCurItemId() ) );
|
|
|
|
if( aURL.isEmpty() )
|
|
return 0;
|
|
|
|
Reference < ::com::sun::star::frame::XDesktop2 > xDesktop =
|
|
::com::sun::star::frame::Desktop::create( ::comphelper::getProcessComponentContext() );
|
|
|
|
URL aTargetURL;
|
|
aTargetURL.Complete = aURL;
|
|
Reference < XURLTransformer > xTrans( URLTransformer::create( ::comphelper::getProcessComponentContext() ) );
|
|
xTrans->parseStrict( aTargetURL );
|
|
|
|
Reference < XDispatch > xDisp;
|
|
if (aTargetURL.Protocol == "slot:")
|
|
xDisp = xDesktop->queryDispatch( aTargetURL, OUString(), 0 );
|
|
else
|
|
{
|
|
OUString aTargetFrame( "_blank" );
|
|
::framework::MenuAttributes* pMenuAttributes =
|
|
reinterpret_cast< ::framework::MenuAttributes*>(pMenu->GetUserValue( pMenu->GetCurItemId() ));
|
|
|
|
if ( pMenuAttributes )
|
|
aTargetFrame = pMenuAttributes->aTargetFrame;
|
|
|
|
xDisp = xDesktop->queryDispatch( aTargetURL, aTargetFrame , 0 );
|
|
}
|
|
|
|
if ( xDisp.is() )
|
|
{
|
|
SfxAppMenuControl_Impl::ExecuteInfo* pExecuteInfo = new SfxAppMenuControl_Impl::ExecuteInfo;
|
|
pExecuteInfo->xDispatch = xDisp;
|
|
pExecuteInfo->aTargetURL = aTargetURL;
|
|
pExecuteInfo->aArgs = Sequence< PropertyValue >();
|
|
Application::PostUserEvent( LINK( 0, SfxAppMenuControl_Impl, ExecuteHdl_Impl), pExecuteInfo );
|
|
}
|
|
|
|
return sal_IntPtr(true);
|
|
}
|
|
|
|
IMPL_STATIC_LINK_NOINSTANCE( SfxAppMenuControl_Impl, ExecuteHdl_Impl, ExecuteInfo*, pExecuteInfo )
|
|
{
|
|
pExecuteInfo->xDispatch->dispatch( pExecuteInfo->aTargetURL, pExecuteInfo->aArgs );
|
|
delete pExecuteInfo;
|
|
return 0;
|
|
}
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|