1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #include <cfgutil.hxx>
22 #include <com/sun/star/beans/XPropertySet.hpp>
23 #include <com/sun/star/container/XEnumerationAccess.hpp>
24 #include <com/sun/star/container/XEnumeration.hpp>
25 #include <com/sun/star/document/XScriptInvocationContext.hpp>
26 #include <com/sun/star/frame/ModuleManager.hpp>
27 #include <com/sun/star/frame/Desktop.hpp>
28 #include <com/sun/star/frame/theUICommandDescription.hpp>
29 #include <com/sun/star/frame/XDispatchInformationProvider.hpp>
30 #include <com/sun/star/script/browse/XBrowseNode.hpp>
31 #include <com/sun/star/script/browse/BrowseNodeTypes.hpp>
32 #include <com/sun/star/script/browse/theBrowseNodeFactory.hpp>
33 #include <com/sun/star/script/browse/BrowseNodeFactoryViewTypes.hpp>
34 #include <com/sun/star/style/XStyleFamiliesSupplier.hpp>
35 #include <com/sun/star/uno/RuntimeException.hpp>
36 #include <com/sun/star/ui/theUICategoryDescription.hpp>
38 #include <basic/basmgr.hxx>
39 #include <tools/urlobj.hxx>
40 #include <strings.hrc>
41 #include <bitmaps.hlst>
42 #include <sfx2/minfitem.hxx>
43 #include <comphelper/SetFlagContextHelper.hxx>
44 #include <comphelper/documentinfo.hxx>
45 #include <comphelper/lok.hxx>
46 #include <comphelper/processfactory.hxx>
47 #include <comphelper/sequenceashashmap.hxx>
48 #include <svtools/imagemgr.hxx>
49 #include <sal/log.hxx>
50 #include <osl/diagnose.h>
51 #include <dialmgr.hxx>
52 #include <comphelper/diagnose_ex.hxx>
53 #include <vcl/commandevent.hxx>
54 #include <vcl/commandinfoprovider.hxx>
55 #include <vcl/help.hxx>
56 #include <vcl/svapp.hxx>
57 #include <o3tl/string_view.hxx>
59 #include <sfx2/sidebar/ResourceManager.hxx>
60 #include <sfx2/sidebar/Context.hxx>
61 #include <unotools/viewoptions.hxx>
63 using namespace ::com::sun::star
;
64 using namespace ::com::sun::star::uno
;
65 using namespace ::com::sun::star::script
;
66 using namespace ::com::sun::star::frame
;
67 using namespace ::com::sun::star::document
;
69 SfxStylesInfo_Impl::SfxStylesInfo_Impl()
72 void SfxStylesInfo_Impl::init(const OUString
& rModuleName
, const css::uno::Reference
< css::frame::XModel
>& xModel
)
74 m_aModuleName
= rModuleName
;
78 const char CMDURL_STYLEPROT_ONLY
[] = ".uno:StyleApply?";
79 const char CMDURL_SPART_ONLY
[] = "Style:string=";
80 const char CMDURL_FPART_ONLY
[] = "FamilyName:string=";
82 constexpr OUStringLiteral STYLEPROP_UINAME
= u
"DisplayName";
83 constexpr OUStringLiteral MACRO_SELECTOR_CONFIGNAME
= u
"MacroSelectorDialog";
84 constexpr OUStringLiteral LAST_RUN_MACRO_INFO
= u
"LastRunMacro";
86 OUString
SfxStylesInfo_Impl::generateCommand(
87 std::u16string_view sFamily
, std::u16string_view sStyle
)
89 return OUString::Concat(".uno:StyleApply?Style:string=")
91 + "&FamilyName:string="
95 bool SfxStylesInfo_Impl::parseStyleCommand(SfxStyleInfo_Impl
& aStyle
)
97 static const sal_Int32 LEN_STYLEPROT
= strlen(CMDURL_STYLEPROT_ONLY
);
98 static const sal_Int32 LEN_SPART
= strlen(CMDURL_SPART_ONLY
);
99 static const sal_Int32 LEN_FPART
= strlen(CMDURL_FPART_ONLY
);
101 if (!aStyle
.sCommand
.startsWith(CMDURL_STYLEPROT_ONLY
))
104 aStyle
.sFamily
.clear();
105 aStyle
.sStyle
.clear();
107 sal_Int32 nCmdLen
= aStyle
.sCommand
.getLength();
108 OUString sCmdArgs
= aStyle
.sCommand
.copy(LEN_STYLEPROT
, nCmdLen
-LEN_STYLEPROT
);
109 sal_Int32 i
= sCmdArgs
.indexOf('&');
113 OUString sArg
= sCmdArgs
.copy(0, i
);
114 if (sArg
.startsWith(CMDURL_SPART_ONLY
))
115 aStyle
.sStyle
= sArg
.copy(LEN_SPART
);
116 else if (sArg
.startsWith(CMDURL_FPART_ONLY
))
117 aStyle
.sFamily
= sArg
.copy(LEN_FPART
);
119 sArg
= sCmdArgs
.copy(i
+1, sCmdArgs
.getLength()-i
-1);
120 if (sArg
.startsWith(CMDURL_SPART_ONLY
))
121 aStyle
.sStyle
= sArg
.copy(LEN_SPART
);
122 else if (sArg
.startsWith(CMDURL_FPART_ONLY
))
123 aStyle
.sFamily
= sArg
.copy(LEN_FPART
);
125 return !(aStyle
.sFamily
.isEmpty() || aStyle
.sStyle
.isEmpty());
128 void SfxStylesInfo_Impl::getLabel4Style(SfxStyleInfo_Impl
& aStyle
)
132 css::uno::Reference
< css::style::XStyleFamiliesSupplier
> xModel(m_xDoc
, css::uno::UNO_QUERY
);
134 css::uno::Reference
< css::container::XNameAccess
> xFamilies
;
136 xFamilies
= xModel
->getStyleFamilies();
138 css::uno::Reference
< css::container::XNameAccess
> xStyleSet
;
140 xFamilies
->getByName(aStyle
.sFamily
) >>= xStyleSet
;
142 css::uno::Reference
< css::beans::XPropertySet
> xStyle
;
144 xStyleSet
->getByName(aStyle
.sStyle
) >>= xStyle
;
146 aStyle
.sLabel
.clear();
148 xStyle
->getPropertyValue(STYLEPROP_UINAME
) >>= aStyle
.sLabel
;
150 catch(const css::uno::RuntimeException
&)
152 catch(const css::uno::Exception
&)
153 { aStyle
.sLabel
.clear(); }
155 if (aStyle
.sLabel
.isEmpty())
157 aStyle
.sLabel
= aStyle
.sCommand
;
161 std::vector
< SfxStyleInfo_Impl
> SfxStylesInfo_Impl::getStyleFamilies() const
163 // It's an optional interface!
164 css::uno::Reference
< css::style::XStyleFamiliesSupplier
> xModel(m_xDoc
, css::uno::UNO_QUERY
);
166 return std::vector
< SfxStyleInfo_Impl
>();
168 css::uno::Reference
< css::container::XNameAccess
> xCont
= xModel
->getStyleFamilies();
169 const css::uno::Sequence
< OUString
> lFamilyNames
= xCont
->getElementNames();
170 std::vector
< SfxStyleInfo_Impl
> lFamilies
;
171 for (const auto& aFamily
: lFamilyNames
)
173 if ((aFamily
== "CellStyles" && m_aModuleName
!= "com.sun.star.sheet.SpreadsheetDocument") ||
174 aFamily
== "cell" || aFamily
== "table" || aFamily
== "Default")
177 SfxStyleInfo_Impl aFamilyInfo
;
178 aFamilyInfo
.sFamily
= aFamily
;
182 css::uno::Reference
< css::beans::XPropertySet
> xFamilyInfo
;
183 xCont
->getByName(aFamilyInfo
.sFamily
) >>= xFamilyInfo
;
184 if (!xFamilyInfo
.is())
186 // TODO_AS currently there is no support for an UIName property .. use internal family name instead
187 aFamilyInfo
.sLabel
= aFamilyInfo
.sFamily
;
190 xFamilyInfo
->getPropertyValue(STYLEPROP_UINAME
) >>= aFamilyInfo
.sLabel
;
192 catch(const css::uno::RuntimeException
&)
194 catch(const css::uno::Exception
&)
195 { return std::vector
< SfxStyleInfo_Impl
>(); }
197 lFamilies
.push_back(aFamilyInfo
);
203 std::vector
< SfxStyleInfo_Impl
> SfxStylesInfo_Impl::getStyles(const OUString
& sFamily
)
205 css::uno::Sequence
< OUString
> lStyleNames
;
206 css::uno::Reference
< css::style::XStyleFamiliesSupplier
> xModel(m_xDoc
, css::uno::UNO_QUERY_THROW
);
207 css::uno::Reference
< css::container::XNameAccess
> xFamilies
= xModel
->getStyleFamilies();
208 css::uno::Reference
< css::container::XNameAccess
> xStyleSet
;
211 xFamilies
->getByName(sFamily
) >>= xStyleSet
;
212 lStyleNames
= xStyleSet
->getElementNames();
214 catch(const css::uno::RuntimeException
&)
216 catch(const css::uno::Exception
&)
217 { return std::vector
< SfxStyleInfo_Impl
>(); }
219 std::vector
< SfxStyleInfo_Impl
> lStyles
;
220 sal_Int32 c
= lStyleNames
.getLength();
224 SfxStyleInfo_Impl aStyleInfo
;
225 aStyleInfo
.sFamily
= sFamily
;
226 aStyleInfo
.sStyle
= lStyleNames
[i
];
227 aStyleInfo
.sCommand
= SfxStylesInfo_Impl::generateCommand(aStyleInfo
.sFamily
, aStyleInfo
.sStyle
);
231 css::uno::Reference
< css::beans::XPropertySet
> xStyle
;
232 xStyleSet
->getByName(aStyleInfo
.sStyle
) >>= xStyle
;
235 xStyle
->getPropertyValue("DisplayName") >>= aStyleInfo
.sLabel
;
237 catch(const css::uno::RuntimeException
&)
239 catch(const css::uno::Exception
&)
242 lStyles
.push_back(aStyleInfo
);
247 OUString
CuiConfigFunctionListBox::GetHelpText( bool bConsiderParent
)
249 SfxGroupInfo_Impl
*pData
= weld::fromId
<SfxGroupInfo_Impl
*>(get_selected_id());
252 if ( pData
->nKind
== SfxCfgKind::FUNCTION_SLOT
)
255 return Application::GetHelp()->GetHelpText(pData
->sCommand
, m_xTreeView
.get());
257 return Application::GetHelp()->GetHelpText(pData
->sCommand
, static_cast<weld::Widget
*>(nullptr));
259 else if ( pData
->nKind
== SfxCfgKind::FUNCTION_SCRIPT
)
261 return pData
->sHelpText
;
267 OUString
CuiConfigFunctionListBox::GetCurCommand() const
269 SfxGroupInfo_Impl
*pData
= weld::fromId
<SfxGroupInfo_Impl
*>(get_selected_id());
272 return pData
->sCommand
;
275 OUString
CuiConfigFunctionListBox::GetCurLabel() const
277 SfxGroupInfo_Impl
*pData
= weld::fromId
<SfxGroupInfo_Impl
*>(get_selected_id());
280 if (!pData
->sLabel
.isEmpty())
281 return pData
->sLabel
;
282 return pData
->sCommand
;
285 CuiConfigFunctionListBox::CuiConfigFunctionListBox(std::unique_ptr
<weld::TreeView
> xTreeView
)
286 : m_xTreeView(std::move(xTreeView
))
287 , m_xScratchIter(m_xTreeView
->make_iterator())
289 m_xTreeView
->make_sorted();
290 m_xTreeView
->set_size_request(m_xTreeView
->get_approximate_digit_width() * 35, m_xTreeView
->get_height_rows(9));
291 m_xTreeView
->connect_query_tooltip(LINK(this, CuiConfigFunctionListBox
, QueryTooltip
));
294 CuiConfigFunctionListBox::~CuiConfigFunctionListBox()
299 IMPL_LINK(CuiConfigFunctionListBox
, QueryTooltip
, const weld::TreeIter
&, rIter
, OUString
)
301 SfxGroupInfo_Impl
*pData
= weld::fromId
<SfxGroupInfo_Impl
*>(m_xTreeView
->get_id(rIter
));
304 OUString aLabel
= CuiResId(RID_CUISTR_COMMANDLABEL
) + ": ";
305 OUString aName
= CuiResId(RID_CUISTR_COMMANDNAME
) + ": ";
306 OUString aTip
= CuiResId(RID_CUISTR_COMMANDTIP
) + ": ";
307 return aLabel
+ pData
->sLabel
+ "\n" + aName
+ pData
->sCommand
+ "\n" + aTip
+ pData
->sTooltip
;
310 void CuiConfigFunctionListBox::ClearAll()
312 Deletes all entries in the FunctionListBox, all UserData and all
313 possibly existing MacroInfo.
316 sal_uInt16 nCount
= aArr
.size();
317 for ( sal_uInt16 i
=0; i
<nCount
; ++i
)
319 SfxGroupInfo_Impl
*pData
= aArr
[i
].get();
321 if ( pData
->nKind
== SfxCfgKind::FUNCTION_SCRIPT
)
323 OUString
* pScriptURI
= static_cast<OUString
*>(pData
->pObject
);
327 if ( pData
->nKind
== SfxCfgKind::GROUP_SCRIPTCONTAINER
)
329 XInterface
* xi
= static_cast<XInterface
*>(pData
->pObject
);
338 m_xTreeView
->clear();
341 OUString
CuiConfigFunctionListBox::GetSelectedScriptURI() const
343 SfxGroupInfo_Impl
*pData
= weld::fromId
<SfxGroupInfo_Impl
*>(get_selected_id());
344 if (pData
&& pData
->nKind
== SfxCfgKind::FUNCTION_SCRIPT
)
345 return *static_cast<OUString
*>(pData
->pObject
);
349 struct SvxConfigGroupBoxResource_Impl
351 OUString m_sMyMacros
;
352 OUString m_sProdMacros
;
353 OUString m_sDlgMacros
;
354 OUString m_aStrGroupStyles
;
355 OUString m_aStrGroupSidebarDecks
;
357 SvxConfigGroupBoxResource_Impl();
360 SvxConfigGroupBoxResource_Impl::SvxConfigGroupBoxResource_Impl() :
361 m_sMyMacros(CuiResId(RID_CUISTR_MYMACROS
)),
362 m_sProdMacros(CuiResId(RID_CUISTR_PRODMACROS
)),
363 m_sDlgMacros(CuiResId(RID_CUISTR_PRODMACROS
)),
364 m_aStrGroupStyles(CuiResId(RID_CUISTR_GROUP_STYLES
)),
365 m_aStrGroupSidebarDecks(CuiResId(RID_CUISTR_GROUP_SIDEBARDECKS
))
369 void CuiConfigGroupListBox::SetStylesInfo(SfxStylesInfo_Impl
* pStyles
)
371 m_pStylesInfo
= pStyles
;
377 /** examines a component whether it supports XEmbeddedScripts, or provides access to such a
378 component by implementing XScriptInvocationContext.
380 the model which supports the embedded scripts, or <NULL/> if it cannot find such a
383 Reference
< XModel
> lcl_getDocumentWithScripts_throw( const Reference
< XInterface
>& _rxComponent
)
385 Reference
< XEmbeddedScripts
> xScripts( _rxComponent
, UNO_QUERY
);
386 if ( !xScripts
.is() )
388 Reference
< XScriptInvocationContext
> xContext( _rxComponent
, UNO_QUERY
);
390 xScripts
= xContext
->getScriptContainer();
393 return Reference
< XModel
>( xScripts
, UNO_QUERY
);
397 Reference
< XModel
> lcl_getScriptableDocument_nothrow( const Reference
< XFrame
>& _rxFrame
)
399 Reference
< XModel
> xDocument
;
401 // examine our associated frame
404 OSL_ENSURE( _rxFrame
.is(), "lcl_getScriptableDocument_nothrow: you need to pass a frame to this dialog/tab page!" );
407 // first try the model in the frame
408 Reference
< XController
> xController( _rxFrame
->getController(), UNO_SET_THROW
);
409 xDocument
= lcl_getDocumentWithScripts_throw( xController
->getModel() );
411 if ( !xDocument
.is() )
413 // if there is no suitable document in the frame, try the controller
414 xDocument
= lcl_getDocumentWithScripts_throw( _rxFrame
->getController() );
418 catch( const Exception
& )
426 CuiConfigGroupListBox::CuiConfigGroupListBox(std::unique_ptr
<weld::TreeView
> xTreeView
)
427 : xImp(new SvxConfigGroupBoxResource_Impl())
428 , m_pFunctionListBox(nullptr)
429 , m_pStylesInfo(nullptr)
430 , m_xTreeView(std::move(xTreeView
))
431 , m_xScratchIter(m_xTreeView
->make_iterator())
433 m_xTreeView
->connect_expanding(LINK(this, CuiConfigGroupListBox
, ExpandingHdl
));
434 m_xTreeView
->set_size_request(m_xTreeView
->get_approximate_digit_width() * 35, m_xTreeView
->get_height_rows(9));
437 CuiConfigGroupListBox::~CuiConfigGroupListBox()
442 void CuiConfigGroupListBox::ClearAll()
444 sal_uInt16 nCount
= aArr
.size();
445 for ( sal_uInt16 i
=0; i
<nCount
; ++i
)
447 SfxGroupInfo_Impl
*pData
= aArr
[i
].get();
448 if (pData
->nKind
== SfxCfgKind::GROUP_STYLES
&& pData
->pObject
)
450 SfxStyleInfo_Impl
* pStyle
= static_cast<SfxStyleInfo_Impl
*>(pData
->pObject
);
453 else if (pData
->nKind
== SfxCfgKind::FUNCTION_SCRIPT
&& pData
->pObject
)
455 OUString
* pScriptURI
= static_cast<OUString
*>(pData
->pObject
);
458 else if (pData
->nKind
== SfxCfgKind::GROUP_SCRIPTCONTAINER
)
460 XInterface
* xi
= static_cast<XInterface
*>(pData
->pObject
);
469 m_xTreeView
->clear();
472 void CuiConfigGroupListBox::InitModule()
476 css::uno::Reference
< css::frame::XDispatchInformationProvider
> xProvider(m_xFrame
, css::uno::UNO_QUERY_THROW
);
477 css::uno::Sequence
< sal_Int16
> lGroups
= xProvider
->getSupportedCommandGroups();
478 sal_Int32 c1
= lGroups
.getLength();
483 // Add All Commands category
484 aArr
.push_back(std::make_unique
<SfxGroupInfo_Impl
>(SfxCfgKind::GROUP_ALLFUNCTIONS
, 0));
485 m_xTreeView
->append(weld::toId(aArr
.back().get()), CuiResId(RID_CUISTR_ALLFUNCTIONS
));
488 for (i1
=0; i1
<c1
; ++i1
)
490 sal_Int16 nGroupID
= lGroups
[i1
];
491 OUString sGroupID
= OUString::number(nGroupID
);
492 OUString sGroupName
;
496 m_xModuleCategoryInfo
->getByName(sGroupID
) >>= sGroupName
;
497 if (sGroupName
.isEmpty())
500 catch(const css::container::NoSuchElementException
&)
503 aArr
.push_back( std::make_unique
<SfxGroupInfo_Impl
>( SfxCfgKind::GROUP_FUNCTION
, nGroupID
) );
504 m_xTreeView
->append(weld::toId(aArr
.back().get()), sGroupName
);
507 catch(const css::uno::RuntimeException
&)
509 catch(const css::uno::Exception
&)
513 void CuiConfigGroupListBox::FillScriptList(const css::uno::Reference
< css::script::browse::XBrowseNode
>& xRootNode
,
514 const weld::TreeIter
* pParentEntry
)
517 if ( xRootNode
->hasChildNodes() )
519 // tdf#120362: Don't ask to enable disabled Java when filling script list
520 css::uno::ContextLayer
layer(comphelper::NoEnableJavaInteractionContext());
522 const Sequence
< Reference
< browse::XBrowseNode
> > children
=
523 xRootNode
->getChildNodes();
524 bool bIsRootNode
= false;
526 OUString
user("user");
527 OUString
share("share");
528 if ( xRootNode
->getName() == "Root" )
533 //To mimic current starbasic behaviour we
534 //need to make sure that only the current document
535 //is displayed in the config tree. Tests below
536 //set the bDisplay flag to FALSE if the current
537 //node is a first level child of the Root and is NOT
538 //either the current document, user or share
539 OUString currentDocTitle
;
540 Reference
< XModel
> xDocument( lcl_getScriptableDocument_nothrow( m_xFrame
) );
541 if ( xDocument
.is() )
543 currentDocTitle
= ::comphelper::DocumentInfo::getDocumentTitle( xDocument
);
546 for ( Reference
< browse::XBrowseNode
> const & theChild
: children
)
548 bool bDisplay
= true;
549 OUString uiName
= theChild
->getName();
552 if ( ! (uiName
== user
|| uiName
== share
||
553 uiName
== currentDocTitle
) )
559 if ( uiName
== user
)
561 uiName
= xImp
->m_sMyMacros
;
563 else if ( uiName
== share
)
565 uiName
= xImp
->m_sProdMacros
;
569 if (theChild
->getType() != browse::BrowseNodeTypes::SCRIPT
&& bDisplay
)
571 // We call acquire on the XBrowseNode so that it does not
572 // get autodestructed and become invalid when accessed later.
575 bool bChildOnDemand
= false;
577 if ( theChild
->hasChildNodes() )
579 const Sequence
< Reference
< browse::XBrowseNode
> > grandchildren
=
580 theChild
->getChildNodes();
582 for ( const auto& rxNode
: grandchildren
)
584 if ( rxNode
->getType() == browse::BrowseNodeTypes::CONTAINER
)
586 bChildOnDemand
= true;
592 OUString aImage
= GetImage(theChild
, m_xContext
, bIsRootNode
);
594 aArr
.push_back( std::make_unique
<SfxGroupInfo_Impl
>(SfxCfgKind::GROUP_SCRIPTCONTAINER
,
595 0, static_cast<void *>( theChild
.get())));
597 OUString
sId(weld::toId(aArr
.back().get()));
598 m_xTreeView
->insert(pParentEntry
, -1, &uiName
, &sId
, nullptr, nullptr, bChildOnDemand
, m_xScratchIter
.get());
599 m_xTreeView
->set_image(*m_xScratchIter
, aImage
);
604 catch (RuntimeException
&) {
605 // do nothing, the entry will not be displayed in the UI
609 void CuiConfigGroupListBox::FillFunctionsList(const css::uno::Sequence
<DispatchInformation
>& xCommands
)
611 m_pFunctionListBox
->freeze();
612 for (const auto & rInfo
: xCommands
)
614 auto aProperties
= vcl::CommandInfoProvider::GetCommandProperties(rInfo
.Command
, m_sModuleLongName
);
616 OUString sUIName
= MapCommand2UIName(rInfo
.Command
);
617 aArr
.push_back( std::make_unique
<SfxGroupInfo_Impl
>( SfxCfgKind::FUNCTION_SLOT
, 0 ) );
618 SfxGroupInfo_Impl
* pGrpInfo
= aArr
.back().get();
619 pGrpInfo
->sCommand
= rInfo
.Command
;
620 pGrpInfo
->sLabel
= sUIName
;
621 pGrpInfo
->sTooltip
= vcl::CommandInfoProvider::GetTooltipForCommand(rInfo
.Command
, aProperties
, m_xFrame
);
622 m_pFunctionListBox
->append(weld::toId(pGrpInfo
), sUIName
);
624 m_pFunctionListBox
->thaw();
627 void CuiConfigGroupListBox::Init(const css::uno::Reference
< css::uno::XComponentContext
>& xContext
,
628 const css::uno::Reference
< css::frame::XFrame
>& xFrame
,
629 const OUString
& sModuleLongName
,
632 m_xTreeView
->freeze();
633 ClearAll(); // Remove all old entries from treelist box
635 m_xContext
= xContext
;
639 m_sModuleLongName
= sModuleLongName
;
640 m_xGlobalCategoryInfo
= css::ui::theUICategoryDescription::get( m_xContext
);
641 m_xModuleCategoryInfo
.set(m_xGlobalCategoryInfo
->getByName(m_sModuleLongName
), css::uno::UNO_QUERY_THROW
);
642 m_xUICmdDescription
= css::frame::theUICommandDescription::get( m_xContext
);
647 SAL_INFO("cui.customize", "** ** About to initialise SF Scripts");
648 // Add Scripting Framework entries
649 Reference
< browse::XBrowseNode
> rootNode
;
652 Reference
< browse::XBrowseNodeFactory
> xFac
= browse::theBrowseNodeFactory::get( m_xContext
);
653 rootNode
.set( xFac
->createView( browse::BrowseNodeFactoryViewTypes::MACROSELECTOR
) );
655 catch( const Exception
& )
657 TOOLS_WARN_EXCEPTION("cui.customize", "Caught some exception whilst retrieving browse nodes from factory");
658 // TODO exception handling
666 //We call acquire on the XBrowseNode so that it does not
667 //get autodestructed and become invalid when accessed later.
670 aArr
.push_back( std::make_unique
<SfxGroupInfo_Impl
>( SfxCfgKind::GROUP_SCRIPTCONTAINER
, 0,
671 static_cast<void *>(rootNode
.get())));
672 OUString
aTitle(xImp
->m_sDlgMacros
);
673 OUString
sId(weld::toId(aArr
.back().get()));
674 m_xTreeView
->insert(nullptr, -1, &aTitle
, &sId
, nullptr, nullptr, true, nullptr);
678 //We are only showing scripts not slot APIs so skip
679 //Root node and show location nodes
680 FillScriptList(rootNode
, nullptr);
684 // add styles and sidebar decks
687 aArr
.push_back( std::make_unique
<SfxGroupInfo_Impl
>( SfxCfgKind::GROUP_STYLES
, 0, nullptr ) ); // TODO last parameter should contain user data
688 OUString
sStyle(xImp
->m_aStrGroupStyles
);
689 OUString
sId(weld::toId(aArr
.back().get()));
690 m_xTreeView
->insert(nullptr, -1, &sStyle
, &sId
, nullptr, nullptr, true, nullptr);
692 aArr
.push_back( std::make_unique
<SfxGroupInfo_Impl
>(SfxCfgKind::GROUP_SIDEBARDECKS
, 0));
693 OUString
sSidebarDecks(xImp
->m_aStrGroupSidebarDecks
);
694 sId
= weld::toId(aArr
.back().get());
695 m_xTreeView
->insert(nullptr, -1, &sSidebarDecks
, &sId
, nullptr, nullptr, false, nullptr);
699 m_xTreeView
->scroll_to_row(0);
700 m_xTreeView
->select(0);
703 OUString
CuiConfigGroupListBox::GetImage(
704 const Reference
< browse::XBrowseNode
>& node
,
705 Reference
< XComponentContext
> const & xCtx
,
711 if (node
->getName() == "user" || node
->getName() == "share" )
713 aImage
= RID_CUIBMP_HARDDISK
;
718 OUString nodeName
= node
->getName();
719 Reference
<XInterface
> xDocumentModel
= getDocumentModel(xCtx
, nodeName
);
720 if ( xDocumentModel
.is() )
722 Reference
< frame::XModuleManager2
> xModuleManager( frame::ModuleManager::create(xCtx
) );
723 // get the long name of the document:
724 OUString
appModule( xModuleManager
->identify(
726 Sequence
<beans::PropertyValue
> moduleDescr
;
727 Any aAny
= xModuleManager
->getByName(appModule
);
728 if( !( aAny
>>= moduleDescr
) )
730 throw RuntimeException("SFTreeListBox::Init: failed to get PropertyValue");
732 beans::PropertyValue
const * pmoduleDescr
=
733 moduleDescr
.getConstArray();
734 for ( sal_Int32 pos
= moduleDescr
.getLength(); pos
--; )
736 if ( pmoduleDescr
[ pos
].Name
== "ooSetupFactoryEmptyDocumentURL" )
738 pmoduleDescr
[ pos
].Value
>>= factoryURL
;
739 SAL_INFO("cui.customize", "factory url for doc images is " << factoryURL
);
744 if( !factoryURL
.isEmpty() )
746 aImage
= SvFileInformationManager::GetFileImageId(INetURLObject(factoryURL
));
750 aImage
= RID_CUIBMP_DOC
;
756 if( node
->getType() == browse::BrowseNodeTypes::SCRIPT
)
757 aImage
= RID_CUIBMP_MACRO
;
759 aImage
= RID_CUIBMP_LIB
;
764 Reference
< XInterface
>
765 CuiConfigGroupListBox::getDocumentModel( Reference
< XComponentContext
> const & xCtx
, std::u16string_view docName
)
767 Reference
< XInterface
> xModel
;
768 Reference
< frame::XDesktop2
> desktop
= frame::Desktop::create( xCtx
);
770 Reference
< container::XEnumerationAccess
> componentsAccess
=
771 desktop
->getComponents();
772 Reference
< container::XEnumeration
> components
=
773 componentsAccess
->createEnumeration();
774 while (components
->hasMoreElements())
776 Reference
< frame::XModel
> model(
777 components
->nextElement(), UNO_QUERY
);
781 ::comphelper::DocumentInfo::getDocumentTitle( model
);
782 if( sTdocUrl
== docName
)
792 OUString
CuiConfigGroupListBox::MapCommand2UIName(const OUString
& sCommand
)
797 css::uno::Reference
< css::container::XNameAccess
> xModuleConf
;
798 m_xUICmdDescription
->getByName(m_sModuleLongName
) >>= xModuleConf
;
799 if (xModuleConf
.is())
801 ::comphelper::SequenceAsHashMap
lProps(xModuleConf
->getByName(sCommand
));
802 sUIName
= lProps
.getUnpackedValueOrDefault("Name", OUString());
805 catch(const css::uno::RuntimeException
&)
807 catch(css::uno::Exception
&)
810 // fallback for missing UINames !?
811 if (sUIName
.isEmpty())
819 void CuiConfigGroupListBox::GroupSelected()
821 A function group or a basic module has been selected.
822 All functions/macros are displayed in the functionlistbox.
825 std::unique_ptr
<weld::TreeIter
> xIter(m_xTreeView
->make_iterator());
826 if (!m_xTreeView
->get_selected(xIter
.get()))
829 SfxGroupInfo_Impl
*pInfo
= weld::fromId
<SfxGroupInfo_Impl
*>(m_xTreeView
->get_id(*xIter
));
830 m_pFunctionListBox
->freeze();
831 m_pFunctionListBox
->ClearAll();
833 switch ( pInfo
->nKind
)
835 case SfxCfgKind::GROUP_ALLFUNCTIONS
:
837 css::uno::Reference
< css::frame::XDispatchInformationProvider
> xProvider( m_xFrame
, UNO_QUERY
);
838 bool bValidIter
= m_xTreeView
->get_iter_first(*xIter
);
841 SfxGroupInfo_Impl
*pCurrentInfo
= weld::fromId
<SfxGroupInfo_Impl
*>(m_xTreeView
->get_id(*xIter
));
842 if (pCurrentInfo
->nKind
== SfxCfgKind::GROUP_FUNCTION
)
844 css::uno::Sequence
< css::frame::DispatchInformation
> lCommands
;
847 lCommands
= xProvider
->getConfigurableDispatchInformation( pCurrentInfo
->nUniqueID
);
848 FillFunctionsList( lCommands
);
850 catch ( container::NoSuchElementException
& )
854 bValidIter
= m_xTreeView
->iter_next(*xIter
);
859 case SfxCfgKind::GROUP_FUNCTION
:
861 sal_uInt16 nGroup
= pInfo
->nUniqueID
;
862 css::uno::Reference
< css::frame::XDispatchInformationProvider
> xProvider (m_xFrame
, css::uno::UNO_QUERY_THROW
);
863 css::uno::Sequence
< css::frame::DispatchInformation
> lCommands
= xProvider
->getConfigurableDispatchInformation(nGroup
);
864 FillFunctionsList( lCommands
);
868 case SfxCfgKind::GROUP_SCRIPTCONTAINER
:
870 if (!m_xTreeView
->iter_has_child(*xIter
))
872 Reference
< browse::XBrowseNode
> rootNode(
873 static_cast< browse::XBrowseNode
* >( pInfo
->pObject
) ) ;
876 if ( rootNode
->hasChildNodes() )
878 const Sequence
< Reference
< browse::XBrowseNode
> > children
=
879 rootNode
->getChildNodes();
881 for ( const Reference
< browse::XBrowseNode
>& childNode
: children
)
883 if (childNode
->getType() == browse::BrowseNodeTypes::SCRIPT
)
885 OUString uri
, description
;
887 Reference
< beans::XPropertySet
>xPropSet( childNode
, UNO_QUERY
);
894 xPropSet
->getPropertyValue("URI");
899 value
= xPropSet
->getPropertyValue("Description");
900 value
>>= description
;
902 catch (Exception
&) {
903 // do nothing, the description will be empty
906 OUString
* pScriptURI
= new OUString( uri
);
908 OUString aImage
= GetImage(childNode
, Reference
< XComponentContext
>(), false);
909 m_pFunctionListBox
->aArr
.push_back( std::make_unique
<SfxGroupInfo_Impl
>( SfxCfgKind::FUNCTION_SCRIPT
, 0, pScriptURI
));
910 m_pFunctionListBox
->aArr
.back()->sCommand
= uri
;
911 m_pFunctionListBox
->aArr
.back()->sLabel
= childNode
->getName();
912 m_pFunctionListBox
->aArr
.back()->sHelpText
= description
;
914 OUString
sId(weld::toId(m_pFunctionListBox
->aArr
.back().get()));
915 m_pFunctionListBox
->append(sId
, childNode
->getName(), aImage
);
920 catch (RuntimeException
&) {
921 // do nothing, the entry will not be displayed in the UI
927 case SfxCfgKind::GROUP_STYLES
:
929 SfxStyleInfo_Impl
* pFamily
= static_cast<SfxStyleInfo_Impl
*>(pInfo
->pObject
);
932 const std::vector
< SfxStyleInfo_Impl
> lStyles
= m_pStylesInfo
->getStyles(pFamily
->sFamily
);
933 for (auto const& lStyle
: lStyles
)
935 SfxStyleInfo_Impl
* pStyle
= new SfxStyleInfo_Impl(lStyle
);
936 m_pFunctionListBox
->aArr
.push_back(std::make_unique
<SfxGroupInfo_Impl
>(SfxCfgKind::GROUP_STYLES
, 0, pStyle
));
937 m_pFunctionListBox
->aArr
.back()->sCommand
= pStyle
->sCommand
;
938 m_pFunctionListBox
->aArr
.back()->sLabel
= pStyle
->sLabel
;
939 OUString
sId(weld::toId(m_pFunctionListBox
->aArr
.back().get()));
940 m_pFunctionListBox
->append(sId
, pStyle
->sLabel
);
946 case SfxCfgKind::GROUP_SIDEBARDECKS
:
948 sfx2::sidebar::ResourceManager aResourceManager
;
949 sfx2::sidebar::Context
aContext(m_sModuleLongName
, OUString());
950 sfx2::sidebar::ResourceManager::DeckContextDescriptorContainer aDecks
;
951 aResourceManager
.GetMatchingDecks(aDecks
, aContext
, false, m_xFrame
->getController());
953 for (auto const& rDeck
: aDecks
)
955 const OUString sCommand
= ".uno:SidebarDeck." + rDeck
.msId
;
956 m_pFunctionListBox
->aArr
.push_back(std::make_unique
<SfxGroupInfo_Impl
>(
957 SfxCfgKind::GROUP_SIDEBARDECKS
, 0,
959 m_pFunctionListBox
->aArr
.back()->sCommand
= sCommand
;
960 m_pFunctionListBox
->aArr
.back()->sLabel
= rDeck
.msId
;
961 m_pFunctionListBox
->aArr
.back()->sTooltip
=
962 vcl::CommandInfoProvider::GetCommandShortcut(sCommand
, m_xFrame
);
963 m_pFunctionListBox
->append(weld::toId(m_pFunctionListBox
->aArr
.back().get()),
971 // Do nothing, the list box will stay empty
972 SAL_INFO( "cui.customize", "Ignoring unexpected SfxCfgKind: " << static_cast<int>(pInfo
->nKind
) );
976 m_pFunctionListBox
->thaw();
978 if (m_pFunctionListBox
->n_children())
979 m_pFunctionListBox
->select(0);
983 A basic or a library is opened.
985 IMPL_LINK(CuiConfigGroupListBox
, ExpandingHdl
, const weld::TreeIter
&, rIter
, bool)
987 SfxGroupInfo_Impl
*pInfo
= weld::fromId
<SfxGroupInfo_Impl
*>(m_xTreeView
->get_id(rIter
));
988 switch ( pInfo
->nKind
)
990 case SfxCfgKind::GROUP_SCRIPTCONTAINER
:
992 if (!m_xTreeView
->iter_has_child(rIter
))
994 Reference
< browse::XBrowseNode
> rootNode(
995 static_cast< browse::XBrowseNode
* >( pInfo
->pObject
) ) ;
996 FillScriptList(rootNode
, &rIter
);
1001 case SfxCfgKind::GROUP_STYLES
:
1003 if (!m_xTreeView
->iter_has_child(rIter
))
1005 const std::vector
<SfxStyleInfo_Impl
> lStyleFamilies
= m_pStylesInfo
->getStyleFamilies();
1006 for (auto const& lStyleFamily
: lStyleFamilies
)
1008 SfxStyleInfo_Impl
* pFamily
= new SfxStyleInfo_Impl(lStyleFamily
);
1009 aArr
.push_back( std::make_unique
<SfxGroupInfo_Impl
>( SfxCfgKind::GROUP_STYLES
, 0, pFamily
));
1010 OUString
sId(weld::toId(aArr
.back().get()));
1011 m_xTreeView
->insert(&rIter
, -1, &pFamily
->sLabel
, &sId
, nullptr, nullptr, false, nullptr);
1018 OSL_FAIL( "Wrong group type!" );
1024 #if HAVE_FEATURE_SCRIPTING
1025 void CuiConfigGroupListBox::SelectMacro( const SfxMacroInfoItem
*pItem
)
1027 auto const rMacro
= pItem
->GetQualifiedName();
1028 sal_Int32 nIdx
{rMacro
.lastIndexOf('.')};
1029 const std::u16string_view
aMethod( rMacro
.subView(nIdx
+ 1) );
1030 std::u16string_view aLib
;
1031 std::u16string_view aModule
;
1034 // string contains at least 2 tokens
1035 nIdx
= rMacro
.lastIndexOf('.', nIdx
);
1038 // string contains at least 3 tokens
1039 aLib
= o3tl::getToken(rMacro
, 0, '.' );
1040 sal_Int32 nIdx2
= nIdx
+ 1;
1041 aModule
= o3tl::getToken(rMacro
, 0, '.', nIdx2
);
1045 std::unique_ptr
<weld::TreeIter
> xIter
= m_xTreeView
->make_iterator();
1046 if (!m_xTreeView
->get_iter_first(*xIter
))
1051 OUString aEntryBas
= m_xTreeView
->get_text(*xIter
);
1052 if (aEntryBas
== xImp
->m_sDlgMacros
)
1054 m_xTreeView
->expand_row(*xIter
);
1055 std::unique_ptr
<weld::TreeIter
> xLocationIter
= m_xTreeView
->make_iterator(xIter
.get());
1056 if (m_xTreeView
->iter_children(*xLocationIter
))
1060 m_xTreeView
->expand_row(*xLocationIter
);
1061 std::unique_ptr
<weld::TreeIter
> xLibIter
= m_xTreeView
->make_iterator(xLocationIter
.get());
1062 if (m_xTreeView
->iter_children(*xLibIter
))
1066 OUString aEntryLib
= m_xTreeView
->get_text(*xLibIter
);
1067 if (aEntryLib
== aLib
)
1069 m_xTreeView
->expand_row(*xLibIter
);
1070 std::unique_ptr
<weld::TreeIter
> xModIter
= m_xTreeView
->make_iterator(xLibIter
.get());
1071 if (m_xTreeView
->iter_children(*xModIter
))
1075 OUString aEntryMod
= m_xTreeView
->get_text(*xModIter
);
1076 if ( aEntryMod
== aModule
)
1078 m_xTreeView
->expand_row(*xModIter
);
1079 m_xTreeView
->scroll_to_row(*xModIter
);
1080 m_xTreeView
->select(*xModIter
);
1082 for (int i
= 0, nCount
= m_pFunctionListBox
->n_children(); i
< nCount
; ++i
)
1084 OUString aEntryMethod
= m_pFunctionListBox
->get_text(i
);
1085 if (aEntryMethod
== aMethod
)
1087 m_pFunctionListBox
->select(i
);
1088 m_pFunctionListBox
->scroll_to_row(i
);
1092 m_xTreeView
->collapse_row(*xModIter
);
1094 } while (m_xTreeView
->iter_next_sibling(*xModIter
));
1096 m_xTreeView
->collapse_row(*xLibIter
);
1098 } while (m_xTreeView
->iter_next_sibling(*xLibIter
));
1100 m_xTreeView
->collapse_row(*xLocationIter
);
1101 } while (m_xTreeView
->iter_next_sibling(*xLocationIter
));
1103 // If the macro can't be located, preselect the "Application Macros" category:
1104 m_xTreeView
->scroll_to_row(*xIter
);
1105 m_xTreeView
->select(*xIter
);
1108 } while (m_xTreeView
->iter_next_sibling(*xIter
));
1113 * Implementation of SvxScriptSelectorDialog
1115 * This dialog is used for selecting Slot API commands
1116 * and Scripting Framework Scripts.
1119 SvxScriptSelectorDialog::SvxScriptSelectorDialog(
1120 weld::Window
* pParent
, const css::uno::Reference
< css::frame::XFrame
>& xFrame
)
1121 : GenericDialogController(pParent
, "cui/ui/macroselectordialog.ui", "MacroSelectorDialog")
1122 , m_xDialogDescription(m_xBuilder
->weld_label("helpmacro"))
1123 , m_xCategories(new CuiConfigGroupListBox(m_xBuilder
->weld_tree_view("categories")))
1124 , m_xCommands(new CuiConfigFunctionListBox(m_xBuilder
->weld_tree_view("commands")))
1125 , m_xLibraryFT(m_xBuilder
->weld_label("libraryft"))
1126 , m_xMacronameFT(m_xBuilder
->weld_label("macronameft"))
1127 , m_xOKButton(m_xBuilder
->weld_button("ok"))
1128 , m_xCancelButton(m_xBuilder
->weld_button("cancel"))
1129 , m_xDescriptionText(m_xBuilder
->weld_text_view("description"))
1130 , m_xDescriptionFrame(m_xBuilder
->weld_frame("descriptionframe"))
1132 m_xCancelButton
->show();
1133 m_xDialogDescription
->show();
1134 m_xOKButton
->show();
1136 m_xLibraryFT
->set_visible(true);
1137 m_xMacronameFT
->set_visible(true);
1139 const OUString
aModuleName(vcl::CommandInfoProvider::GetModuleIdentifier(xFrame
));
1140 m_xCategories
->SetFunctionListBox(m_xCommands
.get());
1141 m_xCategories
->Init(comphelper::getProcessComponentContext(), xFrame
, aModuleName
, /*bShowSlots*/false);
1143 m_xCategories
->connect_changed(
1144 LINK( this, SvxScriptSelectorDialog
, SelectHdl
) );
1145 m_xCommands
->connect_changed( LINK( this, SvxScriptSelectorDialog
, SelectHdl
) );
1146 m_xCommands
->connect_row_activated( LINK( this, SvxScriptSelectorDialog
, FunctionDoubleClickHdl
) );
1147 m_xCommands
->connect_popup_menu( LINK( this, SvxScriptSelectorDialog
, ContextMenuHdl
) );
1149 m_xOKButton
->connect_clicked( LINK( this, SvxScriptSelectorDialog
, ClickHdl
) );
1150 m_xCancelButton
->connect_clicked( LINK( this, SvxScriptSelectorDialog
, ClickHdl
) );
1152 m_sDefaultDesc
= m_xDescriptionText
->get_text();
1154 // Support style commands
1155 uno::Reference
<frame::XController
> xController
;
1156 uno::Reference
<frame::XModel
> xModel
;
1158 xController
= xFrame
->getController();
1159 if (xController
.is())
1160 xModel
= xController
->getModel();
1162 m_aStylesInfo
.init(aModuleName
, xModel
);
1163 m_xCategories
->SetStylesInfo(&m_aStylesInfo
);
1165 // The following call is a workaround to make scroll_to_row work as expected in kf5/x11
1166 m_xDialog
->resize_to_request();
1168 LoadLastUsedMacro();
1171 if (comphelper::LibreOfficeKit::isActive())
1172 m_xDescriptionFrame
->hide();
1175 SvxScriptSelectorDialog::~SvxScriptSelectorDialog()
1179 IMPL_LINK(SvxScriptSelectorDialog
, SelectHdl
, weld::TreeView
&, rCtrl
, void)
1181 if (&rCtrl
== &m_xCategories
->get_widget())
1183 m_xCategories
->GroupSelected();
1188 IMPL_LINK_NOARG(SvxScriptSelectorDialog
, FunctionDoubleClickHdl
, weld::TreeView
&, bool)
1190 if (m_xOKButton
->get_sensitive())
1191 ClickHdl(*m_xOKButton
);
1195 IMPL_LINK(SvxScriptSelectorDialog
, ContextMenuHdl
, const CommandEvent
&, rCEvt
, bool)
1197 weld::TreeView
& xTreeView
= m_xCommands
->get_widget();
1198 if (rCEvt
.GetCommand() != CommandEventId::ContextMenu
|| !xTreeView
.n_children())
1201 std::unique_ptr
<weld::Builder
> xBuilder(Application::CreateBuilder(&xTreeView
, "modules/BasicIDE/ui/sortmenu.ui"));
1202 std::unique_ptr
<weld::Menu
> xPopup(xBuilder
->weld_menu("sortmenu"));
1203 std::unique_ptr
<weld::Menu
> xDropMenu(xBuilder
->weld_menu("sortsubmenu"));
1204 xDropMenu
->set_active("alphabetically", xTreeView
.get_sort_order());
1205 xDropMenu
->set_active("properorder", !xTreeView
.get_sort_order());
1207 OUString
sCommand(xPopup
->popup_at_rect(&xTreeView
, tools::Rectangle(rCEvt
.GetMousePosPixel(), Size(1,1))));
1208 if (sCommand
== "alphabetically")
1210 xTreeView
.make_sorted();
1212 else if (sCommand
== "properorder")
1214 xTreeView
.make_unsorted();
1215 m_xCategories
->GroupSelected();
1217 else if (!sCommand
.isEmpty())
1219 SAL_WARN("cui.customize", "Unknown context menu action: " << sCommand
);
1225 // Check if command is selected and enable the OK button accordingly
1226 // Grab the help text for this id if available and update the description field
1228 SvxScriptSelectorDialog::UpdateUI()
1230 OUString url
= GetScriptURL();
1231 if ( !url
.isEmpty() )
1233 OUString sMessage
= m_xCommands
->GetHelpText();
1234 m_xDescriptionText
->set_text(sMessage
.isEmpty() ? m_sDefaultDesc
: sMessage
);
1235 m_xOKButton
->set_sensitive(true);
1239 m_xDescriptionText
->set_text(m_sDefaultDesc
);
1240 m_xOKButton
->set_sensitive(false);
1244 IMPL_LINK(SvxScriptSelectorDialog
, ClickHdl
, weld::Button
&, rButton
, void)
1246 if (&rButton
== m_xCancelButton
.get())
1248 m_xDialog
->response(RET_CANCEL
);
1250 else if (&rButton
== m_xOKButton
.get())
1252 SaveLastUsedMacro();
1253 m_xDialog
->response(RET_OK
);
1258 SvxScriptSelectorDialog::SetRunLabel()
1260 m_xOKButton
->set_label(CuiResId(RID_CUISTR_SELECTOR_RUN
));
1264 SvxScriptSelectorDialog::GetScriptURL() const
1268 std::unique_ptr
<weld::TreeIter
> xIter
= m_xCommands
->make_iterator();
1269 if (m_xCommands
->get_selected(xIter
.get()))
1271 SfxGroupInfo_Impl
*pData
= weld::fromId
<SfxGroupInfo_Impl
*>(m_xCommands
->get_id(*xIter
));
1272 if ( ( pData
->nKind
== SfxCfgKind::FUNCTION_SLOT
)
1273 || ( pData
->nKind
== SfxCfgKind::FUNCTION_SCRIPT
)
1274 || ( pData
->nKind
== SfxCfgKind::GROUP_STYLES
)
1277 result
= pData
->sCommand
;
1285 SvxScriptSelectorDialog::SaveLastUsedMacro()
1287 // Gets the current selection in the dialog as a series of selected entries
1288 OUString sMacroInfo
;
1289 sMacroInfo
= m_xCommands
->get_selected_text();
1290 weld::TreeView
& xCategories
= m_xCategories
->get_widget();
1291 std::unique_ptr
<weld::TreeIter
> xIter
= xCategories
.make_iterator();
1293 if (!xCategories
.get_selected(xIter
.get()))
1298 sMacroInfo
= xCategories
.get_text(*xIter
) + "|" + sMacroInfo
;
1299 } while (xCategories
.iter_parent(*xIter
));
1301 SvtViewOptions( EViewType::Dialog
, MACRO_SELECTOR_CONFIGNAME
).SetUserItem(
1302 LAST_RUN_MACRO_INFO
, Any(sMacroInfo
));
1306 SvxScriptSelectorDialog::LoadLastUsedMacro()
1308 SvtViewOptions
aDlgOpt( EViewType::Dialog
, MACRO_SELECTOR_CONFIGNAME
);
1309 if (!aDlgOpt
.Exists())
1312 OUString sMacroInfo
;
1313 aDlgOpt
.GetUserItem(LAST_RUN_MACRO_INFO
) >>= sMacroInfo
;
1314 if (sMacroInfo
.isEmpty())
1317 // Counts how many entries exist in the macro info string
1318 sal_Int16 nInfoParts
= 0;
1319 sal_Int16 nLastIndex
= sMacroInfo
.indexOf('|');
1320 if (nLastIndex
> -1)
1323 while ( nLastIndex
!= -1 )
1326 nLastIndex
= sMacroInfo
.indexOf('|', nLastIndex
+ 1);
1330 weld::TreeView
& xCategories
= m_xCategories
->get_widget();
1331 std::unique_ptr
<weld::TreeIter
> xIter
= xCategories
.make_iterator();
1332 if (!xCategories
.get_iter_first(*xIter
))
1335 // Expand the nodes in the category tree
1336 OUString sNodeToExpand
;
1338 sal_Int16 nOpenedNodes
= 0;
1339 for (sal_Int16 i
=0; i
<nInfoParts
- 1; i
++)
1341 sNodeToExpand
= sMacroInfo
.getToken(i
, '|');
1342 bIsIterValid
= true;
1343 while (bIsIterValid
&& xCategories
.get_text(*xIter
) != sNodeToExpand
)
1344 bIsIterValid
= xCategories
.iter_next_sibling(*xIter
);
1348 xCategories
.expand_row(*xIter
);
1351 if (xCategories
.iter_has_child(*xIter
))
1352 (void)xCategories
.iter_children(*xIter
);
1353 else if (nOpenedNodes
< nInfoParts
- 1)
1354 // If the number of levels in the tree is smaller than the
1355 // number of parts in the macro info string, then return
1358 xCategories
.select(*xIter
);
1359 xCategories
.scroll_to_row(*xIter
);
1360 m_xCategories
->GroupSelected();
1362 // Select the macro in the command tree
1363 weld::TreeView
& xCommands
= m_xCommands
->get_widget();
1364 xIter
= xCommands
.make_iterator();
1365 if (!xCommands
.get_iter_first(*xIter
))
1368 OUString sMacroName
= sMacroInfo
.getToken(nInfoParts
- 1, '|');
1369 bIsIterValid
= true;
1370 while (bIsIterValid
&& xCommands
.get_text(*xIter
) != sMacroName
)
1371 bIsIterValid
= xCommands
.iter_next_sibling(*xIter
);
1375 xCommands
.scroll_to_row(*xIter
);
1376 xCommands
.select(*xIter
);
1380 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */