1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_framework.hxx"
26 
27 //_________________________________________________________________________________________________________________
28 //	my own includes
29 //_________________________________________________________________________________________________________________
30 
31 #include <stdio.h>
32 #include <dispatch/dispatchprovider.hxx>
33 #include <loadenv/loadenv.hxx>
34 #include <dispatch/loaddispatcher.hxx>
35 #include <dispatch/closedispatcher.hxx>
36 #include <dispatch/menudispatcher.hxx>
37 #include <dispatch/helpagentdispatcher.hxx>
38 #include <dispatch/startmoduledispatcher.hxx>
39 
40 #include <pattern/window.hxx>
41 #include <threadhelp/transactionguard.hxx>
42 #include <threadhelp/readguard.hxx>
43 #include <threadhelp/writeguard.hxx>
44 #include <dispatchcommands.h>
45 #include <protocols.h>
46 #include <services.h>
47 #include <targets.h>
48 #include <general.h>
49 
50 //_________________________________________________________________________________________________________________
51 //	interface includes
52 //_________________________________________________________________________________________________________________
53 #include <com/sun/star/frame/FrameSearchFlag.hpp>
54 #include <com/sun/star/uno/Exception.hpp>
55 #include <com/sun/star/ucb/XContentProviderManager.hpp>
56 #include <com/sun/star/document/XTypeDetection.hpp>
57 #include <com/sun/star/lang/XInitialization.hpp>
58 
59 //_________________________________________________________________________________________________________________
60 //	includes of other projects
61 //_________________________________________________________________________________________________________________
62 #include <osl/diagnose.h>
63 #include <rtl/string.h>
64 #include <rtl/ustring.hxx>
65 #include <vcl/svapp.hxx>
66 #include <rtl/ustrbuf.hxx>
67 //_________________________________________________________________________________________________________________
68 //	namespace
69 //_________________________________________________________________________________________________________________
70 
71 namespace framework{
72 
73 //_________________________________________________________________________________________________________________
74 //	non exported const
75 //_________________________________________________________________________________________________________________
76 
77 //_________________________________________________________________________________________________________________
78 //	non exported definitions
79 //_________________________________________________________________________________________________________________
80 
81 //_________________________________________________________________________________________________________________
82 //	declarations
83 //_________________________________________________________________________________________________________________
84 
85 //*****************************************************************************************************************
86 //	XInterface, XTypeProvider
87 //*****************************************************************************************************************
88 DEFINE_XINTERFACE_2( DispatchProvider                               ,
89                      OWeakObject                                    ,
90                      DIRECT_INTERFACE(css::lang::XTypeProvider     ),
91                      DIRECT_INTERFACE(css::frame::XDispatchProvider)
92                    )
93 
94 DEFINE_XTYPEPROVIDER_2( DispatchProvider             ,
95                         css::lang::XTypeProvider     ,
96                         css::frame::XDispatchProvider
97                       )
98 
99 //_________________________________________________________________________________________________________________
100 
101 /**
102     @short      standard ctor/dtor
103     @descr      These initialize a new instance of this class with needed informations for work.
104                 We hold a weakreference to our owner frame which start dispatches at us.
105                 We can't use a normal reference because he hold a reference of us too ...
106                 nobody can die so ...!
107 
108     @seealso    using at owner
109 
110     @param      xFactory
111                     reference to servicemanager to create new services.
112     @param      xFrame
113                     reference to our owner frame.
114 
115     @modified   17.05.2002 10:07, as96863
116 */
117 DispatchProvider::DispatchProvider( const css::uno::Reference< css::lang::XMultiServiceFactory >& xFactory  ,
118                                     const css::uno::Reference< css::frame::XFrame >&              xFrame    )
119 		//	Init baseclasses first
120         : ThreadHelpBase( &Application::GetSolarMutex() )
121         , OWeakObject   (                               )
122         // Init member
123         , m_xFactory    ( xFactory                      )
124         , m_xFrame      ( xFrame                        )
125 {
126 }
127 
128 //_________________________________________________________________________________________________________________
129 
130 /**
131     @short      protected(!) dtor for deinitializing
132     @descr      We made it protected to prevent using of us as base class instead as a member.
133 
134     @modified   17.05.2002 10:05, as96863
135  */
136 DispatchProvider::~DispatchProvider()
137 {
138 }
139 
140 //_________________________________________________________________________________________________________________
141 
142 /**
143     @interface  XDispatchProvider
144     @short      search a dispatcher for given URL
145     @descr      If no interceptor is set on owner, we search for right frame and dispatch URL to it.
146                 If no frame was found, we do nothing.
147                 But we doesn't do it directly here. We detect the type of our owner frame and calls
148                 specialized queryDispatch() helper dependen from that. Because a Desktop handle some
149                 requests in another way then a normal frame.
150 
151     @param      aURL
152                     URL to dispatch.
153     @param      sTargetFrameName
154                     name of searched frame.
155     @param      nSearchFlags
156                     flags for searching.
157     @return     A reference to a dispatch object for this URL (if someone was found!).
158 
159     @threadsafe yes
160     @modified   17.05.2002 10:59, as96863
161 */
162 css::uno::Reference< css::frame::XDispatch > SAL_CALL DispatchProvider::queryDispatch( const css::util::URL&  aURL             ,
163                                                                                        const ::rtl::OUString& sTargetFrameName ,
164                                                                                              sal_Int32        nSearchFlags     ) throw( css::uno::RuntimeException )
165 {
166     css::uno::Reference< css::frame::XDispatch > xDispatcher;
167 
168     /* SAFE { */
169     ReadGuard aReadLock( m_aLock );
170     css::uno::Reference< css::frame::XFrame > xOwner( m_xFrame.get(), css::uno::UNO_QUERY );
171     aReadLock.unlock();
172     /* } SAFE */
173 
174     css::uno::Reference< css::frame::XDesktop > xDesktopCheck( xOwner, css::uno::UNO_QUERY );
175 
176     if (xDesktopCheck.is())
177         xDispatcher = implts_queryDesktopDispatch(xOwner, aURL, sTargetFrameName, nSearchFlags);
178     else
179         xDispatcher = implts_queryFrameDispatch(xOwner, aURL, sTargetFrameName, nSearchFlags);
180 
181     return xDispatcher;
182 }
183 
184 //_________________________________________________________________________________________________________________
185 
186 /**
187     @interface  XDispatchProvider
188     @short      do the same like queryDispatch() ... but handle multiple dispatches at the same time
189     @descr      It's an optimism. User give us a list of queries ... and we return a list of dispatcher.
190                 If one of given queries couldn't be solved to a real existing dispatcher ...
191                 we return a list with empty references in it! Order of both lists will be retained!
192 
193     @seealso    method queryDispatch()
194 
195     @param      lDescriptions
196                     a list of all dispatch parameters for multiple requests
197     @return     A reference a list of dispatch objects for these URLs - may with some <NULL/> values inside.
198 
199     @threadsafe yes
200     @modified   17.05.2002 09:55, as96863
201 */
202 css::uno::Sequence< css::uno::Reference< css::frame::XDispatch > > SAL_CALL DispatchProvider::queryDispatches( const css::uno::Sequence< css::frame::DispatchDescriptor >& lDescriptions ) throw( css::uno::RuntimeException )
203 {
204     // Create return list - which must have same size then the given descriptor
205     // It's not allowed to pack it!
206     sal_Int32                                                          nCount     = lDescriptions.getLength();
207     css::uno::Sequence< css::uno::Reference< css::frame::XDispatch > > lDispatcher( nCount );
208 
209     // Step over all descriptors and try to get any dispatcher for it.
210     for( sal_Int32 i=0; i<nCount; ++i )
211     {
212         lDispatcher[i] = queryDispatch( lDescriptions[i].FeatureURL  ,
213                                         lDescriptions[i].FrameName   ,
214                                         lDescriptions[i].SearchFlags );
215     }
216 
217     return lDispatcher;
218 }
219 
220 //_________________________________________________________________________________________________________________
221 
222 ::sal_Bool lcl_isStartModuleDispatch (const css::util::URL& aURL)
223 {
224     return (aURL.Complete.equals(CMD_UNO_SHOWSTARTMODULE));
225 }
226 
227 //_________________________________________________________________________________________________________________
228 
229 /**
230     @short      helper for queryDispatch()
231     @descr      Every member of the frame tree (frame, desktop) must handle such request
232                 in another way. So we implement different specialized metods for every one.
233 
234     @threadsafe yes
235     @modified   20.08.2003 08:32, as96863
236  */
237 css::uno::Reference< css::frame::XDispatch > DispatchProvider::implts_queryDesktopDispatch( const css::uno::Reference< css::frame::XFrame > xDesktop         ,
238                                                                                             const css::util::URL&                           aURL             ,
239                                                                                             const ::rtl::OUString&                          sTargetFrameName ,
240                                                                                                   sal_Int32                                 nSearchFlags     )
241 {
242     css::uno::Reference< css::frame::XDispatch > xDispatcher;
243 
244     // ignore wrong requests which are not supported
245     if (
246         (sTargetFrameName==SPECIALTARGET_MENUBAR  )   ||    // valid for frame dispatches - not for desktop
247         (sTargetFrameName==SPECIALTARGET_HELPAGENT)   ||    // valid for frame dispatches - not for desktop
248         (sTargetFrameName==SPECIALTARGET_PARENT   )   ||    // we have no parent by definition
249         (sTargetFrameName==SPECIALTARGET_BEAMER   )         // beamer frames are allowed as child of tasks only -
250                                                             // and they exist more then ones. We have no idea which our sub tasks is the right one
251        )
252     {
253         return NULL;
254     }
255 
256     //-----------------------------------------------------------------------------------------------------
257     // I) handle special cases which not right for using findFrame() first
258     //-----------------------------------------------------------------------------------------------------
259 
260     //-----------------------------------------------------------------------------------------------------
261     // I.I) "_blank"
262     //  It's not the right place to create a new task here - because we are queried for a dispatch object
263     //  only, which can handle such request. Such dispatcher should create the required task on demand.
264     //  Normaly the functionality for "_blank" is provided by findFrame() - but that would create it directly
265     //  here. Thats why we must "intercept" here.
266     //-----------------------------------------------------------------------------------------------------
267     if (sTargetFrameName==SPECIALTARGET_BLANK)
268     {
269         if (implts_isLoadableContent(aURL))
270             xDispatcher = implts_getOrCreateDispatchHelper( E_BLANKDISPATCHER, xDesktop );
271     }
272 
273     //-----------------------------------------------------------------------------------------------------
274     // I.II) "_default"
275     //  This is a combination of search an empty task for recycling - or create a new one.
276     //-----------------------------------------------------------------------------------------------------
277     else
278     if (sTargetFrameName==SPECIALTARGET_DEFAULT)
279     {
280         if (implts_isLoadableContent(aURL))
281             xDispatcher = implts_getOrCreateDispatchHelper( E_DEFAULTDISPATCHER, xDesktop );
282 
283         if (lcl_isStartModuleDispatch(aURL))
284             xDispatcher = implts_getOrCreateDispatchHelper( E_STARTMODULEDISPATCHER, xDesktop );
285     }
286 
287     //-----------------------------------------------------------------------------------------------------
288     // I.III) "_self", "", "_top"
289     //  The desktop can't load any document - but he can handle some special protocols like "uno", "slot" ...
290     //  Why is "top" here handled too? Because the desktop is the topest frame. Normaly it's superflous
291     //  to use this target - but we can handle it in the same manner then "_self".
292     //-----------------------------------------------------------------------------------------------------
293     else
294     if (
295         (sTargetFrameName==SPECIALTARGET_SELF)  ||
296         (sTargetFrameName==SPECIALTARGET_TOP )  ||
297         (sTargetFrameName.getLength()<1      )
298        )
299     {
300         xDispatcher = implts_searchProtocolHandler(aURL);
301     }
302 
303     //-----------------------------------------------------------------------------------------------------
304     // I.IV) no further special targets exist
305     //  Now we have to search for the right target frame by calling findFrame() - but should provide our code
306     //  against creation of a new task if no frame could be found.
307     //  I said it b efore - it's allowed for dispatch() only.
308     //-----------------------------------------------------------------------------------------------------
309     else
310     {
311         sal_Int32 nRightFlags  = nSearchFlags;
312                   nRightFlags &= ~css::frame::FrameSearchFlag::CREATE;
313 
314         // try to find any existing target and ask him for his dispatcher
315         css::uno::Reference< css::frame::XFrame > xFoundFrame = xDesktop->findFrame(sTargetFrameName, nRightFlags);
316         if (xFoundFrame.is())
317         {
318             css::uno::Reference< css::frame::XDispatchProvider > xProvider( xFoundFrame, css::uno::UNO_QUERY );
319             xDispatcher = xProvider->queryDispatch(aURL,SPECIALTARGET_SELF,0);
320         }
321         else
322         // if it couldn't be found - but creation was allowed
323         // use special dispatcher for creatio or froward it to the browser
324         if (nSearchFlags & css::frame::FrameSearchFlag::CREATE)
325             xDispatcher = implts_getOrCreateDispatchHelper( E_CREATEDISPATCHER, xDesktop, sTargetFrameName, nSearchFlags );
326     }
327 
328     return xDispatcher;
329 }
330 
331 //_________________________________________________________________________________________________________________
332 
333 css::uno::Reference< css::frame::XDispatch > DispatchProvider::implts_queryFrameDispatch( const css::uno::Reference< css::frame::XFrame > xFrame           ,
334                                                                                           const css::util::URL&                           aURL             ,
335                                                                                           const ::rtl::OUString&                          sTargetFrameName ,
336                                                                                                 sal_Int32                                 nSearchFlags     )
337 {
338     css::uno::Reference< css::frame::XDispatch > xDispatcher;
339 
340     //-----------------------------------------------------------------------------------------------------
341     // 0) Some URLs are dispatched in a generic way (e.g. by the menu) using the default target "".
342     //    But they are specified to use her own fix target. Detect such URLs here and use the correct target.
343     //-----------------------------------------------------------------------------------------------------
344 
345     ::rtl::OUString sTargetName = sTargetFrameName;
346 
347     //-----------------------------------------------------------------------------------------------------
348     // I) handle special cases which not right for using findFrame() first
349     //-----------------------------------------------------------------------------------------------------
350 
351     //-----------------------------------------------------------------------------------------------------
352     // I.I) "_blank", "_default"
353     //  It's not the right place to create a new task here. Only the desktop can do that.
354     //  Normaly the functionality for "_blank" is provided by findFrame() - but that would create it directly
355     //  here. Thats why we must "intercept" here.
356     //-----------------------------------------------------------------------------------------------------
357     if (
358         (sTargetName==SPECIALTARGET_BLANK  ) ||
359         (sTargetName==SPECIALTARGET_DEFAULT)
360        )
361     {
362         css::uno::Reference< css::frame::XDispatchProvider > xParent( xFrame->getCreator(), css::uno::UNO_QUERY );
363         if (xParent.is())
364             xDispatcher = xParent->queryDispatch(aURL, sTargetName, 0); // its a special target - ignore search flags
365     }
366 
367     //-----------------------------------------------------------------------------------------------------
368     // I.II) "_menubar"
369     //  Special mode on frame or task to receive the local menu. Not supported by findFrame()
370     //-----------------------------------------------------------------------------------------------------
371     else
372     if (sTargetName==SPECIALTARGET_MENUBAR)
373     {
374         xDispatcher = implts_getOrCreateDispatchHelper( E_MENUDISPATCHER, xFrame );
375     }
376 
377     //-----------------------------------------------------------------------------------------------------
378     // I.III) "_helpagent"
379     //  Special mode on frame or task to start the help agent.
380     //  It's defined for top level frames only.
381     //-----------------------------------------------------------------------------------------------------
382     else
383     if (sTargetName==SPECIALTARGET_HELPAGENT)
384     {
385 		if (WindowHelper::isTopWindow(xFrame->getContainerWindow()))
386             xDispatcher = implts_getOrCreateDispatchHelper( E_HELPAGENTDISPATCHER, xFrame );
387         else
388         {
389             // Don''t use findFrame() here - because it's not possible to find
390             // a top lebel frame without knowing his name. And a frame with name
391             // "" can't be really searched! That's why forward query to any parent
392             // explicitly.
393             css::uno::Reference< css::frame::XDispatchProvider > xProvider( xFrame->getCreator(), css::uno::UNO_QUERY );
394             if (xProvider.is())
395                 xDispatcher = xProvider->queryDispatch(aURL,SPECIALTARGET_HELPAGENT,0);
396         }
397     }
398 
399     //-----------------------------------------------------------------------------------------------------
400     // I.IV) "_helpagent"
401     //  Special sub frame of a top frame only. Search or create it. ... OK it's currently a little bit HACKI.
402     //  Only the sfx (means the controller) can create it it.
403     //-----------------------------------------------------------------------------------------------------
404     else
405     if (sTargetName==SPECIALTARGET_BEAMER)
406     {
407         css::uno::Reference< css::frame::XDispatchProvider > xBeamer( xFrame->findFrame( SPECIALTARGET_BEAMER, css::frame::FrameSearchFlag::CHILDREN | css::frame::FrameSearchFlag::SELF ), css::uno::UNO_QUERY );
408         if (xBeamer.is())
409         {
410             xDispatcher = xBeamer->queryDispatch(aURL, SPECIALTARGET_SELF, 0);
411         }
412         else
413         {
414             css::uno::Reference< css::frame::XDispatchProvider > xController( xFrame->getController(), css::uno::UNO_QUERY );
415             if (xController.is())
416 				// force using of special target - but use original search flags
417 				// May the caller used the CREATE flag or not!
418                 xDispatcher = xController->queryDispatch(aURL, SPECIALTARGET_BEAMER, nSearchFlags);
419         }
420     }
421 
422     //-----------------------------------------------------------------------------------------------------
423     // I.V) "_parent"
424     //  Our parent frame (if it exist) should handle this URL.
425     //-----------------------------------------------------------------------------------------------------
426     else
427     if (sTargetName==SPECIALTARGET_PARENT)
428     {
429         css::uno::Reference< css::frame::XDispatchProvider > xParent( xFrame->getCreator(), css::uno::UNO_QUERY );
430         if (xParent.is())
431             // SELF => we must address the parent directly... and not his parent or any other parent!
432             xDispatcher = xParent->queryDispatch(aURL, SPECIALTARGET_SELF, 0);
433     }
434 
435     //-----------------------------------------------------------------------------------------------------
436     // I.VI) "_top"
437     //  This request must be forwarded to any parent frame, till we reach a top frame.
438     //  If no parent exist, we can handle itself.
439     //-----------------------------------------------------------------------------------------------------
440     else
441     if (sTargetName==SPECIALTARGET_TOP)
442     {
443         if (xFrame->isTop())
444         {
445             // If we are this top frame itself (means our owner frame)
446             // we should call ourself recursiv with a better target "_self".
447             // So we can share the same code! (see reaction for "_self" inside this method too.)
448             xDispatcher = this->queryDispatch(aURL,SPECIALTARGET_SELF,0);
449         }
450         else
451         {
452             css::uno::Reference< css::frame::XDispatchProvider > xParent( xFrame->getCreator(), css::uno::UNO_QUERY );
453             // Normaly if isTop() returned sal_False ... the parent frame MUST(!) exist ...
454             // But it seams to be better to check that here to prevent us against an access violation.
455             if (xParent.is())
456                 xDispatcher = xParent->queryDispatch(aURL, SPECIALTARGET_TOP, 0);
457         }
458     }
459 
460     //-----------------------------------------------------------------------------------------------------
461     // I.VII) "_self", ""
462     //  Our owner frame should handle this URL. But we can't do it for all of them.
463     //  So we ask the internal set controller first. If he disagree we try to find a registered
464     //  protocol handler. If this failed too - we check for a loadable content and in case of true
465     //  we load it into the frame by returning specilized dispatch object.
466     //-----------------------------------------------------------------------------------------------------
467     else
468     if (
469         (sTargetName==SPECIALTARGET_SELF)  ||
470         (sTargetName.getLength()<1      )
471        )
472     {
473         // There exist a hard coded interception for special URLs.
474         if (
475             (aURL.Complete.equalsAscii(".uno:CloseDoc"  )) ||
476             (aURL.Complete.equalsAscii(".uno:CloseWin"  ))
477            )
478         {
479             css::uno::Reference< css::frame::XDispatchProvider > xParent( xFrame->getCreator(), css::uno::UNO_QUERY );
480             // In case the frame is not a top one, is not based on system window and has a parent,
481             // the parent frame should to be queried for the correct dispatcher.
482             // See i93473
483             if (
484                 !WindowHelper::isTopWindow(xFrame->getContainerWindow()) &&
485                 !VCLUnoHelper::GetWindow(xFrame->getContainerWindow())->IsSystemWindow() &&
486                 xParent.is()
487                )
488                 xDispatcher = xParent->queryDispatch(aURL, SPECIALTARGET_SELF, 0);
489             else
490                 xDispatcher = implts_getOrCreateDispatchHelper( E_CLOSEDISPATCHER, xFrame );
491         }
492         else if (aURL.Complete.equalsAscii(".uno:CloseFrame"))
493             xDispatcher = implts_getOrCreateDispatchHelper( E_CLOSEDISPATCHER, xFrame );
494 
495 		if ( ! xDispatcher.is())
496 		{
497 			// Ask our controller for his agreement for these dispatched URL ...
498 			// because some URLs are internal and can be handled faster by SFX - which most is the current controller!
499 			// But in case of e.g. the bibliography not all queries will be handled successfully here.
500 			css::uno::Reference< css::frame::XDispatchProvider > xController( xFrame->getController(), css::uno::UNO_QUERY );
501 			if (xController.is())
502 				xDispatcher = xController->queryDispatch(aURL, SPECIALTARGET_SELF, 0);
503 		}
504 
505 		// If controller has no fun to dispatch these URL - we must search another right dispatcher.
506 		// Search for any registered protocol handler first.
507 		if (!xDispatcher.is())
508 			xDispatcher = implts_searchProtocolHandler(aURL);
509 
510 		// Not for controller - not for protocol handler
511 		// It should be a loadable content - may be a file. Check it ...
512 		// This check is necessary to found out, that
513 		// support for some protocols isn't installed by user. May be
514 		// "ftp" isn't available. So we suppress creation of our self dispatcher.
515 		// The result will be clear. He can't handle it - but he would try it.
516 		if (
517 			( ! xDispatcher.is()             )  &&
518 			( implts_isLoadableContent(aURL) )
519 		   )
520 		{
521 			xDispatcher = implts_getOrCreateDispatchHelper( E_SELFDISPATCHER, xFrame );
522 		}
523     }
524 
525     //-----------------------------------------------------------------------------------------------------
526     // I.VI) no further special handlings exist
527     //  Now we have to search for the right target frame by calling findFrame() - but should provide our code
528     //  against creation of a new task if no frame could be found.
529     //  I said it before - it's allowed for dispatch() only.
530     //-----------------------------------------------------------------------------------------------------
531     else
532     {
533         sal_Int32 nRightFlags  = nSearchFlags;
534                   nRightFlags &= ~css::frame::FrameSearchFlag::CREATE;
535 
536         // try to find any existing target and ask him for his dispatcher
537         css::uno::Reference< css::frame::XFrame > xFoundFrame = xFrame->findFrame(sTargetName, nRightFlags);
538         if (xFoundFrame.is())
539         {
540 			// Attention: Found target is our own owner frame!
541 			// Don't ask him for his dispatcher. We know it already - it's our self dispatch helper.
542 			// Otherwhise we can start a never ending recursiv call. Why?
543 			// Somewere called our owner frame - he called some interceptor objects - and may by this dispatch provider
544 			// is called. If wa use queryDispatch() on our owner frame again - we start this call stack again ... and again.
545 			if (xFoundFrame==xFrame)
546 		        xDispatcher = implts_getOrCreateDispatchHelper( E_SELFDISPATCHER, xFrame );
547 			else
548 			{
549 				css::uno::Reference< css::frame::XDispatchProvider > xProvider( xFoundFrame, css::uno::UNO_QUERY );
550 				xDispatcher = xProvider->queryDispatch(aURL,SPECIALTARGET_SELF,0);
551 			}
552         }
553         else
554         // if it couldn't be found - but creation was allowed
555         // forward request to the desktop.
556         // Note: The given target name must be used to set the name on new created task!
557         //       Don't forward request by changing it to a special one e.g _blank.
558         //       Use the CREATE flag only to prevent call against further searches.
559         //       We already know it - the target must be created new.
560         if (nSearchFlags & css::frame::FrameSearchFlag::CREATE)
561         {
562             css::uno::Reference< css::frame::XDispatchProvider > xParent( xFrame->getCreator(), css::uno::UNO_QUERY );
563             if (xParent.is())
564                 xDispatcher = xParent->queryDispatch(aURL, sTargetName, css::frame::FrameSearchFlag::CREATE);
565         }
566     }
567 
568     return xDispatcher;
569 }
570 
571 //_________________________________________________________________________________________________________________
572 
573 /**
574     @short      search for a registered protocol handler and ask him for a dispatch object
575     @descr      Wes earch a suitable handler inside our cfg package org.openoffice.Office.ProtocolHandler.
576                 If we found anyone, we create and initialize it. Initialize means: we set our owner frame on it
577                 as context information. He can use it or leave it. Of course - we are aware of handler implementations,
578                 which doesn't support initialization. It's an optional feature.
579 
580     @param      aURL
581                     the dispatch URL for which may a handler is registered
582 
583     @return     A dispatch object if a handler was found and agree with the given URL or <NULL/> otherwhise.
584 
585     @threadsafe yes
586     @modified   05.09.2002 13:43, as96863
587 */
588 css::uno::Reference< css::frame::XDispatch > DispatchProvider::implts_searchProtocolHandler( const css::util::URL& aURL )
589 {
590     css::uno::Reference< css::frame::XDispatch > xDispatcher;
591     ProtocolHandler                              aHandler   ;
592 
593     // This member is threadsafe by himself and lives if we live - we doesn't need any mutex here.
594     if (m_aProtocolHandlerCache.search(aURL,&aHandler))
595     {
596         /* SAFE { */
597         ReadGuard aReadLock( m_aLock );
598 
599         // create it
600         css::uno::Reference< css::frame::XDispatchProvider > xHandler;
601         try
602         {
603             xHandler = css::uno::Reference< css::frame::XDispatchProvider >(
604                             m_xFactory->createInstance(aHandler.m_sUNOName),
605                             css::uno::UNO_QUERY);
606         }
607         catch(css::uno::Exception&) {}
608 
609         // look if initialization is necessary
610         css::uno::Reference< css::lang::XInitialization > xInit( xHandler, css::uno::UNO_QUERY );
611         if (xInit.is())
612         {
613             css::uno::Reference< css::frame::XFrame > xOwner( m_xFrame.get(), css::uno::UNO_QUERY );
614             LOG_ASSERT(xOwner.is(), "DispatchProvider::implts_searchProtocolHandler()\nCouldn't get reference to my owner frame. So I can't set may needed context information for this protocol handler.")
615             if (xOwner.is())
616             {
617                 try
618                 {
619                     // but do it only, if all context informations are OK
620                     css::uno::Sequence< css::uno::Any > lContext(1);
621                     lContext[0] <<= xOwner;
622                     xInit->initialize(lContext);
623                 }
624                 catch(css::uno::Exception&) {}
625             }
626         }
627 
628         aReadLock.unlock();
629         /* } SAFE */
630 
631         // ask for his (sub)dispatcher for the given URL
632         if (xHandler.is())
633             xDispatcher = xHandler->queryDispatch(aURL,SPECIALTARGET_SELF,0);
634     }
635 
636     return xDispatcher;
637 }
638 
639 //_________________________________________________________________________________________________________________
640 
641 /**
642     @short      get or create new dispatch helper
643     @descr      Sometimes we need some helper implementations to support dispatching of special URLs or commands.
644                 But it's not a good idea to hold these services for the whole life time of this provider instance.
645                 We should create it on demand ...
646                 Thats why we implement this method. It return an already existing helper or create a new one otherwise.
647 
648     @attention  The parameter sTarget and nSearchFlags are defaulted to "" and 0!
649                 Please use it only, if you can be sure, that the really given by the outside calli!
650                 Mostly it depends from the parameter eHelper is they are required or not.
651 
652     @param      eHelper
653                     specify the requested dispatch helper
654     @param      xOwner
655                     the target of possible dispatch() call on created dispatch helper
656     @param      sTarget
657                     the target parameter of the original queryDispatch() request
658     @param      nSearchFlags
659                     the flags parameter of the original queryDispatch() request
660     @return     A reference to a dispatch helper.
661 
662     @threadsafe yes
663     @modified   20.08.2003 10:22, as96863
664 */
665 css::uno::Reference< css::frame::XDispatch > DispatchProvider::implts_getOrCreateDispatchHelper( EDispatchHelper                                  eHelper     ,
666                                                                                                  const css::uno::Reference< css::frame::XFrame >& xOwner      ,
667                                                                                                  const ::rtl::OUString&                           sTarget     ,
668                                                                                                        sal_Int32                                  nSearchFlags)
669 {
670     css::uno::Reference< css::frame::XDispatch > xDispatchHelper;
671 
672     /* SAFE { */
673     ReadGuard aReadLock( m_aLock );
674     css::uno::Reference< css::lang::XMultiServiceFactory > xFactory = m_xFactory;
675     aReadLock.unlock();
676     /* } SAFE */
677 
678     switch (eHelper)
679     {
680         case E_MENUDISPATCHER :
681                 {
682                     // Attention: Such menue dispatcher must be a singleton for this frame - means our owner frame.
683                     // Otherwhise he can make some trouble.
684                     /* SAFE { */
685                     WriteGuard aWriteLock( m_aLock );
686                     if ( ! m_xMenuDispatcher.is() )
687                     {
688                         MenuDispatcher* pDispatcher = new MenuDispatcher( xFactory, xOwner );
689                         m_xMenuDispatcher = css::uno::Reference< css::frame::XDispatch >( static_cast< ::cppu::OWeakObject* >(pDispatcher), css::uno::UNO_QUERY );
690                     }
691                     xDispatchHelper = m_xMenuDispatcher;
692                     aWriteLock.unlock();
693                     /* } SAFE */
694                 }
695                 break;
696 
697         case E_HELPAGENTDISPATCHER :
698                 {
699                     // Attention: It's not a good idea to create this help agent twice for the same frame (window)
700                     // May it will be shown twice too - and user activate the first one. Then he get the corresponding
701                     // help window ... but there exist another help agent window on bottom side of the frame window.
702                     // It's superflous. Create it on demand - but hold it alive till this provider dies.
703                     /* SAFE { */
704                     WriteGuard aWriteLock( m_aLock );
705                     if ( ! m_xHelpAgentDispatcher.is() )
706                     {
707                         HelpAgentDispatcher* pDispatcher = new HelpAgentDispatcher( xOwner );
708                         m_xHelpAgentDispatcher = css::uno::Reference< css::frame::XDispatch >( static_cast< ::cppu::OWeakObject* >(pDispatcher), css::uno::UNO_QUERY );
709                     }
710                     xDispatchHelper = m_xHelpAgentDispatcher;
711                     aWriteLock.unlock();
712                     /* } SAFE */
713                 }
714                 break;
715 
716         case E_CREATEDISPATCHER :
717                 {
718                     LoadDispatcher* pDispatcher = new LoadDispatcher(xFactory, xOwner, sTarget, nSearchFlags);
719                     xDispatchHelper = css::uno::Reference< css::frame::XDispatch >( static_cast< ::cppu::OWeakObject* >(pDispatcher), css::uno::UNO_QUERY );
720                 }
721                 break;
722 
723         case E_BLANKDISPATCHER :
724                 {
725                     css::uno::Reference< css::frame::XFrame > xDesktop( xOwner, css::uno::UNO_QUERY );
726                     if (xDesktop.is())
727                     {
728                         LoadDispatcher* pDispatcher = new LoadDispatcher(xFactory, xOwner, SPECIALTARGET_BLANK, 0);
729                         xDispatchHelper = css::uno::Reference< css::frame::XDispatch >( static_cast< ::cppu::OWeakObject* >(pDispatcher), css::uno::UNO_QUERY );
730                     }
731                 }
732                 break;
733 
734         case E_DEFAULTDISPATCHER :
735                 {
736                     css::uno::Reference< css::frame::XFrame > xDesktop( xOwner, css::uno::UNO_QUERY );
737                     if (xDesktop.is())
738                     {
739                         LoadDispatcher* pDispatcher = new LoadDispatcher(xFactory, xOwner, SPECIALTARGET_DEFAULT, 0);
740                         xDispatchHelper = css::uno::Reference< css::frame::XDispatch >( static_cast< ::cppu::OWeakObject* >(pDispatcher), css::uno::UNO_QUERY );
741                     }
742                 }
743                 break;
744 
745         case E_SELFDISPATCHER :
746                 {
747                     LoadDispatcher* pDispatcher = new LoadDispatcher(xFactory, xOwner, SPECIALTARGET_SELF, 0);
748                     xDispatchHelper = css::uno::Reference< css::frame::XDispatch >( static_cast< ::cppu::OWeakObject* >(pDispatcher), css::uno::UNO_QUERY );
749                 }
750                 break;
751 
752         case E_CLOSEDISPATCHER :
753                 {
754                     CloseDispatcher* pDispatcher = new CloseDispatcher( xFactory, xOwner, sTarget );
755                     xDispatchHelper = css::uno::Reference< css::frame::XDispatch >( static_cast< ::cppu::OWeakObject* >(pDispatcher), css::uno::UNO_QUERY );
756                 }
757                 break;
758 
759         case E_STARTMODULEDISPATCHER :
760                 {
761                     StartModuleDispatcher* pDispatcher = new StartModuleDispatcher( xFactory, xOwner, sTarget );
762                     xDispatchHelper = css::uno::Reference< css::frame::XDispatch >( static_cast< ::cppu::OWeakObject* >(pDispatcher), css::uno::UNO_QUERY );
763                 }
764                 break;
765     }
766 
767     return xDispatchHelper;
768 }
769 
770 //_________________________________________________________________________________________________________________
771 
772 /**
773     @short      check URL for support by our used loader or handler
774     @descr      If we must return our own dispatch helper implementations (self, blank, create dispatcher!)
775                 we should be shure, that URL describe any loadable content. Otherwise slot/uno URLs
776                 will be detected ... but there exist nothing for ral loading into a target frame!
777 
778     @param      aURL
779                     URL which should be "detected"
780     @return     <TRUE/> if somewhere could handle that - <FALSE/> otherwise.
781 
782     @threadsafe yes
783     @modified   17.05.2002 09:47, as96863
784 */
785 sal_Bool DispatchProvider::implts_isLoadableContent( const css::util::URL& aURL )
786 {
787     LoadEnv::EContentType eType = LoadEnv::classifyContent(aURL.Complete, css::uno::Sequence< css::beans::PropertyValue >());
788     return ( eType == LoadEnv::E_CAN_BE_LOADED );
789 }
790 
791 } // namespace framework
792