1 /**************************************************************
2 *
3 * Licensed to the Apache Software Foundation (ASF) under one
4 * or more contributor license agreements. See the NOTICE file
5 * distributed with this work for additional information
6 * regarding copyright ownership. The ASF licenses this file
7 * to you under the Apache License, Version 2.0 (the
8 * "License"); you may not use this file except in compliance
9 * with the License. You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing,
14 * software distributed under the License is distributed on an
15 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16 * KIND, either express or implied. See the License for the
17 * specific language governing permissions and limitations
18 * under the License.
19 *
20 *************************************************************/
21
22
23
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_comphelper.hxx"
26 #include <comphelper/propertycontainerhelper.hxx>
27 #include <comphelper/property.hxx>
28 #include <osl/diagnose.h>
29 #include <uno/data.h>
30 #include <com/sun/star/uno/genfunc.h>
31 #include <com/sun/star/beans/PropertyAttribute.hpp>
32 #include <com/sun/star/beans/UnknownPropertyException.hpp>
33 #include <rtl/ustrbuf.hxx>
34
35 #include <algorithm>
36
37 //.........................................................................
38 namespace comphelper
39 {
40 //.........................................................................
41
42 using namespace ::com::sun::star::uno;
43 using namespace ::com::sun::star::lang;
44 using namespace ::com::sun::star::beans;
45
46 //--------------------------------------------------------------------------
47 namespace
48 {
49 // comparing two property descriptions
50 struct PropertyDescriptionCompareByHandle : public ::std::binary_function< PropertyDescription, PropertyDescription, bool >
51 {
operator ()comphelper::__anonf96cc37a0111::PropertyDescriptionCompareByHandle52 bool operator() (const PropertyDescription& x, const PropertyDescription& y) const
53 {
54 return x.aProperty.Handle < y.aProperty.Handle;
55 }
56 };
57 // comparing two property descriptions
58 struct PropertyDescriptionHandleCompare : public ::std::binary_function< PropertyDescription, sal_Int32, bool >
59 {
operator ()comphelper::__anonf96cc37a0111::PropertyDescriptionHandleCompare60 bool operator() (const PropertyDescription& x, const sal_Int32& y) const
61 {
62 return x.aProperty.Handle < y;
63 }
operator ()comphelper::__anonf96cc37a0111::PropertyDescriptionHandleCompare64 bool operator() (const sal_Int32& x, const PropertyDescription& y) const
65 {
66 return x < y.aProperty.Handle;
67 }
68 };
69 // comparing two property descriptions (by name)
70 struct PropertyDescriptionNameMatch : public ::std::unary_function< PropertyDescription, bool >
71 {
72 ::rtl::OUString m_rCompare;
PropertyDescriptionNameMatchcomphelper::__anonf96cc37a0111::PropertyDescriptionNameMatch73 PropertyDescriptionNameMatch( const ::rtl::OUString& _rCompare ) : m_rCompare( _rCompare ) { }
74
operator ()comphelper::__anonf96cc37a0111::PropertyDescriptionNameMatch75 bool operator() (const PropertyDescription& x ) const
76 {
77 return x.aProperty.Name.equals(m_rCompare);
78 }
79 };
80 }
81
82 //==========================================================================
83 //= OPropertyContainerHelper
84 //==========================================================================
85 //--------------------------------------------------------------------------
OPropertyContainerHelper()86 OPropertyContainerHelper::OPropertyContainerHelper()
87 :m_bUnused(sal_False)
88 {
89 }
90
91 // -------------------------------------------------------------------------
~OPropertyContainerHelper()92 OPropertyContainerHelper::~OPropertyContainerHelper()
93 {
94 }
95
96 //--------------------------------------------------------------------------
registerProperty(const::rtl::OUString & _rName,sal_Int32 _nHandle,sal_Int32 _nAttributes,void * _pPointerToMember,const Type & _rMemberType)97 void OPropertyContainerHelper::registerProperty(const ::rtl::OUString& _rName, sal_Int32 _nHandle,
98 sal_Int32 _nAttributes, void* _pPointerToMember, const Type& _rMemberType)
99 {
100 OSL_ENSURE((_nAttributes & PropertyAttribute::MAYBEVOID) == 0,
101 "OPropertyContainerHelper::registerProperty: don't use this for properties which may be void ! There is a method called \"registerMayBeVoidProperty\" for this !");
102 OSL_ENSURE(!_rMemberType.equals(::getCppuType(static_cast< Any* >(NULL))),
103 "OPropertyContainerHelper::registerProperty: don't give my the type of an uno::Any ! Really can't handle this !");
104 OSL_ENSURE(_pPointerToMember,
105 "OPropertyContainerHelper::registerProperty: you gave me nonsense : the pointer must be non-NULL");
106
107 PropertyDescription aNewProp;
108 aNewProp.aProperty = Property( _rName, _nHandle, _rMemberType, (sal_Int16)_nAttributes );
109 aNewProp.eLocated = PropertyDescription::ltDerivedClassRealType;
110 aNewProp.aLocation.pDerivedClassMember = _pPointerToMember;
111
112 implPushBackProperty(aNewProp);
113 }
114
115 //--------------------------------------------------------------------------
revokeProperty(sal_Int32 _nHandle)116 void OPropertyContainerHelper::revokeProperty( sal_Int32 _nHandle )
117 {
118 PropertiesIterator aPos = searchHandle( _nHandle );
119 if ( aPos == m_aProperties.end() )
120 throw UnknownPropertyException();
121 m_aProperties.erase( aPos );
122 }
123
124 //--------------------------------------------------------------------------
registerMayBeVoidProperty(const::rtl::OUString & _rName,sal_Int32 _nHandle,sal_Int32 _nAttributes,Any * _pPointerToMember,const Type & _rExpectedType)125 void OPropertyContainerHelper::registerMayBeVoidProperty(const ::rtl::OUString& _rName, sal_Int32 _nHandle, sal_Int32 _nAttributes,
126 Any* _pPointerToMember, const Type& _rExpectedType)
127 {
128 OSL_ENSURE((_nAttributes & PropertyAttribute::MAYBEVOID) != 0,
129 "OPropertyContainerHelper::registerMayBeVoidProperty: why calling this when the attributes say nothing about may-be-void ?");
130 OSL_ENSURE(!_rExpectedType.equals(::getCppuType(static_cast< Any* >(NULL))),
131 "OPropertyContainerHelper::registerMayBeVoidProperty: don't give my the type of an uno::Any ! Really can't handle this !");
132 OSL_ENSURE(_pPointerToMember,
133 "OPropertyContainerHelper::registerMayBeVoidProperty: you gave me nonsense : the pointer must be non-NULL");
134
135 _nAttributes |= PropertyAttribute::MAYBEVOID;
136
137 PropertyDescription aNewProp;
138 aNewProp.aProperty = Property( _rName, _nHandle, _rExpectedType, (sal_Int16)_nAttributes );
139 aNewProp.eLocated = PropertyDescription::ltDerivedClassAnyType;
140 aNewProp.aLocation.pDerivedClassMember = _pPointerToMember;
141
142 implPushBackProperty(aNewProp);
143 }
144
145
146 //--------------------------------------------------------------------------
registerPropertyNoMember(const::rtl::OUString & _rName,sal_Int32 _nHandle,sal_Int32 _nAttributes,const Type & _rType,const void * _pInitialValue)147 void OPropertyContainerHelper::registerPropertyNoMember(const ::rtl::OUString& _rName, sal_Int32 _nHandle, sal_Int32 _nAttributes,
148 const Type& _rType, const void* _pInitialValue)
149 {
150 OSL_ENSURE(!_rType.equals(::getCppuType(static_cast< Any* >(NULL))),
151 "OPropertyContainerHelper::registerPropertyNoMember : don't give my the type of an uno::Any ! Really can't handle this !");
152 OSL_ENSURE(_pInitialValue || ((_nAttributes & PropertyAttribute::MAYBEVOID) != 0),
153 "OPropertyContainerHelper::registerPropertyNoMember : you should not ommit the initial value if the property can't be void ! This will definitivly crash later !");
154
155 PropertyDescription aNewProp;
156 aNewProp.aProperty = Property( _rName, _nHandle, _rType, (sal_Int16)_nAttributes );
157 aNewProp.eLocated = PropertyDescription::ltHoldMyself;
158 aNewProp.aLocation.nOwnClassVectorIndex = m_aHoldProperties.size();
159 if (_pInitialValue)
160 m_aHoldProperties.push_back(Any(_pInitialValue, _rType));
161 else
162 m_aHoldProperties.push_back(Any());
163
164 implPushBackProperty(aNewProp);
165 }
166
167 //--------------------------------------------------------------------------
isRegisteredProperty(sal_Int32 _nHandle) const168 sal_Bool OPropertyContainerHelper::isRegisteredProperty( sal_Int32 _nHandle ) const
169 {
170 return const_cast< OPropertyContainerHelper* >( this )->searchHandle( _nHandle ) != m_aProperties.end();
171 }
172
173 //--------------------------------------------------------------------------
isRegisteredProperty(const::rtl::OUString & _rName) const174 sal_Bool OPropertyContainerHelper::isRegisteredProperty( const ::rtl::OUString& _rName ) const
175 {
176 // TODO: the current structure is from a time where properties were
177 // static, not dynamic. Since we allow that properties are also dynamic,
178 // i.e. registered and revoked even though the XPropertySet has already been
179 // accessed, a vector is not really the best data structure anymore ...
180
181 ConstPropertiesIterator pos = ::std::find_if(
182 m_aProperties.begin(),
183 m_aProperties.end(),
184 PropertyDescriptionNameMatch( _rName )
185 );
186 return pos != m_aProperties.end();
187 }
188
189 //--------------------------------------------------------------------------
190 namespace
191 {
192 struct ComparePropertyWithHandle
193 {
operator ()comphelper::__anonf96cc37a0211::ComparePropertyWithHandle194 bool operator()( const PropertyDescription& _rLHS, sal_Int32 _nRHS ) const
195 {
196 return _rLHS.aProperty.Handle < _nRHS;
197 }
operator ()comphelper::__anonf96cc37a0211::ComparePropertyWithHandle198 bool operator()( sal_Int32 _nLHS, const PropertyDescription& _rRHS ) const
199 {
200 return _nLHS < _rRHS.aProperty.Handle;
201 }
202 };
203 }
204
205 //--------------------------------------------------------------------------
implPushBackProperty(const PropertyDescription & _rProp)206 void OPropertyContainerHelper::implPushBackProperty(const PropertyDescription& _rProp)
207 {
208 #ifdef DBG_UTIL
209 for ( PropertiesIterator checkConflicts = m_aProperties.begin();
210 checkConflicts != m_aProperties.end();
211 ++checkConflicts
212 )
213 {
214 OSL_ENSURE(checkConflicts->aProperty.Name != _rProp.aProperty.Name, "OPropertyContainerHelper::implPushBackProperty: name already exists!");
215 OSL_ENSURE(checkConflicts->aProperty.Handle != _rProp.aProperty.Handle, "OPropertyContainerHelper::implPushBackProperty: handle already exists!");
216 }
217 #endif
218
219 PropertiesIterator pos = ::std::lower_bound(
220 m_aProperties.begin(), m_aProperties.end(),
221 _rProp.aProperty.Handle, ComparePropertyWithHandle() );
222
223 m_aProperties.insert( pos, _rProp );
224 }
225
226 //--------------------------------------------------------------------------
227 namespace
228 {
lcl_throwIllegalPropertyValueTypeException(const PropertyDescription & _rProperty,const Any & _rValue)229 void lcl_throwIllegalPropertyValueTypeException( const PropertyDescription& _rProperty, const Any& _rValue )
230 {
231 ::rtl::OUStringBuffer aErrorMessage;
232 aErrorMessage.appendAscii( "The given value cannot be converted to the required property type." );
233 aErrorMessage.appendAscii( "\n(property name \"" );
234 aErrorMessage.append( _rProperty.aProperty.Name );
235 aErrorMessage.appendAscii( "\", found value type \"" );
236 aErrorMessage.append( _rValue.getValueType().getTypeName() );
237 aErrorMessage.appendAscii( "\", required property type \"" );
238 aErrorMessage.append( _rProperty.aProperty.Type.getTypeName() );
239 aErrorMessage.appendAscii( "\")" );
240 throw IllegalArgumentException( aErrorMessage.makeStringAndClear(), NULL, 4 );
241 }
242 }
243
244 //--------------------------------------------------------------------------
convertFastPropertyValue(Any & _rConvertedValue,Any & _rOldValue,sal_Int32 _nHandle,const Any & _rValue)245 sal_Bool OPropertyContainerHelper::convertFastPropertyValue(
246 Any& _rConvertedValue, Any& _rOldValue, sal_Int32 _nHandle, const Any& _rValue ) SAL_THROW( (IllegalArgumentException) )
247 {
248 sal_Bool bModified = sal_False;
249
250 // get the property somebody is asking for
251 PropertiesIterator aPos = searchHandle(_nHandle);
252 if (aPos == m_aProperties.end())
253 {
254 OSL_ENSURE( false, "OPropertyContainerHelper::convertFastPropertyValue: unknown handle!" );
255 // should not happen if the derived class has built a correct property set info helper to be used by
256 // our base class OPropertySetHelper
257 return bModified;
258 }
259
260 switch (aPos->eLocated)
261 {
262 // similar handling for the two cases where the value is stored in an any
263 case PropertyDescription::ltHoldMyself:
264 case PropertyDescription::ltDerivedClassAnyType:
265 {
266 sal_Bool bMayBeVoid = ((aPos->aProperty.Attributes & PropertyAttribute::MAYBEVOID) != 0);
267
268
269 // non modifiable version of the value-to-be-set
270 Any aNewRequestedValue( _rValue );
271
272 // normalization
273 // (#102329# - 2002-08-14 - fs@openoffice.org)
274 // (#i29490# - 2004-06-16 - fs@openoffice.org)
275 if ( !aNewRequestedValue.getValueType().equals( aPos->aProperty.Type ) )
276 { // the actually given value is not of the same type as the one required
277 Any aProperlyTyped( NULL, aPos->aProperty.Type.getTypeLibType() );
278
279 if ( uno_type_assignData(
280 const_cast< void* >( aProperlyTyped.getValue() ), aProperlyTyped.getValueType().getTypeLibType(),
281 const_cast< void* >( aNewRequestedValue.getValue() ), aNewRequestedValue.getValueType().getTypeLibType(),
282 reinterpret_cast< uno_QueryInterfaceFunc >( cpp_queryInterface ),
283 reinterpret_cast< uno_AcquireFunc >( cpp_acquire ),
284 reinterpret_cast< uno_ReleaseFunc >( cpp_release )
285 )
286 )
287 {
288 // we were able to query the given XInterface-derivee for the interface
289 // which is required for this property
290 aNewRequestedValue = aProperlyTyped;
291 }
292 }
293
294 // argument check
295 if ( ! ( (bMayBeVoid && !aNewRequestedValue.hasValue()) // void is allowed if the attribute says so
296 || (aNewRequestedValue.getValueType().equals(aPos->aProperty.Type)) // else the types have to be equal
297 )
298 )
299 {
300 lcl_throwIllegalPropertyValueTypeException( *aPos, _rValue );
301 }
302
303 Any* pPropContainer = NULL;
304 // the pointer to the any which holds the property value, no matter if located in the derived clas
305 // or in out vector
306
307 if (PropertyDescription::ltHoldMyself == aPos->eLocated)
308 {
309 OSL_ENSURE(aPos->aLocation.nOwnClassVectorIndex < (sal_Int32)m_aHoldProperties.size(),
310 "OPropertyContainerHelper::convertFastPropertyValue: invalid position !");
311 PropertyContainerIterator aIter = m_aHoldProperties.begin() + aPos->aLocation.nOwnClassVectorIndex;
312 pPropContainer = &(*aIter);
313 }
314 else
315 pPropContainer = reinterpret_cast<Any*>(aPos->aLocation.pDerivedClassMember);
316
317 // check if the new value differs from the current one
318 if (!pPropContainer->hasValue() || !aNewRequestedValue.hasValue())
319 bModified = pPropContainer->hasValue() != aNewRequestedValue.hasValue();
320 else
321 bModified = !uno_type_equalData(
322 const_cast< void* >( pPropContainer->getValue() ), aPos->aProperty.Type.getTypeLibType(),
323 const_cast< void* >( aNewRequestedValue.getValue() ), aPos->aProperty.Type.getTypeLibType(),
324 reinterpret_cast< uno_QueryInterfaceFunc >( cpp_queryInterface ),
325 reinterpret_cast< uno_ReleaseFunc >( cpp_release )
326 );
327
328 if (bModified)
329 {
330 _rOldValue = *pPropContainer;
331 _rConvertedValue = aNewRequestedValue;
332 }
333 }
334 break;
335 case PropertyDescription::ltDerivedClassRealType:
336 // let the UNO runtime library do any possible conversion
337 // this may include a change of the type - for instance, if a LONG is required,
338 // but a short is given, then this is valid, as it can be converted without any potential
339 // data loss
340
341 Any aProperlyTyped;
342 const Any* pNewValue = &_rValue;
343
344 if (!_rValue.getValueType().equals(aPos->aProperty.Type))
345 {
346 sal_Bool bConverted = sal_False;
347
348 // a temporary any of the correct (required) type
349 aProperlyTyped = Any( NULL, aPos->aProperty.Type.getTypeLibType() );
350 // (need this as we do not want to overwrite the derived class member here)
351
352 if ( uno_type_assignData(
353 const_cast<void*>(aProperlyTyped.getValue()), aProperlyTyped.getValueType().getTypeLibType(),
354 const_cast<void*>(_rValue.getValue()), _rValue.getValueType().getTypeLibType(),
355 reinterpret_cast< uno_QueryInterfaceFunc >( cpp_queryInterface ),
356 reinterpret_cast< uno_AcquireFunc >( cpp_acquire ),
357 reinterpret_cast< uno_ReleaseFunc >( cpp_release )
358 )
359 )
360 {
361 // could query for the requested interface
362 bConverted = sal_True;
363 pNewValue = &aProperlyTyped;
364 }
365
366 if ( !bConverted )
367 lcl_throwIllegalPropertyValueTypeException( *aPos, _rValue );
368 }
369
370 // from here on, we should have the proper type
371 OSL_ENSURE( pNewValue->getValueType() == aPos->aProperty.Type,
372 "OPropertyContainerHelper::convertFastPropertyValue: conversion failed!" );
373 bModified = !uno_type_equalData(
374 aPos->aLocation.pDerivedClassMember, aPos->aProperty.Type.getTypeLibType(),
375 const_cast<void*>(pNewValue->getValue()), aPos->aProperty.Type.getTypeLibType(),
376 reinterpret_cast< uno_QueryInterfaceFunc >( cpp_queryInterface ),
377 reinterpret_cast< uno_ReleaseFunc >( cpp_release )
378 );
379
380 if (bModified)
381 {
382 _rOldValue.setValue(aPos->aLocation.pDerivedClassMember, aPos->aProperty.Type);
383 _rConvertedValue = *pNewValue;
384 }
385 break;
386 }
387
388 return bModified;
389 }
390
391 //--------------------------------------------------------------------------
setFastPropertyValue(sal_Int32 _nHandle,const Any & _rValue)392 void OPropertyContainerHelper::setFastPropertyValue(sal_Int32 _nHandle, const Any& _rValue) SAL_THROW( (Exception) )
393 {
394 // get the property somebody is asking for
395 PropertiesIterator aPos = searchHandle(_nHandle);
396 if (aPos == m_aProperties.end())
397 {
398 OSL_ENSURE( false, "OPropertyContainerHelper::setFastPropertyValue: unknown handle!" );
399 // should not happen if the derived class has built a correct property set info helper to be used by
400 // our base class OPropertySetHelper
401 return;
402 }
403
404 switch (aPos->eLocated)
405 {
406 case PropertyDescription::ltHoldMyself:
407 m_aHoldProperties[aPos->aLocation.nOwnClassVectorIndex] = _rValue;
408 break;
409
410 case PropertyDescription::ltDerivedClassAnyType:
411 *reinterpret_cast< Any* >(aPos->aLocation.pDerivedClassMember) = _rValue;
412 break;
413
414 case PropertyDescription::ltDerivedClassRealType:
415 #if OSL_DEBUG_LEVEL > 0
416 sal_Bool bSuccess =
417 #endif
418 // copy the data from the to-be-set value
419 uno_type_assignData(
420 aPos->aLocation.pDerivedClassMember, aPos->aProperty.Type.getTypeLibType(),
421 const_cast< void* >( _rValue.getValue() ), _rValue.getValueType().getTypeLibType(),
422 reinterpret_cast< uno_QueryInterfaceFunc >( cpp_queryInterface ),
423 reinterpret_cast< uno_AcquireFunc >( cpp_acquire ),
424 reinterpret_cast< uno_ReleaseFunc >( cpp_release ) );
425
426 OSL_ENSURE( bSuccess,
427 "OPropertyContainerHelper::setFastPropertyValue: ooops .... the value could not be assigned!");
428
429 break;
430 }
431 }
432
433 //--------------------------------------------------------------------------
getFastPropertyValue(Any & _rValue,sal_Int32 _nHandle) const434 void OPropertyContainerHelper::getFastPropertyValue(Any& _rValue, sal_Int32 _nHandle) const
435 {
436 // get the property somebody is asking for
437 PropertiesIterator aPos = const_cast<OPropertyContainerHelper*>(this)->searchHandle(_nHandle);
438 if (aPos == m_aProperties.end())
439 {
440 OSL_ENSURE( false, "OPropertyContainerHelper::getFastPropertyValue: unknown handle!" );
441 // should not happen if the derived class has built a correct property set info helper to be used by
442 // our base class OPropertySetHelper
443 return;
444 }
445
446 switch (aPos->eLocated)
447 {
448 case PropertyDescription::ltHoldMyself:
449 OSL_ENSURE(aPos->aLocation.nOwnClassVectorIndex < (sal_Int32)m_aHoldProperties.size(),
450 "OPropertyContainerHelper::convertFastPropertyValue: invalid position !");
451 _rValue = m_aHoldProperties[aPos->aLocation.nOwnClassVectorIndex];
452 break;
453 case PropertyDescription::ltDerivedClassAnyType:
454 _rValue = *reinterpret_cast<Any*>(aPos->aLocation.pDerivedClassMember);
455 break;
456 case PropertyDescription::ltDerivedClassRealType:
457 _rValue.setValue(aPos->aLocation.pDerivedClassMember, aPos->aProperty.Type);
458 break;
459 }
460 }
461
462 //--------------------------------------------------------------------------
searchHandle(sal_Int32 _nHandle)463 OPropertyContainerHelper::PropertiesIterator OPropertyContainerHelper::searchHandle(sal_Int32 _nHandle)
464 {
465 // search a lower bound
466 PropertiesIterator aLowerBound = ::std::lower_bound(
467 m_aProperties.begin(),
468 m_aProperties.end(),
469 _nHandle,
470 PropertyDescriptionHandleCompare());
471
472 // check for identity
473 if ((aLowerBound != m_aProperties.end()) && aLowerBound->aProperty.Handle != _nHandle)
474 aLowerBound = m_aProperties.end();
475
476 return aLowerBound;
477 }
478
479 //--------------------------------------------------------------------------
getProperty(const::rtl::OUString & _rName) const480 const Property& OPropertyContainerHelper::getProperty( const ::rtl::OUString& _rName ) const
481 {
482 ConstPropertiesIterator pos = ::std::find_if(
483 m_aProperties.begin(),
484 m_aProperties.end(),
485 PropertyDescriptionNameMatch( _rName )
486 );
487 if ( pos == m_aProperties.end() )
488 throw UnknownPropertyException( _rName, NULL );
489
490 return pos->aProperty;
491 }
492
493 //--------------------------------------------------------------------------
modifyAttributes(sal_Int32 _nHandle,sal_Int32 _nAddAttrib,sal_Int32 _nRemoveAttrib)494 void OPropertyContainerHelper::modifyAttributes(sal_Int32 _nHandle, sal_Int32 _nAddAttrib, sal_Int32 _nRemoveAttrib)
495 {
496 // get the property somebody is asking for
497 PropertiesIterator aPos = searchHandle(_nHandle);
498 if (aPos == m_aProperties.end())
499 {
500 OSL_ENSURE( false, "OPropertyContainerHelper::modifyAttributes: unknown handle!" );
501 // should not happen if the derived class has built a correct property set info helper to be used by
502 // our base class OPropertySetHelper
503 return;
504 }
505 aPos->aProperty.Handle |= _nAddAttrib;
506 aPos->aProperty.Handle &= ~_nRemoveAttrib;
507 }
508
509 //--------------------------------------------------------------------------
describeProperties(Sequence<Property> & _rProps) const510 void OPropertyContainerHelper::describeProperties(Sequence< Property >& _rProps) const
511 {
512 Sequence< Property > aOwnProps(m_aProperties.size());
513 Property* pOwnProps = aOwnProps.getArray();
514
515 for ( ConstPropertiesIterator aLoop = m_aProperties.begin();
516 aLoop != m_aProperties.end();
517 ++aLoop, ++pOwnProps
518 )
519 {
520 pOwnProps->Name = aLoop->aProperty.Name;
521 pOwnProps->Handle = aLoop->aProperty.Handle;
522 pOwnProps->Attributes = (sal_Int16)aLoop->aProperty.Attributes;
523 pOwnProps->Type = aLoop->aProperty.Type;
524 }
525
526 // as our property vector is sorted by handles, not by name, we have to sort aOwnProps
527 ::std::sort(aOwnProps.getArray(), aOwnProps.getArray() + aOwnProps.getLength(), PropertyCompareByName());
528
529 // unfortunally the STL merge function does not allow the output range to overlap one of the input ranges,
530 // so we need an extra sequence
531 Sequence< Property > aOutput;
532 aOutput.realloc(_rProps.getLength() + aOwnProps.getLength());
533 // do the merge
534 ::std::merge( _rProps.getConstArray(), _rProps.getConstArray() + _rProps.getLength(), // input 1
535 aOwnProps.getConstArray(), aOwnProps.getConstArray() + aOwnProps.getLength(), // input 2
536 aOutput.getArray(), // output
537 PropertyCompareByName() // compare operator
538 );
539
540 // copy the output
541 _rProps = aOutput;
542 }
543
544 //.........................................................................
545 } // namespace comphelper
546 //.........................................................................
547
548
549