/* -*- 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 namespace sc { struct DocumentLinkManagerImpl : boost::noncopyable { ScDocument& mrDoc; SfxObjectShell* mpShell; boost::scoped_ptr mpDataStream; boost::scoped_ptr mpLinkManager; DocumentLinkManagerImpl( ScDocument& rDoc, SfxObjectShell* pShell ) : mrDoc(rDoc), mpShell(pShell), mpDataStream(NULL), mpLinkManager(NULL) {} ~DocumentLinkManagerImpl() { // Shared base links if (mpLinkManager) { sfx2::SvLinkSources aTemp = mpLinkManager->GetServers(); for (sfx2::SvLinkSources::const_iterator it = aTemp.begin(); it != aTemp.end(); ++it) (*it)->Closed(); if (!mpLinkManager->GetLinks().empty()) mpLinkManager->Remove(0, mpLinkManager->GetLinks().size()); } } }; DocumentLinkManager::DocumentLinkManager( ScDocument& rDoc, SfxObjectShell* pShell ) : mpImpl(new DocumentLinkManagerImpl(rDoc, pShell)) {} DocumentLinkManager::~DocumentLinkManager() { delete mpImpl; } void DocumentLinkManager::setDataStream( DataStream* p ) { mpImpl->mpDataStream.reset(p); } DataStream* DocumentLinkManager::getDataStream() { return mpImpl->mpDataStream.get(); } const DataStream* DocumentLinkManager::getDataStream() const { return mpImpl->mpDataStream.get(); } sfx2::LinkManager* DocumentLinkManager::getLinkManager( bool bCreate ) { if (!mpImpl->mpLinkManager && bCreate && mpImpl->mpShell) mpImpl->mpLinkManager.reset(new sfx2::LinkManager(mpImpl->mpShell)); return mpImpl->mpLinkManager.get(); } const sfx2::LinkManager* DocumentLinkManager::getExistingLinkManager() const { return mpImpl->mpLinkManager.get(); } bool DocumentLinkManager::idleCheckLinks() { if (!mpImpl->mpLinkManager) return false; bool bAnyLeft = false; const sfx2::SvBaseLinks& rLinks = mpImpl->mpLinkManager->GetLinks(); for (size_t i = 0, n = rLinks.size(); i < n; ++i) { sfx2::SvBaseLink* pBase = *rLinks[i]; ScDdeLink* pDdeLink = dynamic_cast(pBase); if (!pDdeLink || !pDdeLink->NeedsUpdate()) continue; pDdeLink->TryUpdate(); if (pDdeLink->NeedsUpdate()) // Was not successful? bAnyLeft = true; } return bAnyLeft; } bool DocumentLinkManager::hasDdeLinks() const { if (!mpImpl->mpLinkManager) return false; const sfx2::SvBaseLinks& rLinks = mpImpl->mpLinkManager->GetLinks(); for (size_t i = 0, n = rLinks.size(); i < n; ++i) { sfx2::SvBaseLink* pBase = *rLinks[i]; if (dynamic_cast(pBase)) return true; } return false; } bool DocumentLinkManager::updateDdeLinks( vcl::Window* pWin ) { if (!mpImpl->mpLinkManager) return false; sfx2::LinkManager* pMgr = mpImpl->mpLinkManager.get(); const sfx2::SvBaseLinks& rLinks = pMgr->GetLinks(); // If the update takes longer, reset all values so that nothing // old (wrong) is left behind bool bAny = false; for (size_t i = 0, n = rLinks.size(); i < n; ++i) { sfx2::SvBaseLink* pBase = *rLinks[i]; ScDdeLink* pDdeLink = dynamic_cast(pBase); if (!pDdeLink) continue; if (pDdeLink->Update()) bAny = true; else { // Update failed. Notify the user. OUString aFile = pDdeLink->GetTopic(); OUString aElem = pDdeLink->GetItem(); OUString aType = pDdeLink->GetAppl(); OUStringBuffer aBuf; aBuf.append(OUString(ScResId(SCSTR_DDEDOC_NOT_LOADED))); aBuf.appendAscii("\n\n"); aBuf.appendAscii("Source : "); aBuf.append(aFile); aBuf.appendAscii("\nElement : "); aBuf.append(aElem); aBuf.appendAscii("\nType : "); aBuf.append(aType); MessageDialog aBox(pWin, aBuf.makeStringAndClear()); aBox.Execute(); } } pMgr->CloseCachedComps(); return bAny; } bool DocumentLinkManager::updateDdeLink( const OUString& rAppl, const OUString& rTopic, const OUString& rItem ) { if (!mpImpl->mpLinkManager) return false; sfx2::LinkManager* pMgr = mpImpl->mpLinkManager.get(); const sfx2::SvBaseLinks& rLinks = pMgr->GetLinks(); bool bFound = false; for (size_t i = 0, n = rLinks.size(); i < n; ++i) { ::sfx2::SvBaseLink* pBase = *rLinks[i]; ScDdeLink* pDdeLink = dynamic_cast(pBase); if (!pDdeLink) continue; if ( OUString(pDdeLink->GetAppl()) == rAppl && OUString(pDdeLink->GetTopic()) == rTopic && OUString(pDdeLink->GetItem()) == rItem ) { pDdeLink->TryUpdate(); bFound = true; // Could be multiple (Mode), so continue searching } } return bFound; } size_t DocumentLinkManager::getDdeLinkCount() const { if (!mpImpl->mpLinkManager) return 0; size_t nDdeCount = 0; const sfx2::SvBaseLinks& rLinks = mpImpl->mpLinkManager->GetLinks(); for (size_t i = 0, n = rLinks.size(); i < n; ++i) { ::sfx2::SvBaseLink* pBase = *rLinks[i]; ScDdeLink* pDdeLink = dynamic_cast(pBase); if (!pDdeLink) continue; ++nDdeCount; } return nDdeCount; } void DocumentLinkManager::disconnectDdeLinks() { if (!mpImpl->mpLinkManager) return; const sfx2::SvBaseLinks& rLinks = mpImpl->mpLinkManager->GetLinks(); for (size_t i = 0, n = rLinks.size(); i < n; ++i) { ::sfx2::SvBaseLink* pBase = *rLinks[i]; ScDdeLink* pDdeLink = dynamic_cast(pBase); if (pDdeLink) pDdeLink->Disconnect(); } } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */