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 .
21 #include <comphelper/multiinterfacecontainer4.hxx>
23 #include <com/sun/star/beans/XPropertySet.hpp>
24 #include <com/sun/star/beans/XPropertySetOption.hpp>
25 #include <com/sun/star/beans/XMultiPropertySet.hpp>
26 #include <com/sun/star/beans/XFastPropertySet.hpp>
28 #include <comphelper/comphelperdllapi.h>
29 #include <comphelper/unoimplbase.hxx>
30 #include <cppuhelper/propshlp.hxx>
34 /*************************************************************************
35 *************************************************************************/
38 This abstract class maps the methods of the interfaces XMultiPropertySet, XFastPropertySet
39 and XPropertySet to the methods getInfoHelper, convertFastPropertyValue,
40 setFastPropertyValue_NoBroadcast and getFastPropertyValue. You must derive from
41 this class and override the methods.
42 It provides a standard implementation of the XPropertySetInfo.
44 This is a modified copy of the cppuhelper::OPropertySetHelper class, except
45 that is uses std::mutex instead of osl::Mutex.
47 class COMPHELPER_DLLPUBLIC OPropertySetHelper
: public virtual comphelper::UnoImplBase
,
48 public css::beans::XMultiPropertySet
,
49 public css::beans::XFastPropertySet
,
50 public css::beans::XPropertySet
57 @param bIgnoreRuntimeExceptionsWhileFiring
58 indicates whether occurring RuntimeExceptions will be
59 ignored when firing notifications
60 (vetoableChange(), propertyChange())
62 PropertyVetoExceptions may still be thrown.
63 This flag is useful in an inter-process scenario when
64 remote bridges may break down
65 (firing DisposedExceptions).
67 OPropertySetHelper(bool bIgnoreRuntimeExceptionsWhileFiring
);
70 Only returns a reference to XMultiPropertySet, XFastPropertySet, XPropertySet and
73 virtual css::uno::Any SAL_CALL
queryInterface(const css::uno::Type
& rType
) override
;
75 /** eases implementing XTypeProvider::getTypes, returns the types of XMultiPropertySet, XFastPropertySet, XPropertySet
77 @throws css::uno::RuntimeException
79 static css::uno::Sequence
<css::uno::Type
> getTypes();
82 Send a disposing notification to the listeners
86 void disposing(std::unique_lock
<std::mutex
>& rGuard
);
89 Throw UnknownPropertyException or PropertyVetoException if the property with the name
90 rPropertyName does not exist or is readonly. Otherwise rPropertyName is changed to its handle
91 value and setFastPropertyValue is called.
93 virtual void SAL_CALL
setPropertyValue(const ::rtl::OUString
& rPropertyName
,
94 const css::uno::Any
& aValue
) override final
;
96 Throw UnknownPropertyException if the property with the name
97 rPropertyName does not exist.
99 virtual css::uno::Any SAL_CALL
100 getPropertyValue(const ::rtl::OUString
& aPropertyName
) override final
;
102 /** Ignored if the property is not bound. */
103 virtual void SAL_CALL
addPropertyChangeListener(
104 const ::rtl::OUString
& aPropertyName
,
105 const css::uno::Reference
<css::beans::XPropertyChangeListener
>& aListener
) override final
;
107 /** Ignored if the property is not bound. */
108 virtual void SAL_CALL
removePropertyChangeListener(
109 const ::rtl::OUString
& aPropertyName
,
110 const css::uno::Reference
<css::beans::XPropertyChangeListener
>& aListener
) override final
;
112 /** Ignored if the property is not constrained. */
113 virtual void SAL_CALL
addVetoableChangeListener(
114 const ::rtl::OUString
& aPropertyName
,
115 const css::uno::Reference
<css::beans::XVetoableChangeListener
>& aListener
) override final
;
117 /** Ignored if the property is not constrained. */
118 virtual void SAL_CALL
removeVetoableChangeListener(
119 const ::rtl::OUString
& aPropertyName
,
120 const css::uno::Reference
<css::beans::XVetoableChangeListener
>& aListener
) override final
;
123 Throw UnknownPropertyException or PropertyVetoException if the property with the name
124 rPropertyName does not exist or is readonly. Otherwise the method convertFastPropertyValue
125 is called, then the vetoable listeners are notified. After this the value of the property
126 is changed with the setFastPropertyValue_NoBroadcast method and the bound listeners are
129 virtual void SAL_CALL
setFastPropertyValue(sal_Int32 nHandle
,
130 const css::uno::Any
& rValue
) override final
;
133 @exception css::beans::UnknownPropertyException
134 if the property with the handle nHandle does not exist.
136 virtual css::uno::Any SAL_CALL
getFastPropertyValue(sal_Int32 nHandle
) override final
;
139 virtual void SAL_CALL
140 setPropertyValues(const css::uno::Sequence
<::rtl::OUString
>& PropertyNames
,
141 const css::uno::Sequence
<css::uno::Any
>& Values
) override
;
143 virtual css::uno::Sequence
<css::uno::Any
> SAL_CALL
144 getPropertyValues(const css::uno::Sequence
<::rtl::OUString
>& PropertyNames
) override final
;
146 virtual void SAL_CALL
addPropertiesChangeListener(
147 const css::uno::Sequence
<::rtl::OUString
>& PropertyNames
,
148 const css::uno::Reference
<css::beans::XPropertiesChangeListener
>& Listener
) override final
;
150 virtual void SAL_CALL
removePropertiesChangeListener(
151 const css::uno::Reference
<css::beans::XPropertiesChangeListener
>& Listener
) override final
;
153 virtual void SAL_CALL
firePropertiesChangeEvent(
154 const css::uno::Sequence
<::rtl::OUString
>& PropertyNames
,
155 const css::uno::Reference
<css::beans::XPropertiesChangeListener
>& Listener
) override final
;
158 The property sequence is created in the call. The interface isn't used after the call.
160 static css::uno::Reference
<css::beans::XPropertySetInfo
>
161 createPropertySetInfo(cppu::IPropertyArrayHelper
& rProperties
);
165 You must call disposing() before destruction.
167 ~OPropertySetHelper();
169 /** Override this if you need to do something special during setFastPropertyValue */
170 virtual void setFastPropertyValueImpl(std::unique_lock
<std::mutex
>& rGuard
, sal_Int32 nHandle
,
171 const css::uno::Any
& rValue
);
172 /** Override this if you need to do something special during getPropertyValue */
173 virtual css::uno::Any
getPropertyValueImpl(std::unique_lock
<std::mutex
>& rGuard
,
174 const ::rtl::OUString
& aPropertyName
);
177 This method fire events to all registered property listeners.
178 @param pnHandles the id's of the properties that changed.
179 @param pNewValues the new values of the properties.
180 @param pOldValues the old values of the properties.
181 @param nCount the number of elements in the arrays pnHandles, pNewValues and pOldValues.
182 @param bVetoable true means fire to VetoableChangeListener, false means fire to
183 XPropertyChangedListener and XMultiPropertyChangedListener.
185 void fire(std::unique_lock
<std::mutex
>& rGuard
, sal_Int32
* pnHandles
,
186 const css::uno::Any
* pNewValues
, const css::uno::Any
* pOldValues
, sal_Int32 nCount
,
190 Set multiple properties with the handles.
191 @param nSeqLen the length of the arrays pHandles and Values.
192 @param pHandles the handles of the properties. The number of elements
193 in the Values sequence is the length of the handle array. A value of -1
194 of a handle means invalid property. These are ignored.
195 @param pValues the values of the properties.
196 @param nHitCount the number of valid entries in the handle array.
198 void setFastPropertyValues(std::unique_lock
<std::mutex
>& rGuard
, sal_Int32 nSeqLen
,
199 sal_Int32
* pHandles
, const css::uno::Any
* pValues
,
200 sal_Int32 nHitCount
);
203 This abstract method must return the name to index table. This table contains all property
204 names and types of this object. The method is not implemented in this class.
206 virtual cppu::IPropertyArrayHelper
& getInfoHelper() = 0;
209 Converted the value rValue and return the result in rConvertedValue and the
210 old value in rOldValue. An IllegalArgumentException is thrown.
211 The method is not implemented in this class. After this call the vetoable
212 listeners are notified.
214 @param rConvertedValue the converted value. Only set if return is true.
215 @param rOldValue the old value. Only set if return is true.
216 @param nHandle the handle of the property.
217 @param rValue the value to be converted
218 @return true if the value converted.
219 @throws css::lang::IllegalArgumentException
220 @throws css::beans::UnknownPropertyException
221 @throws css::uno::RuntimeException
223 virtual bool convertFastPropertyValue(std::unique_lock
<std::mutex
>& rGuard
,
224 css::uno::Any
& rConvertedValue
, css::uno::Any
& rOldValue
,
225 sal_Int32 nHandle
, const css::uno::Any
& rValue
)
228 /** The same as setFastPropertyValue; nHandle is always valid.
229 The changes must not be broadcasted in this method.
230 The method is implemented in a derived class.
233 Although you are permitted to throw any UNO exception, only the following
235 -- css::beans::UnknownPropertyException
236 -- css::beans::PropertyVetoException
237 -- css::lang::IllegalArgumentException
238 -- css::lang::WrappedTargetException
239 -- css::uno::RuntimeException
245 @throws css::uno::Exception
247 virtual void setFastPropertyValue_NoBroadcast(std::unique_lock
<std::mutex
>& rGuard
,
248 sal_Int32 nHandle
, const css::uno::Any
& rValue
)
251 The same as getFastPropertyValue, but return the value through rValue and nHandle
253 The method is not implemented in this class.
255 virtual void getFastPropertyValue(std::unique_lock
<std::mutex
>& rGuard
, css::uno::Any
& rValue
,
256 sal_Int32 nHandle
) const = 0;
258 /** sets an dependent property's value
260 <p>Sometimes setting a given property needs to implicitly modify another property's value. Calling |setPropertyValue|
261 from within |setFastPropertyValue_NoBroadcast| is not an option here, as it would notify the property listeners
262 while our mutex is still locked. Setting the dependent property's value directly (e.g. by calling |setFastPropertyValue_NoBroadcast|
263 recursively) is not an option, too, since it would miss firing the property change event.</p>
265 <p>So, in such cases, you use |setDependentFastPropertyValue| from within |setFastPropertyValue_NoBroadcast|.
266 It will convert and actually set the property value (invoking |convertFastPropertyValue| and |setFastPropertyValue_NoBroadcast|
267 for the given handle and value), and add the property change event to the list of events to be notified
268 when the bottom-most |setFastPropertyValue_NoBroadcast| on the stack returns.</p>
270 <p><strong>Note</strong>: The method will <em>not</em> invoke veto listeners for the property.</p>
272 <p><strong>Note</strong>: It's the caller's responsibility to ensure that our mutex is locked. This is
273 canonically given when the method is invoked from within |setFastPropertyValue_NoBroadcast|, in other
274 contexts, you might need to take own measures.</p>
276 void setDependentFastPropertyValue(std::unique_lock
<std::mutex
>& rGuard
, sal_Int32 i_handle
,
277 const css::uno::Any
& i_value
);
281 Container for the XPropertyChangedListener. The listeners are inserted by handle.
283 OMultiTypeInterfaceContainerHelperVar4
<sal_Int32
, css::beans::XPropertyChangeListener
> aBoundLC
;
285 Container for the XPropertyVetoableListener. The listeners are inserted by handle.
287 OMultiTypeInterfaceContainerHelperVar4
<sal_Int32
, css::beans::XVetoableChangeListener
>
290 Container for the XPropertyChangedListener where the listeners want to listen to all properties.
292 comphelper::OInterfaceContainerHelper4
<css::beans::XPropertyChangeListener
>
293 maPropertyChangeListeners
;
294 comphelper::OInterfaceContainerHelper4
<css::beans::XPropertiesChangeListener
>
295 maPropertiesChangeListeners
;
297 Container for the XVetoableChangeListener where the listeners want to listen to all properties.
299 comphelper::OInterfaceContainerHelper4
<css::beans::XVetoableChangeListener
>
300 maVetoableChangeListeners
;
301 std::vector
<sal_Int32
> m_handles
;
302 std::vector
<css::uno::Any
> m_newValues
;
303 std::vector
<css::uno::Any
> m_oldValues
;
304 bool m_bIgnoreRuntimeExceptionsWhileFiring
= false;
305 bool m_bFireEvents
= true;
307 /** notifies the given changes in property's values, <em>plus</em> all property changes collected during recent
308 |setDependentFastPropertyValue| calls.
310 void impl_fireAll(std::unique_lock
<std::mutex
>& rGuard
, sal_Int32
* i_handles
,
311 const css::uno::Any
* i_newValues
, const css::uno::Any
* i_oldValues
,
314 void fireVetoableChangeListeners(
315 std::unique_lock
<std::mutex
>& rGuard
,
316 comphelper::OInterfaceContainerHelper4
<css::beans::XVetoableChangeListener
>* pListeners
,
317 const css::beans::PropertyChangeEvent
& rChangeEvent
);
318 void firePropertyChangeListeners(
319 std::unique_lock
<std::mutex
>& rGuard
,
320 comphelper::OInterfaceContainerHelper4
<css::beans::XPropertyChangeListener
>* pListeners
,
321 const css::beans::PropertyChangeEvent
& rChangeEvent
);
324 } // end namespace comphelper
326 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */