Files
libreoffice/sc/source/core/tool/ddelink.cxx

258 lines
8.1 KiB
C++
Raw Normal View History

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
re-base on ALv2 code. Includes: Patches contributed by Herbert Duerr i#118735 prevent endless loop if vlookup/hlookup doesn't find anything http://svn.apache.org/viewvc?view=revision&revision=1239673 Patches contributed by Andre Fischer remove lp_solver http://svn.apache.org/viewvc?view=revision&revision=1199180 i#118160: Added external CoinMP library. http://svn.apache.org/viewvc?view=revision&revision=1233909 Patches contributed by Armin Le-Grand i#118485 - Styles for OLEs are not saved. http://svn.apache.org/viewvc?view=revision&revision=1182166 i#118524: apply patch, followup fixes to 118485 http://svn.apache.org/viewvc?view=revision&revision=1186077 Patches contributed by lihuiibm i#108860 - Fix range validation. http://svn.apache.org/viewvc?view=revision&revision=1242846 i#118954 Chart data will lost after copy to different file http://svn.apache.org/viewvc?view=revision&revision=1301345 Patches contributed by Ariel Constenla-Haile Fix Linux build breaker: extra qualification on member http://svn.apache.org/viewvc?view=revision&revision=1301591 i#118696 - i#118697 - Fix some Sheet Tab Color API issues http://svn.apache.org/viewvc?view=revision&revision=1225428 i#118697 - Fix uninitialized variable http://svn.apache.org/viewvc?view=revision&revision=1225859 i#118771 - ScUndoImportTab should preserve tab background color http://svn.apache.org/viewvc?view=revision&revision=1230356 i#118921 - Repaint linked sheet tab background color after updating link http://svn.apache.org/viewvc?view=revision&revision=1245177 i#118927 - Undo/Redo "Update Link" does not reset sheet tab color http://svn.apache.org/viewvc?view=revision&revision=1245241 i#118747 - Copy tab color when transferring sheets across documents http://svn.apache.org/viewvc?view=revision&revision=1230355 Patch contributed by Oliver Rainer-Wittman i#118012 - methods <ScBroadcastAreaSlot::AreaBroadcast(..)> and <ScBroadcastAreaSlot::AreaBroadcastInRange(..)> adapt stl-container iteration in order to avoid destroyed iterators during iteration. http://svn.apache.org/viewvc?view=revision&revision=1297916 Patches contributed by Mathias Bauer gnumake4 work variously http://svn.apache.org/viewvc?view=revision&revision=1394707 http://svn.apache.org/viewvc?view=revision&revision=1394326 http://svn.apache.org/viewvc?view=revision&revision=1396797 http://svn.apache.org/viewvc?view=revision&revision=1397315 Patch contributed by Daniel Rentz calc69: #i116936# fix VBA symbol Cells http://svn.apache.org/viewvc?view=revision&revision=1172135 Patches contributed by leiw: i#118546 CPU 100% on switched off AutoCalculate with Conditional Formatting on date values http://svn.apache.org/viewvc?view=revision&revision=1301380 Re-add new function documentation. Many various cleanups. Add missing calc66: #o11817313# also look at formula result number format, remove redundant binaries.
2012-11-30 12:23:25 +00:00
/*
* 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 .
*/
2000-09-18 23:16:46 +00:00
#include <comphelper/string.hxx>
2010-01-13 22:25:07 +01:00
#include <sfx2/linkmgr.hxx>
2000-09-18 23:16:46 +00:00
#include <sfx2/bindings.hxx>
#include <svl/zforlist.hxx>
2000-09-18 23:16:46 +00:00
#include "ddelink.hxx"
#include "brdcst.hxx"
#include "document.hxx"
#include "scmatrix.hxx"
#include "patattr.hxx"
#include "rechead.hxx"
#include "rangeseq.hxx"
2000-09-18 23:16:46 +00:00
#include "sc.hrc"
#include "hints.hxx"
TYPEINIT2(ScDdeLink,::sfx2::SvBaseLink,SfxBroadcaster);
#define DDE_TXT_ENCODING osl_getThreadTextEncoding()
2000-09-18 23:16:46 +00:00
bool ScDdeLink::bIsInUpdate = false;
2000-09-18 23:16:46 +00:00
//------------------------------------------------------------------------
ScDdeLink::ScDdeLink( ScDocument* pD, const String& rA, const String& rT, const String& rI,
sal_uInt8 nM ) :
::sfx2::SvBaseLink(sfx2::LINKUPDATE_ALWAYS,FORMAT_STRING),
2000-09-18 23:16:46 +00:00
pDoc( pD ),
aAppl( rA ),
aTopic( rT ),
aItem( rI ),
nMode( nM ),
bNeedUpdate( false ),
pResult( NULL )
2000-09-18 23:16:46 +00:00
{
}
2010-12-11 23:25:30 +01:00
ScDdeLink::~ScDdeLink()
2000-09-18 23:16:46 +00:00
{
// Verbindung aufheben
// pResult is refcounted
2000-09-18 23:16:46 +00:00
}
ScDdeLink::ScDdeLink( ScDocument* pD, const ScDdeLink& rOther ) :
::sfx2::SvBaseLink(sfx2::LINKUPDATE_ALWAYS,FORMAT_STRING),
2000-09-18 23:16:46 +00:00
pDoc ( pD ),
aAppl ( rOther.aAppl ),
aTopic ( rOther.aTopic ),
aItem ( rOther.aItem ),
nMode ( rOther.nMode ),
bNeedUpdate( false ),
pResult ( NULL )
2000-09-18 23:16:46 +00:00
{
if (rOther.pResult)
pResult = rOther.pResult->Clone();
}
ScDdeLink::ScDdeLink( ScDocument* pD, SvStream& rStream, ScMultipleReadHeader& rHdr ) :
::sfx2::SvBaseLink(sfx2::LINKUPDATE_ALWAYS,FORMAT_STRING),
2000-09-18 23:16:46 +00:00
pDoc( pD ),
bNeedUpdate( false ),
pResult( NULL )
2000-09-18 23:16:46 +00:00
{
rHdr.StartEntry();
rtl_TextEncoding eCharSet = rStream.GetStreamCharSet();
aAppl = rStream.ReadUniOrByteString( eCharSet );
aTopic = rStream.ReadUniOrByteString( eCharSet );
aItem = rStream.ReadUniOrByteString( eCharSet );
2000-09-18 23:16:46 +00:00
sal_Bool bHasValue;
2000-09-18 23:16:46 +00:00
rStream >> bHasValue;
if ( bHasValue )
pResult = new ScMatrix(0, 0);
2000-09-18 23:16:46 +00:00
if (rHdr.BytesLeft()) // neu in 388b und der 364w (RealTime-Client) Version
rStream >> nMode;
else
nMode = SC_DDE_DEFAULT;
rHdr.EndEntry();
}
void ScDdeLink::Store( SvStream& rStream, ScMultipleWriteHeader& rHdr ) const
{
rHdr.StartEntry();
rtl_TextEncoding eCharSet = rStream.GetStreamCharSet();
rStream.WriteUniOrByteString( aAppl, eCharSet );
rStream.WriteUniOrByteString( aTopic, eCharSet );
rStream.WriteUniOrByteString( aItem, eCharSet );
2000-09-18 23:16:46 +00:00
sal_Bool bHasValue = ( pResult != NULL );
2000-09-18 23:16:46 +00:00
rStream << bHasValue;
if( rStream.GetVersion() > SOFFICE_FILEFORMAT_40 ) // nicht bei 4.0 Export
rStream << nMode; // seit 388b
// Links mit Mode != SC_DDE_DEFAULT werden bei 4.0 Export komplett weggelassen
// (aus ScDocument::SaveDdeLinks)
rHdr.EndEntry();
}
sfx2::SvBaseLink::UpdateResult ScDdeLink::DataChanged(
const String& rMimeType, const ::com::sun::star::uno::Any & rValue )
2000-09-18 23:16:46 +00:00
{
// wir koennen nur Strings...
if ( FORMAT_STRING != SotExchange::GetFormatIdFromMimeType( rMimeType ))
return SUCCESS;
2000-09-18 23:16:46 +00:00
String aLinkStr;
ScByteSequenceToString::GetString( aLinkStr, rValue, DDE_TXT_ENCODING );
2012-01-30 11:31:05 +00:00
aLinkStr = convertLineEnd(aLinkStr, LINEEND_LF);
2000-09-18 23:16:46 +00:00
// wenn String mit Zeilenende aufhoert, streichen:
xub_StrLen nLen = aLinkStr.Len();
if (nLen && aLinkStr.GetChar(nLen-1) == '\n')
aLinkStr.Erase(nLen-1);
String aLine;
SCSIZE nCols = 1; // Leerstring -> eine leere Zelle
SCSIZE nRows = 1;
2000-09-18 23:16:46 +00:00
if (aLinkStr.Len())
{
nRows = static_cast<SCSIZE>(comphelper::string::getTokenCount(aLinkStr, '\n'));
2000-09-18 23:16:46 +00:00
aLine = aLinkStr.GetToken( 0, '\n' );
if (aLine.Len())
nCols = static_cast<SCSIZE>(comphelper::string::getTokenCount(aLine, '\t'));
2000-09-18 23:16:46 +00:00
}
if (!nRows || !nCols) // keine Daten
{
pResult.reset();
2000-09-18 23:16:46 +00:00
}
else // Daten aufteilen
{
// Matrix immer neu anlegen, damit bIsString nicht durcheinanderkommt
pResult = new ScMatrix(nCols, nRows, 0.0);
2000-09-18 23:16:46 +00:00
SvNumberFormatter* pFormatter = pDoc->GetFormatTable();
// nMode bestimmt, wie der Text interpretiert wird (#44455#/#49783#):
// SC_DDE_DEFAULT - Zahlformat aus Zellvorlage "Standard"
// SC_DDE_ENGLISH - Standard-Zahlformat fuer English/US
// SC_DDE_TEXT - ohne NumberFormatter direkt als String
sal_uLong nStdFormat = 0;
2000-09-18 23:16:46 +00:00
if ( nMode == SC_DDE_DEFAULT )
{
ScPatternAttr* pDefPattern = pDoc->GetDefPattern(); // enthaelt Standard-Vorlage
if ( pDefPattern )
nStdFormat = pDefPattern->GetNumberFormat( pFormatter );
}
else if ( nMode == SC_DDE_ENGLISH )
nStdFormat = pFormatter->GetStandardIndex(LANGUAGE_ENGLISH_US);
String aEntry;
for (SCSIZE nR=0; nR<nRows; nR++)
2000-09-18 23:16:46 +00:00
{
aLine = aLinkStr.GetToken( (xub_StrLen) nR, '\n' );
for (SCSIZE nC=0; nC<nCols; nC++)
2000-09-18 23:16:46 +00:00
{
aEntry = aLine.GetToken( (xub_StrLen) nC, '\t' );
sal_uInt32 nIndex = nStdFormat;
double fVal = double();
2000-09-18 23:16:46 +00:00
if ( nMode != SC_DDE_TEXT && pFormatter->IsNumberFormat( aEntry, nIndex, fVal ) )
pResult->PutDouble( fVal, nC, nR );
else if (aEntry.Len() == 0)
// empty cell
pResult->PutEmpty(nC, nR);
2000-09-18 23:16:46 +00:00
else
pResult->PutString( aEntry, nC, nR );
}
}
}
// Es hat sich was getan...
if (HasListeners())
{
Broadcast( ScHint( SC_HINT_DATACHANGED, ScAddress(), NULL ) );
2000-09-18 23:16:46 +00:00
pDoc->TrackFormulas(); // muss sofort passieren
pDoc->StartTrackTimer();
// StartTrackTimer ruft asynchron TrackFormulas, Broadcast(FID_DATACHANGED),
// ResetChanged, SetModified und Invalidate(SID_SAVEDOC/SID_DOC_MODIFIED)
// TrackFormulas zusaetzlich nochmal sofort, damit nicht z.B. durch IdleCalc
// eine Formel berechnet wird, die noch im FormulaTrack steht (#61676#)
// notify Uno objects (for XRefreshListener)
// must be after TrackFormulas
//! do this asynchronously?
ScLinkRefreshedHint aHint;
aHint.SetDdeLink( aAppl, aTopic, aItem, nMode );
pDoc->BroadcastUno( aHint );
}
return SUCCESS;
2000-09-18 23:16:46 +00:00
}
2010-12-11 23:25:30 +01:00
void ScDdeLink::ListenersGone()
2000-09-18 23:16:46 +00:00
{
bool bWas = bIsInUpdate;
bIsInUpdate = true; // Remove() kann Reschedule ausloesen??!?
2000-09-18 23:16:46 +00:00
ScDocument* pStackDoc = pDoc; // member pDoc can't be used after removing the link
2010-01-13 22:25:07 +01:00
sfx2::LinkManager* pLinkMgr = pDoc->GetLinkManager();
pLinkMgr->Remove( this); // deletes this
2000-09-18 23:16:46 +00:00
if ( pLinkMgr->GetLinks().empty() ) // letzten geloescht ?
2000-09-18 23:16:46 +00:00
{
SfxBindings* pBindings = pStackDoc->GetViewBindings(); // don't use member pDoc!
2000-09-22 17:57:10 +00:00
if (pBindings)
pBindings->Invalidate( SID_LINKS );
2000-09-18 23:16:46 +00:00
}
bIsInUpdate = bWas;
}
void ScDdeLink::TryUpdate()
{
if (bIsInUpdate)
bNeedUpdate = true; // kann jetzt nicht ausgefuehrt werden
2000-09-18 23:16:46 +00:00
else
{
bIsInUpdate = true;
pDoc->IncInDdeLinkUpdate();
2000-09-18 23:16:46 +00:00
Update();
pDoc->DecInDdeLinkUpdate();
bIsInUpdate = false;
bNeedUpdate = false;
2000-09-18 23:16:46 +00:00
}
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */