1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski // MARKER(update_precomp.py): autogen include statement, do not remove
25*b1cdbd2cSJim Jagielski #include "precompiled_comphelper.hxx"
26*b1cdbd2cSJim Jagielski 
27*b1cdbd2cSJim Jagielski 
28*b1cdbd2cSJim Jagielski #include <comphelper/property.hxx>
29*b1cdbd2cSJim Jagielski #include <comphelper/sequence.hxx>
30*b1cdbd2cSJim Jagielski #include <comphelper/types.hxx>
31*b1cdbd2cSJim Jagielski #include <osl/diagnose.h>
32*b1cdbd2cSJim Jagielski 
33*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 0
34*b1cdbd2cSJim Jagielski     #include <rtl/strbuf.hxx>
35*b1cdbd2cSJim Jagielski     #include <cppuhelper/exc_hlp.hxx>
36*b1cdbd2cSJim Jagielski     #include <osl/thread.h>
37*b1cdbd2cSJim Jagielski     #include <com/sun/star/lang/XServiceInfo.hpp>
38*b1cdbd2cSJim Jagielski     #include <typeinfo>
39*b1cdbd2cSJim Jagielski #endif
40*b1cdbd2cSJim Jagielski #include <com/sun/star/beans/PropertyAttribute.hpp>
41*b1cdbd2cSJim Jagielski #include <com/sun/star/lang/IllegalArgumentException.hpp>
42*b1cdbd2cSJim Jagielski #include <com/sun/star/uno/genfunc.h>
43*b1cdbd2cSJim Jagielski 
44*b1cdbd2cSJim Jagielski #include <algorithm>
45*b1cdbd2cSJim Jagielski #include <boost/bind.hpp>
46*b1cdbd2cSJim Jagielski 
47*b1cdbd2cSJim Jagielski //.........................................................................
48*b1cdbd2cSJim Jagielski namespace comphelper
49*b1cdbd2cSJim Jagielski {
50*b1cdbd2cSJim Jagielski 
51*b1cdbd2cSJim Jagielski     /** === begin UNO using === **/
52*b1cdbd2cSJim Jagielski     using ::com::sun::star::uno::Reference;
53*b1cdbd2cSJim Jagielski     using ::com::sun::star::beans::XPropertySet;
54*b1cdbd2cSJim Jagielski     using ::com::sun::star::beans::XPropertySetInfo;
55*b1cdbd2cSJim Jagielski     using ::com::sun::star::beans::Property;
56*b1cdbd2cSJim Jagielski     using ::com::sun::star::uno::Sequence;
57*b1cdbd2cSJim Jagielski     using ::com::sun::star::uno::Exception;
58*b1cdbd2cSJim Jagielski     using ::com::sun::star::uno::Any;
59*b1cdbd2cSJim Jagielski     using ::com::sun::star::uno::Type;
60*b1cdbd2cSJim Jagielski     using ::com::sun::star::uno::cpp_queryInterface;
61*b1cdbd2cSJim Jagielski     using ::com::sun::star::uno::cpp_acquire;
62*b1cdbd2cSJim Jagielski     using ::com::sun::star::uno::cpp_release;
63*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 0
64*b1cdbd2cSJim Jagielski     using ::com::sun::star::lang::XServiceInfo;
65*b1cdbd2cSJim Jagielski #endif
66*b1cdbd2cSJim Jagielski     using ::com::sun::star::uno::UNO_QUERY;
67*b1cdbd2cSJim Jagielski     /** === end UNO using === **/
68*b1cdbd2cSJim Jagielski     namespace PropertyAttribute = ::com::sun::star::beans::PropertyAttribute;
69*b1cdbd2cSJim Jagielski 
70*b1cdbd2cSJim Jagielski //------------------------------------------------------------------
copyProperties(const Reference<XPropertySet> & _rxSource,const Reference<XPropertySet> & _rxDest)71*b1cdbd2cSJim Jagielski void copyProperties(const Reference<XPropertySet>& _rxSource,
72*b1cdbd2cSJim Jagielski 					const Reference<XPropertySet>& _rxDest)
73*b1cdbd2cSJim Jagielski {
74*b1cdbd2cSJim Jagielski 	if (!_rxSource.is() || !_rxDest.is())
75*b1cdbd2cSJim Jagielski 	{
76*b1cdbd2cSJim Jagielski 		OSL_ENSURE(sal_False, "copyProperties: invalid arguments !");
77*b1cdbd2cSJim Jagielski         return;
78*b1cdbd2cSJim Jagielski 	}
79*b1cdbd2cSJim Jagielski 
80*b1cdbd2cSJim Jagielski 	Reference< XPropertySetInfo > xSourceProps = _rxSource->getPropertySetInfo();
81*b1cdbd2cSJim Jagielski 	Reference< XPropertySetInfo > xDestProps = _rxDest->getPropertySetInfo();
82*b1cdbd2cSJim Jagielski 
83*b1cdbd2cSJim Jagielski 	Sequence< Property > aSourceProps = xSourceProps->getProperties();
84*b1cdbd2cSJim Jagielski 	const Property* pSourceProps = aSourceProps.getConstArray();
85*b1cdbd2cSJim Jagielski 	Property aDestProp;
86*b1cdbd2cSJim Jagielski 	for (sal_Int32 i=0; i<aSourceProps.getLength(); ++i, ++pSourceProps)
87*b1cdbd2cSJim Jagielski 	{
88*b1cdbd2cSJim Jagielski 		if ( xDestProps->hasPropertyByName(pSourceProps->Name) )
89*b1cdbd2cSJim Jagielski 		{
90*b1cdbd2cSJim Jagielski 			try
91*b1cdbd2cSJim Jagielski 			{
92*b1cdbd2cSJim Jagielski 				aDestProp = xDestProps->getPropertyByName(pSourceProps->Name);
93*b1cdbd2cSJim Jagielski 				if (0 == (aDestProp.Attributes & PropertyAttribute::READONLY) )
94*b1cdbd2cSJim Jagielski                 {
95*b1cdbd2cSJim Jagielski                     const Any aSourceValue = _rxSource->getPropertyValue(pSourceProps->Name);
96*b1cdbd2cSJim Jagielski                     if ( 0 != (aDestProp.Attributes & PropertyAttribute::MAYBEVOID) || aSourceValue.hasValue() )
97*b1cdbd2cSJim Jagielski 					    _rxDest->setPropertyValue(pSourceProps->Name, aSourceValue);
98*b1cdbd2cSJim Jagielski                 }
99*b1cdbd2cSJim Jagielski 			}
100*b1cdbd2cSJim Jagielski 			catch (Exception&)
101*b1cdbd2cSJim Jagielski 			{
102*b1cdbd2cSJim Jagielski #if OSL_DEBUG_LEVEL > 0
103*b1cdbd2cSJim Jagielski                 ::rtl::OStringBuffer aBuffer;
104*b1cdbd2cSJim Jagielski                 aBuffer.append( "::comphelper::copyProperties: could not copy property '" );
105*b1cdbd2cSJim Jagielski                 aBuffer.append( ::rtl::OString( pSourceProps->Name.getStr(), pSourceProps->Name.getLength(), RTL_TEXTENCODING_ASCII_US ) );
106*b1cdbd2cSJim Jagielski                 aBuffer.append( "' to the destination set (a '" );
107*b1cdbd2cSJim Jagielski 
108*b1cdbd2cSJim Jagielski                 Reference< XServiceInfo > xSI( _rxDest, UNO_QUERY );
109*b1cdbd2cSJim Jagielski                 if ( xSI.is() )
110*b1cdbd2cSJim Jagielski                 {
111*b1cdbd2cSJim Jagielski                     aBuffer.append( ::rtl::OUStringToOString( xSI->getImplementationName(), osl_getThreadTextEncoding() ) );
112*b1cdbd2cSJim Jagielski                 }
113*b1cdbd2cSJim Jagielski                 else
114*b1cdbd2cSJim Jagielski                 {
115*b1cdbd2cSJim Jagielski                     aBuffer.append( typeid( *_rxDest.get() ).name() );
116*b1cdbd2cSJim Jagielski                 }
117*b1cdbd2cSJim Jagielski                 aBuffer.append( "' implementation).\n" );
118*b1cdbd2cSJim Jagielski 
119*b1cdbd2cSJim Jagielski                 Any aException( ::cppu::getCaughtException() );
120*b1cdbd2cSJim Jagielski                 aBuffer.append( "Caught an exception of type '" );
121*b1cdbd2cSJim Jagielski                 ::rtl::OUString sExceptionType( aException.getValueTypeName() );
122*b1cdbd2cSJim Jagielski                 aBuffer.append( ::rtl::OString( sExceptionType.getStr(), sExceptionType.getLength(), RTL_TEXTENCODING_ASCII_US ) );
123*b1cdbd2cSJim Jagielski                 aBuffer.append( "'" );
124*b1cdbd2cSJim Jagielski 
125*b1cdbd2cSJim Jagielski                 Exception aBaseException;
126*b1cdbd2cSJim Jagielski                 if ( ( aException >>= aBaseException ) && aBaseException.Message.getLength() )
127*b1cdbd2cSJim Jagielski                 {
128*b1cdbd2cSJim Jagielski                     aBuffer.append( ", saying '" );
129*b1cdbd2cSJim Jagielski                     aBuffer.append( ::rtl::OString( aBaseException.Message.getStr(), aBaseException.Message.getLength(), osl_getThreadTextEncoding() ) );
130*b1cdbd2cSJim Jagielski                     aBuffer.append( "'" );
131*b1cdbd2cSJim Jagielski                 }
132*b1cdbd2cSJim Jagielski                 aBuffer.append( "." );
133*b1cdbd2cSJim Jagielski 
134*b1cdbd2cSJim Jagielski                 OSL_ENSURE( sal_False, aBuffer.getStr() );
135*b1cdbd2cSJim Jagielski #endif
136*b1cdbd2cSJim Jagielski 			}
137*b1cdbd2cSJim Jagielski 		}
138*b1cdbd2cSJim Jagielski 	}
139*b1cdbd2cSJim Jagielski }
140*b1cdbd2cSJim Jagielski 
141*b1cdbd2cSJim Jagielski //------------------------------------------------------------------
hasProperty(const rtl::OUString & _rName,const Reference<XPropertySet> & _rxSet)142*b1cdbd2cSJim Jagielski sal_Bool hasProperty(const rtl::OUString& _rName, const Reference<XPropertySet>& _rxSet)
143*b1cdbd2cSJim Jagielski {
144*b1cdbd2cSJim Jagielski 	if (_rxSet.is())
145*b1cdbd2cSJim Jagielski 	{
146*b1cdbd2cSJim Jagielski 		//	XPropertySetInfoRef xInfo(rxSet->getPropertySetInfo());
147*b1cdbd2cSJim Jagielski 		return _rxSet->getPropertySetInfo()->hasPropertyByName(_rName);
148*b1cdbd2cSJim Jagielski 	}
149*b1cdbd2cSJim Jagielski 	return sal_False;
150*b1cdbd2cSJim Jagielski }
151*b1cdbd2cSJim Jagielski 
152*b1cdbd2cSJim Jagielski //------------------------------------------------------------------
findProperty(Property & o_rProp,Sequence<Property> & i_seqProps,const::rtl::OUString & i_rPropName)153*b1cdbd2cSJim Jagielski bool findProperty(Property&              o_rProp,
154*b1cdbd2cSJim Jagielski                   Sequence<Property>&    i_seqProps,
155*b1cdbd2cSJim Jagielski                   const ::rtl::OUString& i_rPropName)
156*b1cdbd2cSJim Jagielski {
157*b1cdbd2cSJim Jagielski     const Property* pAry(i_seqProps.getConstArray());
158*b1cdbd2cSJim Jagielski     const sal_Int32 nLen(i_seqProps.getLength());
159*b1cdbd2cSJim Jagielski     const Property* pRes(
160*b1cdbd2cSJim Jagielski         std::find_if(pAry,pAry+nLen,
161*b1cdbd2cSJim Jagielski                      boost::bind(PropertyStringEqualFunctor(),
162*b1cdbd2cSJim Jagielski                                  _1,
163*b1cdbd2cSJim Jagielski                                  boost::cref(i_rPropName))));
164*b1cdbd2cSJim Jagielski     if( pRes == pAry+nLen )
165*b1cdbd2cSJim Jagielski         return false;
166*b1cdbd2cSJim Jagielski 
167*b1cdbd2cSJim Jagielski     o_rProp = *pRes;
168*b1cdbd2cSJim Jagielski     return true;
169*b1cdbd2cSJim Jagielski }
170*b1cdbd2cSJim Jagielski 
171*b1cdbd2cSJim Jagielski //------------------------------------------------------------------
RemoveProperty(Sequence<Property> & _rProps,const rtl::OUString & _rPropName)172*b1cdbd2cSJim Jagielski void RemoveProperty(Sequence<Property>& _rProps, const rtl::OUString& _rPropName)
173*b1cdbd2cSJim Jagielski {
174*b1cdbd2cSJim Jagielski 	sal_Int32 nLen = _rProps.getLength();
175*b1cdbd2cSJim Jagielski 
176*b1cdbd2cSJim Jagielski 	// binaere Suche
177*b1cdbd2cSJim Jagielski 	const Property* pProperties = _rProps.getConstArray();
178*b1cdbd2cSJim Jagielski 	const Property* pResult = ::std::lower_bound(pProperties, pProperties + nLen, _rPropName,PropertyStringLessFunctor());
179*b1cdbd2cSJim Jagielski 
180*b1cdbd2cSJim Jagielski 	// gefunden ?
181*b1cdbd2cSJim Jagielski 	if ( pResult && (pResult != pProperties + nLen) && (pResult->Name == _rPropName) )
182*b1cdbd2cSJim Jagielski 	{
183*b1cdbd2cSJim Jagielski 		OSL_ENSURE(pResult->Name.equals(_rPropName), "::RemoveProperty Properties nicht sortiert");
184*b1cdbd2cSJim Jagielski 		removeElementAt(_rProps, pResult - pProperties);
185*b1cdbd2cSJim Jagielski 	}
186*b1cdbd2cSJim Jagielski }
187*b1cdbd2cSJim Jagielski 
188*b1cdbd2cSJim Jagielski //------------------------------------------------------------------
ModifyPropertyAttributes(Sequence<Property> & seqProps,const::rtl::OUString & sPropName,sal_Int16 nAddAttrib,sal_Int16 nRemoveAttrib)189*b1cdbd2cSJim Jagielski void ModifyPropertyAttributes(Sequence<Property>& seqProps, const ::rtl::OUString& sPropName, sal_Int16 nAddAttrib, sal_Int16 nRemoveAttrib)
190*b1cdbd2cSJim Jagielski {
191*b1cdbd2cSJim Jagielski 	sal_Int32 nLen = seqProps.getLength();
192*b1cdbd2cSJim Jagielski 
193*b1cdbd2cSJim Jagielski 	// binaere Suche
194*b1cdbd2cSJim Jagielski 	Property* pProperties = seqProps.getArray();
195*b1cdbd2cSJim Jagielski 	Property* pResult = ::std::lower_bound(pProperties, pProperties + nLen,sPropName, PropertyStringLessFunctor());
196*b1cdbd2cSJim Jagielski 
197*b1cdbd2cSJim Jagielski 	// gefunden ?
198*b1cdbd2cSJim Jagielski 	if ( pResult && (pResult != pProperties + nLen) && (pResult->Name == sPropName) )
199*b1cdbd2cSJim Jagielski 	{
200*b1cdbd2cSJim Jagielski 		pResult->Attributes |= nAddAttrib;
201*b1cdbd2cSJim Jagielski 		pResult->Attributes &= ~nRemoveAttrib;
202*b1cdbd2cSJim Jagielski 	}
203*b1cdbd2cSJim Jagielski }
204*b1cdbd2cSJim Jagielski 
205*b1cdbd2cSJim Jagielski //------------------------------------------------------------------
tryPropertyValue(Any & _rConvertedValue,Any & _rOldValue,const Any & _rValueToSet,const Any & _rCurrentValue,const Type & _rExpectedType)206*b1cdbd2cSJim Jagielski sal_Bool tryPropertyValue(Any& _rConvertedValue, Any& _rOldValue, const Any& _rValueToSet, const Any& _rCurrentValue, const Type& _rExpectedType)
207*b1cdbd2cSJim Jagielski {
208*b1cdbd2cSJim Jagielski 	sal_Bool bModified(sal_False);
209*b1cdbd2cSJim Jagielski 	if (_rCurrentValue.getValue() != _rValueToSet.getValue())
210*b1cdbd2cSJim Jagielski 	{
211*b1cdbd2cSJim Jagielski         if ( _rValueToSet.hasValue() && ( !_rExpectedType.equals( _rValueToSet.getValueType() ) ) )
212*b1cdbd2cSJim Jagielski         {
213*b1cdbd2cSJim Jagielski             _rConvertedValue = Any( NULL, _rExpectedType.getTypeLibType() );
214*b1cdbd2cSJim Jagielski 
215*b1cdbd2cSJim Jagielski             if  ( !uno_type_assignData(
216*b1cdbd2cSJim Jagielski                     const_cast< void* >( _rConvertedValue.getValue() ), _rConvertedValue.getValueType().getTypeLibType(),
217*b1cdbd2cSJim Jagielski                     const_cast< void* >( _rValueToSet.getValue() ), _rValueToSet.getValueType().getTypeLibType(),
218*b1cdbd2cSJim Jagielski                     reinterpret_cast< uno_QueryInterfaceFunc >(
219*b1cdbd2cSJim Jagielski                         cpp_queryInterface),
220*b1cdbd2cSJim Jagielski                     reinterpret_cast< uno_AcquireFunc >(cpp_acquire),
221*b1cdbd2cSJim Jagielski                     reinterpret_cast< uno_ReleaseFunc >(cpp_release)
222*b1cdbd2cSJim Jagielski                   )
223*b1cdbd2cSJim Jagielski                 )
224*b1cdbd2cSJim Jagielski                 throw starlang::IllegalArgumentException();
225*b1cdbd2cSJim Jagielski         }
226*b1cdbd2cSJim Jagielski         else
227*b1cdbd2cSJim Jagielski             _rConvertedValue = _rValueToSet;
228*b1cdbd2cSJim Jagielski 
229*b1cdbd2cSJim Jagielski 		if ( _rCurrentValue != _rConvertedValue )
230*b1cdbd2cSJim Jagielski 		{
231*b1cdbd2cSJim Jagielski 			_rOldValue = _rCurrentValue;
232*b1cdbd2cSJim Jagielski 			bModified = sal_True;
233*b1cdbd2cSJim Jagielski 		}
234*b1cdbd2cSJim Jagielski 	}
235*b1cdbd2cSJim Jagielski 	return bModified;
236*b1cdbd2cSJim Jagielski }
237*b1cdbd2cSJim Jagielski 
238*b1cdbd2cSJim Jagielski //.........................................................................
239*b1cdbd2cSJim Jagielski }
240*b1cdbd2cSJim Jagielski //.........................................................................
241*b1cdbd2cSJim Jagielski 
242