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 .
22 #include "cloneable.hxx"
23 #include "propertybaghelper.hxx"
24 #include "resettable.hxx"
25 #include "windowstateguard.hxx"
27 #include <com/sun/star/awt/XControl.hpp>
28 #include <com/sun/star/beans/XPropertyAccess.hpp>
29 #include <com/sun/star/beans/XPropertyContainer.hpp>
30 #include <com/sun/star/container/XNamed.hpp>
31 #include <com/sun/star/form/binding/XBindableValue.hpp>
32 #include <com/sun/star/form/validation/XValidatableFormComponent.hpp>
33 #include <com/sun/star/form/validation/XValidityConstraintListener.hpp>
34 #include <com/sun/star/form/XBoundComponent.hpp>
35 #include <com/sun/star/form/XBoundControl.hpp>
36 #include <com/sun/star/form/XFormComponent.hpp>
37 #include <com/sun/star/form/XLoadListener.hpp>
38 #include <com/sun/star/form/XReset.hpp>
39 #include <com/sun/star/io/XPersistObject.hpp>
40 #include <com/sun/star/lang/XEventListener.hpp>
41 #include <com/sun/star/lang/XServiceInfo.hpp>
42 #include <com/sun/star/sdb/XColumn.hpp>
43 #include <com/sun/star/sdb/XColumnUpdate.hpp>
44 #include <com/sun/star/sdb/XRowSetChangeListener.hpp>
45 #include <com/sun/star/sdbc/XRowSet.hpp>
46 #include <com/sun/star/uno/XAggregation.hpp>
47 #include <com/sun/star/uno/XComponentContext.hpp>
48 #include <com/sun/star/util/XCloneable.hpp>
49 #include <com/sun/star/util/XModifyListener.hpp>
50 #include <com/sun/star/form/XLoadable.hpp>
52 #include <comphelper/interfacecontainer3.hxx>
53 #include <comphelper/propagg.hxx>
54 #include <comphelper/propmultiplex.hxx>
55 #include <comphelper/uno3.hxx>
56 #include <cppuhelper/component.hxx>
57 #include <cppuhelper/implbase1.hxx>
58 #include <cppuhelper/implbase2.hxx>
59 #include <cppuhelper/implbase3.hxx>
60 #include <cppuhelper/implbase4.hxx>
61 #include <cppuhelper/implbase7.hxx>
62 #include <cppuhelper/propshlp.hxx>
63 #include <osl/mutex.hxx>
64 #include <rtl/ustring.hxx>
71 // default tab index for components
72 const sal_Int16 FRM_DEFAULT_TABINDEX
= 0;
79 /** class whose instances lock an OControlModel
81 Locking here merely means locking the OControlModel's mutex.
83 In addition to the locking facility, the class is also able to fire property
84 change notifications. This happens when the last ControlModelLock instance on a stack
87 class ControlModelLock
90 ControlModelLock( OControlModel
& _rModel
)
102 inline void acquire();
103 inline void release();
105 OControlModel
& getModel() const { return m_rModel
; };
107 /** adds a property change notification, which is to be fired when the last lock on the model
108 (in the current thread) is released.
110 void addPropertyNotification(
111 const sal_Int32 _nHandle
,
112 const css::uno::Any
& _rOldValue
,
113 const css::uno::Any
& _rNewValue
117 void impl_notifyAll_nothrow();
119 OControlModel
& m_rModel
;
121 std::vector
< sal_Int32
> m_aHandles
;
122 std::vector
< css::uno::Any
> m_aOldValues
;
123 std::vector
< css::uno::Any
> m_aNewValues
;
125 ControlModelLock( const ControlModelLock
& ) = delete;
126 ControlModelLock
& operator=( const ControlModelLock
& ) = delete;
131 //= base class for form layer controls
133 typedef ::cppu::ImplHelper3
< css::awt::XControl
134 , css::lang::XEventListener
135 , css::lang::XServiceInfo
138 class OControl
:public ::cppu::OComponentHelper
139 ,public OControl_BASE
142 ::osl::Mutex m_aMutex
;
143 css::uno::Reference
< css::awt::XControl
> m_xControl
;
144 css::uno::Reference
< css::uno::XAggregation
>
147 css::uno::Reference
< css::uno::XComponentContext
>
149 WindowStateGuard m_aWindowStateGuard
;
152 /** constructs a control
155 the service factory for this control
156 @param _rAggregateService
157 the service name of the component to aggregate
158 @param _bSetDelegator
159 set this to <FALSE/> if you don't want the constructor to set the delegator at
160 the aggregate. In this case, you <em>have</em> to call doSetDelegator within your
163 This is helpful, if your derived class wants to cache an interface of the aggregate.
164 In this case, the aggregate needs to be queried for this interface <b>before</b> the
165 <member scope="css::uno">XAggregation::setDelegator</member> call.
167 In such a case, pass <FALSE/> to this parameter. Then, cache the aggregate's interface(s)
168 as needed. Afterwards, call <member>doSetDelegator</member>.
170 In your destructor, you need to call <member>doResetDelegator</member> before
171 resetting the cached interfaces. This will reset the aggregates delegator to <NULL/>,
172 which will ensure that the <member scope="css::uno">XInterface::release</member>
173 calls on the cached interfaces are really applied to the aggregate, instead of
174 the <type>OControl</type> itself.
177 const css::uno::Reference
< css::uno::XComponentContext
>& _rFactory
,
178 const OUString
& _rAggregateService
,
179 const bool _bSetDelegator
= true
183 virtual ~OControl() override
;
185 /** sets the control as delegator at the aggregate
187 This has to be called from within your derived class' constructor, if and only
188 if you passed <FALSE/> to the <arg>_bSetDelegator</arg> parameter of the
189 <type>OControl</type> constructor.
191 void doSetDelegator();
192 void doResetDelegator();
195 DECLARE_UNO3_AGG_DEFAULTS(OControl
, OComponentHelper
)
196 virtual css::uno::Any SAL_CALL
queryAggregation( const css::uno::Type
& _rType
) override
;
199 virtual css::uno::Sequence
<sal_Int8
> SAL_CALL
getImplementationId() override
;
200 virtual css::uno::Sequence
< css::uno::Type
> SAL_CALL
getTypes() override
;
203 virtual void SAL_CALL
disposing() override
;
205 // XComponent (as base of XControl)
206 virtual void SAL_CALL
dispose( ) override
207 { OComponentHelper::dispose(); }
208 virtual void SAL_CALL
addEventListener( const css::uno::Reference
< css::lang::XEventListener
>& _rxListener
) override
209 { OComponentHelper::addEventListener(_rxListener
); }
210 virtual void SAL_CALL
removeEventListener( const css::uno::Reference
< css::lang::XEventListener
>& _rxListener
) override
211 { OComponentHelper::removeEventListener(_rxListener
); }
214 virtual void SAL_CALL
disposing(const css::lang::EventObject
& Source
) override
;
217 virtual sal_Bool SAL_CALL
supportsService(const OUString
& ServiceName
) override
;
218 virtual css::uno::Sequence
<OUString
> SAL_CALL
getSupportedServiceNames() override
;
219 virtual OUString SAL_CALL
getImplementationName() override
= 0;
222 virtual void SAL_CALL
setContext(const css::uno::Reference
<css::uno::XInterface
>& Context
) override
;
223 virtual css::uno::Reference
<css::uno::XInterface
> SAL_CALL
getContext() override
;
224 virtual void SAL_CALL
createPeer(const css::uno::Reference
<css::awt::XToolkit
>& Toolkit
, const css::uno::Reference
<css::awt::XWindowPeer
>& Parent
) override
;
225 virtual css::uno::Reference
<css::awt::XWindowPeer
> SAL_CALL
getPeer() override
;
226 virtual sal_Bool SAL_CALL
setModel(const css::uno::Reference
<css::awt::XControlModel
>& Model
) override
;
227 virtual css::uno::Reference
<css::awt::XControlModel
> SAL_CALL
getModel() override
;
228 virtual css::uno::Reference
<css::awt::XView
> SAL_CALL
getView() override
;
229 virtual void SAL_CALL
setDesignMode(sal_Bool bOn
) override
;
230 virtual sal_Bool SAL_CALL
isDesignMode() override
;
231 virtual sal_Bool SAL_CALL
isTransparent() override
;
234 virtual css::uno::Sequence
< css::uno::Type
> _getTypes();
235 // overwrite this and call the base class if you have additional types
237 css::uno::Sequence
< OUString
> getAggregateServiceNames() const;
240 void impl_resetStateGuard_nothrow();
243 // a form control implementing the XBoundControl interface
244 typedef ::cppu::ImplHelper1
< css::form::XBoundControl
245 > OBoundControl_BASE
;
246 class OBoundControl
:public OControl
247 ,public OBoundControl_BASE
253 const css::uno::Reference
< css::uno::XComponentContext
>& _rxContext
,
254 const OUString
& _rAggregateService
,
255 const bool _bSetDelegator
= true
258 virtual ~OBoundControl() override
;
260 DECLARE_UNO3_AGG_DEFAULTS(OBoundControl
, OControl
)
261 virtual css::uno::Any SAL_CALL
queryAggregation( const css::uno::Type
& _rType
) override
;
264 virtual sal_Bool SAL_CALL
getLock() override
;
265 virtual void SAL_CALL
setLock(sal_Bool _bLock
) override
;
266 // default implementation just disables the controls, overwrite _setLock to change this behaviour
269 virtual sal_Bool SAL_CALL
setModel(const css::uno::Reference
< css::awt::XControlModel
>& Model
) override
;
272 virtual void SAL_CALL
disposing(const css::lang::EventObject
& Source
) override
;
275 virtual void SAL_CALL
disposing() override
;
278 virtual css::uno::Sequence
< css::uno::Type
> _getTypes() override
;
279 // implement the lock setting
280 void _setLock(bool _bLock
);
285 //= model of a form layer control
287 //added for exporting OCX control
288 #define INVALID_OBJ_ID_IN_MSO 0xFFFF
290 typedef ::cppu::ImplHelper7
< css::form::XFormComponent
291 , css::io::XPersistObject
292 , css::container::XNamed
293 , css::lang::XServiceInfo
294 , css::util::XCloneable
295 , css::beans::XPropertyContainer
296 , css::beans::XPropertyAccess
297 > OControlModel_BASE
;
299 class OControlModel
:public ::cppu::OComponentHelper
300 ,public comphelper::OPropertySetAggregationHelper
301 ,public OControlModel_BASE
302 ,public OCloneableAggregation
303 ,public IPropertyBagHelperContext
307 css::uno::Reference
<css::uno::XComponentContext
> m_xContext
;
309 ::osl::Mutex m_aMutex
;
310 oslInterlockedCount m_lockCount
;
312 css::uno::Reference
<css::uno::XInterface
> m_xParent
; // ParentComponent
313 PropertyBagHelper m_aPropertyBagHelper
;
315 const css::uno::Reference
<css::uno::XComponentContext
>&
316 getContext() const { return m_xContext
; }
319 OUString m_aName
; // name of the control
320 OUString m_aTag
; // tag for additional data
321 sal_Int16 m_nTabIndex
; // index within the taborder
322 sal_Int16 m_nClassId
; // type of the control
323 bool m_bNativeLook
; // should the control use the native platform look?
324 bool m_bStandardTheme
; // should the default control colors be 'standard' or use the native platform theme?
325 bool m_bGenerateVbEvents
; // should the control generate fake vba events
326 //added for exporting OCX control
327 sal_Int16 m_nControlTypeinMSO
; //keep the MS office control type for exporting to MS binary file
328 sal_uInt16 m_nObjIDinMSO
; //keep the OCX control obj id for exporting to MS binary file
334 const css::uno::Reference
< css::uno::XComponentContext
>& _rFactory
, // factory to create the aggregate with
335 const OUString
& _rUnoControlModelTypeName
, // service name of te model to aggregate
336 const OUString
& rDefault
= OUString(), // service name of the default control
337 const bool _bSetDelegator
= true // set to sal_False if you want to call setDelegator later (after returning from this ctor)
340 const OControlModel
* _pOriginal
, // the original object to clone
341 const css::uno::Reference
< css::uno::XComponentContext
>& _rFactory
, // factory to create the aggregate with
342 const bool _bCloneAggregate
= true, // should the aggregate of the original be cloned, too?
343 const bool _bSetDelegator
= true // set to sal_False if you want to call setDelegator later (after returning from this ctor)
345 virtual ~OControlModel() override
;
347 /** to be called after an OBoundControlModel (a derivee, respectively) has been cloned
349 <p>This method contains late initializations which cannot be done in the
350 constructor of this base class, since the virtual method of derived classes do
351 not yet work there.</p>
353 virtual void clonedFrom( const OControlModel
* _pOriginal
);
355 using OComponentHelper::rBHelper
;
357 virtual css::uno::Sequence
< css::uno::Type
> _getTypes();
359 void readHelpTextCompatibly(const css::uno::Reference
< css::io::XObjectInputStream
>& _rxInStream
);
360 void writeHelpTextCompatibly(const css::uno::Reference
< css::io::XObjectOutputStream
>& _rxOutStream
);
362 void doSetDelegator();
363 void doResetDelegator();
365 css::uno::Sequence
< OUString
> getAggregateServiceNames() const;
368 DECLARE_UNO3_AGG_DEFAULTS(OControl
, OComponentHelper
)
369 virtual css::uno::Any SAL_CALL
queryAggregation( const css::uno::Type
& _rType
) override
;
372 virtual css::uno::Sequence
<sal_Int8
> SAL_CALL
getImplementationId() override
;
373 virtual css::uno::Sequence
< css::uno::Type
> SAL_CALL
getTypes() override
;
376 virtual void SAL_CALL
disposing() override
;
379 virtual OUString SAL_CALL
getName() override
;
380 virtual void SAL_CALL
setName(const OUString
& aName
) override
;
383 virtual sal_Bool SAL_CALL
supportsService(const OUString
& ServiceName
) override
;
384 virtual css::uno::Sequence
<OUString
> SAL_CALL
getSupportedServiceNames() override
;
385 virtual OUString SAL_CALL
getImplementationName() override
= 0;
387 // XServiceInfo - static version(s)
388 /// @throws css::uno::RuntimeException
389 static css::uno::Sequence
<OUString
> getSupportedServiceNames_Static();
392 virtual OUString SAL_CALL
getServiceName() override
= 0;
393 virtual void SAL_CALL
394 write(const css::uno::Reference
< css::io::XObjectOutputStream
>& _rxOutStream
) override
;
395 virtual void SAL_CALL
396 read(const css::uno::Reference
< css::io::XObjectInputStream
>& _rxInStream
) override
;
398 // XChild (base of XFormComponent)
399 virtual css::uno::Reference
<css::uno::XInterface
> SAL_CALL
getParent() override
;
400 virtual void SAL_CALL
setParent(const css::uno::Reference
<css::uno::XInterface
>& Parent
) override
;
403 virtual void SAL_CALL
disposing(const css::lang::EventObject
& Source
) override
;
406 virtual void SAL_CALL
getFastPropertyValue(css::uno::Any
& rValue
, sal_Int32 nHandle
) const override
;
407 virtual sal_Bool SAL_CALL
convertFastPropertyValue(
408 css::uno::Any
& _rConvertedValue
, css::uno::Any
& _rOldValue
, sal_Int32 _nHandle
, const css::uno::Any
& _rValue
) override
;
409 virtual void SAL_CALL
setFastPropertyValue_NoBroadcast( sal_Int32 nHandle
, const css::uno::Any
& rValue
) override
;
410 using ::cppu::OPropertySetHelper::getFastPropertyValue
;
412 // css::beans::XPropertyState
413 virtual css::beans::PropertyState
getPropertyStateByHandle(sal_Int32 nHandle
) override
;
414 virtual void setPropertyToDefaultByHandle(sal_Int32 nHandle
) override
;
415 virtual css::uno::Any
getPropertyDefaultByHandle( sal_Int32 nHandle
) const override
;
418 virtual css::uno::Reference
< css::util::XCloneable
> SAL_CALL
createClone( ) override
= 0;
420 // XPropertyContainer
421 virtual void SAL_CALL
addProperty( const OUString
& Name
, ::sal_Int16 Attributes
, const css::uno::Any
& DefaultValue
) override
;
422 virtual void SAL_CALL
removeProperty( const OUString
& Name
) override
;
425 virtual css::uno::Sequence
< css::beans::PropertyValue
> SAL_CALL
getPropertyValues( ) override
;
426 virtual void SAL_CALL
setPropertyValues( const css::uno::Sequence
< css::beans::PropertyValue
>& aProps
) override
;
429 using OPropertySetAggregationHelper::setPropertyValues
;
430 using OPropertySetAggregationHelper::getPropertyValues
;
433 virtual void writeAggregate( const css::uno::Reference
< css::io::XObjectOutputStream
>& _rxOutStream
) const;
434 virtual void readAggregate( const css::uno::Reference
< css::io::XObjectInputStream
>& _rxInStream
);
438 virtual css::uno::Reference
< css::beans::XPropertySetInfo
> SAL_CALL
getPropertySetInfo() override
;
439 // OPropertySetHelper
440 virtual cppu::IPropertyArrayHelper
& SAL_CALL
getInfoHelper() override
;
442 /** describes the properties provided by this class, or its respective
445 Derived classes usually call the base class first, and then append own properties.
447 virtual void describeFixedProperties(
448 css::uno::Sequence
< css::beans::Property
>& /* [out] */ _rProps
451 // IPropertyBagHelperContext
452 virtual ::osl::Mutex
& getMutex() override
;
453 virtual void describeFixedAndAggregateProperties(
454 css::uno::Sequence
< css::beans::Property
>& _out_rFixedProperties
,
455 css::uno::Sequence
< css::beans::Property
>& _out_rAggregateProperties
457 virtual css::uno::Reference
< css::beans::XMultiPropertySet
>
458 getPropertiesInterface() override
;
460 /** describes the properties of our aggregate
462 The default implementation simply asks m_xAggregateSet for its properties.
464 You usually only need to override this method if you want to filter the
465 aggregate properties.
467 virtual void describeAggregateProperties(
468 css::uno::Sequence
< css::beans::Property
>& /* [out] */ _rAggregateProps
472 struct LockAccess
{ friend class ControlModelLock
; private: LockAccess() { } };
474 void lockInstance( LockAccess
);
475 oslInterlockedCount
unlockInstance( LockAccess
);
477 void firePropertyChanges(
478 const std::vector
< sal_Int32
>& _rHandles
,
479 const std::vector
< css::uno::Any
>& _rOldValues
,
480 const std::vector
< css::uno::Any
>& _rNewValues
,
485 getInstanceMutex() { return m_aMutex
; }
488 //= OBoundControlModel
489 //= model of a form layer control which is bound to a data source field
491 typedef ::cppu::ImplHelper4
< css::form::XLoadListener
493 , css::beans::XPropertyChangeListener
494 , css::sdb::XRowSetChangeListener
495 > OBoundControlModel_BASE1
;
497 // separated into an own base class since derivees can disable the support for this
498 // interface, thus we want to easily exclude it in the queryInterface and getTypes
499 typedef ::cppu::ImplHelper1
< css::form::XBoundComponent
500 > OBoundControlModel_COMMITTING
;
503 typedef ::cppu::ImplHelper2
< css::form::binding::XBindableValue
504 , css::util::XModifyListener
505 > OBoundControlModel_BINDING
;
508 typedef ::cppu::ImplHelper2
< css::form::validation::XValidityConstraintListener
509 , css::form::validation::XValidatableFormComponent
510 > OBoundControlModel_VALIDATION
;
512 class OBoundControlModel
:public OControlModel
513 ,public OBoundControlModel_BASE1
514 ,public OBoundControlModel_COMMITTING
515 ,public OBoundControlModel_BINDING
516 ,public OBoundControlModel_VALIDATION
517 ,public ::comphelper::OPropertyChangeListener
520 enum ValueChangeInstigator
528 css::uno::Reference
< css::beans::XPropertySet
>
530 // the form which controls supplies the field we bind to.
531 css::uno::Reference
< css::form::XLoadable
>
534 OUString m_sValuePropertyName
;
535 sal_Int32 m_nValuePropertyAggregateHandle
;
536 sal_Int32 m_nFieldType
;
537 css::uno::Type m_aValuePropertyType
;
538 bool m_bValuePropertyMayBeVoid
;
540 ResetHelper m_aResetHelper
;
541 ::comphelper::OInterfaceContainerHelper3
<css::form::XUpdateListener
>
543 ::comphelper::OInterfaceContainerHelper3
<css::form::validation::XFormComponentValidityListener
>
544 m_aFormComponentListeners
;
546 css::uno::Reference
< css::form::binding::XValueBinding
>
548 css::uno::Reference
< css::form::validation::XValidator
>
550 css::uno::Type m_aExternalValueType
;
553 OUString m_aControlSource
; // data source, name of the field
554 css::uno::Reference
< css::beans::XPropertySet
>
555 m_xLabelControl
; // reference to a sibling control (model) which is our label
556 bool m_bInputRequired
;
559 rtl::Reference
<::comphelper::OPropertyChangeMultiplexer
>
560 m_pAggPropMultiplexer
;
562 bool m_bFormListening
: 1; // are we currently a XLoadListener at our ambient form?
564 bool m_bRequired
: 1;
565 const bool m_bCommitable
: 1; // do we support XBoundComponent?
566 const bool m_bSupportsExternalBinding
: 1; // do we support XBindableValue?
567 const bool m_bSupportsValidation
: 1; // do we support XValidatable?
568 bool m_bForwardValueChanges
: 1; // do we currently handle changes in the bound database field?
569 bool m_bTransferringValue
: 1; // true if we're currently transferring our value to an external binding
570 bool m_bIsCurrentValueValid
: 1; // flag specifying whether our current value is valid, relative to our external validator
571 bool m_bBindingControlsRO
: 1; // is our ReadOnly property currently controlled by our external binding?
572 bool m_bBindingControlsEnable
: 1; // is our Enabled property currently controlled by our external binding?
574 ValueChangeInstigator m_eControlValueChangeInstigator
;
577 OUString m_aLabelServiceName
;
578 // when setting the label for our control (property FM_PROP_CONTROLLABEL, member m_xLabelControl),
579 // we accept only objects supporting an XControlModel interface, an XServiceInfo interface and
580 // support for a service (XServiceInfo::supportsService) determined by this string.
581 // Any other arguments will throw an IllegalArgumentException.
582 // The default value is FM_COMPONENT_FIXEDTEXT.
584 css::uno::Reference
< css::sdbc::XRowSet
>
586 css::uno::Reference
< css::sdb::XColumnUpdate
>
588 css::uno::Reference
< css::sdb::XColumn
>
592 sal_Int32
getValuePropertyAggHandle( ) const { return m_nValuePropertyAggregateHandle
; }
593 const OUString
& getControlSource( ) const { return m_aControlSource
; }
594 bool isRequired() const { return m_bRequired
; }
595 bool isLoaded() const { return m_bLoaded
; }
600 const css::uno::Reference
< css::uno::XComponentContext
>& _rxContext
,
601 // factory to create the aggregate with
602 const OUString
& _rUnoControlModelTypeName
, // service name of te model to aggregate
603 const OUString
& _rDefault
, // service name of the default control
604 const bool _bCommitable
, // is the control (model) committable?
605 const bool _bSupportExternalBinding
, // set to sal_True if you want to support XBindableValue
606 const bool _bSupportsValidation
// set to sal_True if you want to support XValidatable
609 const OBoundControlModel
* _pOriginal
, // the original object to clone
610 const css::uno::Reference
< css::uno::XComponentContext
>& _rxContext
611 // factory to create the aggregate with
613 virtual ~OBoundControlModel() override
;
615 /// late ctor after cloning
616 virtual void clonedFrom( const OControlModel
* _pOriginal
) override
;
618 /** initializes the part of the class which is related to the control value.
620 <p>Kind of late ctor, to be called for derivees which have a dedicated value property.<br/>
621 The value property is the property which's value is synced with either the database
622 column the object is bound to, or with the external value binding, if present.<br/>
623 E.g. for a text control model, this property will most probably be "Text".</p>
625 <p>Derived classes are strongly recommended to call this method - at least the
626 "DataFieldProperty" (exposed in getFastPropertyValue) relies on the information
627 given herein, and needs to be supplied otherwise else.</p>
629 <p>If this method has been called properly, then <member>setControlValue</member>
630 does not need to be overridden - it will simply set the property value at the
634 The method has not be called before during the life time of the object.
636 @param _rValuePropertyName
637 the name of the value property
638 @param _nValuePropertyExternalHandle
639 the handle of the property, as exposed to external components.<br/>
640 Normally, this information can be obtained dynamically (e.g. from describeFixedProperties),
641 but since this method is to be called from within the constructor of derived classes,
642 we prefer to be on the *really* safe side here...
645 @see suspendValueListening
646 @see resumeValueListening
647 @see describeFixedProperties
649 void initValueProperty(
650 const OUString
& _rValuePropertyName
,
651 sal_Int32 _nValuePropertyExternalHandle
654 /** initializes the part of the class which is related to the control value.
656 <p>In opposite to ->initValueProperty, this method is to be used for value properties which are <em>not</em>
657 implemented by our aggregate, but by ourselves.</p>
659 <p>Certain functionality is not available when using own value properties. This includes binding to an external
660 value and external validation. (This is not a conceptual limit, but simply missing implementation.)</p>
662 void initOwnValueProperty(
663 const OUString
& i_rValuePropertyName
666 /** suspends listening at the value property
668 <p>As long as this listening is suspended, changes in the value property will not be
669 recognized and not be handled.</p>
671 @see initValueProperty
672 @see resumeValueListening
674 void suspendValueListening( );
676 /** resumes listening at the value property
678 <p>As long as this listening is suspended, changes in the value property will not be
679 recognized and not be handled.</p>
682 listening at the value property is currently suspended
684 @see initValueProperty
685 @see resumeValueListening
687 void resumeValueListening( );
689 /** (to be) called when the value property changed
691 Normally, this is done automatically, since the value property is a property of our aggregate, and we're
692 a listener at this property.
693 However, in some cases the value property might not be an aggregate property, but a property of the
694 delegator instance. In this case, you'll need to call <code>onValuePropertyChange</code> whenever this
697 void onValuePropertyChange( ControlModelLock
& i_rControLock
);
699 /** starts listening at the aggregate, for changes in the given property
701 <p>The OBoundControlModel automatically registers a multiplexer which listens for
702 changes in the aggregate property values. By default, only the control value property
703 is observed. You may add additional properties to be observed with this method.</p>
705 @see initValueProperty
706 @see _propertyChanged
708 void startAggregatePropertyListening( const OUString
& _rPropertyName
);
710 /** returns the default which should be used when resetting the control
712 <p>The default implementation returns an empty Any.</p>
714 @see resetNoBroadcast
716 virtual css::uno::Any
717 getDefaultForReset() const;
719 /** translates a db column value into a control value.
721 <p>Must transform the very current value of the database column we're bound to
722 (<member>m_xColumn</member>) into a value which can be used as current value
728 virtual css::uno::Any
729 translateDbColumnToControlValue( ) = 0;
731 /** returns the data types which the control could use to exchange data with
732 an external value binding
734 The types returned here are completely independent from the concrete value binding,
735 they're just candidates which depend on the control type, and possible the concrete state
736 of the control (i.e. some property value).
738 If a control implementation supports multiple types, the ordering in the returned
739 sequence indicates preference: Preferred types are mentioned first.
741 The default implementation returns the type of our value property.
743 virtual css::uno::Sequence
< css::uno::Type
>
744 getSupportedBindingTypes();
746 /** translates the given value, which was obtained from the current external value binding,
747 to a value which can be used in setControlValue
749 <p>The default implementation returns the value itself, exception when it is VOID, and
750 our value property is not allowed to be void - in this case, the returned value is a
751 default-constructed value of the type required by our value property.
753 @see hasExternalValueBinding
754 @see getExternalValueType
756 virtual css::uno::Any
757 translateExternalValueToControlValue( const css::uno::Any
& _rExternalValue
) const;
759 /** commits the current control value to our external value binding
761 <p>The default implementation simply calls getControlValue.</p>
763 @see hasExternalValueBinding
764 @see initValueProperty
766 virtual css::uno::Any
767 translateControlValueToExternalValue( ) const;
769 /** commits the current control value to the database column we're bound to
771 we're properly bound to a database column, especially <member>m_xColumnUpdate</member>
774 <TRUE/> if and only if the current control value results from a reset (<member>getDefaultForReset</member>)
777 virtual bool commitControlValueToDbColumn(
781 /** sets the given value as new current value for the control
783 Besides some administrative work (such as caring for <member>m_eControlValueChangeInstigator</member>),
784 this method simply calls <member>doSetControlValue</member>.
787 Our own mutex is locked.
789 The value to set. This value is guaranteed to be created by
790 <member>translateDbColumnToControlValue</member> or
791 <member>translateExternalValueToControlValue</member>
793 the instigator of the value change
795 void setControlValue(
796 const css::uno::Any
& _rValue
,
797 ValueChangeInstigator _eInstigator
800 <p>The default implementation will forward the given value to the aggregate, using
801 m_nValuePropertyAggregateHandle and/or m_sValuePropertyName.</p>
804 Our own mutex is locked.
806 The value to set. This value is guaranteed to be created by
807 <member>translateDbColumnToControlValue</member> or
808 <member>translateExternalValueToControlValue</member>
810 virtual void doSetControlValue(
811 const css::uno::Any
& _rValue
814 /** retrieves the current value of the control
816 <p>The default implementation will ask the aggregate for the property value
817 determined by either m_nValuePropertyAggregateHandle and/or m_sValuePropertyName.</p>
820 Our own mutex is locked.
822 virtual css::uno::Any
823 getControlValue( ) const;
825 /** called whenever a connection to a database column has been established
827 virtual void onConnectedDbColumn( const css::uno::Reference
< css::uno::XInterface
>& _rxForm
);
828 /** called whenever a connection to a database column has been suspended
830 virtual void onDisconnectedDbColumn();
832 /** called whenever a connection to an external supplier of values (XValueBinding) has been established
833 @see m_xExternalBinding
835 virtual void onConnectedExternalValue( );
837 /** called whenever an external validator has been registered
839 void onConnectedValidator( );
840 /** called whenever an external validator has been revoked
842 void onDisconnectedValidator( );
844 /** nFieldType is the type of the field, on which the model will be linked.
845 The linking happens when sal_True is returned.
846 The default-implementation allows everything but the three binary types
849 virtual bool approveDbColumnType(sal_Int32 _nColumnType
);
851 /** retrieves the current value of the control, in a shape which can be used with our
854 The default implementation simply calls <member>>translateControlValueToExternalValue</member>.
857 Our own mutex is locked.
859 virtual css::uno::Any
860 translateControlValueToValidatableValue( ) const;
862 /** retrieves the current value of the form component
864 This is the implementation method for XValidatableFormComponent::getCurrentValue. The default implementation
865 calls translateControlValueToValidatableValue if a validator is present, otherwise getControlValue.
868 our mutex is locked when this method is called
870 virtual css::uno::Any
871 getCurrentFormComponentValue() const;
873 /** We can't write (new) common properties in this base class, as the file format doesn't allow this
874 (unfortunately). So derived classes may use the following two methods. They secure the written
875 data with marks, so any new common properties in newer versions will be skipped by older ones.
877 void writeCommonProperties(const css::uno::Reference
< css::io::XObjectOutputStream
>& _rxOutStream
);
878 void readCommonProperties(const css::uno::Reference
< css::io::XObjectInputStream
>& _rxInStream
);
879 // the next method may be used in derived classes's read when an unknown version is encountered
880 void defaultCommonProperties();
882 /** called to reset the control to some kind of default.
884 <p>The semantics of "default" is finally defined by the derived class (in particular,
885 by <member>getDefaultForReset</member>).</p>
887 <p>No listener notification needs to be done in the derived class.</p>
889 <p>Normally, you won't override this method, but <member>getDefaultForReset</member> instead.</p>
891 @see getDefaultForReset
893 virtual void resetNoBroadcast();
895 virtual css::uno::Sequence
< css::uno::Type
> _getTypes() override
;
897 /// sets m_xField to the given new value, without notifying our listeners
898 void impl_setField_noNotify(
899 const css::uno::Reference
< css::beans::XPropertySet
>& _rxField
901 bool hasField() const
903 return m_xField
.is();
905 sal_Int32
getFieldType() const
910 // OControlModel's property handling
911 virtual void describeFixedProperties(
912 css::uno::Sequence
< css::beans::Property
>& /* [out] */ _rProps
916 const css::uno::Reference
< css::beans::XPropertySet
>& getField() const
923 DECLARE_UNO3_AGG_DEFAULTS(OBoundControlModel
, OControlModel
)
924 virtual css::uno::Any SAL_CALL
queryAggregation( const css::uno::Type
& _rType
) override
;
927 virtual void SAL_CALL
disposing() override
;
930 virtual void SAL_CALL
reset( ) override
;
931 virtual void SAL_CALL
addResetListener( const css::uno::Reference
< css::form::XResetListener
>& aListener
) override
;
932 virtual void SAL_CALL
removeResetListener( const css::uno::Reference
< css::form::XResetListener
>& aListener
) override
;
935 virtual css::uno::Sequence
<OUString
> SAL_CALL
getSupportedServiceNames( ) override
;
937 // XServiceInfo - static version
938 /// @throws css::uno::RuntimeException
939 static css::uno::Sequence
<OUString
> getSupportedServiceNames_Static();
942 virtual void SAL_CALL
setParent( const css::uno::Reference
< css::uno::XInterface
>& Parent
) override
;
945 virtual void SAL_CALL
write( const css::uno::Reference
< css::io::XObjectOutputStream
>& OutStream
) override
;
946 virtual void SAL_CALL
read( const css::uno::Reference
< css::io::XObjectInputStream
>& InStream
) override
;
949 virtual sal_Bool SAL_CALL
commit() override
;
951 // XUpdateBroadcaster (base of XBoundComponent)
952 virtual void SAL_CALL
addUpdateListener( const css::uno::Reference
< css::form::XUpdateListener
>& aListener
) override
;
953 virtual void SAL_CALL
removeUpdateListener( const css::uno::Reference
< css::form::XUpdateListener
>& aListener
) override
;
956 virtual void SAL_CALL
getFastPropertyValue(css::uno::Any
& rValue
, sal_Int32 nHandle
) const override
;
957 virtual sal_Bool SAL_CALL
convertFastPropertyValue(
958 css::uno::Any
& _rConvertedValue
, css::uno::Any
& _rOldValue
, sal_Int32 _nHandle
, const css::uno::Any
& _rValue
) override
;
959 virtual void SAL_CALL
setFastPropertyValue_NoBroadcast( sal_Int32 nHandle
, const css::uno::Any
& rValue
) override
;
960 using ::cppu::OPropertySetHelper::getFastPropertyValue
;
962 // css::beans::XPropertyState
963 virtual css::uno::Any
getPropertyDefaultByHandle( sal_Int32 nHandle
) const override
;
966 virtual void SAL_CALL
disposing(const css::lang::EventObject
& Source
) override
;
968 // XPropertyChangeListener
969 virtual void SAL_CALL
propertyChange( const css::beans::PropertyChangeEvent
& evt
) override
;
971 // XRowSetChangeListener
972 virtual void SAL_CALL
onRowSetChanged( const css::lang::EventObject
& i_Event
) override
;
975 virtual void SAL_CALL
loaded( const css::lang::EventObject
& aEvent
) override
;
976 virtual void SAL_CALL
unloading( const css::lang::EventObject
& aEvent
) override
;
977 virtual void SAL_CALL
unloaded( const css::lang::EventObject
& aEvent
) override
;
978 virtual void SAL_CALL
reloading( const css::lang::EventObject
& aEvent
) override
;
979 virtual void SAL_CALL
reloaded( const css::lang::EventObject
& aEvent
) override
;
983 virtual void SAL_CALL
setValueBinding( const css::uno::Reference
< css::form::binding::XValueBinding
>& _rxBinding
) override
;
984 virtual css::uno::Reference
< css::form::binding::XValueBinding
> SAL_CALL
getValueBinding( ) override
;
987 virtual void SAL_CALL
modified( const css::lang::EventObject
& _rEvent
) override
;
990 virtual void SAL_CALL
setValidator( const css::uno::Reference
< css::form::validation::XValidator
>& Validator
) override
;
991 virtual css::uno::Reference
< css::form::validation::XValidator
> SAL_CALL
getValidator( ) override
;
993 // XValidityConstraintListener
994 virtual void SAL_CALL
validityConstraintChanged( const css::lang::EventObject
& Source
) override
;
996 // XValidatableFormComponent
997 virtual sal_Bool SAL_CALL
isValid( ) override
;
998 virtual css::uno::Any SAL_CALL
getCurrentValue( ) override
;
999 virtual void SAL_CALL
addFormComponentValidityListener( const css::uno::Reference
< css::form::validation::XFormComponentValidityListener
>& Listener
) override
;
1000 virtual void SAL_CALL
removeFormComponentValidityListener( const css::uno::Reference
< css::form::validation::XFormComponentValidityListener
>& Listener
) override
;
1003 // OPropertyChangeListener
1005 _propertyChanged( const css::beans::PropertyChangeEvent
& _rEvt
) override
;
1007 /// checks whether we currently have an external value binding in place
1008 bool hasExternalValueBinding() const { return m_xExternalBinding
.is(); }
1010 // checks whether we currently have an external validator
1011 bool hasValidator() const { return m_xValidator
.is(); }
1013 /** transfers the very current value of the db column we're bound to the control
1015 our own mutex is locked
1017 we don't have an external binding in place
1019 void transferDbValueToControl( );
1021 /** transfers the current value of the active external binding to the control
1023 we do have an active external binding in place
1025 void transferExternalValueToControl( ControlModelLock
& _rInstanceLock
);
1027 /** transfers the control value to the external binding
1029 our own mutex is locked, and _rInstanceLock is the guard locking it
1031 we do have an active external binding in place
1033 void transferControlValueToExternal( ControlModelLock
& _rInstanceLock
);
1035 /** calculates the type which is to be used to communicate with the current external binding,
1036 and stores it in m_aExternalValueType
1038 The method checks the possible type candidates as returned by getSupportedBindingTypes,
1039 and the types supported by the current external binding, if any.
1041 void calculateExternalValueType();
1043 /** returns the type which should be used to exchange data with our external value binding
1045 @see initValueProperty
1047 const css::uno::Type
&
1048 getExternalValueType() const { return m_aExternalValueType
; }
1050 /** initializes the control from m_xField
1052 Basically, this method calls transferDbValueToControl - but only if our cursor is positioned
1053 on a valid row. Otherwise, the control is reset.
1056 m_xField is not <NULL/>
1058 void initFromField( const css::uno::Reference
< css::sdbc::XRowSet
>& _rxForm
);
1061 void connectToField( const css::uno::Reference
< css::sdbc::XRowSet
>& _rxForm
);
1064 /** does a new validation of the control value
1066 If necessary, our <member>m_bIsCurrentValueValid</member> member will be adjusted,
1067 and changes will be notified.
1069 Note that it's not necessary that we're connected to a validator. If we are not,
1070 it's assumed that our value is valid, and this is handled appropriately.
1072 Use this method if there is a potential that <b>only</b> the validity flag changed. If
1073 any of the other aspects (our current value, or our current text) changed, then
1074 pass <TRUE/> for <member>_bForceNotification</member>.
1076 @param _bForceNotification
1077 if <TRUE/>, then the validity listeners will be notified, not matter whether the validity
1080 void recheckValidity( bool _bForceNotification
);
1082 /// initializes m_pAggPropMultiplexer
1083 void implInitAggMultiplexer( );
1085 /// initializes listening at the value property
1086 void implInitValuePropertyListening( ) const;
1088 /** adds or removes the component as load listener to/from our form, and (if necessary) as RowSetChange listener at
1091 @precond there must no external value binding be in place
1093 void doFormListening( const bool _bStart
);
1095 bool isFormListening() const { return m_bFormListening
; }
1097 /** determines the new value of m_xAmbientForm
1099 void impl_determineAmbientForm_nothrow();
1101 /** connects to a value supplier which is a database column.
1103 The column is taken from our parent, which must be a database form respectively row set.
1105 @precond The control does not have an external value supplier
1108 Determines whether the connection is made after the row set has been loaded (<FALSE/>)
1109 or reloaded (<TRUE/>)
1111 @see impl_disconnectDatabaseColumn_noNotify
1113 void impl_connectDatabaseColumn_noNotify(
1117 /** disconnects from a value supplier which is a database column
1119 @precond The control does not have an external value supplier
1120 @see impl_connectDatabaseColumn_noNotify
1122 void impl_disconnectDatabaseColumn_noNotify();
1124 /** connects to an external value binding
1126 <p>Note that by definition, external data bindings supersede the SQL data binding which
1127 is defined by our RowSet-column-related properties. This means that in case we're currently
1128 connected to a database column when this is called, this connection is suspended.</p>
1131 the new external binding has already been approved (see <member>impl_approveValueBinding_nolock</member>)
1133 there currently is no external binding in place
1135 void connectExternalValueBinding(
1136 const css::uno::Reference
< css::form::binding::XValueBinding
>& _rxBinding
,
1137 ControlModelLock
& _rInstanceLock
1140 /** disconnects from an external value binding
1143 there currently is an external binding in place
1145 void disconnectExternalValueBinding( );
1147 /** connects the component to an external validator
1150 there currently is no active validator
1152 our mutex is currently locked exactly once
1154 void connectValidator(
1155 const css::uno::Reference
< css::form::validation::XValidator
>& _rxValidator
1158 /** disconnects the component from its current an external validator
1161 there currently is an active validator
1163 our mutex is currently locked exactly once
1165 void disconnectValidator( );
1167 /** called from within <member scope="css:::form::binding">XBindableValue::setValueBinding</member>
1168 to approve the new binding
1170 The default implementation approves the binding if and only if it is not <NULL/>, and supports
1171 the type returned by getExternalValueType.
1174 the binding which applies for being responsible for our value, Must not be
1177 <TRUE/> if and only if the given binding can supply values in the proper type
1179 @seealso getExternalValueType
1181 bool impl_approveValueBinding_nolock(
1182 const css::uno::Reference
< css::form::binding::XValueBinding
>& _rxBinding
1189 inline void ControlModelLock::acquire()
1191 m_rModel
.lockInstance( OControlModel::LockAccess() );
1194 inline void ControlModelLock::release()
1196 OSL_ENSURE( m_bLocked
, "ControlModelLock::release: not locked!" );
1199 if ( 0 == m_rModel
.unlockInstance( OControlModel::LockAccess() ) )
1200 impl_notifyAll_nothrow();
1207 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */