/* -*- 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 "appdata.hxx" #include #include #include #include "sfxtypes.hxx" #include #include "helper.hxx" #include #include #include #if defined(_WIN32) OUString SfxDdeServiceName_Impl( const OUString& sIn ) { OUStringBuffer sReturn(sIn.getLength()); for ( sal_uInt16 n = sIn.getLength(); n; --n ) { sal_Unicode cChar = sIn[n-1]; if (rtl::isAsciiAlphanumeric(cChar)) sReturn.append(cChar); } return sReturn.makeStringAndClear(); } class ImplDdeService : public DdeService { public: explicit ImplDdeService( const OUString& rNm ) : DdeService( rNm ) {} virtual bool MakeTopic( const OUString& ); virtual OUString Topics(); virtual bool SysTopicExecute( const OUString* pStr ); }; namespace { sal_Bool lcl_IsDocument( const OUString& rContent ) { using namespace com::sun::star; sal_Bool bRet = sal_False; INetURLObject aObj( rContent ); DBG_ASSERT( aObj.GetProtocol() != INetProtocol::NotValid, "Invalid URL!" ); try { ::ucbhelper::Content aCnt( aObj.GetMainURL( INetURLObject::NO_DECODE ), uno::Reference< ucb::XCommandEnvironment >(), comphelper::getProcessComponentContext() ); bRet = aCnt.isDocument(); } catch( const ucb::CommandAbortedException& ) { SAL_INFO( "sfx2.appl", "CommandAbortedException" ); } catch( const ucb::IllegalIdentifierException& ) { SAL_INFO( "sfx2.appl", "IllegalIdentifierException" ); } catch( const ucb::ContentCreationException& ) { SAL_INFO( "sfx2.appl", "IllegalIdentifierException" ); } catch( const uno::Exception& ) { SAL_WARN( "sfx.appl", "Any other exception" ); } return bRet; } } bool ImplDdeService::MakeTopic( const OUString& rNm ) { // Workaround for Event after Main() under OS/2 // happens when exiting starts the App again if ( !Application::IsInExecute() ) return false; // The Topic rNm is sought, do we have it? // First only loop over the ObjectShells to find those // with the specific name: sal_Bool bRet = sal_False; OUString sNm( rNm.toAsciiLowerCase() ); SfxObjectShell* pShell = SfxObjectShell::GetFirst(); while( pShell ) { OUString sTmp( pShell->GetTitle(SFX_TITLE_FULLNAME) ); if( sNm == sTmp.toAsciiLowerCase() ) { SfxGetpApp()->AddDdeTopic( pShell ); bRet = true; break; } pShell = SfxObjectShell::GetNext( *pShell ); } if( !bRet ) { INetURLObject aWorkPath( SvtPathOptions().GetWorkPath() ); INetURLObject aFile; if ( aWorkPath.GetNewAbsURL( rNm, &aFile ) && lcl_IsDocument( aFile.GetMainURL( INetURLObject::NO_DECODE ) ) ) { // File exists? then try to load it: SfxStringItem aName( SID_FILE_NAME, aFile.GetMainURL( INetURLObject::NO_DECODE ) ); SfxBoolItem aNewView(SID_OPEN_NEW_VIEW, sal_True); SfxBoolItem aSilent(SID_SILENT, sal_True); SfxDispatcher* pDispatcher = SfxGetpApp()->GetDispatcher_Impl(); const SfxPoolItem* pRet = pDispatcher->Execute( SID_OPENDOC, SfxCallMode::SYNCHRON, &aName, &aNewView, &aSilent, 0L ); if( pRet && dynamic_cast< const SfxViewFrameItem *>( pRet ) != nullptr && ((SfxViewFrameItem*)pRet)->GetFrame() && 0 != ( pShell = ((SfxViewFrameItem*)pRet) ->GetFrame()->GetObjectShell() ) ) { SfxGetpApp()->AddDdeTopic( pShell ); bRet = true; } } } return bRet; } OUString ImplDdeService::Topics() { OUString sRet; if( GetSysTopic() ) sRet += GetSysTopic()->GetName(); SfxObjectShell* pShell = SfxObjectShell::GetFirst(); while( pShell ) { if( SfxViewFrame::GetFirst( pShell ) ) { if( !sRet.isEmpty() ) sRet += "\t"; sRet += pShell->GetTitle(SFX_TITLE_FULLNAME); } pShell = SfxObjectShell::GetNext( *pShell ); } if( !sRet.isEmpty() ) sRet += "\r\n"; return sRet; } bool ImplDdeService::SysTopicExecute( const OUString* pStr ) { return SfxGetpApp()->DdeExecute( *pStr ); } #endif class SfxDdeTriggerTopic_Impl : public DdeTopic { #if defined(_WIN32) public: SfxDdeTriggerTopic_Impl() : DdeTopic( "TRIGGER" ) {} virtual bool Execute( const OUString* ) override { return true; } #endif }; class SfxDdeDocTopic_Impl : public DdeTopic { #if defined(_WIN32) public: SfxObjectShell* pSh; DdeData aData; css::uno::Sequence< sal_Int8 > aSeq; explicit SfxDdeDocTopic_Impl( SfxObjectShell* pShell ) : DdeTopic( pShell->GetTitle(SFX_TITLE_FULLNAME) ), pSh( pShell ) {} virtual DdeData* Get( SotClipboardFormatId ) override; virtual bool Put( const DdeData* ) override; virtual bool Execute( const OUString* ) override; virtual bool StartAdviseLoop() override; virtual bool MakeItem( const OUString& rItem ) override; #endif }; #if defined(_WIN32) namespace { /* [Description] Checks if 'rCmd' of the event 'rEvent' is (without '(') and then assemble this data into a , which is then executed through . If 'rCmd' is the given event 'rEvent', then TRUE is returned, otherwise FALSE. [Example] rCmd = "Open(\"d:\doc\doc.sdw\")" rEvent = "Open" */ sal_Bool SfxAppEvent_Impl( const OUString& rCmd, const OUString& rEvent, ApplicationEvent::Type eType ) { OUString sEvent(rEvent + "("); if (rCmd.startsWithIgnoreAsciiCase(sEvent)) { sal_Int32 start = sEvent.getLength(); if ( rCmd.getLength() - start >= 2 ) { // Transform into the ApplicationEvent Format //TODO: I /assume/ that rCmd should match the syntax of // // "WM_DDE_EXECUTE message" but does not (handle commands enclosed // in [...]; handle commas separating multiple arguments; handle // double "", ((, )), [[, ]] in quoted arguments); see also the mail // thread starting at "DDE on Windows." std::vector aData; for ( sal_Int32 n = start; n < rCmd.getLength() - 1; ) { // Resiliently read arguments either starting with " and // spanning to the next " (if any; TODO: do we need to undo any // escaping within the string?) or with neither " nor SPC and // spanning to the next SPC (if any; TODO: is this from not // wrapped in "..." relevant? it would have been parsed by the // original code even if that was only by accident, so I left it // in), with runs of SPCs treated like single ones: switch ( rCmd[n] ) { case '"': { sal_Int32 i = rCmd.indexOf('"', ++n); if (i < 0 || i > rCmd.getLength() - 1) { i = rCmd.getLength() - 1; } aData.push_back(rCmd.copy(n, i - n)); n = i + 1; break; } case ' ': ++n; break; default: { sal_Int32 i = rCmd.indexOf(' ', n); if (i < 0 || i > rCmd.getLength() - 1) { i = rCmd.getLength() - 1; } aData.push_back(rCmd.copy(n, i - n)); n = i + 1; break; } } } GetpApp()->AppEvent( ApplicationEvent(eType, aData) ); return sal_True; } } return sal_False; } } /* Description] This method can be overridden by application developers, to receive DDE-commands directed to their SfxApplication subclass. The base implementation understands the API functionality of the relevant SfxApplication subclass in BASIC syntax. Return values can not be transferred, unfortunately. */ long SfxApplication::DdeExecute( const OUString& rCmd ) // Expressed in our BASIC-Syntax { // Print or Open-Event? if ( !( SfxAppEvent_Impl( rCmd, "Print", ApplicationEvent::TYPE_PRINT ) || SfxAppEvent_Impl( rCmd, "Open", ApplicationEvent::TYPE_OPEN ) ) ) { // all others are BASIC StarBASIC* pBasic = GetBasic(); DBG_ASSERT( pBasic, "Where is the Basic???" ); SbxVariable* pRet = pBasic->Execute( rCmd ); if( !pRet ) { SbxBase::ResetError(); return 0; } } return 1; } /* [Description] This method can be overridden by application developers, to receive DDE-commands directed to the their SfxApplication subclass. The base implementation does nothing and returns 0. */ long SfxObjectShell::DdeExecute( const OUString& rCmd ) // Expressed in our BASIC-Syntax { #if !HAVE_FEATURE_SCRIPTING (void) rCmd; #else StarBASIC* pBasic = GetBasic(); DBG_ASSERT( pBasic, "Where is the Basic???" ) ; SbxVariable* pRet = pBasic->Execute( rCmd ); if( !pRet ) { SbxBase::ResetError(); return 0; } #endif return 1; } /* [Description] This method can be overridden by application developers, to receive DDE-data-requests directed to their SfxApplication subclass. The base implementation provides no data and returns false. */ bool SfxObjectShell::DdeGetData( const OUString&, // the Item to be addressed const OUString&, // in: Format css::uno::Any& )// out: requested data { return false; } /* [Description] This method can be overridden by application developers, to receive DDE-data directed to their SfxApplication subclass. The base implementation is not receiving any data and returns false. */ bool SfxObjectShell::DdeSetData( const OUString&, // the Item to be addressed const OUString&, // in: Format const css::uno::Any& )// out: requested data { return false; } #endif /* [Description] This method can be overridden by application developers, to establish a DDE-hotlink to their SfxApplication subclass. The base implementation is not generate a link and returns 0. */ ::sfx2::SvLinkSource* SfxObjectShell::DdeCreateLinkSource( const OUString& ) // the Item to be addressed { return nullptr; } void SfxObjectShell::ReconnectDdeLink(SfxObjectShell& /*rServer*/) { } void SfxObjectShell::ReconnectDdeLinks(SfxObjectShell& rServer) { SfxObjectShell* p = GetFirst(nullptr, false); while (p) { if (&rServer != p) p->ReconnectDdeLink(rServer); p = GetNext(*p, nullptr, false); } } bool SfxApplication::InitializeDde() { int nError = 0; #if defined(_WIN32) DBG_ASSERT( !pAppData_Impl->pDdeService, "Dde can not be initialized multiple times" ); pAppData_Impl->pDdeService = new ImplDdeService( Application::GetAppName() ); nError = pAppData_Impl->pDdeService->GetError(); if( !nError ) { pAppData_Impl->pDocTopics = new SfxDdeDocTopics_Impl; // we certainly want to support RTF! pAppData_Impl->pDdeService->AddFormat( SotClipboardFormatId::RTF ); // Config path as a topic because of multiple starts INetURLObject aOfficeLockFile( SvtPathOptions().GetUserConfigPath() ); aOfficeLockFile.insertName( "soffice.lck" ); OUString aService( SfxDdeServiceName_Impl( aOfficeLockFile.GetMainURL(INetURLObject::DECODE_TO_IURI) ) ); aService = aService.toAsciiUpperCase(); pAppData_Impl->pDdeService2 = new ImplDdeService( aService ); pAppData_Impl->pTriggerTopic = new SfxDdeTriggerTopic_Impl; pAppData_Impl->pDdeService2->AddTopic( *pAppData_Impl->pTriggerTopic ); } #endif return !nError; } void SfxAppData_Impl::DeInitDDE() { DELETEZ( pTriggerTopic ); DELETEZ( pDdeService2 ); DELETEZ( pDocTopics ); DELETEZ( pDdeService ); } #if defined(_WIN32) void SfxApplication::AddDdeTopic( SfxObjectShell* pSh ) { //OV: DDE is disconnected in server mode! if( !pAppData_Impl->pDocTopics ) return; // prevent double submit OUString sShellNm; sal_Bool bFnd = sal_False; for (size_t n = pAppData_Impl->pDocTopics->size(); n;) { if( (*pAppData_Impl->pDocTopics)[ --n ]->pSh == pSh ) { // If the document is untitled, is still a new Topic is created! if( !bFnd ) { bFnd = sal_True; sShellNm = pSh->GetTitle(SFX_TITLE_FULLNAME).toAsciiLowerCase(); } OUString sNm( (*pAppData_Impl->pDocTopics)[ n ]->GetName() ); if( sShellNm == sNm.toAsciiLowerCase() ) return ; } } SfxDdeDocTopic_Impl *const pTopic = new SfxDdeDocTopic_Impl(pSh); pAppData_Impl->pDocTopics->push_back(pTopic); pAppData_Impl->pDdeService->AddTopic( *pTopic ); } #endif void SfxApplication::RemoveDdeTopic( SfxObjectShell* pSh ) { #if defined(_WIN32) //OV: DDE is disconnected in server mode! if( !pAppData_Impl->pDocTopics ) return; for (size_t n = pAppData_Impl->pDocTopics->size(); n; ) { SfxDdeDocTopic_Impl *const pTopic = (*pAppData_Impl->pDocTopics)[ --n ]; if (pTopic->pSh == pSh) { pAppData_Impl->pDdeService->RemoveTopic( *pTopic ); delete pTopic; pAppData_Impl->pDocTopics->erase( pAppData_Impl->pDocTopics->begin() + n ); } } #else (void) pSh; #endif } const DdeService* SfxApplication::GetDdeService() const { return pAppData_Impl->pDdeService; } DdeService* SfxApplication::GetDdeService() { return pAppData_Impl->pDdeService; } #if defined(_WIN32) DdeData* SfxDdeDocTopic_Impl::Get(SotClipboardFormatId nFormat) { OUString sMimeType( SotExchange::GetFormatMimeType( nFormat )); css::uno::Any aValue; bool bRet = pSh->DdeGetData( GetCurItem(), sMimeType, aValue ); if( bRet && aValue.hasValue() && ( aValue >>= aSeq ) ) { aData = DdeData( aSeq.getConstArray(), aSeq.getLength(), nFormat ); return &aData; } aSeq.realloc( 0 ); return 0; } bool SfxDdeDocTopic_Impl::Put( const DdeData* pData ) { aSeq = css::uno::Sequence< sal_Int8 >( (sal_Int8*)(const void*)*pData, (long)*pData ); bool bRet; if( aSeq.getLength() ) { css::uno::Any aValue; aValue <<= aSeq; OUString sMimeType( SotExchange::GetFormatMimeType( pData->GetFormat() )); bRet = pSh->DdeSetData( GetCurItem(), sMimeType, aValue ); } else bRet = false; return bRet; } bool SfxDdeDocTopic_Impl::Execute( const OUString* pStr ) { long nRet = pStr ? pSh->DdeExecute( *pStr ) : 0; return 0 != nRet; } bool SfxDdeDocTopic_Impl::MakeItem( const OUString& rItem ) { AddItem( DdeItem( rItem ) ); return true; } bool SfxDdeDocTopic_Impl::StartAdviseLoop() { bool bRet = false; ::sfx2::SvLinkSource* pNewObj = pSh->DdeCreateLinkSource( GetCurItem() ); if( pNewObj ) { // then we also establish a corresponding SvBaseLink OUString sNm, sTmp( Application::GetAppName() ); ::sfx2::MakeLnkName( sNm, &sTmp, pSh->GetTitle(SFX_TITLE_FULLNAME), GetCurItem() ); new ::sfx2::SvBaseLink( sNm, OBJECT_DDE_EXTERN, pNewObj ); bRet = true; } return bRet; } #endif /* vim:set shiftwidth=4 softtabstop=4 expandtab: */