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