/* -*- 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 "scitems.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "document.hxx" #include "table.hxx" #include "attrib.hxx" #include "patattr.hxx" #include "rangenam.hxx" #include "dbdata.hxx" #include "pivot.hxx" #include "docpool.hxx" #include "stlpool.hxx" #include "stlsheet.hxx" #include "globstr.hrc" #include "chartarr.hxx" #include "chartlock.hxx" #include "rechead.hxx" #include "global.hxx" #include "brdcst.hxx" #include "bcaslot.hxx" #include "adiasync.hxx" #include "addinlis.hxx" #include "chartlis.hxx" #include "markdata.hxx" #include "conditio.hxx" #include "colorscale.hxx" #include "validat.hxx" #include "progress.hxx" #include "detdata.hxx" #include "sc.hrc" #include "ddelink.hxx" #include "chgtrack.hxx" #include "chgviset.hxx" #include "editutil.hxx" #include "hints.hxx" #include "dpobject.hxx" #include "scrdata.hxx" #include "poolhelp.hxx" #include "unoreflist.hxx" #include "listenercalls.hxx" #include "recursionhelper.hxx" #include "lookupcache.hxx" #include "externalrefmgr.hxx" #include "appoptio.hxx" #include "scmod.hxx" #include "../../ui/inc/viewutil.hxx" #include "tabprotection.hxx" #include "formulaparserpool.hxx" #include "clipparam.hxx" #include "macromgr.hxx" #include "formulacell.hxx" #include "clipcontext.hxx" #include "refupdatecontext.hxx" #include "refreshtimerprotector.hxx" #include "scopetools.hxx" #include "formulagroup.hxx" #include "documentlinkmgr.hxx" #include "interpre.hxx" #include #include "docsh.hxx" #include using namespace com::sun::star; // pImpl because including lookupcache.hxx in document.hxx isn't wanted, and // dtor plus helpers are convenient. struct ScLookupCacheMapImpl { ScLookupCacheMap aCacheMap; ~ScLookupCacheMapImpl() { freeCaches(); } void clear() { freeCaches(); // free mapping ScLookupCacheMap aTmp; aCacheMap.swap( aTmp); } private: void freeCaches() { for (ScLookupCacheMap::iterator it( aCacheMap.begin()); it != aCacheMap.end(); ++it) delete (*it).second; } }; // STATIC DATA ----------------------------------------------------------- ScDocument::ScDocument( ScDocumentMode eMode, SfxObjectShell* pDocShell ) : mpCellStringPool(new svl::SharedStringPool(ScGlobal::pCharClass)), mpFormulaGroupCxt(NULL), maCalcConfig( ScInterpreter::GetGlobalConfig()), mpUndoManager( NULL ), pEditEngine( NULL ), pNoteEngine( NULL ), pShell( pDocShell ), pPrinter( NULL ), pVirtualDevice_100th_mm( NULL ), pDrawLayer( NULL ), pValidationList( NULL ), pFormatExchangeList( NULL ), pRangeName(NULL), pDPCollection( NULL ), pFormulaTree( NULL ), pEOFormulaTree( NULL ), pFormulaTrack( NULL ), pEOFormulaTrack( NULL ), pClipData( NULL ), pDetOpList(NULL), pChangeTrack( NULL ), pUnoBroadcaster( NULL ), pUnoListenerCalls( NULL ), pUnoRefUndoList( NULL ), pChangeViewSettings( NULL ), pScriptTypeData( NULL ), pCacheFieldEditEngine( NULL ), pViewOptions( NULL ), pDocOptions( NULL ), pExtDocOptions( NULL ), pConsolidateDlgData( NULL ), pRecursionHelper( NULL ), pAutoNameCache( NULL ), pLookupCacheMapImpl( NULL ), pPreviewFont( NULL ), pPreviewCellStyle( NULL ), nUnoObjectId( 0 ), nRangeOverflowType( 0 ), aCurTextWidthCalcPos(MAXCOL,0,0), aTableOpList( 0 ), nFormulaCodeInTree(0), nXMLImportedFormulaCount( 0 ), nInterpretLevel(0), nMacroInterpretLevel(0), nInterpreterTableOpLevel(0), nSrcVer( SC_CURRENT_VERSION ), nSrcMaxRow( MAXROW ), nFormulaTrackCount(0), bHardRecalcState(false), nVisibleTab( 0 ), eLinkMode(LM_UNKNOWN), bAutoCalc( eMode == SCDOCMODE_DOCUMENT ), bAutoCalcShellDisabled( false ), bForcedFormulaPending( false ), bCalculatingFormulaTree( false ), bIsClip( eMode == SCDOCMODE_CLIP ), bIsUndo( eMode == SCDOCMODE_UNDO ), bIsVisible( false ), bIsEmbedded( false ), bInsertingFromOtherDoc( false ), bLoadingMedium( false ), bImportingXML( false ), bCalcingAfterLoad( false ), bNoListening( false ), mbIdleEnabled(true), bInLinkUpdate( false ), bChartListenerCollectionNeedsUpdate( false ), bHasForcedFormulas( false ), bInDtorClear( false ), bExpandRefs( false ), bDetectiveDirty( false ), nMacroCallMode( SC_MACROCALL_ALLOWED ), bHasMacroFunc( false ), nVisSpellState( 0 ), nAsianCompression(SC_ASIANCOMPRESSION_INVALID), nAsianKerning(SC_ASIANKERNING_INVALID), bPastingDrawFromOtherDoc( false ), nInDdeLinkUpdate( 0 ), bInUnoBroadcast( false ), bInUnoListenerCall( false ), eGrammar( formula::FormulaGrammar::GRAM_NATIVE ), bStyleSheetUsageInvalid( true ), mbUndoEnabled( true ), mbAdjustHeightEnabled( true ), mbExecuteLinkEnabled( true ), mbChangeReadOnlyEnabled( false ), mbStreamValidLocked( false ), mbUserInteractionEnabled(true), mnNamedRangesLockCount(0), mbUseEmbedFonts(false) { SetStorageGrammar( formula::FormulaGrammar::GRAM_STORAGE_DEFAULT); eSrcSet = osl_getThreadTextEncoding(); if ( eMode == SCDOCMODE_DOCUMENT ) { xPoolHelper = new ScPoolHelper( this ); pBASM = new ScBroadcastAreaSlotMachine( this ); pChartListenerCollection = new ScChartListenerCollection( this ); pRefreshTimerControl = new ScRefreshTimerControl; } else { pBASM = NULL; pChartListenerCollection = NULL; pRefreshTimerControl = NULL; } pDBCollection = new ScDBCollection(this); pSelectionAttr = NULL; pChartCollection = new ScChartCollection; apTemporaryChartLock.reset( new ScTemporaryChartLock(this) ); xColNameRanges = new ScRangePairList; xRowNameRanges = new ScRangePairList; ImplCreateOptions(); // languages for a visible document are set by docshell later (from options) SetLanguage( ScGlobal::eLnge, ScGlobal::eLnge, ScGlobal::eLnge ); aTrackTimer.SetTimeoutHdl( LINK( this, ScDocument, TrackTimeHdl ) ); aTrackTimer.SetTimeout( 100 ); } sfx2::LinkManager* ScDocument::GetLinkManager() { return GetDocLinkManager().getLinkManager(true); } const sfx2::LinkManager* ScDocument::GetLinkManager() const { return GetDocLinkManager().getExistingLinkManager(); } sc::DocumentLinkManager& ScDocument::GetDocLinkManager() { if (!mpDocLinkMgr) mpDocLinkMgr.reset(new sc::DocumentLinkManager(*this, pShell)); return *mpDocLinkMgr; } const sc::DocumentLinkManager& ScDocument::GetDocLinkManager() const { return const_cast(this)->GetDocLinkManager(); } void ScDocument::SetStorageGrammar( formula::FormulaGrammar::Grammar eGram ) { OSL_PRECOND( eGram == formula::FormulaGrammar::GRAM_ODFF || eGram == formula::FormulaGrammar::GRAM_PODF, "ScDocument::SetStorageGrammar: wrong storage grammar"); eStorageGrammar = eGram; // FIXME: the XML import shouldn't strip brackets, the compiler should // digest them instead, which could also speedup reference recognition // during import. eXmlImportGrammar = formula::FormulaGrammar::mergeToGrammar( eGram, formula::FormulaGrammar::CONV_OOO); } void ScDocument::SetDocVisible( bool bSet ) { // called from view ctor - only for a visible document, // each new sheet's RTL flag is initialized from the locale bIsVisible = bSet; } sal_uInt32 ScDocument::GetDocumentID() const { const ScDocument* pThis = this; sal_uInt32 nCrc = rtl_crc32( 0, &pThis, sizeof(ScDocument*) ); // the this pointer only might not be sufficient nCrc = rtl_crc32( nCrc, &pShell, sizeof(SfxObjectShell*) ); return nCrc; } void ScDocument::StartChangeTracking() { if (!pChangeTrack) pChangeTrack = new ScChangeTrack( this ); } void ScDocument::EndChangeTracking() { delete pChangeTrack; pChangeTrack = NULL; } void ScDocument::SetChangeTrack( ScChangeTrack* pTrack ) { OSL_ENSURE( pTrack->GetDocument() == this, "SetChangeTrack: different documents" ); if ( !pTrack || pTrack == pChangeTrack || pTrack->GetDocument() != this ) return ; EndChangeTracking(); pChangeTrack = pTrack; } IMPL_LINK_NOARG(ScDocument, TrackTimeHdl) { if ( ScDdeLink::IsInUpdate() ) // do not nest { aTrackTimer.Start(); // try again later } else if (pShell) // execute { TrackFormulas(); pShell->Broadcast( SfxSimpleHint( FID_DATACHANGED ) ); // modified... if (!pShell->IsModified()) { pShell->SetModified( true ); SfxBindings* pBindings = GetViewBindings(); if (pBindings) { pBindings->Invalidate( SID_SAVEDOC ); pBindings->Invalidate( SID_DOC_MODIFIED ); } } } return 0; } void ScDocument::SetExpandRefs( bool bVal ) { bExpandRefs = bVal; } void ScDocument::StartTrackTimer() { if (!aTrackTimer.IsActive()) // do not postpone for forever aTrackTimer.Start(); } ScDocument::~ScDocument() { OSL_PRECOND( !bInLinkUpdate, "bInLinkUpdate in dtor" ); bInDtorClear = true; // first of all disable all refresh timers by deleting the control if ( pRefreshTimerControl ) { // To be sure there isn't anything running do it with a protector, // this ensures also that nothing needs the control anymore. ScRefreshTimerProtector aProt( GetRefreshTimerControlAddress() ); delete pRefreshTimerControl, pRefreshTimerControl = NULL; } mxFormulaParserPool.reset(); // Destroy the external ref mgr instance here because it has a timer // which needs to be stopped before the app closes. pExternalRefMgr.reset(); ScAddInAsync::RemoveDocument( this ); ScAddInListener::RemoveDocument( this ); DELETEZ( pChartListenerCollection); // before pBASM because of potential Listener! DELETEZ( pLookupCacheMapImpl); // before pBASM because of listeners // destroy BroadcastAreas first to avoid un-needed Single-EndListenings of Formula-Cells delete pBASM; // BroadcastAreaSlotMachine pBASM = NULL; delete pUnoBroadcaster; // broadcasted nochmal SFX_HINT_DYING pUnoBroadcaster = NULL; delete pUnoRefUndoList; delete pUnoListenerCalls; Clear( true ); // true = from destructor (needed for SdrModel::ClearModel) if (pValidationList) { for( ScValidationDataList::iterator it = pValidationList->begin(); it != pValidationList->end(); ++it ) delete *it; pValidationList->clear(); DELETEZ(pValidationList); } delete pRangeName; delete pDBCollection; delete pSelectionAttr; apTemporaryChartLock.reset(); delete pChartCollection; DeleteDrawLayer(); delete pFormatExchangeList; delete pPrinter; ImplDeleteOptions(); delete pConsolidateDlgData; delete pClipData; delete pDetOpList; // also deletes entries delete pChangeTrack; delete pEditEngine; delete pNoteEngine; delete pChangeViewSettings; // and delete delete pVirtualDevice_100th_mm; delete pDPCollection; // delete the EditEngine before destroying the xPoolHelper delete pCacheFieldEditEngine; if ( xPoolHelper.is() && !bIsClip ) xPoolHelper->SourceDocumentGone(); xPoolHelper.clear(); delete pScriptTypeData; delete pRecursionHelper; delete pPreviewFont; SAL_WARN_IF( pAutoNameCache, "sc.core", "AutoNameCache still set in dtor" ); mpFormulaGroupCxt.reset(); mpCellStringPool.reset(); } void ScDocument::InitClipPtrs( ScDocument* pSourceDoc ) { OSL_ENSURE(bIsClip, "InitClipPtrs and not bIsClip"); if (pValidationList) { for(ScValidationDataList::iterator it = pValidationList->begin(); it != pValidationList->end(); ++it ) delete *it; pValidationList->clear(); DELETEZ(pValidationList); } Clear(); SharePooledResources(pSourceDoc); // conditional Formats / validations // TODO: Copy Templates? const ScValidationDataList* pSourceValid = pSourceDoc->pValidationList; if ( pSourceValid ) pValidationList = new ScValidationDataList(this, *pSourceValid); // store Links in Stream delete pClipData; if (pSourceDoc->GetDocLinkManager().hasDdeLinks()) { pClipData = new SvMemoryStream; pSourceDoc->SaveDdeLinks(*pClipData); } else pClipData = NULL; // Options pointers exist (ImplCreateOptions) for any document. // Must be copied for correct results in OLE objects (#i42666#). SetDocOptions( pSourceDoc->GetDocOptions() ); SetViewOptions( pSourceDoc->GetViewOptions() ); } SvNumberFormatter* ScDocument::GetFormatTable() const { return xPoolHelper->GetFormTable(); } SvNumberFormatter* ScDocument::CreateFormatTable() const { return xPoolHelper->CreateNumberFormatter(); } SfxItemPool* ScDocument::GetEditPool() const { return xPoolHelper->GetEditPool(); } SfxItemPool* ScDocument::GetEnginePool() const { return xPoolHelper->GetEnginePool(); } ScFieldEditEngine& ScDocument::GetEditEngine() { if ( !pEditEngine ) { pEditEngine = new ScFieldEditEngine(this, GetEnginePool(), GetEditPool()); pEditEngine->SetUpdateMode( false ); pEditEngine->EnableUndo( false ); pEditEngine->SetRefMapMode( MAP_100TH_MM ); ApplyAsianEditSettings( *pEditEngine ); } return *pEditEngine; } ScNoteEditEngine& ScDocument::GetNoteEngine() { if ( !pNoteEngine ) { pNoteEngine = new ScNoteEditEngine( GetEnginePool(), GetEditPool() ); pNoteEngine->SetUpdateMode( false ); pNoteEngine->EnableUndo( false ); pNoteEngine->SetRefMapMode( MAP_100TH_MM ); ApplyAsianEditSettings( *pNoteEngine ); const SfxItemSet& rItemSet = GetDefPattern()->GetItemSet(); SfxItemSet* pEEItemSet = new SfxItemSet( pNoteEngine->GetEmptyItemSet() ); ScPatternAttr::FillToEditItemSet( *pEEItemSet, rItemSet ); pNoteEngine->SetDefaults( pEEItemSet ); // edit engine takes ownership } return *pNoteEngine; } void ScDocument::ResetClip( ScDocument* pSourceDoc, const ScMarkData* pMarks ) { if (bIsClip) { InitClipPtrs(pSourceDoc); for (SCTAB i = 0; i < static_cast(pSourceDoc->maTabs.size()); i++) if (pSourceDoc->maTabs[i]) if (!pMarks || pMarks->GetTableSelect(i)) { OUString aString; pSourceDoc->maTabs[i]->GetName(aString); if ( i < static_cast(maTabs.size()) ) { maTabs[i] = new ScTable(this, i, aString); } else { if( i > static_cast(maTabs.size()) ) { maTabs.resize(i, NULL ); } maTabs.push_back(new ScTable(this, i, aString)); } maTabs[i]->SetLayoutRTL( pSourceDoc->maTabs[i]->IsLayoutRTL() ); } } else { OSL_FAIL("ResetClip"); } } void ScDocument::ResetClip( ScDocument* pSourceDoc, SCTAB nTab ) { if (bIsClip) { InitClipPtrs(pSourceDoc); if (nTab >= static_cast(maTabs.size())) { maTabs.resize(nTab+1, NULL ); } maTabs[nTab] = new ScTable(this, nTab, OUString("baeh")); if (nTab < static_cast(pSourceDoc->maTabs.size()) && pSourceDoc->maTabs[nTab]) maTabs[nTab]->SetLayoutRTL( pSourceDoc->maTabs[nTab]->IsLayoutRTL() ); } else { OSL_FAIL("ResetClip"); } } void ScDocument::EnsureTable( SCTAB nTab ) { bool bExtras = !bIsUndo; // Column-Widths, Row-Heights, Flags if (static_cast(nTab) >= maTabs.size()) maTabs.resize(nTab+1, NULL); if (!maTabs[nTab]) maTabs[nTab] = new ScTable(this, nTab, "temp", bExtras, bExtras); } ScRefCellValue ScDocument::GetRefCellValue( const ScAddress& rPos ) { if (!TableExists(rPos.Tab())) return ScRefCellValue(); // empty return maTabs[rPos.Tab()]->GetRefCellValue(rPos.Col(), rPos.Row()); } svl::SharedStringPool& ScDocument::GetSharedStringPool() { return *mpCellStringPool; } const svl::SharedStringPool& ScDocument::GetSharedStringPool() const { return *mpCellStringPool; } bool ScDocument::GetPrintArea( SCTAB nTab, SCCOL& rEndCol, SCROW& rEndRow, bool bNotes ) const { if (ValidTab(nTab) && nTab < static_cast(maTabs.size()) && maTabs[nTab]) { bool bAny = maTabs[nTab]->GetPrintArea( rEndCol, rEndRow, bNotes ); if (pDrawLayer) { ScRange aDrawRange(0,0,nTab, MAXCOL,MAXROW,nTab); if (DrawGetPrintArea( aDrawRange, true, true )) { if (aDrawRange.aEnd.Col()>rEndCol) rEndCol=aDrawRange.aEnd.Col(); if (aDrawRange.aEnd.Row()>rEndRow) rEndRow=aDrawRange.aEnd.Row(); bAny = true; } } return bAny; } rEndCol = 0; rEndRow = 0; return false; } bool ScDocument::GetPrintAreaHor( SCTAB nTab, SCROW nStartRow, SCROW nEndRow, SCCOL& rEndCol, bool bNotes ) const { if (ValidTab(nTab) && nTab < static_cast(maTabs.size()) && maTabs[nTab]) { bool bAny = maTabs[nTab]->GetPrintAreaHor( nStartRow, nEndRow, rEndCol, bNotes ); if (pDrawLayer) { ScRange aDrawRange(0,nStartRow,nTab, MAXCOL,nEndRow,nTab); if (DrawGetPrintArea( aDrawRange, true, false )) { if (aDrawRange.aEnd.Col()>rEndCol) rEndCol=aDrawRange.aEnd.Col(); bAny = true; } } return bAny; } rEndCol = 0; return false; } bool ScDocument::GetPrintAreaVer( SCTAB nTab, SCCOL nStartCol, SCCOL nEndCol, SCROW& rEndRow, bool bNotes ) const { if (ValidTab(nTab) && nTab < static_cast(maTabs.size()) && maTabs[nTab]) { bool bAny = maTabs[nTab]->GetPrintAreaVer( nStartCol, nEndCol, rEndRow, bNotes ); if (pDrawLayer) { ScRange aDrawRange(nStartCol,0,nTab, nEndCol,MAXROW,nTab); if (DrawGetPrintArea( aDrawRange, false, true )) { if (aDrawRange.aEnd.Row()>rEndRow) rEndRow=aDrawRange.aEnd.Row(); bAny = true; } } return bAny; } rEndRow = 0; return false; } bool ScDocument::GetDataStart( SCTAB nTab, SCCOL& rStartCol, SCROW& rStartRow ) const { if (ValidTab(nTab) && nTab < static_cast(maTabs.size()) && maTabs[nTab]) { bool bAny = maTabs[nTab]->GetDataStart( rStartCol, rStartRow ); if (pDrawLayer) { ScRange aDrawRange(0,0,nTab, MAXCOL,MAXROW,nTab); if (DrawGetPrintArea( aDrawRange, true, true )) { if (aDrawRange.aStart.Col()(maTabs.size()); if(nTabCount < 2) return false; bool bValid = false; if (ValidTab(nOldPos) && nOldPos < nTabCount ) { if (maTabs[nOldPos]) { sc::AutoCalcSwitch aACSwitch(*this, false); SetNoListening( true ); if (nNewPos == SC_TAB_APPEND || nNewPos >= nTabCount) nNewPos = nTabCount-1; // Update Reference // TODO: combine with UpdateReference! sc::RefUpdateMoveTabContext aCxt(nOldPos, nNewPos); SCsTAB nDz = ((SCsTAB)nNewPos) - (SCsTAB)nOldPos; ScRange aSourceRange( 0,0,nOldPos, MAXCOL,MAXROW,nOldPos ); if (pRangeName) pRangeName->UpdateMoveTab(aCxt); pDBCollection->UpdateMoveTab( nOldPos, nNewPos ); xColNameRanges->UpdateReference( URM_REORDER, this, aSourceRange, 0,0,nDz ); xRowNameRanges->UpdateReference( URM_REORDER, this, aSourceRange, 0,0,nDz ); if (pDPCollection) pDPCollection->UpdateReference( URM_REORDER, aSourceRange, 0,0,nDz ); if (pDetOpList) pDetOpList->UpdateReference( this, URM_REORDER, aSourceRange, 0,0,nDz ); UpdateChartRef( URM_REORDER, 0,0,nOldPos, MAXCOL,MAXROW,nOldPos, 0,0,nDz ); UpdateRefAreaLinks( URM_REORDER, aSourceRange, 0,0,nDz ); if ( pValidationList ) pValidationList->UpdateMoveTab(aCxt); if ( pUnoBroadcaster ) pUnoBroadcaster->Broadcast( ScUpdateRefHint( URM_REORDER, aSourceRange, 0,0,nDz ) ); ScTable* pSaveTab = maTabs[nOldPos]; maTabs.erase(maTabs.begin()+nOldPos); maTabs.insert(maTabs.begin()+nNewPos, pSaveTab); TableContainer::iterator it = maTabs.begin(); for (SCTAB i = 0; i < nTabCount; i++) if (maTabs[i]) maTabs[i]->UpdateMoveTab(aCxt, i, pProgress); it = maTabs.begin(); for (; it != maTabs.end(); ++it) if (*it) (*it)->UpdateCompile(); SetNoListening( false ); StartAllListeners(); // sheet names of references may not be valid until sheet is moved pChartListenerCollection->UpdateScheduledSeriesRanges(); sc::SetFormulaDirtyContext aFormulaDirtyCxt; SetAllFormulasDirty(aFormulaDirtyCxt); if (pDrawLayer) DrawMovePage( static_cast(nOldPos), static_cast(nNewPos) ); bValid = true; } } return bValid; } bool ScDocument::CopyTab( SCTAB nOldPos, SCTAB nNewPos, const ScMarkData* pOnlyMarked ) { if (SC_TAB_APPEND == nNewPos || nNewPos >= static_cast(maTabs.size())) nNewPos = static_cast(maTabs.size()); OUString aName; GetName(nOldPos, aName); // check first if Prefix is valid; if not, then only avoid duplicates bool bPrefix = ValidTabName( aName ); OSL_ENSURE(bPrefix, "invalid table name"); SCTAB nDummy; CreateValidTabName(aName); bool bValid; if (bPrefix) bValid = ValidNewTabName(aName); else bValid = !GetTable( aName, nDummy ); sc::AutoCalcSwitch aACSwitch(*this, false); sc::RefUpdateInsertTabContext aCxt(nNewPos, 1); sc::StartListeningContext aSLCxt(*this); if (bValid) { if (nNewPos >= static_cast(maTabs.size())) { nNewPos = static_cast(maTabs.size()); maTabs.push_back(new ScTable(this, nNewPos, aName)); } else { if (ValidTab(nNewPos) && (nNewPos < static_cast(maTabs.size()))) { SetNoListening( true ); ScRange aRange( 0,0,nNewPos, MAXCOL,MAXROW,MAXTAB ); xColNameRanges->UpdateReference( URM_INSDEL, this, aRange, 0,0,1 ); xRowNameRanges->UpdateReference( URM_INSDEL, this, aRange, 0,0,1 ); if (pRangeName) pRangeName->UpdateInsertTab(aCxt); pDBCollection->UpdateReference( URM_INSDEL, 0,0,nNewPos, MAXCOL,MAXROW,MAXTAB, 0,0,1 ); if (pDPCollection) pDPCollection->UpdateReference( URM_INSDEL, aRange, 0,0,1 ); if (pDetOpList) pDetOpList->UpdateReference( this, URM_INSDEL, aRange, 0,0,1 ); UpdateChartRef( URM_INSDEL, 0,0,nNewPos, MAXCOL,MAXROW,MAXTAB, 0,0,1 ); UpdateRefAreaLinks( URM_INSDEL, aRange, 0,0,1 ); if ( pUnoBroadcaster ) pUnoBroadcaster->Broadcast( ScUpdateRefHint( URM_INSDEL, aRange, 0,0,1 ) ); SCTAB i; for (TableContainer::iterator it = maTabs.begin(); it != maTabs.end(); ++it) if (*it && it != (maTabs.begin() + nOldPos)) (*it)->UpdateInsertTab(aCxt); maTabs.push_back(NULL); for (i = static_cast(maTabs.size())-1; i > nNewPos; i--) maTabs[i] = maTabs[i - 1]; if (nNewPos <= nOldPos) nOldPos++; maTabs[nNewPos] = new ScTable(this, nNewPos, aName); bValid = true; for (TableContainer::iterator it = maTabs.begin(); it != maTabs.end(); ++it) if (*it && it != maTabs.begin()+nOldPos && it != maTabs.begin() + nNewPos) (*it)->UpdateCompile(); SetNoListening( false ); for (TableContainer::iterator it = maTabs.begin(); it != maTabs.end(); ++it) if (*it && it != maTabs.begin()+nOldPos && it != maTabs.begin()+nNewPos) (*it)->StartListeners(aSLCxt, true); if (pValidationList) pValidationList->UpdateInsertTab(aCxt); // sheet names of references may not be valid until sheet is copied pChartListenerCollection->UpdateScheduledSeriesRanges(); } else bValid = false; } } if (bValid) { SetNoListening( true ); // noch nicht bei CopyToTable/Insert sc::CopyToDocContext aCopyDocCxt(*this); maTabs[nOldPos]->CopyToTable(aCopyDocCxt, 0, 0, MAXCOL, MAXROW, IDF_ALL, (pOnlyMarked != NULL), maTabs[nNewPos], pOnlyMarked ); maTabs[nNewPos]->SetTabBgColor(maTabs[nOldPos]->GetTabBgColor()); SCTAB nDz = nNewPos - nOldPos; sc::RefUpdateContext aRefCxt(*this); aRefCxt.meMode = URM_COPY; aRefCxt.maRange = ScRange(0, 0, nNewPos, MAXCOL, MAXROW, nNewPos); aRefCxt.mnTabDelta = nDz; maTabs[nNewPos]->UpdateReference(aRefCxt, NULL); maTabs[nNewPos]->UpdateInsertTabAbs(nNewPos); // move all paragraphs up by one!! maTabs[nOldPos]->UpdateInsertTab(aCxt); maTabs[nOldPos]->UpdateCompile(); maTabs[nNewPos]->UpdateCompile( true ); // maybe already compiled in Clone, but used names need recompilation SetNoListening( false ); maTabs[nOldPos]->StartListeners(aSLCxt, true); maTabs[nNewPos]->StartListeners(aSLCxt, true); sc::SetFormulaDirtyContext aFormulaDirtyCxt; SetAllFormulasDirty(aFormulaDirtyCxt); if (pDrawLayer) DrawCopyPage( static_cast(nOldPos), static_cast(nNewPos) ); if (pDPCollection) pDPCollection->CopyToTab(nOldPos, nNewPos); maTabs[nNewPos]->SetPageStyle( maTabs[nOldPos]->GetPageStyle() ); maTabs[nNewPos]->SetPendingRowHeights( maTabs[nOldPos]->IsPendingRowHeights() ); // Copy the custom print range if exists. maTabs[nNewPos]->CopyPrintRange(*maTabs[nOldPos]); // Copy the RTL settings maTabs[nNewPos]->SetLayoutRTL(maTabs[nOldPos]->IsLayoutRTL()); maTabs[nNewPos]->SetLoadingRTL(maTabs[nOldPos]->IsLoadingRTL()); } return bValid; } sal_uLong ScDocument::TransferTab( ScDocument* pSrcDoc, SCTAB nSrcPos, SCTAB nDestPos, bool bInsertNew, bool bResultsOnly ) { sal_uLong nRetVal = 1; // 0 => error 1 = ok // 3 => NameBox // 4 => both if (pSrcDoc->pShell->GetMedium()) { pSrcDoc->maFileURL = pSrcDoc->pShell->GetMedium()->GetURLObject().GetMainURL(INetURLObject::DECODE_TO_IURI); // for unsaved files use the title name and adjust during save of file if (pSrcDoc->maFileURL.isEmpty()) pSrcDoc->maFileURL = pSrcDoc->pShell->GetName(); } else { pSrcDoc->maFileURL = pSrcDoc->pShell->GetName(); } bool bValid = true; if (bInsertNew) // re-insert { OUString aName; pSrcDoc->GetName(nSrcPos, aName); CreateValidTabName(aName); bValid = InsertTab(nDestPos, aName); // Copy the RTL settings maTabs[nDestPos]->SetLayoutRTL(pSrcDoc->maTabs[nSrcPos]->IsLayoutRTL()); maTabs[nDestPos]->SetLoadingRTL(pSrcDoc->maTabs[nSrcPos]->IsLoadingRTL()); } else // replace existing tables { if (ValidTab(nDestPos) && nDestPos < static_cast(maTabs.size()) && maTabs[nDestPos]) { maTabs[nDestPos]->DeleteArea( 0,0, MAXCOL,MAXROW, IDF_ALL ); } else bValid = false; } if (bValid) { bool bOldAutoCalcSrc = false; bool bOldAutoCalc = GetAutoCalc(); SetAutoCalc( false ); // avoid repeated calculations SetNoListening( true ); if ( bResultsOnly ) { bOldAutoCalcSrc = pSrcDoc->GetAutoCalc(); pSrcDoc->SetAutoCalc( true ); // in case something needs calculation } { NumFmtMergeHandler aNumFmtMergeHdl(this, pSrcDoc); sc::CopyToDocContext aCxt(*this); nDestPos = std::min(nDestPos, (SCTAB)(GetTableCount() - 1)); { // scope for bulk broadcast ScBulkBroadcast aBulkBroadcast( pBASM); pSrcDoc->maTabs[nSrcPos]->CopyToTable(aCxt, 0, 0, MAXCOL, MAXROW, ( bResultsOnly ? IDF_ALL & ~IDF_FORMULA : IDF_ALL), false, maTabs[nDestPos] ); } } maTabs[nDestPos]->SetTabNo(nDestPos); maTabs[nDestPos]->SetTabBgColor(pSrcDoc->maTabs[nSrcPos]->GetTabBgColor()); if ( !bResultsOnly ) { sc::RefUpdateContext aRefCxt(*this); aRefCxt.meMode = URM_COPY; aRefCxt.maRange = ScRange(0, 0, nDestPos, MAXCOL, MAXROW, nDestPos); aRefCxt.mnTabDelta = nDestPos - nSrcPos; maTabs[nDestPos]->UpdateReference(aRefCxt, NULL); // Readjust self-contained absolute references to this sheet maTabs[nDestPos]->TestTabRefAbs(nSrcPos); sc::CompileFormulaContext aFormulaCxt(this); maTabs[nDestPos]->CompileAll(aFormulaCxt); } SetNoListening( false ); if ( !bResultsOnly ) { sc::StartListeningContext aSLCxt(*this); maTabs[nDestPos]->StartListeners(aSLCxt, true); } SetDirty( ScRange( 0, 0, nDestPos, MAXCOL, MAXROW, nDestPos), false); if ( bResultsOnly ) pSrcDoc->SetAutoCalc( bOldAutoCalcSrc ); SetAutoCalc( bOldAutoCalc ); // copy Drawing if (bInsertNew) TransferDrawPage( pSrcDoc, nSrcPos, nDestPos ); maTabs[nDestPos]->SetPendingRowHeights( pSrcDoc->maTabs[nSrcPos]->IsPendingRowHeights() ); } if (!bValid) nRetVal = 0; bool bVbaEnabled = IsInVBAMode(); if ( bVbaEnabled ) { SfxObjectShell* pSrcShell = pSrcDoc->GetDocumentShell(); if ( pSrcShell ) { OUString aLibName("Standard"); const BasicManager *pBasicManager = pSrcShell->GetBasicManager(); if (pBasicManager && !pBasicManager->GetName().isEmpty()) { aLibName = pSrcShell->GetBasicManager()->GetName(); } OUString sCodeName; OUString sSource; uno::Reference< script::XLibraryContainer > xLibContainer = pSrcShell->GetBasicContainer(); uno::Reference< container::XNameContainer > xLib; if( xLibContainer.is() ) { uno::Any aLibAny = xLibContainer->getByName(aLibName); aLibAny >>= xLib; } if( xLib.is() ) { OUString sSrcCodeName; pSrcDoc->GetCodeName( nSrcPos, sSrcCodeName ); OUString sRTLSource; xLib->getByName( sSrcCodeName ) >>= sRTLSource; sSource = sRTLSource; } VBA_InsertModule( *this, nDestPos, sCodeName, sSource ); } } return nRetVal; } void ScDocument::SetError( SCCOL nCol, SCROW nRow, SCTAB nTab, const sal_uInt16 nError) { if (ValidTab(nTab) && nTab < static_cast(maTabs.size())) if (maTabs[nTab]) maTabs[nTab]->SetError( nCol, nRow, nError ); } void ScDocument::SetFormula( const ScAddress& rPos, const ScTokenArray& rArray, formula::FormulaGrammar::Grammar eGram ) { if (!TableExists(rPos.Tab())) return; maTabs[rPos.Tab()]->SetFormula(rPos.Col(), rPos.Row(), rArray, eGram); } void ScDocument::SetFormula( const ScAddress& rPos, const OUString& rFormula, formula::FormulaGrammar::Grammar eGram ) { if (!TableExists(rPos.Tab())) return; maTabs[rPos.Tab()]->SetFormula(rPos.Col(), rPos.Row(), rFormula, eGram); } ScFormulaCell* ScDocument::SetFormulaCell( const ScAddress& rPos, ScFormulaCell* pCell ) { if (!TableExists(rPos.Tab())) { delete pCell; return NULL; } return maTabs[rPos.Tab()]->SetFormulaCell(rPos.Col(), rPos.Row(), pCell); } bool ScDocument::SetFormulaCells( const ScAddress& rPos, std::vector& rCells ) { if (rCells.empty()) return false; ScTable* pTab = FetchTable(rPos.Tab()); if (!pTab) return false; return pTab->SetFormulaCells(rPos.Col(), rPos.Row(), rCells); } void ScDocument::SetConsolidateDlgData( const ScConsolidateParam* pData ) { delete pConsolidateDlgData; if ( pData ) pConsolidateDlgData = new ScConsolidateParam( *pData ); else pConsolidateDlgData = NULL; } void ScDocument::SetChangeViewSettings(const ScChangeViewSettings& rNew) { if (pChangeViewSettings==NULL) pChangeViewSettings = new ScChangeViewSettings; OSL_ENSURE( pChangeViewSettings, "Oops. No ChangeViewSettings :-( by!" ); *pChangeViewSettings=rNew; } ScFieldEditEngine* ScDocument::CreateFieldEditEngine() { ScFieldEditEngine* pNewEditEngine = NULL; if (!pCacheFieldEditEngine) { pNewEditEngine = new ScFieldEditEngine( this, GetEnginePool(), GetEditPool(), false); } else { if ( !bImportingXML ) { // #i66209# previous use might not have restored update mode, // ensure same state as for a new EditEngine (UpdateMode = true) if ( !pCacheFieldEditEngine->GetUpdateMode() ) pCacheFieldEditEngine->SetUpdateMode(true); } pNewEditEngine = pCacheFieldEditEngine; pCacheFieldEditEngine = NULL; } return pNewEditEngine; } void ScDocument::DisposeFieldEditEngine(ScFieldEditEngine*& rpEditEngine) { if (!pCacheFieldEditEngine && rpEditEngine) { pCacheFieldEditEngine = rpEditEngine; pCacheFieldEditEngine->Clear(); } else delete rpEditEngine; rpEditEngine = NULL; } ScRecursionHelper* ScDocument::CreateRecursionHelperInstance() { return new ScRecursionHelper; } ScLookupCache & ScDocument::GetLookupCache( const ScRange & rRange ) { ScLookupCache* pCache = 0; if (!pLookupCacheMapImpl) pLookupCacheMapImpl = new ScLookupCacheMapImpl; ScLookupCacheMap::iterator it( pLookupCacheMapImpl->aCacheMap.find( rRange)); if (it == pLookupCacheMapImpl->aCacheMap.end()) { pCache = new ScLookupCache( this, rRange); AddLookupCache( *pCache); } else pCache = (*it).second; return *pCache; } void ScDocument::AddLookupCache( ScLookupCache & rCache ) { if (!pLookupCacheMapImpl->aCacheMap.insert( ::std::pair< const ScRange, ScLookupCache*>( rCache.getRange(), &rCache)).second) { OSL_FAIL( "ScDocument::AddLookupCache: couldn't add to hash map"); } else StartListeningArea( rCache.getRange(), false, &rCache); } void ScDocument::RemoveLookupCache( ScLookupCache & rCache ) { ScLookupCacheMap::iterator it( pLookupCacheMapImpl->aCacheMap.find( rCache.getRange())); if (it == pLookupCacheMapImpl->aCacheMap.end()) { OSL_FAIL( "ScDocument::RemoveLookupCache: range not found in hash map"); } else { ScLookupCache* pCache = (*it).second; pLookupCacheMapImpl->aCacheMap.erase( it); EndListeningArea( pCache->getRange(), false, &rCache); } } void ScDocument::ClearLookupCaches() { if( pLookupCacheMapImpl ) pLookupCacheMapImpl->clear(); } bool ScDocument::IsCellInChangeTrack(const ScAddress &cell,Color *pColCellBoder) { ScChangeTrack* pTrack = GetChangeTrack(); ScChangeViewSettings* pSettings = GetChangeViewSettings(); if ( !pTrack || !pTrack->GetFirst() || !pSettings || !pSettings->ShowChanges() ) return false; // missing or turned-off ScActionColorChanger aColorChanger(*pTrack); // Clipping happens from outside //! TODO: without Clipping; only paint affected cells ??!??!? const ScChangeAction* pAction = pTrack->GetFirst(); while (pAction) { ScChangeActionType eType; if ( pAction->IsVisible() ) { eType = pAction->GetType(); const ScBigRange& rBig = pAction->GetBigRange(); if ( rBig.aStart.Tab() == cell.Tab()) { ScRange aRange = rBig.MakeRange(); if ( eType == SC_CAT_DELETE_ROWS ) aRange.aEnd.SetRow( aRange.aStart.Row() ); else if ( eType == SC_CAT_DELETE_COLS ) aRange.aEnd.SetCol( aRange.aStart.Col() ); if (ScViewUtil::IsActionShown( *pAction, *pSettings, *this ) ) { if (aRange.In(cell)) { if (pColCellBoder != NULL) { aColorChanger.Update( *pAction ); Color aColor( aColorChanger.GetColor() ); *pColCellBoder = aColor; } return true; } } } if ( eType == SC_CAT_MOVE && static_cast(pAction)-> GetFromRange().aStart.Tab() == cell.Col() ) { ScRange aRange = static_cast(pAction)-> GetFromRange().MakeRange(); if (ScViewUtil::IsActionShown( *pAction, *pSettings, *this ) ) { if (aRange.In(cell)) { if (pColCellBoder != NULL) { aColorChanger.Update( *pAction ); Color aColor( aColorChanger.GetColor() ); *pColCellBoder = aColor; } return true; } } } } pAction = pAction->GetNext(); } return false; } void ScDocument::GetCellChangeTrackNote( const ScAddress &aCellPos, OUString &aTrackText,bool &bLeftEdge) { aTrackText.clear(); // Change-Tracking ScChangeTrack* pTrack = GetChangeTrack(); ScChangeViewSettings* pSettings = GetChangeViewSettings(); if ( pTrack && pTrack->GetFirst() && pSettings && pSettings->ShowChanges()) { const ScChangeAction* pFound = NULL; const ScChangeAction* pFoundContent = NULL; const ScChangeAction* pFoundMove = NULL; long nModified = 0; const ScChangeAction* pAction = pTrack->GetFirst(); while (pAction) { if ( pAction->IsVisible() && ScViewUtil::IsActionShown( *pAction, *pSettings, *this ) ) { ScChangeActionType eType = pAction->GetType(); const ScBigRange& rBig = pAction->GetBigRange(); if ( rBig.aStart.Tab() == aCellPos.Tab()) { ScRange aRange = rBig.MakeRange(); if ( eType == SC_CAT_DELETE_ROWS ) aRange.aEnd.SetRow( aRange.aStart.Row() ); else if ( eType == SC_CAT_DELETE_COLS ) aRange.aEnd.SetCol( aRange.aStart.Col() ); if ( aRange.In( aCellPos ) ) { pFound = pAction; // the last wins switch ( eType ) { case SC_CAT_CONTENT : pFoundContent = pAction; break; case SC_CAT_MOVE : pFoundMove = pAction; break; default: break; } ++nModified; } } if ( eType == SC_CAT_MOVE ) { ScRange aRange = static_cast(pAction)-> GetFromRange().MakeRange(); if ( aRange.In( aCellPos ) ) { pFound = pAction; ++nModified; } } } pAction = pAction->GetNext(); } if ( pFound ) { if ( pFoundContent && pFound->GetType() != SC_CAT_CONTENT ) pFound = pFoundContent; // Content wins if ( pFoundMove && pFound->GetType() != SC_CAT_MOVE && pFoundMove->GetActionNumber() > pFound->GetActionNumber() ) pFound = pFoundMove; // Move wins // for deleted columns: arrow on left side of row if ( pFound->GetType() == SC_CAT_DELETE_COLS ) bLeftEdge = true; DateTime aDT = pFound->GetDateTime(); aTrackText = pFound->GetUser(); aTrackText += ", "; aTrackText += ScGlobal::pLocaleData->getDate(aDT); aTrackText += " "; aTrackText += ScGlobal::pLocaleData->getTime(aDT); aTrackText += ":\n"; OUString aComStr = pFound->GetComment(); if(!aComStr.isEmpty()) { aTrackText += aComStr; aTrackText += "\n( "; } pFound->GetDescription( aTrackText, this ); if (!aComStr.isEmpty()) { aTrackText += ")"; } } } } void ScDocument::SetPreviewFont( SfxItemSet* pFont ) { delete pPreviewFont; pPreviewFont = pFont; } void ScDocument::SetPreviewSelection( ScMarkData& rSel ) { maPreviewSelection = rSel; } SfxItemSet* ScDocument::GetPreviewFont( SCCOL nCol, SCROW nRow, SCTAB nTab ) { SfxItemSet* pRet = NULL; if ( pPreviewFont ) { ScMarkData aSel = GetPreviewSelection(); if ( aSel.IsCellMarked( nCol, nRow ) && aSel.GetFirstSelected() == nTab ) pRet = pPreviewFont; } return pRet; } ScStyleSheet* ScDocument::GetPreviewCellStyle( SCCOL nCol, SCROW nRow, SCTAB nTab ) { ScStyleSheet* pRet = NULL; ScMarkData aSel = GetPreviewSelection(); if ( pPreviewCellStyle && aSel.IsCellMarked( nCol, nRow ) && aSel.GetFirstSelected() == nTab ) pRet = pPreviewCellStyle; return pRet; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */