xref: /AOO41X/main/sfx2/source/doc/guisaveas.cxx (revision d119d52d53d0b2180f2ae51341d882123be2af2b)
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_sfx2.hxx"
26 #include <com/sun/star/ui/dialogs/XExecutableDialog.hpp>
27 #include <com/sun/star/ui/dialogs/XFilePicker.hpp>
28 #include <com/sun/star/ui/dialogs/XFilePickerControlAccess.hpp>
29 #include <com/sun/star/ui/dialogs/CommonFilePickerElementIds.hpp>
30 #include <com/sun/star/ui/dialogs/ExtendedFilePickerElementIds.hpp>
31 #include "com/sun/star/ui/dialogs/TemplateDescription.hpp"
32 #include <com/sun/star/view/XSelectionSupplier.hpp>
33 #include <com/sun/star/beans/XPropertyAccess.hpp>
34 #include <com/sun/star/beans/XPropertySet.hpp>
35 #include <com/sun/star/beans/XPropertyContainer.hpp>
36 #include <com/sun/star/beans/PropertyAttribute.hpp>
37 #include <com/sun/star/document/XExporter.hpp>
38 #include <com/sun/star/document/XDocumentInfoSupplier.hpp>
39 #include <com/sun/star/document/XDocumentInfo.hpp>
40 #include <com/sun/star/task/XInteractionHandler.hpp>
41 #include <com/sun/star/util/DateTime.hpp>
42 #include <com/sun/star/util/XURLTransformer.hpp>
43 #include <com/sun/star/frame/XStorable.hpp>
44 #include <com/sun/star/frame/XStorable2.hpp>
45 #include <com/sun/star/frame/XDispatchProvider.hpp>
46 #include <com/sun/star/frame/XDispatch.hpp>
47 #include <com/sun/star/frame/XTitle.hpp>
48 #include <com/sun/star/util/XModifyListener.hpp>
49 #include <com/sun/star/util/XModifiable.hpp>
50 #include <com/sun/star/util/XModifyBroadcaster.hpp>
51 
52 #include <com/sun/star/util/XCloneable.hpp>
53 #include <com/sun/star/frame/XModuleManager.hpp>
54 #include <com/sun/star/io/IOException.hpp>
55 
56 #include "guisaveas.hxx"
57 
58 #include <unotools/pathoptions.hxx>
59 #include <unotools/pathoptions.hxx>
60 #include <svl/itemset.hxx>
61 #include <svl/eitem.hxx>
62 #include <svl/stritem.hxx>
63 #include <svl/intitem.hxx>
64 #include <unotools/useroptions.hxx>
65 #include <unotools/saveopt.hxx>
66 #include <tools/debug.hxx>
67 #include <tools/urlobj.hxx>
68 #include <comphelper/processfactory.hxx>
69 #include <comphelper/configurationhelper.hxx>
70 #include <comphelper/mimeconfighelper.hxx>
71 #include <vcl/msgbox.hxx>
72 #include <vcl/window.hxx>
73 #include <toolkit/awt/vclxwindow.hxx>
74 
75 #include <sfx2/sfxsids.hrc>
76 #include <doc.hrc>
77 #include <sfx2/sfxresid.hxx>
78 #include <sfx2/docfilt.hxx>
79 #include <sfx2/filedlghelper.hxx>
80 #include <sfx2/app.hxx>
81 #include <sfx2/objsh.hxx>
82 #include <sfx2/dinfdlg.hxx>
83 #include <sfx2/request.hxx>
84 #include <sfxtypes.hxx>
85 #include "alienwarn.hxx"
86 
87 #include "../appl/app.hrc"
88 
89 #define DOCPROPSNUM 17
90 
91 // flags that specify requested operation
92 #define EXPORT_REQUESTED            1
93 #define PDFEXPORT_REQUESTED         2
94 #define PDFDIRECTEXPORT_REQUESTED   4
95 #define WIDEEXPORT_REQUESTED        8
96 #define SAVE_REQUESTED              16
97 #define SAVEAS_REQUESTED            32
98 
99 // possible statuses of save operation
100 #define STATUS_NO_ACTION            0
101 #define STATUS_SAVE                 1
102 #define STATUS_SAVEAS               2
103 #define STATUS_SAVEAS_STANDARDNAME  3
104 
105 const ::rtl::OUString aFilterNameString = ::rtl::OUString::createFromAscii( "FilterName" );
106 const ::rtl::OUString aFilterOptionsString = ::rtl::OUString::createFromAscii( "FilterOptions" );
107 const ::rtl::OUString aFilterDataString    = ::rtl::OUString::createFromAscii( "FilterData" );
108 const ::rtl::OUString aFilterFlagsString   = ::rtl::OUString::createFromAscii( "FilterFlags" );
109 
110 using namespace ::com::sun::star;
111 
112 namespace {
113 //-------------------------------------------------------------------------
getSlotIDFromMode(sal_Int8 nStoreMode)114 static sal_uInt16 getSlotIDFromMode( sal_Int8 nStoreMode )
115 {
116     // This is a temporary hardcoded solution must be removed when
117     // dialogs do not need parameters in SidSet representation any more
118 
119     sal_uInt16 nResult = 0;
120     if ( nStoreMode == EXPORT_REQUESTED )
121         nResult = SID_EXPORTDOC;
122     else if ( nStoreMode == ( EXPORT_REQUESTED | PDFEXPORT_REQUESTED ) )
123         nResult = SID_EXPORTDOCASPDF;
124     else if ( nStoreMode == ( EXPORT_REQUESTED | PDFEXPORT_REQUESTED | PDFDIRECTEXPORT_REQUESTED ) )
125         nResult = SID_DIRECTEXPORTDOCASPDF;
126     else if ( nStoreMode == SAVEAS_REQUESTED || nStoreMode == ( EXPORT_REQUESTED | WIDEEXPORT_REQUESTED ) )
127         nResult = SID_SAVEASDOC;
128     else {
129         DBG_ASSERT( sal_False, "Unacceptable slot name is provided!\n" );
130     }
131 
132     return nResult;
133 }
134 
135 //-------------------------------------------------------------------------
getStoreModeFromSlotName(const::rtl::OUString & aSlotName)136 static sal_uInt8 getStoreModeFromSlotName( const ::rtl::OUString& aSlotName )
137 {
138     sal_uInt8 nResult = 0;
139     if ( aSlotName.equalsAscii( "ExportTo" ) )
140         nResult = EXPORT_REQUESTED;
141     else if ( aSlotName.equalsAscii( "ExportToPDF" ) )
142         nResult = EXPORT_REQUESTED | PDFEXPORT_REQUESTED;
143     else if ( aSlotName.equalsAscii( "ExportDirectToPDF" ) )
144         nResult = EXPORT_REQUESTED | PDFEXPORT_REQUESTED | PDFDIRECTEXPORT_REQUESTED;
145     else if ( aSlotName.equalsAscii( "Save" ) )
146         nResult = SAVE_REQUESTED;
147     else if ( aSlotName.equalsAscii( "SaveAs" ) )
148         nResult = SAVEAS_REQUESTED;
149     else
150         throw task::ErrorCodeIOException( ::rtl::OUString(),
151                                             uno::Reference< uno::XInterface >(),
152                                             ERRCODE_IO_INVALIDPARAMETER );
153 
154     return nResult;
155 }
156 
157 //-------------------------------------------------------------------------
getMustFlags(sal_Int8 nStoreMode)158 static sal_Int32 getMustFlags( sal_Int8 nStoreMode )
159 {
160     return ( SFX_FILTER_EXPORT
161             | ( ( ( nStoreMode & EXPORT_REQUESTED ) && !( nStoreMode & WIDEEXPORT_REQUESTED ) ) ? 0 : SFX_FILTER_IMPORT ) );
162 }
163 
164 //-------------------------------------------------------------------------
getDontFlags(sal_Int8 nStoreMode)165 static sal_Int32 getDontFlags( sal_Int8 nStoreMode )
166 {
167     return ( SFX_FILTER_INTERNAL
168             | SFX_FILTER_NOTINFILEDLG
169             | ( ( ( nStoreMode & EXPORT_REQUESTED ) && !( nStoreMode & WIDEEXPORT_REQUESTED ) ) ? SFX_FILTER_IMPORT : 0 ) );
170 }
171 
172 //=========================================================================
173 // class DocumentSettingsGuard
174 //=========================================================================
175 
176 class DocumentSettingsGuard
177 {
178     uno::Reference< beans::XPropertySet > m_xDocumentSettings;
179     sal_Bool m_bPreserveReadOnly;
180     sal_Bool m_bReadOnlySupported;
181 
182     sal_Bool m_bRestoreSettings;
183 public:
DocumentSettingsGuard(const uno::Reference<frame::XModel> & xModel,sal_Bool bReadOnly,sal_Bool bRestore)184     DocumentSettingsGuard( const uno::Reference< frame::XModel >& xModel, sal_Bool bReadOnly, sal_Bool bRestore )
185     : m_bPreserveReadOnly( sal_False )
186     , m_bReadOnlySupported( sal_False )
187     , m_bRestoreSettings( bRestore )
188     {
189         try
190         {
191             uno::Reference< lang::XMultiServiceFactory > xDocSettingsSupplier( xModel, uno::UNO_QUERY_THROW );
192             m_xDocumentSettings.set(
193                 xDocSettingsSupplier->createInstance(
194                     ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "com.sun.star.document.Settings" ) ) ),
195                 uno::UNO_QUERY_THROW );
196 
197             ::rtl::OUString aLoadReadonlyString( RTL_CONSTASCII_USTRINGPARAM( "LoadReadonly" ) );
198 
199             try
200             {
201                 m_xDocumentSettings->getPropertyValue( aLoadReadonlyString ) >>= m_bPreserveReadOnly;
202                 m_xDocumentSettings->setPropertyValue( aLoadReadonlyString, uno::makeAny( bReadOnly ) );
203                 m_bReadOnlySupported = sal_True;
204             }
205             catch( uno::Exception& )
206             {}
207         }
208         catch( uno::Exception& )
209         {}
210 
211         if ( ( bReadOnly && !m_bReadOnlySupported ) )
212             throw uno::RuntimeException(); // the user could provide the data, so it must be stored
213     }
214 
~DocumentSettingsGuard()215     ~DocumentSettingsGuard()
216     {
217         if ( m_bRestoreSettings )
218         {
219             ::rtl::OUString aLoadReadonlyString( RTL_CONSTASCII_USTRINGPARAM( "LoadReadonly" ) );
220 
221             try
222             {
223                 if ( m_bReadOnlySupported )
224                     m_xDocumentSettings->setPropertyValue( aLoadReadonlyString, uno::makeAny( m_bPreserveReadOnly ) );
225             }
226             catch( uno::Exception& )
227             {
228                 OSL_ASSERT( "Unexpected exception!" );
229             }
230         }
231     }
232 };
233 } // anonymous namespace
234 
235 //=========================================================================
236 // class ModelData_Impl
237 //=========================================================================
238 class ModelData_Impl
239 {
240     SfxStoringHelper* m_pOwner;
241     uno::Reference< frame::XModel > m_xModel;
242     uno::Reference< frame::XStorable > m_xStorable;
243     uno::Reference< frame::XStorable2 > m_xStorable2;
244     uno::Reference< util::XModifiable > m_xModifiable;
245 
246     ::rtl::OUString m_aModuleName;
247     ::comphelper::SequenceAsHashMap* m_pDocumentPropsHM;
248     ::comphelper::SequenceAsHashMap* m_pModulePropsHM;
249 
250     ::comphelper::SequenceAsHashMap m_aMediaDescrHM;
251 
252     sal_Bool m_bRecommendReadOnly;
253 
254 public:
255     ModelData_Impl( SfxStoringHelper& aOwner,
256                     const uno::Reference< frame::XModel >& xModel,
257                     const uno::Sequence< beans::PropertyValue >& aMediaDescr );
258 
259     ~ModelData_Impl();
260 
261     void FreeDocumentProps();
262 
263     uno::Reference< frame::XModel > GetModel();
264     uno::Reference< frame::XStorable > GetStorable();
265     uno::Reference< frame::XStorable2 > GetStorable2();
266     uno::Reference< util::XModifiable > GetModifiable();
267 
GetMediaDescr()268     ::comphelper::SequenceAsHashMap& GetMediaDescr() { return m_aMediaDescrHM; }
269 
IsRecommendReadOnly()270     sal_Bool IsRecommendReadOnly() { return m_bRecommendReadOnly; }
271 
272     const ::comphelper::SequenceAsHashMap& GetDocProps();
273 
274     ::rtl::OUString GetModuleName();
275     const ::comphelper::SequenceAsHashMap& GetModuleProps();
276 
277     void CheckInteractionHandler();
278 
279 
280     ::rtl::OUString GetDocServiceName();
281     uno::Sequence< beans::PropertyValue > GetDocServiceDefaultFilterCheckFlags( sal_Int32 nMust, sal_Int32 nDont );
282     uno::Sequence< beans::PropertyValue > GetDocServiceAnyFilter( sal_Int32 nMust, sal_Int32 nDont );
283     uno::Sequence< beans::PropertyValue > GetPreselectedFilter_Impl( sal_Int8 nStoreMode );
284     uno::Sequence< beans::PropertyValue > GetDocServiceDefaultFilter();
285 
286     sal_Bool ExecuteFilterDialog_Impl( const ::rtl::OUString& aFilterName );
287 
288     sal_Int8 CheckSaveAcceptable( sal_Int8 nCurStatus );
289     sal_Int8 CheckStateForSave();
290 
291     sal_Int8 CheckFilter( const ::rtl::OUString& );
292 
293     sal_Bool CheckFilterOptionsDialogExistence();
294 
295     sal_Bool OutputFileDialog( sal_Int8 nStoreMode,
296                                 const ::comphelper::SequenceAsHashMap& aPreselectedFilterPropsHM,
297                                 sal_Bool bSetStandardName,
298                                 ::rtl::OUString& aSuggestedName,
299                                 sal_Bool bPreselectPassword,
300                                 const ::rtl::OUString& aSuggestedDir,
301                                 sal_Int16 nDialog,
302                                 const ::rtl::OUString& rStandardDir,
303                                 const ::com::sun::star::uno::Sequence< ::rtl::OUString >& rBlackList
304                                 );
305 
306     sal_Bool ShowDocumentInfoDialog();
307 
308     ::rtl::OUString GetReccomendedDir( const ::rtl::OUString& aSuggestedDir,
309                                        const sfx2::FileDialogHelper::Context& aCtxt );
310     ::rtl::OUString GetReccomendedName( const ::rtl::OUString& aSuggestedName,
311                                         const ::rtl::OUString& aTypeName );
312 
313 };
314 
315 //-------------------------------------------------------------------------
ModelData_Impl(SfxStoringHelper & aOwner,const uno::Reference<frame::XModel> & xModel,const uno::Sequence<beans::PropertyValue> & aMediaDescr)316 ModelData_Impl::ModelData_Impl( SfxStoringHelper& aOwner,
317                                 const uno::Reference< frame::XModel >& xModel,
318                                 const uno::Sequence< beans::PropertyValue >& aMediaDescr )
319 : m_pOwner( &aOwner )
320 , m_xModel( xModel )
321 , m_pDocumentPropsHM( NULL )
322 , m_pModulePropsHM( NULL )
323 , m_aMediaDescrHM( aMediaDescr )
324 , m_bRecommendReadOnly( sal_False )
325 {
326     CheckInteractionHandler();
327 }
328 
329 //-------------------------------------------------------------------------
~ModelData_Impl()330 ModelData_Impl::~ModelData_Impl()
331 {
332     FreeDocumentProps();
333     if ( m_pDocumentPropsHM )
334         delete m_pDocumentPropsHM;
335 
336     if ( m_pModulePropsHM )
337         delete m_pModulePropsHM;
338 }
339 
340 //-------------------------------------------------------------------------
FreeDocumentProps()341 void ModelData_Impl::FreeDocumentProps()
342 {
343     if ( m_pDocumentPropsHM )
344     {
345         delete m_pDocumentPropsHM;
346         m_pDocumentPropsHM = NULL;
347     }
348 }
349 
350 //-------------------------------------------------------------------------
GetModel()351 uno::Reference< frame::XModel > ModelData_Impl::GetModel()
352 {
353     if ( !m_xModel.is() )
354         throw uno::RuntimeException();
355 
356     return m_xModel;
357 }
358 
359 //-------------------------------------------------------------------------
GetStorable()360 uno::Reference< frame::XStorable > ModelData_Impl::GetStorable()
361 {
362     if ( !m_xStorable.is() )
363     {
364         m_xStorable = uno::Reference< frame::XStorable >( m_xModel, uno::UNO_QUERY );
365         if ( !m_xStorable.is() )
366             throw uno::RuntimeException();
367     }
368 
369     return m_xStorable;
370 }
371 
372 //-------------------------------------------------------------------------
GetStorable2()373 uno::Reference< frame::XStorable2 > ModelData_Impl::GetStorable2()
374 {
375     if ( !m_xStorable2.is() )
376     {
377         m_xStorable2 = uno::Reference< frame::XStorable2 >( m_xModel, uno::UNO_QUERY );
378         if ( !m_xStorable2.is() )
379             throw uno::RuntimeException();
380     }
381 
382     return m_xStorable2;
383 }
384 
385 //-------------------------------------------------------------------------
GetModifiable()386 uno::Reference< util::XModifiable > ModelData_Impl::GetModifiable()
387 {
388     if ( !m_xModifiable.is() )
389     {
390         m_xModifiable = uno::Reference< util::XModifiable >( m_xModel, uno::UNO_QUERY );
391         if ( !m_xModifiable.is() )
392             throw uno::RuntimeException();
393     }
394 
395     return m_xModifiable;
396 }
397 
398 //-------------------------------------------------------------------------
GetDocProps()399 const ::comphelper::SequenceAsHashMap& ModelData_Impl::GetDocProps()
400 {
401     if ( !m_pDocumentPropsHM )
402         m_pDocumentPropsHM = new ::comphelper::SequenceAsHashMap( GetModel()->getArgs() );
403 
404     return *m_pDocumentPropsHM;
405 }
406 
407 //-------------------------------------------------------------------------
GetModuleName()408 ::rtl::OUString ModelData_Impl::GetModuleName()
409 {
410     if ( !m_aModuleName.getLength() )
411     {
412         m_aModuleName = m_pOwner->GetModuleManager()->identify(
413                                                 uno::Reference< uno::XInterface >( m_xModel, uno::UNO_QUERY ) );
414         if ( !m_aModuleName.getLength() )
415             throw uno::RuntimeException(); // TODO:
416     }
417     return m_aModuleName;
418 }
419 
420 //-------------------------------------------------------------------------
GetModuleProps()421 const ::comphelper::SequenceAsHashMap& ModelData_Impl::GetModuleProps()
422 {
423     if ( !m_pModulePropsHM )
424     {
425         uno::Sequence< beans::PropertyValue > aModuleProps;
426         m_pOwner->GetNamedModuleManager()->getByName( GetModuleName() ) >>= aModuleProps;
427         if ( !aModuleProps.getLength() )
428             throw uno::RuntimeException(); // TODO;
429         m_pModulePropsHM = new ::comphelper::SequenceAsHashMap( aModuleProps );
430     }
431 
432     return *m_pModulePropsHM;
433 }
434 
435 //-------------------------------------------------------------------------
GetDocServiceName()436 ::rtl::OUString ModelData_Impl::GetDocServiceName()
437 {
438     return GetModuleProps().getUnpackedValueOrDefault(::rtl::OUString::createFromAscii( "ooSetupFactoryDocumentService" ), ::rtl::OUString());
439 }
440 
441 //-------------------------------------------------------------------------
CheckInteractionHandler()442 void ModelData_Impl::CheckInteractionHandler()
443 {
444     ::comphelper::SequenceAsHashMap::const_iterator aInteractIter =
445             m_aMediaDescrHM.find( ::rtl::OUString::createFromAscii( "InteractionHandler" ) );
446 
447     if ( aInteractIter == m_aMediaDescrHM.end() )
448     {
449         try {
450             m_aMediaDescrHM[ ::rtl::OUString::createFromAscii( "InteractionHandler" ) ]
451                 <<= uno::Reference< task::XInteractionHandler >(
452                             m_pOwner->GetServiceFactory()->createInstance(
453                                             DEFINE_CONST_UNICODE("com.sun.star.task.InteractionHandler") ),
454                             uno::UNO_QUERY );
455         }
456         catch( uno::Exception& )
457         {
458         }
459     }
460     else
461     {
462         uno::Reference< task::XInteractionHandler > xInteract;
463         DBG_ASSERT( ( aInteractIter->second >>= xInteract ) && xInteract.is(), "Broken interaction handler is provided!\n" );
464     }
465 }
466 
467 //-------------------------------------------------------------------------
GetDocServiceDefaultFilter()468 uno::Sequence< beans::PropertyValue > ModelData_Impl::GetDocServiceDefaultFilter()
469 {
470     uno::Sequence< beans::PropertyValue > aProps;
471 
472     ::rtl::OUString aFilterName = GetModuleProps().getUnpackedValueOrDefault(
473                                                                 ::rtl::OUString::createFromAscii( "ooSetupFactoryDefaultFilter" ),
474                                                                 ::rtl::OUString() );
475 
476     m_pOwner->GetFilterConfiguration()->getByName( aFilterName ) >>= aProps;
477 
478     return aProps;
479 }
480 
481 //-------------------------------------------------------------------------
GetDocServiceDefaultFilterCheckFlags(sal_Int32 nMust,sal_Int32 nDont)482 uno::Sequence< beans::PropertyValue > ModelData_Impl::GetDocServiceDefaultFilterCheckFlags( sal_Int32 nMust,
483                                                                                                 sal_Int32 nDont )
484 {
485     uno::Sequence< beans::PropertyValue > aFilterProps;
486     uno::Sequence< beans::PropertyValue > aProps = GetDocServiceDefaultFilter();
487     if ( aProps.getLength() )
488     {
489         ::comphelper::SequenceAsHashMap aFiltHM( aProps );
490         sal_Int32 nFlags = aFiltHM.getUnpackedValueOrDefault( ::rtl::OUString::createFromAscii( "Flags" ),
491                                                         (sal_Int32)0 );
492         if ( ( ( nFlags & nMust ) == nMust ) && !( nFlags & nDont ) )
493             aFilterProps = aProps;
494     }
495 
496     return aFilterProps;
497 }
498 
499 
500 //-------------------------------------------------------------------------
GetDocServiceAnyFilter(sal_Int32 nMust,sal_Int32 nDont)501 uno::Sequence< beans::PropertyValue > ModelData_Impl::GetDocServiceAnyFilter( sal_Int32 nMust, sal_Int32 nDont )
502 {
503     uno::Sequence< beans::NamedValue > aSearchRequest( 1 );
504     aSearchRequest[0].Name = ::rtl::OUString::createFromAscii( "DocumentService" );
505     aSearchRequest[0].Value <<= GetDocServiceName();
506 
507     return ::comphelper::MimeConfigurationHelper::SearchForFilter( m_pOwner->GetFilterQuery(), aSearchRequest, nMust, nDont );
508 }
509 
510 //-------------------------------------------------------------------------
GetPreselectedFilter_Impl(sal_Int8 nStoreMode)511 uno::Sequence< beans::PropertyValue > ModelData_Impl::GetPreselectedFilter_Impl( sal_Int8 nStoreMode )
512 {
513     uno::Sequence< beans::PropertyValue > aFilterProps;
514 
515     sal_Int32 nMust = getMustFlags( nStoreMode );
516     sal_Int32 nDont = getDontFlags( nStoreMode );
517 
518     if ( nStoreMode & PDFEXPORT_REQUESTED )
519     {
520         // Preselect PDF-Filter for EXPORT
521         uno::Sequence< beans::NamedValue > aSearchRequest( 2 );
522         aSearchRequest[0].Name = ::rtl::OUString::createFromAscii( "Type" );
523         aSearchRequest[0].Value <<= ::rtl::OUString::createFromAscii( "pdf_Portable_Document_Format" );
524         aSearchRequest[1].Name = ::rtl::OUString::createFromAscii( "DocumentService" );
525         aSearchRequest[1].Value <<= GetDocServiceName();
526 
527         aFilterProps = ::comphelper::MimeConfigurationHelper::SearchForFilter( m_pOwner->GetFilterQuery(), aSearchRequest, nMust, nDont );
528     }
529     else
530     {
531         aFilterProps = GetDocServiceDefaultFilterCheckFlags( nMust, nDont );
532 
533         if ( !aFilterProps.getLength() )
534         {
535             // the default filter was not faund, use just the first acceptable one
536             aFilterProps = GetDocServiceAnyFilter( nMust, nDont );
537         }
538     }
539 
540     return aFilterProps;
541 }
542 
543 //-------------------------------------------------------------------------
ExecuteFilterDialog_Impl(const::rtl::OUString & aFilterName)544 sal_Bool ModelData_Impl::ExecuteFilterDialog_Impl( const ::rtl::OUString& aFilterName )
545 {
546     sal_Bool bDialogUsed = sal_False;
547 
548     try {
549         uno::Sequence < beans::PropertyValue > aProps;
550         uno::Any aAny = m_pOwner->GetFilterConfiguration()->getByName( aFilterName );
551         if ( aAny >>= aProps )
552         {
553             sal_Int32 nPropertyCount = aProps.getLength();
554             for( sal_Int32 nProperty=0; nProperty < nPropertyCount; ++nProperty )
555                 if( aProps[nProperty].Name.equals( ::rtl::OUString::createFromAscii("UIComponent")) )
556                 {
557                     ::rtl::OUString aServiceName;
558                     aProps[nProperty].Value >>= aServiceName;
559                     if( aServiceName.getLength() )
560                     {
561                         uno::Reference< ui::dialogs::XExecutableDialog > xFilterDialog(
562                                                     m_pOwner->GetServiceFactory()->createInstance( aServiceName ), uno::UNO_QUERY );
563                         uno::Reference< beans::XPropertyAccess > xFilterProperties( xFilterDialog, uno::UNO_QUERY );
564 
565                         if( xFilterDialog.is() && xFilterProperties.is() )
566                         {
567                             bDialogUsed = sal_True;
568 
569                             uno::Reference< document::XExporter > xExporter( xFilterDialog, uno::UNO_QUERY );
570                             if( xExporter.is() )
571                                 xExporter->setSourceDocument(
572                                     uno::Reference< lang::XComponent >( GetModel(), uno::UNO_QUERY ) );
573 
574                             uno::Sequence< beans::PropertyValue > aPropsForDialog;
575                             GetMediaDescr() >> aPropsForDialog;
576                             xFilterProperties->setPropertyValues( aPropsForDialog );
577 
578                             if( xFilterDialog->execute() )
579                             {
580                                 uno::Sequence< beans::PropertyValue > aPropsFromDialog =
581                                                                             xFilterProperties->getPropertyValues();
582                                 for ( sal_Int32 nInd = 0; nInd < aPropsFromDialog.getLength(); nInd++ )
583                                     GetMediaDescr()[aPropsFromDialog[nInd].Name] = aPropsFromDialog[nInd].Value;
584                             }
585                             else
586                             {
587                                 throw task::ErrorCodeIOException( ::rtl::OUString(),
588                                                                     uno::Reference< uno::XInterface >(),
589                                                                     ERRCODE_IO_ABORT );
590                             }
591                         }
592                     }
593 
594                     break;
595                 }
596         }
597     }
598     catch( container::NoSuchElementException& )
599     {
600         // the filter name is unknown
601         throw task::ErrorCodeIOException( ::rtl::OUString(),
602                                             uno::Reference< uno::XInterface >(),
603                                             ERRCODE_IO_INVALIDPARAMETER );
604     }
605     catch( task::ErrorCodeIOException& )
606     {
607         throw;
608     }
609     catch( uno::Exception& )
610     {
611     }
612 
613     return bDialogUsed;
614 }
615 
616 //-------------------------------------------------------------------------
CheckSaveAcceptable(sal_Int8 nCurStatus)617 sal_Int8 ModelData_Impl::CheckSaveAcceptable( sal_Int8 nCurStatus )
618 {
619     sal_Int8 nResult = nCurStatus;
620 
621     if ( nResult != STATUS_NO_ACTION && GetStorable()->hasLocation() )
622     {
623         // check whether save is acceptable by the configuration
624         // it is done only for documents that have persistence already
625         uno::Reference< uno::XInterface > xCommonConfig = ::comphelper::ConfigurationHelper::openConfig(
626                             m_pOwner->GetServiceFactory(),
627                             ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "/org.openoffice.Office.Common" ) ),
628                             ::comphelper::ConfigurationHelper::E_STANDARD );
629         if ( !xCommonConfig.is() )
630             throw uno::RuntimeException(); // should the saving proceed as usual instead?
631 
632         try
633         {
634             sal_Bool bAlwaysSaveAs = sal_False;
635 
636             // the saving is acceptable
637             // in case the configuration entry is not set or set to false
638             // or in case of version creation
639             ::rtl::OUString aVersionCommentString = ::rtl::OUString::createFromAscii( "VersionComment" );
640             if ( ( ::comphelper::ConfigurationHelper::readRelativeKey(
641                     xCommonConfig,
642                     ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Save/Document/" ) ),
643                     ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "AlwaysSaveAs" ) ) ) >>= bAlwaysSaveAs )
644             && bAlwaysSaveAs
645             && GetMediaDescr().find( aVersionCommentString ) == GetMediaDescr().end() )
646             {
647                 // notify the user that SaveAs is going to be done
648                 String aString( SfxResId( STR_NEW_FILENAME_SAVE ) );
649                 Window* pWin = SfxStoringHelper::GetModelWindow( m_xModel );
650                 QueryBox aMessageBox( pWin, WB_OK_CANCEL | WB_DEF_OK, aString );
651                 if ( aMessageBox.Execute() == RET_OK )
652                     nResult = STATUS_SAVEAS;
653                 else
654                     nResult = STATUS_NO_ACTION;
655             }
656         }
657         catch( uno::Exception& )
658         {
659             // impossibility to get the configuration access means normal saving flow for now
660         }
661     }
662 
663     return nResult;
664 }
665 
666 //-------------------------------------------------------------------------
CheckStateForSave()667 sal_Int8 ModelData_Impl::CheckStateForSave()
668 {
669     // if the document is readonly or a new one a SaveAs operation must be used
670     if ( !GetStorable()->hasLocation() || GetStorable()->isReadonly() )
671         return STATUS_SAVEAS;
672 
673     // check acceptable entries for media descriptor
674     sal_Bool bVersInfoNeedsStore = sal_False;
675     ::comphelper::SequenceAsHashMap aAcceptedArgs;
676 
677     ::rtl::OUString aVersionCommentString = ::rtl::OUString::createFromAscii( "VersionComment" );
678     ::rtl::OUString aAuthorString = ::rtl::OUString::createFromAscii( "Author" );
679     ::rtl::OUString aInteractionHandlerString = ::rtl::OUString::createFromAscii( "InteractionHandler" );
680     ::rtl::OUString aStatusIndicatorString = ::rtl::OUString::createFromAscii( "StatusIndicator" );
681     ::rtl::OUString aFailOnWarningString = ::rtl::OUString::createFromAscii( "FailOnWarning" );
682 
683     if ( GetMediaDescr().find( aVersionCommentString ) != GetMediaDescr().end() )
684     {
685         bVersInfoNeedsStore = sal_True;
686         aAcceptedArgs[ aVersionCommentString ] = GetMediaDescr()[ aVersionCommentString ];
687     }
688     if ( GetMediaDescr().find( aAuthorString ) != GetMediaDescr().end() )
689         aAcceptedArgs[ aAuthorString ] = GetMediaDescr()[ aAuthorString ];
690     if ( GetMediaDescr().find( aInteractionHandlerString ) != GetMediaDescr().end() )
691         aAcceptedArgs[ aInteractionHandlerString ] = GetMediaDescr()[ aInteractionHandlerString ];
692     if ( GetMediaDescr().find( aStatusIndicatorString ) != GetMediaDescr().end() )
693         aAcceptedArgs[ aStatusIndicatorString ] = GetMediaDescr()[ aStatusIndicatorString ];
694     if ( GetMediaDescr().find( aFailOnWarningString ) != GetMediaDescr().end() )
695         aAcceptedArgs[ aFailOnWarningString ] = GetMediaDescr()[ aFailOnWarningString ];
696 
697     // remove unacceptable entry if there is any
698     DBG_ASSERT( GetMediaDescr().size() == aAcceptedArgs.size(),
699                 "Unacceptable parameters are provided in Save request!\n" );
700     if ( GetMediaDescr().size() != aAcceptedArgs.size() )
701         GetMediaDescr() = aAcceptedArgs;
702 
703     // the document must be modified
704     if ( !GetModifiable()->isModified() && !bVersInfoNeedsStore )
705         return STATUS_NO_ACTION;
706 
707     // check that the old filter is acceptable
708     ::rtl::OUString aOldFilterName = GetDocProps().getUnpackedValueOrDefault(
709                                                     aFilterNameString,
710                                                     ::rtl::OUString() );
711     sal_Int8 nResult = CheckFilter( aOldFilterName );
712 
713     return nResult;
714 }
715 
CheckFilter(const::rtl::OUString & aFilterName)716 sal_Int8 ModelData_Impl::CheckFilter( const ::rtl::OUString& aFilterName )
717 {
718     ::comphelper::SequenceAsHashMap aFiltPropsHM;
719     sal_Int32 nFiltFlags = 0;
720     if ( aFilterName.getLength() )
721     {
722         // get properties of filter
723         uno::Sequence< beans::PropertyValue > aFilterProps;
724         if ( aFilterName.getLength() )
725             m_pOwner->GetFilterConfiguration()->getByName( aFilterName ) >>= aFilterProps;
726 
727         aFiltPropsHM = ::comphelper::SequenceAsHashMap( aFilterProps );
728         nFiltFlags = aFiltPropsHM.getUnpackedValueOrDefault( ::rtl::OUString::createFromAscii( "Flags" ), (sal_Int32)0 );
729     }
730 
731     // only a temporary solution until default filter retrieving feature is implemented
732     // then GetDocServiceDefaultFilter() must be used
733     ::comphelper::SequenceAsHashMap aDefFiltPropsHM = GetDocServiceDefaultFilterCheckFlags( 3, 0 );
734     sal_Int32 nDefFiltFlags = aDefFiltPropsHM.getUnpackedValueOrDefault( ::rtl::OUString::createFromAscii( "Flags" ), (sal_Int32)0 );
735 
736     // if the old filter is not acceptable
737     // and there is no default filter or it is not acceptable for requested parameters then proceed with saveAs
738     if ( ( !aFiltPropsHM.size() || !( nFiltFlags & SFX_FILTER_EXPORT ) )
739       && ( !aDefFiltPropsHM.size() || !( nDefFiltFlags & SFX_FILTER_EXPORT ) || nDefFiltFlags & SFX_FILTER_INTERNAL ) )
740         return STATUS_SAVEAS;
741 
742     // so at this point there is either an acceptable old filter or default one
743     if ( !aFiltPropsHM.size() || !( nFiltFlags & SFX_FILTER_EXPORT ) )
744     {
745         // so the default filter must be acceptable
746         return STATUS_SAVEAS_STANDARDNAME;
747     }
748     else if ( ( !( nFiltFlags & SFX_FILTER_OWN ) || ( nFiltFlags & SFX_FILTER_ALIEN ) )
749            && aDefFiltPropsHM.size()
750            && ( nDefFiltFlags & SFX_FILTER_EXPORT ) && !( nDefFiltFlags & SFX_FILTER_INTERNAL ))
751     {
752         // the default filter is acceptable and the old filter is alian one
753         // so ask to make a saveAs operation
754         ::rtl::OUString aUIName = aFiltPropsHM.getUnpackedValueOrDefault( ::rtl::OUString::createFromAscii( "UIName" ),
755                                                                                 ::rtl::OUString() );
756         ::rtl::OUString aDefUIName = aDefFiltPropsHM.getUnpackedValueOrDefault( ::rtl::OUString::createFromAscii( "UIName" ),
757                                                                                 ::rtl::OUString() );
758         ::rtl::OUString aPreusedFilterName = GetDocProps().getUnpackedValueOrDefault(
759                                                     ::rtl::OUString::createFromAscii( "PreusedFilterName" ),
760                                                     ::rtl::OUString() );
761         if ( !aPreusedFilterName.equals( aFilterName ) && !aUIName.equals( aDefUIName ) )
762         {
763             if ( !SfxStoringHelper::WarnUnacceptableFormat( GetModel(), aUIName, aDefUIName, sal_True ) )
764                 return STATUS_SAVEAS_STANDARDNAME;
765         }
766     }
767 
768     return STATUS_SAVE;
769 }
770 
771 //-------------------------------------------------------------------------
CheckFilterOptionsDialogExistence()772 sal_Bool ModelData_Impl::CheckFilterOptionsDialogExistence()
773 {
774     uno::Sequence< beans::NamedValue > aSearchRequest( 1 );
775     aSearchRequest[0].Name = ::rtl::OUString::createFromAscii( "DocumentService" );
776     aSearchRequest[0].Value <<= GetDocServiceName();
777 
778     uno::Reference< container::XEnumeration > xFilterEnum =
779                                     m_pOwner->GetFilterQuery()->createSubSetEnumerationByProperties( aSearchRequest );
780 
781     while ( xFilterEnum->hasMoreElements() )
782     {
783         uno::Sequence< beans::PropertyValue > pProps;
784         if ( xFilterEnum->nextElement() >>= pProps )
785         {
786             ::comphelper::SequenceAsHashMap aPropsHM( pProps );
787             ::rtl::OUString aUIServName = aPropsHM.getUnpackedValueOrDefault(
788                                             ::rtl::OUString::createFromAscii( "UIComponent" ),
789                                             ::rtl::OUString() );
790             if ( aUIServName.getLength() )
791                 return sal_True;
792         }
793     }
794 
795     return sal_False;
796 }
797 
798 //-------------------------------------------------------------------------
OutputFileDialog(sal_Int8 nStoreMode,const::comphelper::SequenceAsHashMap & aPreselectedFilterPropsHM,sal_Bool bSetStandardName,::rtl::OUString & aSuggestedName,sal_Bool bPreselectPassword,const::rtl::OUString & aSuggestedDir,sal_Int16 nDialog,const::rtl::OUString & rStandardDir,const::com::sun::star::uno::Sequence<::rtl::OUString> & rBlackList)799 sal_Bool ModelData_Impl::OutputFileDialog( sal_Int8 nStoreMode,
800                                             const ::comphelper::SequenceAsHashMap& aPreselectedFilterPropsHM,
801                                             sal_Bool bSetStandardName,
802                                             ::rtl::OUString& aSuggestedName,
803                                             sal_Bool bPreselectPassword,
804                                             const ::rtl::OUString& aSuggestedDir,
805                                             sal_Int16 nDialog,
806                                             const ::rtl::OUString& rStandardDir,
807                                             const ::com::sun::star::uno::Sequence< ::rtl::OUString >& rBlackList)
808 {
809     sal_Bool bUseFilterOptions = sal_False;
810 
811     ::comphelper::SequenceAsHashMap::const_iterator aOverwriteIter =
812                 GetMediaDescr().find( ::rtl::OUString::createFromAscii( "Overwrite" ) );
813 
814     // the file name must be specified if overwrite option is set
815     if ( aOverwriteIter != GetMediaDescr().end() )
816         throw task::ErrorCodeIOException( ::rtl::OUString(),
817                                             uno::Reference< uno::XInterface >(),
818                                             ERRCODE_IO_INVALIDPARAMETER );
819 
820     // no target file name is specified
821     // we need to show the file dialog
822 
823     // check if we have a filter which allows for filter options, so we need a corresponding checkbox in the dialog
824     sal_Bool bAllowOptions = sal_False;
825 
826     // in case of Export, filter options dialog is used if available
827     if( !( nStoreMode & EXPORT_REQUESTED ) || ( nStoreMode & WIDEEXPORT_REQUESTED ) )
828         bAllowOptions = CheckFilterOptionsDialogExistence();
829 
830     // get the filename by dialog ...
831     // create the file dialog
832     sal_Int16  aDialogMode = bAllowOptions
833         ? (com::sun::star::ui::dialogs::TemplateDescription::
834            FILESAVE_AUTOEXTENSION_PASSWORD_FILTEROPTIONS)
835         : (com::sun::star::ui::dialogs::TemplateDescription::
836            FILESAVE_AUTOEXTENSION_PASSWORD);
837     sal_Int64 aDialogFlags = 0;
838 
839     if( ( nStoreMode & EXPORT_REQUESTED ) && !( nStoreMode & WIDEEXPORT_REQUESTED ) )
840     {
841         if ( nStoreMode & PDFEXPORT_REQUESTED )
842             aDialogMode = com::sun::star::ui::dialogs::TemplateDescription::
843                 FILESAVE_AUTOEXTENSION;
844         else
845             aDialogMode = com::sun::star::ui::dialogs::TemplateDescription::
846                 FILESAVE_AUTOEXTENSION_SELECTION;
847         aDialogFlags = SFXWB_EXPORT;
848     }
849 
850     sfx2::FileDialogHelper* pFileDlg = NULL;
851 
852     ::rtl::OUString aDocServiceName = GetDocServiceName();
853     DBG_ASSERT( aDocServiceName.getLength(), "No document service for this module set!" );
854 
855     sal_Int32 nMust = getMustFlags( nStoreMode );
856     sal_Int32 nDont = getDontFlags( nStoreMode );
857     sfx2::FileDialogHelper::Context eCtxt = sfx2::FileDialogHelper::UNKNOWN_CONTEXT;
858 
859     if ( ( nStoreMode & EXPORT_REQUESTED ) && !( nStoreMode & WIDEEXPORT_REQUESTED ) )
860     {
861         if ( ( nStoreMode & PDFEXPORT_REQUESTED ) && aPreselectedFilterPropsHM.size() )
862         {
863             // this is a PDF export
864             // the filter options has been shown already
865             ::rtl::OUString aFilterUIName = aPreselectedFilterPropsHM.getUnpackedValueOrDefault(
866                                                         ::rtl::OUString::createFromAscii( "UIName" ),
867                                                         ::rtl::OUString() );
868 
869             pFileDlg = new sfx2::FileDialogHelper( aDialogMode, aDialogFlags, aFilterUIName, ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "pdf" ) ), rStandardDir, rBlackList );
870             pFileDlg->SetCurrentFilter( aFilterUIName );
871         }
872         else
873         {
874             // This is the normal dialog
875             pFileDlg = new sfx2::FileDialogHelper( aDialogMode, aDialogFlags, aDocServiceName, nDialog, nMust, nDont, rStandardDir, rBlackList );
876         }
877 
878         if( aDocServiceName.equalsAscii( "com.sun.star.drawing.DrawingDocument" ) )
879             eCtxt = sfx2::FileDialogHelper::SD_EXPORT;
880         if( aDocServiceName.equalsAscii( "com.sun.star.presentation.PresentationDocument" ) )
881             eCtxt = sfx2::FileDialogHelper::SI_EXPORT;
882         if( aDocServiceName.equalsAscii( "com.sun.star.text.TextDocument" ) )
883             eCtxt = sfx2::FileDialogHelper::SW_EXPORT;
884 
885         if ( eCtxt != sfx2::FileDialogHelper::UNKNOWN_CONTEXT )
886             pFileDlg->SetContext( eCtxt );
887 
888         pFileDlg->CreateMatcher( aDocServiceName );
889 
890         uno::Reference< ui::dialogs::XFilePicker > xFilePicker = pFileDlg->GetFilePicker();
891         uno::Reference< ui::dialogs::XFilePickerControlAccess > xControlAccess =
892         uno::Reference< ui::dialogs::XFilePickerControlAccess >( xFilePicker, uno::UNO_QUERY );
893 
894         if ( xControlAccess.is() )
895         {
896             ::rtl::OUString aCtrlText = String( SfxResId( STR_EXPORTBUTTON ) );
897             xControlAccess->setLabel( ui::dialogs::CommonFilePickerElementIds::PUSHBUTTON_OK, aCtrlText );
898 
899             aCtrlText = ::rtl::OUString( String( SfxResId( STR_LABEL_FILEFORMAT ) ) );
900             xControlAccess->setLabel( ui::dialogs::CommonFilePickerElementIds::LISTBOX_FILTER_LABEL, aCtrlText );
901         }
902     }
903     else
904     {
905         // This is the normal dialog
906         pFileDlg = new sfx2::FileDialogHelper( aDialogMode, aDialogFlags, aDocServiceName, nDialog, nMust, nDont, rStandardDir, rBlackList );
907         pFileDlg->CreateMatcher( aDocServiceName );
908     }
909 
910     ::rtl::OUString aAdjustToType;
911 
912     if ( ( nStoreMode & EXPORT_REQUESTED ) && !( nStoreMode & WIDEEXPORT_REQUESTED ) )
913     {
914         // it is export, set the preselected filter
915         ::rtl::OUString aFilterUIName = aPreselectedFilterPropsHM.getUnpackedValueOrDefault(
916                                         ::rtl::OUString::createFromAscii( "UIName" ),
917                                         ::rtl::OUString() );
918         pFileDlg->SetCurrentFilter( aFilterUIName );
919     }
920     // it is no export, bSetStandardName == true means that user agreed to store document in the default (default default ;-)) format
921     else if ( bSetStandardName || GetStorable()->hasLocation() )
922     {
923         uno::Sequence< beans::PropertyValue > aOldFilterProps;
924         ::rtl::OUString aOldFilterName = GetDocProps().getUnpackedValueOrDefault(
925                                                         aFilterNameString,
926                                                         ::rtl::OUString() );
927 
928         if ( aOldFilterName.getLength() )
929             m_pOwner->GetFilterConfiguration()->getByName( aOldFilterName ) >>= aOldFilterProps;
930 
931         ::comphelper::SequenceAsHashMap aOldFiltPropsHM( aOldFilterProps );
932         sal_Int32 nOldFiltFlags = aOldFiltPropsHM.getUnpackedValueOrDefault( ::rtl::OUString::createFromAscii( "Flags" ), (sal_Int32)0 );
933 
934         if ( bSetStandardName || ( nOldFiltFlags & nMust ) != nMust || nOldFiltFlags & nDont )
935         {
936             // the suggested type will be changed, the extension should be adjusted
937             aAdjustToType = aPreselectedFilterPropsHM.getUnpackedValueOrDefault(
938                                             ::rtl::OUString::createFromAscii( "Type" ),
939                                             ::rtl::OUString() );
940 
941             ::rtl::OUString aFilterUIName = aPreselectedFilterPropsHM.getUnpackedValueOrDefault(
942                                             ::rtl::OUString::createFromAscii( "UIName" ),
943                                             ::rtl::OUString() );
944             pFileDlg->SetCurrentFilter( aFilterUIName );
945         }
946         else
947         {
948             pFileDlg->SetCurrentFilter( aOldFiltPropsHM.getUnpackedValueOrDefault(
949                                                         ::rtl::OUString::createFromAscii( "UIName" ),
950                                                         ::rtl::OUString() ) );
951         }
952     }
953 
954     ::rtl::OUString aReccomendedDir = GetReccomendedDir( aSuggestedDir, eCtxt );
955     if ( aReccomendedDir.getLength() )
956         pFileDlg->SetDisplayDirectory( aReccomendedDir );
957     ::rtl::OUString aReccomendedName = GetReccomendedName( aSuggestedName, aAdjustToType );
958     if ( aReccomendedName.getLength() )
959         pFileDlg->SetFileName( aReccomendedName );
960 
961     uno::Reference < view::XSelectionSupplier > xSel( GetModel()->getCurrentController(), uno::UNO_QUERY );
962     if ( xSel.is() && xSel->getSelection().hasValue() )
963         GetMediaDescr()[::rtl::OUString::createFromAscii( "SelectionOnly" )] <<= sal_True;
964 
965     // This is a temporary hardcoded solution must be removed when
966     // dialogs do not need parameters in SidSet representation any more
967     sal_uInt16 nSlotID = getSlotIDFromMode( nStoreMode );
968     if ( !nSlotID )
969         throw lang::IllegalArgumentException(); // TODO:
970 
971     // generate SidSet from MediaDescriptor and provide it into FileDialog
972     // than merge changed SidSet back
973     SfxAllItemSet aDialogParams( SFX_APP()->GetPool() );
974     SfxItemSet* pDialogParams = &aDialogParams;
975     TransformParameters( nSlotID,
976                          GetMediaDescr().getAsConstPropertyValueList(),
977                          aDialogParams,
978                          NULL );
979 
980     const SfxPoolItem* pItem = NULL;
981     if ( bPreselectPassword && aDialogParams.GetItemState( SID_ENCRYPTIONDATA, sal_True, &pItem ) != SFX_ITEM_SET )
982     {
983         // the file dialog preselects the password checkbox if the provided mediadescriptor has encryption data entry
984         // after dialog execution the password interaction flag will be either removed or not
985         aDialogParams.Put( SfxBoolItem( SID_PASSWORDINTERACTION, sal_True ) );
986     }
987 
988     // aStringTypeFN is a pure output parameter, pDialogParams is an in/out parameter
989     String aStringTypeFN;
990     if ( pFileDlg->Execute( pDialogParams, aStringTypeFN ) != ERRCODE_NONE )
991     {
992         delete pFileDlg;
993         throw task::ErrorCodeIOException( ::rtl::OUString(), uno::Reference< uno::XInterface >(), ERRCODE_IO_ABORT );
994     }
995 
996     ::rtl::OUString aFilterName = aStringTypeFN;
997 
998     // the following two arguments can not be converted in MediaDescriptor,
999     // so they should be removed from the ItemSet after retrieving
1000     SFX_ITEMSET_ARG( pDialogParams, pRecommendReadOnly, SfxBoolItem, SID_RECOMMENDREADONLY, sal_False );
1001     m_bRecommendReadOnly = ( pRecommendReadOnly && pRecommendReadOnly->GetValue() );
1002     pDialogParams->ClearItem( SID_RECOMMENDREADONLY );
1003 
1004     uno::Sequence< beans::PropertyValue > aPropsFromDialog;
1005     TransformItems( nSlotID, *pDialogParams, aPropsFromDialog, NULL );
1006     GetMediaDescr() << aPropsFromDialog;
1007 
1008     // get the path from the dialog
1009     INetURLObject aURL( pFileDlg->GetPath() );
1010     // the path should be provided outside since it might be used for further calls to the dialog
1011     aSuggestedName = aURL.GetName( INetURLObject::DECODE_WITH_CHARSET );
1012 
1013     // old filter options should be cleared in case different filter is used
1014 
1015     ::rtl::OUString aFilterFromMediaDescr = GetMediaDescr().getUnpackedValueOrDefault(
1016                                                     aFilterNameString,
1017                                                     ::rtl::OUString() );
1018     ::rtl::OUString aOldFilterName = GetDocProps().getUnpackedValueOrDefault(
1019                                                     aFilterNameString,
1020                                                     ::rtl::OUString() );
1021     if ( aFilterName.equals( aFilterFromMediaDescr ) )
1022     {
1023         // preserv current settings if any
1024         // if there no current settings and the name is the same
1025         // as old filter name use old filter settings
1026 
1027         if ( aFilterFromMediaDescr.equals( aOldFilterName ) )
1028         {
1029             ::comphelper::SequenceAsHashMap::const_iterator aIter =
1030                                         GetDocProps().find( aFilterOptionsString );
1031             if ( aIter != GetDocProps().end()
1032               && GetMediaDescr().find( aFilterOptionsString ) == GetMediaDescr().end() )
1033                 GetMediaDescr()[aIter->first] = aIter->second;
1034 
1035             aIter = GetDocProps().find( aFilterDataString );
1036             if ( aIter != GetDocProps().end()
1037               && GetMediaDescr().find( aFilterDataString ) == GetMediaDescr().end() )
1038                 GetMediaDescr()[aIter->first] = aIter->second;
1039         }
1040     }
1041     else
1042     {
1043         GetMediaDescr().erase( aFilterDataString );
1044         GetMediaDescr().erase( aFilterOptionsString );
1045 
1046         if ( aFilterName.equals( aOldFilterName ) )
1047         {
1048             // merge filter option of the document filter
1049 
1050             ::comphelper::SequenceAsHashMap::const_iterator aIter =
1051                                 GetDocProps().find( aFilterOptionsString );
1052             if ( aIter != GetDocProps().end() )
1053                 GetMediaDescr()[aIter->first] = aIter->second;
1054 
1055             aIter = GetDocProps().find( aFilterDataString );
1056             if ( aIter != GetDocProps().end() )
1057                 GetMediaDescr()[aIter->first] = aIter->second;
1058         }
1059     }
1060 
1061     uno::Reference< ui::dialogs::XFilePickerControlAccess > xExtFileDlg( pFileDlg->GetFilePicker(), uno::UNO_QUERY );
1062     if ( xExtFileDlg.is() )
1063     {
1064         if ( SfxStoringHelper::CheckFilterOptionsAppearence( m_pOwner->GetFilterConfiguration(), aFilterName ) )
1065             bUseFilterOptions = sal_True;
1066 
1067         if ( ( !( nStoreMode & EXPORT_REQUESTED ) || ( nStoreMode & WIDEEXPORT_REQUESTED ) ) && bUseFilterOptions )
1068         {
1069             try
1070             {
1071                 // for exporters: always show dialog if format uses options
1072                 // for save: show dialog if format uses options and no options given or if forced by user
1073                 uno::Any aVal =
1074                         xExtFileDlg->getValue( ui::dialogs::ExtendedFilePickerElementIds::CHECKBOX_FILTEROPTIONS, 0 );
1075 
1076                 aVal >>= bUseFilterOptions;
1077                 if ( !bUseFilterOptions )
1078                     bUseFilterOptions =
1079                       ( GetMediaDescr().find( aFilterDataString ) == GetMediaDescr().end()
1080                       && GetMediaDescr().find( aFilterOptionsString ) == GetMediaDescr().end() );
1081             }
1082             catch( lang::IllegalArgumentException& )
1083             {}
1084         }
1085     }
1086 
1087     delete pFileDlg;
1088 
1089     // merge in results of the dialog execution
1090     GetMediaDescr()[::rtl::OUString::createFromAscii( "URL" )] <<=
1091                                                 ::rtl::OUString( aURL.GetMainURL( INetURLObject::NO_DECODE ));
1092     GetMediaDescr()[aFilterNameString] <<= aFilterName;
1093 
1094     return bUseFilterOptions;
1095 }
1096 
1097 //-------------------------------------------------------------------------
ShowDocumentInfoDialog()1098 sal_Bool ModelData_Impl::ShowDocumentInfoDialog()
1099 {
1100     sal_Bool bDialogUsed = sal_False;
1101 
1102     try {
1103         uno::Reference< frame::XController > xController = GetModel()->getCurrentController();
1104         if ( xController.is() )
1105         {
1106             uno::Reference< frame::XDispatchProvider > xFrameDispatch( xController->getFrame(), uno::UNO_QUERY );
1107             if ( xFrameDispatch.is() )
1108             {
1109                 util::URL aURL;
1110                 aURL.Complete = ::rtl::OUString::createFromAscii( ".uno:SetDocumentProperties" );
1111 
1112                 uno::Reference< util::XURLTransformer > xTransformer(
1113                             m_pOwner->GetServiceFactory()->createInstance(
1114                                             DEFINE_CONST_UNICODE("com.sun.star.util.URLTransformer") ),
1115                             uno::UNO_QUERY );
1116                 if ( xTransformer.is() && xTransformer->parseStrict( aURL ) )
1117                 {
1118                     uno::Reference< frame::XDispatch > xDispatch = xFrameDispatch->queryDispatch(
1119                                                                                 aURL,
1120                                                                                 ::rtl::OUString::createFromAscii( "_self" ),
1121                                                                                 0 );
1122                     if ( xDispatch.is() )
1123                     {
1124                         xDispatch->dispatch( aURL, uno::Sequence< beans::PropertyValue >() );
1125                         bDialogUsed = sal_True;
1126                     }
1127                 }
1128             }
1129         }
1130     }
1131     catch ( uno::Exception& )
1132     {
1133     }
1134 
1135     return bDialogUsed;
1136 }
1137 
1138 //-------------------------------------------------------------------------
GetReccomendedDir(const::rtl::OUString & aSuggestedDir,const sfx2::FileDialogHelper::Context & aCtxt)1139 ::rtl::OUString ModelData_Impl::GetReccomendedDir( const ::rtl::OUString& aSuggestedDir, const sfx2::FileDialogHelper::Context& aCtxt )
1140 {
1141     ::rtl::OUString aReccomendedDir;
1142 
1143     if ( ( aSuggestedDir.getLength() || GetStorable()->hasLocation() )
1144       && !GetMediaDescr().getUnpackedValueOrDefault( ::rtl::OUString::createFromAscii( "RepairPackage" ),
1145                                                                     sal_False ) )
1146     {
1147         INetURLObject aLocation;
1148         if ( aSuggestedDir.getLength() )
1149             aLocation = INetURLObject( aSuggestedDir );
1150         else
1151         {
1152             ::rtl::OUString aOldURL = GetStorable()->getLocation();
1153             if ( aOldURL.getLength() )
1154             {
1155                 INetURLObject aTmp( aOldURL );
1156                 if ( aTmp.removeSegment() )
1157                     aLocation = aTmp;
1158             }
1159 
1160             if ( aLocation.HasError() )
1161                 aLocation = INetURLObject( SvtPathOptions().GetWorkPath() );
1162         }
1163 
1164         aLocation.setFinalSlash();
1165         if ( !aLocation.HasError() )
1166             aReccomendedDir = aLocation.GetMainURL( INetURLObject::NO_DECODE );
1167     }
1168     else
1169     {
1170         // pb: set graphic path if context == SD_EXPORT or SI_EXPORT else work path
1171         ::rtl::OUString aConfigSuggestion( ( aCtxt != sfx2::FileDialogHelper::UNKNOWN_CONTEXT ) ? SvtPathOptions().GetGraphicPath() : SvtPathOptions().GetWorkPath() );
1172         aReccomendedDir = INetURLObject( aConfigSuggestion ).GetMainURL( INetURLObject::NO_DECODE );
1173     }
1174 
1175     return aReccomendedDir;
1176 }
1177 
1178 //-------------------------------------------------------------------------
GetReccomendedName(const::rtl::OUString & aSuggestedName,const::rtl::OUString & aTypeName)1179 ::rtl::OUString ModelData_Impl::GetReccomendedName( const ::rtl::OUString& aSuggestedName, const ::rtl::OUString& aTypeName )
1180 {
1181     // the last used name might be provided by aSuggestedName from the old selection, or from the MediaDescriptor
1182     ::rtl::OUString aReccomendedName;
1183 
1184     if ( aSuggestedName.getLength() )
1185         aReccomendedName = aSuggestedName;
1186     else
1187     {
1188         aReccomendedName = INetURLObject( GetStorable()->getLocation() ).GetName( INetURLObject::DECODE_WITH_CHARSET );
1189         if ( !aReccomendedName.getLength() )
1190         {
1191             try {
1192                 uno::Reference< frame::XTitle > xTitle( GetModel(), uno::UNO_QUERY_THROW );
1193                 aReccomendedName = xTitle->getTitle();
1194             } catch( uno::Exception& ) {}
1195         }
1196 
1197         if ( aReccomendedName.getLength() && aTypeName.getLength() )
1198         {
1199             // adjust the extension to the type
1200             uno::Reference< container::XNameAccess > xTypeDetection = uno::Reference< container::XNameAccess >(
1201                 m_pOwner->GetServiceFactory()->createInstance( ::rtl::OUString::createFromAscii( "com.sun.star.document.TypeDetection" ) ),
1202                 uno::UNO_QUERY );
1203             if ( xTypeDetection.is() )
1204             {
1205                 INetURLObject aObj( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "file:///c:/" ) ) + aReccomendedName );
1206 
1207                 uno::Sequence< beans::PropertyValue > aTypeNameProps;
1208                 if ( ( xTypeDetection->getByName( aTypeName ) >>= aTypeNameProps ) && aTypeNameProps.getLength() )
1209                 {
1210                     ::comphelper::SequenceAsHashMap aTypeNamePropsHM( aTypeNameProps );
1211                     uno::Sequence< ::rtl::OUString > aExtensions = aTypeNamePropsHM.getUnpackedValueOrDefault(
1212                                                     ::rtl::OUString::createFromAscii( "Extensions" ),
1213                                                     ::uno::Sequence< ::rtl::OUString >() );
1214                     if ( aExtensions.getLength() )
1215                         aObj.SetExtension( aExtensions[0] );
1216                 }
1217 
1218                 aReccomendedName = aObj.GetName( INetURLObject::DECODE_WITH_CHARSET );
1219             }
1220         }
1221     }
1222 
1223     return aReccomendedName;
1224 }
1225 
1226 
1227 //=========================================================================
1228 // class SfxStoringHelper
1229 //=========================================================================
1230 //-------------------------------------------------------------------------
SfxStoringHelper(const uno::Reference<lang::XMultiServiceFactory> & xFactory)1231 SfxStoringHelper::SfxStoringHelper( const uno::Reference< lang::XMultiServiceFactory >& xFactory )
1232 : m_xFactory( xFactory )
1233 {
1234 }
1235 
1236 //-------------------------------------------------------------------------
GetServiceFactory()1237 uno::Reference< lang::XMultiServiceFactory > SfxStoringHelper::GetServiceFactory()
1238 {
1239     if ( !m_xFactory.is() )
1240     {
1241         m_xFactory = ::comphelper::getProcessServiceFactory();
1242         if( !m_xFactory.is() )
1243             throw uno::RuntimeException(); // TODO:
1244     }
1245 
1246     return m_xFactory;
1247 }
1248 
1249 //-------------------------------------------------------------------------
GetFilterConfiguration()1250 uno::Reference< container::XNameAccess > SfxStoringHelper::GetFilterConfiguration()
1251 {
1252     if ( !m_xFilterCFG.is() )
1253     {
1254         m_xFilterCFG = uno::Reference< container::XNameAccess >(
1255             GetServiceFactory()->createInstance( ::rtl::OUString::createFromAscii( "com.sun.star.document.FilterFactory" ) ),
1256             uno::UNO_QUERY );
1257 
1258         if ( !m_xFilterCFG.is() )
1259             throw uno::RuntimeException();
1260     }
1261 
1262     return m_xFilterCFG;
1263 }
1264 
1265 //-------------------------------------------------------------------------
GetFilterQuery()1266 uno::Reference< container::XContainerQuery > SfxStoringHelper::GetFilterQuery()
1267 {
1268     if ( !m_xFilterQuery.is() )
1269     {
1270         m_xFilterQuery = uno::Reference< container::XContainerQuery >( GetFilterConfiguration(), uno::UNO_QUERY );
1271         if ( !m_xFilterQuery.is() )
1272             throw uno::RuntimeException();
1273     }
1274 
1275     return m_xFilterQuery;
1276 }
1277 
1278 //-------------------------------------------------------------------------
GetModuleManager()1279 uno::Reference< ::com::sun::star::frame::XModuleManager > SfxStoringHelper::GetModuleManager()
1280 {
1281     if ( !m_xModuleManager.is() )
1282     {
1283         m_xModuleManager = uno::Reference< ::com::sun::star::frame::XModuleManager >(
1284             GetServiceFactory()->createInstance(
1285                     ::rtl::OUString::createFromAscii( "com.sun.star.frame.ModuleManager" ) ),
1286             uno::UNO_QUERY );
1287 
1288         if ( !m_xModuleManager.is() )
1289             throw uno::RuntimeException();
1290     }
1291 
1292     return m_xModuleManager;
1293 }
1294 
1295 //-------------------------------------------------------------------------
GetNamedModuleManager()1296 uno::Reference< container::XNameAccess > SfxStoringHelper::GetNamedModuleManager()
1297 {
1298     if ( !m_xNamedModManager.is() )
1299     {
1300         m_xNamedModManager = uno::Reference< container::XNameAccess >( GetModuleManager(), uno::UNO_QUERY );
1301         if ( !m_xNamedModManager.is() )
1302             throw uno::RuntimeException();
1303     }
1304 
1305     return m_xNamedModManager;
1306 }
1307 
1308 //-------------------------------------------------------------------------
GUIStoreModel(const uno::Reference<frame::XModel> & xModel,const::rtl::OUString & aSlotName,uno::Sequence<beans::PropertyValue> & aArgsSequence,sal_Bool bPreselectPassword,::rtl::OUString aSuggestedName,sal_uInt16 nDocumentSignatureState)1309 sal_Bool SfxStoringHelper::GUIStoreModel( const uno::Reference< frame::XModel >& xModel,
1310                                             const ::rtl::OUString& aSlotName,
1311                                             uno::Sequence< beans::PropertyValue >& aArgsSequence,
1312                                             sal_Bool bPreselectPassword,
1313                                             ::rtl::OUString aSuggestedName,
1314                                             sal_uInt16 nDocumentSignatureState )
1315 {
1316     ModelData_Impl aModelData( *this, xModel, aArgsSequence );
1317 
1318     sal_Bool bDialogUsed = sal_False;
1319 
1320     INetURLObject aURL;
1321 
1322     sal_Bool bSetStandardName = sal_False; // can be set only for SaveAs
1323 
1324     // parse the slot name
1325     sal_Int8 nStoreMode = getStoreModeFromSlotName( aSlotName );
1326     sal_Int8 nStatusSave = STATUS_NO_ACTION;
1327 
1328     // handle the special cases
1329     if ( nStoreMode & SAVEAS_REQUESTED )
1330     {
1331         ::comphelper::SequenceAsHashMap::const_iterator aSaveToIter =
1332                         aModelData.GetMediaDescr().find( ::rtl::OUString::createFromAscii( "SaveTo" ) );
1333         if ( aSaveToIter != aModelData.GetMediaDescr().end() )
1334         {
1335             sal_Bool bWideExport = sal_False;
1336             aSaveToIter->second >>= bWideExport;
1337             if ( bWideExport )
1338                 nStoreMode = EXPORT_REQUESTED | WIDEEXPORT_REQUESTED;
1339         }
1340 
1341         // if saving is not acceptable the warning must be shown even in case of SaveAs operation
1342         if ( ( nStoreMode & SAVEAS_REQUESTED ) && aModelData.CheckSaveAcceptable( STATUS_SAVEAS ) == STATUS_NO_ACTION )
1343             throw task::ErrorCodeIOException( ::rtl::OUString(), uno::Reference< uno::XInterface >(), ERRCODE_IO_ABORT );
1344     }
1345     else if ( nStoreMode & SAVE_REQUESTED )
1346     {
1347         // if saving is not acceptable by the configuration the warning must be shown
1348         nStatusSave = aModelData.CheckSaveAcceptable( STATUS_SAVE );
1349 
1350         if ( nStatusSave == STATUS_NO_ACTION )
1351             throw task::ErrorCodeIOException( ::rtl::OUString(), uno::Reference< uno::XInterface >(), ERRCODE_IO_ABORT );
1352         else if ( nStatusSave == STATUS_SAVE )
1353         {
1354             // check whether it is possible to use save operation
1355             nStatusSave = aModelData.CheckStateForSave();
1356         }
1357 
1358         if ( nStatusSave == STATUS_NO_ACTION )
1359         {
1360             throw task::ErrorCodeIOException( ::rtl::OUString(), uno::Reference< uno::XInterface >(), ERRCODE_IO_ABORT );
1361         }
1362         else if ( nStatusSave != STATUS_SAVE )
1363         {
1364             // this should be a usual SaveAs operation
1365             nStoreMode = SAVEAS_REQUESTED;
1366             if ( nStatusSave == STATUS_SAVEAS_STANDARDNAME )
1367                 bSetStandardName = sal_True;
1368         }
1369     }
1370 
1371     if ( !( nStoreMode & EXPORT_REQUESTED ) )
1372     {
1373         // if it is no export, warn user that the signature will be removed
1374         if (  SIGNATURESTATE_SIGNATURES_OK == nDocumentSignatureState
1375            || SIGNATURESTATE_SIGNATURES_INVALID == nDocumentSignatureState
1376            || SIGNATURESTATE_SIGNATURES_NOTVALIDATED == nDocumentSignatureState
1377            || SIGNATURESTATE_SIGNATURES_PARTIAL_OK == nDocumentSignatureState)
1378         {
1379             if ( QueryBox( NULL, SfxResId( RID_XMLSEC_QUERY_LOSINGSIGNATURE ) ).Execute() != RET_YES )
1380             {
1381                 // the user has decided not to store the document
1382                 throw task::ErrorCodeIOException( ::rtl::OUString(),
1383                                                   uno::Reference< uno::XInterface >(),
1384                                                   ERRCODE_IO_ABORT );
1385             }
1386         }
1387     }
1388 
1389     if ( nStoreMode & SAVE_REQUESTED && nStatusSave == STATUS_SAVE )
1390     {
1391         // Document properties can contain streams that should be freed before storing
1392         aModelData.FreeDocumentProps();
1393 
1394         if ( aModelData.GetStorable2().is() )
1395         {
1396             try
1397             {
1398                 aModelData.GetStorable2()->storeSelf( aModelData.GetMediaDescr().getAsConstPropertyValueList() );
1399             }
1400             catch( lang::IllegalArgumentException& )
1401             {
1402                 OSL_ENSURE( sal_False, "ModelData didn't handle illegal parameters, all the parameters are ignored!\n" );
1403                 aModelData.GetStorable()->store();
1404             }
1405         }
1406         else
1407         {
1408             OSL_ENSURE( sal_False, "XStorable2 is not supported by the model!\n" );
1409             aModelData.GetStorable()->store();
1410         }
1411 
1412         return sal_False;
1413     }
1414 
1415     // preselect a filter for the storing process
1416     uno::Sequence< beans::PropertyValue > aFilterProps = aModelData.GetPreselectedFilter_Impl( nStoreMode );
1417 
1418     DBG_ASSERT( aFilterProps.getLength(), "No filter for storing!\n" );
1419     if ( !aFilterProps.getLength() )
1420         throw task::ErrorCodeIOException( ::rtl::OUString(),
1421                                             uno::Reference< uno::XInterface >(),
1422                                             ERRCODE_IO_INVALIDPARAMETER );
1423 
1424     ::comphelper::SequenceAsHashMap aFilterPropsHM( aFilterProps );
1425     ::rtl::OUString aFilterName = aFilterPropsHM.getUnpackedValueOrDefault(
1426                                                                     ::rtl::OUString::createFromAscii( "Name" ),
1427                                                                     ::rtl::OUString() );
1428 
1429     ::rtl::OUString aFilterFromMediaDescr = aModelData.GetMediaDescr().getUnpackedValueOrDefault(
1430                                                     aFilterNameString,
1431                                                     ::rtl::OUString() );
1432     ::rtl::OUString aOldFilterName = aModelData.GetDocProps().getUnpackedValueOrDefault(
1433                                                     aFilterNameString,
1434                                                     ::rtl::OUString() );
1435 
1436     sal_Bool bUseFilterOptions = sal_False;
1437     ::comphelper::SequenceAsHashMap::const_iterator aFileNameIter = aModelData.GetMediaDescr().find( ::rtl::OUString::createFromAscii( "URL" ) );
1438 
1439     if ( ( nStoreMode & EXPORT_REQUESTED ) && ( nStoreMode & PDFEXPORT_REQUESTED ) && !( nStoreMode & PDFDIRECTEXPORT_REQUESTED ) )
1440     {
1441         // this is PDF export, the filter options dialog should be shown before the export
1442         aModelData.GetMediaDescr()[aFilterNameString] <<= aFilterName;
1443         if ( aModelData.GetMediaDescr().find( aFilterFlagsString ) == aModelData.GetMediaDescr().end()
1444           && aModelData.GetMediaDescr().find( aFilterOptionsString ) == aModelData.GetMediaDescr().end()
1445           && aModelData.GetMediaDescr().find( aFilterDataString ) == aModelData.GetMediaDescr().end() )
1446         {
1447             // execute filter options dialog since no options are set in the media descriptor
1448             if ( aModelData.ExecuteFilterDialog_Impl( aFilterName ) )
1449                 bDialogUsed = sal_True;
1450         }
1451     }
1452 
1453     if ( aFileNameIter == aModelData.GetMediaDescr().end() )
1454     {
1455         sal_Int16 nDialog = SFX2_IMPL_DIALOG_CONFIG;
1456         ::comphelper::SequenceAsHashMap::const_iterator aDlgIter =
1457             aModelData.GetMediaDescr().find( ::rtl::OUString::createFromAscii( "UseSystemDialog" ) );
1458         if ( aDlgIter != aModelData.GetMediaDescr().end() )
1459         {
1460             sal_Bool bUseSystemDialog = sal_True;
1461             if ( aDlgIter->second >>= bUseSystemDialog )
1462             {
1463                 if ( bUseSystemDialog )
1464                     nDialog = SFX2_IMPL_DIALOG_SYSTEM;
1465                 else
1466                     nDialog = SFX2_IMPL_DIALOG_OOO;
1467             }
1468         }
1469 
1470         // The Dispatch supports parameter FolderName that overwrites SuggestedSaveAsDir
1471         ::rtl::OUString aSuggestedDir = aModelData.GetMediaDescr().getUnpackedValueOrDefault( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "FolderName" ) ), ::rtl::OUString() );
1472         if ( !aSuggestedDir.getLength() )
1473         {
1474             aSuggestedDir = aModelData.GetMediaDescr().getUnpackedValueOrDefault( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "SuggestedSaveAsDir" ) ), ::rtl::OUString() );
1475             if ( !aSuggestedDir.getLength() )
1476                 aSuggestedDir = aModelData.GetDocProps().getUnpackedValueOrDefault( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "SuggestedSaveAsDir" ) ), ::rtl::OUString() );
1477         }
1478 
1479     aSuggestedName = aModelData.GetMediaDescr().getUnpackedValueOrDefault( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "SuggestedSaveAsName" ) ), ::rtl::OUString() );
1480         if ( !aSuggestedName.getLength() )
1481             aSuggestedName = aModelData.GetDocProps().getUnpackedValueOrDefault( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "SuggestedSaveAsName" ) ), ::rtl::OUString() );
1482 
1483         ::rtl::OUString sStandardDir;
1484         ::comphelper::SequenceAsHashMap::const_iterator aStdDirIter =
1485             aModelData.GetMediaDescr().find( ::rtl::OUString::createFromAscii( "StandardDir" ) );
1486         if ( aStdDirIter != aModelData.GetMediaDescr().end() )
1487             aStdDirIter->second >>= sStandardDir;
1488 
1489         ::com::sun::star::uno::Sequence< ::rtl::OUString >  aBlackList;
1490 
1491         ::comphelper::SequenceAsHashMap::const_iterator aBlackListIter =
1492             aModelData.GetMediaDescr().find( ::rtl::OUString::createFromAscii( "BlackList" ) );
1493         if ( aBlackListIter != aModelData.GetMediaDescr().end() )
1494             aBlackListIter->second >>= aBlackList;
1495 
1496         sal_Bool bExit = sal_False;
1497         while ( !bExit )
1498         {
1499             bUseFilterOptions = aModelData.OutputFileDialog( nStoreMode, aFilterProps, bSetStandardName, aSuggestedName, bPreselectPassword, aSuggestedDir, nDialog, sStandardDir, aBlackList );
1500 
1501             // in case the dialog is opend a second time the folder should be the same as before, not what was handed over by parameters
1502             aSuggestedDir = ::rtl::OUString();
1503             if ( nStoreMode == SAVEAS_REQUESTED )
1504             {
1505                 // in case of saving check filter for possible alien warning
1506                 ::rtl::OUString aSelFilterName = aModelData.GetMediaDescr().getUnpackedValueOrDefault(
1507                                                                                 aFilterNameString,
1508                                                                                 ::rtl::OUString() );
1509                 sal_Int8 nStatusFilterSave = aModelData.CheckFilter( aSelFilterName );
1510                 if ( nStatusFilterSave == STATUS_SAVEAS_STANDARDNAME )
1511                 {
1512                     // switch to best filter
1513                     bSetStandardName = sal_True;
1514                 }
1515                 else if ( nStatusFilterSave == STATUS_SAVE )
1516                 {
1517                     // user confirmed alien filter or "good" filter is used
1518                     bExit = sal_True;
1519                 }
1520             }
1521             else
1522                 bExit = sal_True;
1523         }
1524 
1525         bDialogUsed = sal_True;
1526         aFileNameIter = aModelData.GetMediaDescr().find( ::rtl::OUString::createFromAscii( "URL" ) );
1527     }
1528     else
1529     {
1530         // the target file name is provided so check if new filter options
1531         // are provided or old options can be used
1532         if ( aFilterFromMediaDescr.equals( aOldFilterName ) )
1533         {
1534             ::comphelper::SequenceAsHashMap::const_iterator aIter =
1535                                             aModelData.GetDocProps().find( aFilterOptionsString );
1536             if ( aIter != aModelData.GetDocProps().end()
1537               && aModelData.GetMediaDescr().find( aFilterOptionsString ) == aModelData.GetMediaDescr().end() )
1538                 aModelData.GetMediaDescr()[aIter->first] = aIter->second;
1539 
1540             aIter = aModelData.GetDocProps().find( aFilterDataString );
1541             if ( aIter != aModelData.GetDocProps().end()
1542               && aModelData.GetMediaDescr().find( aFilterDataString ) == aModelData.GetMediaDescr().end() )
1543                 aModelData.GetMediaDescr()[aIter->first] = aIter->second;
1544         }
1545     }
1546 
1547     if ( aFileNameIter != aModelData.GetMediaDescr().end() )
1548     {
1549         ::rtl::OUString aFileName;
1550         aFileNameIter->second >>= aFileName;
1551         aURL.SetURL( aFileName );
1552         DBG_ASSERT( aURL.GetProtocol() != INET_PROT_NOT_VALID, "Illegal URL!" );
1553 
1554         ::comphelper::SequenceAsHashMap::const_iterator aIter =
1555                                 aModelData.GetMediaDescr().find( aFilterNameString );
1556 
1557         if ( aIter != aModelData.GetMediaDescr().end() )
1558             aIter->second >>= aFilterName;
1559         else
1560             aModelData.GetMediaDescr()[aFilterNameString] <<= aFilterName;
1561 
1562         DBG_ASSERT( aFilterName.getLength(), "Illegal filter!" );
1563     }
1564     else
1565     {
1566         DBG_ASSERT( sal_False, "This code must be unreachable!\n" );
1567         throw task::ErrorCodeIOException( ::rtl::OUString(),
1568                                             uno::Reference< uno::XInterface >(),
1569                                             ERRCODE_IO_INVALIDPARAMETER );
1570     }
1571 
1572     ::comphelper::SequenceAsHashMap::const_iterator aIter =
1573                             aModelData.GetMediaDescr().find( ::rtl::OUString::createFromAscii( "FilterFlags" ) );
1574     sal_Bool bFilterFlagsSet = ( aIter != aModelData.GetMediaDescr().end() );
1575 
1576     if( !( nStoreMode & PDFEXPORT_REQUESTED ) && !bFilterFlagsSet
1577         && ( ( nStoreMode & EXPORT_REQUESTED ) || bUseFilterOptions ) )
1578     {
1579         // execute filter options dialog
1580         if ( aModelData.ExecuteFilterDialog_Impl( aFilterName ) )
1581             bDialogUsed = sal_True;
1582     }
1583 
1584     // so the arguments will not change any more and can be stored to the main location
1585     aArgsSequence = aModelData.GetMediaDescr().getAsConstPropertyValueList();
1586 
1587     // store the document and handle it's docinfo
1588     SvtSaveOptions aOptions;
1589 
1590     DocumentSettingsGuard aSettingsGuard( aModelData.GetModel(), aModelData.IsRecommendReadOnly(), nStoreMode & EXPORT_REQUESTED );
1591 
1592     OSL_ENSURE( aModelData.GetMediaDescr().find( ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Password" ) ) ) == aModelData.GetMediaDescr().end(), "The Password property of MediaDescriptor should not be used here!" );
1593     if ( aOptions.IsDocInfoSave()
1594       && ( !aModelData.GetStorable()->hasLocation()
1595           || INetURLObject( aModelData.GetStorable()->getLocation() ) != aURL ) )
1596     {
1597         // this is defenitly not a Save operation
1598         // so the document info can be updated
1599 
1600         // on export document info must be preserved
1601         uno::Reference<document::XDocumentInfoSupplier> xDIS(
1602             aModelData.GetModel(), uno::UNO_QUERY_THROW);
1603         uno::Reference<util::XCloneable> xCloneable(
1604             xDIS->getDocumentInfo(), uno::UNO_QUERY_THROW);
1605         uno::Reference<document::XDocumentInfo> xOldDocInfo(
1606             xCloneable->createClone(), uno::UNO_QUERY_THROW);
1607 
1608         // use dispatch API to show document info dialog
1609         if ( aModelData.ShowDocumentInfoDialog() )
1610             bDialogUsed = sal_True;
1611         else
1612         {
1613             DBG_ERROR( "Can't execute document info dialog!\n" );
1614         }
1615 
1616         try {
1617             // Document properties can contain streams that should be freed before storing
1618             aModelData.FreeDocumentProps();
1619             if ( ( nStoreMode & EXPORT_REQUESTED ) )
1620                 aModelData.GetStorable()->storeToURL( aURL.GetMainURL( INetURLObject::NO_DECODE ), aArgsSequence );
1621             else
1622                 aModelData.GetStorable()->storeAsURL( aURL.GetMainURL( INetURLObject::NO_DECODE ), aArgsSequence );
1623         }
1624         catch( uno::Exception& )
1625         {
1626             if ( ( nStoreMode & EXPORT_REQUESTED ) )
1627                 SetDocInfoState( aModelData.GetModel(), xOldDocInfo, sal_True );
1628 
1629             throw;
1630         }
1631 
1632         if ( ( nStoreMode & EXPORT_REQUESTED ) )
1633             SetDocInfoState( aModelData.GetModel(), xOldDocInfo, sal_True );
1634     }
1635     else
1636     {
1637         // Document properties can contain streams that should be freed before storing
1638         aModelData.FreeDocumentProps();
1639 
1640         // this is actually a save operation with different parameters
1641         // so storeTo or storeAs without DocInfo operations are used
1642         if ( ( nStoreMode & EXPORT_REQUESTED ) )
1643             aModelData.GetStorable()->storeToURL( aURL.GetMainURL( INetURLObject::NO_DECODE ), aArgsSequence );
1644         else
1645             aModelData.GetStorable()->storeAsURL( aURL.GetMainURL( INetURLObject::NO_DECODE ), aArgsSequence );
1646     }
1647 
1648     return bDialogUsed;
1649 }
1650 
1651 //-------------------------------------------------------------------------
1652 // static
CheckFilterOptionsAppearence(const uno::Reference<container::XNameAccess> & xFilterCFG,const::rtl::OUString & aFilterName)1653 sal_Bool SfxStoringHelper::CheckFilterOptionsAppearence(
1654                                                     const uno::Reference< container::XNameAccess >& xFilterCFG,
1655                                                     const ::rtl::OUString& aFilterName )
1656 {
1657     sal_Bool bUseFilterOptions = sal_False;
1658 
1659     DBG_ASSERT( xFilterCFG.is(), "No filter configuration!\n" );
1660     if( xFilterCFG.is() )
1661     {
1662         try {
1663             uno::Sequence < beans::PropertyValue > aProps;
1664             uno::Any aAny = xFilterCFG->getByName( aFilterName );
1665             if ( aAny >>= aProps )
1666             {
1667                 ::comphelper::SequenceAsHashMap aPropsHM( aProps );
1668                 ::rtl::OUString aServiceName = aPropsHM.getUnpackedValueOrDefault(
1669                                                     ::rtl::OUString::createFromAscii( "UIComponent" ),
1670                                                     ::rtl::OUString() );
1671                 if( aServiceName.getLength() )
1672                     bUseFilterOptions = sal_True;
1673             }
1674         }
1675         catch( uno::Exception& )
1676         {
1677         }
1678     }
1679 
1680     return bUseFilterOptions;
1681 }
1682 
1683 //-------------------------------------------------------------------------
1684 // static
SetDocInfoState(const uno::Reference<frame::XModel> & xModel,const uno::Reference<document::XDocumentInfo> & i_xOldDocInfo,sal_Bool bNoModify)1685 void SfxStoringHelper::SetDocInfoState(
1686         const uno::Reference< frame::XModel >& xModel,
1687         const uno::Reference< document::XDocumentInfo >& i_xOldDocInfo,
1688         sal_Bool bNoModify )
1689 {
1690     uno::Reference< document::XDocumentInfoSupplier > xModelDocInfoSupplier( xModel, uno::UNO_QUERY );
1691     if ( !xModelDocInfoSupplier.is() )
1692         throw uno::RuntimeException(); // TODO:
1693 
1694     uno::Reference< document::XDocumentInfo > xDocInfoToFill = xModelDocInfoSupplier->getDocumentInfo();
1695     uno::Reference< beans::XPropertySet > xPropSet( i_xOldDocInfo,
1696         uno::UNO_QUERY_THROW );
1697 
1698     uno::Reference< util::XModifiable > xModifiable( xModel, uno::UNO_QUERY );
1699     if ( bNoModify && !xModifiable.is() )
1700         throw uno::RuntimeException();
1701 
1702     sal_Bool bIsModified = bNoModify && xModifiable->isModified();
1703 
1704     try
1705     {
1706         uno::Reference< beans::XPropertySet > xSet( xDocInfoToFill, uno::UNO_QUERY );
1707         uno::Reference< beans::XPropertyContainer > xContainer( xSet, uno::UNO_QUERY );
1708         uno::Reference< beans::XPropertySetInfo > xSetInfo = xSet->getPropertySetInfo();
1709         uno::Sequence< beans::Property > lProps = xSetInfo->getProperties();
1710         const beans::Property* pProps = lProps.getConstArray();
1711         sal_Int32 c = lProps.getLength();
1712         sal_Int32 i = 0;
1713         for (i=0; i<c; ++i)
1714         {
1715             uno::Any aValue = xPropSet->getPropertyValue( pProps[i].Name );
1716             if ( pProps[i].Attributes & ::com::sun::star::beans::PropertyAttribute::REMOVABLE )
1717                 // QUESTION: DefaultValue?!
1718                 xContainer->addProperty( pProps[i].Name, pProps[i].Attributes, aValue );
1719             try
1720             {
1721                 // it is possible that the propertysets from XML and binary files differ; we shouldn't break then
1722                 xSet->setPropertyValue( pProps[i].Name, aValue );
1723             }
1724             catch ( uno::Exception& ) {}
1725         }
1726 
1727         sal_Int16 nCount = i_xOldDocInfo->getUserFieldCount();
1728         sal_Int16 nSupportedCount = xDocInfoToFill->getUserFieldCount();
1729         for ( sal_Int16 nInd = 0; nInd < nCount && nInd < nSupportedCount; nInd++ )
1730         {
1731             ::rtl::OUString aPropName = i_xOldDocInfo->getUserFieldName( nInd );
1732             xDocInfoToFill->setUserFieldName( nInd, aPropName );
1733             ::rtl::OUString aPropVal = i_xOldDocInfo->getUserFieldValue( nInd );
1734             xDocInfoToFill->setUserFieldValue( nInd, aPropVal );
1735         }
1736     }
1737     catch ( uno::Exception& ) {}
1738 
1739     // set the modified flag back if required
1740     if ( bNoModify && bIsModified != xModifiable->isModified() )
1741         xModifiable->setModified( bIsModified );
1742 }
1743 
1744 //-------------------------------------------------------------------------
1745 // static
WarnUnacceptableFormat(const uno::Reference<frame::XModel> & xModel,::rtl::OUString aOldUIName,::rtl::OUString,sal_Bool)1746 sal_Bool SfxStoringHelper::WarnUnacceptableFormat( const uno::Reference< frame::XModel >& xModel,
1747                                                     ::rtl::OUString aOldUIName,
1748                                                     ::rtl::OUString /*aDefUIName*/,
1749                                                     sal_Bool /*bCanProceedFurther*/ )
1750 {
1751     if ( !SvtSaveOptions().IsWarnAlienFormat() )
1752         return sal_True;
1753 
1754     Window* pWin = SfxStoringHelper::GetModelWindow( xModel );
1755     SfxAlienWarningDialog aDlg( pWin, aOldUIName );
1756 
1757     return aDlg.Execute() == RET_OK;
1758 }
1759 
1760 // static
ExecuteFilterDialog(SfxStoringHelper & _rStorageHelper,const::rtl::OUString & _sFilterName,const::com::sun::star::uno::Reference<::com::sun::star::frame::XModel> & _xModel,::com::sun::star::uno::Sequence<::com::sun::star::beans::PropertyValue> & _rArgsSequence)1761 void SfxStoringHelper::ExecuteFilterDialog( SfxStoringHelper& _rStorageHelper
1762                                             ,const ::rtl::OUString& _sFilterName
1763                                             ,const ::com::sun::star::uno::Reference< ::com::sun::star::frame::XModel >& _xModel
1764                                             ,/*OUT*/::com::sun::star::uno::Sequence< ::com::sun::star::beans::PropertyValue >& _rArgsSequence)
1765 {
1766     ModelData_Impl aModelData( _rStorageHelper, _xModel, _rArgsSequence );
1767     if ( aModelData.ExecuteFilterDialog_Impl( _sFilterName ) )
1768         _rArgsSequence = aModelData.GetMediaDescr().getAsConstPropertyValueList();
1769 }
1770 
1771 // static
GetModelWindow(const uno::Reference<frame::XModel> & xModel)1772 Window* SfxStoringHelper::GetModelWindow( const uno::Reference< frame::XModel >& xModel )
1773 {
1774     Window* pWin = 0;
1775     try {
1776         if ( xModel.is() )
1777         {
1778             uno::Reference< frame::XController > xController = xModel->getCurrentController();
1779             if ( xController.is() )
1780             {
1781                 uno::Reference< frame::XFrame > xFrame = xController->getFrame();
1782                 if ( xFrame.is() )
1783                 {
1784                     uno::Reference< awt::XWindow > xWindow = xFrame->getContainerWindow();
1785                     if ( xWindow.is() )
1786                     {
1787                         VCLXWindow* pVCLWindow = VCLXWindow::GetImplementation( xWindow );
1788                         if ( pVCLWindow )
1789                             pWin = pVCLWindow->GetWindow();
1790                     }
1791                 }
1792             }
1793         }
1794     }
1795     catch ( uno::Exception& )
1796     {
1797     }
1798 
1799     return pWin;
1800 }
1801 
1802