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/.
10 #ifndef INCLUDED_VCL_BUILDER_HXX
11 #define INCLUDED_VCL_BUILDER_HXX
14 #include <sal/log.hxx>
15 #include <unotools/resmgr.hxx>
16 #include <tools/fldunit.hxx>
17 #include <vcl/dllapi.h>
18 #include <vcl/window.hxx>
19 #include <vcl/vclptr.hxx>
20 #include <tools/wintypes.hxx>
21 #include <vcl/EnumContext.hxx>
25 #include <string_view>
28 # //some problem with MacOSX and a check define
38 class NumericFormatter
;
40 class SalInstanceBuilder
;
48 class VclMultiLineEdit
;
49 struct NotebookBarAddonsItem
;
50 namespace xmlreader
{ class XmlReader
; }
51 namespace com::sun::star::frame
{ class XFrame
; }
53 struct ComboBoxTextItem
57 ComboBoxTextItem(const OUString
& rItem
, const OString
& rId
)
64 /// Creates a hierarchy of vcl::Windows (widgets) from a .ui file for dialogs, sidebar, etc.
65 class VCL_DLLPUBLIC VclBuilder
68 typedef std::map
<OString
, OUString
> stringmap
;
69 typedef std::map
<OString
, std::pair
<OString
, OString
>> accelmap
;
70 /// These functions create a new widget with parent pParent and return it in rRet
71 typedef void (*customMakeWidget
)(VclPtr
<vcl::Window
> &rRet
, const VclPtr
<vcl::Window
> &pParent
, stringmap
&rVec
);
74 VclBuilder(vcl::Window
* pParent
, const OUString
& sUIRootDir
, const OUString
& sUIFile
,
75 const OString
& sID
= OString(),
76 const css::uno::Reference
<css::frame::XFrame
>& rFrame
77 = css::uno::Reference
<css::frame::XFrame
>(),
79 const NotebookBarAddonsItem
* pNotebookBarAddonsItem
= nullptr);
81 ///releases references and disposes all children.
82 void disposeBuilder();
83 //sID must exist and be of type T
84 template <typename T
> T
* get(VclPtr
<T
>& ret
, const OString
& sID
);
86 //sID may not exist, but must be of type T if it does
87 template <typename T
= vcl::Window
> T
* get(const OString
& sID
);
89 vcl::Window
* get_widget_root();
92 PopupMenu
* get_menu(const OString
& sID
);
94 //release ownership of pWindow, i.e. don't delete it
95 void drop_ownership(const vcl::Window
*pWindow
);
97 //see m_aDeferredProperties, you need this for toplevel dialogs
98 //which build themselves from their ctor. The properties on
99 //the top level are stored in m_aDeferredProperties and need
100 //to be applied post ctor
101 void setDeferredProperties();
103 /// return UI-File name (without '.ui')
104 const OString
& getUIFile() const
109 /// Pre-loads all modules containing UI information
110 static void preload();
112 static SymbolType
mapStockToSymbol(std::u16string_view icon_name
);
115 VclBuilder(const VclBuilder
&) = delete;
116 VclBuilder
& operator=(const VclBuilder
&) = delete;
118 // owner for ListBox/ComboBox UserData
119 std::vector
<std::unique_ptr
<OUString
>> m_aUserData
;
121 //If the toplevel window has any properties which need to be set on it,
122 //but the toplevel is the owner of the builder, then its ctor
123 //has not been completed during the building, so properties for it
124 //are collected here and need to be set afterwards, e.g. during
126 stringmap m_aDeferredProperties
;
128 std::unique_ptr
<NotebookBarAddonsItem
> m_pNotebookBarAddonsItem
;
132 bool m_bVerticalOrient
;
133 sal_Int32 m_nPosition
;
134 PackingData(bool bVerticalOrient
= false)
135 : m_bVerticalOrient(bVerticalOrient
)
144 VclPtr
<vcl::Window
> m_pWindow
;
145 PackingData m_aPackingData
;
146 WinAndId(const OString
&rId
, vcl::Window
*pWindow
, bool bVertical
)
149 , m_aPackingData(bVertical
)
153 std::vector
<WinAndId
> m_aChildren
;
158 VclPtr
<Menu
> m_pMenu
;
159 MenuAndId(const OString
&rId
, Menu
*pMenu
);
161 std::vector
<MenuAndId
> m_aMenus
;
167 StringPair(const OString
&rId
, const OString
&rValue
)
178 UStringPair(const OString
&rId
, const OUString
&rValue
)
185 typedef StringPair RadioButtonGroupMap
;
187 struct ButtonImageWidgetMap
192 ButtonImageWidgetMap(const OString
&rId
, const OUString
&rValue
, bool bRadio
)
200 typedef UStringPair TextBufferMap
;
201 typedef UStringPair WidgetAdjustmentMap
;
202 typedef UStringPair ButtonMenuMap
;
203 typedef UStringPair MnemonicWidgetMap
;
205 struct ComboBoxModelMap
209 sal_Int32 m_nActiveId
;
210 ComboBoxModelMap(const OString
&rId
, const OUString
&rValue
, sal_Int32 nActiveId
)
213 , m_nActiveId(nActiveId
)
220 typedef std::vector
<OUString
> row
;
221 std::vector
<row
> m_aEntries
;
224 const ListStore
* get_model_by_name(const OString
& sID
) const;
225 void mungeModel(ListBox
&rTarget
, const ListStore
&rStore
, sal_uInt16 nActiveId
);
226 void mungeModel(ComboBox
&rTarget
, const ListStore
&rStore
, sal_uInt16 nActiveId
);
227 void mungeModel(SvTabListBox
&rTarget
, const ListStore
&rStore
, sal_uInt16 nActiveId
);
229 typedef stringmap TextBuffer
;
230 const TextBuffer
* get_buffer_by_name(const OString
& sID
) const;
232 static void mungeTextBuffer(VclMultiLineEdit
&rTarget
, const TextBuffer
&rTextBuffer
);
234 typedef stringmap Adjustment
;
235 const Adjustment
* get_adjustment_by_name(const OString
& sID
) const;
237 static void mungeAdjustment(NumericFormatter
&rTarget
, const Adjustment
&rAdjustment
);
238 static void mungeAdjustment(FormattedField
&rTarget
, const Adjustment
&rAdjustment
);
239 static void mungeAdjustment(ScrollBar
&rTarget
, const Adjustment
&rAdjustment
);
240 static void mungeAdjustment(Slider
&rTarget
, const Adjustment
&rAdjustment
);
242 typedef std::map
<OString
, int> ImageSizeMap
;
246 std::vector
<OString
> m_aWidgets
;
247 stringmap m_aProperties
;
254 std::locale m_aResLocale
;
256 std::vector
<RadioButtonGroupMap
> m_aGroupMaps
;
258 std::vector
<ComboBoxModelMap
> m_aModelMaps
;
259 std::map
<OString
, ListStore
> m_aModels
;
261 std::vector
<TextBufferMap
> m_aTextBufferMaps
;
262 std::map
<OString
, TextBuffer
> m_aTextBuffers
;
264 std::vector
<WidgetAdjustmentMap
> m_aNumericFormatterAdjustmentMaps
;
265 std::vector
<WidgetAdjustmentMap
> m_aFormattedFormatterAdjustmentMaps
;
266 std::vector
<WidgetAdjustmentMap
> m_aScrollAdjustmentMaps
;
267 std::vector
<WidgetAdjustmentMap
> m_aSliderAdjustmentMaps
;
269 std::map
<OString
, Adjustment
> m_aAdjustments
;
271 std::vector
<ButtonImageWidgetMap
> m_aButtonImageWidgetMaps
;
272 ImageSizeMap m_aImageSizeMap
;
274 std::vector
<ButtonMenuMap
> m_aButtonMenuMaps
;
276 std::map
<VclPtr
<vcl::Window
>, VclPtr
<vcl::Window
>> m_aRedundantParentWidgets
;
278 std::vector
<SizeGroup
> m_aSizeGroups
;
280 std::map
<VclPtr
<vcl::Window
>, stringmap
> m_aAtkInfo
;
282 std::vector
<MnemonicWidgetMap
> m_aMnemonicWidgetMaps
;
284 std::vector
< VclPtr
<VclExpander
> > m_aExpanderWidgets
;
286 std::vector
< VclPtr
<MessageDialog
> > m_aMessageDialogs
;
288 sal_uInt16 m_nLastToolbarId
;
290 sal_uInt16 m_nLastMenuItemId
;
297 ResHookProc m_pStringReplace
;
298 VclPtr
<vcl::Window
> m_pParent
;
299 bool m_bToplevelHasDeferredInit
;
300 bool m_bToplevelHasDeferredProperties
;
301 bool m_bToplevelParentFound
;
303 std::unique_ptr
<ParserState
> m_pParserState
;
305 vcl::Window
*get_by_name(const OString
& sID
);
306 void delete_by_name(const OString
& sID
);
308 class sortIntoBestTabTraversalOrder
311 sortIntoBestTabTraversalOrder(VclBuilder
*pBuilder
)
312 : m_pBuilder(pBuilder
) {}
314 bool operator()(const vcl::Window
*pA
, const vcl::Window
*pB
) const;
317 VclBuilder
*m_pBuilder
;
320 /// XFrame to be able to extract labels and other properties of the UNO commands (like of .uno:Bold).
321 css::uno::Reference
<css::frame::XFrame
> m_xFrame
;
324 VclPtr
<vcl::Window
> insertObject(vcl::Window
*pParent
,
325 const OString
&rClass
, const OString
&rID
,
326 stringmap
&rProps
, stringmap
&rPangoAttributes
,
327 stringmap
&rAtkProps
);
329 VclPtr
<vcl::Window
> makeObject(vcl::Window
*pParent
,
330 const OString
&rClass
, const OString
&rID
,
333 void connectNumericFormatterAdjustment(const OString
&id
, const OUString
&rAdjustment
);
334 void connectFormattedFormatterAdjustment(const OString
&id
, const OUString
&rAdjustment
);
336 static int getImageSize(const stringmap
&rMap
);
338 void extractGroup(const OString
&id
, stringmap
&rVec
);
339 void extractModel(const OString
&id
, stringmap
&rVec
);
340 void extractBuffer(const OString
&id
, stringmap
&rVec
);
341 static bool extractAdjustmentToMap(const OString
&id
, stringmap
&rVec
, std::vector
<WidgetAdjustmentMap
>& rAdjustmentMap
);
342 void extractButtonImage(const OString
&id
, stringmap
&rMap
, bool bRadio
);
343 void extractMnemonicWidget(const OString
&id
, stringmap
&rMap
);
345 // either pParent or pAtkProps must be set, pParent for a child of a widget, pAtkProps for
346 // collecting the atk info for a GtkMenuItem
347 void handleChild(vcl::Window
*pParent
, stringmap
*pAtkProps
, xmlreader::XmlReader
&reader
);
348 VclPtr
<vcl::Window
> handleObject(vcl::Window
*pParent
, stringmap
*pAtkProps
, xmlreader::XmlReader
&reader
);
349 void handlePacking(vcl::Window
*pCurrent
, vcl::Window
*pParent
, xmlreader::XmlReader
&reader
);
350 static std::vector
<vcl::EnumContext::Context
> handleStyle(xmlreader::XmlReader
&reader
, int &nPriority
);
351 static OString
getStyleClass(xmlreader::XmlReader
&reader
);
352 void applyPackingProperty(vcl::Window
*pCurrent
, vcl::Window
*pParent
, xmlreader::XmlReader
&reader
);
353 void collectProperty(xmlreader::XmlReader
&reader
, stringmap
&rVec
) const;
354 static void collectPangoAttribute(xmlreader::XmlReader
&reader
, stringmap
&rMap
);
355 static void collectAtkRelationAttribute(xmlreader::XmlReader
&reader
, stringmap
&rMap
);
356 static void collectAtkRoleAttribute(xmlreader::XmlReader
&reader
, stringmap
&rMap
);
357 static void collectAccelerator(xmlreader::XmlReader
&reader
, accelmap
&rMap
);
359 void insertMenuObject(
362 const OString
&rClass
,
365 stringmap
&rAtkProps
,
368 void handleMenuChild(Menu
*pParent
, xmlreader::XmlReader
&reader
);
369 void handleMenuObject(Menu
*pParent
, xmlreader::XmlReader
&reader
);
371 void handleListStore(xmlreader::XmlReader
&reader
, const OString
&rID
, const OString
&rClass
);
372 void handleRow(xmlreader::XmlReader
&reader
, const OString
&rID
);
373 void handleTabChild(vcl::Window
*pParent
, xmlreader::XmlReader
&reader
);
374 VclPtr
<Menu
> handleMenu(xmlreader::XmlReader
&reader
, const OString
&rID
, bool bMenuBar
);
375 std::vector
<ComboBoxTextItem
> handleItems(xmlreader::XmlReader
&reader
) const;
377 void handleSizeGroup(xmlreader::XmlReader
&reader
);
379 stringmap
handleAtkObject(xmlreader::XmlReader
&reader
);
381 static void applyAtkProperties(vcl::Window
*pWindow
, const stringmap
& rProperties
);
383 void handleActionWidget(xmlreader::XmlReader
&reader
);
385 PackingData
get_window_packing_data(const vcl::Window
*pWindow
) const;
386 void set_window_packing_position(const vcl::Window
*pWindow
, sal_Int32 nPosition
);
388 static vcl::Window
* prepareWidgetOwnScrolling(vcl::Window
*pParent
, WinBits
&rWinStyle
);
389 void cleanupWidgetOwnScrolling(vcl::Window
*pScrollParent
, vcl::Window
*pWindow
, stringmap
&rMap
);
391 void set_response(const OString
& sID
, short nResponse
);
393 OString
get_by_window(const vcl::Window
*pWindow
) const;
394 void delete_by_window(vcl::Window
*pWindow
);
397 namespace BuilderUtils
399 //apply the properties of rProps to pWindow
400 VCL_DLLPUBLIC
void set_properties(vcl::Window
*pWindow
, const VclBuilder::stringmap
&rProps
);
402 //Convert _ gtk markup to ~ vcl markup
403 VCL_DLLPUBLIC OUString
convertMnemonicMarkup(const OUString
&rIn
);
405 VCL_DLLPUBLIC OUString
extractCustomProperty(VclBuilder::stringmap
&rMap
);
407 VCL_DLLPUBLIC
bool extractDropdown(VclBuilder::stringmap
&rMap
);
409 //add a default value of 25 width-chars to a map if width-chars not set
410 VCL_DLLPUBLIC
void ensureDefaultWidthChars(VclBuilder::stringmap
&rMap
);
412 //Helpers to retrofit all the existing code to the builder
413 VCL_DLLPUBLIC
void reorderWithinParent(std::vector
< vcl::Window
*>& rChilds
, bool bIsButtonBox
);
414 VCL_DLLPUBLIC
void reorderWithinParent(vcl::Window
&rWindow
, sal_uInt16 nNewPosition
);
416 //Convert an accessibility role name to accessibility role number
417 VCL_DLLPUBLIC sal_Int16
getRoleFromName(const OString
& roleName
);
420 template <typename T
>
421 inline T
* VclBuilder::get(VclPtr
<T
>& ret
, const OString
& sID
)
423 vcl::Window
*w
= get_by_name(sID
);
424 SAL_WARN_IF(!w
, "vcl.layout", "widget \"" << sID
<< "\" not found in .ui");
425 SAL_WARN_IF(!dynamic_cast<T
*>(w
),
426 "vcl.layout", ".ui widget \"" << sID
<< "\" needs to correspond to vcl type " << typeid(T
).name());
428 assert(dynamic_cast<T
*>(w
));
429 ret
= static_cast<T
*>(w
);
433 //sID may not exist, but must be of type T if it does
434 template <typename T
>
435 inline T
* VclBuilder::get(const OString
& sID
)
437 vcl::Window
*w
= get_by_name(sID
);
438 SAL_WARN_IF(w
&& !dynamic_cast<T
*>(w
),
439 "vcl.layout", ".ui widget \"" << sID
<< "\" needs to correspond to vcl type " << typeid(T
).name());
440 assert(!w
|| dynamic_cast<T
*>(w
));
441 return static_cast<T
*>(w
);
445 //helper baseclass to ease retro fitting dialogs/tabpages that load a resource
446 //to load a .ui file instead
448 //vcl requires the Window Children of a Parent Window to be destroyed before
449 //the Parent Window. VclBuilderContainer owns the VclBuilder which owns the
450 //Children Window. So the VclBuilderContainer dtor must be called before
451 //the Parent Window dtor.
453 //i.e. class Dialog : public SystemWindow, public VclBuilderContainer
454 //not class Dialog : public VclBuilderContainer, public SystemWindow
456 //With the new 'dispose' framework, it is necessary to force the builder
457 //dispose before the Window dispose; so a Dialog::dispose() method would
458 //finish: disposeBuilder(); SystemWindow::dispose() to capture this ordering.
460 class VCL_DLLPUBLIC VclBuilderContainer
463 VclBuilderContainer();
464 virtual ~VclBuilderContainer();
465 void disposeBuilder();
467 template <typename T
> T
* get(VclPtr
<T
>& ret
, const OString
& sID
)
469 return m_pUIBuilder
->get
<T
>(ret
, sID
);
471 template <typename T
= vcl::Window
> T
* get(const OString
& sID
)
473 return m_pUIBuilder
->get
<T
>(sID
);
475 void setDeferredProperties()
479 m_pUIBuilder
->setDeferredProperties();
483 std::unique_ptr
<VclBuilder
> m_pUIBuilder
;
485 friend class ::SalInstanceBuilder
;
486 friend class ::ScreenshotTest
;
490 * @return true if rValue is "True", "true", "1", etc.
492 bool toBool(std::u16string_view rValue
);
496 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */