Files
libreoffice/sc/source/ui/docshell/servobj.cxx

258 lines
8.0 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 16:07:07 +00:00
#include <sot/formats.hxx>
#include <sfx2/app.hxx>
2010-01-13 22:25:07 +01:00
#include <sfx2/linkmgr.hxx>
2000-09-18 16:07:07 +00:00
#include "servobj.hxx"
#include "docsh.hxx"
#include "impex.hxx"
#include "brdcst.hxx"
#include "rangenam.hxx"
#include "sc.hrc"
2000-09-18 16:07:07 +00:00
using namespace formula;
static bool lcl_FillRangeFromName( ScRange& rRange, ScDocShell* pDocSh, const OUString& rName )
2000-09-18 16:07:07 +00:00
{
if (pDocSh)
{
ScDocument& rDoc = pDocSh->GetDocument();
ScRangeName* pNames = rDoc.GetRangeName();
2000-09-18 16:07:07 +00:00
if (pNames)
{
const ScRangeData* pData = pNames->findByUpperName(ScGlobal::pCharClass->uppercase(rName));
if (pData)
2000-09-18 16:07:07 +00:00
{
if ( pData->IsValidReference( rRange ) )
return true;
2000-09-18 16:07:07 +00:00
}
}
}
return false;
2000-09-18 16:07:07 +00:00
}
ScServerObjectSvtListenerForwarder::ScServerObjectSvtListenerForwarder(
ScServerObject* pObjP)
: pObj(pObjP)
{
}
ScServerObjectSvtListenerForwarder::~ScServerObjectSvtListenerForwarder()
{
//! do NOT access pObj
}
void ScServerObjectSvtListenerForwarder::Notify( const SfxHint& rHint )
{
pObj->Notify( aBroadcaster, rHint);
}
ScServerObject::ScServerObject( ScDocShell* pShell, const OUString& rItem ) :
aForwarder( this ),
2000-09-18 16:07:07 +00:00
pDocSh( pShell ),
bRefreshListener( false )
2000-09-18 16:07:07 +00:00
{
// parse item string
if ( lcl_FillRangeFromName( aRange, pDocSh, rItem ) )
{
aItemStr = rItem; // must be parsed again on ref update
}
else
{
// parse ref
ScDocument& rDoc = pDocSh->GetDocument();
SCTAB nTab = ScDocShell::GetCurTab();
2000-09-18 16:07:07 +00:00
aRange.aStart.SetTab( nTab );
// For DDE link, we always must parse references using OOO A1 convention.
if ( aRange.Parse( rItem, &rDoc, FormulaGrammar::CONV_OOO ) & SCA_VALID )
2000-09-18 16:07:07 +00:00
{
// area reference
}
else if ( aRange.aStart.Parse( rItem, &rDoc, FormulaGrammar::CONV_OOO ) & SCA_VALID )
2000-09-18 16:07:07 +00:00
{
// cell reference
aRange.aEnd = aRange.aStart;
}
else
{
2011-03-01 19:05:02 +01:00
OSL_FAIL("ScServerObject: invalid item");
2000-09-18 16:07:07 +00:00
}
}
pDocSh->GetDocument().GetLinkManager()->InsertServer( this );
pDocSh->GetDocument().StartListeningArea( aRange, false, &aForwarder );
2000-09-18 16:07:07 +00:00
StartListening(*pDocSh); // um mitzubekommen, wenn die DocShell geloescht wird
StartListening(*SfxGetpApp()); // for SC_HINT_AREAS_CHANGED
2000-09-18 16:07:07 +00:00
}
2010-12-11 23:25:30 +01:00
ScServerObject::~ScServerObject()
2000-09-18 16:07:07 +00:00
{
Clear();
}
void ScServerObject::Clear()
{
if (pDocSh)
{
ScDocShell* pTemp = pDocSh;
pDocSh = NULL;
pTemp->GetDocument().EndListeningArea(aRange, false, &aForwarder);
pTemp->GetDocument().GetLinkManager()->RemoveServer( this );
2000-09-18 16:07:07 +00:00
EndListening(*pTemp);
EndListening(*SfxGetpApp());
2000-09-18 16:07:07 +00:00
}
}
void ScServerObject::EndListeningAll()
{
aForwarder.EndListeningAll();
SfxListener::EndListeningAll();
}
bool ScServerObject::GetData(
::com::sun::star::uno::Any & rData /*out param*/,
const OUString & rMimeType, bool /* bSynchron */ )
2000-09-18 16:07:07 +00:00
{
if (!pDocSh)
return false;
2000-09-18 16:07:07 +00:00
// named ranges may have changed -> update aRange
if ( !aItemStr.isEmpty() )
2000-09-18 16:07:07 +00:00
{
ScRange aNew;
if ( lcl_FillRangeFromName( aNew, pDocSh, aItemStr ) && aNew != aRange )
{
aRange = aNew;
bRefreshListener = true;
2000-09-18 16:07:07 +00:00
}
}
if ( bRefreshListener )
{
// refresh the listeners now (this is called from a timer)
EndListeningAll();
pDocSh->GetDocument().StartListeningArea( aRange, false, &aForwarder );
2000-09-18 16:07:07 +00:00
StartListening(*pDocSh);
StartListening(*SfxGetpApp());
bRefreshListener = false;
2000-09-18 16:07:07 +00:00
}
OUString aDdeTextFmt = pDocSh->GetDdeTextFmt();
ScDocument& rDoc = pDocSh->GetDocument();
2000-09-18 16:07:07 +00:00
if( FORMAT_STRING == SotExchange::GetFormatIdFromMimeType( rMimeType ))
2000-09-18 16:07:07 +00:00
{
ScImportExport aObj( &rDoc, aRange );
if( aDdeTextFmt[0] == 'F' )
aObj.SetFormulas( true );
if( aDdeTextFmt == "SYLK" || aDdeTextFmt == "FSYLK" )
2000-09-18 16:07:07 +00:00
{
OString aByteData;
if( aObj.ExportByteString( aByteData, osl_getThreadTextEncoding(), SOT_FORMATSTR_ID_SYLK ) )
2000-09-18 16:07:07 +00:00
{
rData <<= ::com::sun::star::uno::Sequence< sal_Int8 >(
reinterpret_cast<const sal_Int8*>(aByteData.getStr()),
2011-11-10 22:11:00 +00:00
aByteData.getLength() + 1 );
return true;
2000-09-18 16:07:07 +00:00
}
return false;
2000-09-18 16:07:07 +00:00
}
if( aDdeTextFmt == "CSV" || aDdeTextFmt == "FCSV" )
2000-09-18 16:07:07 +00:00
aObj.SetSeparator( ',' );
aObj.SetExportTextOptions( ScExportTextOptions( ScExportTextOptions::ToSpace, ' ', false ) );
return aObj.ExportData( rMimeType, rData );
2000-09-18 16:07:07 +00:00
}
ScImportExport aObj( &rDoc, aRange );
aObj.SetExportTextOptions( ScExportTextOptions( ScExportTextOptions::ToSpace, ' ', false ) );
2000-09-18 16:07:07 +00:00
if( aObj.IsRef() )
return aObj.ExportData( rMimeType, rData );
return false;
2000-09-18 16:07:07 +00:00
}
2010-12-11 23:25:30 +01:00
void ScServerObject::Notify( SfxBroadcaster& rBC, const SfxHint& rHint )
2000-09-18 16:07:07 +00:00
{
bool bDataChanged = false;
2000-09-18 16:07:07 +00:00
// DocShell can't be tested via type info, because SFX_HINT_DYING comes from the dtor
if ( &rBC == pDocSh )
2000-09-18 16:07:07 +00:00
{
// from DocShell, only SFX_HINT_DYING is interesting
const SfxSimpleHint* pSimpleHint = dynamic_cast<const SfxSimpleHint*>( &rHint );
if ( pSimpleHint && pSimpleHint->GetId() == SFX_HINT_DYING )
{
pDocSh = NULL;
EndListening(*SfxGetpApp());
// don't access DocShell anymore for EndListening etc.
}
2000-09-18 16:07:07 +00:00
}
else if (rBC.ISA(SfxApplication))
{
const SfxSimpleHint* pSimpleHint = dynamic_cast<const SfxSimpleHint*>( &rHint );
if ( !aItemStr.isEmpty() && pSimpleHint &&
pSimpleHint->GetId() == SC_HINT_AREAS_CHANGED )
2000-09-18 16:07:07 +00:00
{
// check if named range was modified
ScRange aNew;
if ( lcl_FillRangeFromName( aNew, pDocSh, aItemStr ) && aNew != aRange )
bDataChanged = true;
2000-09-18 16:07:07 +00:00
}
}
else
{
// must be from Area broadcasters
const ScHint* pScHint = dynamic_cast<const ScHint*>( &rHint );
if (pScHint && (pScHint->GetId() & SC_HINT_DATACHANGED))
bDataChanged = true;
else if ( dynamic_cast<const ScAreaChangedHint*>(&rHint) ) // position of broadcaster changed
2000-09-18 16:07:07 +00:00
{
ScRange aNewRange = static_cast<const ScAreaChangedHint&>(rHint).GetRange();
2000-09-18 16:07:07 +00:00
if ( aRange != aNewRange )
{
bRefreshListener = true;
bDataChanged = true;
2000-09-18 16:07:07 +00:00
}
}
else if ( dynamic_cast<const SfxSimpleHint*>(&rHint) )
2000-09-18 16:07:07 +00:00
{
sal_uLong nId = static_cast<const SfxSimpleHint&>(rHint).GetId();
2000-09-18 16:07:07 +00:00
if (nId == SFX_HINT_DYING)
{
// If the range is being deleted, listening must be restarted
// after the deletion is complete (done in GetData)
bRefreshListener = true;
bDataChanged = true;
2000-09-18 16:07:07 +00:00
}
}
}
if ( bDataChanged && HasDataLinks() )
SvLinkSource::NotifyDataChanged();
2000-09-18 16:07:07 +00:00
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */