/* -*- 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 "cfgutil.hxx" #include "cfg.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "helpid.hrc" #include #include #include #include #include #include #include #include "cuires.hrc" #include "bitmaps.hlst" #include #include #include #include #include #include #include #include "svtools/treelistentry.hxx" #include #include #include "dialmgr.hxx" #include #include #include #include #include #include #include #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::script; using namespace ::com::sun::star::frame; using namespace ::com::sun::star::document; SfxStylesInfo_Impl::SfxStylesInfo_Impl() {} void SfxStylesInfo_Impl::init(const OUString& rModuleName, const css::uno::Reference< css::frame::XModel >& xModel) { m_aModuleName = rModuleName; m_xDoc = xModel; } static const char CMDURL_STYLEPROT_ONLY[] = ".uno:StyleApply?"; static const char CMDURL_SPART_ONLY [] = "Style:string="; static const char CMDURL_FPART_ONLY [] = "FamilyName:string="; static const char STYLEPROP_UINAME[] = "DisplayName"; OUString SfxStylesInfo_Impl::generateCommand(const OUString& sFamily, const OUString& sStyle) { return ".uno:StyleApply?Style:string=" + sStyle + "&FamilyName:string=" + sFamily; } bool SfxStylesInfo_Impl::parseStyleCommand(SfxStyleInfo_Impl& aStyle) { static const sal_Int32 LEN_STYLEPROT = strlen(CMDURL_STYLEPROT_ONLY); static const sal_Int32 LEN_SPART = strlen(CMDURL_SPART_ONLY); static const sal_Int32 LEN_FPART = strlen(CMDURL_FPART_ONLY); if (!aStyle.sCommand.startsWith(CMDURL_STYLEPROT_ONLY)) return false; aStyle.sFamily.clear(); aStyle.sStyle.clear(); sal_Int32 nCmdLen = aStyle.sCommand.getLength(); OUString sCmdArgs = aStyle.sCommand.copy(LEN_STYLEPROT, nCmdLen-LEN_STYLEPROT); sal_Int32 i = sCmdArgs.indexOf('&'); if (i<0) return false; OUString sArg = sCmdArgs.copy(0, i); if (sArg.startsWith(CMDURL_SPART_ONLY)) aStyle.sStyle = sArg.copy(LEN_SPART, sArg.getLength()-LEN_SPART); else if (sArg.startsWith(CMDURL_FPART_ONLY)) aStyle.sFamily = sArg.copy(LEN_FPART, sArg.getLength()-LEN_FPART); sArg = sCmdArgs.copy(i+1, sCmdArgs.getLength()-i-1); if (sArg.startsWith(CMDURL_SPART_ONLY)) aStyle.sStyle = sArg.copy(LEN_SPART, sArg.getLength()-LEN_SPART); else if (sArg.startsWith(CMDURL_FPART_ONLY)) aStyle.sFamily = sArg.copy(LEN_FPART, sArg.getLength()-LEN_FPART); return !(aStyle.sFamily.isEmpty() || aStyle.sStyle.isEmpty()); } void SfxStylesInfo_Impl::getLabel4Style(SfxStyleInfo_Impl& aStyle) { try { css::uno::Reference< css::style::XStyleFamiliesSupplier > xModel(m_xDoc, css::uno::UNO_QUERY); css::uno::Reference< css::container::XNameAccess > xFamilies; if (xModel.is()) xFamilies = xModel->getStyleFamilies(); css::uno::Reference< css::container::XNameAccess > xStyleSet; if (xFamilies.is()) xFamilies->getByName(aStyle.sFamily) >>= xStyleSet; css::uno::Reference< css::beans::XPropertySet > xStyle; if (xStyleSet.is()) xStyleSet->getByName(aStyle.sStyle) >>= xStyle; aStyle.sLabel.clear(); if (xStyle.is()) xStyle->getPropertyValue(STYLEPROP_UINAME) >>= aStyle.sLabel; } catch(const css::uno::RuntimeException&) { throw; } catch(const css::uno::Exception&) { aStyle.sLabel.clear(); } if (aStyle.sLabel.isEmpty()) { aStyle.sLabel = aStyle.sCommand; } } std::vector< SfxStyleInfo_Impl > SfxStylesInfo_Impl::getStyleFamilies() { // It's an optional interface! css::uno::Reference< css::style::XStyleFamiliesSupplier > xModel(m_xDoc, css::uno::UNO_QUERY); if (!xModel.is()) return std::vector< SfxStyleInfo_Impl >(); css::uno::Reference< css::container::XNameAccess > xCont = xModel->getStyleFamilies(); css::uno::Sequence< OUString > lFamilyNames = xCont->getElementNames(); std::vector< SfxStyleInfo_Impl > lFamilies; for (const auto& aFamily : lFamilyNames) { if ((aFamily == "CellStyles" && m_aModuleName != "com.sun.star.sheet.SpreadsheetDocument") || aFamily == "cell" || aFamily == "table" || aFamily == "Default") continue; SfxStyleInfo_Impl aFamilyInfo; aFamilyInfo.sFamily = aFamily; try { css::uno::Reference< css::beans::XPropertySet > xFamilyInfo; xCont->getByName(aFamilyInfo.sFamily) >>= xFamilyInfo; if (!xFamilyInfo.is()) { // TODO_AS currently there is no support for an UIName property .. use internal family name instead aFamilyInfo.sLabel = aFamilyInfo.sFamily; } else xFamilyInfo->getPropertyValue(STYLEPROP_UINAME) >>= aFamilyInfo.sLabel; } catch(const css::uno::RuntimeException&) { throw; } catch(const css::uno::Exception&) { return std::vector< SfxStyleInfo_Impl >(); } lFamilies.push_back(aFamilyInfo); } return lFamilies; } std::vector< SfxStyleInfo_Impl > SfxStylesInfo_Impl::getStyles(const OUString& sFamily) { css::uno::Sequence< OUString > lStyleNames; css::uno::Reference< css::style::XStyleFamiliesSupplier > xModel(m_xDoc, css::uno::UNO_QUERY_THROW); css::uno::Reference< css::container::XNameAccess > xFamilies = xModel->getStyleFamilies(); css::uno::Reference< css::container::XNameAccess > xStyleSet; try { xFamilies->getByName(sFamily) >>= xStyleSet; lStyleNames = xStyleSet->getElementNames(); } catch(const css::uno::RuntimeException&) { throw; } catch(const css::uno::Exception&) { return std::vector< SfxStyleInfo_Impl >(); } std::vector< SfxStyleInfo_Impl > lStyles; sal_Int32 c = lStyleNames.getLength(); sal_Int32 i = 0; for (i=0; i xStyle; xStyleSet->getByName(aStyleInfo.sStyle) >>= xStyle; if (!xStyle.is()) continue; xStyle->getPropertyValue("DisplayName") >>= aStyleInfo.sLabel; } catch(const css::uno::RuntimeException&) { throw; } catch(const css::uno::Exception&) { continue; } lStyles.push_back(aStyleInfo); } return lStyles; } SfxConfigFunctionListBox::SfxConfigFunctionListBox(vcl::Window* pParent, WinBits nStyle) : SvTreeListBox( pParent, nStyle ) , pStylesInfo( nullptr ) { SetStyle( GetStyle() | WB_CLIPCHILDREN | WB_HSCROLL | WB_SORT ); GetModel()->SetSortMode( SortAscending ); } VCL_BUILDER_FACTORY_CONSTRUCTOR(SfxConfigFunctionListBox, WB_TABSTOP) SfxConfigFunctionListBox::~SfxConfigFunctionListBox() { disposeOnce(); } void SfxConfigFunctionListBox::dispose() { ClearAll(); SvTreeListBox::dispose(); } void SfxConfigFunctionListBox::MouseMove( const MouseEvent& ) { } void SfxConfigFunctionListBox::ClearAll() /* Description Deletes all entries in the FunctionListBox, all UserData and all possibly existing MacroInfo. */ { sal_uInt16 nCount = aArr.size(); for ( sal_uInt16 i=0; inKind == SfxCfgKind::FUNCTION_SCRIPT ) { OUString* pScriptURI = static_cast(pData->pObject); delete pScriptURI; } if ( pData->nKind == SfxCfgKind::GROUP_SCRIPTCONTAINER ) { XInterface* xi = static_cast(pData->pObject); if (xi != nullptr) { xi->release(); } } } aArr.clear(); Clear(); } OUString SfxConfigFunctionListBox::GetSelectedScriptURI() { SvTreeListEntry *pEntry = FirstSelected(); if ( pEntry ) { SfxGroupInfo_Impl *pData = static_cast(pEntry->GetUserData()); if ( pData && ( pData->nKind == SfxCfgKind::FUNCTION_SCRIPT ) ) return *static_cast(pData->pObject); } return OUString(); } OUString SfxConfigFunctionListBox::GetHelpText() { SvTreeListEntry *pEntry = FirstSelected(); if ( pEntry ) { SfxGroupInfo_Impl *pData = static_cast(pEntry->GetUserData()); if ( pData ) { if ( pData->nKind == SfxCfgKind::FUNCTION_SLOT ) { return Application::GetHelp()->GetHelpText( pData->sCommand, this ); } else if ( pData->nKind == SfxCfgKind::FUNCTION_SCRIPT ) { return pData->sHelpText; } } } return OUString(); } OUString SfxConfigFunctionListBox::GetCurCommand() { SvTreeListEntry *pEntry = FirstSelected(); if (!pEntry) return OUString(); SfxGroupInfo_Impl *pData = static_cast(pEntry->GetUserData()); if (!pData) return OUString(); return pData->sCommand; } OUString SfxConfigFunctionListBox::GetCurLabel() { SvTreeListEntry *pEntry = FirstSelected(); if (!pEntry) return OUString(); SfxGroupInfo_Impl *pData = static_cast(pEntry->GetUserData()); if (!pData) return OUString(); if (!pData->sLabel.isEmpty()) return pData->sLabel; return pData->sCommand; } void SfxConfigFunctionListBox::SetStylesInfo(SfxStylesInfo_Impl* pStyles) { pStylesInfo = pStyles; } struct SvxConfigGroupBoxResource_Impl { Image m_hdImage; Image m_libImage; Image m_macImage; Image m_docImage; OUString m_sMyMacros; OUString m_sProdMacros; OUString m_sMacros; OUString m_sDlgMacros; OUString m_aStrGroupStyles; Image m_collapsedImage; Image m_expandedImage; SvxConfigGroupBoxResource_Impl(); }; SvxConfigGroupBoxResource_Impl::SvxConfigGroupBoxResource_Impl() : m_hdImage(BitmapEx(RID_CUIBMP_HARDDISK)), m_libImage(BitmapEx(RID_CUIBMP_LIB)), m_macImage(BitmapEx(RID_CUIBMP_MACRO)), m_docImage(BitmapEx(RID_CUIBMP_DOC)), m_sMyMacros(CuiResId(RID_SVXSTR_MYMACROS)), m_sProdMacros(CuiResId(RID_SVXSTR_PRODMACROS)), m_sMacros(CuiResId(RID_SVXSTR_BASICMACROS)), m_sDlgMacros(CuiResId(RID_SVXSTR_PRODMACROS)), m_aStrGroupStyles(CuiResId(RID_SVXSTR_GROUP_STYLES)), m_collapsedImage(BitmapEx(RID_CUIBMP_COLLAPSED)), m_expandedImage(BitmapEx(RID_CUIBMP_EXPANDED)) { } SfxConfigGroupListBox::SfxConfigGroupListBox(vcl::Window* pParent, WinBits nStyle) : SvTreeListBox(pParent, nStyle) , xImp(new SvxConfigGroupBoxResource_Impl()) , pFunctionListBox(nullptr) , m_pImageProvider(nullptr) , pStylesInfo(nullptr) { SetStyle( GetStyle() | WB_CLIPCHILDREN | WB_HSCROLL | WB_HASBUTTONS | WB_HASLINES | WB_HASLINESATROOT | WB_HASBUTTONSATROOT ); SetNodeBitmaps(xImp->m_collapsedImage, xImp->m_expandedImage); } VCL_BUILDER_FACTORY_CONSTRUCTOR(SfxConfigGroupListBox, WB_TABSTOP) SfxConfigGroupListBox::~SfxConfigGroupListBox() { disposeOnce(); } void SfxConfigGroupListBox::dispose() { ClearAll(); pFunctionListBox.clear(); SvTreeListBox::dispose(); } void SfxConfigGroupListBox::ClearAll() { sal_uInt16 nCount = aArr.size(); for ( sal_uInt16 i=0; inKind == SfxCfgKind::GROUP_SCRIPTCONTAINER) { XInterface* xi = static_cast(pData->pObject); if (xi != nullptr) { xi->release(); } } } aArr.clear(); Clear(); } void SfxConfigGroupListBox::SetStylesInfo(SfxStylesInfo_Impl* pStyles) { pStylesInfo = pStyles; } void SfxConfigGroupListBox::InitModule() { try { css::uno::Reference< css::frame::XDispatchInformationProvider > xProvider(m_xFrame, css::uno::UNO_QUERY_THROW); css::uno::Sequence< sal_Int16 > lGroups = xProvider->getSupportedCommandGroups(); sal_Int32 c1 = lGroups.getLength(); sal_Int32 i1 = 0; if ( c1 ) { // Add All Commands category SvTreeListEntry* pEntry = InsertEntry( CuiResId(RID_SVXSTR_ALLFUNCTIONS) ); aArr.push_back( o3tl::make_unique( SfxCfgKind::GROUP_ALLFUNCTIONS, 0 ) ); pEntry->SetUserData(aArr.back().get()); } for (i1=0; i1getByName(sGroupID) >>= sGroupName; if (sGroupName.isEmpty()) continue; } catch(const css::container::NoSuchElementException&) { continue; } SvTreeListEntry* pEntry = InsertEntry(sGroupName); aArr.push_back( o3tl::make_unique( SfxCfgKind::GROUP_FUNCTION, rGroupID ) ); pEntry->SetUserData(aArr.back().get()); } } catch(const css::uno::RuntimeException&) { throw; } catch(const css::uno::Exception&) {} } namespace { /** examines a component whether it supports XEmbeddedScripts, or provides access to such a component by implementing XScriptInvocationContext. @return the model which supports the embedded scripts, or if it cannot find such a model */ Reference< XModel > lcl_getDocumentWithScripts_throw( const Reference< XInterface >& _rxComponent ) { Reference< XEmbeddedScripts > xScripts( _rxComponent, UNO_QUERY ); if ( !xScripts.is() ) { Reference< XScriptInvocationContext > xContext( _rxComponent, UNO_QUERY ); if ( xContext.is() ) xScripts.set( xContext->getScriptContainer(), UNO_QUERY ); } return Reference< XModel >( xScripts, UNO_QUERY ); } Reference< XModel > lcl_getScriptableDocument_nothrow( const Reference< XFrame >& _rxFrame ) { Reference< XModel > xDocument; // examine our associated frame try { OSL_ENSURE( _rxFrame.is(), "lcl_getScriptableDocument_nothrow: you need to pass a frame to this dialog/tab page!" ); if ( _rxFrame.is() ) { // first try the model in the frame Reference< XController > xController( _rxFrame->getController(), UNO_SET_THROW ); xDocument = lcl_getDocumentWithScripts_throw( xController->getModel() ); if ( !xDocument.is() ) { // if there is no suitable document in the frame, try the controller xDocument = lcl_getDocumentWithScripts_throw( _rxFrame->getController() ); } } } catch( const Exception& ) { } return xDocument; } } void SfxConfigGroupListBox::FillScriptList(const css::uno::Reference< css::script::browse::XBrowseNode >& xRootNode, SvTreeListEntry* pParentEntry, bool bCheapChildrenOnDemand) { try { if ( xRootNode->hasChildNodes() ) { Sequence< Reference< browse::XBrowseNode > > children = xRootNode->getChildNodes(); bool bIsRootNode = false; OUString user("user"); OUString share("share"); if ( xRootNode->getName() == "Root" ) { bIsRootNode = true; } //To mimic current starbasic behaviour we //need to make sure that only the current document //is displayed in the config tree. Tests below //set the bDisplay flag to FALSE if the current //node is a first level child of the Root and is NOT //either the current document, user or share OUString currentDocTitle; Reference< XModel > xDocument( lcl_getScriptableDocument_nothrow( m_xFrame ) ); if ( xDocument.is() ) { currentDocTitle = ::comphelper::DocumentInfo::getDocumentTitle( xDocument ); } for ( sal_Int32 n = 0; n < children.getLength(); ++n ) { Reference< browse::XBrowseNode >& theChild = children[n]; bool bDisplay = true; OUString uiName = theChild->getName(); if ( bIsRootNode ) { if ( ! ((theChild->getName().equals( user ) || theChild->getName().equals( share ) || theChild->getName().equals( currentDocTitle ) ) ) ) { bDisplay=false; } else { if ( uiName.equals( user ) ) { uiName = xImp->m_sMyMacros; } else if ( uiName.equals( share ) ) { uiName = xImp->m_sProdMacros; } } } if (children[n]->getType() != browse::BrowseNodeTypes::SCRIPT && bDisplay ) { // We call acquire on the XBrowseNode so that it does not // get autodestructed and become invalid when accessed later. theChild->acquire(); Image aImage = GetImage( theChild, m_xContext, bIsRootNode ); SvTreeListEntry* pNewEntry = InsertEntry( uiName, pParentEntry ); SetExpandedEntryBmp( pNewEntry, aImage ); SetCollapsedEntryBmp( pNewEntry, aImage ); aArr.push_back( o3tl::make_unique(SfxCfgKind::GROUP_SCRIPTCONTAINER, 0, static_cast( theChild.get()))); pNewEntry->SetUserData( aArr.back().get() ); if ( !bCheapChildrenOnDemand && children[n]->hasChildNodes() ) { Sequence< Reference< browse::XBrowseNode > > grandchildren = children[n]->getChildNodes(); for ( sal_Int32 m = 0; m < grandchildren.getLength(); ++m ) { if ( grandchildren[m]->getType() == browse::BrowseNodeTypes::CONTAINER ) { pNewEntry->EnableChildrenOnDemand(); m = grandchildren.getLength(); } } } else { /* i30923 - Would be nice if there was a better * way to determine if a basic lib had children * without having to ask for them (which forces * the library to be loaded */ pNewEntry->EnableChildrenOnDemand(); } } } } } catch (RuntimeException&) { // do nothing, the entry will not be displayed in the UI } } void SfxConfigGroupListBox::FillFunctionsList(const css::uno::Sequence& xCommands) { for (const auto & rInfo : xCommands) { OUString sUIName = MapCommand2UIName(rInfo.Command); Image aImage; if ( m_pImageProvider ) aImage = m_pImageProvider->GetImage( rInfo.Command ); SvTreeListEntry* pFuncEntry = nullptr; if ( !!aImage ) pFuncEntry = pFunctionListBox->InsertEntry( sUIName, aImage, aImage ); else pFuncEntry = pFunctionListBox->InsertEntry(sUIName ); aArr.push_back( o3tl::make_unique( SfxCfgKind::FUNCTION_SLOT, 0 ) ); SfxGroupInfo_Impl* pGrpInfo = aArr.back().get(); pGrpInfo->sCommand = rInfo.Command; pGrpInfo->sLabel = sUIName; pFuncEntry->SetUserData(pGrpInfo); } } void SfxConfigGroupListBox::Init(const css::uno::Reference< css::uno::XComponentContext >& xContext, const css::uno::Reference< css::frame::XFrame >& xFrame, const OUString& sModuleLongName, bool bEventMode) { SetUpdateMode(false); ClearAll(); // Remove all old entries from treelist box m_xContext = xContext; m_xFrame = xFrame; if( bEventMode ) { m_sModuleLongName = sModuleLongName; m_xGlobalCategoryInfo = css::ui::theUICategoryDescription::get( m_xContext ); m_xModuleCategoryInfo.set(m_xGlobalCategoryInfo->getByName(m_sModuleLongName), css::uno::UNO_QUERY_THROW); m_xUICmdDescription = css::frame::theUICommandDescription::get( m_xContext ); InitModule(); } SAL_INFO("cui.customize", "** ** About to initialise SF Scripts"); // Add Scripting Framework entries Reference< browse::XBrowseNode > rootNode; try { Reference< browse::XBrowseNodeFactory > xFac = browse::theBrowseNodeFactory::get( m_xContext ); rootNode.set( xFac->createView( browse::BrowseNodeFactoryViewTypes::MACROSELECTOR ) ); } catch( Exception& e ) { SAL_INFO("cui.customize", "Caught some exception whilst retrieving browse nodes from factory... Exception: " << e.Message); // TODO exception handling } if ( rootNode.is() ) { if ( bEventMode ) { //We call acquire on the XBrowseNode so that it does not //get autodestructed and become invalid when accessed later. rootNode->acquire(); aArr.push_back( o3tl::make_unique( SfxCfgKind::GROUP_SCRIPTCONTAINER, 0, static_cast(rootNode.get()))); OUString aTitle(xImp->m_sDlgMacros); SvTreeListEntry *pNewEntry = InsertEntry( aTitle ); pNewEntry->SetUserData( aArr.back().get() ); pNewEntry->EnableChildrenOnDemand(); } else { //We are only showing scripts not slot APIs so skip //Root node and show location nodes FillScriptList(rootNode, nullptr, false); } } // add styles if ( bEventMode ) { OUString sStyle(xImp->m_aStrGroupStyles); SvTreeListEntry *pEntry = InsertEntry( sStyle ); aArr.push_back( o3tl::make_unique( SfxCfgKind::GROUP_STYLES, 0, nullptr ) ); // TODO last parameter should contain user data pEntry->SetUserData( aArr.back().get() ); pEntry->EnableChildrenOnDemand(); } MakeVisible( GetEntry( nullptr,0 ) ); SetUpdateMode( true ); } Image SfxConfigGroupListBox::GetImage( const Reference< browse::XBrowseNode >& node, Reference< XComponentContext > const & xCtx, bool bIsRootNode ) { Image aImage; if ( bIsRootNode ) { OUString user("user"); OUString share("share"); if (node->getName().equals( user ) || node->getName().equals(share ) ) { aImage = xImp->m_hdImage; } else { OUString factoryURL; OUString nodeName = node->getName(); Reference xDocumentModel = getDocumentModel(xCtx, nodeName ); if ( xDocumentModel.is() ) { Reference< frame::XModuleManager2 > xModuleManager( frame::ModuleManager::create(xCtx) ); // get the long name of the document: OUString appModule( xModuleManager->identify( xDocumentModel ) ); Sequence moduleDescr; Any aAny = xModuleManager->getByName(appModule); if( !( aAny >>= moduleDescr ) ) { throw RuntimeException("SFTreeListBox::Init: failed to get PropertyValue"); } beans::PropertyValue const * pmoduleDescr = moduleDescr.getConstArray(); for ( sal_Int32 pos = moduleDescr.getLength(); pos--; ) { if ( pmoduleDescr[ pos ].Name == "ooSetupFactoryEmptyDocumentURL" ) { pmoduleDescr[ pos ].Value >>= factoryURL; SAL_INFO("cui.customize", "factory url for doc images is " << factoryURL); break; } } } if( !factoryURL.isEmpty() ) { aImage = SvFileInformationManager::GetFileImage( INetURLObject(factoryURL) ); } else { aImage = xImp->m_docImage; } } } else { if( node->getType() == browse::BrowseNodeTypes::SCRIPT ) aImage = xImp->m_macImage; else aImage = xImp->m_libImage; } return aImage; } Reference< XInterface > SfxConfigGroupListBox::getDocumentModel( Reference< XComponentContext > const & xCtx, OUString& docName ) { Reference< XInterface > xModel; Reference< frame::XDesktop2 > desktop = frame::Desktop::create( xCtx ); Reference< container::XEnumerationAccess > componentsAccess = desktop->getComponents(); Reference< container::XEnumeration > components = componentsAccess->createEnumeration(); while (components->hasMoreElements()) { Reference< frame::XModel > model( components->nextElement(), UNO_QUERY ); if ( model.is() ) { OUString sTdocUrl = ::comphelper::DocumentInfo::getDocumentTitle( model ); if( sTdocUrl.equals( docName ) ) { xModel = model; break; } } } return xModel; } OUString SfxConfigGroupListBox::MapCommand2UIName(const OUString& sCommand) { OUString sUIName; try { css::uno::Reference< css::container::XNameAccess > xModuleConf; m_xUICmdDescription->getByName(m_sModuleLongName) >>= xModuleConf; if (xModuleConf.is()) { ::comphelper::SequenceAsHashMap lProps(xModuleConf->getByName(sCommand)); sUIName = lProps.getUnpackedValueOrDefault("Name", OUString()); } } catch(const css::uno::RuntimeException&) { throw; } catch(css::uno::Exception&) { sUIName.clear(); } // fallback for missing UINames !? if (sUIName.isEmpty()) { sUIName = sCommand; } return sUIName; } void SfxConfigGroupListBox::GroupSelected() /* Description A function group or a basic module has been selected. All functions/macros are displayed in the functionlistbox. */ { SvTreeListEntry *pEntry = FirstSelected(); SfxGroupInfo_Impl *pInfo = static_cast(pEntry->GetUserData()); pFunctionListBox->SetUpdateMode(false); pFunctionListBox->ClearAll(); if ( pInfo->nKind != SfxCfgKind::GROUP_FUNCTION && pInfo->nKind != SfxCfgKind::GROUP_ALLFUNCTIONS && pInfo->nKind != SfxCfgKind::GROUP_SCRIPTCONTAINER && pInfo->nKind != SfxCfgKind::GROUP_STYLES ) { pFunctionListBox->SetUpdateMode(true); return; } switch ( pInfo->nKind ) { case SfxCfgKind::GROUP_ALLFUNCTIONS: { css::uno::Reference< css::frame::XDispatchInformationProvider > xProvider( m_xFrame, UNO_QUERY ); SvTreeListEntry *pCurrEntry = First(); while( pCurrEntry ) { SfxGroupInfo_Impl *pCurrentInfo = static_cast(pCurrEntry->GetUserData()); if (pCurrentInfo->nKind == SfxCfgKind::GROUP_FUNCTION) { css::uno::Sequence< css::frame::DispatchInformation > lCommands; try { lCommands = xProvider->getConfigurableDispatchInformation( pCurrentInfo->nUniqueID ); FillFunctionsList( lCommands ); } catch ( container::NoSuchElementException& ) { } } pCurrEntry = Next( pCurrEntry ); } break; } case SfxCfgKind::GROUP_FUNCTION : { sal_uInt16 nGroup = pInfo->nUniqueID; css::uno::Reference< css::frame::XDispatchInformationProvider > xProvider (m_xFrame, css::uno::UNO_QUERY_THROW); css::uno::Sequence< css::frame::DispatchInformation > lCommands = xProvider->getConfigurableDispatchInformation(nGroup); FillFunctionsList( lCommands ); break; } case SfxCfgKind::GROUP_SCRIPTCONTAINER: { if ( !GetChildCount( pEntry ) ) { Reference< browse::XBrowseNode > rootNode( static_cast< browse::XBrowseNode* >( pInfo->pObject ) ) ; try { if ( rootNode->hasChildNodes() ) { Sequence< Reference< browse::XBrowseNode > > children = rootNode->getChildNodes(); for ( sal_Int32 n = 0; n < children.getLength(); ++n ) { if (children[n]->getType() == browse::BrowseNodeTypes::SCRIPT) { OUString uri, description; Reference < beans::XPropertySet >xPropSet( children[n], UNO_QUERY ); if (!xPropSet.is()) { continue; } Any value = xPropSet->getPropertyValue("URI"); value >>= uri; try { value = xPropSet->getPropertyValue("Description"); value >>= description; } catch (Exception &) { // do nothing, the description will be empty } OUString* pScriptURI = new OUString( uri ); Image aImage = GetImage( children[n], Reference< XComponentContext >(), false ); SvTreeListEntry* pNewEntry = pFunctionListBox->InsertEntry( children[n]->getName() ); pFunctionListBox->SetExpandedEntryBmp( pNewEntry, aImage ); pFunctionListBox->SetCollapsedEntryBmp(pNewEntry, aImage ); pFunctionListBox->aArr.push_back( o3tl::make_unique( SfxCfgKind::FUNCTION_SCRIPT, 0, pScriptURI )); pFunctionListBox->aArr.back()->sCommand = uri; pFunctionListBox->aArr.back()->sLabel = children[n]->getName(); pFunctionListBox->aArr.back()->sHelpText = description; pNewEntry->SetUserData( pFunctionListBox->aArr.back().get() ); } } } } catch (RuntimeException&) { // do nothing, the entry will not be displayed in the UI } } break; } case SfxCfgKind::GROUP_STYLES : { SfxStyleInfo_Impl* pFamily = static_cast(pInfo->pObject); if (pFamily) { const std::vector< SfxStyleInfo_Impl > lStyles = pStylesInfo->getStyles(pFamily->sFamily); std::vector< SfxStyleInfo_Impl >::const_iterator pIt; for ( pIt = lStyles.begin(); pIt != lStyles.end() ; ++pIt ) { SfxStyleInfo_Impl* pStyle = new SfxStyleInfo_Impl(*pIt); SvTreeListEntry* pFuncEntry = pFunctionListBox->InsertEntry( pStyle->sLabel ); pFunctionListBox->aArr.push_back( o3tl::make_unique( SfxCfgKind::GROUP_STYLES, 0, pStyle ) ); pFunctionListBox->aArr.back()->sCommand = pStyle->sCommand; pFunctionListBox->aArr.back()->sLabel = pStyle->sLabel; pFuncEntry->SetUserData( pFunctionListBox->aArr.back().get() ); } } break; } default: return; } if ( pFunctionListBox->GetEntryCount() ) pFunctionListBox->Select( pFunctionListBox->GetEntry( nullptr, 0 ) ); pFunctionListBox->SetUpdateMode(true); } bool SfxConfigGroupListBox::Expand( SvTreeListEntry* pParent ) { bool bRet = SvTreeListBox::Expand( pParent ); if ( bRet ) { sal_uLong nEntries = GetOutputSizePixel().Height() / GetEntryHeight(); sal_uLong nChildCount = GetVisibleChildCount( pParent ); if ( nChildCount+1 > nEntries ) { MakeVisible( pParent, true ); } else { SvTreeListEntry *pEntry = GetFirstEntryInView(); sal_uLong nParentPos = 0; while ( pEntry && pEntry != pParent ) { ++nParentPos; pEntry = GetNextEntryInView( pEntry ); } if ( nParentPos + nChildCount + 1 > nEntries ) ScrollOutputArea( (short)( nEntries - ( nParentPos + nChildCount + 1 ) ) ); } } return bRet; } void SfxConfigGroupListBox::RequestingChildren( SvTreeListEntry *pEntry ) /* Description A basic or a library is opened. */ { SfxGroupInfo_Impl *pInfo = static_cast(pEntry->GetUserData()); pInfo->bWasOpened = true; switch ( pInfo->nKind ) { case SfxCfgKind::GROUP_SCRIPTCONTAINER: { if ( !GetChildCount( pEntry ) ) { Reference< browse::XBrowseNode > rootNode( static_cast< browse::XBrowseNode* >( pInfo->pObject ) ) ; FillScriptList(rootNode, pEntry, true /* i30923 */ ); } break; } case SfxCfgKind::GROUP_STYLES: { if ( !GetChildCount( pEntry ) ) { const std::vector< SfxStyleInfo_Impl > lStyleFamilies = pStylesInfo->getStyleFamilies(); std::vector< SfxStyleInfo_Impl >::const_iterator pIt; for ( pIt = lStyleFamilies.begin(); pIt != lStyleFamilies.end() ; ++pIt ) { SfxStyleInfo_Impl* pFamily = new SfxStyleInfo_Impl(*pIt); SvTreeListEntry* pStyleEntry = InsertEntry( pFamily->sLabel, pEntry ); aArr.push_back( o3tl::make_unique( SfxCfgKind::GROUP_STYLES, 0, pFamily )); pStyleEntry->SetUserData( aArr.back().get() ); pStyleEntry->EnableChildrenOnDemand( false ); } } break; } default: OSL_FAIL( "Falscher Gruppentyp!" ); break; } } void SfxConfigGroupListBox::SelectMacro( const SfxMacroInfoItem *pItem ) { SelectMacro( pItem->GetBasicManager()->GetName(), pItem->GetQualifiedName() ); } void SfxConfigGroupListBox::SelectMacro( const OUString& rBasic, const OUString& rMacro ) { const OUString aBasicName(rBasic + " " + xImp->m_sMacros); const sal_Int32 nCount = comphelper::string::getTokenCount(rMacro, '.'); const OUString aMethod( rMacro.getToken( nCount-1, '.' ) ); OUString aLib; OUString aModule; if ( nCount > 2 ) { aLib = rMacro.getToken( 0, '.' ); aModule = rMacro.getToken( nCount-2, '.' ); } SvTreeListEntry *pEntry = FirstChild(nullptr); while ( pEntry ) { OUString aEntryBas = GetEntryText( pEntry ); if ( aEntryBas == aBasicName ) { Expand( pEntry ); SvTreeListEntry *pLib = FirstChild( pEntry ); while ( pLib ) { OUString aEntryLib = GetEntryText( pLib ); if ( aEntryLib == aLib ) { Expand( pLib ); SvTreeListEntry *pMod = FirstChild( pLib ); while ( pMod ) { OUString aEntryMod = GetEntryText( pMod ); if ( aEntryMod == aModule ) { Expand( pMod ); MakeVisible( pMod ); Select( pMod ); SvTreeListEntry *pMethod = pFunctionListBox->First(); while ( pMethod ) { OUString aEntryMethod = GetEntryText( pMethod ); if ( aEntryMethod == aMethod ) { pFunctionListBox->Select( pMethod ); pFunctionListBox->MakeVisible( pMethod ); return; } pMethod = pFunctionListBox->Next( pMethod ); } } pMod = NextSibling( pMod ); } } pLib = NextSibling( pLib ); } } pEntry = NextSibling( pEntry ); } } /* * Implementation of SvxScriptSelectorDialog * * This dialog is used for selecting Slot API commands * and Scripting Framework Scripts. */ SvxScriptSelectorDialog::SvxScriptSelectorDialog( vcl::Window* pParent, bool bShowSlots, const css::uno::Reference< css::frame::XFrame >& xFrame) : ModalDialog(pParent, "MacroSelectorDialog", "cui/ui/macroselectordialog.ui") , m_bShowSlots(bShowSlots) { get("libraryft")->Show(!m_bShowSlots); get("categoryft")->Show(m_bShowSlots); get("macronameft")->Show(!m_bShowSlots); get("commandsft")->Show(m_bShowSlots); get(m_pDescriptionText, "description"); get(m_pCommands, "commands"); if (m_bShowSlots) { // If we are showing Slot API commands update labels in the UI SetText(CuiResId(RID_SVXSTR_SELECTOR_ADD_COMMANDS)); get(m_pCancelButton, "close"); get(m_pDialogDescription, "helptoolbar"); get(m_pOKButton, "add"); } else { get(m_pCancelButton, "cancel"); get(m_pDialogDescription, "helpmacro"); get(m_pOKButton, "ok"); } m_pCancelButton->Show(); m_pDialogDescription->Show(); m_pOKButton->Show(); get(m_pCategories, "categories"); const OUString aModuleName(vcl::CommandInfoProvider::GetModuleIdentifier(xFrame)); m_pCategories->SetFunctionListBox(m_pCommands); m_pCategories->Init(comphelper::getProcessComponentContext(), xFrame, aModuleName, bShowSlots); m_pCategories->SetSelectHdl( LINK( this, SvxScriptSelectorDialog, SelectHdl ) ); m_pCommands->SetSelectHdl( LINK( this, SvxScriptSelectorDialog, SelectHdl ) ); m_pCommands->SetDoubleClickHdl( LINK( this, SvxScriptSelectorDialog, FunctionDoubleClickHdl ) ); m_pOKButton->SetClickHdl( LINK( this, SvxScriptSelectorDialog, ClickHdl ) ); m_pCancelButton->SetClickHdl( LINK( this, SvxScriptSelectorDialog, ClickHdl ) ); m_sDefaultDesc = m_pDescriptionText->GetText(); // Support style commands uno::Reference xController; uno::Reference xModel; if (xFrame.is()) xController = xFrame->getController(); if (xController.is()) xModel = xController->getModel(); m_aStylesInfo.init(aModuleName, xModel); m_pCommands->SetStylesInfo(&m_aStylesInfo); m_pCategories->SetStylesInfo(&m_aStylesInfo); UpdateUI(); } SvxScriptSelectorDialog::~SvxScriptSelectorDialog() { disposeOnce(); } void SvxScriptSelectorDialog::dispose() { m_pDialogDescription.clear(); m_pCategories.clear(); m_pCommands.clear(); m_pOKButton.clear(); m_pCancelButton.clear(); m_pDescriptionText.clear(); ModalDialog::dispose(); } IMPL_LINK( SvxScriptSelectorDialog, SelectHdl, SvTreeListBox*, pCtrl, void ) { if (pCtrl == m_pCategories) { m_pCategories->GroupSelected(); } UpdateUI(); } IMPL_LINK_NOARG( SvxScriptSelectorDialog, FunctionDoubleClickHdl, SvTreeListBox*, bool ) { if (m_pOKButton->IsEnabled()) ClickHdl(m_pOKButton); return false; } // Check if command is selected and enable the OK button accordingly // Grab the help text for this id if available and update the description field void SvxScriptSelectorDialog::UpdateUI() { OUString url = GetScriptURL(); if ( !url.isEmpty() ) { OUString sMessage = m_pCommands->GetHelpText(); m_pDescriptionText->SetText(sMessage.isEmpty() ? m_sDefaultDesc : sMessage); m_pOKButton->Enable(); } else { m_pDescriptionText->SetText(m_sDefaultDesc); m_pOKButton->Enable( false ); } } IMPL_LINK( SvxScriptSelectorDialog, ClickHdl, Button *, pButton, void ) { if (pButton == m_pCancelButton) { EndDialog(); } else if (pButton == m_pOKButton) { GetAddHdl().Call( *this ); // If we are displaying Slot API commands then this the dialog is being // run from Tools/Configure and we should not close it if ( !m_bShowSlots ) { EndDialog( RET_OK ); } else { // Select the next entry in the list if possible SvTreeListEntry* current = m_pCommands->FirstSelected(); SvTreeListEntry* next = SvTreeListBox::NextSibling( current ); if ( next != nullptr ) { m_pCommands->Select( next ); } } } } void SvxScriptSelectorDialog::SetRunLabel() { m_pOKButton->SetText(CuiResId(RID_SVXSTR_SELECTOR_RUN)); } void SvxScriptSelectorDialog::SetDialogDescription( const OUString& rDescription ) { m_pDialogDescription->SetText( rDescription ); } OUString SvxScriptSelectorDialog::GetScriptURL() const { OUString result; SvTreeListEntry *pEntry = const_cast< SvxScriptSelectorDialog* >( this )->m_pCommands->FirstSelected(); if ( pEntry ) { SfxGroupInfo_Impl *pData = static_cast(pEntry->GetUserData()); if ( ( pData->nKind == SfxCfgKind::FUNCTION_SLOT ) || ( pData->nKind == SfxCfgKind::FUNCTION_SCRIPT ) || ( pData->nKind == SfxCfgKind::GROUP_STYLES ) ) { result = pData->sCommand; } } return result; } OUString SvxScriptSelectorDialog::GetSelectedDisplayName() { return m_pCommands->GetEntryText( m_pCommands->FirstSelected() ); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */