/* -*- 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 #if defined UNX #include #else // UNX #include #ifndef PATH_MAX #define PATH_MAX _MAX_PATH #endif #endif // UNX #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 #include "sfxtypes.hxx" #include #include "arrdecl.hxx" #include #include #include #include #include #include #include #include #include #include #include "appdata.hxx" #include "openflag.hxx" #include "app.hrc" #include "virtmenu.hxx" #include #include #include "imestatuswindow.hxx" #include "workwin.hxx" #include #include #include #include #include "eventsupplier.hxx" #include #include "shellimpl.hxx" #ifdef DBG_UTIL #include #endif #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace ::com::sun::star; // Static member SfxApplication* SfxApplication::pApp = nullptr; #if HAVE_FEATURE_SCRIPTING static BasicDLL* pBasic = nullptr; #endif #if HAVE_FEATURE_DESKTOP static SfxHelp* pSfxHelp = nullptr; #endif namespace { class theApplicationMutex : public rtl::Static {}; } #include #include #include #include "fwkhelper.hxx" SfxApplication* SfxApplication::GetOrCreate() { // SFX on demand ::osl::MutexGuard aGuard(theApplicationMutex::get()); if (!pApp) { SAL_INFO( "sfx.appl", "SfxApplication::SetApp" ); pApp = new SfxApplication; // at the moment a bug may occur when Initialize_Impl returns FALSE, // but this is only temporary because all code that may cause such // a fault will be moved outside the SFX pApp->Initialize_Impl(); ::framework::SetImageProducer( GetImage ); ::framework::SetRefreshToolbars( RefreshToolbars ); ::framework::SetToolBoxControllerCreator( SfxToolBoxControllerFactory ); ::framework::SetStatusBarControllerCreator( SfxStatusBarControllerFactory ); ::framework::SetDockingWindowCreator( SfxDockingWindowFactory ); ::framework::SetIsDockingWindowVisible( IsDockingWindowVisible ); ::framework::SetActivateToolPanel( &SfxViewFrame::ActivateToolPanel ); #if HAVE_FEATURE_DESKTOP Application::SetHelp( pSfxHelp ); if (!utl::ConfigManager::IsAvoidConfig() && SvtHelpOptions().IsHelpTips()) Help::EnableQuickHelp(); else Help::DisableQuickHelp(); if (!utl::ConfigManager::IsAvoidConfig() && SvtHelpOptions().IsHelpTips() && SvtHelpOptions().IsExtendedHelp()) Help::EnableBalloonHelp(); else Help::DisableBalloonHelp(); #endif } return pApp; } SfxApplication::SfxApplication() : pAppData_Impl( nullptr ) { SetName( "StarOffice" ); if (!utl::ConfigManager::IsAvoidConfig()) SvtViewOptions::AcquireOptions(); pAppData_Impl = new SfxAppData_Impl( this ); pAppData_Impl->m_xImeStatusWindow->init(); SAL_INFO( "sfx.appl", "{ initialize DDE" ); bool bOk = InitializeDde(); #ifdef DBG_UTIL if( !bOk ) { OStringBuffer aStr("No DDE-Service possible. Error: "); if( GetDdeService() ) aStr.append(static_cast(GetDdeService()->GetError())); else aStr.append('?'); DBG_ASSERT( false, aStr.getStr() ); } #else (void)bOk; #endif #if HAVE_FEATURE_DESKTOP pSfxHelp = new SfxHelp; #endif #if HAVE_FEATURE_SCRIPTING pBasic = new BasicDLL; StarBASIC::SetGlobalErrorHdl( LINK( this, SfxApplication, GlobalBasicErrorHdl_Impl ) ); #endif SAL_INFO( "sfx.appl", "} initialize DDE" ); } SfxApplication::~SfxApplication() { OSL_ENSURE( GetObjectShells_Impl().size() == 0, "Memory leak: some object shells were not removed!" ); Broadcast( SfxSimpleHint(SFX_HINT_DYING) ); SfxModule::DestroyModules_Impl(); #if HAVE_FEATURE_DESKTOP delete pSfxHelp; Application::SetHelp(); #endif // delete global options if (!utl::ConfigManager::IsAvoidConfig()) SvtViewOptions::ReleaseOptions(); if ( !pAppData_Impl->bDowning ) Deinitialize(); #if HAVE_FEATURE_SCRIPTING delete pBasic; #endif delete pAppData_Impl; pApp = nullptr; } const OUString& SfxApplication::GetLastDir_Impl() const /* [Description] Internal method by which the last set directory with the method in SFX is returned. This is usually the most recently addressed by the SfxFileDialog directory. [Cross-reference] */ { return pAppData_Impl->aLastDir; } const OUString& SfxApplication::GetLastSaveDirectory() const /* [Description] As , only external [Cross-reference] */ { return GetLastDir_Impl(); } void SfxApplication::SetLastDir_Impl ( const OUString& rNewDir /* Complete directory path as a string */ ) /* [Description] Internal Method, by which a directory path is set that was last addressed (eg by the SfxFileDialog). [Cross-reference] */ { pAppData_Impl->aLastDir = rNewDir; } void SfxApplication::ResetLastDir() { pAppData_Impl->aLastDir.clear(); } SfxDispatcher* SfxApplication::GetDispatcher_Impl() { return pAppData_Impl->pViewFrame? pAppData_Impl->pViewFrame->GetDispatcher(): pAppData_Impl->pAppDispat; } void SfxApplication::SetViewFrame_Impl( SfxViewFrame *pFrame ) { if ( pFrame != pAppData_Impl->pViewFrame ) { // get the containerframes ( if one of the frames is an InPlaceFrame ) SfxViewFrame *pOldContainerFrame = pAppData_Impl->pViewFrame; while ( pOldContainerFrame && pOldContainerFrame->GetParentViewFrame_Impl() ) pOldContainerFrame = pOldContainerFrame->GetParentViewFrame_Impl(); SfxViewFrame *pNewContainerFrame = pFrame; while ( pNewContainerFrame && pNewContainerFrame->GetParentViewFrame_Impl() ) pNewContainerFrame = pNewContainerFrame->GetParentViewFrame_Impl(); // DocWinActivate : both frames belong to the same TopWindow // TopWinActivate : both frames belong to different TopWindows bool bTaskActivate = pOldContainerFrame != pNewContainerFrame; if ( pOldContainerFrame ) { if ( bTaskActivate ) NotifyEvent( SfxViewEventHint( SFX_EVENT_DEACTIVATEDOC, GlobalEventConfig::GetEventName(GlobalEventId::DEACTIVATEDOC), pOldContainerFrame->GetObjectShell(), pOldContainerFrame->GetFrame().GetController() ) ); pOldContainerFrame->DoDeactivate( bTaskActivate, pFrame ); if( pOldContainerFrame->GetProgress() ) pOldContainerFrame->GetProgress()->Suspend(); } pAppData_Impl->pViewFrame = pFrame; if( pNewContainerFrame ) { pNewContainerFrame->DoActivate( bTaskActivate ); if ( bTaskActivate && pNewContainerFrame->GetObjectShell() ) { pNewContainerFrame->GetObjectShell()->PostActivateEvent_Impl( pNewContainerFrame ); NotifyEvent(SfxViewEventHint(SFX_EVENT_ACTIVATEDOC, GlobalEventConfig::GetEventName(GlobalEventId::ACTIVATEDOC), pNewContainerFrame->GetObjectShell(), pNewContainerFrame->GetFrame().GetController() ) ); } SfxProgress *pProgress = pNewContainerFrame->GetProgress(); if ( pProgress ) { if( pProgress->IsSuspended() ) pProgress->Resume(); else pProgress->SetState( pProgress->GetState() ); } if ( pAppData_Impl->pViewFrame->GetViewShell() ) { SfxDispatcher* pDisp = pAppData_Impl->pViewFrame->GetDispatcher(); pDisp->Flush(); pDisp->Update_Impl(true); } } } // even if the frame actually didn't change, ensure its document is forwarded // to SfxObjectShell::SetCurrentComponent. // Otherwise, the CurrentComponent might not be correct, in case it has meanwhile // been reset to some other document, by some non-SFX component. // #i49133# / 2007-12-19 / frank.schoenheit@sun.com if ( pFrame && pFrame->GetViewShell() ) pFrame->GetViewShell()->SetCurrentDocument(); } ResMgr* SfxApplication::GetSfxResManager() { return SfxResId::GetResMgr(); } void SfxApplication::SetProgress_Impl ( SfxProgress *pProgress ) { DBG_ASSERT( ( !pAppData_Impl->pProgress && pProgress ) || ( pAppData_Impl->pProgress && !pProgress ), "Progress activation/deactivation mismatch" ); if ( pAppData_Impl->pProgress && pProgress ) { pAppData_Impl->pProgress->Suspend(); pAppData_Impl->pProgress->UnLock(); delete pAppData_Impl->pProgress; } pAppData_Impl->pProgress = pProgress; } sal_uInt16 SfxApplication::GetFreeIndex() { return pAppData_Impl->aIndexBitSet.GetFreeIndex()+1; } void SfxApplication::ReleaseIndex(sal_uInt16 i) { pAppData_Impl->aIndexBitSet.ReleaseIndex(i-1); } vcl::Window* SfxApplication::GetTopWindow() const { SfxWorkWindow* pWork = GetWorkWindow_Impl( SfxViewFrame::Current() ); return pWork ? pWork->GetWindow() : nullptr; } SfxTbxCtrlFactArr_Impl& SfxApplication::GetTbxCtrlFactories_Impl() const { return *pAppData_Impl->pTbxCtrlFac; } SfxStbCtrlFactArr_Impl& SfxApplication::GetStbCtrlFactories_Impl() const { return *pAppData_Impl->pStbCtrlFac; } SfxMenuCtrlFactArr_Impl& SfxApplication::GetMenuCtrlFactories_Impl() const { return *pAppData_Impl->pMenuCtrlFac; } SfxViewFrameArr_Impl& SfxApplication::GetViewFrames_Impl() const { return *pAppData_Impl->pViewFrames; } SfxViewShellArr_Impl& SfxApplication::GetViewShells_Impl() const { return *pAppData_Impl->pViewShells; } SfxObjectShellArr_Impl& SfxApplication::GetObjectShells_Impl() const { return *pAppData_Impl->pObjShells; } void SfxApplication::Invalidate( sal_uInt16 nId ) { for( SfxViewFrame* pFrame = SfxViewFrame::GetFirst(); pFrame; pFrame = SfxViewFrame::GetNext( *pFrame ) ) Invalidate_Impl( pFrame->GetBindings(), nId ); } #if HAVE_FEATURE_SCRIPTING #ifndef DISABLE_DYNLOADING typedef long (SAL_CALL *basicide_handle_basic_error)(void*); typedef void (SAL_CALL *basicide_macro_organizer)(sal_Int16); extern "C" { static void SAL_CALL thisModule() {} } #else extern "C" long basicide_handle_basic_error(void*); extern "C" void basicide_macro_organizer(sal_Int16); #endif #endif IMPL_LINK_TYPED( SfxApplication, GlobalBasicErrorHdl_Impl, StarBASIC*, pStarBasic, bool ) { #if !HAVE_FEATURE_SCRIPTING (void) pStarBasic; return false; #else #ifndef DISABLE_DYNLOADING // load basctl module osl::Module aMod; aMod.loadRelative(&thisModule, SVLIBRARY("basctl")); // get symbol basicide_handle_basic_error pSymbol = reinterpret_cast(aMod.getFunctionSymbol("basicide_handle_basic_error")); aMod.release(); // call basicide_handle_basic_error in basctl bool bRet = pSymbol && pSymbol( pStarBasic ); #else bool bRet = basicide_handle_basic_error( pStarBasic ); #endif return bRet; #endif } bool SfxApplication::IsXScriptURL( const OUString& rScriptURL ) { bool result = false; #if !HAVE_FEATURE_SCRIPTING (void) rScriptURL; #else css::uno::Reference< css::uno::XComponentContext > xContext = ::comphelper::getProcessComponentContext(); css::uno::Reference< css::uri::XUriReferenceFactory > xFactory = css::uri::UriReferenceFactory::create( xContext ); try { css::uno::Reference< css::uri::XVndSunStarScriptUrl > xUrl( xFactory->parse( rScriptURL ), css::uno::UNO_QUERY ); if ( xUrl.is() ) { result = true; } } catch (const css::uno::RuntimeException&) { // ignore, will just return FALSE } #endif return result; } OUString SfxApplication::ChooseScript() { OUString aScriptURL; #if HAVE_FEATURE_SCRIPTING SfxAbstractDialogFactory* pFact = SfxAbstractDialogFactory::Create(); if ( pFact ) { SAL_INFO( "sfx.appl", "create selector dialog"); const SfxViewFrame* pViewFrame = SfxViewFrame::Current(); const SfxFrame* pFrame = pViewFrame ? &pViewFrame->GetFrame() : nullptr; uno::Reference< frame::XFrame > xFrame( pFrame ? pFrame->GetFrameInterface() : uno::Reference< frame::XFrame >() ); std::unique_ptr pDlg( pFact->CreateScriptSelectorDialog( nullptr, false, xFrame )); SAL_INFO( "sfx.appl", "done, now exec it"); sal_uInt16 nRet = pDlg->Execute(); SAL_INFO( "sfx.appl", "has returned"); if ( nRet == RET_OK ) { aScriptURL = pDlg->GetScriptURL(); } } #endif return aScriptURL; } void SfxApplication::MacroOrganizer( sal_Int16 nTabId ) { #if !HAVE_FEATURE_SCRIPTING (void) nTabId; #else #ifndef DISABLE_DYNLOADING // load basctl module osl::Module aMod; aMod.loadRelative(&thisModule, SVLIBRARY("basctl")); // get symbol basicide_macro_organizer pSymbol = reinterpret_cast(aMod.getFunctionSymbol("basicide_macro_organizer")); aMod.release(); SAL_WARN_IF(!pSymbol, "sfx.doc", "SfxApplication::MacroOrganizer, no symbol!"); if (!pSymbol) return; // call basicide_macro_organizer in basctl pSymbol( nTabId ); #else basicide_macro_organizer( nTabId ); #endif #endif } ErrCode SfxApplication::CallBasic( const OUString& rCode, BasicManager* pMgr, SbxArray* pArgs, SbxValue* pRet ) { #if !HAVE_FEATURE_SCRIPTING (void) rCode; (void) pMgr; (void) pArgs; (void) pRet; return ERRCODE_BASIC_CANNOT_LOAD; #else return pMgr->ExecuteMacro( rCode, pArgs, pRet); #endif } sfx2::sidebar::Theme & SfxApplication::GetSidebarTheme() { if (!pAppData_Impl->m_pSidebarTheme.is()) { pAppData_Impl->m_pSidebarTheme.set(new sfx2::sidebar::Theme); pAppData_Impl->m_pSidebarTheme->InitializeTheme(); } return *pAppData_Impl->m_pSidebarTheme; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */