1 /*************************************************************************
2  *
3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4  *
5  * Copyright 2000, 2010 Oracle and/or its affiliates.
6  *
7  * OpenOffice.org - a multi-platform office productivity suite
8  *
9  * This file is part of OpenOffice.org.
10  *
11  * OpenOffice.org is free software: you can redistribute it and/or modify
12  * it under the terms of the GNU Lesser General Public License version 3
13  * only, as published by the Free Software Foundation.
14  *
15  * OpenOffice.org is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
18  * GNU Lesser General Public License version 3 for more details
19  * (a copy is included in the LICENSE file that accompanied this code).
20  *
21  * You should have received a copy of the GNU Lesser General Public License
22  * version 3 along with OpenOffice.org.  If not, see
23  * <http://www.openoffice.org/license.html>
24  * for a copy of the LGPLv3 License.
25  *
26  ************************************************************************/
27 
28 // MARKER(update_precomp.py): autogen include statement, do not remove
29 #include "precompiled_framework.hxx"
30 //_______________________________________________
31 // includes of own project
32 #include <loadenv/loadenv.hxx>
33 
34 #ifndef __FRAMEWORK_LOADENV_TARGETHELPER_HXX_
35 #include <loadenv/targethelper.hxx>
36 #endif
37 #include <framework/framelistanalyzer.hxx>
38 
39 #ifndef __FRAMEWORK_CONSTANT_FRAMELOADER_HXX_
40 #include <constant/frameloader.hxx>
41 #endif
42 
43 #ifndef __FRAMEWORK_CONSTANT_CONTENTHANDLER_HXX_
44 #include <constant/contenthandler.hxx>
45 #endif
46 
47 #ifndef __FRAMEWORK_CONSTANT_CONTAINERQUERY_HXX_
48 #include <constant/containerquery.hxx>
49 #endif
50 #include <interaction/quietinteraction.hxx>
51 #include <threadhelp/writeguard.hxx>
52 #include <threadhelp/readguard.hxx>
53 #include <threadhelp/resetableguard.hxx>
54 #include <properties.h>
55 #include <protocols.h>
56 #include <services.h>
57 #include <comphelper/interaction.hxx>
58 #include <framework/interaction.hxx>
59 
60 //_______________________________________________
61 // includes of uno interface
62 #include <com/sun/star/task/ErrorCodeRequest.hpp>
63 #include <com/sun/star/uno/RuntimeException.hpp>
64 #include <com/sun/star/frame/DispatchResultState.hpp>
65 #include <com/sun/star/frame/FrameSearchFlag.hpp>
66 #include <com/sun/star/util/XURLTransformer.hpp>
67 #include <com/sun/star/ucb/XContentProviderManager.hpp>
68 #include <com/sun/star/util/XCloseable.hpp>
69 #include <com/sun/star/lang/XComponent.hpp>
70 #include <com/sun/star/lang/XServiceInfo.hpp>
71 #include <com/sun/star/lang/DisposedException.hpp>
72 #include <com/sun/star/awt/XWindow.hpp>
73 #include <com/sun/star/awt/XWindow2.hpp>
74 #include <com/sun/star/awt/XTopWindow.hpp>
75 #include <com/sun/star/frame/XModel.hpp>
76 #include <com/sun/star/frame/XFrameLoader.hpp>
77 #include <com/sun/star/frame/XSynchronousFrameLoader.hpp>
78 #include <com/sun/star/frame/XNotifyingDispatch.hpp>
79 #include <com/sun/star/task/XStatusIndicatorFactory.hpp>
80 #include <com/sun/star/task/XStatusIndicator.hpp>
81 #include <com/sun/star/util/XModifiable.hpp>
82 #include <com/sun/star/frame/XDispatchProvider.hpp>
83 #include <com/sun/star/document/XTypeDetection.hpp>
84 #include <com/sun/star/document/XActionLockable.hpp>
85 #include <com/sun/star/io/XInputStream.hpp>
86 #include <com/sun/star/task/XInteractionHandler.hpp>
87 #include <com/sun/star/container/XNameAccess.hpp>
88 #include <com/sun/star/container/XContainerQuery.hpp>
89 #include <com/sun/star/container/XEnumeration.hpp>
90 #include <com/sun/star/document/MacroExecMode.hpp>
91 #include <com/sun/star/document/UpdateDocMode.hpp>
92 
93 //_______________________________________________
94 // includes of an other project
95 #include <vcl/window.hxx>
96 #include <vcl/wrkwin.hxx>
97 #include <vcl/syswin.hxx>
98 
99 #ifndef _TOOLKIT_HELPER_VCLUNOHELPER_HXX_
100 #include <toolkit/unohlp.hxx>
101 #endif
102 #include <unotools/moduleoptions.hxx>
103 #include <svtools/sfxecode.hxx>
104 #include <unotools/processfactory.hxx>
105 #include <unotools/ucbhelper.hxx>
106 #include <comphelper/configurationhelper.hxx>
107 #include <rtl/ustrbuf.hxx>
108 #include <vcl/svapp.hxx>
109 
110 //_______________________________________________
111 // namespace
112 
113 namespace framework{
114 
115 // may there exist already a define .-(
116 #ifndef css
117 namespace css = ::com::sun::star;
118 #endif
119 
120 //_______________________________________________
121 // declarations
122 
123 class LoadEnvListener : private ThreadHelpBase
124                       , public ::cppu::WeakImplHelper2< css::frame::XLoadEventListener      ,
125                                                         css::frame::XDispatchResultListener >
126 {
127     private:
128 
129         void**   m_ppCheck ;
130         LoadEnv* m_pLoadEnv;
131 
132     public:
133 
134         //_______________________________________
135         LoadEnvListener(void*    pCheck  ,
136                         LoadEnv* pLoadEnv)
137         {
138             m_ppCheck  = &pCheck ;
139             m_pLoadEnv = pLoadEnv;
140         }
141 
142         //_______________________________________
143         // frame.XLoadEventListener
144         virtual void SAL_CALL loadFinished(const css::uno::Reference< css::frame::XFrameLoader >& xLoader)
145             throw(css::uno::RuntimeException);
146 
147         virtual void SAL_CALL loadCancelled(const css::uno::Reference< css::frame::XFrameLoader >& xLoader)
148             throw(css::uno::RuntimeException);
149 
150         //_______________________________________
151         // frame.XDispatchResultListener
152         virtual void SAL_CALL dispatchFinished(const css::frame::DispatchResultEvent& aEvent)
153             throw(css::uno::RuntimeException);
154 
155         //_______________________________________
156         // lang.XEventListener
157         virtual void SAL_CALL disposing(const css::lang::EventObject& aEvent)
158             throw(css::uno::RuntimeException);
159 };
160 
161 /*-----------------------------------------------
162     14.10.2003 13:43
163 -----------------------------------------------*/
164 LoadEnv::LoadEnv(const css::uno::Reference< css::lang::XMultiServiceFactory >& xSMGR)
165     throw(LoadEnvException, css::uno::RuntimeException)
166     : ThreadHelpBase(     )
167     , m_xSMGR       (xSMGR)
168     , m_pCheck      (this )
169 	, m_pQuietInteraction( 0 )
170 {
171 }
172 
173 /*-----------------------------------------------
174     14.10.2003 13:43
175 -----------------------------------------------*/
176 LoadEnv::~LoadEnv()
177 {
178     m_pCheck = 0;
179 }
180 
181 /*-----------------------------------------------
182     10.09.2003 14:05
183 -----------------------------------------------*/
184 css::uno::Reference< css::lang::XComponent > LoadEnv::loadComponentFromURL(const css::uno::Reference< css::frame::XComponentLoader >&    xLoader,
185                                                                            const css::uno::Reference< css::lang::XMultiServiceFactory >& xSMGR  ,
186                                                                            const ::rtl::OUString&                                        sURL   ,
187                                                                            const ::rtl::OUString&                                        sTarget,
188                                                                                  sal_Int32                                               nFlags ,
189                                                                            const css::uno::Sequence< css::beans::PropertyValue >&        lArgs  )
190     throw(css::lang::IllegalArgumentException,
191           css::io::IOException               ,
192           css::uno::RuntimeException         )
193 {
194     css::uno::Reference< css::lang::XComponent > xComponent;
195 
196     try
197     {
198         LoadEnv aEnv(xSMGR);
199 
200         aEnv.initializeLoading(sURL,
201                                lArgs,
202                                css::uno::Reference< css::frame::XFrame >(xLoader, css::uno::UNO_QUERY),
203                                sTarget,
204                                nFlags,
205                                LoadEnv::E_NO_FEATURE);
206         aEnv.startLoading();
207         aEnv.waitWhileLoading(); // wait for ever!
208 
209         xComponent = aEnv.getTargetComponent();
210     }
211     catch(const LoadEnvException& ex)
212     {
213         switch(ex.m_nID)
214         {
215             case LoadEnvException::ID_INVALID_MEDIADESCRIPTOR:
216                     throw css::lang::IllegalArgumentException(
217                             ::rtl::OUString::createFromAscii("Optional list of arguments seem to be corrupted."),
218                             xLoader,
219                             4);
220 
221             case LoadEnvException::ID_UNSUPPORTED_CONTENT:
222                     throw css::lang::IllegalArgumentException(
223                             ::rtl::OUString::createFromAscii("URL seems to be an unsupported one."),
224                             xLoader,
225                             1);
226 
227 			default: xComponent.clear();
228                     break;
229         }
230     }
231 
232     return xComponent;
233 }
234 
235 //-----------------------------------------------
236 ::comphelper::MediaDescriptor impl_mergeMediaDescriptorWithMightExistingModelArgs(const css::uno::Sequence< css::beans::PropertyValue >& lOutsideDescriptor)
237 {
238     ::comphelper::MediaDescriptor lDescriptor(lOutsideDescriptor);
239     css::uno::Reference< css::frame::XModel > xModel     = lDescriptor.getUnpackedValueOrDefault(
240                                                             ::comphelper::MediaDescriptor::PROP_MODEL (),
241                                                             css::uno::Reference< css::frame::XModel > ());
242     if (xModel.is ())
243     {
244         ::comphelper::MediaDescriptor lModelDescriptor(xModel->getArgs());
245         ::comphelper::MediaDescriptor::iterator pIt = lModelDescriptor.find( ::comphelper::MediaDescriptor::PROP_MACROEXECUTIONMODE() );
246 		if ( pIt != lModelDescriptor.end() )
247 			lDescriptor[::comphelper::MediaDescriptor::PROP_MACROEXECUTIONMODE()] = pIt->second;
248     }
249 
250     return lDescriptor;
251 }
252 
253 /*-----------------------------------------------
254     20.08.2003 09:49
255 -----------------------------------------------*/
256 void LoadEnv::initializeLoading(const ::rtl::OUString&                                           sURL            ,
257                                 const css::uno::Sequence< css::beans::PropertyValue >&           lMediaDescriptor,
258                                 const css::uno::Reference< css::frame::XFrame >&                 xBaseFrame      ,
259                                 const ::rtl::OUString&                                           sTarget         ,
260                                       sal_Int32                                                  nSearchFlags    ,
261                                       EFeature                                                   eFeature        , // => use default ...
262                                       EContentType                                               eContentType    ) // => use default ...
263     throw(LoadEnvException, css::uno::RuntimeException)
264 {
265     // SAFE -> ----------------------------------
266     WriteGuard aWriteLock(m_aLock);
267 
268     // Handle still running processes!
269     if (m_xAsynchronousJob.is())
270         throw LoadEnvException(LoadEnvException::ID_STILL_RUNNING);
271 
272     // take over all new parameters.
273     m_xTargetFrame.clear();
274     m_xBaseFrame                    = xBaseFrame        ;
275     m_lMediaDescriptor              = impl_mergeMediaDescriptorWithMightExistingModelArgs(lMediaDescriptor);
276     m_sTarget                       = sTarget           ;
277     m_nSearchFlags                  = nSearchFlags      ;
278     m_eFeature                      = eFeature          ;
279     m_eContentType                  = eContentType      ;
280     m_bCloseFrameOnError            = sal_False         ;
281     m_bReactivateControllerOnError  = sal_False         ;
282     m_bLoaded                       = sal_False         ;
283 
284     // try to find out, if its realy a content, which can be loaded or must be "handled"
285     // We use a default value for this in-parameter. Then we have to start a complex check method
286     // internaly. But if this check was already done outside it can be supressed to perform
287     // the load request. We take over the result then!
288     if (m_eContentType == E_UNSUPPORTED_CONTENT)
289     {
290         m_eContentType = LoadEnv::classifyContent(sURL, lMediaDescriptor);
291         if (m_eContentType == E_UNSUPPORTED_CONTENT)
292             throw LoadEnvException(LoadEnvException::ID_UNSUPPORTED_CONTENT);
293     }
294 
295     // make URL part of the MediaDescriptor
296     // It doesnt mater, if its already an item of it.
297     // It must be the same value ... so we can overwrite it :-)
298     m_lMediaDescriptor[::comphelper::MediaDescriptor::PROP_URL()] <<= sURL;
299 
300     // parse it - because some following code require that
301     m_aURL.Complete = sURL;
302     css::uno::Reference< css::util::XURLTransformer > xParser(m_xSMGR->createInstance(SERVICENAME_URLTRANSFORMER), css::uno::UNO_QUERY);
303     xParser->parseStrict(m_aURL);
304 
305     // BTW: Split URL and JumpMark ...
306     // Because such mark is an explicit value of the media descriptor!
307     if (m_aURL.Mark.getLength())
308         m_lMediaDescriptor[::comphelper::MediaDescriptor::PROP_JUMPMARK()] <<= m_aURL.Mark;
309 
310     // By the way: remove the old and deprecated value "FileName" from the descriptor!
311     ::comphelper::MediaDescriptor::iterator pIt = m_lMediaDescriptor.find(::comphelper::MediaDescriptor::PROP_FILENAME());
312     if (pIt != m_lMediaDescriptor.end())
313         m_lMediaDescriptor.erase(pIt);
314 
315     // patch the MediaDescriptor, so it fullfill the outside requirements
316     // Means especialy items like e.g. UI InteractionHandler, Status Indicator,
317     // MacroExecutionMode etcpp.
318 
319     /*TODO progress is bound to a frame ... How can we set it here? */
320 
321     // UI mode
322     const bool bUIMode =
323         ( ( m_eFeature & E_WORK_WITH_UI )                                                                          == E_WORK_WITH_UI ) &&
324         ( m_lMediaDescriptor.getUnpackedValueOrDefault( ::comphelper::MediaDescriptor::PROP_HIDDEN() , sal_False ) == sal_False      ) &&
325         ( m_lMediaDescriptor.getUnpackedValueOrDefault( ::comphelper::MediaDescriptor::PROP_PREVIEW(), sal_False ) == sal_False      );
326 
327     initializeUIDefaults(
328         m_xSMGR,
329         m_lMediaDescriptor,
330         bUIMode,
331         &m_pQuietInteraction
332     );
333 
334     aWriteLock.unlock();
335     // <- SAFE ----------------------------------
336 }
337 
338 /*-----------------------------------------------
339     22.01.2010
340 -----------------------------------------------*/
341 void LoadEnv::initializeUIDefaults( const css::uno::Reference< css::lang::XMultiServiceFactory >& i_rSMGR,
342                                     ::comphelper::MediaDescriptor& io_lMediaDescriptor, const bool i_bUIMode,
343                                     QuietInteraction** o_ppQuietInteraction )
344 {
345     css::uno::Reference< css::task::XInteractionHandler > xInteractionHandler;
346     sal_Int16                                             nMacroMode         ;
347     sal_Int16                                             nUpdateMode        ;
348 
349     if ( i_bUIMode )
350     {
351         nMacroMode  = css::document::MacroExecMode::USE_CONFIG;
352         nUpdateMode = css::document::UpdateDocMode::ACCORDING_TO_CONFIG;
353         try
354         {
355             xInteractionHandler = css::uno::Reference< css::task::XInteractionHandler >(i_rSMGR->createInstance(IMPLEMENTATIONNAME_UIINTERACTIONHANDLER), css::uno::UNO_QUERY);
356         }
357         catch(const css::uno::RuntimeException&) {throw;}
358         catch(const css::uno::Exception&       ) {      }
359     }
360     // hidden mode
361     else
362     {
363         nMacroMode  = css::document::MacroExecMode::NEVER_EXECUTE;
364         nUpdateMode = css::document::UpdateDocMode::NO_UPDATE;
365         QuietInteraction* pQuietInteraction = new QuietInteraction();
366         xInteractionHandler = css::uno::Reference< css::task::XInteractionHandler >(static_cast< css::task::XInteractionHandler* >(pQuietInteraction), css::uno::UNO_QUERY);
367         if ( o_ppQuietInteraction != NULL )
368         {
369             *o_ppQuietInteraction = pQuietInteraction;
370             (*o_ppQuietInteraction)->acquire();
371         }
372     }
373 
374     if (
375         (xInteractionHandler.is()                                                                                       ) &&
376         (io_lMediaDescriptor.find(::comphelper::MediaDescriptor::PROP_INTERACTIONHANDLER()) == io_lMediaDescriptor.end())
377        )
378     {
379         io_lMediaDescriptor[::comphelper::MediaDescriptor::PROP_INTERACTIONHANDLER()] <<= xInteractionHandler;
380     }
381 
382     if (io_lMediaDescriptor.find(::comphelper::MediaDescriptor::PROP_MACROEXECUTIONMODE()) == io_lMediaDescriptor.end())
383         io_lMediaDescriptor[::comphelper::MediaDescriptor::PROP_MACROEXECUTIONMODE()] <<= nMacroMode;
384 
385     if (io_lMediaDescriptor.find(::comphelper::MediaDescriptor::PROP_UPDATEDOCMODE()) == io_lMediaDescriptor.end())
386         io_lMediaDescriptor[::comphelper::MediaDescriptor::PROP_UPDATEDOCMODE()] <<= nUpdateMode;
387 }
388 
389 /*-----------------------------------------------
390     15.08.2003 08:16
391 -----------------------------------------------*/
392 void LoadEnv::startLoading()
393     throw(LoadEnvException, css::uno::RuntimeException)
394 {
395     // SAFE ->
396     ReadGuard aReadLock(m_aLock);
397 
398     // Handle still running processes!
399     if (m_xAsynchronousJob.is())
400         throw LoadEnvException(LoadEnvException::ID_STILL_RUNNING);
401 
402     // content can not be loaded or handled
403     // check "classifyContent()" failed before ...
404     if (m_eContentType == E_UNSUPPORTED_CONTENT)
405         throw LoadEnvException(LoadEnvException::ID_UNSUPPORTED_CONTENT);
406 
407     // <- SAFE
408     aReadLock.unlock();
409 
410     // detect its type/filter etcpp.
411     // These information will be available by the
412     // used descriptor member afterwards and is needed
413     // for all following operations!
414     // Note: An exception will be thrown, in case operation was not successfully ...
415     if (m_eContentType != E_CAN_BE_SET)/* Attention: special feature to set existing component on a frame must ignore type detection! */
416         impl_detectTypeAndFilter();
417 
418     // start loading the content ...
419     // Attention: Dont check m_eContentType deeper then UNSUPPORTED/SUPPORTED!
420     // Because it was made in th easiest way ... may a flat detection was made only.
421     // And such simple detection can fail some times .-)
422     // Use another strategy here. Try it and let it run into the case "loading not possible".
423     sal_Bool bStarted = sal_False;
424     if (
425         ((m_eFeature & E_ALLOW_CONTENTHANDLER) == E_ALLOW_CONTENTHANDLER) &&
426         (m_eContentType                        != E_CAN_BE_SET          )   /* Attention: special feature to set existing component on a frame must ignore type detection! */
427        )
428     {
429         bStarted = impl_handleContent();
430     }
431 
432     if (!bStarted)
433         bStarted = impl_loadContent();
434 
435     // not started => general error
436     // We cant say - what was the reason for.
437     if (!bStarted)
438         throw LoadEnvException(LoadEnvException::ID_GENERAL_ERROR);
439 }
440 
441 /*-----------------------------------------------
442     15.08.2003 09:50
443     TODO
444         First draft does not implement timeout using [ms].
445         Current implementation counts yield calls only ...
446 -----------------------------------------------*/
447 sal_Bool LoadEnv::waitWhileLoading(sal_uInt32 nTimeout)
448     throw(LoadEnvException, css::uno::RuntimeException)
449 {
450     // Because its not a good idea to block the main thread
451     // (and we cant be shure that we are currently not used inside the
452     // main thread!), we cant use conditions here realy. We must yield
453     // in an intellegent manner :-)
454 
455     sal_Int32 nTime = nTimeout;
456     while(true)
457     {
458         // SAFE -> ------------------------------
459         ReadGuard aReadLock1(m_aLock);
460         if (!m_xAsynchronousJob.is())
461             break;
462         aReadLock1.unlock();
463         // <- SAFE ------------------------------
464 
465         Application::Yield();
466 
467         // forever!
468         if (nTimeout==0)
469             continue;
470 
471         // timed out?
472         --nTime;
473         if (nTime<1)
474             break;
475     }
476 
477     // SAFE -> ----------------------------------
478     ReadGuard aReadLock2(m_aLock);
479     return !m_xAsynchronousJob.is();
480     // <- SAFE ----------------------------------
481 }
482 
483 /*-----------------------------------------------
484     20.08.2003 10:00
485 -----------------------------------------------*/
486 void LoadEnv::cancelLoading()
487     throw(LoadEnvException, css::uno::RuntimeException)
488 {
489     // PARTIAL(!) SAFE -> ------------------------------
490     ReadGuard aReadLock(m_aLock);
491 
492     // Still running? Might waitWhileLoading()
493     // runned into the timeout!
494     if (m_xAsynchronousJob.is())
495     {
496         // try to cancel it ... if its an asynchronous frame loader
497         css::uno::Reference< css::frame::XFrameLoader > xAsyncLoader(m_xAsynchronousJob, css::uno::UNO_QUERY);
498         if (xAsyncLoader.is())
499         {
500             aReadLock.unlock();
501             // <- BREAK SAFE ------------------------------
502             xAsyncLoader->cancel();
503             // <- RESTART SAFE ----------------------------
504             aReadLock.lock();
505             /* Attention:
506                 After returning from any cancel/dispose call, neither the frame nor weself
507                 may be called back. Because only we can cancel this job, we already know
508                 the result! => Thats why its not usefull nor neccessary to wait for any
509                 asynchronous listener notification.
510             */
511             m_bLoaded = sal_False;
512             m_xAsynchronousJob.clear();
513         }
514         // or may be its a content handler? Such handler cant be cancelled in its running
515         // operation :-( And we cant deregister us there again :-(
516         // => The only chance is an exception :-)
517         else
518             throw LoadEnvException(LoadEnvException::ID_STILL_RUNNING);
519     }
520 
521     impl_reactForLoadingState();
522 
523     aReadLock.unlock();
524     // <- PARTIAL(!) SAFE ------------------------------
525 }
526 
527 /*-----------------------------------------------
528     14.08.2003 13:33
529 -----------------------------------------------*/
530 css::uno::Reference< css::frame::XFrame > LoadEnv::getTarget() const
531 {
532     // SAFE ->
533     ReadGuard aReadLock(m_aLock);
534     return m_xTargetFrame;
535     // <- SAFE
536 }
537 
538 /*-----------------------------------------------
539     14.08.2003 13:35
540 -----------------------------------------------*/
541 css::uno::Reference< css::lang::XComponent > LoadEnv::getTargetComponent() const
542 {
543     // SAFE ->
544     ReadGuard aReadLock(m_aLock);
545 
546     if (!m_xTargetFrame.is())
547         return css::uno::Reference< css::lang::XComponent >();
548 
549     css::uno::Reference< css::frame::XController > xController = m_xTargetFrame->getController();
550     if (!xController.is())
551         return css::uno::Reference< css::lang::XComponent >(m_xTargetFrame->getComponentWindow(), css::uno::UNO_QUERY);
552 
553     css::uno::Reference< css::frame::XModel > xModel = xController->getModel();
554     if (!xModel.is())
555         return css::uno::Reference< css::lang::XComponent >(xController, css::uno::UNO_QUERY);
556 
557     return css::uno::Reference< css::lang::XComponent >(xModel, css::uno::UNO_QUERY);
558     // <- SAFE
559 }
560 
561 /*-----------------------------------------------
562     15.08.2003 11:15
563 -----------------------------------------------*/
564 void SAL_CALL LoadEnvListener::loadFinished(const css::uno::Reference< css::frame::XFrameLoader >&)
565     throw(css::uno::RuntimeException)
566 {
567     // SAFE -> ----------------------------------
568     WriteGuard aWriteLock(m_aLock);
569 
570     if (m_ppCheck && *m_ppCheck)
571         m_pLoadEnv->impl_setResult(sal_True);
572     m_ppCheck = NULL;
573 
574     aWriteLock.unlock();
575     // <- SAFE ----------------------------------
576 }
577 
578 /*-----------------------------------------------
579     14.10.2003 12:23
580 -----------------------------------------------*/
581 void SAL_CALL LoadEnvListener::loadCancelled(const css::uno::Reference< css::frame::XFrameLoader >&)
582     throw(css::uno::RuntimeException)
583 {
584     // SAFE -> ----------------------------------
585     WriteGuard aWriteLock(m_aLock);
586 
587     if (m_ppCheck && *m_ppCheck)
588         m_pLoadEnv->impl_setResult(sal_False);
589     m_ppCheck = NULL;
590 
591     aWriteLock.unlock();
592     // <- SAFE ----------------------------------
593 }
594 
595 /*-----------------------------------------------
596     14.10.2003 12:23
597 -----------------------------------------------*/
598 void SAL_CALL LoadEnvListener::dispatchFinished(const css::frame::DispatchResultEvent& aEvent)
599     throw(css::uno::RuntimeException)
600 {
601     // SAFE -> ----------------------------------
602     WriteGuard aWriteLock(m_aLock);
603 
604     if (!m_ppCheck || !*m_ppCheck)
605         return;
606 
607     switch(aEvent.State)
608     {
609         case css::frame::DispatchResultState::FAILURE :
610             m_pLoadEnv->impl_setResult(sal_False);
611             break;
612 
613         case css::frame::DispatchResultState::SUCCESS :
614             m_pLoadEnv->impl_setResult(sal_False);
615             break;
616 
617         case css::frame::DispatchResultState::DONTKNOW :
618             m_pLoadEnv->impl_setResult(sal_False);
619             break;
620     }
621     m_ppCheck = NULL;
622 
623     aWriteLock.unlock();
624     // <- SAFE ----------------------------------
625 }
626 
627 /*-----------------------------------------------
628     14.10.2003 12:24
629 -----------------------------------------------*/
630 void SAL_CALL LoadEnvListener::disposing(const css::lang::EventObject&)
631     throw(css::uno::RuntimeException)
632 {
633     // SAFE -> ----------------------------------
634     WriteGuard aWriteLock(m_aLock);
635 
636     if (m_ppCheck && *m_ppCheck)
637         m_pLoadEnv->impl_setResult(sal_False);
638     m_ppCheck = NULL;
639 
640     aWriteLock.unlock();
641     // <- SAFE ----------------------------------
642 }
643 
644 /*-----------------------------------------------
645     14.10.2003 12:20
646 -----------------------------------------------*/
647 void LoadEnv::impl_setResult(sal_Bool bResult)
648 {
649     // SAFE -> ----------------------------------
650     WriteGuard aWriteLock(m_aLock);
651 
652     m_bLoaded = bResult;
653 
654     impl_reactForLoadingState();
655 
656     // clearing of this reference will unblock waitWhileLoading()!
657     // So we must be shure, that loading process was realy finished.
658     // => do it as last operation of this method ...
659     m_xAsynchronousJob.clear();
660 
661     aWriteLock.unlock();
662     // <- SAFE ----------------------------------
663 }
664 
665 /*-----------------------------------------------
666     06.02.2004 14:03
667     TODO: Is it a good idea to change Sequence<>
668           parameter to stl-adapter?
669 -----------------------------------------------*/
670 LoadEnv::EContentType LoadEnv::classifyContent(const ::rtl::OUString&                                 sURL            ,
671                                                const css::uno::Sequence< css::beans::PropertyValue >& lMediaDescriptor)
672 {
673     //-------------------------------------------
674     // (i) Filter some special well known URL protocols,
675     //     which can not be handled or loaded in general.
676     //     Of course an empty URL must be ignored here too.
677     //     Note: These URL schemata are fix and well known ...
678     //     But there can be some additional ones, which was not
679     //     defined at implementation time of this class :-(
680     //     So we have to make shure, that the following code
681     //     can detect such protocol schemata too :-)
682 
683     if(
684         (!sURL.getLength()                                       ) ||
685         (ProtocolCheck::isProtocol(sURL,ProtocolCheck::E_UNO    )) ||
686         (ProtocolCheck::isProtocol(sURL,ProtocolCheck::E_SLOT   )) ||
687         (ProtocolCheck::isProtocol(sURL,ProtocolCheck::E_MACRO  )) ||
688         (ProtocolCheck::isProtocol(sURL,ProtocolCheck::E_SERVICE)) ||
689         (ProtocolCheck::isProtocol(sURL,ProtocolCheck::E_MAILTO )) ||
690         (ProtocolCheck::isProtocol(sURL,ProtocolCheck::E_NEWS   ))
691       )
692     {
693         return E_UNSUPPORTED_CONTENT;
694     }
695 
696     //-------------------------------------------
697     // (ii) Some special URLs indicates a given input stream,
698     //      a full featured document model directly or
699     //      specify a request for opening an empty document.
700     //      Such contents are loadable in general.
701     //      But we have to check, if the media descriptor contains
702     //      all needed resources. If they are missing - the following
703     //      load request will fail.
704 
705     /* Attention: The following code cant work on such special URLs!
706                   It should not break the office .. but it make no sense
707                   to start expensive object creations and complex search
708                   algorithm if its clear, that such URLs must be handled
709                   in a special way .-)
710     */
711 
712     // creation of new documents
713     if (ProtocolCheck::isProtocol(sURL,ProtocolCheck::E_PRIVATE_FACTORY))
714         return E_CAN_BE_LOADED;
715 
716     // using of an existing input stream
717     ::comphelper::MediaDescriptor                 stlMediaDescriptor(lMediaDescriptor);
718     ::comphelper::MediaDescriptor::const_iterator pIt;
719     if (ProtocolCheck::isProtocol(sURL,ProtocolCheck::E_PRIVATE_STREAM))
720     {
721         pIt = stlMediaDescriptor.find(::comphelper::MediaDescriptor::PROP_INPUTSTREAM());
722         css::uno::Reference< css::io::XInputStream > xStream;
723         if (pIt != stlMediaDescriptor.end())
724             pIt->second >>= xStream;
725         if (xStream.is())
726             return E_CAN_BE_LOADED;
727         LOG_WARNING("LoadEnv::classifyContent()", "loading from stream with right URL but invalid stream detected")
728         return E_UNSUPPORTED_CONTENT;
729     }
730 
731     // using of a full featured document
732     if (ProtocolCheck::isProtocol(sURL,ProtocolCheck::E_PRIVATE_OBJECT))
733     {
734         pIt = stlMediaDescriptor.find(::comphelper::MediaDescriptor::PROP_MODEL());
735         css::uno::Reference< css::frame::XModel > xModel;
736         if (pIt != stlMediaDescriptor.end())
737             pIt->second >>= xModel;
738         if (xModel.is())
739             return E_CAN_BE_SET;
740         LOG_WARNING("LoadEnv::classifyContent()", "loading with object with right URL but invalid object detected")
741         return E_UNSUPPORTED_CONTENT;
742     }
743 
744     // following operatons can work on an internal type name only :-(
745     css::uno::Reference< css::lang::XMultiServiceFactory > xSMGR = ::utl::getProcessServiceFactory();
746     css::uno::Reference< css::document::XTypeDetection > xDetect(xSMGR->createInstance(SERVICENAME_TYPEDETECTION), css::uno::UNO_QUERY);
747 
748     ::rtl::OUString sType = xDetect->queryTypeByURL(sURL);
749 
750     css::uno::Sequence< css::beans::NamedValue >           lQuery(1)   ;
751     css::uno::Reference< css::container::XContainerQuery > xContainer  ;
752     css::uno::Reference< css::container::XEnumeration >    xSet        ;
753     css::uno::Sequence< ::rtl::OUString >                  lTypesReg(1);
754 
755     /*
756     //-------------------------------------------
757     lQuery[0].Name    = ::framework::constant::Filter::PROP_TYPE;
758     lQuery[0].Value <<= sType;
759 
760     xContainer = css::uno::Reference< css::container::XContainerQuery >(xSMGR->createInstance(SERVICENAME_FILTERFACTORY), css::uno::UNO_QUERY);
761     xSet       = xContainer->createSubSetEnumerationByProperties(lQuery);
762     // at least one registered frame loader is enough!
763     if (xSet->hasMoreElements())
764         return E_CAN_BE_LOADED;
765     */
766 
767     //-------------------------------------------
768     // (iii) If a FrameLoader service (or at least
769     //      a Filter) can be found, which supports
770     //      this URL - it must be a loadable content.
771     //      Because both items are registered for types
772     //      its enough to check for frame loaders only.
773     //      Mos of our filters are handled by our global
774     //      default loader. But there exist some specialized
775     //      loader, which does not work on top of filters!
776     //      So its not enough to search on the filter configuration.
777     //      Further its not enough to search for types!
778     //      Because there exist some types, which are referenced by
779     //      other objects ... but not by filters nor frame loaders!
780 
781     lTypesReg[0]      = sType;
782     lQuery[0].Name    = ::framework::constant::FrameLoader::PROP_TYPES;
783     lQuery[0].Value <<= lTypesReg;
784 
785     xContainer = css::uno::Reference< css::container::XContainerQuery >(xSMGR->createInstance(SERVICENAME_FRAMELOADERFACTORY), css::uno::UNO_QUERY);
786     xSet       = xContainer->createSubSetEnumerationByProperties(lQuery);
787     // at least one registered frame loader is enough!
788     if (xSet->hasMoreElements())
789         return E_CAN_BE_LOADED;
790 
791     //-------------------------------------------
792     // (iv) Some URL protocols are supported by special services.
793     //      E.g. ContentHandler.
794     //      Such contents can be handled ... but not loaded.
795 
796     lTypesReg[0]      = sType;
797     lQuery[0].Name    = ::framework::constant::ContentHandler::PROP_TYPES;
798     lQuery[0].Value <<= lTypesReg;
799 
800     xContainer = css::uno::Reference< css::container::XContainerQuery >(xSMGR->createInstance(SERVICENAME_CONTENTHANDLERFACTORY), css::uno::UNO_QUERY);
801     xSet       = xContainer->createSubSetEnumerationByProperties(lQuery);
802     // at least one registered content handler is enough!
803     if (xSet->hasMoreElements())
804         return E_CAN_BE_HANDLED;
805 
806     //-------------------------------------------
807     // (v) Last but not least the UCB is used inside office to
808     //     load contents. He has a special configuration to know
809     //     which URL schemata can be used inside office.
810     css::uno::Reference< css::ucb::XContentProviderManager > xUCB(xSMGR->createInstance(SERVICENAME_UCBCONTENTBROKER), css::uno::UNO_QUERY);
811     if (xUCB->queryContentProvider(sURL).is())
812         return E_CAN_BE_LOADED;
813 
814     //-------------------------------------------
815     // (TODO) At this point, we have no idea .-)
816     //        But it seems to be better, to break all
817     //        further requests for this URL. Otherwhise
818     //        we can run into some trouble.
819     return E_UNSUPPORTED_CONTENT;
820 }
821 
822 /*-----------------------------------------------
823     03.11.2003 09:31
824 -----------------------------------------------*/
825 void LoadEnv::impl_detectTypeAndFilter()
826     throw(LoadEnvException, css::uno::RuntimeException)
827 {
828     static ::rtl::OUString TYPEPROP_PREFERREDFILTER = ::rtl::OUString::createFromAscii("PreferredFilter");
829     static ::rtl::OUString FILTERPROP_FLAGS         = ::rtl::OUString::createFromAscii("Flags"          );
830     static sal_Int32       FILTERFLAG_TEMPLATEPATH  = 16;
831 
832     // SAFE ->
833     ReadGuard aReadLock(m_aLock);
834 
835     // Attention: Because our stl media descriptor is a copy of an uno sequence
836     // we cant use as an in/out parameter here. Copy it before and dont forget to
837     // update structure afterwards again!
838     css::uno::Sequence< css::beans::PropertyValue >        lDescriptor = m_lMediaDescriptor.getAsConstPropertyValueList();
839     css::uno::Reference< css::lang::XMultiServiceFactory > xSMGR       = m_xSMGR;
840 
841     aReadLock.unlock();
842     // <- SAFE
843 
844     ::rtl::OUString sType;
845     css::uno::Reference< css::document::XTypeDetection > xDetect(xSMGR->createInstance(SERVICENAME_TYPEDETECTION), css::uno::UNO_QUERY);
846     if (xDetect.is())
847         sType = xDetect->queryTypeByDescriptor(lDescriptor, sal_True); /*TODO should deep detection be able for enable/disable it from outside? */
848 
849     // no valid content -> loading not possible
850     if (!sType.getLength())
851         throw LoadEnvException(LoadEnvException::ID_UNSUPPORTED_CONTENT);
852 
853     // SAFE ->
854     WriteGuard aWriteLock(m_aLock);
855 
856     // detection was successfully => update the descriptor member of this class
857     m_lMediaDescriptor << lDescriptor;
858     m_lMediaDescriptor[::comphelper::MediaDescriptor::PROP_TYPENAME()] <<= sType;
859     // Is there an already detected (may be preselected) filter?
860     // see below ...
861     ::rtl::OUString sFilter = m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_FILTERNAME(), ::rtl::OUString());
862 
863     aWriteLock.unlock();
864     // <- SAFE
865 
866     // But the type isnt enough. For loading sometimes we need more informations.
867     // E.g. for our "_default" feature, where we recylce any frame which contains
868     // and "Untitled" document, we must know if the new document is based on a template!
869     // But this information is available as a filter property only.
870     // => We must try(!) to detect the right filter for this load request.
871     // On the other side ... if no filter is available .. ignore it.
872     // Then the type information must be enough.
873     if (!sFilter.getLength())
874     {
875         // no -> try to find a preferred filter for the detected type.
876         // Dont forget to updatet he media descriptor.
877         css::uno::Reference< css::container::XNameAccess > xTypeCont(xDetect, css::uno::UNO_QUERY_THROW);
878         try
879         {
880             ::comphelper::SequenceAsHashMap lTypeProps(xTypeCont->getByName(sType));
881             sFilter = lTypeProps.getUnpackedValueOrDefault(TYPEPROP_PREFERREDFILTER, ::rtl::OUString());
882             if (sFilter.getLength())
883             {
884                 // SAFE ->
885                 aWriteLock.lock();
886                 m_lMediaDescriptor[::comphelper::MediaDescriptor::PROP_FILTERNAME()] <<= sFilter;
887                 aWriteLock.unlock();
888                 // <- SAFE
889             }
890         }
891         catch(const css::container::NoSuchElementException&)
892             {}
893     }
894 
895     // check if the filter (if one exists) points to a template format filter.
896     // Then we have to add the property "AsTemplate".
897     // We need this information to decide afterwards if we can use a "recycle frame"
898     // for target "_default" or has to create a new one everytimes.
899     // On the other side we have to supress that, if this property already exists
900     // and should trigger a special handling. Then the outside calli of this method here,
901     // has to know, what he is doing .-)
902 
903     sal_Bool bIsOwnTemplate = sal_False;
904     if (sFilter.getLength())
905     {
906         css::uno::Reference< css::container::XNameAccess > xFilterCont(xSMGR->createInstance(SERVICENAME_FILTERFACTORY), css::uno::UNO_QUERY_THROW);
907         try
908         {
909             ::comphelper::SequenceAsHashMap lFilterProps(xFilterCont->getByName(sFilter));
910             sal_Int32 nFlags         = lFilterProps.getUnpackedValueOrDefault(FILTERPROP_FLAGS, (sal_Int32)0);
911                       bIsOwnTemplate = ((nFlags & FILTERFLAG_TEMPLATEPATH) == FILTERFLAG_TEMPLATEPATH);
912         }
913         catch(const css::container::NoSuchElementException&)
914             {}
915     }
916     if (bIsOwnTemplate)
917     {
918         // SAFE ->
919         aWriteLock.lock();
920         // Dont overwrite external decisions! See comments before ...
921         ::comphelper::MediaDescriptor::const_iterator pAsTemplateItem = m_lMediaDescriptor.find(::comphelper::MediaDescriptor::PROP_ASTEMPLATE());
922         if (pAsTemplateItem == m_lMediaDescriptor.end())
923             m_lMediaDescriptor[::comphelper::MediaDescriptor::PROP_ASTEMPLATE()] <<= sal_True;
924         aWriteLock.unlock();
925         // <- SAFE
926     }
927 }
928 
929 /*-----------------------------------------------
930     15.08.2003 09:38
931 -----------------------------------------------*/
932 sal_Bool LoadEnv::impl_handleContent()
933     throw(LoadEnvException, css::uno::RuntimeException)
934 {
935     // SAFE -> -----------------------------------
936     ReadGuard aReadLock(m_aLock);
937 
938     // the type must exist inside the descriptor ... otherwhise this class is implemented wrong :-)
939     ::rtl::OUString sType = m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_TYPENAME(), ::rtl::OUString());
940     if (!sType.getLength())
941         throw LoadEnvException(LoadEnvException::ID_INVALID_MEDIADESCRIPTOR);
942 
943     // convert media descriptor and URL to right format for later interface call!
944     css::uno::Sequence< css::beans::PropertyValue > lDescriptor;
945     m_lMediaDescriptor >> lDescriptor;
946     css::util::URL aURL = m_aURL;
947 
948     // get neccessary container to query for a handler object
949     css::uno::Reference< css::lang::XMultiServiceFactory > xFactory(m_xSMGR->createInstance(SERVICENAME_CONTENTHANDLERFACTORY), css::uno::UNO_QUERY);
950     css::uno::Reference< css::container::XContainerQuery > xQuery  (xFactory                                                  , css::uno::UNO_QUERY);
951 
952     aReadLock.unlock();
953     // <- SAFE -----------------------------------
954 
955     // query
956     css::uno::Sequence< ::rtl::OUString > lTypeReg(1);
957     lTypeReg[0] = sType;
958 
959     css::uno::Sequence< css::beans::NamedValue > lQuery(1);
960     lQuery[0].Name    = ::framework::constant::ContentHandler::PROP_TYPES;
961     lQuery[0].Value <<= lTypeReg;
962 
963     css::uno::Reference< css::container::XEnumeration > xSet = xQuery->createSubSetEnumerationByProperties(lQuery);
964     while(xSet->hasMoreElements())
965     {
966         ::comphelper::SequenceAsHashMap lProps   (xSet->nextElement());
967         ::rtl::OUString                 sHandler = lProps.getUnpackedValueOrDefault(::framework::constant::ContentHandler::PROP_NAME, ::rtl::OUString());
968 
969         css::uno::Reference< css::frame::XNotifyingDispatch > xHandler;
970         try
971         {
972             xHandler = css::uno::Reference< css::frame::XNotifyingDispatch >(xFactory->createInstance(sHandler), css::uno::UNO_QUERY);
973             if (!xHandler.is())
974                 continue;
975         }
976         catch(const css::uno::RuntimeException&)
977             { throw; }
978         catch(const css::uno::Exception&)
979             { continue; }
980 
981         // SAFE -> -----------------------------------
982         WriteGuard aWriteLock(m_aLock);
983         m_xAsynchronousJob = xHandler;
984         m_pCheck           = this;
985         LoadEnvListener* pListener = new LoadEnvListener(m_pCheck, this);
986         aWriteLock.unlock();
987         // <- SAFE -----------------------------------
988 
989         css::uno::Reference< css::frame::XDispatchResultListener > xListener(static_cast< css::frame::XDispatchResultListener* >(pListener), css::uno::UNO_QUERY);
990         xHandler->dispatchWithNotification(aURL, lDescriptor, xListener);
991 
992         return sal_True;
993     }
994 
995     return sal_False;
996 }
997 
998 //-----------------------------------------------
999 sal_Bool LoadEnv::impl_furtherDocsAllowed()
1000 {
1001     // SAFE ->
1002     ReadGuard aReadLock(m_aLock);
1003     css::uno::Reference< css::lang::XMultiServiceFactory > xSMGR = m_xSMGR;
1004     aReadLock.unlock();
1005     // <- SAFE
1006 
1007     sal_Bool bAllowed = sal_True;
1008 
1009     try
1010     {
1011         css::uno::Any aVal = ::comphelper::ConfigurationHelper::readDirectKey(
1012                                 xSMGR,
1013                                 ::rtl::OUString::createFromAscii("org.openoffice.Office.Common/"),
1014                                 ::rtl::OUString::createFromAscii("Misc"),
1015                                 ::rtl::OUString::createFromAscii("MaxOpenDocuments"),
1016                                 ::comphelper::ConfigurationHelper::E_READONLY);
1017 
1018         // NIL means: count of allowed documents = infinite !
1019         //     => return sal_True
1020         if ( ! aVal.hasValue())
1021             bAllowed = sal_True;
1022         else
1023         {
1024             sal_Int32 nMaxOpenDocuments = 0;
1025             aVal >>= nMaxOpenDocuments;
1026 
1027             css::uno::Reference< css::frame::XFramesSupplier > xDesktop(
1028                 xSMGR->createInstance(SERVICENAME_DESKTOP),
1029                 css::uno::UNO_QUERY_THROW);
1030 
1031             FrameListAnalyzer aAnalyzer(xDesktop,
1032                                         css::uno::Reference< css::frame::XFrame >(),
1033                                         FrameListAnalyzer::E_HELP |
1034                                         FrameListAnalyzer::E_BACKINGCOMPONENT |
1035                                         FrameListAnalyzer::E_HIDDEN);
1036 
1037             sal_Int32 nOpenDocuments = aAnalyzer.m_lOtherVisibleFrames.getLength();
1038                       bAllowed       = (nOpenDocuments < nMaxOpenDocuments);
1039         }
1040     }
1041     catch(const css::uno::Exception&)
1042         { bAllowed = sal_True; } // !! internal errors are no reason to disturb the office from opening documents .-)
1043 
1044     if ( ! bAllowed )
1045     {
1046         // SAFE ->
1047         aReadLock.lock();
1048         css::uno::Reference< css::task::XInteractionHandler > xInteraction = m_lMediaDescriptor.getUnpackedValueOrDefault(
1049                                                                                 ::comphelper::MediaDescriptor::PROP_INTERACTIONHANDLER(),
1050                                                                                 css::uno::Reference< css::task::XInteractionHandler >());
1051         aReadLock.unlock();
1052         // <- SAFE
1053 
1054         if (xInteraction.is())
1055         {
1056             css::uno::Any                                                                    aInteraction;
1057             css::uno::Sequence< css::uno::Reference< css::task::XInteractionContinuation > > lContinuations(2);
1058 
1059             comphelper::OInteractionAbort*   pAbort   = new comphelper::OInteractionAbort();
1060             comphelper::OInteractionApprove* pApprove = new comphelper::OInteractionApprove();
1061 
1062             lContinuations[0] = css::uno::Reference< css::task::XInteractionContinuation >(
1063                                     static_cast< css::task::XInteractionContinuation* >(pAbort),
1064                                     css::uno::UNO_QUERY_THROW);
1065             lContinuations[1] = css::uno::Reference< css::task::XInteractionContinuation >(
1066                                     static_cast< css::task::XInteractionContinuation* >(pApprove),
1067                                     css::uno::UNO_QUERY_THROW);
1068 
1069             css::task::ErrorCodeRequest aErrorCode;
1070             aErrorCode.ErrCode = ERRCODE_SFX_NOMOREDOCUMENTSALLOWED;
1071             aInteraction <<= aErrorCode;
1072             xInteraction->handle( InteractionRequest::CreateRequest(aInteraction, lContinuations) );
1073         }
1074     }
1075 
1076     return bAllowed;
1077 }
1078 
1079 //-----------------------------------------------
1080 sal_Bool LoadEnv::impl_loadContent()
1081     throw(LoadEnvException, css::uno::RuntimeException)
1082 {
1083     // SAFE -> -----------------------------------
1084     WriteGuard aWriteLock(m_aLock);
1085 
1086     // search or create right target frame
1087     ::rtl::OUString sTarget = m_sTarget;
1088     if (TargetHelper::matchSpecialTarget(sTarget, TargetHelper::E_DEFAULT))
1089     {
1090         m_xTargetFrame = impl_searchAlreadyLoaded();
1091         if (m_xTargetFrame.is())
1092 		{
1093 			impl_setResult(sal_True);
1094 			return sal_True;
1095 		}
1096         m_xTargetFrame = impl_searchRecycleTarget();
1097     }
1098 
1099     if (! m_xTargetFrame.is())
1100     {
1101         if (
1102             (TargetHelper::matchSpecialTarget(sTarget, TargetHelper::E_BLANK  )) ||
1103             (TargetHelper::matchSpecialTarget(sTarget, TargetHelper::E_DEFAULT))
1104            )
1105         {
1106             if (! impl_furtherDocsAllowed())
1107                 return sal_False;
1108             m_xTargetFrame       = m_xBaseFrame->findFrame(SPECIALTARGET_BLANK, 0);
1109             m_bCloseFrameOnError = m_xTargetFrame.is();
1110         }
1111         else
1112         {
1113             sal_Int32 nFlags = m_nSearchFlags & ~css::frame::FrameSearchFlag::CREATE;
1114             m_xTargetFrame   = m_xBaseFrame->findFrame(sTarget, nFlags);
1115             if (! m_xTargetFrame.is())
1116             {
1117                 if (! impl_furtherDocsAllowed())
1118                     return sal_False;
1119                 m_xTargetFrame       = m_xBaseFrame->findFrame(SPECIALTARGET_BLANK, 0);
1120                 m_bCloseFrameOnError = m_xTargetFrame.is();
1121             }
1122         }
1123     }
1124 
1125     // If we couldn't find a valid frame or the frame has no container window
1126     // we have to throw an exception.
1127     if (
1128         ( ! m_xTargetFrame.is()                       ) ||
1129         ( ! m_xTargetFrame->getContainerWindow().is() )
1130        )
1131         throw LoadEnvException(LoadEnvException::ID_NO_TARGET_FOUND);
1132 
1133     css::uno::Reference< css::frame::XFrame > xTargetFrame = m_xTargetFrame;
1134 
1135     // Now we have a valid frame ... and type detection was already done.
1136     // We should apply the module dependend window position and size to the
1137     // frame window.
1138     impl_applyPersistentWindowState(xTargetFrame->getContainerWindow());
1139 
1140     // Don't forget to lock task for following load process. Otherwise it could die
1141     // during this operation runs by terminating the office or closing this task via api.
1142     // If we set this lock "close()" will return false and closing will be broken.
1143     // Attention: Don't forget to reset this lock again after finishing operation.
1144     // Otherwise task AND office couldn't die!!!
1145     // This includes gracefully handling of Exceptions (Runtime!) too ...
1146     // Thats why we use a specialized guard, which will reset the lock
1147     // if it will be run out of scope.
1148 
1149     // Note further: ignore if this internal guard already contains a resource.
1150     // Might impl_searchRecylcTarget() set it before. But incase this impl-method wasnt used
1151     // and the target frame was new created ... this lock here must be set!
1152     css::uno::Reference< css::document::XActionLockable > xTargetLock(xTargetFrame, css::uno::UNO_QUERY);
1153     m_aTargetLock.setResource(xTargetLock);
1154 
1155     // Add status indicator to descriptor. Loader can show an progresses then.
1156     // But don't do it, if loading should be hidden or preview is used ...!
1157     // So we prevent our code against wrong using. Why?
1158     // It could be, that using of this progress could make trouble. e.g. He make window visible ...
1159     // but shouldn't do that. But if no indicator is available ... nobody has a chance to do that!
1160     sal_Bool                                           bHidden    = m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_HIDDEN()         , sal_False                                           );
1161     sal_Bool                                           bMinimized = m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_MINIMIZED()      , sal_False                                           );
1162     sal_Bool                                           bPreview   = m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_PREVIEW()        , sal_False                                           );
1163     css::uno::Reference< css::task::XStatusIndicator > xProgress  = m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_STATUSINDICATOR(), css::uno::Reference< css::task::XStatusIndicator >());
1164 
1165     if (!bHidden && !bMinimized && !bPreview && !xProgress.is())
1166     {
1167         // Note: its an optional interface!
1168         css::uno::Reference< css::task::XStatusIndicatorFactory > xProgressFactory(xTargetFrame, css::uno::UNO_QUERY);
1169         if (xProgressFactory.is())
1170         {
1171             xProgress = xProgressFactory->createStatusIndicator();
1172             if (xProgress.is())
1173                 m_lMediaDescriptor[::comphelper::MediaDescriptor::PROP_STATUSINDICATOR()] <<= xProgress;
1174         }
1175     }
1176 
1177     // convert media descriptor and URL to right format for later interface call!
1178     css::uno::Sequence< css::beans::PropertyValue > lDescriptor;
1179     m_lMediaDescriptor >> lDescriptor;
1180     ::rtl::OUString sURL = m_aURL.Complete;
1181 
1182     // try to locate any interested frame loader
1183     css::uno::Reference< css::uno::XInterface >                xLoader     = impl_searchLoader();
1184     css::uno::Reference< css::frame::XFrameLoader >            xAsyncLoader(xLoader, css::uno::UNO_QUERY);
1185     css::uno::Reference< css::frame::XSynchronousFrameLoader > xSyncLoader (xLoader, css::uno::UNO_QUERY);
1186 
1187     if (xAsyncLoader.is())
1188     {
1189         // SAFE -> -----------------------------------
1190         aWriteLock.lock();
1191         m_xAsynchronousJob = xAsyncLoader;
1192         m_pCheck           = this;
1193         LoadEnvListener* pListener = new LoadEnvListener(m_pCheck, this);
1194         aWriteLock.unlock();
1195         // <- SAFE -----------------------------------
1196 
1197         css::uno::Reference< css::frame::XLoadEventListener > xListener(static_cast< css::frame::XLoadEventListener* >(pListener), css::uno::UNO_QUERY);
1198         xAsyncLoader->load(xTargetFrame, sURL, lDescriptor, xListener);
1199 
1200         return sal_True;
1201     }
1202     else
1203     if (xSyncLoader.is())
1204     {
1205         sal_Bool bResult = xSyncLoader->load(lDescriptor, xTargetFrame);
1206         // react for the result here, so the outside waiting
1207         // code can ask for it later.
1208         impl_setResult(bResult);
1209         // But the return value indicates a valid started(!) operation.
1210         // And thats true everxtimes, we reach this line :-)
1211         return sal_True;
1212     }
1213 
1214     aWriteLock.unlock();
1215     // <- SAFE
1216 
1217     return sal_False;
1218 }
1219 
1220 /*-----------------------------------------------
1221     06.02.2004 14:40
1222 -----------------------------------------------*/
1223 css::uno::Reference< css::uno::XInterface > LoadEnv::impl_searchLoader()
1224 {
1225     // SAFE -> -----------------------------------
1226     ReadGuard aReadLock(m_aLock);
1227 
1228     // special mode to set an existing component on this frame
1229     // In such case the laoder is fix. It must be the SFX based implementation,
1230     // which can create a view on top of such xModel components :-)
1231     if (m_eContentType == E_CAN_BE_SET)
1232     {
1233         try
1234         {
1235             return m_xSMGR->createInstance(IMPLEMENTATIONNAME_GENERICFRAMELOADER);
1236         }
1237         catch(const css::uno::RuntimeException&)
1238             { throw; }
1239         catch(const css::uno::Exception&)
1240             {}
1241         throw LoadEnvException(LoadEnvException::ID_INVALID_ENVIRONMENT);
1242     }
1243 
1244     // Otherwhise ...
1245     // We need this type information to locate an registered frame loader
1246     // Without such information we cant work!
1247     ::rtl::OUString sType = m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_TYPENAME(), ::rtl::OUString());
1248     if (!sType.getLength())
1249         throw LoadEnvException(LoadEnvException::ID_INVALID_MEDIADESCRIPTOR);
1250 
1251     // try to locate any interested frame loader
1252     css::uno::Reference< css::lang::XMultiServiceFactory > xLoaderFactory(m_xSMGR->createInstance(SERVICENAME_FRAMELOADERFACTORY), css::uno::UNO_QUERY);
1253     css::uno::Reference< css::container::XContainerQuery > xQuery        (xLoaderFactory                                         , css::uno::UNO_QUERY);
1254 
1255     aReadLock.unlock();
1256     // <- SAFE -----------------------------------
1257 
1258     css::uno::Sequence< ::rtl::OUString > lTypesReg(1);
1259     lTypesReg[0] = sType;
1260 
1261     css::uno::Sequence< css::beans::NamedValue > lQuery(1);
1262     lQuery[0].Name    = ::framework::constant::FrameLoader::PROP_TYPES;
1263     lQuery[0].Value <<= lTypesReg;
1264 
1265     css::uno::Reference< css::container::XEnumeration > xSet = xQuery->createSubSetEnumerationByProperties(lQuery);
1266     while(xSet->hasMoreElements())
1267     {
1268         // try everyone ...
1269         // Ignore any loader, which makes trouble :-)
1270         ::comphelper::SequenceAsHashMap             lLoaderProps(xSet->nextElement());
1271         ::rtl::OUString                             sLoader     = lLoaderProps.getUnpackedValueOrDefault(::framework::constant::FrameLoader::PROP_NAME, ::rtl::OUString());
1272         css::uno::Reference< css::uno::XInterface > xLoader     ;
1273         try
1274         {
1275             xLoader = xLoaderFactory->createInstance(sLoader);
1276             if (xLoader.is())
1277                 return xLoader;
1278         }
1279         catch(const css::uno::RuntimeException&)
1280             { throw; }
1281         catch(const css::uno::Exception&)
1282             { continue; }
1283     }
1284 
1285     return css::uno::Reference< css::uno::XInterface >();
1286 }
1287 
1288 /*-----------------------------------------------
1289     24.01.2006 15:11
1290 -----------------------------------------------*/
1291 void LoadEnv::impl_jumpToMark(const css::uno::Reference< css::frame::XFrame >& xFrame,
1292                               const css::util::URL&                            aURL  )
1293 {
1294     if (! aURL.Mark.getLength())
1295         return;
1296 
1297     css::uno::Reference< css::frame::XDispatchProvider > xProvider(xFrame, css::uno::UNO_QUERY);
1298     if (! xProvider.is())
1299         return;
1300 
1301     // SAFE ->
1302     ReadGuard aReadLock(m_aLock);
1303     css::uno::Reference< css::lang::XMultiServiceFactory > xSMGR = m_xSMGR;
1304     aReadLock.unlock();
1305     // <- SAFE
1306 
1307     css::util::URL aCmd;
1308     aCmd.Complete = ::rtl::OUString::createFromAscii(".uno:JumpToMark");
1309 
1310     css::uno::Reference< css::util::XURLTransformer > xParser(xSMGR->createInstance(SERVICENAME_URLTRANSFORMER), css::uno::UNO_QUERY_THROW);
1311     xParser->parseStrict(aCmd);
1312 
1313     css::uno::Reference< css::frame::XDispatch > xDispatcher = xProvider->queryDispatch(aCmd, SPECIALTARGET_SELF, 0);
1314     if (! xDispatcher.is())
1315         return;
1316 
1317     ::comphelper::SequenceAsHashMap lArgs;
1318     lArgs[::rtl::OUString::createFromAscii("Bookmark")] <<= aURL.Mark;
1319     xDispatcher->dispatch(aCmd, lArgs.getAsConstPropertyValueList());
1320 }
1321 
1322 /*-----------------------------------------------
1323     31.07.2003 09:02
1324 -----------------------------------------------*/
1325 css::uno::Reference< css::frame::XFrame > LoadEnv::impl_searchAlreadyLoaded()
1326     throw(LoadEnvException, css::uno::RuntimeException)
1327 {
1328     // SAFE ->
1329     ReadGuard aReadLock(m_aLock);
1330 
1331     // such search is allowed for special requests only ...
1332     // or better its not allowed for some requests in general :-)
1333     if (
1334         ( ! TargetHelper::matchSpecialTarget(m_sTarget, TargetHelper::E_DEFAULT)                                               ) ||
1335         (m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_ASTEMPLATE() , sal_False) == sal_True) ||
1336 //      (m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_HIDDEN()     , sal_False) == sal_True) ||
1337         (m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_OPENNEWVIEW(), sal_False) == sal_True)
1338        )
1339     {
1340         return css::uno::Reference< css::frame::XFrame >();
1341     }
1342 
1343     // check URL
1344     // May its not usefull to start expensive document search, if it
1345     // can fail only .. because we load from a stream or model directly!
1346     if (
1347         (ProtocolCheck::isProtocol(m_aURL.Complete, ProtocolCheck::E_PRIVATE_STREAM )) ||
1348         (ProtocolCheck::isProtocol(m_aURL.Complete, ProtocolCheck::E_PRIVATE_OBJECT ))
1349         /*TODO should be private:factory here tested too? */
1350        )
1351     {
1352         return css::uno::Reference< css::frame::XFrame >();
1353     }
1354 
1355     // otherwhise - iterate through the tasks of the desktop container
1356     // to find out, which of them might contains the requested document
1357     css::uno::Reference< css::frame::XFramesSupplier >  xSupplier(m_xSMGR->createInstance(SERVICENAME_DESKTOP), css::uno::UNO_QUERY);
1358     css::uno::Reference< css::container::XIndexAccess > xTaskList(xSupplier->getFrames()                      , css::uno::UNO_QUERY);
1359 
1360     if (!xTaskList.is())
1361         return css::uno::Reference< css::frame::XFrame >(); // task list can be empty!
1362 
1363     // Note: To detect if a document was alrady loaded before
1364     // we check URLs here only. But might the existing and the requred
1365     // document has different versions! Then its URLs are the same ...
1366     sal_Int16 nNewVersion = m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_VERSION(), (sal_Int16)(-1));
1367 
1368     // will be used to save the first hidden frame referring the searched model
1369     // Normaly we are interested on visible frames ... but if there is no such visible
1370     // frame we referr to any hidden frame also (but as fallback only).
1371     css::uno::Reference< css::frame::XFrame > xHiddenTask;
1372     css::uno::Reference< css::frame::XFrame > xTask;
1373 
1374     sal_Int32 count = xTaskList->getCount();
1375     for (sal_Int32 i=0; i<count; ++i)
1376     {
1377         try
1378         {
1379             // locate model of task
1380             // Note: Without a model there is no chance to decide if
1381             // this task contains the searched document or not!
1382             xTaskList->getByIndex(i) >>= xTask;
1383             if (!xTask.is())
1384                 continue;
1385 
1386             css::uno::Reference< css::frame::XController > xController = xTask->getController();
1387             if (!xController.is())
1388 			{
1389 				xTask.clear ();
1390                 continue;
1391 			}
1392 
1393             css::uno::Reference< css::frame::XModel > xModel = xController->getModel();
1394             if (!xModel.is())
1395 			{
1396 				xTask.clear ();
1397                 continue;
1398 			}
1399 
1400             // don't check the complete URL here.
1401             // use its main part - ignore optional jumpmarks!
1402 			const ::rtl::OUString sURL = xModel->getURL();
1403             if (!::utl::UCBContentHelper::EqualURLs( m_aURL.Main, sURL ))
1404 			{
1405 				xTask.clear ();
1406                 continue;
1407 			}
1408 
1409             // get the original load arguments from the current document
1410             // and decide if its realy the same then the one will be.
1411             // It must be visible and must use the same file revision ...
1412             // or must not have any file revision set (-1 == -1!)
1413             ::comphelper::MediaDescriptor lOldDocDescriptor(xModel->getArgs());
1414 
1415             if (lOldDocDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_VERSION(), (sal_Int32)(-1)) != nNewVersion)
1416 			{
1417 				xTask.clear ();
1418                 continue;
1419 			}
1420 
1421             // Hidden frames are special.
1422             // They will be used as "last chance" if there is no visible frame pointing to the same model.
1423             // Safe the result but continue with current loop might be looking for other visible frames.
1424             ::sal_Bool bIsHidden = lOldDocDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_HIDDEN(), sal_False);
1425             if (
1426                 (   bIsHidden       ) &&
1427                 ( ! xHiddenTask.is())
1428                )
1429             {
1430                 xHiddenTask = xTask;
1431                 xTask.clear ();
1432                 continue;
1433             }
1434 
1435             // We found a visible task pointing to the right model ...
1436             // Break search.
1437             break;
1438         }
1439         catch(const css::uno::RuntimeException& exRun)
1440             { throw exRun; }
1441         catch(const css::uno::Exception&)
1442             { continue; }
1443     }
1444 
1445     css::uno::Reference< css::frame::XFrame > xResult;
1446     if (xTask.is())
1447         xResult = xTask;
1448     else
1449     if (xHiddenTask.is())
1450         xResult = xHiddenTask;
1451 
1452     if (xResult.is())
1453     {
1454         // Now we are shure, that this task includes the searched document.
1455         // It's time to activate it. As special feature we try to jump internaly
1456         // if an optional jumpmark is given too.
1457         if (m_aURL.Mark.getLength())
1458             impl_jumpToMark(xResult, m_aURL);
1459 
1460         // bring it to front and make sure it's visible...
1461         impl_makeFrameWindowVisible(xResult->getContainerWindow(), sal_True);
1462     }
1463 
1464     aReadLock.unlock();
1465     // <- SAFE
1466 
1467     return xResult;
1468 }
1469 
1470 /*-----------------------------------------------
1471     30.03.2004 09:12
1472 -----------------------------------------------*/
1473 sal_Bool LoadEnv::impl_isFrameAlreadyUsedForLoading(const css::uno::Reference< css::frame::XFrame >& xFrame) const
1474 {
1475     css::uno::Reference< css::document::XActionLockable > xLock(xFrame, css::uno::UNO_QUERY);
1476 
1477     // ? no lock interface ?
1478     // Might its an external written frame implementation :-(
1479     // Allowing using of it ... but it can fail if its not synchronized with our processes !
1480     if (!xLock.is())
1481         return sal_False;
1482 
1483     // Otherwhise we have to look for any other existing lock.
1484     return xLock->isActionLocked();
1485 }
1486 
1487 /*-----------------------------------------------
1488     30.03.2004 09:12
1489 -----------------------------------------------*/
1490 css::uno::Reference< css::frame::XFrame > LoadEnv::impl_searchRecycleTarget()
1491     throw(LoadEnvException, css::uno::RuntimeException)
1492 {
1493     // SAFE -> ..................................
1494     ReadGuard aReadLock(m_aLock);
1495 
1496     // The special backing mode frame will be recycled by definition!
1497     // It does'nt matter if somehwere whish to create a new view
1498     // or open a new untitled document ...
1499     // The only exception form that - hidden frames!
1500     if (m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_HIDDEN(), sal_False) == sal_True)
1501         return css::uno::Reference< css::frame::XFrame >();
1502 
1503     css::uno::Reference< css::frame::XFramesSupplier > xSupplier(m_xSMGR->createInstance(SERVICENAME_DESKTOP), css::uno::UNO_QUERY);
1504     FrameListAnalyzer aTasksAnalyzer(xSupplier, css::uno::Reference< css::frame::XFrame >(), FrameListAnalyzer::E_BACKINGCOMPONENT);
1505     if (aTasksAnalyzer.m_xBackingComponent.is())
1506     {
1507         if (!impl_isFrameAlreadyUsedForLoading(aTasksAnalyzer.m_xBackingComponent))
1508         {
1509             // bring it to front ...
1510             impl_makeFrameWindowVisible(aTasksAnalyzer.m_xBackingComponent->getContainerWindow(), sal_True);
1511             return aTasksAnalyzer.m_xBackingComponent;
1512         }
1513     }
1514 
1515     // These states indicates the wishing for creation of a new view in general.
1516     if (
1517         (m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_ASTEMPLATE() , sal_False) == sal_True) ||
1518         (m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_OPENNEWVIEW(), sal_False) == sal_True)
1519        )
1520     {
1521         return css::uno::Reference< css::frame::XFrame >();
1522     }
1523 
1524 	// On the other side some special URLs will open a new frame everytimes (expecting
1525 	// they can use the backing-mode frame!)
1526     if (
1527         (ProtocolCheck::isProtocol(m_aURL.Complete, ProtocolCheck::E_PRIVATE_FACTORY )) ||
1528         (ProtocolCheck::isProtocol(m_aURL.Complete, ProtocolCheck::E_PRIVATE_STREAM  )) ||
1529         (ProtocolCheck::isProtocol(m_aURL.Complete, ProtocolCheck::E_PRIVATE_OBJECT  ))
1530        )
1531     {
1532         return css::uno::Reference< css::frame::XFrame >();
1533     }
1534 
1535 	// No backing frame! No special URL => recycle active task - if possible.
1536 	// Means - if it does not already contains a modified document, or
1537 	// use another office module.
1538     css::uno::Reference< css::frame::XFrame > xTask = xSupplier->getActiveFrame();
1539 
1540     // not a real error - but might a focus problem!
1541     if (!xTask.is())
1542         return css::uno::Reference< css::frame::XFrame >();
1543 
1544     // not a real error - may its a view only
1545     css::uno::Reference< css::frame::XController > xController = xTask->getController();
1546     if (!xController.is())
1547         return css::uno::Reference< css::frame::XFrame >();
1548 
1549     // not a real error - may its a db component instead of a full feartured office document
1550     css::uno::Reference< css::frame::XModel > xModel = xController->getModel();
1551     if (!xModel.is())
1552         return css::uno::Reference< css::frame::XFrame >();
1553 
1554     // get some more informations ...
1555 
1556     // A valid set URL means: there is already a location for this document.
1557     // => it was saved there or opened from there. Such Documents can not be used here.
1558     // We search for empty document ... created by a private:factory/ URL!
1559     if (xModel->getURL().getLength()>0)
1560         return css::uno::Reference< css::frame::XFrame >();
1561 
1562     // The old document must be unmodified ...
1563     css::uno::Reference< css::util::XModifiable > xModified(xModel, css::uno::UNO_QUERY);
1564     if (xModified->isModified())
1565         return css::uno::Reference< css::frame::XFrame >();
1566 
1567     Window* pWindow = VCLUnoHelper::GetWindow(xTask->getContainerWindow());
1568     if (pWindow && pWindow->IsInModalMode())
1569         return css::uno::Reference< css::frame::XFrame >();
1570 
1571     // find out the application type of this document
1572     // We can recycle only documents, which uses the same application
1573     // then the new one.
1574     SvtModuleOptions::EFactory eOldApp = SvtModuleOptions::ClassifyFactoryByModel(xModel);
1575     SvtModuleOptions::EFactory eNewApp = SvtModuleOptions::ClassifyFactoryByURL  (m_aURL.Complete, m_lMediaDescriptor.getAsConstPropertyValueList());
1576 
1577     aReadLock.unlock();
1578     // <- SAFE ..................................
1579 
1580     if (eOldApp != eNewApp)
1581         return css::uno::Reference< css::frame::XFrame >();
1582 
1583     // OK this task seams to be useable for recycling
1584     // But we should mark it as such - means set an action lock.
1585     // Otherwhise it would be used more then ones or will be destroyed
1586     // by a close() or terminate() request.
1587     // But if such lock already exist ... it means this task is used for
1588     // any other operation already. Don't use it then.
1589     if (impl_isFrameAlreadyUsedForLoading(xTask))
1590         return css::uno::Reference< css::frame::XFrame >();
1591 
1592     // OK - there is a valid target frame.
1593     // But may be it contains already a document.
1594     // Then we have to ask it, if it allows recylcing of this frame .-)
1595     sal_Bool bReactivateOldControllerOnError = sal_False;
1596     css::uno::Reference< css::frame::XController > xOldDoc = xTask->getController();
1597     if (xOldDoc.is())
1598     {
1599         bReactivateOldControllerOnError = xOldDoc->suspend(sal_True);
1600         if (! bReactivateOldControllerOnError)
1601             return css::uno::Reference< css::frame::XFrame >();
1602     }
1603 
1604     // SAFE -> ..................................
1605     WriteGuard aWriteLock(m_aLock);
1606 
1607     css::uno::Reference< css::document::XActionLockable > xLock(xTask, css::uno::UNO_QUERY);
1608     if (!m_aTargetLock.setResource(xLock))
1609         return css::uno::Reference< css::frame::XFrame >();
1610 
1611     m_bReactivateControllerOnError = bReactivateOldControllerOnError;
1612     aWriteLock.unlock();
1613     // <- SAFE ..................................
1614 
1615     // bring it to front ...
1616     impl_makeFrameWindowVisible(xTask->getContainerWindow(), sal_True);
1617 
1618     return xTask;
1619 }
1620 
1621 /*-----------------------------------------------
1622     15.08.2003 12:39
1623 -----------------------------------------------*/
1624 void LoadEnv::impl_reactForLoadingState()
1625     throw(LoadEnvException, css::uno::RuntimeException)
1626 {
1627     /*TODO reset action locks */
1628 
1629     // SAFE -> ----------------------------------
1630     ReadGuard aReadLock(m_aLock);
1631 
1632     if (m_bLoaded)
1633     {
1634         // Bring the new loaded document to front (if allowed!).
1635         // Note: We show new created frames here only.
1636         // We dont hide already visible frames here ...
1637         css::uno::Reference< css::awt::XWindow > xWindow      = m_xTargetFrame->getContainerWindow();
1638         sal_Bool                                 bHidden      = m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_HIDDEN(), sal_False);
1639         sal_Bool                                 bMinimized = m_lMediaDescriptor.getUnpackedValueOrDefault(::comphelper::MediaDescriptor::PROP_MINIMIZED(), sal_False);
1640 
1641         if (bMinimized)
1642         {
1643             ::vos::OClearableGuard aSolarGuard(Application::GetSolarMutex());
1644             Window* pWindow = VCLUnoHelper::GetWindow(xWindow);
1645             // check for system window is neccessary to guarantee correct pointer cast!
1646             if (pWindow && pWindow->IsSystemWindow())
1647                 ((WorkWindow*)pWindow)->Minimize();
1648         }
1649         else
1650         if (!bHidden)
1651         {
1652             // show frame ... if it's not still visible ...
1653             // But do nothing if it's already visible!
1654             impl_makeFrameWindowVisible(xWindow, sal_False);
1655         }
1656 
1657         // Note: Only if an existing property "FrameName" is given by this media descriptor,
1658         // it should be used. Otherwhise we should do nothing. May be the outside code has already
1659         // set a frame name on the target!
1660         ::comphelper::MediaDescriptor::const_iterator pFrameName = m_lMediaDescriptor.find(::comphelper::MediaDescriptor::PROP_FRAMENAME());
1661         if (pFrameName != m_lMediaDescriptor.end())
1662         {
1663             ::rtl::OUString sFrameName;
1664             pFrameName->second >>= sFrameName;
1665             // Check the name again. e.g. "_default" isnt allowed.
1666             // On the other side "_beamer" is a valid name :-)
1667             if (TargetHelper::isValidNameForFrame(sFrameName))
1668                 m_xTargetFrame->setName(sFrameName);
1669         }
1670     }
1671     else if (m_bReactivateControllerOnError)
1672 	{
1673 		// Try to reactivate the old document (if any exists!)
1674 		css::uno::Reference< css::frame::XController > xOldDoc = m_xTargetFrame->getController();
1675 		// clear does not depend from reactivation state of a might existing old document!
1676 		// We must make shure, that a might following getTargetComponent() call does not return
1677 		// the old document!
1678 		m_xTargetFrame.clear();
1679 		if (xOldDoc.is())
1680 		{
1681 			sal_Bool bReactivated = xOldDoc->suspend(sal_False);
1682 			if (!bReactivated)
1683 				throw LoadEnvException(LoadEnvException::ID_COULD_NOT_REACTIVATE_CONTROLLER);
1684 			m_bReactivateControllerOnError = sal_False;
1685 		}
1686 	}
1687 	else if (m_bCloseFrameOnError)
1688 	{
1689 		// close empty frames
1690 		css::uno::Reference< css::util::XCloseable > xCloseable (m_xTargetFrame, css::uno::UNO_QUERY);
1691 		css::uno::Reference< css::lang::XComponent > xDisposable(m_xTargetFrame, css::uno::UNO_QUERY);
1692 
1693 		try
1694 		{
1695 			if (xCloseable.is())
1696 				xCloseable->close(sal_True);
1697 			else
1698 			if (xDisposable.is())
1699 				xDisposable->dispose();
1700 		}
1701 		catch(const css::util::CloseVetoException&)
1702 		{}
1703 		catch(const css::lang::DisposedException&)
1704 		{}
1705 		m_xTargetFrame.clear();
1706 	}
1707 
1708     // This max force an implicit closing of our target frame ...
1709     // e.g. in case close(sal_True) was called before and the frame
1710     // kill itself if our external use-lock is released here!
1711     // Thats why we releas this lock AFTER ALL OPERATIONS on this frame
1712     // are finished. The frame itslef must handle then
1713     // this situation gracefully.
1714     m_aTargetLock.freeResource();
1715 
1716     // Last but not least :-)
1717     // We have to clear the current media descriptor.
1718     // Otherwhise it hold a might existing stream open!
1719     m_lMediaDescriptor.clear();
1720 
1721 	css::uno::Any aRequest;
1722 	bool bThrow = false;
1723 	if ( !m_bLoaded && m_pQuietInteraction && m_pQuietInteraction->wasUsed() )
1724 	{
1725 		aRequest = m_pQuietInteraction->getRequest();
1726 		m_pQuietInteraction->release();
1727 		m_pQuietInteraction = 0;
1728 		bThrow = true;
1729 	}
1730 
1731     aReadLock.unlock();
1732 
1733 	if (bThrow)
1734 	{
1735         if  ( aRequest.isExtractableTo( ::cppu::UnoType< css::uno::Exception >::get() ) )
1736 			throw LoadEnvException( LoadEnvException::ID_GENERAL_ERROR, aRequest );
1737 	}
1738 
1739     // <- SAFE ----------------------------------
1740 }
1741 
1742 /*-----------------------------------------------
1743     16.01.2005 13:04
1744 -----------------------------------------------*/
1745 void LoadEnv::impl_makeFrameWindowVisible(const css::uno::Reference< css::awt::XWindow >& xWindow      ,
1746                                                 sal_Bool bForceToFront)
1747 {
1748     // SAFE -> ----------------------------------
1749     ReadGuard aReadLock(m_aLock);
1750     css::uno::Reference< css::lang::XMultiServiceFactory > xSMGR( m_xSMGR.get(), css::uno::UNO_QUERY );
1751     aReadLock.unlock();
1752     // <- SAFE ----------------------------------
1753 
1754     ::vos::OClearableGuard aSolarGuard(Application::GetSolarMutex());
1755     Window* pWindow = VCLUnoHelper::GetWindow(xWindow);
1756     if ( pWindow )
1757     {
1758         bool bForceFrontAndFocus(false);
1759         css::uno::Any a = ::comphelper::ConfigurationHelper::readDirectKey(
1760             xSMGR,
1761             ::rtl::OUString::createFromAscii("org.openoffice.Office.Common/View"),
1762             ::rtl::OUString::createFromAscii("NewDocumentHandling"),
1763             ::rtl::OUString::createFromAscii("ForceFocusAndToFront"),
1764             ::comphelper::ConfigurationHelper::E_READONLY);
1765         a >>= bForceFrontAndFocus;
1766 
1767         if( pWindow->IsVisible() && (bForceFrontAndFocus || bForceToFront) )
1768             pWindow->ToTop();
1769         else
1770             pWindow->Show(sal_True, (bForceFrontAndFocus || bForceToFront) ? SHOW_FOREGROUNDTASK : 0 );
1771     }
1772 
1773 /* #i19976#
1774     We tried to prevent a toFront() call in case the user putted the
1775     loading document into the background ..
1776     But we had several errors trying that. So we decided to
1777     rollback these changes and bring the new loaded document to front hardly !
1778 
1779     css::uno::Reference< css::awt::XWindow2 > xWindow2(xWindow, css::uno::UNO_QUERY);
1780 
1781     sal_Bool bIsVisible = sal_False;
1782     if (xWindow2.is())
1783         bIsVisible = xWindow2->isVisible(); // TODO is parent visible too ? .-)
1784 
1785     if (!bIsVisible)
1786     {
1787         xWindow->setVisible(sal_True);
1788         bForceToFront = sal_True;
1789     }
1790 
1791     if (
1792         (bForceToFront  ) &&
1793         (xTopWindow.is())
1794        )
1795     {
1796         xTopWindow->toFront();
1797     }
1798 */
1799 }
1800 
1801 /*-----------------------------------------------
1802     15.03.2005 11:12
1803 -----------------------------------------------*/
1804 void LoadEnv::impl_applyPersistentWindowState(const css::uno::Reference< css::awt::XWindow >& xWindow)
1805 {
1806     static ::rtl::OUString PACKAGE_SETUP_MODULES = ::rtl::OUString::createFromAscii("/org.openoffice.Setup/Office/Factories");
1807 
1808     // no window -> action not possible
1809     if (!xWindow.is())
1810         return;
1811 
1812     // window already visible -> do nothing! If we use a "recycle frame" for loading ...
1813     // the current position and size must be used.
1814     css::uno::Reference< css::awt::XWindow2 > xVisibleCheck(xWindow, css::uno::UNO_QUERY);
1815     if (
1816         (xVisibleCheck.is()        ) &&
1817         (xVisibleCheck->isVisible())
1818        )
1819        return;
1820 
1821     // SOLAR SAFE ->
1822     ::vos::OClearableGuard aSolarLock1(Application::GetSolarMutex());
1823 
1824     Window*  pWindow       = VCLUnoHelper::GetWindow(xWindow);
1825     sal_Bool bSystemWindow = pWindow->IsSystemWindow();
1826     sal_Bool bWorkWindow   = (pWindow->GetType() == WINDOW_WORKWINDOW);
1827 
1828     if (!bSystemWindow && !bWorkWindow)
1829         return;
1830 
1831     // dont overwrite this special state!
1832     WorkWindow* pWorkWindow = (WorkWindow*)pWindow;
1833     if (pWorkWindow->IsMinimized())
1834         return;
1835 
1836     aSolarLock1.clear();
1837     // <- SOLAR SAFE
1838 
1839     // SAFE ->
1840     ReadGuard aReadLock(m_aLock);
1841 
1842     // no filter -> no module -> no persistent window state
1843     ::rtl::OUString sFilter = m_lMediaDescriptor.getUnpackedValueOrDefault(
1844                                     ::comphelper::MediaDescriptor::PROP_FILTERNAME(),
1845                                     ::rtl::OUString());
1846     if (!sFilter.getLength())
1847         return;
1848 
1849     css::uno::Reference< css::lang::XMultiServiceFactory > xSMGR = m_xSMGR;
1850 
1851     aReadLock.unlock();
1852     // <- SAFE
1853 
1854     try
1855     {
1856         // retrieve the module name from the filter configuration
1857         css::uno::Reference< css::container::XNameAccess > xFilterCfg(
1858             xSMGR->createInstance(SERVICENAME_FILTERFACTORY),
1859             css::uno::UNO_QUERY_THROW);
1860         ::comphelper::SequenceAsHashMap lProps (xFilterCfg->getByName(sFilter));
1861         ::rtl::OUString                 sModule = lProps.getUnpackedValueOrDefault(FILTER_PROPNAME_DOCUMENTSERVICE, ::rtl::OUString());
1862 
1863         // get access to the configuration of this office module
1864         css::uno::Reference< css::container::XNameAccess > xModuleCfg(::comphelper::ConfigurationHelper::openConfig(
1865                                                                         xSMGR,
1866                                                                         PACKAGE_SETUP_MODULES,
1867                                                                         ::comphelper::ConfigurationHelper::E_READONLY),
1868                                                                       css::uno::UNO_QUERY_THROW);
1869 
1870         // read window state from the configuration
1871         // and apply it on the window.
1872         // Do nothing, if no configuration entry exists!
1873         ::rtl::OUString sWindowState ;
1874         ::comphelper::ConfigurationHelper::readRelativeKey(xModuleCfg, sModule, OFFICEFACTORY_PROPNAME_WINDOWATTRIBUTES) >>= sWindowState;
1875         if (sWindowState.getLength())
1876         {
1877             // SOLAR SAFE ->
1878             ::vos::OClearableGuard aSolarLock2(Application::GetSolarMutex());
1879 
1880             // We have to retrieve the window pointer again. Because nobody can guarantee
1881             // that the XWindow was not disposed inbetween .-)
1882             // But if we get a valid pointer we can be sure, that it's the system window pointer
1883             // we already checked and used before. Because nobody recylce the same uno reference for
1884             // a new internal c++ implementation ... hopefully .-))
1885             Window* pWindowCheck  = VCLUnoHelper::GetWindow(xWindow);
1886             if (! pWindowCheck)
1887                 return;
1888 
1889             SystemWindow* pSystemWindow = (SystemWindow*)pWindowCheck;
1890             pSystemWindow->SetWindowState(U2B_ENC(sWindowState,RTL_TEXTENCODING_UTF8));
1891 
1892             aSolarLock2.clear();
1893             // <- SOLAR SAFE
1894         }
1895     }
1896     catch(const css::uno::RuntimeException& exRun)
1897         { throw exRun; }
1898     catch(const css::uno::Exception&)
1899         {}
1900 }
1901 
1902 } // namespace framework
1903 
1904