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 .
19 #ifndef INCLUDED_SVX_SOURCE_INC_FMEXPL_HXX
20 #define INCLUDED_SVX_SOURCE_INC_FMEXPL_HXX
22 #include <config_options.h>
23 #include <svl/lstner.hxx>
24 #include <svl/SfxBroadcaster.hxx>
25 #include <vcl/window.hxx>
26 #include <sfx2/childwin.hxx>
27 #include <svl/poolitem.hxx>
28 #include <sfx2/bindings.hxx>
29 #include <sfx2/dockwin.hxx>
30 #include <sfx2/ctrlitem.hxx>
31 #include <vcl/image.hxx>
33 #include <com/sun/star/form/XForm.hpp>
34 #include <com/sun/star/form/XFormComponent.hpp>
35 #include <com/sun/star/beans/PropertyChangeEvent.hpp>
36 #include <com/sun/star/container/XContainerListener.hpp>
37 #include <com/sun/star/beans/XPropertyChangeListener.hpp>
38 #include <com/sun/star/container/XIndexContainer.hpp>
40 #include <svx/fmview.hxx>
45 #include <cppuhelper/implbase.hxx>
56 class FmNavInsertedHint
: public SfxHint
58 FmEntryData
* pEntryData
;
62 FmNavInsertedHint( FmEntryData
* pInsertedEntryData
, sal_uInt32 nRelPos
);
63 virtual ~FmNavInsertedHint() override
;
65 FmEntryData
* GetEntryData() const { return pEntryData
; }
66 sal_uInt32
GetRelPos() const { return nPos
; }
70 class FmNavModelReplacedHint
: public SfxHint
72 FmEntryData
* pEntryData
; // the data of the entry that has got a new model
75 FmNavModelReplacedHint( FmEntryData
* pAffectedEntryData
);
76 virtual ~FmNavModelReplacedHint() override
;
78 FmEntryData
* GetEntryData() const { return pEntryData
; }
82 class FmNavRemovedHint
: public SfxHint
84 FmEntryData
* pEntryData
;
87 FmNavRemovedHint( FmEntryData
* pInsertedEntryData
);
88 virtual ~FmNavRemovedHint() override
;
90 FmEntryData
* GetEntryData() const { return pEntryData
; }
94 class FmNavNameChangedHint
: public SfxHint
96 FmEntryData
* pEntryData
;
100 FmNavNameChangedHint( FmEntryData
* pData
, const OUString
& rNewName
);
101 virtual ~FmNavNameChangedHint() override
;
103 FmEntryData
* GetEntryData() const { return pEntryData
; }
104 const OUString
& GetNewName() const { return aNewName
; }
108 class FmNavClearedHint
: public SfxHint
112 virtual ~FmNavClearedHint() override
;
116 class FmNavViewMarksChanged
: public SfxHint
120 FmNavViewMarksChanged(FmFormView
* pWhichView
) { pView
= pWhichView
; }
122 const FmFormView
* GetAffectedView() const { return pView
; }
126 class FmEntryDataList
;
130 css::uno::Reference
< css::uno::XInterface
> m_xNormalizedIFace
;
131 css::uno::Reference
< css::beans::XPropertySet
> m_xProperties
;
132 css::uno::Reference
< css::container::XChild
> m_xChild
;
135 OUString m_aNormalImage
;
138 std::unique_ptr
<FmEntryDataList
>
140 FmEntryData
* pParent
;
143 void newObject( const css::uno::Reference
< css::uno::XInterface
>& _rxIFace
);
147 FmEntryData( FmEntryData
* pParentData
, const css::uno::Reference
< css::uno::XInterface
>& _rIFace
);
148 FmEntryData( const FmEntryData
& rEntryData
);
149 virtual ~FmEntryData();
151 void SetText( const OUString
& rText
){ aText
= rText
; }
152 void SetParent( FmEntryData
* pParentData
){ pParent
= pParentData
; }
154 const OUString
& GetNormalImage() const { return m_aNormalImage
; }
156 const OUString
& GetText() const { return aText
; }
157 FmEntryData
* GetParent() const { return pParent
; }
158 FmEntryDataList
* GetChildList() const { return pChildList
.get(); }
160 virtual bool IsEqualWithoutChildren( FmEntryData
* pEntryData
);
161 virtual std::unique_ptr
<FmEntryData
> Clone() = 0;
163 // note that the interface returned is normalized, i.e. querying the given XInterface of the object
164 // for XInterface must return the interface itself.
165 const css::uno::Reference
< css::uno::XInterface
>& GetElement() const
167 return m_xNormalizedIFace
;
170 const css::uno::Reference
< css::beans::XPropertySet
>& GetPropertySet() const
172 return m_xProperties
;
175 const css::uno::Reference
< css::container::XChild
>& GetChildIFace() const
182 class FmEntryDataList final
185 std::vector
< std::unique_ptr
<FmEntryData
> > maEntryDataList
;
191 FmEntryData
* at( size_t Index
)
192 { return maEntryDataList
.at(Index
).get(); }
194 size_t size() const { return maEntryDataList
.size(); }
195 void removeNoDelete( FmEntryData
* pItem
);
196 void insert( std::unique_ptr
<FmEntryData
> pItem
, size_t Index
);
201 // FmNavRequestSelectHint - someone tells the NavigatorTree to select certain entries
203 typedef std::set
<FmEntryData
*> FmEntryDataArray
;
205 class FmNavRequestSelectHint
: public SfxHint
207 FmEntryDataArray m_arredToSelect
;
208 bool m_bMixedSelection
;
210 FmNavRequestSelectHint()
211 : m_bMixedSelection(false)
215 void SetMixedSelection(bool bMixedSelection
) { m_bMixedSelection
= bMixedSelection
; }
216 bool IsMixedSelection() const { return m_bMixedSelection
; }
217 void AddItem(FmEntryData
* pEntry
) { m_arredToSelect
.insert(pEntry
); }
218 void ClearItems() { m_arredToSelect
.clear(); }
219 FmEntryDataArray
& GetItems() { return m_arredToSelect
; }
223 class FmFormData
: public FmEntryData
225 css::uno::Reference
< css::form::XForm
> m_xForm
;
228 FmFormData(const css::uno::Reference
< css::form::XForm
>& _rxForm
, FmFormData
* _pParent
);
229 FmFormData( const FmFormData
& rFormData
);
230 virtual ~FmFormData() override
;
232 const css::uno::Reference
< css::form::XForm
>& GetFormIface() const { return m_xForm
; }
234 virtual bool IsEqualWithoutChildren( FmEntryData
* pEntryData
) override
;
235 virtual std::unique_ptr
<FmEntryData
> Clone() override
;
239 class FmControlData
: public FmEntryData
241 css::uno::Reference
< css::form::XFormComponent
> m_xFormComponent
;
243 OUString
GetImage() const;
248 const css::uno::Reference
< css::form::XFormComponent
>& _rxComponent
,
251 FmControlData( const FmControlData
& rControlData
);
252 virtual ~FmControlData() override
;
254 const css::uno::Reference
< css::form::XFormComponent
>& GetFormComponent() const { return m_xFormComponent
; }
255 virtual bool IsEqualWithoutChildren( FmEntryData
* pEntryData
) override
;
256 virtual std::unique_ptr
<FmEntryData
> Clone() override
;
258 void ModelReplaced(const css::uno::Reference
< css::form::XFormComponent
>& _rxNew
);
266 class NavigatorTreeModel
;
268 class OFormComponentObserver final
269 :public ::cppu::WeakImplHelper
< css::beans::XPropertyChangeListener
270 , css::container::XContainerListener
273 ::svxform::NavigatorTreeModel
* m_pNavModel
;
278 OFormComponentObserver( ::svxform::NavigatorTreeModel
* pModel
);
280 // XEventListenerListener
281 virtual void SAL_CALL
disposing(const css::lang::EventObject
& Source
) override
;
283 // css::beans::XPropertyChangeListener
284 virtual void SAL_CALL
propertyChange(const css::beans::PropertyChangeEvent
& evt
) override
;
286 // css::container::XContainerListener
288 virtual void SAL_CALL
elementInserted(const css::container::ContainerEvent
& rEvent
) override
;
289 virtual void SAL_CALL
elementReplaced(const css::container::ContainerEvent
& rEvent
) override
;
290 virtual void SAL_CALL
elementRemoved(const css::container::ContainerEvent
& rEvent
) override
;
292 void Lock() { m_nLocks
++; }
293 void UnLock() { m_nLocks
--; }
294 bool IsLocked() const { return m_nLocks
!= 0; }
295 bool CanUndo() const { return m_bCanUndo
; }
296 void ReleaseModel() { m_pNavModel
= nullptr; }
298 void Insert(const css::uno::Reference
< css::uno::XInterface
>& xIface
, sal_Int32 nIndex
);
299 void Remove( const css::uno::Reference
< css::uno::XInterface
>& _rxElement
);
302 class NavigatorTreeModel
: public SfxBroadcaster
305 friend class NavigatorTree
;
306 friend class OFormComponentObserver
;
308 std::unique_ptr
<FmEntryDataList
>
310 FmFormShell
* m_pFormShell
;
311 FmFormPage
* m_pFormPage
;
312 FmFormModel
* m_pFormModel
;
313 rtl::Reference
<OFormComponentObserver
> m_pPropChangeList
;
315 void UpdateContent( const css::uno::Reference
< css::form::XForms
>& xForms
);
317 void InsertForm(const css::uno::Reference
< css::form::XForm
>& xForm
, sal_uInt32 nRelPos
);
318 void RemoveForm(FmFormData
const * pFormData
);
320 void InsertFormComponent(const css::uno::Reference
< css::form::XFormComponent
>& xComp
, sal_uInt32 nRelPos
);
321 void RemoveFormComponent(FmControlData
const * pControlData
);
322 void InsertSdrObj(const SdrObject
* pSdrObj
);
323 void RemoveSdrObj(const SdrObject
* pSdrObj
);
325 void ReplaceFormComponent(const css::uno::Reference
< css::form::XFormComponent
>& xOld
, const css::uno::Reference
< css::form::XFormComponent
>& xNew
);
327 void BroadcastMarkedObjects(const SdrMarkList
& mlMarked
);
328 // send a RequestSelectHint with the currently selected objects
329 bool InsertFormComponent(FmNavRequestSelectHint
& rHint
, SdrObject
* pObject
);
330 // is a helper for previous, manages the ... in SdrObjGroups;
331 // returns sal_True if the object is a FormComponent (or recursively consists only of such)
334 NavigatorTreeModel();
335 virtual ~NavigatorTreeModel() override
;
337 void FillBranch( FmFormData
* pParentData
);
338 void UpdateContent( FmFormShell
* pNewShell
);
340 void Insert(FmEntryData
* pEntryData
, sal_uInt32 nRelPos
= SAL_MAX_UINT32
,
341 bool bAlterModel
= false);
342 void Remove(FmEntryData
* pEntryData
, bool bAlterModel
= false);
344 static bool Rename( FmEntryData
* pEntryData
, const OUString
& rNewText
);
349 css::uno::Reference
< css::form::XForms
> GetForms() const;
350 FmFormShell
* GetFormShell() const { return m_pFormShell
; }
351 FmFormPage
* GetFormPage() const { return m_pFormPage
; }
352 FmEntryData
* FindData( const css::uno::Reference
< css::uno::XInterface
>& xElement
, FmEntryDataList
* pDataList
, bool bRecurs
=true );
353 FmEntryData
* FindData( const OUString
& rText
, FmFormData
const * pParentData
, bool bRecurs
);
354 FmEntryDataList
* GetRootList() const { return m_pRootList
.get(); }
355 static css::uno::Reference
< css::container::XIndexContainer
> GetFormComponents( FmFormData
const * pParentFormData
);
357 virtual void Notify( SfxBroadcaster
& rBC
, const SfxHint
& rHint
) override
;
362 class NavigatorTreeDropTarget
: public DropTargetHelper
365 NavigatorTree
& m_rTreeView
;
367 virtual sal_Int8
AcceptDrop( const AcceptDropEvent
& rEvt
) override
;
368 virtual sal_Int8
ExecuteDrop( const ExecuteDropEvent
& rEvt
) override
;
371 NavigatorTreeDropTarget(NavigatorTree
& rTreeView
);
374 typedef std::set
<std::unique_ptr
<weld::TreeIter
>> SvLBoxEntrySortedArray
;
376 class NavigatorTree final
: public SfxListener
378 std::unique_ptr
<weld::TreeView
> m_xTreeView
;
379 NavigatorTreeDropTarget m_aDropTargetHelper
;
381 enum SELDATA_ITEMS
{ SDI_DIRTY
, SDI_ALL
, SDI_NORMALIZED
, SDI_NORMALIZED_FORMARK
};
383 Timer m_aSynchronizeTimer
;
384 // the meta-data about my current selection
385 SvLBoxEntrySortedArray m_arrCurrentSelection
;
386 // the entries which, in the view, are currently marked as "cut" (painted semi-transparent)
387 ListBoxEntrySet m_aCutEntries
;
389 ::svxform::OControlExchangeHelper m_aControlExchange
;
391 std::unique_ptr
<NavigatorTreeModel
> m_pNavModel
;
392 std::unique_ptr
<weld::TreeIter
> m_xRootEntry
;
393 std::unique_ptr
<weld::TreeIter
> m_xEditEntry
;
395 ImplSVEvent
* nEditEvent
;
397 SELDATA_ITEMS m_sdiState
;
399 sal_uInt16 m_nSelectLock
;
400 sal_uInt16 m_nFormsSelected
;
401 sal_uInt16 m_nControlsSelected
;
402 sal_uInt16 m_nHiddenControls
; // (the number is included in m_nControlsSelected)
404 bool m_bDragDataDirty
: 1; // ditto
405 bool m_bPrevSelectionMixed
: 1;
406 bool m_bRootSelected
: 1;
407 bool m_bInitialUpdate
: 1; // am I the first time in the UpdateContent?
408 bool m_bKeyboardCut
: 1;
411 FmControlData
* NewControl(const OUString
& rServiceName
, const weld::TreeIter
& rParentEntry
, bool bEditName
);
412 void NewForm(const weld::TreeIter
& rParentEntry
);
413 std::unique_ptr
<weld::TreeIter
> Insert(FmEntryData
* pEntryData
, int nRelPos
);
414 void Remove( FmEntryData
* pEntryData
);
417 void CollectSelectionData(SELDATA_ITEMS sdiHow
);
418 // Collects the currently selected entries in m_arrCurrentSelection, normalizes the list if requested.
419 // - SDI_NORMALIZED simply means that all entries that already have a selected ancestor are not collected.
420 // - SDI_NORMALIZED_FORMARK means that the procedure is the same as for SDI_NORMALIZED,
421 // but entries whose direct parent is not selected are collected (independent of the
422 // status of further ancestors). The same applies for forms that are selected,
423 // regardless of the status of any ancestors.
424 // For both normalized modes, the m_nFormsSelected, ... contain the correct number,
425 // even if not all of these entries end up in m_arrCurrentSelection.
426 // SDI_DIRTY is of course not allowed as a parameter.
428 // a single interface for all selected entries
429 void ShowSelectionProperties(bool bForce
= false);
430 // delete all selected elements
431 void DeleteSelection();
433 void SynchronizeSelection(FmEntryDataArray
& arredToSelect
);
434 // after calling this method, exactly the entries marked in the array are selected
435 void SynchronizeSelection();
436 // makes the same, takes the MarkList of the View
437 void SynchronizeMarkList();
438 // reverse direction of SynchronizeMarkList: selects in the view all controls corresponding to the current selection
440 void CollectObjects(FmFormData
const * pFormData
, bool bDeep
, ::std::set
< css::uno::Reference
< css::form::XFormComponent
> >& _rObjects
);
442 // in the Select I usually update the Marklist of the corresponding view,
443 // with the following functions I can control the locking of this behavior
444 void LockSelectionHandling() { ++m_nSelectLock
; }
445 void UnlockSelectionHandling() { --m_nSelectLock
; }
446 bool IsSelectionHandlingLocked() const { return m_nSelectLock
>0; }
448 bool IsEditingActive() const { return m_bEditing
; }
450 static bool IsHiddenControl(FmEntryData
const * pEntryData
);
452 DECL_LINK( KeyInputHdl
, const KeyEvent
&, bool );
453 DECL_LINK( PopupMenuHdl
, const CommandEvent
&, bool );
455 DECL_LINK(EditingEntryHdl
, const weld::TreeIter
&, bool);
456 typedef std::pair
<const weld::TreeIter
&, OUString
> IterString
;
457 DECL_LINK(EditedEntryHdl
, const IterString
&, bool);
459 DECL_LINK( OnEdit
, void*, void );
461 DECL_LINK( OnEntrySelDesel
, weld::TreeView
&, void );
462 DECL_LINK( OnSynchronizeTimer
, Timer
*, void );
464 DECL_LINK( OnClipboardAction
, OLocalExchange
&, void );
466 DECL_LINK( DragBeginHdl
, bool&, bool );
469 NavigatorTree(std::unique_ptr
<weld::TreeView
> xTreeView
);
470 virtual ~NavigatorTree() override
;
473 void UpdateContent( FmFormShell
* pFormShell
);
474 void MarkViewObj( FmFormData
const * pFormData
, bool bDeep
);
475 void MarkViewObj( FmControlData
const * pControlData
);
476 void UnmarkAllViewObj();
478 void GrabFocus() { m_xTreeView
->grab_focus(); }
480 bool IsFormEntry(const weld::TreeIter
& rEntry
);
481 bool IsFormComponentEntry(const weld::TreeIter
& rEntry
);
483 OUString
GenerateName( FmEntryData
const * pEntryData
);
485 NavigatorTreeModel
* GetNavModel() const { return m_pNavModel
.get(); }
486 std::unique_ptr
<weld::TreeIter
> FindEntry(FmEntryData
* pEntryData
);
488 virtual void Notify( SfxBroadcaster
& rBC
, const SfxHint
& rHint
) override
;
490 weld::TreeView
& get_widget() { return *m_xTreeView
; }
492 sal_Int8
AcceptDrop(const AcceptDropEvent
& rEvt
);
493 sal_Int8
ExecuteDrop(const ExecuteDropEvent
& rEvt
);
496 sal_Int8
implAcceptDataTransfer( const DataFlavorExVector
& _rFlavors
, sal_Int8 _nAction
, const weld::TreeIter
* _pTargetEntry
, bool _bDnD
);
498 sal_Int8
implExecuteDataTransfer( const OControlTransferData
& _rData
, sal_Int8 _nAction
, const Point
& _rDropPos
, bool _bDnD
);
499 sal_Int8
implExecuteDataTransfer( const OControlTransferData
& _rData
, sal_Int8 _nAction
, const weld::TreeIter
* _pTargetEntry
, bool _bDnD
);
501 // check if a cut, copy, or drag operation can be started in the current situation
502 bool implAllowExchange( sal_Int8 _nAction
, bool* _pHasNonHidden
= nullptr );
503 // check if a paste with the current clipboard content can be accepted
504 bool implAcceptPaste( );
506 // fills m_aControlExchange in preparation of a DnD or clipboard operation
507 bool implPrepareExchange( sal_Int8 _nAction
);
509 void ModelHasRemoved(const weld::TreeIter
* _pEntry
);
515 bool doingKeyboardCut( ) const { return m_bKeyboardCut
; }
518 class NavigatorFrame
: public SfxDockingWindow
, public SfxControllerItem
521 std::unique_ptr
<NavigatorTree
> m_xNavigatorTree
;
524 virtual bool Close() override
;
525 virtual void GetFocus() override
;
526 virtual Size
CalcDockingSize( SfxChildAlignment
) override
;
527 virtual SfxChildAlignment
CheckAlignment( SfxChildAlignment
, SfxChildAlignment
) override
;
529 using SfxDockingWindow::StateChanged
;
532 NavigatorFrame( SfxBindings
*pBindings
, SfxChildWindow
*pMgr
,
533 vcl::Window
* pParent
);
534 virtual ~NavigatorFrame() override
;
535 virtual void dispose() override
;
537 void UpdateContent( FmFormShell
* pFormShell
);
538 void StateChanged( sal_uInt16 nSID
, SfxItemState eState
, const SfxPoolItem
* pState
) override
;
539 void FillInfo( SfxChildWinInfo
& rInfo
) const override
;
542 class UNLESS_MERGELIBS(SVXCORE_DLLPUBLIC
) NavigatorFrameManager
: public SfxChildWindow
545 SVX_DLLPRIVATE
NavigatorFrameManager( vcl::Window
*pParent
, sal_uInt16 nId
, SfxBindings
*pBindings
,
546 SfxChildWinInfo
*pInfo
);
547 SFX_DECL_CHILDWINDOW( NavigatorFrameManager
);
551 #endif // INCLUDED_SVX_SOURCE_INC_FMEXPL_HXX
553 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */