/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef UNX // need symlink #include #include #endif #include #include using namespace ::com::sun::star; using namespace ::com::sun::star::uno; using namespace ::com::sun::star::frame; using namespace ::com::sun::star::container; using namespace ::com::sun::star::io; using namespace ::com::sun::star::lang; using namespace ::com::sun::star::beans; using namespace ::com::sun::star::util; using namespace ::com::sun::star::ui::dialogs; using namespace ::sfx2; #ifdef ENABLE_QUICKSTART_APPLET # if !defined(WIN32) && !defined(MACOSX) extern "C" { static void SAL_CALL thisModule() {} } # endif #endif #if defined(UNX) && defined(ENABLE_SYSTRAY_GTK) && !defined(PLUGIN_NAME) #define PLUGIN_NAME "libqstart_gtklo.so" #endif class SfxNotificationListener_Impl : public cppu::WeakImplHelper1< XDispatchResultListener > { public: virtual void SAL_CALL dispatchFinished( const DispatchResultEvent& aEvent ) throw( RuntimeException, std::exception ) SAL_OVERRIDE; virtual void SAL_CALL disposing( const EventObject& aEvent ) throw( RuntimeException, std::exception ) SAL_OVERRIDE; }; void SAL_CALL SfxNotificationListener_Impl::dispatchFinished( const DispatchResultEvent& ) throw( RuntimeException, std::exception ) { ShutdownIcon::LeaveModalMode(); } void SAL_CALL SfxNotificationListener_Impl::disposing( const EventObject& ) throw( RuntimeException, std::exception ) { } OUString SAL_CALL ShutdownIcon::getImplementationName() throw (css::uno::RuntimeException, std::exception) { return OUString("com.sun.star.comp.desktop.QuickstartWrapper"); } sal_Bool SAL_CALL ShutdownIcon::supportsService(OUString const & ServiceName) throw (css::uno::RuntimeException, std::exception) { return cppu::supportsService(this, ServiceName); } css::uno::Sequence SAL_CALL ShutdownIcon::getSupportedServiceNames() throw (css::uno::RuntimeException, std::exception) { css::uno::Sequence< OUString > aSeq(1); aSeq[0] = OUString("com.sun.star.office.Quickstart"); return aSeq; } bool ShutdownIcon::bModalMode = false; ShutdownIcon* ShutdownIcon::pShutdownIcon = NULL; #if !defined( ENABLE_QUICKSTART_APPLET ) || defined( UNX ) // To remove conditionals extern "C" { static void disabled_initSystray() { } static void disabled_deInitSystray() { } } #endif bool ShutdownIcon::LoadModule( osl::Module **pModule, oslGenericFunction *pInit, oslGenericFunction *pDeInit ) { if ( pModule ) { OSL_ASSERT ( pInit && pDeInit ); *pInit = *pDeInit = NULL; *pModule = NULL; } #ifdef ENABLE_QUICKSTART_APPLET # ifdef WIN32 if ( pModule ) { *pInit = win32_init_sys_tray; *pDeInit = win32_shutdown_sys_tray; } return true; # elif defined MACOSX *pInit = aqua_init_systray; *pDeInit = aqua_shutdown_systray; return true; # else // UNX osl::Module *pPlugin; pPlugin = new osl::Module(); oslGenericFunction pTmpInit = NULL; oslGenericFunction pTmpDeInit = NULL; #define DOSTRING( x ) #x #define STRING( x ) DOSTRING( x ) if ( pPlugin->loadRelative( &thisModule, OUString (STRING( PLUGIN_NAME ) ) ) ) { pTmpInit = pPlugin->getFunctionSymbol( OUString( "plugin_init_sys_tray" ) ); pTmpDeInit = pPlugin->getFunctionSymbol( OUString( "plugin_shutdown_sys_tray" ) ); } if ( !pTmpInit || !pTmpDeInit ) { delete pPlugin; pPlugin = NULL; } if ( pModule ) { *pModule = pPlugin; *pInit = pTmpInit; *pDeInit = pTmpDeInit; } else { bool bRet = pPlugin != NULL; delete pPlugin; return bRet; } # endif // UNX #endif // ENABLE_QUICKSTART_APPLET #if !defined( ENABLE_QUICKSTART_APPLET ) || defined( UNX ) // Avoid unreachable code. In the ENABLE_QUICKSTART_APPLET && !UNX // case, we have already returned. if ( pModule ) { if ( !*pInit ) *pInit = disabled_initSystray; if ( !*pDeInit ) *pDeInit = disabled_deInitSystray; } return true; #endif // !ENABLE_QUICKSTART_APPLET || UNX } // These two timeouts are necessary to avoid there being // plugin frames still on the stack, after unloading that // code, causing a crash during disabling / termination. class IdleUnloader : Timer { ::osl::Module *m_pModule; public: IdleUnloader (::osl::Module **pModule) : m_pModule (*pModule) { *pModule = NULL; Start(); } virtual void Timeout() SAL_OVERRIDE { delete m_pModule; delete this; } }; class IdleTerminate : Timer { ::com::sun::star::uno::Reference< XDesktop2 > m_xDesktop; public: IdleTerminate (::com::sun::star::uno::Reference< XDesktop2 > xDesktop) { m_xDesktop = xDesktop; Start(); } virtual void Timeout() SAL_OVERRIDE { m_xDesktop->terminate(); delete this; } }; void ShutdownIcon::initSystray() { if (m_bInitialized) return; m_bInitialized = true; (void) LoadModule( &m_pPlugin, &m_pInitSystray, &m_pDeInitSystray ); m_bVeto = true; m_pInitSystray(); } void ShutdownIcon::deInitSystray() { if (!m_bInitialized) return; if (m_pDeInitSystray) m_pDeInitSystray(); m_bVeto = false; m_pInitSystray = 0; m_pDeInitSystray = 0; new IdleUnloader (&m_pPlugin); delete m_pFileDlg; m_pFileDlg = NULL; m_bInitialized = false; } ShutdownIcon::ShutdownIcon( const ::com::sun::star::uno::Reference< XComponentContext > & rxContext ) : ShutdownIconServiceBase( m_aMutex ), m_bVeto ( false ), m_bListenForTermination ( false ), m_bSystemDialogs( false ), m_pResMgr( NULL ), m_pFileDlg( NULL ), m_xContext( rxContext ), m_pInitSystray( 0 ), m_pDeInitSystray( 0 ), m_pPlugin( 0 ), m_bInitialized( false ) { m_bSystemDialogs = SvtMiscOptions().UseSystemFileDialog(); } ShutdownIcon::~ShutdownIcon() { deInitSystray(); new IdleUnloader (&m_pPlugin); } void ShutdownIcon::OpenURL( const OUString& aURL, const OUString& rTarget, const Sequence< PropertyValue >& aArgs ) { if ( getInstance() && getInstance()->m_xDesktop.is() ) { ::com::sun::star::uno::Reference < XDispatchProvider > xDispatchProvider( getInstance()->m_xDesktop, UNO_QUERY ); if ( xDispatchProvider.is() ) { com::sun::star::util::URL aDispatchURL; aDispatchURL.Complete = aURL; ::com::sun::star::uno::Reference< util::XURLTransformer > xURLTransformer( util::URLTransformer::create( ::comphelper::getProcessComponentContext() ) ); try { ::com::sun::star::uno::Reference< com::sun::star::frame::XDispatch > xDispatch; xURLTransformer->parseStrict( aDispatchURL ); xDispatch = xDispatchProvider->queryDispatch( aDispatchURL, rTarget, 0 ); if ( xDispatch.is() ) xDispatch->dispatch( aDispatchURL, aArgs ); } catch ( com::sun::star::uno::RuntimeException& ) { throw; } catch ( com::sun::star::uno::Exception& ) { } } } } void ShutdownIcon::FileOpen() { if ( getInstance() && getInstance()->m_xDesktop.is() ) { ::SolarMutexGuard aGuard; EnterModalMode(); getInstance()->StartFileDialog(); } } void ShutdownIcon::FromTemplate() { if ( getInstance() && getInstance()->m_xDesktop.is() ) { ::com::sun::star::uno::Reference < ::com::sun::star::frame::XFramesSupplier > xDesktop ( getInstance()->m_xDesktop, UNO_QUERY); ::com::sun::star::uno::Reference < ::com::sun::star::frame::XFrame > xFrame( xDesktop->getActiveFrame() ); if ( !xFrame.is() ) xFrame = ::com::sun::star::uno::Reference < ::com::sun::star::frame::XFrame >( xDesktop, UNO_QUERY ); URL aTargetURL; aTargetURL.Complete = "slot:5500"; ::com::sun::star::uno::Reference< util::XURLTransformer > xTrans( util::URLTransformer::create( ::comphelper::getProcessComponentContext() ) ); xTrans->parseStrict( aTargetURL ); ::com::sun::star::uno::Reference < ::com::sun::star::frame::XDispatchProvider > xProv( xFrame, UNO_QUERY ); ::com::sun::star::uno::Reference < ::com::sun::star::frame::XDispatch > xDisp; if ( xProv.is() ) { if (aTargetURL.Protocol == "slot:") xDisp = xProv->queryDispatch( aTargetURL, OUString(), 0 ); else xDisp = xProv->queryDispatch( aTargetURL, OUString("_blank"), 0 ); } if ( xDisp.is() ) { Sequence aArgs(1); PropertyValue* pArg = aArgs.getArray(); pArg[0].Name = "Referer"; pArg[0].Value <<= OUString("private:user"); ::com::sun::star::uno::Reference< ::com::sun::star::frame::XNotifyingDispatch > xNotifyer( xDisp, UNO_QUERY ); if ( xNotifyer.is() ) { EnterModalMode(); xNotifyer->dispatchWithNotification( aTargetURL, aArgs, new SfxNotificationListener_Impl() ); } else xDisp->dispatch( aTargetURL, aArgs ); } } } #include OUString ShutdownIcon::GetResString( int id ) { ::SolarMutexGuard aGuard; if( ! m_pResMgr ) m_pResMgr = SfxResId::GetResMgr(); ResId aResId( id, *m_pResMgr ); aResId.SetRT( RSC_STRING ); if( !m_pResMgr || !m_pResMgr->IsAvailable( aResId ) ) return OUString(); return ResId(id, *m_pResMgr).toString(); } OUString ShutdownIcon::GetUrlDescription( const OUString& aUrl ) { ::SolarMutexGuard aGuard; return OUString( SvFileInformationManager::GetDescription( INetURLObject( aUrl ) ) ); } void ShutdownIcon::StartFileDialog() { ::SolarMutexGuard aGuard; bool bDirty = ( m_bSystemDialogs != static_cast(SvtMiscOptions().UseSystemFileDialog()) ); if ( m_pFileDlg && bDirty ) { // Destroy instance as changing the system file dialog setting // forces us to create a new FileDialogHelper instance! delete m_pFileDlg; m_pFileDlg = NULL; } if ( !m_pFileDlg ) m_pFileDlg = new FileDialogHelper( ui::dialogs::TemplateDescription::FILEOPEN_READONLY_VERSION, SFXWB_MULTISELECTION, OUString() ); m_pFileDlg->StartExecuteModal( STATIC_LINK( this, ShutdownIcon, DialogClosedHdl_Impl ) ); } IMPL_STATIC_LINK( ShutdownIcon, DialogClosedHdl_Impl, FileDialogHelper*, EMPTYARG ) { DBG_ASSERT( pThis->m_pFileDlg, "ShutdownIcon, DialogClosedHdl_Impl(): no file dialog" ); // use constructor for filling up filters automatically! if ( ERRCODE_NONE == pThis->m_pFileDlg->GetError() ) { ::com::sun::star::uno::Reference< XFilePicker > xPicker = pThis->m_pFileDlg->GetFilePicker(); try { if ( xPicker.is() ) { ::com::sun::star::uno::Reference < XFilePickerControlAccess > xPickerControls ( xPicker, UNO_QUERY ); ::com::sun::star::uno::Reference < XFilterManager > xFilterManager ( xPicker, UNO_QUERY ); Sequence< OUString > sFiles = xPicker->getFiles(); int nFiles = sFiles.getLength(); int nArgs=3; Sequence< PropertyValue > aArgs(3); ::com::sun::star::uno::Reference < com::sun::star::task::XInteractionHandler2 > xInteraction( task::InteractionHandler::createWithParent(::comphelper::getProcessComponentContext(), 0) ); aArgs[0].Name = "InteractionHandler"; aArgs[0].Value <<= xInteraction; sal_Int16 nMacroExecMode = ::com::sun::star::document::MacroExecMode::USE_CONFIG; aArgs[1].Name = "MacroExecutionMode"; aArgs[1].Value <<= nMacroExecMode; sal_Int16 nUpdateDoc = ::com::sun::star::document::UpdateDocMode::ACCORDING_TO_CONFIG; aArgs[2].Name = "UpdateDocMode"; aArgs[2].Value <<= nUpdateDoc; // use the filedlghelper to get the current filter name, // because it removes the extensions before you get the filter name. OUString aFilterName( pThis->m_pFileDlg->GetCurrentFilter() ); if ( xPickerControls.is() ) { // Set readonly flag bool bReadOnly = false; xPickerControls->getValue( ExtendedFilePickerElementIds::CHECKBOX_READONLY, 0 ) >>= bReadOnly; // Only set porperty if readonly is set to TRUE if ( bReadOnly ) { aArgs.realloc( ++nArgs ); aArgs[nArgs-1].Name = "ReadOnly"; aArgs[nArgs-1].Value <<= bReadOnly; } // Get version string sal_Int32 iVersion = -1; xPickerControls->getValue( ExtendedFilePickerElementIds::LISTBOX_VERSION, ControlActions::GET_SELECTED_ITEM_INDEX ) >>= iVersion; if ( iVersion >= 0 ) { sal_Int16 uVersion = (sal_Int16)iVersion; aArgs.realloc( ++nArgs ); aArgs[nArgs-1].Name = "Version"; aArgs[nArgs-1].Value <<= uVersion; } // Retrieve the current filter if ( aFilterName.isEmpty() ) xPickerControls->getValue( CommonFilePickerElementIds::LISTBOX_FILTER, ControlActions::GET_SELECTED_ITEM ) >>= aFilterName; } // Convert UI filter name to internal filter name if ( !aFilterName.isEmpty() ) { const SfxFilter* pFilter = SfxGetpApp()->GetFilterMatcher().GetFilter4UIName( aFilterName, 0, SFX_FILTER_NOTINFILEDLG ); if ( pFilter ) { aFilterName = pFilter->GetFilterName(); if ( !aFilterName.isEmpty() ) { aArgs.realloc( ++nArgs ); aArgs[nArgs-1].Name = "FilterName"; aArgs[nArgs-1].Value <<= aFilterName; } } } if ( 1 == nFiles ) OpenURL( sFiles[0], OUString( "_default" ), aArgs ); else { OUString aBaseDirURL = sFiles[0]; if ( !aBaseDirURL.isEmpty() && !aBaseDirURL.endsWith("/") ) aBaseDirURL += "/"; int iFiles; for ( iFiles = 1; iFiles < nFiles; iFiles++ ) { OUString aURL = aBaseDirURL; aURL += sFiles[iFiles]; OpenURL( aURL, OUString( "_default" ), aArgs ); } } } } catch ( ... ) { } } #ifdef WNT // Destroy dialog to prevent problems with custom controls // This fix is dependent on the dialog settings. Destroying the dialog here will // crash the non-native dialog implementation! Therefore make this dependent on // the settings. if ( SvtMiscOptions().UseSystemFileDialog() ) { delete pThis->m_pFileDlg; pThis->m_pFileDlg = NULL; } #endif LeaveModalMode(); return 0; } void ShutdownIcon::addTerminateListener() { ShutdownIcon* pInst = getInstance(); if ( ! pInst) return; if (pInst->m_bListenForTermination) return; ::com::sun::star::uno::Reference< XDesktop2 > xDesktop = pInst->m_xDesktop; if ( ! xDesktop.is()) return; xDesktop->addTerminateListener( pInst ); pInst->m_bListenForTermination = true; } void ShutdownIcon::terminateDesktop() { ShutdownIcon* pInst = getInstance(); if ( ! pInst) return; ::com::sun::star::uno::Reference< XDesktop2 > xDesktop = pInst->m_xDesktop; if ( ! xDesktop.is()) return; // always remove ourselves as listener pInst->m_bListenForTermination = true; xDesktop->removeTerminateListener( pInst ); // terminate desktop only if no tasks exist ::com::sun::star::uno::Reference< XIndexAccess > xTasks ( xDesktop->getFrames(), UNO_QUERY ); if( xTasks.is() && xTasks->getCount() < 1 ) new IdleTerminate( xDesktop ); // remove the instance pointer ShutdownIcon::pShutdownIcon = 0; } ShutdownIcon* ShutdownIcon::getInstance() { OSL_ASSERT( pShutdownIcon ); return pShutdownIcon; } ShutdownIcon* ShutdownIcon::createInstance() { if (pShutdownIcon) return pShutdownIcon; ShutdownIcon *pIcon = NULL; try { pIcon = new ShutdownIcon( comphelper::getProcessComponentContext() ); pIcon->init (); pShutdownIcon = pIcon; } catch (...) { delete pIcon; } return pShutdownIcon; } void ShutdownIcon::init() throw( ::com::sun::star::uno::Exception ) { // access resource system and sfx only protected by solarmutex ::SolarMutexGuard aSolarGuard; ResMgr *pResMgr = SfxResId::GetResMgr(); ::osl::ResettableMutexGuard aGuard( m_aMutex ); m_pResMgr = pResMgr; aGuard.clear(); ::com::sun::star::uno::Reference < XDesktop2 > xDesktop = Desktop::create( m_xContext ); aGuard.reset(); m_xDesktop = xDesktop; } void SAL_CALL ShutdownIcon::disposing() { m_xContext.clear(); m_xDesktop.clear(); } // XEventListener void SAL_CALL ShutdownIcon::disposing( const ::com::sun::star::lang::EventObject& ) throw(::com::sun::star::uno::RuntimeException, std::exception) { } // XTerminateListener void SAL_CALL ShutdownIcon::queryTermination( const ::com::sun::star::lang::EventObject& ) throw(::com::sun::star::frame::TerminationVetoException, ::com::sun::star::uno::RuntimeException, std::exception) { SAL_INFO("sfx.appl", "ShutdownIcon::queryTermination: veto is " << m_bVeto); ::osl::ClearableMutexGuard aGuard( m_aMutex ); if ( m_bVeto ) throw ::com::sun::star::frame::TerminationVetoException(); } void SAL_CALL ShutdownIcon::notifyTermination( const ::com::sun::star::lang::EventObject& ) throw(::com::sun::star::uno::RuntimeException, std::exception) { } void SAL_CALL ShutdownIcon::initialize( const ::com::sun::star::uno::Sequence< ::com::sun::star::uno::Any>& aArguments ) throw( ::com::sun::star::uno::Exception, std::exception ) { ::osl::ResettableMutexGuard aGuard( m_aMutex ); // third argument only sets veto, everything else will be ignored! if (aArguments.getLength() > 2) { bool bVeto = ::cppu::any2bool(aArguments[2]); m_bVeto = bVeto; return; } if ( aArguments.getLength() > 0 ) { if ( !ShutdownIcon::pShutdownIcon ) { try { bool bQuickstart = ::cppu::any2bool( aArguments[0] ); if( !bQuickstart && !GetAutostart() ) return; aGuard.clear(); init (); aGuard.reset(); if ( !m_xDesktop.is() ) return; /* Create a sub-classed instance - foo */ ShutdownIcon::pShutdownIcon = this; initSystray(); } catch(const ::com::sun::star::lang::IllegalArgumentException&) { } } } if ( aArguments.getLength() > 1 ) { bool bAutostart = ::cppu::any2bool( aArguments[1] ); if (bAutostart && !GetAutostart()) SetAutostart( true ); if (!bAutostart && GetAutostart()) SetAutostart( false ); } } void ShutdownIcon::EnterModalMode() { bModalMode = true; } void ShutdownIcon::LeaveModalMode() { bModalMode = false; } #ifdef WNT // defined in shutdowniconw32.cxx #elif defined MACOSX // defined in shutdowniconaqua.cxx #else bool ShutdownIcon::IsQuickstarterInstalled() { #ifndef ENABLE_QUICKSTART_APPLET return false; #else // !ENABLE_QUICKSTART_APPLET #ifdef UNX return LoadModule( NULL, NULL, NULL); #endif // UNX #endif // !ENABLE_QUICKSTART_APPLET } #endif // !WNT #if defined (ENABLE_QUICKSTART_APPLET) && defined (UNX) /** * Return the XDG autostart directory. * http://standards.freedesktop.org/autostart-spec/autostart-spec-latest.html * Available in Unix and with Quickstart enabled. * @param bCreate Create the directory if it does not exist yet. * @return OUString containing the autostart directory path. */ static OUString getAutostartDir( bool bCreate = false ) { OUString aShortcut; const char *pConfigHome; if( (pConfigHome = getenv("XDG_CONFIG_HOME") ) ) aShortcut = OStringToOUString( OString( pConfigHome ), RTL_TEXTENCODING_UTF8 ); else { OUString aHomeURL; osl::Security().getHomeDir( aHomeURL ); ::osl::File::getSystemPathFromFileURL( aHomeURL, aShortcut ); aShortcut += "/.config"; } aShortcut += "/autostart"; if (bCreate) { OUString aShortcutUrl; osl::File::getFileURLFromSystemPath( aShortcut, aShortcutUrl ); osl::Directory::createPath( aShortcutUrl ); } return aShortcut; } #endif OUString ShutdownIcon::getShortcutName() { #ifndef ENABLE_QUICKSTART_APPLET return OUString(); #else OUString aShortcutName( "StarOffice 6.0" ); ResMgr* pMgr = SfxResId::GetResMgr(); if( pMgr ) { ::SolarMutexGuard aGuard; aShortcutName = SFX2_RESSTR(STR_QUICKSTART_LNKNAME); } #ifdef WNT aShortcutName += ".lnk"; OUString aShortcut(GetAutostartFolderNameW32()); aShortcut += "\\"; aShortcut += aShortcutName; #else // UNX OUString aShortcut = getAutostartDir(); aShortcut += "/qstart.desktop"; #endif // UNX return aShortcut; #endif // ENABLE_QUICKSTART_APPLET } bool ShutdownIcon::GetAutostart( ) { #if defined MACOSX return true; #else bool bRet = false; #ifdef ENABLE_QUICKSTART_APPLET OUString aShortcut( getShortcutName() ); OUString aShortcutUrl; osl::File::getFileURLFromSystemPath( aShortcut, aShortcutUrl ); osl::File f( aShortcutUrl ); osl::File::RC error = f.open( osl_File_OpenFlag_Read ); if( error == osl::File::E_None ) { f.close(); bRet = true; } #endif // ENABLE_QUICKSTART_APPLET return bRet; #endif } void ShutdownIcon::SetAutostart( bool bActivate ) { #ifdef ENABLE_QUICKSTART_APPLET OUString aShortcut( getShortcutName() ); if( bActivate && IsQuickstarterInstalled() ) { #ifdef WNT EnableAutostartW32( aShortcut ); #else // UNX getAutostartDir( true ); OUString aPath( "${BRAND_BASE_DIR}/" LIBO_SHARE_FOLDER "/xdg/qstart.desktop" ); rtl::Bootstrap::expandMacros( aPath ); OUString aDesktopFile; ::osl::File::getSystemPathFromFileURL( aPath, aDesktopFile ); OString aDesktopFileUnx = OUStringToOString( aDesktopFile, osl_getThreadTextEncoding() ); OString aShortcutUnx = OUStringToOString( aShortcut, osl_getThreadTextEncoding() ); if ((0 != symlink(aDesktopFileUnx.getStr(), aShortcutUnx.getStr())) && (errno == EEXIST)) { unlink(aShortcutUnx.getStr()); int ret = symlink(aDesktopFileUnx.getStr(), aShortcutUnx.getStr()); (void)ret; //deliberately ignore return value, it's non-critical if it fails } ShutdownIcon *pIcon = ShutdownIcon::createInstance(); if( pIcon ) pIcon->initSystray(); #endif // UNX } else { OUString aShortcutUrl; ::osl::File::getFileURLFromSystemPath( aShortcut, aShortcutUrl ); ::osl::File::remove( aShortcutUrl ); #ifdef UNX if (pShutdownIcon) { ShutdownIcon *pIcon = getInstance(); pIcon->deInitSystray(); } #endif } #else (void)bActivate; // unused variable #endif // ENABLE_QUICKSTART_APPLET } static const ::sal_Int32 PROPHANDLE_TERMINATEVETOSTATE = 0; // XFastPropertySet void SAL_CALL ShutdownIcon::setFastPropertyValue( ::sal_Int32 nHandle, const ::com::sun::star::uno::Any& aValue ) throw (::com::sun::star::beans::UnknownPropertyException, ::com::sun::star::beans::PropertyVetoException, ::com::sun::star::lang::IllegalArgumentException, ::com::sun::star::lang::WrappedTargetException, ::com::sun::star::uno::RuntimeException, std::exception) { switch(nHandle) { case PROPHANDLE_TERMINATEVETOSTATE : { // use new value in case it's a valid information only bool bState( false ); if (! (aValue >>= bState)) return; m_bVeto = bState; if (m_bVeto && ! m_bListenForTermination) addTerminateListener(); } break; default : throw ::com::sun::star::beans::UnknownPropertyException(); } } // XFastPropertySet ::com::sun::star::uno::Any SAL_CALL ShutdownIcon::getFastPropertyValue( ::sal_Int32 nHandle ) throw (::com::sun::star::beans::UnknownPropertyException, ::com::sun::star::lang::WrappedTargetException, ::com::sun::star::uno::RuntimeException, std::exception) { ::com::sun::star::uno::Any aValue; switch(nHandle) { case PROPHANDLE_TERMINATEVETOSTATE : { bool bState = (m_bListenForTermination && m_bVeto); aValue <<= bState; } break; default : throw ::com::sun::star::beans::UnknownPropertyException(); } return aValue; } extern "C" SAL_DLLPUBLIC_EXPORT css::uno::XInterface * SAL_CALL com_sun_star_comp_desktop_QuickstartWrapper_get_implementation( css::uno::XComponentContext *context, css::uno::Sequence const &) { return cppu::acquire(new ShutdownIcon(context)); } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */