1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski // MARKER(update_precomp.py): autogen include statement, do not remove
25*b1cdbd2cSJim Jagielski #include "precompiled_scripting.hxx"
26*b1cdbd2cSJim Jagielski #include "scripthandler.hxx"
27*b1cdbd2cSJim Jagielski 
28*b1cdbd2cSJim Jagielski #include <osl/mutex.hxx>
29*b1cdbd2cSJim Jagielski 
30*b1cdbd2cSJim Jagielski #include <com/sun/star/frame/DispatchResultEvent.hpp>
31*b1cdbd2cSJim Jagielski #include <com/sun/star/frame/DispatchResultState.hpp>
32*b1cdbd2cSJim Jagielski #include <com/sun/star/frame/XController.hpp>
33*b1cdbd2cSJim Jagielski #include <com/sun/star/frame/XModel.hpp>
34*b1cdbd2cSJim Jagielski 
35*b1cdbd2cSJim Jagielski #include <com/sun/star/document/XEmbeddedScripts.hpp>
36*b1cdbd2cSJim Jagielski #include <com/sun/star/document/XScriptInvocationContext.hpp>
37*b1cdbd2cSJim Jagielski 
38*b1cdbd2cSJim Jagielski #include <com/sun/star/lang/XSingleServiceFactory.hpp>
39*b1cdbd2cSJim Jagielski 
40*b1cdbd2cSJim Jagielski #include <com/sun/star/script/provider/XScriptProviderSupplier.hpp>
41*b1cdbd2cSJim Jagielski #include <com/sun/star/script/provider/XScriptProviderFactory.hpp>
42*b1cdbd2cSJim Jagielski #include <com/sun/star/script/provider/ScriptFrameworkErrorType.hpp>
43*b1cdbd2cSJim Jagielski 
44*b1cdbd2cSJim Jagielski #include <rtl/uri.hxx>
45*b1cdbd2cSJim Jagielski #include <sfx2/objsh.hxx>
46*b1cdbd2cSJim Jagielski #include <sfx2/frame.hxx>
47*b1cdbd2cSJim Jagielski #include <sfx2/sfxdlg.hxx>
48*b1cdbd2cSJim Jagielski #include <vcl/abstdlg.hxx>
49*b1cdbd2cSJim Jagielski #include <tools/diagnose_ex.h>
50*b1cdbd2cSJim Jagielski 
51*b1cdbd2cSJim Jagielski #include <cppuhelper/factory.hxx>
52*b1cdbd2cSJim Jagielski #include <cppuhelper/exc_hlp.hxx>
53*b1cdbd2cSJim Jagielski #include <util/util.hxx>
54*b1cdbd2cSJim Jagielski #include <framework/documentundoguard.hxx>
55*b1cdbd2cSJim Jagielski 
56*b1cdbd2cSJim Jagielski #include "com/sun/star/uno/XComponentContext.hpp"
57*b1cdbd2cSJim Jagielski #include "com/sun/star/uri/XUriReference.hpp"
58*b1cdbd2cSJim Jagielski #include "com/sun/star/uri/XUriReferenceFactory.hpp"
59*b1cdbd2cSJim Jagielski #include "com/sun/star/uri/XVndSunStarScriptUrl.hpp"
60*b1cdbd2cSJim Jagielski #include "com/sun/star/beans/XPropertySet.hpp"
61*b1cdbd2cSJim Jagielski 
62*b1cdbd2cSJim Jagielski using namespace ::com::sun::star;
63*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::uno;
64*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::frame;
65*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::util;
66*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::beans;
67*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::lang;
68*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::script;
69*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::script::provider;
70*b1cdbd2cSJim Jagielski using namespace ::com::sun::star::document;
71*b1cdbd2cSJim Jagielski 
72*b1cdbd2cSJim Jagielski namespace scripting_protocolhandler
73*b1cdbd2cSJim Jagielski {
74*b1cdbd2cSJim Jagielski 
75*b1cdbd2cSJim Jagielski const sal_Char * const MYSERVICENAME = "com.sun.star.frame.ProtocolHandler";
76*b1cdbd2cSJim Jagielski const sal_Char * const MYIMPLNAME = "com.sun.star.comp.ScriptProtocolHandler";
77*b1cdbd2cSJim Jagielski const sal_Char * MYSCHEME = "vnd.sun.star.script";
78*b1cdbd2cSJim Jagielski const sal_Int32 MYSCHEME_LEN = 20;
79*b1cdbd2cSJim Jagielski 
initialize(const css::uno::Sequence<css::uno::Any> & aArguments)80*b1cdbd2cSJim Jagielski void SAL_CALL ScriptProtocolHandler::initialize(
81*b1cdbd2cSJim Jagielski     const css::uno::Sequence < css::uno::Any >& aArguments )
82*b1cdbd2cSJim Jagielski     throw ( css::uno::Exception )
83*b1cdbd2cSJim Jagielski {
84*b1cdbd2cSJim Jagielski     if ( m_bInitialised )
85*b1cdbd2cSJim Jagielski     {
86*b1cdbd2cSJim Jagielski         return ;
87*b1cdbd2cSJim Jagielski     }
88*b1cdbd2cSJim Jagielski 
89*b1cdbd2cSJim Jagielski     // first argument contains a reference to the frame (may be empty or the desktop,
90*b1cdbd2cSJim Jagielski     // but usually it's a "real" frame)
91*b1cdbd2cSJim Jagielski     if ( aArguments.getLength() &&
92*b1cdbd2cSJim Jagielski          sal_False == ( aArguments[ 0 ] >>= m_xFrame ) )
93*b1cdbd2cSJim Jagielski     {
94*b1cdbd2cSJim Jagielski         ::rtl::OUString temp = OUSTR( "ScriptProtocolHandler::initialize: could not extract reference to the frame" );
95*b1cdbd2cSJim Jagielski         throw RuntimeException( temp, Reference< XInterface >() );
96*b1cdbd2cSJim Jagielski     }
97*b1cdbd2cSJim Jagielski 
98*b1cdbd2cSJim Jagielski     ENSURE_OR_THROW( m_xFactory.is(), "ScriptProtocolHandler::initialize: No Service Manager available" );
99*b1cdbd2cSJim Jagielski     m_bInitialised = true;
100*b1cdbd2cSJim Jagielski }
101*b1cdbd2cSJim Jagielski 
queryDispatch(const URL & aURL,const::rtl::OUString & sTargetFrameName,sal_Int32 nSearchFlags)102*b1cdbd2cSJim Jagielski Reference< XDispatch > SAL_CALL ScriptProtocolHandler::queryDispatch(
103*b1cdbd2cSJim Jagielski     const URL& aURL, const ::rtl::OUString& sTargetFrameName, sal_Int32 nSearchFlags )
104*b1cdbd2cSJim Jagielski     throw( ::com::sun::star::uno::RuntimeException )
105*b1cdbd2cSJim Jagielski {
106*b1cdbd2cSJim Jagielski 	(void)sTargetFrameName;
107*b1cdbd2cSJim Jagielski 	(void)nSearchFlags;
108*b1cdbd2cSJim Jagielski 
109*b1cdbd2cSJim Jagielski     Reference< XDispatch > xDispatcher;
110*b1cdbd2cSJim Jagielski     // get scheme of url
111*b1cdbd2cSJim Jagielski 
112*b1cdbd2cSJim Jagielski     Reference< uri::XUriReferenceFactory > xFac (
113*b1cdbd2cSJim Jagielski          m_xFactory->createInstance( rtl::OUString::createFromAscii(
114*b1cdbd2cSJim Jagielski             "com.sun.star.uri.UriReferenceFactory") ) , UNO_QUERY );
115*b1cdbd2cSJim Jagielski     if ( xFac.is() )
116*b1cdbd2cSJim Jagielski     {
117*b1cdbd2cSJim Jagielski         Reference<  uri::XUriReference > uriRef(
118*b1cdbd2cSJim Jagielski             xFac->parse( aURL.Complete ), UNO_QUERY );
119*b1cdbd2cSJim Jagielski         if ( uriRef.is() )
120*b1cdbd2cSJim Jagielski         {
121*b1cdbd2cSJim Jagielski             if ( uriRef->getScheme().equals( ::rtl::OUString::createFromAscii( ::scripting_protocolhandler::MYSCHEME ) ) )
122*b1cdbd2cSJim Jagielski             {
123*b1cdbd2cSJim Jagielski                 xDispatcher = this;
124*b1cdbd2cSJim Jagielski             }
125*b1cdbd2cSJim Jagielski         }
126*b1cdbd2cSJim Jagielski     }
127*b1cdbd2cSJim Jagielski 
128*b1cdbd2cSJim Jagielski     return xDispatcher;
129*b1cdbd2cSJim Jagielski }
130*b1cdbd2cSJim Jagielski 
131*b1cdbd2cSJim Jagielski Sequence< Reference< XDispatch > > SAL_CALL
queryDispatches(const Sequence<DispatchDescriptor> & seqDescriptor)132*b1cdbd2cSJim Jagielski ScriptProtocolHandler::queryDispatches(
133*b1cdbd2cSJim Jagielski const Sequence < DispatchDescriptor >& seqDescriptor )
134*b1cdbd2cSJim Jagielski throw( RuntimeException )
135*b1cdbd2cSJim Jagielski {
136*b1cdbd2cSJim Jagielski     sal_Int32 nCount = seqDescriptor.getLength();
137*b1cdbd2cSJim Jagielski     Sequence< Reference< XDispatch > > lDispatcher( nCount );
138*b1cdbd2cSJim Jagielski     for ( sal_Int32 i = 0; i < nCount; ++i )
139*b1cdbd2cSJim Jagielski     {
140*b1cdbd2cSJim Jagielski         lDispatcher[ i ] = this->queryDispatch( seqDescriptor[ i ].FeatureURL,
141*b1cdbd2cSJim Jagielski                                                 seqDescriptor[ i ].FrameName,
142*b1cdbd2cSJim Jagielski                                                 seqDescriptor[ i ].SearchFlags );
143*b1cdbd2cSJim Jagielski     }
144*b1cdbd2cSJim Jagielski     return lDispatcher;
145*b1cdbd2cSJim Jagielski }
146*b1cdbd2cSJim Jagielski 
dispatchWithNotification(const URL & aURL,const Sequence<PropertyValue> & lArgs,const Reference<XDispatchResultListener> & xListener)147*b1cdbd2cSJim Jagielski void SAL_CALL ScriptProtocolHandler::dispatchWithNotification(
148*b1cdbd2cSJim Jagielski     const URL& aURL, const Sequence < PropertyValue >& lArgs,
149*b1cdbd2cSJim Jagielski     const Reference< XDispatchResultListener >& xListener )
150*b1cdbd2cSJim Jagielski     throw ( RuntimeException )
151*b1cdbd2cSJim Jagielski {
152*b1cdbd2cSJim Jagielski 
153*b1cdbd2cSJim Jagielski     sal_Bool bSuccess = sal_False;
154*b1cdbd2cSJim Jagielski     Any invokeResult;
155*b1cdbd2cSJim Jagielski 	bool bCaughtException = sal_False;
156*b1cdbd2cSJim Jagielski 	Any aException;
157*b1cdbd2cSJim Jagielski 
158*b1cdbd2cSJim Jagielski     if ( m_bInitialised )
159*b1cdbd2cSJim Jagielski     {
160*b1cdbd2cSJim Jagielski         try
161*b1cdbd2cSJim Jagielski         {
162*b1cdbd2cSJim Jagielski             ::rtl::OUString xStringUri = ::rtl::Uri::decode( aURL.Complete,
163*b1cdbd2cSJim Jagielski                 rtl_UriDecodeWithCharset, RTL_TEXTENCODING_UTF8 );
164*b1cdbd2cSJim Jagielski             bool bIsDocumentScript = ( xStringUri.indexOfAsciiL( RTL_CONSTASCII_STRINGPARAM( "document" ) ) !=-1 );
165*b1cdbd2cSJim Jagielski 
166*b1cdbd2cSJim Jagielski             if ( bIsDocumentScript )
167*b1cdbd2cSJim Jagielski             {
168*b1cdbd2cSJim Jagielski                 // obtain the component for our security check
169*b1cdbd2cSJim Jagielski                 Reference< XEmbeddedScripts > xDocumentScripts;
170*b1cdbd2cSJim Jagielski                 if ( getScriptInvocation() )
171*b1cdbd2cSJim Jagielski                     xDocumentScripts.set( m_xScriptInvocation->getScriptContainer(), UNO_SET_THROW );
172*b1cdbd2cSJim Jagielski 
173*b1cdbd2cSJim Jagielski                 OSL_ENSURE( xDocumentScripts.is(), "ScriptProtocolHandler::dispatchWithNotification: can't do the security check!" );
174*b1cdbd2cSJim Jagielski                 if ( !xDocumentScripts.is() || !xDocumentScripts->getAllowMacroExecution() )
175*b1cdbd2cSJim Jagielski                     return;
176*b1cdbd2cSJim Jagielski             }
177*b1cdbd2cSJim Jagielski 
178*b1cdbd2cSJim Jagielski             // Creates a ScriptProvider ( if one is not created allready )
179*b1cdbd2cSJim Jagielski             createScriptProvider();
180*b1cdbd2cSJim Jagielski 
181*b1cdbd2cSJim Jagielski             Reference< provider::XScript > xFunc =
182*b1cdbd2cSJim Jagielski                 m_xScriptProvider->getScript( aURL.Complete );
183*b1cdbd2cSJim Jagielski             ENSURE_OR_THROW( xFunc.is(),
184*b1cdbd2cSJim Jagielski                 "ScriptProtocolHandler::dispatchWithNotification: validate xFunc - unable to obtain XScript interface" );
185*b1cdbd2cSJim Jagielski 
186*b1cdbd2cSJim Jagielski 
187*b1cdbd2cSJim Jagielski             Sequence< Any > inArgs( 0 );
188*b1cdbd2cSJim Jagielski             Sequence< Any > outArgs( 0 );
189*b1cdbd2cSJim Jagielski             Sequence< sal_Int16 > outIndex;
190*b1cdbd2cSJim Jagielski 
191*b1cdbd2cSJim Jagielski             if ( lArgs.getLength() > 0 )
192*b1cdbd2cSJim Jagielski             {
193*b1cdbd2cSJim Jagielski                int argCount = 0;
194*b1cdbd2cSJim Jagielski                for ( int index = 0; index < lArgs.getLength(); index++ )
195*b1cdbd2cSJim Jagielski                {
196*b1cdbd2cSJim Jagielski                    // Sometimes we get a propertyval with name = "Referer"
197*b1cdbd2cSJim Jagielski                    // this is not an argument to be passed to script, so
198*b1cdbd2cSJim Jagielski                    // ignore.
199*b1cdbd2cSJim Jagielski                    if ( lArgs[ index ].Name.compareToAscii("Referer") != 0  ||
200*b1cdbd2cSJim Jagielski                         lArgs[ index ].Name.getLength() == 0 )
201*b1cdbd2cSJim Jagielski                    {
202*b1cdbd2cSJim Jagielski                        inArgs.realloc( ++argCount );
203*b1cdbd2cSJim Jagielski                        inArgs[ argCount - 1 ] = lArgs[ index ].Value;
204*b1cdbd2cSJim Jagielski                    }
205*b1cdbd2cSJim Jagielski                }
206*b1cdbd2cSJim Jagielski             }
207*b1cdbd2cSJim Jagielski 
208*b1cdbd2cSJim Jagielski             // attempt to protect the document against the script tampering with its Undo Context
209*b1cdbd2cSJim Jagielski             ::std::auto_ptr< ::framework::DocumentUndoGuard > pUndoGuard;
210*b1cdbd2cSJim Jagielski             if ( bIsDocumentScript )
211*b1cdbd2cSJim Jagielski                 pUndoGuard.reset( new ::framework::DocumentUndoGuard( m_xScriptInvocation ) );
212*b1cdbd2cSJim Jagielski 
213*b1cdbd2cSJim Jagielski             bSuccess = sal_False;
214*b1cdbd2cSJim Jagielski             while ( !bSuccess )
215*b1cdbd2cSJim Jagielski             {
216*b1cdbd2cSJim Jagielski                 Any aFirstCaughtException;
217*b1cdbd2cSJim Jagielski                 try
218*b1cdbd2cSJim Jagielski                 {
219*b1cdbd2cSJim Jagielski                     invokeResult = xFunc->invoke( inArgs, outIndex, outArgs );
220*b1cdbd2cSJim Jagielski                     bSuccess = sal_True;
221*b1cdbd2cSJim Jagielski                 }
222*b1cdbd2cSJim Jagielski                 catch( const provider::ScriptFrameworkErrorException& se )
223*b1cdbd2cSJim Jagielski                 {
224*b1cdbd2cSJim Jagielski                     if  ( !aFirstCaughtException.hasValue() )
225*b1cdbd2cSJim Jagielski                         aFirstCaughtException = ::cppu::getCaughtException();
226*b1cdbd2cSJim Jagielski 
227*b1cdbd2cSJim Jagielski                     if ( se.errorType != provider::ScriptFrameworkErrorType::NO_SUCH_SCRIPT )
228*b1cdbd2cSJim Jagielski                         // the only condition which allows us to retry is if there is no method with the
229*b1cdbd2cSJim Jagielski                         // given name/signature
230*b1cdbd2cSJim Jagielski                         ::cppu::throwException( aFirstCaughtException );
231*b1cdbd2cSJim Jagielski 
232*b1cdbd2cSJim Jagielski                     if ( inArgs.getLength() == 0 )
233*b1cdbd2cSJim Jagielski                         // no chance to retry if we can't strip more in-args
234*b1cdbd2cSJim Jagielski                         ::cppu::throwException( aFirstCaughtException );
235*b1cdbd2cSJim Jagielski 
236*b1cdbd2cSJim Jagielski                     // strip one argument, then retry
237*b1cdbd2cSJim Jagielski                     inArgs.realloc( inArgs.getLength() - 1 );
238*b1cdbd2cSJim Jagielski                 }
239*b1cdbd2cSJim Jagielski             }
240*b1cdbd2cSJim Jagielski         }
241*b1cdbd2cSJim Jagielski         // Office doesn't handle exceptions rethrown here very well, it cores,
242*b1cdbd2cSJim Jagielski         // all we can is log them and then set fail for the dispatch event!
243*b1cdbd2cSJim Jagielski         // (if there is a listener of course)
244*b1cdbd2cSJim Jagielski         catch ( const Exception & e )
245*b1cdbd2cSJim Jagielski         {
246*b1cdbd2cSJim Jagielski             aException = ::cppu::getCaughtException();
247*b1cdbd2cSJim Jagielski 
248*b1cdbd2cSJim Jagielski             ::rtl::OUString reason = ::rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "ScriptProtocolHandler::dispatch: caught " ) );
249*b1cdbd2cSJim Jagielski 
250*b1cdbd2cSJim Jagielski             invokeResult <<= reason.concat( aException.getValueTypeName() ).concat( e.Message );
251*b1cdbd2cSJim Jagielski 
252*b1cdbd2cSJim Jagielski 			bCaughtException = sal_True;
253*b1cdbd2cSJim Jagielski         }
254*b1cdbd2cSJim Jagielski     }
255*b1cdbd2cSJim Jagielski     else
256*b1cdbd2cSJim Jagielski     {
257*b1cdbd2cSJim Jagielski         ::rtl::OUString reason = ::rtl::OUString::createFromAscii(
258*b1cdbd2cSJim Jagielski         "ScriptProtocolHandler::dispatchWithNotification failed, ScriptProtocolHandler not initialised"
259*b1cdbd2cSJim Jagielski         );
260*b1cdbd2cSJim Jagielski         invokeResult <<= reason;
261*b1cdbd2cSJim Jagielski     }
262*b1cdbd2cSJim Jagielski 
263*b1cdbd2cSJim Jagielski 	if ( bCaughtException )
264*b1cdbd2cSJim Jagielski 	{
265*b1cdbd2cSJim Jagielski 		SfxAbstractDialogFactory* pFact = SfxAbstractDialogFactory::Create();
266*b1cdbd2cSJim Jagielski 
267*b1cdbd2cSJim Jagielski 		if ( pFact != NULL )
268*b1cdbd2cSJim Jagielski 		{
269*b1cdbd2cSJim Jagielski 			VclAbstractDialog* pDlg =
270*b1cdbd2cSJim Jagielski 				pFact->CreateScriptErrorDialog( NULL, aException );
271*b1cdbd2cSJim Jagielski 
272*b1cdbd2cSJim Jagielski 			if ( pDlg != NULL )
273*b1cdbd2cSJim Jagielski 			{
274*b1cdbd2cSJim Jagielski 				pDlg->Execute();
275*b1cdbd2cSJim Jagielski 				delete pDlg;
276*b1cdbd2cSJim Jagielski 			}
277*b1cdbd2cSJim Jagielski 		}
278*b1cdbd2cSJim Jagielski    	}
279*b1cdbd2cSJim Jagielski 
280*b1cdbd2cSJim Jagielski     if ( xListener.is() )
281*b1cdbd2cSJim Jagielski     {
282*b1cdbd2cSJim Jagielski         // always call dispatchFinished(), because we didn't load a document but
283*b1cdbd2cSJim Jagielski         // executed a macro instead!
284*b1cdbd2cSJim Jagielski         ::com::sun::star::frame::DispatchResultEvent aEvent;
285*b1cdbd2cSJim Jagielski 
286*b1cdbd2cSJim Jagielski         aEvent.Source = static_cast< ::cppu::OWeakObject* >( this );
287*b1cdbd2cSJim Jagielski         aEvent.Result = invokeResult;
288*b1cdbd2cSJim Jagielski         if ( bSuccess )
289*b1cdbd2cSJim Jagielski         {
290*b1cdbd2cSJim Jagielski             aEvent.State = ::com::sun::star::frame::DispatchResultState::SUCCESS;
291*b1cdbd2cSJim Jagielski         }
292*b1cdbd2cSJim Jagielski         else
293*b1cdbd2cSJim Jagielski         {
294*b1cdbd2cSJim Jagielski             aEvent.State = ::com::sun::star::frame::DispatchResultState::FAILURE;
295*b1cdbd2cSJim Jagielski         }
296*b1cdbd2cSJim Jagielski 
297*b1cdbd2cSJim Jagielski         try
298*b1cdbd2cSJim Jagielski         {
299*b1cdbd2cSJim Jagielski             xListener->dispatchFinished( aEvent ) ;
300*b1cdbd2cSJim Jagielski         }
301*b1cdbd2cSJim Jagielski         catch(RuntimeException & e)
302*b1cdbd2cSJim Jagielski         {
303*b1cdbd2cSJim Jagielski             OSL_TRACE(
304*b1cdbd2cSJim Jagielski             "ScriptProtocolHandler::dispatchWithNotification: caught RuntimeException"
305*b1cdbd2cSJim Jagielski             "while dispatchFinished %s",
306*b1cdbd2cSJim Jagielski             ::rtl::OUStringToOString( e.Message,
307*b1cdbd2cSJim Jagielski             RTL_TEXTENCODING_ASCII_US ).pData->buffer );
308*b1cdbd2cSJim Jagielski         }
309*b1cdbd2cSJim Jagielski     }
310*b1cdbd2cSJim Jagielski }
311*b1cdbd2cSJim Jagielski 
dispatch(const URL & aURL,const Sequence<PropertyValue> & lArgs)312*b1cdbd2cSJim Jagielski void SAL_CALL ScriptProtocolHandler::dispatch(
313*b1cdbd2cSJim Jagielski const URL& aURL, const Sequence< PropertyValue >& lArgs )
314*b1cdbd2cSJim Jagielski throw ( RuntimeException )
315*b1cdbd2cSJim Jagielski {
316*b1cdbd2cSJim Jagielski     dispatchWithNotification( aURL, lArgs, Reference< XDispatchResultListener >() );
317*b1cdbd2cSJim Jagielski }
318*b1cdbd2cSJim Jagielski 
addStatusListener(const Reference<XStatusListener> & xControl,const URL & aURL)319*b1cdbd2cSJim Jagielski void SAL_CALL ScriptProtocolHandler::addStatusListener(
320*b1cdbd2cSJim Jagielski const Reference< XStatusListener >& xControl, const URL& aURL )
321*b1cdbd2cSJim Jagielski throw ( RuntimeException )
322*b1cdbd2cSJim Jagielski {
323*b1cdbd2cSJim Jagielski 	(void)xControl;
324*b1cdbd2cSJim Jagielski 	(void)aURL;
325*b1cdbd2cSJim Jagielski 
326*b1cdbd2cSJim Jagielski     // implement if status is supported
327*b1cdbd2cSJim Jagielski }
328*b1cdbd2cSJim Jagielski 
removeStatusListener(const Reference<XStatusListener> & xControl,const URL & aURL)329*b1cdbd2cSJim Jagielski void SAL_CALL ScriptProtocolHandler::removeStatusListener(
330*b1cdbd2cSJim Jagielski const Reference< XStatusListener >& xControl, const URL& aURL )
331*b1cdbd2cSJim Jagielski throw ( RuntimeException )
332*b1cdbd2cSJim Jagielski {
333*b1cdbd2cSJim Jagielski 	(void)xControl;
334*b1cdbd2cSJim Jagielski 	(void)aURL;
335*b1cdbd2cSJim Jagielski }
336*b1cdbd2cSJim Jagielski 
337*b1cdbd2cSJim Jagielski bool
getScriptInvocation()338*b1cdbd2cSJim Jagielski ScriptProtocolHandler::getScriptInvocation()
339*b1cdbd2cSJim Jagielski {
340*b1cdbd2cSJim Jagielski     if ( !m_xScriptInvocation.is() && m_xFrame.is() )
341*b1cdbd2cSJim Jagielski     {
342*b1cdbd2cSJim Jagielski         Reference< XController > xController = m_xFrame->getController();
343*b1cdbd2cSJim Jagielski         if ( xController .is() )
344*b1cdbd2cSJim Jagielski         {
345*b1cdbd2cSJim Jagielski             // try to obtain an XScriptInvocationContext interface, preferred from the
346*b1cdbd2cSJim Jagielski             // mode, then from the controller
347*b1cdbd2cSJim Jagielski             if ( !m_xScriptInvocation.set( xController->getModel(), UNO_QUERY ) )
348*b1cdbd2cSJim Jagielski                 m_xScriptInvocation.set( xController, UNO_QUERY );
349*b1cdbd2cSJim Jagielski         }
350*b1cdbd2cSJim Jagielski     }
351*b1cdbd2cSJim Jagielski     return m_xScriptInvocation.is();
352*b1cdbd2cSJim Jagielski }
353*b1cdbd2cSJim Jagielski 
createScriptProvider()354*b1cdbd2cSJim Jagielski void ScriptProtocolHandler::createScriptProvider()
355*b1cdbd2cSJim Jagielski {
356*b1cdbd2cSJim Jagielski     if ( m_xScriptProvider.is() )
357*b1cdbd2cSJim Jagielski         return;
358*b1cdbd2cSJim Jagielski 
359*b1cdbd2cSJim Jagielski     try
360*b1cdbd2cSJim Jagielski     {
361*b1cdbd2cSJim Jagielski         // first, ask the component supporting the XScriptInvocationContext interface
362*b1cdbd2cSJim Jagielski         // (if there is one) for a script provider
363*b1cdbd2cSJim Jagielski         if ( getScriptInvocation() )
364*b1cdbd2cSJim Jagielski         {
365*b1cdbd2cSJim Jagielski             Reference< XScriptProviderSupplier > xSPS( m_xScriptInvocation, UNO_QUERY );
366*b1cdbd2cSJim Jagielski             if ( xSPS.is() )
367*b1cdbd2cSJim Jagielski                 m_xScriptProvider = xSPS->getScriptProvider();
368*b1cdbd2cSJim Jagielski         }
369*b1cdbd2cSJim Jagielski 
370*b1cdbd2cSJim Jagielski         // second, ask the model in our frame
371*b1cdbd2cSJim Jagielski         if ( !m_xScriptProvider.is() && m_xFrame.is() )
372*b1cdbd2cSJim Jagielski         {
373*b1cdbd2cSJim Jagielski             Reference< XController > xController = m_xFrame->getController();
374*b1cdbd2cSJim Jagielski             if ( xController .is() )
375*b1cdbd2cSJim Jagielski             {
376*b1cdbd2cSJim Jagielski                 Reference< XScriptProviderSupplier > xSPS( xController->getModel(), UNO_QUERY );
377*b1cdbd2cSJim Jagielski                 if ( xSPS.is() )
378*b1cdbd2cSJim Jagielski                     m_xScriptProvider = xSPS->getScriptProvider();
379*b1cdbd2cSJim Jagielski             }
380*b1cdbd2cSJim Jagielski         }
381*b1cdbd2cSJim Jagielski 
382*b1cdbd2cSJim Jagielski 
383*b1cdbd2cSJim Jagielski         // as a fallback, ask the controller
384*b1cdbd2cSJim Jagielski         if ( !m_xScriptProvider.is() && m_xFrame.is() )
385*b1cdbd2cSJim Jagielski         {
386*b1cdbd2cSJim Jagielski             Reference< XScriptProviderSupplier > xSPS( m_xFrame->getController(), UNO_QUERY );
387*b1cdbd2cSJim Jagielski             if ( xSPS.is() )
388*b1cdbd2cSJim Jagielski                 m_xScriptProvider = xSPS->getScriptProvider();
389*b1cdbd2cSJim Jagielski         }
390*b1cdbd2cSJim Jagielski 
391*b1cdbd2cSJim Jagielski         // if nothing of this is successful, use the master script provider
392*b1cdbd2cSJim Jagielski         if ( !m_xScriptProvider.is() )
393*b1cdbd2cSJim Jagielski         {
394*b1cdbd2cSJim Jagielski             Reference< XPropertySet > xProps( m_xFactory, UNO_QUERY_THROW );
395*b1cdbd2cSJim Jagielski 
396*b1cdbd2cSJim Jagielski             ::rtl::OUString dc(
397*b1cdbd2cSJim Jagielski                 RTL_CONSTASCII_USTRINGPARAM( "DefaultContext" ) );
398*b1cdbd2cSJim Jagielski 
399*b1cdbd2cSJim Jagielski             Reference< XComponentContext > xCtx(
400*b1cdbd2cSJim Jagielski                 xProps->getPropertyValue( dc ), UNO_QUERY_THROW );
401*b1cdbd2cSJim Jagielski 
402*b1cdbd2cSJim Jagielski             ::rtl::OUString tmspf = ::rtl::OUString::createFromAscii(
403*b1cdbd2cSJim Jagielski                 "/singletons/com.sun.star.script.provider.theMasterScriptProviderFactory");
404*b1cdbd2cSJim Jagielski 
405*b1cdbd2cSJim Jagielski             Reference< provider::XScriptProviderFactory > xFac(
406*b1cdbd2cSJim Jagielski                 xCtx->getValueByName( tmspf ), UNO_QUERY_THROW );
407*b1cdbd2cSJim Jagielski 
408*b1cdbd2cSJim Jagielski             Any aContext;
409*b1cdbd2cSJim Jagielski             if ( getScriptInvocation() )
410*b1cdbd2cSJim Jagielski                 aContext = makeAny( m_xScriptInvocation );
411*b1cdbd2cSJim Jagielski             m_xScriptProvider = Reference< provider::XScriptProvider > (
412*b1cdbd2cSJim Jagielski                 xFac->createScriptProvider( aContext ), UNO_QUERY_THROW );
413*b1cdbd2cSJim Jagielski         }
414*b1cdbd2cSJim Jagielski     }
415*b1cdbd2cSJim Jagielski     catch ( RuntimeException & e )
416*b1cdbd2cSJim Jagielski     {
417*b1cdbd2cSJim Jagielski         ::rtl::OUString temp = OUSTR( "ScriptProtocolHandler::createScriptProvider(),  " );
418*b1cdbd2cSJim Jagielski         throw RuntimeException( temp.concat( e.Message ), Reference< XInterface >() );
419*b1cdbd2cSJim Jagielski     }
420*b1cdbd2cSJim Jagielski     catch ( Exception & e )
421*b1cdbd2cSJim Jagielski     {
422*b1cdbd2cSJim Jagielski         ::rtl::OUString temp = OUSTR( "ScriptProtocolHandler::createScriptProvider: " );
423*b1cdbd2cSJim Jagielski         throw RuntimeException( temp.concat( e.Message ), Reference< XInterface >() );
424*b1cdbd2cSJim Jagielski     }
425*b1cdbd2cSJim Jagielski }
426*b1cdbd2cSJim Jagielski 
ScriptProtocolHandler(Reference<css::lang::XMultiServiceFactory> const & rFact)427*b1cdbd2cSJim Jagielski ScriptProtocolHandler::ScriptProtocolHandler(
428*b1cdbd2cSJim Jagielski Reference< css::lang::XMultiServiceFactory > const& rFact ) :
429*b1cdbd2cSJim Jagielski m_bInitialised( false ), m_xFactory( rFact )
430*b1cdbd2cSJim Jagielski {
431*b1cdbd2cSJim Jagielski }
432*b1cdbd2cSJim Jagielski 
~ScriptProtocolHandler()433*b1cdbd2cSJim Jagielski ScriptProtocolHandler::~ScriptProtocolHandler()
434*b1cdbd2cSJim Jagielski {
435*b1cdbd2cSJim Jagielski }
436*b1cdbd2cSJim Jagielski 
437*b1cdbd2cSJim Jagielski /* XServiceInfo */
getImplementationName()438*b1cdbd2cSJim Jagielski ::rtl::OUString SAL_CALL ScriptProtocolHandler::getImplementationName( )
439*b1cdbd2cSJim Jagielski throw( RuntimeException )
440*b1cdbd2cSJim Jagielski {
441*b1cdbd2cSJim Jagielski     return impl_getStaticImplementationName();
442*b1cdbd2cSJim Jagielski }
443*b1cdbd2cSJim Jagielski 
444*b1cdbd2cSJim Jagielski /* XServiceInfo */
supportsService(const::rtl::OUString & sServiceName)445*b1cdbd2cSJim Jagielski sal_Bool SAL_CALL ScriptProtocolHandler::supportsService(
446*b1cdbd2cSJim Jagielski const ::rtl::OUString& sServiceName )
447*b1cdbd2cSJim Jagielski throw( RuntimeException )
448*b1cdbd2cSJim Jagielski {
449*b1cdbd2cSJim Jagielski     Sequence< ::rtl::OUString > seqServiceNames = getSupportedServiceNames();
450*b1cdbd2cSJim Jagielski     const ::rtl::OUString* pArray = seqServiceNames.getConstArray();
451*b1cdbd2cSJim Jagielski     for ( sal_Int32 nCounter = 0; nCounter < seqServiceNames.getLength(); nCounter++ )
452*b1cdbd2cSJim Jagielski     {
453*b1cdbd2cSJim Jagielski         if ( pArray[ nCounter ] == sServiceName )
454*b1cdbd2cSJim Jagielski         {
455*b1cdbd2cSJim Jagielski             return sal_True ;
456*b1cdbd2cSJim Jagielski         }
457*b1cdbd2cSJim Jagielski     }
458*b1cdbd2cSJim Jagielski 
459*b1cdbd2cSJim Jagielski     return sal_False ;
460*b1cdbd2cSJim Jagielski }
461*b1cdbd2cSJim Jagielski 
462*b1cdbd2cSJim Jagielski /* XServiceInfo */
getSupportedServiceNames()463*b1cdbd2cSJim Jagielski Sequence< ::rtl::OUString > SAL_CALL ScriptProtocolHandler::getSupportedServiceNames()
464*b1cdbd2cSJim Jagielski throw( RuntimeException )
465*b1cdbd2cSJim Jagielski {
466*b1cdbd2cSJim Jagielski     return impl_getStaticSupportedServiceNames();
467*b1cdbd2cSJim Jagielski }
468*b1cdbd2cSJim Jagielski 
469*b1cdbd2cSJim Jagielski /* Helper for XServiceInfo */
impl_getStaticSupportedServiceNames()470*b1cdbd2cSJim Jagielski Sequence< ::rtl::OUString > ScriptProtocolHandler::impl_getStaticSupportedServiceNames()
471*b1cdbd2cSJim Jagielski {
472*b1cdbd2cSJim Jagielski     ::osl::MutexGuard aGuard( ::osl::Mutex::getGlobalMutex() );
473*b1cdbd2cSJim Jagielski     Sequence< ::rtl::OUString > seqServiceNames( 1 );
474*b1cdbd2cSJim Jagielski     seqServiceNames.getArray() [ 0 ] =
475*b1cdbd2cSJim Jagielski         ::rtl::OUString::createFromAscii( ::scripting_protocolhandler::MYSERVICENAME );
476*b1cdbd2cSJim Jagielski     return seqServiceNames ;
477*b1cdbd2cSJim Jagielski }
478*b1cdbd2cSJim Jagielski 
479*b1cdbd2cSJim Jagielski /* Helper for XServiceInfo */
impl_getStaticImplementationName()480*b1cdbd2cSJim Jagielski ::rtl::OUString ScriptProtocolHandler::impl_getStaticImplementationName()
481*b1cdbd2cSJim Jagielski {
482*b1cdbd2cSJim Jagielski     return ::rtl::OUString::createFromAscii( ::scripting_protocolhandler::MYIMPLNAME );
483*b1cdbd2cSJim Jagielski }
484*b1cdbd2cSJim Jagielski 
485*b1cdbd2cSJim Jagielski /* Helper for registry */
impl_createInstance(const Reference<css::lang::XMultiServiceFactory> & xServiceManager)486*b1cdbd2cSJim Jagielski Reference< XInterface > SAL_CALL ScriptProtocolHandler::impl_createInstance(
487*b1cdbd2cSJim Jagielski const Reference< css::lang::XMultiServiceFactory >& xServiceManager )
488*b1cdbd2cSJim Jagielski throw( RuntimeException )
489*b1cdbd2cSJim Jagielski {
490*b1cdbd2cSJim Jagielski     return Reference< XInterface > ( *new ScriptProtocolHandler( xServiceManager ) );
491*b1cdbd2cSJim Jagielski }
492*b1cdbd2cSJim Jagielski 
493*b1cdbd2cSJim Jagielski /* Factory for registration */
impl_createFactory(const Reference<XMultiServiceFactory> & xServiceManager)494*b1cdbd2cSJim Jagielski Reference< XSingleServiceFactory > ScriptProtocolHandler::impl_createFactory(
495*b1cdbd2cSJim Jagielski const Reference< XMultiServiceFactory >& xServiceManager )
496*b1cdbd2cSJim Jagielski {
497*b1cdbd2cSJim Jagielski     Reference< XSingleServiceFactory > xReturn (
498*b1cdbd2cSJim Jagielski         cppu::createSingleFactory( xServiceManager,
499*b1cdbd2cSJim Jagielski             ScriptProtocolHandler::impl_getStaticImplementationName(),
500*b1cdbd2cSJim Jagielski             ScriptProtocolHandler::impl_createInstance,
501*b1cdbd2cSJim Jagielski             ScriptProtocolHandler::impl_getStaticSupportedServiceNames() )
502*b1cdbd2cSJim Jagielski     );
503*b1cdbd2cSJim Jagielski     return xReturn;
504*b1cdbd2cSJim Jagielski }
505*b1cdbd2cSJim Jagielski 
506*b1cdbd2cSJim Jagielski } // namespace scripting_protocolhandler
507*b1cdbd2cSJim Jagielski 
508*b1cdbd2cSJim Jagielski /* exported functions for registration */
509*b1cdbd2cSJim Jagielski extern "C"
510*b1cdbd2cSJim Jagielski {
511*b1cdbd2cSJim Jagielski 
512*b1cdbd2cSJim Jagielski #undef css
513*b1cdbd2cSJim Jagielski #define css ::com::sun::star
514*b1cdbd2cSJim Jagielski 
component_getImplementationEnvironment(const sal_Char ** ppEnvironmentTypeName,uno_Environment ** ppEnvironment)515*b1cdbd2cSJim Jagielski     void SAL_CALL component_getImplementationEnvironment(
516*b1cdbd2cSJim Jagielski         const sal_Char** ppEnvironmentTypeName, uno_Environment** ppEnvironment )
517*b1cdbd2cSJim Jagielski     {
518*b1cdbd2cSJim Jagielski 		(void)ppEnvironment;
519*b1cdbd2cSJim Jagielski 
520*b1cdbd2cSJim Jagielski         *ppEnvironmentTypeName = CPPU_CURRENT_LANGUAGE_BINDING_NAME ;
521*b1cdbd2cSJim Jagielski     }
522*b1cdbd2cSJim Jagielski 
component_getFactory(const sal_Char * pImplementationName,void * pServiceManager,void * pRegistryKey)523*b1cdbd2cSJim Jagielski     void* SAL_CALL component_getFactory( const sal_Char * pImplementationName ,
524*b1cdbd2cSJim Jagielski                                          void * pServiceManager ,
525*b1cdbd2cSJim Jagielski                                          void * pRegistryKey )
526*b1cdbd2cSJim Jagielski     {
527*b1cdbd2cSJim Jagielski 		(void)pRegistryKey;
528*b1cdbd2cSJim Jagielski 
529*b1cdbd2cSJim Jagielski         // Set default return value for this operation - if it failed.
530*b1cdbd2cSJim Jagielski         void * pReturn = NULL ;
531*b1cdbd2cSJim Jagielski 
532*b1cdbd2cSJim Jagielski         if (
533*b1cdbd2cSJim Jagielski             ( pImplementationName != NULL ) &&
534*b1cdbd2cSJim Jagielski             ( pServiceManager != NULL )
535*b1cdbd2cSJim Jagielski         )
536*b1cdbd2cSJim Jagielski         {
537*b1cdbd2cSJim Jagielski             // Define variables which are used in following macros.
538*b1cdbd2cSJim Jagielski             ::com::sun::star::uno::Reference<
539*b1cdbd2cSJim Jagielski             ::com::sun::star::lang::XSingleServiceFactory > xFactory ;
540*b1cdbd2cSJim Jagielski             ::com::sun::star::uno::Reference< ::com::sun::star::lang::XMultiServiceFactory >
541*b1cdbd2cSJim Jagielski             xServiceManager( reinterpret_cast<
542*b1cdbd2cSJim Jagielski             ::com::sun::star::lang::XMultiServiceFactory* >( pServiceManager ) ) ;
543*b1cdbd2cSJim Jagielski 
544*b1cdbd2cSJim Jagielski             if ( ::scripting_protocolhandler::ScriptProtocolHandler::impl_getStaticImplementationName().equals(
545*b1cdbd2cSJim Jagielski                 ::rtl::OUString::createFromAscii( pImplementationName ) ) )
546*b1cdbd2cSJim Jagielski             {
547*b1cdbd2cSJim Jagielski                 xFactory = ::scripting_protocolhandler::ScriptProtocolHandler::impl_createFactory( xServiceManager );
548*b1cdbd2cSJim Jagielski             }
549*b1cdbd2cSJim Jagielski 
550*b1cdbd2cSJim Jagielski             // Factory is valid - service was found.
551*b1cdbd2cSJim Jagielski             if ( xFactory.is() )
552*b1cdbd2cSJim Jagielski             {
553*b1cdbd2cSJim Jagielski                 xFactory->acquire();
554*b1cdbd2cSJim Jagielski                 pReturn = xFactory.get();
555*b1cdbd2cSJim Jagielski             }
556*b1cdbd2cSJim Jagielski         }
557*b1cdbd2cSJim Jagielski 
558*b1cdbd2cSJim Jagielski         // Return with result of this operation.
559*b1cdbd2cSJim Jagielski         return pReturn ;
560*b1cdbd2cSJim Jagielski     }
561*b1cdbd2cSJim Jagielski } // extern "C"
562*b1cdbd2cSJim Jagielski 
563*b1cdbd2cSJim Jagielski 
564