/* -*- 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 #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "sfxtypes.hxx" #include "virtmenu.hxx" #include #include #include #include #include #include #include #include #include #include "../doc/doc.hrc" #include 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 = nullptr; 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(nullptr), pSubMenu(nullptr), b_ShowStrings(bShowStrings) { } // Constructor for array SfxMenuControl::SfxMenuControl(): pOwnMenu(nullptr), pSubMenu(nullptr), b_ShowStrings(false) { } SfxMenuControl::SfxMenuControl(sal_uInt16 nSlotId, SfxBindings& rBindings): SfxControllerItem(nSlotId, rBindings), pOwnMenu(nullptr), pSubMenu(nullptr), 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 != nullptr, "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 ? nullptr != 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() ) { SAL_INFO("sfx", "Title of menu item changed - please check if this needs correction!"); } return; } bool bCheck = false; if ( dynamic_cast< const SfxBoolItem *>( pState ) != nullptr ) { // BoolItem for check DBG_ASSERT( GetId() < SID_OBJECTMENU0 || GetId() > SID_OBJECTMENU_LAST, "SfxBoolItem not allowed for SID_OBJECTMENUx" ); bCheck = static_cast(pState)->GetValue(); Menu* pMenu = pOwnMenu->GetSVMenu(); pMenu->SetItemBits( GetId() , pMenu->GetItemBits( GetId() ) | MenuItemBits::CHECKABLE); } else if ( dynamic_cast< const SfxEnumItemInterface *>( pState ) != nullptr && static_cast(pState)->HasBoolValue() ) { // Treat EnumItem as Bool DBG_ASSERT( GetId() < SID_OBJECTMENU0 || GetId() > SID_OBJECTMENU_LAST, "SfxEnumItem not allowed for SID_OBJECTMENUx" ); bCheck = static_cast(pState)->GetBoolValue(); Menu* pMenu = pOwnMenu->GetSVMenu(); pMenu->SetItemBits( GetId() , pMenu->GetItemBits( GetId() ) | MenuItemBits::CHECKABLE); } else if ( ( b_ShowStrings || bIsObjMenu ) && dynamic_cast< const SfxStringItem *>( pState ) != nullptr ) { // Get MenuText from SfxStringItem OUString aStr( static_cast(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, SfxMenuCtrlFactory( SfxMenuControl::CreateImpl, typeid(SfxStringItem), nSlotId ) ); } void SfxMenuControl::RegisterMenuControl(SfxModule* pMod, const SfxMenuCtrlFactory& rFact) { SfxGetpApp()->RegisterMenuControl_Impl( pMod, rFact ); } SfxMenuControl* SfxMenuControl::CreateControl( sal_uInt16 nId, Menu &rMenu, SfxBindings &rBindings ) { const std::type_info* aSlotType = SFX_SLOTPOOL().GetSlotType(nId); if ( aSlotType ) { SfxApplication *pApp = SfxGetpApp(); SfxDispatcher *pDisp = rBindings.GetDispatcher_Impl(); SfxModule *pMod = pDisp ? SfxModule::GetActiveModule( pDisp->GetFrame() ) :nullptr; if ( pMod ) { SfxMenuCtrlFactArr_Impl *pFactories = pMod->GetMenuCtrlFactories_Impl(); if ( pFactories ) { SfxMenuCtrlFactArr_Impl &rFactories = *pFactories; for ( size_t 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 ( size_t 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 nullptr; } 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(); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */