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_sc.hxx"
26 #include "fapihelper.hxx"
27 
28 #include <algorithm>
29 #include <com/sun/star/lang/XServiceName.hpp>
30 #include <com/sun/star/lang/XMultiServiceFactory.hpp>
31 #include <com/sun/star/beans/XPropertyState.hpp>
32 #include <comphelper/docpasswordhelper.hxx>
33 #include <comphelper/processfactory.hxx>
34 #include <tools/urlobj.hxx>
35 #include <sfx2/objsh.hxx>
36 #include <sfx2/docfile.hxx>
37 #include <sfx2/request.hxx>
38 #include <sfx2/frame.hxx>
39 #include <sfx2/sfxsids.hrc>
40 #include <svl/stritem.hxx>
41 #include <svl/itemset.hxx>
42 #include "miscuno.hxx"
43 
44 using ::rtl::OUString;
45 using ::com::sun::star::uno::Any;
46 using ::com::sun::star::uno::Reference;
47 using ::com::sun::star::uno::Sequence;
48 using ::com::sun::star::uno::Exception;
49 using ::com::sun::star::uno::UNO_QUERY;
50 using ::com::sun::star::uno::UNO_QUERY_THROW;
51 using ::com::sun::star::uno::UNO_SET_THROW;
52 using ::com::sun::star::uno::TypeClass_BOOLEAN;
53 using ::com::sun::star::uno::XInterface;
54 using ::com::sun::star::beans::XPropertySet;
55 using ::com::sun::star::beans::XPropertyState;
56 using ::com::sun::star::lang::XServiceName;
57 using ::com::sun::star::lang::XMultiServiceFactory;
58 using ::com::sun::star::task::PasswordRequestMode_PASSWORD_ENTER;
59 
60 using namespace ::com::sun::star;
61 
62 // Static helper functions ====================================================
63 
GetServiceName(Reference<XInterface> xInt)64 OUString ScfApiHelper::GetServiceName( Reference< XInterface > xInt )
65 {
66     OUString aService;
67     Reference< XServiceName > xServiceName( xInt, UNO_QUERY );
68     if( xServiceName.is() )
69         aService = xServiceName->getServiceName();
70     return aService;
71 }
72 
GetServiceFactory(SfxObjectShell * pShell)73 Reference< XMultiServiceFactory > ScfApiHelper::GetServiceFactory( SfxObjectShell* pShell )
74 {
75     Reference< XMultiServiceFactory > xFactory;
76     if( pShell )
77         xFactory.set( pShell->GetModel(), UNO_QUERY );
78     return xFactory;
79 }
80 
CreateInstance(Reference<XMultiServiceFactory> xFactory,const OUString & rServiceName)81 Reference< XInterface > ScfApiHelper::CreateInstance(
82         Reference< XMultiServiceFactory > xFactory, const OUString& rServiceName )
83 {
84     Reference< XInterface > xInt;
85     if( xFactory.is() )
86     {
87         try
88         {
89             xInt = xFactory->createInstance( rServiceName );
90         }
91         catch( Exception& )
92         {
93             DBG_ERRORFILE( "ScfApiHelper::CreateInstance - cannot create instance" );
94         }
95     }
96     return xInt;
97 }
98 
CreateInstance(SfxObjectShell * pShell,const OUString & rServiceName)99 Reference< XInterface > ScfApiHelper::CreateInstance( SfxObjectShell* pShell, const OUString& rServiceName )
100 {
101     return CreateInstance( GetServiceFactory( pShell ), rServiceName );
102 }
103 
CreateInstance(const OUString & rServiceName)104 Reference< XInterface > ScfApiHelper::CreateInstance( const OUString& rServiceName )
105 {
106     return CreateInstance( ::comphelper::getProcessServiceFactory(), rServiceName );
107 }
108 
CreateInstanceWithArgs(Reference<XMultiServiceFactory> xFactory,const OUString & rServiceName,const Sequence<Any> & rArgs)109 Reference< XInterface > ScfApiHelper::CreateInstanceWithArgs(
110         Reference< XMultiServiceFactory > xFactory, const OUString& rServiceName, const Sequence< Any >& rArgs )
111 {
112     Reference< XInterface > xInt;
113     if( xFactory.is() )
114     {
115         try
116         {
117             xInt = xFactory->createInstanceWithArguments( rServiceName, rArgs );
118         }
119         catch( Exception& )
120         {
121             DBG_ERRORFILE( "ScfApiHelper::CreateInstanceWithArgs - cannot create instance" );
122         }
123     }
124     return xInt;
125 }
126 
127 //UNUSED2008-05  Reference< XInterface > ScfApiHelper::CreateInstanceWithArgs(
128 //UNUSED2008-05          SfxObjectShell* pShell, const OUString& rServiceName, const Sequence< Any >& rArgs )
129 //UNUSED2008-05  {
130 //UNUSED2008-05      return CreateInstanceWithArgs( GetServiceFactory( pShell ), rServiceName, rArgs );
131 //UNUSED2008-05  }
132 
CreateInstanceWithArgs(const OUString & rServiceName,const Sequence<Any> & rArgs)133 Reference< XInterface > ScfApiHelper::CreateInstanceWithArgs(
134         const OUString& rServiceName, const Sequence< Any >& rArgs )
135 {
136     return CreateInstanceWithArgs( ::comphelper::getProcessServiceFactory(), rServiceName, rArgs );
137 }
138 
QueryEncryptionDataForMedium(SfxMedium & rMedium,::comphelper::IDocPasswordVerifier & rVerifier,const::std::vector<OUString> * pDefaultPasswords)139 uno::Sequence< beans::NamedValue > ScfApiHelper::QueryEncryptionDataForMedium( SfxMedium& rMedium,
140         ::comphelper::IDocPasswordVerifier& rVerifier, const ::std::vector< OUString >* pDefaultPasswords )
141 {
142     uno::Sequence< beans::NamedValue > aEncryptionData;
143     SFX_ITEMSET_ARG( rMedium.GetItemSet(), pEncryptionDataItem, SfxUnoAnyItem, SID_ENCRYPTIONDATA, sal_False);
144     if ( pEncryptionDataItem )
145         pEncryptionDataItem->GetValue() >>= aEncryptionData;
146 
147     ::rtl::OUString aPassword;
148     SFX_ITEMSET_ARG( rMedium.GetItemSet(), pPasswordItem, SfxStringItem, SID_PASSWORD, sal_False);
149     if ( pPasswordItem )
150         aPassword = pPasswordItem->GetValue();
151 
152     OUString aDocName = INetURLObject( rMedium.GetOrigURL() ).GetName( INetURLObject::DECODE_WITH_CHARSET );
153 
154     bool bIsDefaultPassword = false;
155     aEncryptionData = ::comphelper::DocPasswordHelper::requestAndVerifyDocPassword(
156         rVerifier, aEncryptionData, aPassword, rMedium.GetInteractionHandler(), aDocName,
157         ::comphelper::DocPasswordRequestType_MS, pDefaultPasswords, &bIsDefaultPassword );
158 
159     rMedium.GetItemSet()->ClearItem( SID_PASSWORD );
160     rMedium.GetItemSet()->ClearItem( SID_ENCRYPTIONDATA );
161 
162     if( !bIsDefaultPassword && (aEncryptionData.getLength() > 0) )
163         rMedium.GetItemSet()->Put( SfxUnoAnyItem( SID_ENCRYPTIONDATA, uno::makeAny( aEncryptionData ) ) );
164 
165     return aEncryptionData;
166 }
167 
168 // Property sets ==============================================================
169 
Set(Reference<XPropertySet> xPropSet)170 void ScfPropertySet::Set( Reference< XPropertySet > xPropSet )
171 {
172     mxPropSet = xPropSet;
173     mxMultiPropSet.set( mxPropSet, UNO_QUERY );
174 }
175 
GetServiceName() const176 OUString ScfPropertySet::GetServiceName() const
177 {
178     return ScfApiHelper::GetServiceName( mxPropSet );
179 }
180 
181 // Get properties -------------------------------------------------------------
182 
HasProperty(const OUString & rPropName) const183 bool ScfPropertySet::HasProperty( const OUString& rPropName ) const
184 {
185     bool bHasProp = false;
186     try
187     {
188         Reference< XPropertyState > xPropState( mxPropSet, UNO_QUERY_THROW );
189         bHasProp = xPropState->getPropertyState( rPropName ) == ::com::sun::star::beans::PropertyState_DIRECT_VALUE;
190     }
191     catch( Exception& )
192     {
193     }
194     return bHasProp;
195 }
196 
GetAnyProperty(Any & rValue,const OUString & rPropName) const197 bool ScfPropertySet::GetAnyProperty( Any& rValue, const OUString& rPropName ) const
198 {
199     bool bHasValue = false;
200     try
201     {
202         if( mxPropSet.is() )
203         {
204             rValue = mxPropSet->getPropertyValue( rPropName );
205             bHasValue = true;
206         }
207     }
208     catch( Exception& )
209     {
210     }
211     return bHasValue;
212 }
213 
GetBoolProperty(const::rtl::OUString & rPropName) const214 bool ScfPropertySet::GetBoolProperty( const ::rtl::OUString& rPropName ) const
215 {
216     Any aAny;
217     return GetAnyProperty( aAny, rPropName ) && ScUnoHelpFunctions::GetBoolFromAny( aAny );
218 }
219 
GetStringProperty(String & rValue,const OUString & rPropName) const220 bool ScfPropertySet::GetStringProperty( String& rValue, const OUString& rPropName ) const
221 {
222     OUString aOUString;
223     bool bRet = GetProperty( aOUString, rPropName );
224     rValue = aOUString;
225     return bRet;
226 }
227 
GetColorProperty(Color & rColor,const::rtl::OUString & rPropName) const228 bool ScfPropertySet::GetColorProperty( Color& rColor, const ::rtl::OUString& rPropName ) const
229 {
230     sal_Int32 nApiColor = 0;
231     bool bRet = GetProperty( nApiColor, rPropName );
232     rColor = ScfApiHelper::ConvertFromApiColor( nApiColor );
233     return bRet;
234 }
235 
GetProperties(Sequence<Any> & rValues,const Sequence<OUString> & rPropNames) const236 void ScfPropertySet::GetProperties( Sequence< Any >& rValues, const Sequence< OUString >& rPropNames ) const
237 {
238     try
239     {
240         DBG_ASSERT( mxMultiPropSet.is(), "ScfPropertySet::GetProperties - multi property set not available" );
241         if( mxMultiPropSet.is() )   // first try the XMultiPropertySet
242         {
243             rValues = mxMultiPropSet->getPropertyValues( rPropNames );
244         }
245         else if( mxPropSet.is() )
246         {
247             sal_Int32 nLen = rPropNames.getLength();
248             const OUString* pPropName = rPropNames.getConstArray();
249             const OUString* pPropNameEnd = pPropName + nLen;
250             rValues.realloc( nLen );
251             Any* pValue = rValues.getArray();
252             for( ; pPropName != pPropNameEnd; ++pPropName, ++pValue )
253                 *pValue = mxPropSet->getPropertyValue( *pPropName );
254         }
255     }
256     catch( Exception& )
257     {
258     }
259 }
260 
261 // Set properties -------------------------------------------------------------
262 
SetAnyProperty(const OUString & rPropName,const Any & rValue)263 void ScfPropertySet::SetAnyProperty( const OUString& rPropName, const Any& rValue )
264 {
265     try
266     {
267         if( mxPropSet.is() )
268             mxPropSet->setPropertyValue( rPropName, rValue );
269     }
270     catch( Exception& )
271     {
272         DBG_ERRORFILE(
273             ByteString( "ScfPropertySet::SetAnyProperty - cannot set property \"" ).
274                 Append( ByteString( String( rPropName ), RTL_TEXTENCODING_ASCII_US ) ).
275                 Append( '"' ).
276                 GetBuffer() );
277     }
278 }
279 
SetProperties(const Sequence<OUString> & rPropNames,const Sequence<Any> & rValues)280 void ScfPropertySet::SetProperties( const Sequence< OUString >& rPropNames, const Sequence< Any >& rValues )
281 {
282     DBG_ASSERT( rPropNames.getLength() == rValues.getLength(), "ScfPropertySet::SetProperties - length of sequences different" );
283     try
284     {
285         if( mxMultiPropSet.is() )   // first try the XMultiPropertySet
286         {
287             mxMultiPropSet->setPropertyValues( rPropNames, rValues );
288         }
289         else if( mxPropSet.is() )
290         {
291             DBG_ERRORFILE( "ScfPropertySet::SetProperties - multi property set not available" );
292             const OUString* pPropName = rPropNames.getConstArray();
293             const OUString* pPropNameEnd = pPropName + rPropNames.getLength();
294             const Any* pValue = rValues.getConstArray();
295             for( ; pPropName != pPropNameEnd; ++pPropName, ++pValue )
296                 mxPropSet->setPropertyValue( *pPropName, *pValue );
297         }
298     }
299     catch( Exception& )
300     {
301         DBG_ERRORFILE( "ScfPropertySet::SetAnyProperty - cannot set multiple properties" );
302     }
303 }
304 
305 // ============================================================================
306 
ScfPropSetHelper(const sal_Char * const * ppcPropNames)307 ScfPropSetHelper::ScfPropSetHelper( const sal_Char* const* ppcPropNames ) :
308     mnNextIdx( 0 )
309 {
310     DBG_ASSERT( ppcPropNames, "ScfPropSetHelper::ScfPropSetHelper - no strings found" );
311 
312     // create OUStrings from ASCII property names
313     typedef ::std::pair< OUString, size_t >     IndexedOUString;
314     typedef ::std::vector< IndexedOUString >    IndexedOUStringVec;
315     IndexedOUStringVec aPropNameVec;
316     for( size_t nVecIdx = 0; *ppcPropNames; ++ppcPropNames, ++nVecIdx )
317     {
318         OUString aPropName = OUString::createFromAscii( *ppcPropNames );
319         aPropNameVec.push_back( IndexedOUString( aPropName, nVecIdx ) );
320     }
321 
322     // sorts the pairs, which will be sorted by first component (the property name)
323     ::std::sort( aPropNameVec.begin(), aPropNameVec.end() );
324 
325     // resize member sequences
326     size_t nSize = aPropNameVec.size();
327     maNameSeq.realloc( static_cast< sal_Int32 >( nSize ) );
328     maValueSeq.realloc( static_cast< sal_Int32 >( nSize ) );
329     maNameOrder.resize( nSize );
330 
331     // fill the property name sequence and store original sort order
332     sal_Int32 nSeqIdx = 0;
333     for( IndexedOUStringVec::const_iterator aIt = aPropNameVec.begin(),
334             aEnd = aPropNameVec.end(); aIt != aEnd; ++aIt, ++nSeqIdx )
335     {
336         maNameSeq[ nSeqIdx ] = aIt->first;
337         maNameOrder[ aIt->second ] = nSeqIdx;
338     }
339 }
340 
341 // read properties ------------------------------------------------------------
342 
ReadFromPropertySet(const ScfPropertySet & rPropSet)343 void ScfPropSetHelper::ReadFromPropertySet( const ScfPropertySet& rPropSet )
344 {
345     rPropSet.GetProperties( maValueSeq, maNameSeq );
346     mnNextIdx = 0;
347 }
348 
ReadValue(UnoAny & rAny)349 bool ScfPropSetHelper::ReadValue( UnoAny& rAny )
350 {
351     UnoAny* pAny = GetNextAny();
352     if( pAny )
353         rAny = *pAny;
354     return pAny != 0;
355 }
356 
ReadValue(String & rString)357 bool ScfPropSetHelper::ReadValue( String& rString )
358 {
359     OUString aOUString;
360     bool bRet = ReadValue( aOUString );
361     rString = aOUString;
362     return bRet;
363 }
364 
ReadValue(Color & rColor)365 bool ScfPropSetHelper::ReadValue( Color& rColor )
366 {
367     sal_Int32 nApiColor(0);
368     bool bRet = ReadValue( nApiColor );
369     rColor = ScfApiHelper::ConvertFromApiColor( nApiColor );
370     return bRet;
371 }
372 
ReadValue(bool & rbValue)373 bool ScfPropSetHelper::ReadValue( bool& rbValue )
374 {
375     Any aAny;
376     bool bRet = ReadValue( aAny );
377     rbValue = ScUnoHelpFunctions::GetBoolFromAny( aAny );
378     return bRet;
379 }
380 
381 // write properties -----------------------------------------------------------
382 
InitializeWrite(bool bClearAllAnys)383 void ScfPropSetHelper::InitializeWrite( bool bClearAllAnys )
384 {
385     mnNextIdx = 0;
386     if( bClearAllAnys )
387         for( sal_Int32 nIdx = 0, nLen = maValueSeq.getLength(); nIdx < nLen; ++nIdx )
388             maValueSeq[ nIdx ].clear();
389 }
390 
WriteValue(const Any & rAny)391 void ScfPropSetHelper::WriteValue( const Any& rAny )
392 {
393     if( UnoAny* pAny = GetNextAny() )
394         *pAny = rAny;
395 }
396 
WriteValue(const bool & rbValue)397 void ScfPropSetHelper::WriteValue( const bool& rbValue )
398 {
399     if( Any* pAny = GetNextAny() )
400         ScUnoHelpFunctions::SetBoolInAny( *pAny, rbValue );
401 }
402 
WriteToPropertySet(ScfPropertySet & rPropSet) const403 void ScfPropSetHelper::WriteToPropertySet( ScfPropertySet& rPropSet ) const
404 {
405     rPropSet.SetProperties( maNameSeq, maValueSeq );
406 }
407 
408 // private --------------------------------------------------------------------
409 
GetNextAny()410 Any* ScfPropSetHelper::GetNextAny()
411 {
412     DBG_ASSERT( mnNextIdx < maNameOrder.size(), "ScfPropSetHelper::GetNextAny - sequence overflow" );
413     Any* pAny = 0;
414     if( mnNextIdx < maNameOrder.size() )
415         pAny = &maValueSeq[ maNameOrder[ mnNextIdx++ ] ];
416     return pAny;
417 }
418 
419 // ============================================================================
420 
421