xref: /trunk/main/sfx2/source/appl/shutdownicon.cxx (revision d119d52d)
1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_sfx2.hxx"
26 
27 #include <shutdownicon.hxx>
28 #include <app.hrc>
29 #include <sfx2/app.hxx>
30 #include <vos/mutex.hxx>
31 #include <svtools/imagemgr.hxx>
32 #include <svtools/miscopt.hxx>
33 // #include <cmdlineargs.hxx>
34 #include <com/sun/star/task/XInteractionHandler.hpp>
35 #include <com/sun/star/frame/XDispatchResultListener.hpp>
36 #include <com/sun/star/frame/XNotifyingDispatch.hpp>
37 #include <com/sun/star/frame/XFramesSupplier.hpp>
38 #include <com/sun/star/frame/XComponentLoader.hpp>
39 #include <com/sun/star/frame/XFrame.hpp>
40 #include <com/sun/star/util/XURLTransformer.hpp>
41 #include <com/sun/star/frame/XFramesSupplier.hpp>
42 #include <com/sun/star/ui/dialogs/XFilePickerControlAccess.hpp>
43 #include <com/sun/star/ui/dialogs/XFilterManager.hpp>
44 #include <com/sun/star/ui/dialogs/ExtendedFilePickerElementIds.hpp>
45 #include <com/sun/star/ui/dialogs/CommonFilePickerElementIds.hpp>
46 #include <com/sun/star/ui/dialogs/ControlActions.hpp>
47 #include <com/sun/star/document/MacroExecMode.hpp>
48 #include <com/sun/star/document/UpdateDocMode.hpp>
49 #include <sfx2/filedlghelper.hxx>
50 #include <sfx2/fcontnr.hxx>
51 #ifndef _UNOTOOLS_PROCESSFACTORY_HXX
52 #include <comphelper/processfactory.hxx>
53 #endif
54 #include <cppuhelper/compbase1.hxx>
55 #include <sfx2/dispatch.hxx>
56 #include <comphelper/extract.hxx>
57 #include <tools/urlobj.hxx>
58 #include <osl/security.hxx>
59 #include <osl/file.hxx>
60 #include <rtl/bootstrap.hxx>
61 #include <rtl/ustrbuf.hxx>
62 #include <tools/link.hxx>
63 #ifdef UNX // need symlink
64 #include <unistd.h>
65 #include <errno.h>
66 #endif
67 #include <vcl/timer.hxx>
68 
69 #include "sfx2/sfxresid.hxx"
70 
71 using namespace ::com::sun::star::uno;
72 using namespace ::com::sun::star::frame;
73 using namespace ::com::sun::star::container;
74 using namespace ::com::sun::star::io;
75 using namespace ::com::sun::star::lang;
76 using namespace ::com::sun::star::beans;
77 using namespace ::com::sun::star::util;
78 using namespace ::com::sun::star::ui::dialogs;
79 using namespace ::vos;
80 #ifdef WNT
81 using ::rtl::OUString;
82 #else
83 using namespace ::rtl;
84 #endif
85 using namespace ::sfx2;
86 
87 #ifdef ENABLE_QUICKSTART_APPLET
88 # if !defined(WIN32) && !defined(QUARTZ)
89 extern "C" { static void SAL_CALL thisModule() {} }
90 # endif
91 #endif
92 
93 #if defined(UNX) && defined(ENABLE_SYSTRAY_GTK)
94 #define PLUGIN_NAME libqstart_gtk.so
95 #endif
96 
97 class SfxNotificationListener_Impl : public cppu::WeakImplHelper1< XDispatchResultListener >
98 {
99 public:
100     virtual void SAL_CALL dispatchFinished( const DispatchResultEvent& aEvent ) throw( RuntimeException );
101     virtual void SAL_CALL disposing( const EventObject& aEvent ) throw( RuntimeException );
102 };
103 
104 void SAL_CALL SfxNotificationListener_Impl::dispatchFinished( const DispatchResultEvent& ) throw( RuntimeException )
105 {
106 	ShutdownIcon::LeaveModalMode();
107 }
108 
109 void SAL_CALL SfxNotificationListener_Impl::disposing( const EventObject& ) throw( RuntimeException )
110 {
111 }
112 
113 SFX_IMPL_XSERVICEINFO( ShutdownIcon, "com.sun.star.office.Quickstart", "com.sun.star.comp.desktop.QuickstartWrapper" )	\
114 SFX_IMPL_ONEINSTANCEFACTORY( ShutdownIcon );
115 
116 bool ShutdownIcon::bModalMode = false;
117 ShutdownIcon* ShutdownIcon::pShutdownIcon = NULL;
118 
119 // To remove conditionals
120 extern "C" {
121 	static void disabled_initSystray() { }
122 	static void disabled_deInitSystray() { }
123 }
124 #define DOSTRING( x )			   			#x
125 #define STRING( x )				   			DOSTRING( x )
126 
127 bool ShutdownIcon::LoadModule( osl::Module **pModule,
128 							   oslGenericFunction *pInit,
129 							   oslGenericFunction *pDeInit )
130 {
131 	if ( pModule )
132 	{
133 		OSL_ASSERT ( pInit && pDeInit );
134 		*pInit = *pDeInit = NULL;
135 		*pModule = NULL;
136 	}
137 
138 #ifdef ENABLE_QUICKSTART_APPLET
139 #  ifdef WIN32
140 	if ( pModule )
141 	{
142 		*pInit = win32_init_sys_tray;
143 		*pDeInit = win32_shutdown_sys_tray;
144 	}
145 	return true;
146 #  elif defined QUARTZ
147     *pInit = aqua_init_systray;
148     *pDeInit = aqua_shutdown_systray;
149     return true;
150 #  else // UNX
151 	osl::Module *pPlugin;
152 	pPlugin = new osl::Module();
153 
154 	oslGenericFunction pTmpInit = NULL;
155 	oslGenericFunction pTmpDeInit = NULL;
156 	if ( pPlugin->loadRelative( &thisModule, OUString( RTL_CONSTASCII_USTRINGPARAM( STRING( PLUGIN_NAME ) ) ) ) )
157 	{
158 		pTmpInit = pPlugin->getFunctionSymbol(
159 			OUString( RTL_CONSTASCII_USTRINGPARAM( "plugin_init_sys_tray" ) ) );
160 		pTmpDeInit = pPlugin->getFunctionSymbol(
161 			OUString( RTL_CONSTASCII_USTRINGPARAM( "plugin_shutdown_sys_tray" ) ) );
162 	}
163 	if ( !pTmpInit || !pTmpDeInit )
164 	{
165 		delete pPlugin;
166 		pPlugin = NULL;
167 	}
168 	if ( pModule )
169 	{
170 		*pModule = pPlugin;
171 		*pInit = pTmpInit;
172 		*pDeInit = pTmpDeInit;
173 	}
174 	else
175 	{
176 		bool bRet = pPlugin != NULL;
177 		delete pPlugin;
178 		return bRet;
179 	}
180 #  endif // UNX
181 #endif // ENABLE_QUICKSTART_APPLET
182 	if ( pModule )
183 	{
184 		if ( !*pInit )
185 			*pInit = disabled_initSystray;
186 		if ( !*pDeInit )
187 			*pDeInit = disabled_deInitSystray;
188 	}
189 
190 	return true;
191 }
192 
193 class IdleUnloader : Timer
194 {
195     ::osl::Module *m_pModule;
196 public:
197     IdleUnloader (::osl::Module **pModule) :
198         m_pModule (*pModule)
199     {
200         *pModule = NULL;
201         Start();
202     }
203     virtual void Timeout()
204     {
205         delete m_pModule;
206         delete this;
207     }
208 };
209 
210 void ShutdownIcon::initSystray()
211 {
212 	if (m_bInitialized)
213 		return;
214 	m_bInitialized = true;
215 
216 	(void) LoadModule( &m_pPlugin, &m_pInitSystray, &m_pDeInitSystray );
217 	m_bVeto = true;
218 	m_pInitSystray();
219 }
220 
221 void ShutdownIcon::deInitSystray()
222 {
223 	if (!m_bInitialized)
224 		return;
225 
226     if (m_pDeInitSystray)
227 		m_pDeInitSystray();
228 
229 	m_bVeto = false;
230 	m_pInitSystray = 0;
231 	m_pDeInitSystray = 0;
232     new IdleUnloader (&m_pPlugin);
233 
234     delete m_pFileDlg;
235 	m_pFileDlg = NULL;
236 	m_bInitialized = false;
237 }
238 
239 
240 ShutdownIcon::ShutdownIcon( Reference< XMultiServiceFactory > aSMgr ) :
241 	ShutdownIconServiceBase( m_aMutex ),
242 	m_bVeto ( false ),
243     m_bListenForTermination ( false ),
244     m_bSystemDialogs( false ),
245 	m_pResMgr( NULL ),
246     m_pFileDlg( NULL ),
247 	m_xServiceManager( aSMgr ),
248 	m_pInitSystray( 0 ),
249 	m_pDeInitSystray( 0 ),
250 	m_pPlugin( 0 ),
251 	m_bInitialized( false )
252 {
253     m_bSystemDialogs = SvtMiscOptions().UseSystemFileDialog();
254 }
255 
256 ShutdownIcon::~ShutdownIcon()
257 {
258 	deInitSystray();
259     new IdleUnloader (&m_pPlugin);
260 }
261 
262 // ---------------------------------------------------------------------------
263 
264 void ShutdownIcon::OpenURL( const ::rtl::OUString& aURL, const ::rtl::OUString& rTarget, const Sequence< PropertyValue >& aArgs )
265 {
266     if ( getInstance() && getInstance()->m_xDesktop.is() )
267     {
268         Reference < XDispatchProvider > xDispatchProvider( getInstance()->m_xDesktop, UNO_QUERY );
269         if ( xDispatchProvider.is() )
270         {
271             com::sun::star::util::URL aDispatchURL;
272             aDispatchURL.Complete = aURL;
273 
274             Reference < com::sun::star::util::XURLTransformer > xURLTransformer(
275                 ::comphelper::getProcessServiceFactory()->createInstance( OUString::createFromAscii("com.sun.star.util.URLTransformer") ),
276                 com::sun::star::uno::UNO_QUERY );
277             if ( xURLTransformer.is() )
278             {
279                 try
280                 {
281                     Reference< com::sun::star::frame::XDispatch > xDispatch;
282 
283                     xURLTransformer->parseStrict( aDispatchURL );
284                     xDispatch = xDispatchProvider->queryDispatch( aDispatchURL, rTarget, 0 );
285                     if ( xDispatch.is() )
286                         xDispatch->dispatch( aDispatchURL, aArgs );
287                 }
288                 catch ( com::sun::star::uno::RuntimeException& )
289                 {
290                     throw;
291                 }
292                 catch ( com::sun::star::uno::Exception& )
293                 {
294                 }
295             }
296         }
297     }
298 }
299 
300 // ---------------------------------------------------------------------------
301 
302 void ShutdownIcon::FileOpen()
303 {
304     if ( getInstance() && getInstance()->m_xDesktop.is() )
305     {
306         ::vos::OGuard aGuard( Application::GetSolarMutex() );
307 		EnterModalMode();
308         getInstance()->StartFileDialog();
309     }
310 }
311 
312 // ---------------------------------------------------------------------------
313 
314 void ShutdownIcon::FromTemplate()
315 {
316     if ( getInstance() && getInstance()->m_xDesktop.is() )
317     {
318         Reference < ::com::sun::star::frame::XFramesSupplier > xDesktop ( getInstance()->m_xDesktop, UNO_QUERY);
319         Reference < ::com::sun::star::frame::XFrame > xFrame( xDesktop->getActiveFrame() );
320         if ( !xFrame.is() )
321             xFrame = Reference < ::com::sun::star::frame::XFrame >( xDesktop, UNO_QUERY );
322 
323         URL aTargetURL;
324         aTargetURL.Complete = OUString( RTL_CONSTASCII_USTRINGPARAM( "slot:5500" ) );
325         Reference < XURLTransformer > xTrans( ::comphelper::getProcessServiceFactory()->createInstance( rtl::OUString::createFromAscii("com.sun.star.util.URLTransformer" )), UNO_QUERY );
326         xTrans->parseStrict( aTargetURL );
327 
328         Reference < ::com::sun::star::frame::XDispatchProvider > xProv( xFrame, UNO_QUERY );
329         Reference < ::com::sun::star::frame::XDispatch > xDisp;
330 	    if ( xProv.is() )
331 	    {
332             if ( aTargetURL.Protocol.compareToAscii("slot:") == COMPARE_EQUAL )
333                 xDisp = xProv->queryDispatch( aTargetURL, ::rtl::OUString(), 0 );
334             else
335                 xDisp = xProv->queryDispatch( aTargetURL, ::rtl::OUString::createFromAscii("_blank"), 0 );
336 	    }
337         if ( xDisp.is() )
338 	    {
339 		    Sequence<PropertyValue> aArgs(1);
340 		    PropertyValue* pArg = aArgs.getArray();
341 		    pArg[0].Name = rtl::OUString::createFromAscii("Referer");
342             pArg[0].Value <<= ::rtl::OUString::createFromAscii("private:user");
343             Reference< ::com::sun::star::frame::XNotifyingDispatch > xNotifyer( xDisp, UNO_QUERY );
344             if ( xNotifyer.is() )
345 			{
346 				EnterModalMode();
347                 xNotifyer->dispatchWithNotification( aTargetURL, aArgs, new SfxNotificationListener_Impl() );
348 			}
349             else
350                 xDisp->dispatch( aTargetURL, aArgs );
351 	    }
352     }
353 }
354 
355 // ---------------------------------------------------------------------------
356 #include <tools/rcid.h>
357 OUString ShutdownIcon::GetResString( int id )
358 {
359     ::vos::OGuard aGuard( Application::GetSolarMutex() );
360 
361     if( ! m_pResMgr )
362         m_pResMgr = SfxResId::GetResMgr();
363 	ResId aResId( id, *m_pResMgr );
364 	aResId.SetRT( RSC_STRING );
365 	if( !m_pResMgr || !m_pResMgr->IsAvailable( aResId ) )
366         return OUString();
367 
368     UniString aRes( ResId(id, *m_pResMgr) );
369     return OUString( aRes );
370 }
371 
372 // ---------------------------------------------------------------------------
373 
374 OUString ShutdownIcon::GetUrlDescription( const OUString& aUrl )
375 {
376     ::vos::OGuard aGuard( Application::GetSolarMutex() );
377 
378     return OUString( SvFileInformationManager::GetDescription( INetURLObject( aUrl ) ) );
379 }
380 
381 // ---------------------------------------------------------------------------
382 
383 void ShutdownIcon::StartFileDialog()
384 {
385     ::vos::OGuard aGuard( Application::GetSolarMutex() );
386 
387     bool bDirty = ( m_bSystemDialogs != static_cast<bool>(SvtMiscOptions().UseSystemFileDialog()) );
388 
389     if ( m_pFileDlg && bDirty )
390     {
391         // Destroy instance as changing the system file dialog setting
392         // forces us to create a new FileDialogHelper instance!
393         delete m_pFileDlg;
394         m_pFileDlg = NULL;
395     }
396 
397     if ( !m_pFileDlg )
398         m_pFileDlg = new FileDialogHelper( WB_OPEN | SFXWB_MULTISELECTION, String() );
399     m_pFileDlg->StartExecuteModal( STATIC_LINK( this, ShutdownIcon, DialogClosedHdl_Impl ) );
400 }
401 
402 // ---------------------------------------------------------------------------
403 
404 IMPL_STATIC_LINK( ShutdownIcon, DialogClosedHdl_Impl, FileDialogHelper*, EMPTYARG )
405 {
406     DBG_ASSERT( pThis->m_pFileDlg, "ShutdownIcon, DialogClosedHdl_Impl(): no file dialog" );
407 
408     // use ctor for filling up filters automatically! #89169#
409     if ( ERRCODE_NONE == pThis->m_pFileDlg->GetError() )
410     {
411         Reference< XFilePicker >    xPicker = pThis->m_pFileDlg->GetFilePicker();
412 
413         try
414         {
415 
416             if ( xPicker.is() )
417             {
418 
419                 Reference < XFilePickerControlAccess > xPickerControls ( xPicker, UNO_QUERY );
420                 Reference < XFilterManager > xFilterManager ( xPicker, UNO_QUERY );
421 
422                 Sequence< OUString >        sFiles = xPicker->getFiles();
423                 int                         nFiles = sFiles.getLength();
424 
425                 int                         nArgs=3;
426                 Sequence< PropertyValue >   aArgs(3);
427 
428                 Reference < com::sun::star::task::XInteractionHandler > xInteraction(
429                     ::comphelper::getProcessServiceFactory()->createInstance( OUString::createFromAscii("com.sun.star.task.InteractionHandler") ),
430                     com::sun::star::uno::UNO_QUERY );
431 
432                 aArgs[0].Name = OUString::createFromAscii( "InteractionHandler" );
433                 aArgs[0].Value <<= xInteraction;
434 
435                 sal_Int16 nMacroExecMode = ::com::sun::star::document::MacroExecMode::USE_CONFIG;
436                 aArgs[1].Name = OUString::createFromAscii( "MacroExecutionMode" );
437                 aArgs[1].Value <<= nMacroExecMode;
438 
439                 sal_Int16 nUpdateDoc = ::com::sun::star::document::UpdateDocMode::ACCORDING_TO_CONFIG;
440                 aArgs[2].Name = OUString::createFromAscii( "UpdateDocMode" );
441                 aArgs[2].Value <<= nUpdateDoc;
442 
443                 // pb: #102643# use the filedlghelper to get the current filter name,
444                 // because it removes the extensions before you get the filter name.
445                 OUString aFilterName( pThis->m_pFileDlg->GetCurrentFilter() );
446 
447                 if ( xPickerControls.is() )
448                 {
449 
450                     // Set readonly flag
451 
452                     sal_Bool    bReadOnly = sal_False;
453 
454 
455                     xPickerControls->getValue( ExtendedFilePickerElementIds::CHECKBOX_READONLY, 0 ) >>= bReadOnly;
456 
457                     // #95239#: Only set porperty if readonly is set to TRUE
458 
459                     if ( bReadOnly )
460                     {
461                         aArgs.realloc( ++nArgs );
462                         aArgs[nArgs-1].Name  = OUString::createFromAscii( "ReadOnly" );
463                         aArgs[nArgs-1].Value <<= bReadOnly;
464                     }
465 
466                     // Get version string
467 
468                     sal_Int32   iVersion = -1;
469 
470                     xPickerControls->getValue( ExtendedFilePickerElementIds::LISTBOX_VERSION, ControlActions::GET_SELECTED_ITEM_INDEX ) >>= iVersion;
471 
472                     if ( iVersion >= 0 )
473                     {
474                         sal_Int16   uVersion = (sal_Int16)iVersion;
475 
476                         aArgs.realloc( ++nArgs );
477                         aArgs[nArgs-1].Name  = OUString::createFromAscii( "Version" );
478                         aArgs[nArgs-1].Value <<= uVersion;
479                     }
480 
481                     // Retrieve the current filter
482 
483                     if ( !aFilterName.getLength() )
484                         xPickerControls->getValue( CommonFilePickerElementIds::LISTBOX_FILTER, ControlActions::GET_SELECTED_ITEM ) >>= aFilterName;
485 
486                 }
487 
488 
489                 // Convert UI filter name to internal filter name
490 
491                 if ( aFilterName.getLength() )
492                 {
493                     const SfxFilter* pFilter = SFX_APP()->GetFilterMatcher().GetFilter4UIName( aFilterName, 0, SFX_FILTER_NOTINFILEDLG );
494 
495                     if ( pFilter )
496                     {
497                         aFilterName = pFilter->GetFilterName();
498 
499                         if ( aFilterName.getLength() )
500                         {
501                             aArgs.realloc( ++nArgs );
502                             aArgs[nArgs-1].Name  = OUString::createFromAscii( "FilterName" );
503                             aArgs[nArgs-1].Value <<= aFilterName;
504                         }
505                     }
506                 }
507 
508                 if ( 1 == nFiles )
509                     OpenURL( sFiles[0], OUString( RTL_CONSTASCII_USTRINGPARAM( "_default" ) ), aArgs );
510                 else
511                 {
512                     OUString    aBaseDirURL = sFiles[0];
513                     if ( aBaseDirURL.getLength() > 0 && aBaseDirURL[aBaseDirURL.getLength()-1] != '/' )
514                         aBaseDirURL += OUString::createFromAscii("/");
515 
516                     int iFiles;
517                     for ( iFiles = 1; iFiles < nFiles; iFiles++ )
518                     {
519                         OUString    aURL = aBaseDirURL;
520                         aURL += sFiles[iFiles];
521                         OpenURL( aURL, OUString( RTL_CONSTASCII_USTRINGPARAM( "_default" ) ), aArgs );
522                     }
523                 }
524             }
525         }
526         catch ( ... )
527         {
528         }
529     }
530 
531 #ifdef WNT
532     // #103346 Destroy dialog to prevent problems with custom controls
533     // This fix is dependent on the dialog settings. Destroying the dialog here will
534     // crash the non-native dialog implementation! Therefore make this dependent on
535     // the settings.
536     if ( SvtMiscOptions().UseSystemFileDialog() )
537     {
538         delete pThis->m_pFileDlg;
539         pThis->m_pFileDlg = NULL;
540     }
541 #endif
542 
543     LeaveModalMode();
544     return 0;
545 }
546 
547 // ---------------------------------------------------------------------------
548 
549 void ShutdownIcon::addTerminateListener()
550 {
551     ShutdownIcon* pInst = getInstance();
552     if ( ! pInst)
553         return;
554 
555     if (pInst->m_bListenForTermination)
556         return;
557 
558     Reference< XDesktop > xDesktop = pInst->m_xDesktop;
559     if ( ! xDesktop.is())
560         return;
561 
562 	xDesktop->addTerminateListener( pInst );
563     pInst->m_bListenForTermination = true;
564 }
565 
566 // ---------------------------------------------------------------------------
567 
568 void ShutdownIcon::terminateDesktop()
569 {
570     ShutdownIcon* pInst = getInstance();
571     if ( ! pInst)
572         return;
573 
574     Reference< XDesktop > xDesktop = pInst->m_xDesktop;
575     if ( ! xDesktop.is())
576         return;
577 
578     // always remove ourselves as listener
579     pInst->m_bListenForTermination = true;
580     xDesktop->removeTerminateListener( pInst );
581 
582     // terminate desktop only if no tasks exist
583     Reference< XFramesSupplier > xSupplier( xDesktop, UNO_QUERY );
584     if ( xSupplier.is() )
585     {
586         Reference< XIndexAccess > xTasks ( xSupplier->getFrames(), UNO_QUERY );
587         if( xTasks.is() )
588         {
589             if( xTasks->getCount() < 1 )
590                 xDesktop->terminate();
591         }
592     }
593 
594     // remove the instance pointer
595     ShutdownIcon::pShutdownIcon = 0;
596 }
597 
598 // ---------------------------------------------------------------------------
599 
600 ShutdownIcon* ShutdownIcon::getInstance()
601 {
602 	OSL_ASSERT( pShutdownIcon );
603 	return pShutdownIcon;
604 }
605 
606 // ---------------------------------------------------------------------------
607 
608 ShutdownIcon* ShutdownIcon::createInstance()
609 {
610 	if (pShutdownIcon)
611         return pShutdownIcon;
612 
613 	ShutdownIcon *pIcon = NULL;
614 	try {
615 		Reference< XMultiServiceFactory > xSMgr( comphelper::getProcessServiceFactory() );
616 		pIcon = new ShutdownIcon( xSMgr );
617 		pIcon->init ();
618 		pShutdownIcon = pIcon;
619 	} catch (...) {
620 		delete pIcon;
621 	}
622 
623 	return pShutdownIcon;
624 }
625 
626 void ShutdownIcon::init() throw( ::com::sun::star::uno::Exception )
627 {
628 	// access resource system and sfx only protected by solarmutex
629 	vos::OGuard aSolarGuard( Application::GetSolarMutex() );
630 	ResMgr *pResMgr = SfxResId::GetResMgr();
631 
632 	::osl::ResettableMutexGuard	aGuard(	m_aMutex );
633 	m_pResMgr = pResMgr;
634 	aGuard.clear();
635 	Reference < XDesktop > xDesktop( m_xServiceManager->createInstance(
636 											 DEFINE_CONST_UNICODE( "com.sun.star.frame.Desktop" )),
637 									 UNO_QUERY );
638 	aGuard.reset();
639 	m_xDesktop = xDesktop;
640 }
641 
642 // ---------------------------------------------------------------------------
643 
644 void SAL_CALL ShutdownIcon::disposing()
645 {
646 	m_xServiceManager = Reference< XMultiServiceFactory >();
647 	m_xDesktop = Reference< XDesktop >();
648 }
649 
650 // ---------------------------------------------------------------------------
651 
652 // XEventListener
653 void SAL_CALL ShutdownIcon::disposing( const ::com::sun::star::lang::EventObject& )
654 	throw(::com::sun::star::uno::RuntimeException)
655 {
656 }
657 
658 // ---------------------------------------------------------------------------
659 
660 // XTerminateListener
661 void SAL_CALL ShutdownIcon::queryTermination( const ::com::sun::star::lang::EventObject& )
662 throw(::com::sun::star::frame::TerminationVetoException, ::com::sun::star::uno::RuntimeException)
663 {
664 	::osl::ClearableMutexGuard	aGuard(	m_aMutex );
665 
666 	if ( m_bVeto )
667 		throw ::com::sun::star::frame::TerminationVetoException();
668 }
669 
670 
671 // ---------------------------------------------------------------------------
672 
673 void SAL_CALL ShutdownIcon::notifyTermination( const ::com::sun::star::lang::EventObject& )
674 throw(::com::sun::star::uno::RuntimeException)
675 {
676 }
677 
678 
679 // ---------------------------------------------------------------------------
680 
681 void SAL_CALL ShutdownIcon::initialize( const ::com::sun::star::uno::Sequence< ::com::sun::star::uno::Any>& aArguments )
682 	throw( ::com::sun::star::uno::Exception )
683 {
684 	::osl::ResettableMutexGuard	aGuard(	m_aMutex );
685 
686     // third argument only sets veto, everything else will be ignored!
687     if (aArguments.getLength() > 2)
688     {
689         sal_Bool bVeto = sal_True;
690         bVeto = ::cppu::any2bool(aArguments[2]);
691         m_bVeto = bVeto;
692         return;
693     }
694 
695 	if ( aArguments.getLength() > 0 )
696 	{
697 		if ( !ShutdownIcon::pShutdownIcon )
698 		{
699 			try
700 			{
701 				sal_Bool bQuickstart = sal_False;
702 				bQuickstart = ::cppu::any2bool( aArguments[0] );
703 				if( !bQuickstart && !GetAutostart() )
704 					return;
705                 aGuard.clear();
706 				init ();
707                 aGuard.reset();
708 				if ( !m_xDesktop.is() )
709 					return;
710 
711 				/* Create a sub-classed instance - foo */
712 				ShutdownIcon::pShutdownIcon = this;
713 				initSystray();
714 #ifdef OS2
715 				// above win32 starts the quickstart thread, but we have
716 				// quickstart running only when -quickstart is specified
717 				// on command line (next boot).
718 				// so if -quickstart was not specified, we cannot issue
719 				// quickstart veto on shutdown.
720 				if (bQuickstart)
721 				{
722 					// disable shutdown
723 					ShutdownIcon::getInstance()->SetVeto( true );
724 					ShutdownIcon::getInstance()->addTerminateListener();
725 				}
726 #endif
727 			}
728 			catch(const ::com::sun::star::lang::IllegalArgumentException&)
729 			{
730 			}
731 		}
732 	}
733     if ( aArguments.getLength() > 1 )
734     {
735 			sal_Bool bAutostart = sal_False;
736 			bAutostart = ::cppu::any2bool( aArguments[1] );
737             if (bAutostart && !GetAutostart())
738                 SetAutostart( sal_True );
739             if (!bAutostart && GetAutostart())
740                 SetAutostart( sal_False );
741     }
742 
743 }
744 
745 // -------------------------------
746 
747 void ShutdownIcon::EnterModalMode()
748 {
749 	bModalMode = sal_True;
750 }
751 
752 // -------------------------------
753 
754 void ShutdownIcon::LeaveModalMode()
755 {
756 	bModalMode = sal_False;
757 }
758 
759 #ifdef WNT
760 // defined in shutdowniconw32.cxx
761 #elif defined(OS2)
762 // defined in shutdowniconOs2.cxx
763 #elif defined QUARTZ
764 // defined in shutdowniconaqua.cxx
765 #else
766 bool ShutdownIcon::IsQuickstarterInstalled()
767 {
768 #ifndef ENABLE_QUICKSTART_APPLET
769 	return false;
770 #else // !ENABLE_QUICKSTART_APPLET
771 #ifdef UNX
772 	return LoadModule( NULL, NULL, NULL);
773 #endif // UNX
774 #endif // !ENABLE_QUICKSTART_APPLET
775 }
776 #endif // !WNT
777 
778 // ---------------------------------------------------------------------------
779 
780 #if defined (ENABLE_QUICKSTART_APPLET) && defined (UNX)
781 static OUString getDotAutostart( bool bCreate = false )
782 {
783 	OUString aShortcut;
784 	const char *pConfigHome;
785 	if( (pConfigHome = getenv("XDG_CONFIG_HOME") ) )
786 		aShortcut = OStringToOUString( OString( pConfigHome ), RTL_TEXTENCODING_UTF8 );
787 	else
788 	{
789 		OUString aHomeURL;
790 		osl::Security().getHomeDir( aHomeURL );
791 		::osl::File::getSystemPathFromFileURL( aHomeURL, aShortcut );
792 		aShortcut += OUString( RTL_CONSTASCII_USTRINGPARAM( "/.config" ) );
793 	}
794 	aShortcut += OUString( RTL_CONSTASCII_USTRINGPARAM( "/autostart" ) );
795 	if (bCreate)
796 	{
797 		OUString aShortcutUrl;
798 		osl::File::getFileURLFromSystemPath( aShortcut, aShortcutUrl );
799 		osl::Directory::createPath( aShortcutUrl );
800 	}
801 	return aShortcut;
802 }
803 #endif
804 
805 rtl::OUString ShutdownIcon::getShortcutName()
806 {
807 #ifndef ENABLE_QUICKSTART_APPLET
808 	return OUString();
809 #else
810 
811     OUString aShortcutName( RTL_CONSTASCII_USTRINGPARAM( "StarOffice 6.0" ) );
812 	ResMgr* pMgr = SfxResId::GetResMgr();
813     if( pMgr )
814     {
815         ::vos::OGuard aGuard( Application::GetSolarMutex() );
816         UniString aRes( SfxResId( STR_QUICKSTART_LNKNAME ) );
817         aShortcutName = OUString( aRes );
818     }
819 #ifdef WNT
820     aShortcutName += OUString( RTL_CONSTASCII_USTRINGPARAM( ".lnk" ) );
821 
822 	OUString aShortcut(GetAutostartFolderNameW32());
823 	aShortcut += OUString( RTL_CONSTASCII_USTRINGPARAM( "\\" ) );
824 	aShortcut += aShortcutName;
825 #else // UNX
826     OUStringBuffer aStrBuff( getDotAutostart() );
827     aStrBuff.appendAscii( RTL_CONSTASCII_STRINGPARAM( "/" ) );
828     if ( sal_Int32 len = aShortcutName.getLength() )
829         aStrBuff.append( aShortcutName.getStr(), len );
830     else
831         aStrBuff.appendAscii( RTL_CONSTASCII_STRINGPARAM( "qstart" ) );
832     aStrBuff.appendAscii( RTL_CONSTASCII_STRINGPARAM( ".desktop" ) );
833 
834     OUString aShortcut( aStrBuff.makeStringAndClear() );
835 #endif // UNX
836 	return aShortcut;
837 #endif // ENABLE_QUICKSTART_APPLET
838 }
839 
840 bool ShutdownIcon::GetAutostart( )
841 {
842 #if defined(OS2)
843     return GetAutostartOs2( );
844 #elif defined QUARTZ
845     return true;
846 #else
847 	bool bRet = false;
848 #ifdef ENABLE_QUICKSTART_APPLET
849 	OUString aShortcut( getShortcutName() );
850 	OUString aShortcutUrl;
851 	osl::File::getFileURLFromSystemPath( aShortcut, aShortcutUrl );
852 	osl::File f( aShortcutUrl );
853 	osl::File::RC error = f.open( OpenFlag_Read );
854 	if( error == osl::File::E_None )
855 	{
856 		f.close();
857 		bRet = true;
858 	}
859 #endif // ENABLE_QUICKSTART_APPLET
860     return bRet;
861 #endif
862 }
863 
864 void ShutdownIcon::SetAutostart( bool bActivate )
865 {
866 #ifdef ENABLE_QUICKSTART_APPLET
867 	OUString aShortcut( getShortcutName() );
868 
869     if( bActivate && IsQuickstarterInstalled() )
870     {
871 #ifdef WNT
872 		EnableAutostartW32( aShortcut );
873 #else // UNX
874 		getDotAutostart( true );
875 
876 		OUString aPath( RTL_CONSTASCII_USTRINGPARAM("${BRAND_BASE_DIR}/share/xdg/qstart.desktop" ) );
877 		Bootstrap::expandMacros( aPath );
878 
879 		OUString aDesktopFile;
880 		::osl::File::getSystemPathFromFileURL( aPath, aDesktopFile );
881 
882 		OString aDesktopFileUnx = OUStringToOString( aDesktopFile,
883 													 osl_getThreadTextEncoding() );
884 		OString aShortcutUnx = OUStringToOString( aShortcut,
885 												  osl_getThreadTextEncoding() );
886 		if ((0 != symlink(aDesktopFileUnx, aShortcutUnx)) && (errno == EEXIST))
887 		{
888 		unlink(aShortcutUnx);
889 		symlink(aDesktopFileUnx, aShortcutUnx);
890 		}
891 
892 		ShutdownIcon *pIcon = ShutdownIcon::createInstance();
893 		if( pIcon )
894 			pIcon->initSystray();
895 #endif // UNX
896     }
897     else
898     {
899         OUString aShortcutUrl;
900         ::osl::File::getFileURLFromSystemPath( aShortcut, aShortcutUrl );
901         ::osl::File::remove( aShortcutUrl );
902 #ifdef UNX
903 		if (pShutdownIcon)
904 		{
905 		    ShutdownIcon *pIcon = getInstance();
906 			pIcon->deInitSystray();
907 		}
908 #endif
909     }
910 #elif defined OS2
911     SetAutostartOs2( bActivate );
912 #else
913     (void)bActivate; // unused variable
914 #endif // ENABLE_QUICKSTART_APPLET
915 }
916 
917 static const ::sal_Int32 PROPHANDLE_TERMINATEVETOSTATE = 0;
918 
919 // XFastPropertySet
920 void SAL_CALL ShutdownIcon::setFastPropertyValue(       ::sal_Int32                  nHandle,
921                                                   const ::com::sun::star::uno::Any& aValue )
922     throw (::com::sun::star::beans::UnknownPropertyException,
923             ::com::sun::star::beans::PropertyVetoException,
924             ::com::sun::star::lang::IllegalArgumentException,
925             ::com::sun::star::lang::WrappedTargetException,
926             ::com::sun::star::uno::RuntimeException)
927 {
928     switch(nHandle)
929     {
930         case PROPHANDLE_TERMINATEVETOSTATE :
931              {
932                 // use new value in case it's a valid information only
933                 ::sal_Bool bState( sal_False );
934                 if (! (aValue >>= bState))
935                     return;
936 
937                 m_bVeto = bState;
938                 if (m_bVeto && ! m_bListenForTermination)
939                     addTerminateListener();
940              }
941              break;
942 
943         default :
944             throw ::com::sun::star::beans::UnknownPropertyException();
945     }
946 }
947 
948 // XFastPropertySet
949 ::com::sun::star::uno::Any SAL_CALL ShutdownIcon::getFastPropertyValue( ::sal_Int32 nHandle )
950     throw (::com::sun::star::beans::UnknownPropertyException,
951             ::com::sun::star::lang::WrappedTargetException,
952             ::com::sun::star::uno::RuntimeException)
953 {
954     ::com::sun::star::uno::Any aValue;
955     switch(nHandle)
956     {
957         case PROPHANDLE_TERMINATEVETOSTATE :
958              {
959                 bool bState   = (m_bListenForTermination && m_bVeto);
960                      aValue <<= bState;
961              }
962              break;
963 
964         default :
965             throw ::com::sun::star::beans::UnknownPropertyException();
966     }
967 
968     return aValue;
969 }
970