summaryrefslogtreecommitdiff
path: root/sc/source/core/data/column3.cxx
diff options
context:
space:
mode:
Diffstat (limited to 'sc/source/core/data/column3.cxx')
-rw-r--r--sc/source/core/data/column3.cxx2065
1 files changed, 0 insertions, 2065 deletions
diff --git a/sc/source/core/data/column3.cxx b/sc/source/core/data/column3.cxx
deleted file mode 100644
index b4c68edfd..000000000
--- a/sc/source/core/data/column3.cxx
+++ /dev/null
@@ -1,2065 +0,0 @@
-/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
-/*************************************************************************
- *
- * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
- *
- * Copyright 2000, 2010 Oracle and/or its affiliates.
- *
- * OpenOffice.org - a multi-platform office productivity suite
- *
- * This file is part of OpenOffice.org.
- *
- * OpenOffice.org is free software: you can redistribute it and/or modify
- * it under the terms of the GNU Lesser General Public License version 3
- * only, as published by the Free Software Foundation.
- *
- * OpenOffice.org is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU Lesser General Public License version 3 for more details
- * (a copy is included in the LICENSE file that accompanied this code).
- *
- * You should have received a copy of the GNU Lesser General Public License
- * version 3 along with OpenOffice.org. If not, see
- * <http://www.openoffice.org/license.html>
- * for a copy of the LGPLv3 License.
- *
- ************************************************************************/
-
-// MARKER(update_precomp.py): autogen include statement, do not remove
-#include "precompiled_sc.hxx"
-// INCLUDE ---------------------------------------------------------------
-
-#include <boost/scoped_ptr.hpp>
-
-#include <mdds/flat_segment_tree.hpp>
-
-#include <sfx2/objsh.hxx>
-#include <svl/zforlist.hxx>
-#include <svl/zformat.hxx>
-
-#include "scitems.hxx"
-#include "column.hxx"
-#include "cell.hxx"
-#include "document.hxx"
-#include "attarray.hxx"
-#include "patattr.hxx"
-#include "cellform.hxx"
-#include "collect.hxx"
-#include "formula/errorcodes.hxx"
-#include "formula/token.hxx"
-#include "brdcst.hxx"
-#include "docoptio.hxx" // GetStdPrecision fuer GetMaxNumberStringLen
-#include "subtotal.hxx"
-#include "markdata.hxx"
-#include "detfunc.hxx" // fuer Notizen bei DeleteRange
-#include "postit.hxx"
-#include "stringutil.hxx"
-#include "docpool.hxx"
-
-#include <com/sun/star/i18n/LocaleDataItem.hpp>
-
-using ::com::sun::star::i18n::LocaleDataItem;
-using ::rtl::OUString;
-using ::rtl::OUStringBuffer;
-
-// Err527 Workaround
-extern const ScFormulaCell* pLastFormulaTreeTop; // in cellform.cxx
-using namespace formula;
-// STATIC DATA -----------------------------------------------------------
-
-bool ScColumn::bDoubleAlloc = false; // fuer Import: Groesse beim Allozieren verdoppeln
-
-
-void ScColumn::Insert( SCROW nRow, ScBaseCell* pNewCell )
-{
- sal_Bool bIsAppended = false;
- if (pItems && nCount>0)
- {
- if (pItems[nCount-1].nRow < nRow)
- {
- Append(nRow, pNewCell );
- bIsAppended = sal_True;
- }
- }
- if ( !bIsAppended )
- {
- SCSIZE nIndex;
- if (Search(nRow, nIndex))
- {
- ScBaseCell* pOldCell = pItems[nIndex].pCell;
-
- // move broadcaster and note to new cell, if not existing in new cell
- if (pOldCell->HasBroadcaster() && !pNewCell->HasBroadcaster())
- pNewCell->TakeBroadcaster( pOldCell->ReleaseBroadcaster() );
- if (pOldCell->HasNote() && !pNewCell->HasNote())
- pNewCell->TakeNote( pOldCell->ReleaseNote() );
-
- if ( pOldCell->GetCellType() == CELLTYPE_FORMULA && !pDocument->IsClipOrUndo() )
- {
- pOldCell->EndListeningTo( pDocument );
- // falls in EndListening NoteCell in gleicher Col zerstoert
- if ( nIndex >= nCount || pItems[nIndex].nRow != nRow )
- Search(nRow, nIndex);
- }
- pOldCell->Delete();
- pItems[nIndex].pCell = pNewCell;
- }
- else
- {
- if (nCount + 1 > nLimit)
- {
- if (bDoubleAlloc)
- {
- if (nLimit < COLUMN_DELTA)
- nLimit = COLUMN_DELTA;
- else
- {
- nLimit *= 2;
- if ( nLimit > sal::static_int_cast<SCSIZE>(MAXROWCOUNT) )
- nLimit = MAXROWCOUNT;
- }
- }
- else
- nLimit += COLUMN_DELTA;
-
- ColEntry* pNewItems = new ColEntry[nLimit];
- if (pItems)
- {
- memmove( pNewItems, pItems, nCount * sizeof(ColEntry) );
- delete[] pItems;
- }
- pItems = pNewItems;
- }
- memmove( &pItems[nIndex + 1], &pItems[nIndex], (nCount - nIndex) * sizeof(ColEntry) );
- pItems[nIndex].pCell = pNewCell;
- pItems[nIndex].nRow = nRow;
- ++nCount;
- }
- }
- // Bei aus Clipboard sind hier noch falsche (alte) Referenzen!
- // Werden in CopyBlockFromClip per UpdateReference umgesetzt,
- // danach StartListeningFromClip und BroadcastFromClip gerufen.
- // Wird ins Clipboard/UndoDoc gestellt, wird kein Broadcast gebraucht.
- // Nach Import wird CalcAfterLoad gerufen, dort Listening.
- if ( !(pDocument->IsClipOrUndo() || pDocument->IsInsertingFromOtherDoc()) )
- {
- pNewCell->StartListeningTo( pDocument );
- CellType eCellType = pNewCell->GetCellType();
- // Notizzelle entsteht beim Laden nur durch StartListeningCell,
- // ausloesende Formelzelle muss sowieso dirty sein.
- if ( !(pDocument->IsCalcingAfterLoad() && eCellType == CELLTYPE_NOTE) )
- {
- if ( eCellType == CELLTYPE_FORMULA )
- ((ScFormulaCell*)pNewCell)->SetDirty();
- else
- pDocument->Broadcast( ScHint( SC_HINT_DATACHANGED,
- ScAddress( nCol, nRow, nTab ), pNewCell ) );
- }
- }
-}
-
-
-void ScColumn::Insert( SCROW nRow, sal_uInt32 nNumberFormat, ScBaseCell* pCell )
-{
- Insert(nRow, pCell);
- short eOldType = pDocument->GetFormatTable()->
- GetType( (sal_uLong)
- ((SfxUInt32Item*)GetAttr( nRow, ATTR_VALUE_FORMAT ))->
- GetValue() );
- short eNewType = pDocument->GetFormatTable()->GetType(nNumberFormat);
- if (!pDocument->GetFormatTable()->IsCompatible(eOldType, eNewType))
- ApplyAttr( nRow, SfxUInt32Item( ATTR_VALUE_FORMAT, (sal_uInt32) nNumberFormat) );
-}
-
-
-void ScColumn::Append( SCROW nRow, ScBaseCell* pCell )
-{
- if (nCount + 1 > nLimit)
- {
- if (bDoubleAlloc)
- {
- if (nLimit < COLUMN_DELTA)
- nLimit = COLUMN_DELTA;
- else
- {
- nLimit *= 2;
- if ( nLimit > sal::static_int_cast<SCSIZE>(MAXROWCOUNT) )
- nLimit = MAXROWCOUNT;
- }
- }
- else
- nLimit += COLUMN_DELTA;
-
- ColEntry* pNewItems = new ColEntry[nLimit];
- if (pItems)
- {
- memmove( pNewItems, pItems, nCount * sizeof(ColEntry) );
- delete[] pItems;
- }
- pItems = pNewItems;
- }
- pItems[nCount].pCell = pCell;
- pItems[nCount].nRow = nRow;
- ++nCount;
-}
-
-
-void ScColumn::Delete( SCROW nRow )
-{
- SCSIZE nIndex;
-
- if (Search(nRow, nIndex))
- {
- ScBaseCell* pCell = pItems[nIndex].pCell;
- ScNoteCell* pNoteCell = new ScNoteCell;
- pItems[nIndex].pCell = pNoteCell; // Dummy fuer Interpret
- pDocument->Broadcast( ScHint( SC_HINT_DYING,
- ScAddress( nCol, nRow, nTab ), pCell ) );
- if ( SvtBroadcaster* pBC = pCell->ReleaseBroadcaster() )
- {
- pNoteCell->TakeBroadcaster( pBC );
- }
- else
- {
- delete pNoteCell;
- --nCount;
- memmove( &pItems[nIndex], &pItems[nIndex + 1], (nCount - nIndex) * sizeof(ColEntry) );
- pItems[nCount].nRow = 0;
- pItems[nCount].pCell = NULL;
- // Soll man hier den Speicher freigeben (delta)? Wird dann langsamer!
- }
- pCell->EndListeningTo( pDocument );
- pCell->Delete();
- }
-}
-
-
-void ScColumn::DeleteAtIndex( SCSIZE nIndex )
-{
- ScBaseCell* pCell = pItems[nIndex].pCell;
- ScNoteCell* pNoteCell = new ScNoteCell;
- pItems[nIndex].pCell = pNoteCell; // Dummy fuer Interpret
- pDocument->Broadcast( ScHint( SC_HINT_DYING,
- ScAddress( nCol, pItems[nIndex].nRow, nTab ), pCell ) );
- delete pNoteCell;
- --nCount;
- memmove( &pItems[nIndex], &pItems[nIndex + 1], (nCount - nIndex) * sizeof(ColEntry) );
- pItems[nCount].nRow = 0;
- pItems[nCount].pCell = NULL;
- pCell->EndListeningTo( pDocument );
- pCell->Delete();
-}
-
-
-void ScColumn::FreeAll()
-{
- if (pItems)
- {
- for (SCSIZE i = 0; i < nCount; i++)
- pItems[i].pCell->Delete();
- delete[] pItems;
- pItems = NULL;
- }
- nCount = 0;
- nLimit = 0;
-}
-
-
-void ScColumn::DeleteRow( SCROW nStartRow, SCSIZE nSize )
-{
- pAttrArray->DeleteRow( nStartRow, nSize );
-
- if ( !pItems || !nCount )
- return ;
-
- SCSIZE nFirstIndex;
- Search( nStartRow, nFirstIndex );
- if ( nFirstIndex >= nCount )
- return ;
-
- sal_Bool bOldAutoCalc = pDocument->GetAutoCalc();
- pDocument->SetAutoCalc( false ); // Mehrfachberechnungen vermeiden
-
- sal_Bool bFound=false;
- SCROW nEndRow = nStartRow + nSize - 1;
- SCSIZE nStartIndex = 0;
- SCSIZE nEndIndex = 0;
- SCSIZE i;
-
- for ( i = nFirstIndex; i < nCount && pItems[i].nRow <= nEndRow; i++ )
- {
- if (!bFound)
- {
- nStartIndex = i;
- bFound = sal_True;
- }
- nEndIndex = i;
-
- ScBaseCell* pCell = pItems[i].pCell;
- SvtBroadcaster* pBC = pCell->GetBroadcaster();
- if (pBC)
- {
-// gibt jetzt invalid reference, kein Aufruecken der direkten Referenzen
-// MoveListeners( *pBC, nRow+nSize );
- pCell->DeleteBroadcaster();
- // in DeleteRange werden leere Broadcaster geloescht
- }
- }
- if (bFound)
- {
- DeleteRange( nStartIndex, nEndIndex, IDF_CONTENTS );
- Search( nStartRow, i );
- if ( i >= nCount )
- {
- pDocument->SetAutoCalc( bOldAutoCalc );
- return ;
- }
- }
- else
- i = nFirstIndex;
-
- ScAddress aAdr( nCol, 0, nTab );
- ScHint aHint( SC_HINT_DATACHANGED, aAdr, NULL ); // only areas (ScBaseCell* == NULL)
- ScAddress& rAddress = aHint.GetAddress();
- // for sparse occupation use single broadcasts, not ranges
- sal_Bool bSingleBroadcasts = (((pItems[nCount-1].nRow - pItems[i].nRow) /
- (nCount - i)) > 1);
- if ( bSingleBroadcasts )
- {
- SCROW nLastBroadcast = MAXROW+1;
- for ( ; i < nCount; i++ )
- {
- SCROW nOldRow = pItems[i].nRow;
- // Aenderung Quelle broadcasten
- rAddress.SetRow( nOldRow );
- pDocument->AreaBroadcast( aHint );
- SCROW nNewRow = (pItems[i].nRow -= nSize);
- // Aenderung Ziel broadcasten
- if ( nLastBroadcast != nNewRow )
- { // direkt aufeinanderfolgende nicht doppelt broadcasten
- rAddress.SetRow( nNewRow );
- pDocument->AreaBroadcast( aHint );
- }
- nLastBroadcast = nOldRow;
- ScBaseCell* pCell = pItems[i].pCell;
- if ( pCell->GetCellType() == CELLTYPE_FORMULA )
- ((ScFormulaCell*)pCell)->aPos.SetRow( nNewRow );
- }
- }
- else
- {
- rAddress.SetRow( pItems[i].nRow );
- ScRange aRange( rAddress );
- aRange.aEnd.SetRow( pItems[nCount-1].nRow );
- for ( ; i < nCount; i++ )
- {
- SCROW nNewRow = (pItems[i].nRow -= nSize);
- ScBaseCell* pCell = pItems[i].pCell;
- if ( pCell->GetCellType() == CELLTYPE_FORMULA )
- ((ScFormulaCell*)pCell)->aPos.SetRow( nNewRow );
- }
- pDocument->AreaBroadcastInRange( aRange, aHint );
- }
-
- pDocument->SetAutoCalc( bOldAutoCalc );
-}
-
-
-void ScColumn::DeleteRange( SCSIZE nStartIndex, SCSIZE nEndIndex, sal_uInt16 nDelFlag )
-{
- /* If caller specifies to not remove the note caption objects, all cells
- have to forget the pointers to them. This is used e.g. while undoing a
- "paste cells" operation, which removes the caption objects later in
- drawing undo. */
- bool bDeleteNote = (nDelFlag & IDF_NOTE) != 0;
- bool bNoCaptions = (nDelFlag & IDF_NOCAPTIONS) != 0;
- if (bDeleteNote && bNoCaptions)
- for ( SCSIZE nIdx = nStartIndex; nIdx <= nEndIndex; ++nIdx )
- if ( ScPostIt* pNote = pItems[ nIdx ].pCell->GetNote() )
- pNote->ForgetCaption();
-
- ScHint aHint( SC_HINT_DYING, ScAddress( nCol, 0, nTab ), 0 );
-
- // cache all formula cells, they will be deleted at end of this function
- typedef ::std::vector< ScFormulaCell* > FormulaCellVector;
- FormulaCellVector aDelCells;
- aDelCells.reserve( nEndIndex - nStartIndex + 1 );
-
- typedef mdds::flat_segment_tree<SCSIZE, bool> RemovedSegments_t;
- RemovedSegments_t aRemovedSegments(nStartIndex, nEndIndex + 1, false);
- SCSIZE nFirst(nStartIndex);
-
- // dummy replacement for old cells, to prevent that interpreter uses old cell
- boost::scoped_ptr<ScNoteCell> pDummyCell(new ScNoteCell);
-
- for ( SCSIZE nIdx = nStartIndex; nIdx <= nEndIndex; ++nIdx )
- {
- // all contents is deleted and cell do not contain broadcaster
- if (((nDelFlag & IDF_CONTENTS) == IDF_CONTENTS) && pItems[ nIdx ].pCell->GetBroadcaster())
- {
- ScBaseCell* pOldCell = pItems[ nIdx ].pCell;
- if (pOldCell->GetCellType() == CELLTYPE_FORMULA)
- {
- // cache formula cell, will be deleted below
- aDelCells.push_back( static_cast< ScFormulaCell* >( pOldCell ) );
- }
- else
- {
- // interpret in broadcast must not use the old cell
- pItems[ nIdx ].pCell = pDummyCell.get();
- aHint.GetAddress().SetRow( pItems[ nIdx ].nRow );
- aHint.SetCell( pOldCell );
- pDocument->Broadcast( aHint );
- pOldCell->Delete();
- }
- }
- // delete some contents of the cells
- else
- {
- // decide whether to delete the cell object according to passed flags
- bool bDelete = false;
- ScBaseCell* pOldCell = pItems[nIdx].pCell;
- CellType eCellType = pOldCell->GetCellType();
- switch ( eCellType )
- {
- case CELLTYPE_VALUE:
- {
- sal_uInt16 nValFlags = nDelFlag & (IDF_DATETIME|IDF_VALUE);
- // delete values and dates?
- bDelete = nValFlags == (IDF_DATETIME|IDF_VALUE);
- // if not, decide according to cell number format
- if( !bDelete && (nValFlags != 0) )
- {
- sal_uLong nIndex = (sal_uLong)((SfxUInt32Item*)GetAttr( pItems[nIdx].nRow, ATTR_VALUE_FORMAT ))->GetValue();
- short nType = pDocument->GetFormatTable()->GetType(nIndex);
- bool bIsDate = (nType == NUMBERFORMAT_DATE) || (nType == NUMBERFORMAT_TIME) || (nType == NUMBERFORMAT_DATETIME);
- bDelete = nValFlags == (bIsDate ? IDF_DATETIME : IDF_VALUE);
- }
- }
- break;
-
- case CELLTYPE_STRING:
- case CELLTYPE_EDIT:
- bDelete = (nDelFlag & IDF_STRING) != 0;
- break;
-
- case CELLTYPE_FORMULA:
- bDelete = (nDelFlag & IDF_FORMULA) != 0;
- break;
-
- case CELLTYPE_NOTE:
- // do note delete note cell with broadcaster
- bDelete = bDeleteNote && !pOldCell->GetBroadcaster();
- break;
-
- default:; // added to avoid warnings
- }
-
- if (bDelete)
- {
- // try to create a replacement note cell, if note or broadcaster exists
- ScNoteCell* pNoteCell = 0;
- if (eCellType != CELLTYPE_NOTE)
- {
- // do not rescue note if it has to be deleted according to passed flags
- ScPostIt* pNote = bDeleteNote ? 0 : pOldCell->ReleaseNote();
- // #i99844# do not release broadcaster from old cell, it still has to notify deleted content
- SvtBroadcaster* pBC = pOldCell->GetBroadcaster();
- if( pNote || pBC )
- pNoteCell = new ScNoteCell( pNote, pBC );
- }
-
- // remove cell entry in cell item list
- SCROW nOldRow = pItems[nIdx].nRow;
- if (pNoteCell)
- {
- // replace old cell with the replacement note cell
- pItems[nIdx].pCell = pNoteCell;
- // ... so it's not really deleted
- bDelete = false;
- }
- else
- pItems[nIdx].pCell = pDummyCell.get();
-
- // cache formula cells (will be deleted later), delete cell of other type
- if (eCellType == CELLTYPE_FORMULA)
- {
- aDelCells.push_back( static_cast< ScFormulaCell* >( pOldCell ) );
- }
- else
- {
- aHint.GetAddress().SetRow( nOldRow );
- aHint.SetCell( pOldCell );
- pDocument->Broadcast( aHint );
- // #i99844# after broadcasting, old cell has to forget the broadcaster (owned by pNoteCell)
- pOldCell->ReleaseBroadcaster();
- pOldCell->Delete();
- }
- }
- else
- {
- // delete cell note
- if (bDeleteNote)
- pItems[nIdx].pCell->DeleteNote();
- }
-
- if (!bDelete)
- {
- // We just came to a non-deleted cell after a segment of
- // deleted ones. So we need to remember the segment
- // before moving on.
- if (nFirst < nIdx)
- aRemovedSegments.insert_back(nFirst, nIdx, true);
- nFirst = nIdx + 1;
- }
- }
- }
- // there is a segment of deleted cells at the end
- if (nFirst < nEndIndex)
- aRemovedSegments.insert_back(nFirst, nEndIndex + 1, true);
-
- {
- RemovedSegments_t::const_iterator aIt(aRemovedSegments.begin());
- RemovedSegments_t::const_iterator aEnd(aRemovedSegments.end());
- if (aIt != aEnd)
- {
- SCSIZE nStartSegment(aIt->first);
- bool bMoveSegment(aIt->second);
- // The indexes in aRemovedSegments denote cell positions in the
- // original array. But as we are shifting it from the left, we have
- // to compensate for already performed shifts for latter segments.
- // TODO: use reverse iterators instead
- SCSIZE nShift(0);
- ++aIt;
- do
- {
- SCSIZE const nEndSegment(aIt->first);
- if (bMoveSegment)
- {
- memmove(
- &pItems[nStartSegment - nShift],
- &pItems[nEndSegment - nShift],
- (nCount - nEndSegment) * sizeof(ColEntry));
- SCSIZE const nNewShift(nEndSegment - nStartSegment);
- nShift += nNewShift;
- nCount -= nNewShift;
- }
- nStartSegment = nEndSegment;
- bMoveSegment = aIt->second;
- ++aIt;
- } while (aIt != aEnd);
- }
- }
-
- // *** delete all formula cells ***
-
- // first, all cells stop listening, may save unneeded recalcualtions
- for ( FormulaCellVector::iterator aIt = aDelCells.begin(), aEnd = aDelCells.end(); aIt != aEnd; ++aIt )
- (*aIt)->EndListeningTo( pDocument );
-
- // #i101869# if the note cell with the broadcaster was deleted in EndListening,
- // forget the pointer to the broadcaster
- for ( FormulaCellVector::iterator aIt = aDelCells.begin(), aEnd = aDelCells.end(); aIt != aEnd; ++aIt )
- {
- SCSIZE nIndex;
- if ( !Search( (*aIt)->aPos.Row(), nIndex ) )
- (*aIt)->ReleaseBroadcaster();
- }
-
- // broadcast SC_HINT_DYING for all cells and delete them
- for ( FormulaCellVector::iterator aIt = aDelCells.begin(), aEnd = aDelCells.end(); aIt != aEnd; ++aIt )
- {
- aHint.SetAddress( (*aIt)->aPos );
- aHint.SetCell( *aIt );
- pDocument->Broadcast( aHint );
- // #i99844# after broadcasting, old cell has to forget the broadcaster (owned by replacement note cell)
- (*aIt)->ReleaseBroadcaster();
- (*aIt)->Delete();
- }
-}
-
-
-void ScColumn::DeleteArea(SCROW nStartRow, SCROW nEndRow, sal_uInt16 nDelFlag)
-{
- // FreeAll darf hier nicht gerufen werden wegen Broadcastern
-
- // Attribute erst am Ende, damit vorher noch zwischen Zahlen und Datum
- // unterschieden werden kann (#47901#)
-
- sal_uInt16 nContMask = IDF_CONTENTS;
- // IDF_NOCAPTIONS needs to be passed too, if IDF_NOTE is set
- if( nDelFlag & IDF_NOTE )
- nContMask |= IDF_NOCAPTIONS;
- sal_uInt16 nContFlag = nDelFlag & nContMask;
-
- if (pItems && nCount>0 && nContFlag)
- {
- if (nStartRow==0 && nEndRow==MAXROW)
- DeleteRange( 0, nCount-1, nContFlag );
- else
- {
- sal_Bool bFound=false;
- SCSIZE nStartIndex = 0;
- SCSIZE nEndIndex = 0;
- for (SCSIZE i = 0; i < nCount; i++)
- if ((pItems[i].nRow >= nStartRow) && (pItems[i].nRow <= nEndRow))
- {
- if (!bFound)
- {
- nStartIndex = i;
- bFound = sal_True;
- }
- nEndIndex = i;
- }
- if (bFound)
- DeleteRange( nStartIndex, nEndIndex, nContFlag );
- }
- }
-
- if ( nDelFlag & IDF_EDITATTR )
- {
- OSL_ENSURE( nContFlag == 0, "DeleteArea: falsche Flags" );
- RemoveEditAttribs( nStartRow, nEndRow );
- }
-
- // Attribute erst hier
- if ((nDelFlag & IDF_ATTRIB) == IDF_ATTRIB) pAttrArray->DeleteArea( nStartRow, nEndRow );
- else if ((nDelFlag & IDF_ATTRIB) != 0) pAttrArray->DeleteHardAttr( nStartRow, nEndRow );
-}
-
-
-ScFormulaCell* ScColumn::CreateRefCell( ScDocument* pDestDoc, const ScAddress& rDestPos,
- SCSIZE nIndex, sal_uInt16 nFlags ) const
-{
- sal_uInt16 nContFlags = nFlags & IDF_CONTENTS;
- if (!nContFlags)
- return NULL;
-
- // Testen, ob Zelle kopiert werden soll
- // auch bei IDF_CONTENTS komplett, wegen Notes / Broadcastern
-
- sal_Bool bMatch = false;
- ScBaseCell* pCell = pItems[nIndex].pCell;
- CellType eCellType = pCell->GetCellType();
- switch ( eCellType )
- {
- case CELLTYPE_VALUE:
- {
- sal_uInt16 nValFlags = nFlags & (IDF_DATETIME|IDF_VALUE);
-
- if ( nValFlags == (IDF_DATETIME|IDF_VALUE) )
- bMatch = sal_True;
- else if ( nValFlags )
- {
- sal_uLong nNumIndex = (sal_uLong)((SfxUInt32Item*)GetAttr(
- pItems[nIndex].nRow, ATTR_VALUE_FORMAT ))->GetValue();
- short nTyp = pDocument->GetFormatTable()->GetType(nNumIndex);
- if ((nTyp == NUMBERFORMAT_DATE) || (nTyp == NUMBERFORMAT_TIME) || (nTyp == NUMBERFORMAT_DATETIME))
- bMatch = ((nFlags & IDF_DATETIME) != 0);
- else
- bMatch = ((nFlags & IDF_VALUE) != 0);
- }
- }
- break;
- case CELLTYPE_STRING:
- case CELLTYPE_EDIT: bMatch = ((nFlags & IDF_STRING) != 0); break;
- case CELLTYPE_FORMULA: bMatch = ((nFlags & IDF_FORMULA) != 0); break;
- default:
- {
- // added to avoid warnings
- }
- }
- if (!bMatch)
- return NULL;
-
-
- // Referenz einsetzen
- ScSingleRefData aRef;
- aRef.nCol = nCol;
- aRef.nRow = pItems[nIndex].nRow;
- aRef.nTab = nTab;
- aRef.InitFlags(); // -> alles absolut
- aRef.SetFlag3D(true);
-
- //! 3D(FALSE) und TabRel(TRUE), wenn die endgueltige Position auf der selben Tabelle ist?
- //! (bei TransposeClip ist die Zielposition noch nicht bekannt)
-
- aRef.CalcRelFromAbs( rDestPos );
-
- ScTokenArray aArr;
- aArr.AddSingleReference( aRef );
-
- return new ScFormulaCell( pDestDoc, rDestPos, &aArr );
-}
-
-
-// rColumn = Quelle
-// nRow1, nRow2 = Zielposition
-
-void ScColumn::CopyFromClip(SCROW nRow1, SCROW nRow2, long nDy,
- sal_uInt16 nInsFlag, bool bAsLink, bool bSkipAttrForEmpty,
- ScColumn& rColumn)
-{
- if ((nInsFlag & IDF_ATTRIB) != 0)
- {
- if ( bSkipAttrForEmpty )
- {
- // copy only attributes for non-empty cells
- // (notes are not counted as non-empty here, to match the content behavior)
-
- SCSIZE nStartIndex;
- rColumn.Search( nRow1-nDy, nStartIndex );
- while ( nStartIndex < rColumn.nCount && rColumn.pItems[nStartIndex].nRow <= nRow2-nDy )
- {
- SCSIZE nEndIndex = nStartIndex;
- if ( rColumn.pItems[nStartIndex].pCell->GetCellType() != CELLTYPE_NOTE )
- {
- SCROW nStartRow = rColumn.pItems[nStartIndex].nRow;
- SCROW nEndRow = nStartRow;
-
- // find consecutive non-empty cells
-
- while ( nEndRow < nRow2-nDy &&
- nEndIndex+1 < rColumn.nCount &&
- rColumn.pItems[nEndIndex+1].nRow == nEndRow+1 &&
- rColumn.pItems[nEndIndex+1].pCell->GetCellType() != CELLTYPE_NOTE )
- {
- ++nEndIndex;
- ++nEndRow;
- }
-
- rColumn.pAttrArray->CopyAreaSafe( nStartRow+nDy, nEndRow+nDy, nDy, *pAttrArray );
- }
- nStartIndex = nEndIndex + 1;
- }
- }
- else
- rColumn.pAttrArray->CopyAreaSafe( nRow1, nRow2, nDy, *pAttrArray );
- }
- if ((nInsFlag & IDF_CONTENTS) == 0)
- return;
-
- if ( bAsLink && nInsFlag == IDF_ALL )
- {
- // bei "alles" werden auch leere Zellen referenziert
- //! IDF_ALL muss immer mehr Flags enthalten, als bei "Inhalte Einfuegen"
- //! einzeln ausgewaehlt werden koennen!
-
- Resize( nCount + static_cast<SCSIZE>(nRow2-nRow1+1) );
-
- ScAddress aDestPos( nCol, 0, nTab ); // Row wird angepasst
-
- // Referenz erzeugen (Quell-Position)
- ScSingleRefData aRef;
- aRef.nCol = rColumn.nCol;
- // nRow wird angepasst
- aRef.nTab = rColumn.nTab;
- aRef.InitFlags(); // -> alles absolut
- aRef.SetFlag3D(true);
-
- for (SCROW nDestRow = nRow1; nDestRow <= nRow2; nDestRow++)
- {
- aRef.nRow = nDestRow - nDy; // Quell-Zeile
- aDestPos.SetRow( nDestRow );
-
- aRef.CalcRelFromAbs( aDestPos );
- ScTokenArray aArr;
- aArr.AddSingleReference( aRef );
- Insert( nDestRow, new ScFormulaCell( pDocument, aDestPos, &aArr ) );
- }
-
- return;
- }
-
- SCSIZE nColCount = rColumn.nCount;
-
- // ignore IDF_FORMULA - "all contents but no formulas" results in the same number of cells
- if ((nInsFlag & ( IDF_CONTENTS & ~IDF_FORMULA )) == ( IDF_CONTENTS & ~IDF_FORMULA ) && nRow2-nRow1 >= 64)
- {
- //! Always do the Resize from the outside, where the number of repetitions is known
- //! (then it can be removed here)
-
- SCSIZE nNew = nCount + nColCount;
- if ( nLimit < nNew )
- Resize( nNew );
- }
-
- // IDF_ADDNOTES must be passed without other content flags than IDF_NOTE
- bool bAddNotes = (nInsFlag & (IDF_CONTENTS | IDF_ADDNOTES)) == (IDF_NOTE | IDF_ADDNOTES);
-
- sal_Bool bAtEnd = false;
- for (SCSIZE i = 0; i < nColCount && !bAtEnd; i++)
- {
- SCsROW nDestRow = rColumn.pItems[i].nRow + nDy;
- if ( nDestRow > (SCsROW) nRow2 )
- bAtEnd = sal_True;
- else if ( nDestRow >= (SCsROW) nRow1 )
- {
- // rows at the beginning may be skipped if filtered rows are left out,
- // nDestRow may be negative then
-
- ScAddress aDestPos( nCol, (SCROW)nDestRow, nTab );
-
- /* #i102056# Paste from clipboard needs to paste the cell notes in
- a second pass. This must not overwrite the existing cells
- already copied to the destination position in the first pass.
- To indicate this special case, the modifier IDF_ADDNOTES is
- passed together with IDF_NOTE in nInsFlag. Of course, there is
- still the need to create a new cell, if there is no cell at the
- destination position at all. */
- ScBaseCell* pAddNoteCell = bAddNotes ? GetCell( aDestPos.Row() ) : 0;
- if (pAddNoteCell)
- {
- // do nothing if source cell does not contain a note
- const ScBaseCell* pSourceCell = rColumn.pItems[i].pCell;
- const ScPostIt* pSourceNote = pSourceCell ? pSourceCell->GetNote() : 0;
- if (pSourceNote)
- {
- OSL_ENSURE( !pAddNoteCell->HasNote(), "ScColumn::CopyFromClip - unexpected note at destination cell" );
- bool bCloneCaption = (nInsFlag & IDF_NOCAPTIONS) == 0;
- // #i52342# if caption is cloned, the note must be constructed with the destination document
- ScAddress aSourcePos( rColumn.nCol, rColumn.pItems[i].nRow, rColumn.nTab );
- ScPostIt* pNewNote = pSourceNote->Clone( aSourcePos, *pDocument, aDestPos, bCloneCaption );
- pAddNoteCell->TakeNote( pNewNote );
- }
- }
- else
- {
- ScBaseCell* pNewCell = bAsLink ?
- rColumn.CreateRefCell( pDocument, aDestPos, i, nInsFlag ) :
- rColumn.CloneCell( i, nInsFlag, *pDocument, aDestPos );
- if (pNewCell)
- Insert( aDestPos.Row(), pNewCell );
- }
- }
- }
-}
-
-
-namespace {
-
-/** Helper for ScColumn::CloneCell - decides whether to clone a value cell depending on clone flags and number format. */
-bool lclCanCloneValue( ScDocument& rDoc, const ScColumn& rCol, SCROW nRow, bool bCloneValue, bool bCloneDateTime )
-{
- // values and dates, or nothing to be cloned -> not needed to check number format
- if( bCloneValue == bCloneDateTime )
- return bCloneValue;
-
- // check number format of value cell
- sal_uLong nNumIndex = (sal_uLong)((SfxUInt32Item*)rCol.GetAttr( nRow, ATTR_VALUE_FORMAT ))->GetValue();
- short nTyp = rDoc.GetFormatTable()->GetType( nNumIndex );
- bool bIsDateTime = (nTyp == NUMBERFORMAT_DATE) || (nTyp == NUMBERFORMAT_TIME) || (nTyp == NUMBERFORMAT_DATETIME);
- return bIsDateTime ? bCloneDateTime : bCloneValue;
-}
-
-} // namespace
-
-
-ScBaseCell* ScColumn::CloneCell(SCSIZE nIndex, sal_uInt16 nFlags, ScDocument& rDestDoc, const ScAddress& rDestPos)
-{
- bool bCloneValue = (nFlags & IDF_VALUE) != 0;
- bool bCloneDateTime = (nFlags & IDF_DATETIME) != 0;
- bool bCloneString = (nFlags & IDF_STRING) != 0;
- bool bCloneSpecialBoolean = (nFlags & IDF_SPECIAL_BOOLEAN) != 0;
- bool bCloneFormula = (nFlags & IDF_FORMULA) != 0;
- bool bCloneNote = (nFlags & IDF_NOTE) != 0;
- bool bForceFormula = false;
-
- ScBaseCell* pNew = 0;
- ScBaseCell& rSource = *pItems[nIndex].pCell;
- switch (rSource.GetCellType())
- {
- case CELLTYPE_NOTE:
- // note will be cloned below
- break;
-
- case CELLTYPE_STRING:
- case CELLTYPE_EDIT:
- // note will be cloned below
- if (bCloneString)
- pNew = rSource.CloneWithoutNote( rDestDoc, rDestPos );
- break;
-
- case CELLTYPE_VALUE:
- // note will be cloned below
- if (lclCanCloneValue( *pDocument, *this, pItems[nIndex].nRow, bCloneValue, bCloneDateTime ))
- pNew = rSource.CloneWithoutNote( rDestDoc, rDestPos );
- break;
-
- case CELLTYPE_FORMULA:
- if ( bCloneSpecialBoolean )
- {
- ScFormulaCell& rForm = (ScFormulaCell&)rSource;
- rtl::OUStringBuffer aBuf;
- // #TODO #FIXME do we have a localisation issue here?
- rForm.GetFormula( aBuf );
- rtl::OUString aVal( aBuf.makeStringAndClear() );
- if ( aVal.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "=TRUE()" ) )
- || aVal.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "=FALSE()" ) ) )
- bForceFormula = true;
- }
- if (bForceFormula || bCloneFormula)
- {
- // note will be cloned below
- pNew = rSource.CloneWithoutNote( rDestDoc, rDestPos );
- }
- else if ( (bCloneValue || bCloneDateTime || bCloneString) && !rDestDoc.IsUndo() )
- {
- // ins Undo-Dokument immer nur die Original-Zelle kopieren,
- // aus Formeln keine Value/String-Zellen erzeugen
- ScFormulaCell& rForm = (ScFormulaCell&)rSource;
- sal_uInt16 nErr = rForm.GetErrCode();
- if ( nErr )
- {
- // error codes are cloned with values
- if (bCloneValue)
- {
- ScFormulaCell* pErrCell = new ScFormulaCell( &rDestDoc, rDestPos );
- pErrCell->SetErrCode( nErr );
- pNew = pErrCell;
- }
- }
- else if (rForm.IsValue())
- {
- if (lclCanCloneValue( *pDocument, *this, pItems[nIndex].nRow, bCloneValue, bCloneDateTime ))
- {
- double nVal = rForm.GetValue();
- pNew = new ScValueCell(nVal);
- }
- }
- else if (bCloneString)
- {
- String aString;
- rForm.GetString( aString );
- // do not clone empty string
- if (aString.Len() > 0)
- {
- if ( rForm.IsMultilineResult() )
- {
- pNew = new ScEditCell( aString, &rDestDoc );
- }
- else
- {
- pNew = new ScStringCell( aString );
- }
- }
- }
- }
- break;
-
- default: OSL_FAIL( "ScColumn::CloneCell - unknown cell type" );
- }
-
- // clone the cell note
- if (bCloneNote)
- {
- if (ScPostIt* pNote = rSource.GetNote())
- {
- bool bCloneCaption = (nFlags & IDF_NOCAPTIONS) == 0;
- // #i52342# if caption is cloned, the note must be constructed with the destination document
- ScAddress aOwnPos( nCol, pItems[nIndex].nRow, nTab );
- ScPostIt* pNewNote = pNote->Clone( aOwnPos, rDestDoc, rDestPos, bCloneCaption );
- if (!pNew)
- pNew = new ScNoteCell( pNewNote );
- else
- pNew->TakeNote( pNewNote );
- }
- }
-
- return pNew;
-}
-
-
-void ScColumn::MixMarked( const ScMarkData& rMark, sal_uInt16 nFunction,
- bool bSkipEmpty, ScColumn& rSrcCol )
-{
- SCROW nRow1, nRow2;
-
- if (rMark.IsMultiMarked())
- {
- ScMarkArrayIter aIter( rMark.GetArray()+nCol );
- while (aIter.Next( nRow1, nRow2 ))
- MixData( nRow1, nRow2, nFunction, bSkipEmpty, rSrcCol );
- }
-}
-
-
-// Ergebnis in rVal1
-
-sal_Bool lcl_DoFunction( double& rVal1, double nVal2, sal_uInt16 nFunction )
-{
- sal_Bool bOk = false;
- switch (nFunction)
- {
- case PASTE_ADD:
- bOk = SubTotal::SafePlus( rVal1, nVal2 );
- break;
- case PASTE_SUB:
- nVal2 = -nVal2; //! geht das immer ohne Fehler?
- bOk = SubTotal::SafePlus( rVal1, nVal2 );
- break;
- case PASTE_MUL:
- bOk = SubTotal::SafeMult( rVal1, nVal2 );
- break;
- case PASTE_DIV:
- bOk = SubTotal::SafeDiv( rVal1, nVal2 );
- break;
- }
- return bOk;
-}
-
-
-void lcl_AddCode( ScTokenArray& rArr, ScFormulaCell* pCell )
-{
- rArr.AddOpCode(ocOpen);
-
- ScTokenArray* pCode = pCell->GetCode();
- if (pCode)
- {
- const formula::FormulaToken* pToken = pCode->First();
- while (pToken)
- {
- rArr.AddToken( *pToken );
- pToken = pCode->Next();
- }
- }
-
- rArr.AddOpCode(ocClose);
-}
-
-
-void ScColumn::MixData( SCROW nRow1, SCROW nRow2,
- sal_uInt16 nFunction, bool bSkipEmpty,
- ScColumn& rSrcCol )
-{
- SCSIZE nSrcCount = rSrcCol.nCount;
-
- SCSIZE nIndex;
- Search( nRow1, nIndex );
-
-// SCSIZE nSrcIndex = 0;
- SCSIZE nSrcIndex;
- rSrcCol.Search( nRow1, nSrcIndex ); //! Testen, ob Daten ganz vorne
-
- SCROW nNextThis = MAXROW+1;
- if ( nIndex < nCount )
- nNextThis = pItems[nIndex].nRow;
- SCROW nNextSrc = MAXROW+1;
- if ( nSrcIndex < nSrcCount )
- nNextSrc = rSrcCol.pItems[nSrcIndex].nRow;
-
- while ( nNextThis <= nRow2 || nNextSrc <= nRow2 )
- {
- SCROW nRow = Min( nNextThis, nNextSrc );
-
- ScBaseCell* pSrc = NULL;
- ScBaseCell* pDest = NULL;
- ScBaseCell* pNew = NULL;
- sal_Bool bDelete = false;
-
- if ( nSrcIndex < nSrcCount && nNextSrc == nRow )
- pSrc = rSrcCol.pItems[nSrcIndex].pCell;
-
- if ( nIndex < nCount && nNextThis == nRow )
- pDest = pItems[nIndex].pCell;
-
- OSL_ENSURE( pSrc || pDest, "Nanu ?" );
-
- CellType eSrcType = pSrc ? pSrc->GetCellType() : CELLTYPE_NONE;
- CellType eDestType = pDest ? pDest->GetCellType() : CELLTYPE_NONE;
-
- sal_Bool bSrcEmpty = ( eSrcType == CELLTYPE_NONE || eSrcType == CELLTYPE_NOTE );
- sal_Bool bDestEmpty = ( eDestType == CELLTYPE_NONE || eDestType == CELLTYPE_NOTE );
-
- if ( bSkipEmpty && bDestEmpty ) // Originalzelle wiederherstellen
- {
- if ( pSrc ) // war da eine Zelle?
- {
- pNew = pSrc->CloneWithoutNote( *pDocument );
- }
- }
- else if ( nFunction ) // wirklich Rechenfunktion angegeben
- {
- double nVal1;
- double nVal2;
- if ( eSrcType == CELLTYPE_VALUE )
- nVal1 = ((ScValueCell*)pSrc)->GetValue();
- else
- nVal1 = 0.0;
- if ( eDestType == CELLTYPE_VALUE )
- nVal2 = ((ScValueCell*)pDest)->GetValue();
- else
- nVal2 = 0.0;
-
- // leere Zellen werden als Werte behandelt
-
- sal_Bool bSrcVal = ( bSrcEmpty || eSrcType == CELLTYPE_VALUE );
- sal_Bool bDestVal = ( bDestEmpty || eDestType == CELLTYPE_VALUE );
-
- sal_Bool bSrcText = ( eSrcType == CELLTYPE_STRING ||
- eSrcType == CELLTYPE_EDIT );
- sal_Bool bDestText = ( eDestType == CELLTYPE_STRING ||
- eDestType == CELLTYPE_EDIT );
-
- // sonst bleibt nur Formel...
-
- if ( bSrcEmpty && bDestEmpty )
- {
- // beide leer -> nix
- }
- else if ( bSrcVal && bDestVal )
- {
- // neuen Wert eintragen, oder Fehler bei Ueberlauf
-
- sal_Bool bOk = lcl_DoFunction( nVal1, nVal2, nFunction );
-
- if (bOk)
- pNew = new ScValueCell( nVal1 );
- else
- {
- ScFormulaCell* pFC = new ScFormulaCell( pDocument,
- ScAddress( nCol, nRow, nTab ) );
- pFC->SetErrCode( errNoValue );
- //! oder NOVALUE, dann auch in consoli,
- //! sonst in Interpreter::GetCellValue die Abfrage auf errNoValue raus
- //! (dann geht Stringzelle+Wertzelle nicht mehr)
- pNew = pFC;
- }
- }
- else if ( bSrcText || bDestText )
- {
- // mit Texten wird nicht gerechnet - immer "alte" Zelle, also pSrc
-
- if (pSrc)
- pNew = pSrc->CloneWithoutNote( *pDocument );
- else if (pDest)
- bDelete = sal_True;
- }
- else
- {
- // Kombination aus Wert und mindestens einer Formel -> Formel erzeugen
-
- ScTokenArray aArr;
-
- // erste Zelle
- if ( eSrcType == CELLTYPE_FORMULA )
- lcl_AddCode( aArr, (ScFormulaCell*)pSrc );
- else
- aArr.AddDouble( nVal1 );
-
- // Operator
- OpCode eOp = ocAdd;
- switch ( nFunction )
- {
- case PASTE_ADD: eOp = ocAdd; break;
- case PASTE_SUB: eOp = ocSub; break;
- case PASTE_MUL: eOp = ocMul; break;
- case PASTE_DIV: eOp = ocDiv; break;
- }
- aArr.AddOpCode(eOp); // Funktion
-
- // zweite Zelle
- if ( eDestType == CELLTYPE_FORMULA )
- lcl_AddCode( aArr, (ScFormulaCell*)pDest );
- else
- aArr.AddDouble( nVal2 );
-
- pNew = new ScFormulaCell( pDocument, ScAddress( nCol, nRow, nTab ), &aArr );
- }
- }
-
-
- if ( pNew || bDelete ) // neues Ergebnis ?
- {
- if (pDest && !pNew) // alte Zelle da ?
- {
- if ( pDest->GetBroadcaster() )
- pNew = new ScNoteCell; // Broadcaster uebernehmen
- else
- Delete(nRow); // -> loeschen
- }
- if (pNew)
- Insert(nRow, pNew); // neue einfuegen
-
- Search( nRow, nIndex ); // alles kann sich verschoben haben
- if (pNew)
- nNextThis = nRow; // nIndex zeigt jetzt genau auf nRow
- else
- nNextThis = ( nIndex < nCount ) ? pItems[nIndex].nRow : MAXROW+1;
- }
-
- if ( nNextThis == nRow )
- {
- ++nIndex;
- nNextThis = ( nIndex < nCount ) ? pItems[nIndex].nRow : MAXROW+1;
- }
- if ( nNextSrc == nRow )
- {
- ++nSrcIndex;
- nNextSrc = ( nSrcIndex < nSrcCount ) ?
- rSrcCol.pItems[nSrcIndex].nRow :
- MAXROW+1;
- }
- }
-}
-
-
-ScAttrIterator* ScColumn::CreateAttrIterator( SCROW nStartRow, SCROW nEndRow ) const
-{
- return new ScAttrIterator( pAttrArray, nStartRow, nEndRow );
-}
-
-
-void ScColumn::StartAllListeners()
-{
- if (pItems)
- for (SCSIZE i = 0; i < nCount; i++)
- {
- ScBaseCell* pCell = pItems[i].pCell;
- if ( pCell->GetCellType() == CELLTYPE_FORMULA )
- {
- SCROW nRow = pItems[i].nRow;
- ((ScFormulaCell*)pCell)->StartListeningTo( pDocument );
- if ( nRow != pItems[i].nRow )
- Search( nRow, i ); // Listener eingefuegt?
- }
- }
-}
-
-
-void ScColumn::StartNeededListeners()
-{
- if (pItems)
- {
- for (SCSIZE i = 0; i < nCount; i++)
- {
- ScBaseCell* pCell = pItems[i].pCell;
- if ( pCell->GetCellType() == CELLTYPE_FORMULA )
- {
- ScFormulaCell* pFCell = static_cast<ScFormulaCell*>(pCell);
- if (pFCell->NeedsListening())
- {
- SCROW nRow = pItems[i].nRow;
- pFCell->StartListeningTo( pDocument );
- if ( nRow != pItems[i].nRow )
- Search( nRow, i ); // Listener eingefuegt?
- }
- }
- }
- }
-}
-
-
-void ScColumn::BroadcastInArea( SCROW nRow1, SCROW nRow2 )
-{
- if ( pItems )
- {
- SCROW nRow;
- SCSIZE nIndex;
- Search( nRow1, nIndex );
- while ( nIndex < nCount && (nRow = pItems[nIndex].nRow) <= nRow2 )
- {
- ScBaseCell* pCell = pItems[nIndex].pCell;
- if ( pCell->GetCellType() == CELLTYPE_FORMULA )
- ((ScFormulaCell*)pCell)->SetDirty();
- else
- pDocument->Broadcast( ScHint( SC_HINT_DATACHANGED,
- ScAddress( nCol, nRow, nTab ), pCell ) );
- nIndex++;
- }
- }
-}
-
-
-void ScColumn::StartListeningInArea( SCROW nRow1, SCROW nRow2 )
-{
- if ( pItems )
- {
- SCROW nRow;
- SCSIZE nIndex;
- Search( nRow1, nIndex );
- while ( nIndex < nCount && (nRow = pItems[nIndex].nRow) <= nRow2 )
- {
- ScBaseCell* pCell = pItems[nIndex].pCell;
- if ( pCell->GetCellType() == CELLTYPE_FORMULA )
- ((ScFormulaCell*)pCell)->StartListeningTo( pDocument );
- if ( nRow != pItems[nIndex].nRow )
- Search( nRow, nIndex ); // durch Listening eingefuegt
- nIndex++;
- }
- }
-}
-
-
-// TRUE = Zahlformat gesetzt
-bool ScColumn::SetString( SCROW nRow, SCTAB nTabP, const String& rString,
- formula::FormulaGrammar::AddressConvention eConv,
- ScSetStringParam* pParam )
-{
- bool bNumFmtSet = false;
- if (!ValidRow(nRow))
- return false;
-
- ScBaseCell* pNewCell = NULL;
- sal_Bool bIsLoading = false;
- if (rString.Len() > 0)
- {
- ScSetStringParam aParam;
- if (pParam)
- aParam = *pParam;
-
- sal_uInt32 nIndex, nOldIndex = 0;
- sal_Unicode cFirstChar;
- if (!aParam.mpNumFormatter)
- aParam.mpNumFormatter = pDocument->GetFormatTable();
- SfxObjectShell* pDocSh = pDocument->GetDocumentShell();
- if ( pDocSh )
- bIsLoading = pDocSh->IsLoading();
- // IsLoading bei ConvertFrom Import
- if ( !bIsLoading )
- {
- nIndex = nOldIndex = GetNumberFormat( nRow );
- if ( rString.Len() > 1
- && aParam.mpNumFormatter->GetType(nIndex) != NUMBERFORMAT_TEXT )
- cFirstChar = rString.GetChar(0);
- else
- cFirstChar = 0; // Text
- }
- else
- { // waehrend ConvertFrom Import gibt es keine gesetzten Formate
- cFirstChar = rString.GetChar(0);
- }
-
- if ( cFirstChar == '=' )
- {
- if ( rString.Len() == 1 ) // = Text
- pNewCell = new ScStringCell( rString );
- else // =Formel
- pNewCell = new ScFormulaCell( pDocument,
- ScAddress( nCol, nRow, nTabP ), rString,
- formula::FormulaGrammar::mergeToGrammar( formula::FormulaGrammar::GRAM_DEFAULT,
- eConv), MM_NONE );
- }
- else if ( cFirstChar == '\'') // 'Text
- {
- // Cell format is not 'Text', and the first char
- // is an apostrophe. Check if the input is considered a number.
- String aTest = rString.Copy(1);
- double fTest;
- if (aParam.mpNumFormatter->IsNumberFormat(aTest, nIndex, fTest))
- // This is a number. Strip out the first char.
- pNewCell = new ScStringCell(aTest);
- else
- // This is a normal text. Take it as-is.
- pNewCell = new ScStringCell(rString);
- }
- else
- {
- double nVal;
- sal_Bool bIsText = false;
- if ( bIsLoading )
- {
- if ( pItems && nCount )
- {
- String aStr;
- SCSIZE i = nCount;
- SCSIZE nStop = (i >= 3 ? i - 3 : 0);
- // die letzten Zellen vergleichen, ob gleicher String
- // und IsNumberFormat eingespart werden kann
- do
- {
- i--;
- ScBaseCell* pCell = pItems[i].pCell;
- switch ( pCell->GetCellType() )
- {
- case CELLTYPE_STRING :
- ((ScStringCell*)pCell)->GetString( aStr );
- if ( rString == aStr )
- bIsText = true;
- break;
- case CELLTYPE_NOTE : // durch =Formel referenziert
- break;
- default:
- if ( i == nCount - 1 )
- i = 0;
- // wahrscheinlich ganze Spalte kein String
- }
- } while ( i && i > nStop && !bIsText );
- }
- // nIndex fuer IsNumberFormat vorbelegen
- if ( !bIsText )
- nIndex = nOldIndex = aParam.mpNumFormatter->GetStandardIndex();
- }
-
- do
- {
- if (bIsText)
- break;
-
- if (aParam.mbDetectNumberFormat)
- {
- if (!aParam.mpNumFormatter->IsNumberFormat(rString, nIndex, nVal))
- break;
-
- if ( aParam.mpNumFormatter )
- {
- // convert back to the original language if a built-in format was detected
- const SvNumberformat* pOldFormat = aParam.mpNumFormatter->GetEntry( nOldIndex );
- if ( pOldFormat )
- nIndex = aParam.mpNumFormatter->GetFormatForLanguageIfBuiltIn( nIndex, pOldFormat->GetLanguage() );
- }
-
- pNewCell = new ScValueCell( nVal );
- if ( nIndex != nOldIndex)
- {
- // #i22345# New behavior: Apply the detected number format only if
- // the old one was the default number, date, time or boolean format.
- // Exception: If the new format is boolean, always apply it.
-
- sal_Bool bOverwrite = false;
- const SvNumberformat* pOldFormat = aParam.mpNumFormatter->GetEntry( nOldIndex );
- if ( pOldFormat )
- {
- short nOldType = pOldFormat->GetType() & ~NUMBERFORMAT_DEFINED;
- if ( nOldType == NUMBERFORMAT_NUMBER || nOldType == NUMBERFORMAT_DATE ||
- nOldType == NUMBERFORMAT_TIME || nOldType == NUMBERFORMAT_LOGICAL )
- {
- if ( nOldIndex == aParam.mpNumFormatter->GetStandardFormat(
- nOldType, pOldFormat->GetLanguage() ) )
- {
- bOverwrite = true; // default of these types can be overwritten
- }
- }
- }
- if ( !bOverwrite && aParam.mpNumFormatter->GetType( nIndex ) == NUMBERFORMAT_LOGICAL )
- {
- bOverwrite = true; // overwrite anything if boolean was detected
- }
-
- if ( bOverwrite )
- {
- ApplyAttr( nRow, SfxUInt32Item( ATTR_VALUE_FORMAT,
- (sal_uInt32) nIndex) );
- bNumFmtSet = true;
- }
- }
- }
- else
- {
- // Only check if the string is a regular number.
- const LocaleDataWrapper* pLocale = aParam.mpNumFormatter->GetLocaleData();
- if (!pLocale)
- break;
-
- LocaleDataItem aLocaleItem = pLocale->getLocaleItem();
- const OUString& rDecSep = aLocaleItem.decimalSeparator;
- const OUString& rGroupSep = aLocaleItem.thousandSeparator;
- if (rDecSep.getLength() != 1 || rGroupSep.getLength() != 1)
- break;
-
- sal_Unicode dsep = rDecSep.getStr()[0];
- sal_Unicode gsep = rGroupSep.getStr()[0];
-
- if (!ScStringUtil::parseSimpleNumber(rString, dsep, gsep, nVal))
- break;
-
- pNewCell = new ScValueCell(nVal);
- }
- }
- while (false);
-
- if (!pNewCell)
- {
- if (aParam.mbSetTextCellFormat && aParam.mpNumFormatter->IsNumberFormat(rString, nIndex, nVal))
- {
- // Set the cell format type to Text.
- sal_uInt32 nFormat = aParam.mpNumFormatter->GetStandardFormat(NUMBERFORMAT_TEXT);
- ScPatternAttr aNewAttrs(pDocument->GetPool());
- SfxItemSet& rSet = aNewAttrs.GetItemSet();
- rSet.Put( SfxUInt32Item(ATTR_VALUE_FORMAT, nFormat) );
- ApplyPattern(nRow, aNewAttrs);
- }
-
- pNewCell = new ScStringCell(rString);
- }
- }
- }
-
- if ( bIsLoading && (!nCount || nRow > pItems[nCount-1].nRow) )
- { // Search einsparen und ohne Umweg ueber Insert, Listener aufbauen
- // und Broadcast kommt eh erst nach dem Laden
- if ( pNewCell )
- Append( nRow, pNewCell );
- }
- else
- {
- SCSIZE i;
- if (Search(nRow, i))
- {
- ScBaseCell* pOldCell = pItems[i].pCell;
- ScPostIt* pNote = pOldCell->ReleaseNote();
- SvtBroadcaster* pBC = pOldCell->ReleaseBroadcaster();
- if (pNewCell || pNote || pBC)
- {
- if (pNewCell)
- pNewCell->TakeNote( pNote );
- else
- pNewCell = new ScNoteCell( pNote );
- if (pBC)
- {
- pNewCell->TakeBroadcaster(pBC);
- pLastFormulaTreeTop = 0; // Err527 Workaround
- }
-
- if ( pOldCell->GetCellType() == CELLTYPE_FORMULA )
- {
- pOldCell->EndListeningTo( pDocument );
- // falls in EndListening NoteCell in gleicher Col zerstoert
- if ( i >= nCount || pItems[i].nRow != nRow )
- Search(nRow, i);
- }
- pOldCell->Delete();
- pItems[i].pCell = pNewCell; // ersetzen
- if ( pNewCell->GetCellType() == CELLTYPE_FORMULA )
- {
- pNewCell->StartListeningTo( pDocument );
- ((ScFormulaCell*)pNewCell)->SetDirty();
- }
- else
- pDocument->Broadcast( ScHint( SC_HINT_DATACHANGED,
- ScAddress( nCol, nRow, nTabP ), pNewCell ) );
- }
- else
- {
- DeleteAtIndex(i); // loeschen und Broadcast
- }
- }
- else if (pNewCell)
- {
- Insert(nRow, pNewCell); // neu eintragen und Broadcast
- }
- }
-
- // hier keine Formate mehr fuer Formeln setzen!
- // (werden bei der Ausgabe abgefragt)
-
- return bNumFmtSet;
-}
-
-
-void ScColumn::GetFilterEntries(SCROW nStartRow, SCROW nEndRow, TypedScStrCollection& rStrings, bool& rHasDates)
-{
- bool bHasDates = false;
- SvNumberFormatter* pFormatter = pDocument->GetFormatTable();
- String aString;
- SCROW nRow = 0;
- SCSIZE nIndex;
-
- Search( nStartRow, nIndex );
-
- while ( (nIndex < nCount) ? ((nRow=pItems[nIndex].nRow) <= nEndRow) : false )
- {
- ScBaseCell* pCell = pItems[nIndex].pCell;
- TypedStrData* pData;
- sal_uLong nFormat = GetNumberFormat( nRow );
-
- ScCellFormat::GetInputString( pCell, nFormat, aString, *pFormatter );
-
- if ( pDocument->HasStringData( nCol, nRow, nTab ) )
- pData = new TypedStrData( aString );
- else
- {
- double nValue;
-
- switch ( pCell->GetCellType() )
- {
- case CELLTYPE_VALUE:
- nValue = ((ScValueCell*)pCell)->GetValue();
- break;
-
- case CELLTYPE_FORMULA:
- nValue = ((ScFormulaCell*)pCell)->GetValue();
- break;
-
- default:
- nValue = 0.0;
- }
-
- if (pFormatter)
- {
- short nType = pFormatter->GetType(nFormat);
- if ((nType & NUMBERFORMAT_DATE) && !(nType & NUMBERFORMAT_TIME))
- {
- // special case for date values. Disregard the time
- // element if the number format is of date type.
- nValue = ::rtl::math::approxFloor(nValue);
- bHasDates = true;
- }
- }
-
- pData = new TypedStrData( aString, nValue, SC_STRTYPE_VALUE );
- }
-
- if ( !rStrings.Insert( pData ) )
- delete pData; // doppelt
-
- ++nIndex;
- }
-
- rHasDates = bHasDates;
-}
-
-//
-// GetDataEntries - Strings aus zusammenhaengendem Bereich um nRow
-//
-
-// DATENT_MAX - max. Anzahl Eintrage in Liste fuer Auto-Eingabe
-// DATENT_SEARCH - max. Anzahl Zellen, die durchsucht werden - neu: nur Strings zaehlen
-#define DATENT_MAX 200
-#define DATENT_SEARCH 2000
-
-
-bool ScColumn::GetDataEntries(SCROW nStartRow, TypedScStrCollection& rStrings, bool bLimit)
-{
- sal_Bool bFound = false;
- SCSIZE nThisIndex;
- sal_Bool bThisUsed = Search( nStartRow, nThisIndex );
- String aString;
- sal_uInt16 nCells = 0;
-
- // Die Beschraenkung auf angrenzende Zellen (ohne Luecken) ist nicht mehr gewollt
- // (Featurekommission zur 5.1), stattdessen abwechselnd nach oben und unten suchen,
- // damit naheliegende Zellen wenigstens zuerst gefunden werden.
- //! Abstaende der Zeilennummern vergleichen? (Performance??)
-
- SCSIZE nUpIndex = nThisIndex; // zeigt hinter die Zelle
- SCSIZE nDownIndex = nThisIndex; // zeigt auf die Zelle
- if (bThisUsed)
- ++nDownIndex; // Startzelle ueberspringen
-
- while ( nUpIndex || nDownIndex < nCount )
- {
- if ( nUpIndex ) // nach oben
- {
- ScBaseCell* pCell = pItems[nUpIndex-1].pCell;
- CellType eType = pCell->GetCellType();
- if (eType == CELLTYPE_STRING || eType == CELLTYPE_EDIT) // nur Strings interessieren
- {
- if (eType == CELLTYPE_STRING)
- ((ScStringCell*)pCell)->GetString(aString);
- else
- ((ScEditCell*)pCell)->GetString(aString);
-
- TypedStrData* pData = new TypedStrData(aString);
- if ( !rStrings.Insert( pData ) )
- delete pData; // doppelt
- else if ( bLimit && rStrings.GetCount() >= DATENT_MAX )
- break; // Maximum erreicht
- bFound = true;
-
- if ( bLimit )
- if (++nCells >= DATENT_SEARCH)
- break; // genug gesucht
- }
- --nUpIndex;
- }
-
- if ( nDownIndex < nCount ) // nach unten
- {
- ScBaseCell* pCell = pItems[nDownIndex].pCell;
- CellType eType = pCell->GetCellType();
- if (eType == CELLTYPE_STRING || eType == CELLTYPE_EDIT) // nur Strings interessieren
- {
- if (eType == CELLTYPE_STRING)
- ((ScStringCell*)pCell)->GetString(aString);
- else
- ((ScEditCell*)pCell)->GetString(aString);
-
- TypedStrData* pData = new TypedStrData(aString);
- if ( !rStrings.Insert( pData ) )
- delete pData; // doppelt
- else if ( bLimit && rStrings.GetCount() >= DATENT_MAX )
- break; // Maximum erreicht
- bFound = true;
-
- if ( bLimit )
- if (++nCells >= DATENT_SEARCH)
- break; // genug gesucht
- }
- ++nDownIndex;
- }
- }
-
- return bFound;
-}
-
-#undef DATENT_MAX
-#undef DATENT_SEARCH
-
-
-void ScColumn::RemoveProtected( SCROW nStartRow, SCROW nEndRow )
-{
- ScAttrIterator aAttrIter( pAttrArray, nStartRow, nEndRow );
- SCROW nTop = -1;
- SCROW nBottom = -1;
- SCSIZE nIndex;
- const ScPatternAttr* pPattern = aAttrIter.Next( nTop, nBottom );
- while (pPattern)
- {
- const ScProtectionAttr* pAttr = (const ScProtectionAttr*)&pPattern->GetItem(ATTR_PROTECTION);
- if ( pAttr->GetHideCell() )
- DeleteArea( nTop, nBottom, IDF_CONTENTS );
- else if ( pAttr->GetHideFormula() )
- {
- Search( nTop, nIndex );
- while ( nIndex<nCount && pItems[nIndex].nRow<=nBottom )
- {
- if ( pItems[nIndex].pCell->GetCellType() == CELLTYPE_FORMULA )
- {
- ScFormulaCell* pFormula = (ScFormulaCell*)pItems[nIndex].pCell;
- if (pFormula->IsValue())
- {
- double nVal = pFormula->GetValue();
- pItems[nIndex].pCell = new ScValueCell( nVal );
- }
- else
- {
- String aString;
- pFormula->GetString(aString);
- pItems[nIndex].pCell = new ScStringCell( aString );
- }
- delete pFormula;
- }
- ++nIndex;
- }
- }
-
- pPattern = aAttrIter.Next( nTop, nBottom );
- }
-}
-
-
-void ScColumn::SetError( SCROW nRow, const sal_uInt16 nError)
-{
- if (VALIDROW(nRow))
- {
- ScFormulaCell* pCell = new ScFormulaCell
- ( pDocument, ScAddress( nCol, nRow, nTab ) );
- pCell->SetErrCode( nError );
- Insert( nRow, pCell );
- }
-}
-
-
-void ScColumn::SetValue( SCROW nRow, const double& rVal)
-{
- if (VALIDROW(nRow))
- {
- ScBaseCell* pCell = new ScValueCell(rVal);
- Insert( nRow, pCell );
- }
-}
-
-
-void ScColumn::GetString( SCROW nRow, String& rString ) const
-{
- SCSIZE nIndex;
- Color* pColor;
- if (Search(nRow, nIndex))
- {
- ScBaseCell* pCell = pItems[nIndex].pCell;
- if (pCell->GetCellType() != CELLTYPE_NOTE)
- {
- sal_uLong nFormat = GetNumberFormat( nRow );
- ScCellFormat::GetString( pCell, nFormat, rString, &pColor, *(pDocument->GetFormatTable()) );
- }
- else
- rString.Erase();
- }
- else
- rString.Erase();
-}
-
-
-void ScColumn::GetInputString( SCROW nRow, String& rString ) const
-{
- SCSIZE nIndex;
- if (Search(nRow, nIndex))
- {
- ScBaseCell* pCell = pItems[nIndex].pCell;
- if (pCell->GetCellType() != CELLTYPE_NOTE)
- {
- sal_uLong nFormat = GetNumberFormat( nRow );
- ScCellFormat::GetInputString( pCell, nFormat, rString, *(pDocument->GetFormatTable()) );
- }
- else
- rString.Erase();
- }
- else
- rString.Erase();
-}
-
-
-double ScColumn::GetValue( SCROW nRow ) const
-{
- SCSIZE nIndex;
- if (Search(nRow, nIndex))
- {
- ScBaseCell* pCell = pItems[nIndex].pCell;
- switch (pCell->GetCellType())
- {
- case CELLTYPE_VALUE:
- return ((ScValueCell*)pCell)->GetValue();
-// break;
- case CELLTYPE_FORMULA:
- {
- if (((ScFormulaCell*)pCell)->IsValue())
- return ((ScFormulaCell*)pCell)->GetValue();
- else
- return 0.0;
- }
-// break;
- default:
- return 0.0;
-// break;
- }
- }
- return 0.0;
-}
-
-
-void ScColumn::GetFormula( SCROW nRow, String& rFormula ) const
-{
- SCSIZE nIndex;
- if (Search(nRow, nIndex))
- {
- ScBaseCell* pCell = pItems[nIndex].pCell;
- if (pCell->GetCellType() == CELLTYPE_FORMULA)
- ((ScFormulaCell*)pCell)->GetFormula( rFormula );
- else
- rFormula.Erase();
- }
- else
- rFormula.Erase();
-}
-
-
-CellType ScColumn::GetCellType( SCROW nRow ) const
-{
- SCSIZE nIndex;
- if (Search(nRow, nIndex))
- return pItems[nIndex].pCell->GetCellType();
- return CELLTYPE_NONE;
-}
-
-
-sal_uInt16 ScColumn::GetErrCode( SCROW nRow ) const
-{
- SCSIZE nIndex;
- if (Search(nRow, nIndex))
- {
- ScBaseCell* pCell = pItems[nIndex].pCell;
- if (pCell->GetCellType() == CELLTYPE_FORMULA)
- return ((ScFormulaCell*)pCell)->GetErrCode();
- }
- return 0;
-}
-
-
-bool ScColumn::HasStringData( SCROW nRow ) const
-{
- SCSIZE nIndex;
- if (Search(nRow, nIndex))
- return (pItems[nIndex].pCell)->HasStringData();
- return false;
-}
-
-
-bool ScColumn::HasValueData( SCROW nRow ) const
-{
- SCSIZE nIndex;
- if (Search(nRow, nIndex))
- return (pItems[nIndex].pCell)->HasValueData();
- return false;
-}
-
-bool ScColumn::HasStringCells( SCROW nStartRow, SCROW nEndRow ) const
-{
- // TRUE, wenn String- oder Editzellen im Bereich
-
- if ( pItems )
- {
- SCSIZE nIndex;
- Search( nStartRow, nIndex );
- while ( nIndex < nCount && pItems[nIndex].nRow <= nEndRow )
- {
- CellType eType = pItems[nIndex].pCell->GetCellType();
- if ( eType == CELLTYPE_STRING || eType == CELLTYPE_EDIT )
- return sal_True;
- ++nIndex;
- }
- }
- return false;
-}
-
-
-ScPostIt* ScColumn::GetNote( SCROW nRow )
-{
- SCSIZE nIndex;
- return Search( nRow, nIndex ) ? pItems[ nIndex ].pCell->GetNote() : 0;
-}
-
-
-void ScColumn::TakeNote( SCROW nRow, ScPostIt* pNote )
-{
- SCSIZE nIndex;
- if( Search( nRow, nIndex ) )
- pItems[ nIndex ].pCell->TakeNote( pNote );
- else
- Insert( nRow, new ScNoteCell( pNote ) );
-}
-
-
-ScPostIt* ScColumn::ReleaseNote( SCROW nRow )
-{
- ScPostIt* pNote = 0;
- SCSIZE nIndex;
- if( Search( nRow, nIndex ) )
- {
- ScBaseCell* pCell = pItems[ nIndex ].pCell;
- pNote = pCell->ReleaseNote();
- if( (pCell->GetCellType() == CELLTYPE_NOTE) && !pCell->GetBroadcaster() )
- DeleteAtIndex( nIndex );
- }
- return pNote;
-}
-
-
-void ScColumn::DeleteNote( SCROW nRow )
-{
- delete ReleaseNote( nRow );
-}
-
-
-sal_Int32 ScColumn::GetMaxStringLen( SCROW nRowStart, SCROW nRowEnd, CharSet eCharSet ) const
-{
- sal_Int32 nStringLen = 0;
- if ( pItems )
- {
- String aString;
- rtl::OString aOString;
- bool bIsOctetTextEncoding = rtl_isOctetTextEncoding( eCharSet);
- SvNumberFormatter* pNumFmt = pDocument->GetFormatTable();
- SCSIZE nIndex;
- SCROW nRow;
- Search( nRowStart, nIndex );
- while ( nIndex < nCount && (nRow = pItems[nIndex].nRow) <= nRowEnd )
- {
- ScBaseCell* pCell = pItems[nIndex].pCell;
- if ( pCell->GetCellType() != CELLTYPE_NOTE )
- {
- Color* pColor;
- sal_uLong nFormat = (sal_uLong) ((SfxUInt32Item*) GetAttr(
- nRow, ATTR_VALUE_FORMAT ))->GetValue();
- ScCellFormat::GetString( pCell, nFormat, aString, &pColor,
- *pNumFmt );
- sal_Int32 nLen;
- if (bIsOctetTextEncoding)
- {
- rtl::OUString aOUString( aString);
- if (!aOUString.convertToString( &aOString, eCharSet,
- RTL_UNICODETOTEXT_FLAGS_UNDEFINED_ERROR |
- RTL_UNICODETOTEXT_FLAGS_INVALID_ERROR))
- {
- // TODO: anything? this is used by the dBase export filter
- // that throws an error anyway, but in case of another
- // context we might want to indicate a conversion error
- // early.
- }
- nLen = aOString.getLength();
- }
- else
- nLen = aString.Len() * sizeof(sal_Unicode);
- if ( nStringLen < nLen)
- nStringLen = nLen;
- }
- nIndex++;
- }
- }
- return nStringLen;
-}
-
-
-xub_StrLen ScColumn::GetMaxNumberStringLen(
- sal_uInt16& nPrecision, SCROW nRowStart, SCROW nRowEnd ) const
-{
- xub_StrLen nStringLen = 0;
- nPrecision = pDocument->GetDocOptions().GetStdPrecision();
- if ( nPrecision == SvNumberFormatter::UNLIMITED_PRECISION )
- // In case of unlimited precision, use 2 instead.
- nPrecision = 2;
-
- if ( pItems )
- {
- String aString;
- SvNumberFormatter* pNumFmt = pDocument->GetFormatTable();
- SCSIZE nIndex;
- SCROW nRow;
- Search( nRowStart, nIndex );
- while ( nIndex < nCount && (nRow = pItems[nIndex].nRow) <= nRowEnd )
- {
- ScBaseCell* pCell = pItems[nIndex].pCell;
- CellType eType = pCell->GetCellType();
- if ( eType == CELLTYPE_VALUE || (eType == CELLTYPE_FORMULA
- && ((ScFormulaCell*)pCell)->IsValue()) )
- {
- sal_uLong nFormat = (sal_uLong) ((SfxUInt32Item*) GetAttr(
- nRow, ATTR_VALUE_FORMAT ))->GetValue();
- ScCellFormat::GetInputString( pCell, nFormat, aString, *pNumFmt );
- xub_StrLen nLen = aString.Len();
- if ( nLen )
- {
- if ( nFormat )
- {
- const SvNumberformat* pEntry = pNumFmt->GetEntry( nFormat );
- sal_uInt16 nPrec;
- if (pEntry)
- {
- sal_Bool bThousand, bNegRed;
- sal_uInt16 nLeading;
- pEntry->GetFormatSpecialInfo(bThousand, bNegRed, nPrec, nLeading);
- }
- else
- nPrec = pNumFmt->GetFormatPrecision( nFormat );
-
- if ( nPrec != SvNumberFormatter::UNLIMITED_PRECISION && nPrec > nPrecision )
- nPrecision = nPrec;
- }
- if ( nPrecision )
- { // less than nPrecision in string => widen it
- // more => shorten it
- String aSep = pNumFmt->GetFormatDecimalSep( nFormat );
- xub_StrLen nTmp = aString.Search( aSep );
- if ( nTmp == STRING_NOTFOUND )
- nLen += nPrecision + aSep.Len();
- else
- {
- nTmp = aString.Len() - (nTmp + aSep.Len());
- if ( nTmp != nPrecision )
- nLen += nPrecision - nTmp;
- // nPrecision > nTmp : nLen + Diff
- // nPrecision < nTmp : nLen - Diff
- }
- }
- if ( nStringLen < nLen )
- nStringLen = nLen;
- }
- }
- nIndex++;
- }
- }
- return nStringLen;
-}
-
-/* vim:set shiftwidth=4 softtabstop=4 expandtab: */