blob: 2819aac5062057da23296e13135107447385ab33 [file] [log] [blame]
/**************************************************************
*
* 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
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*
*************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_sc.hxx"
// INCLUDE ---------------------------------------------------------------
#include <sfx2/objsh.hxx>
#include <svl/zforlist.hxx>
#include <svl/zformat.hxx>
#include "boost/tuple/tuple.hpp"
#include <boost/function.hpp>
#include "boost/lambda/bind.hpp"
#include "boost/bind.hpp"
#include "boost/lambda/lambda.hpp"
#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 "dpglobal.hxx"
#include <dptablecache.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 -----------------------------------------------------------
sal_Bool ScColumn::bDoubleAlloc = sal_False; // fuer Import: Groesse beim Allozieren verdoppeln
void ScColumn::Insert( SCROW nRow, ScBaseCell* pNewCell )
{
sal_Bool bIsAppended = sal_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_uLong 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( sal_False ); // Mehrfachberechnungen vermeiden
sal_Bool bFound=sal_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;
// #43940# Aenderung Quelle broadcasten
rAddress.SetRow( nOldRow );
pDocument->AreaBroadcast( aHint );
SCROW nNewRow = (pItems[i].nRow -= nSize);
// #43940# 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();
// special simple mode if all contents are deleted and cells do not contain broadcasters
bool bSimple = ((nDelFlag & IDF_CONTENTS) == IDF_CONTENTS);
if (bSimple)
for ( SCSIZE nIdx = nStartIndex; bSimple && (nIdx <= nEndIndex); ++nIdx )
if (pItems[ nIdx ].pCell->GetBroadcaster())
bSimple = false;
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 );
// simple deletion of the cell objects
if (bSimple)
{
// pNoteCell: dummy replacement for old cells, to prevent that interpreter uses old cell
ScNoteCell* pNoteCell = new ScNoteCell;
for ( SCSIZE nIdx = nStartIndex; nIdx <= nEndIndex; ++nIdx )
{
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 = pNoteCell;
aHint.GetAddress().SetRow( pItems[ nIdx ].nRow );
aHint.SetCell( pOldCell );
pDocument->Broadcast( aHint );
pOldCell->Delete();
}
}
delete pNoteCell;
memmove( &pItems[nStartIndex], &pItems[nEndIndex + 1], (nCount - nEndIndex - 1) * sizeof(ColEntry) );
nCount -= nEndIndex-nStartIndex+1;
}
// else: delete some contents of the cells
else
{
SCSIZE j = nStartIndex;
for ( SCSIZE nIdx = nStartIndex; nIdx <= nEndIndex; ++nIdx )
{
// decide whether to delete the cell object according to passed flags
bool bDelete = false;
ScBaseCell* pOldCell = pItems[j].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[j].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[j].nRow;
if (pNoteCell)
{
// replace old cell with the replacement note cell
pItems[j].pCell = pNoteCell;
++j;
}
else
{
// remove the old cell from the cell item list
--nCount;
memmove( &pItems[j], &pItems[j + 1], (nCount - j) * sizeof(ColEntry) );
pItems[nCount].nRow = 0;
pItems[nCount].pCell = 0;
}
// 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[j].pCell->DeleteNote();
// cell not deleted, move index to next cell
++j;
}
}
}
// *** 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=sal_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 )
{
DBG_ASSERT( 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 = sal_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(sal_True);
//! 3D(sal_False) und TabRel(sal_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, sal_Bool bAsLink, sal_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(sal_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 = sal_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)
{
DBG_ASSERT( !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 bCloneFormula = (nFlags & IDF_FORMULA) != 0;
bool bCloneNote = (nFlags & IDF_NOTE) != 0;
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 (bCloneFormula)
{
// note will be cloned below
pNew = rSource.CloneWithoutNote( rDestDoc, rDestPos );
}
else if ( (bCloneValue || bCloneDateTime || bCloneString) && !rDestDoc.IsUndo() )
{
// #48491# 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 );
// #33224# do not clone empty string
if (aString.Len() > 0)
{
if ( rForm.IsMultilineResult() )
{
pNew = new ScEditCell( aString, &rDestDoc );
}
else
{
pNew = new ScStringCell( aString );
}
}
}
}
break;
default: DBG_ERRORFILE( "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,
sal_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 = sal_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, sal_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 = sal_False;
if ( nSrcIndex < nSrcCount && nNextSrc == nRow )
pSrc = rSrcCol.pItems[nSrcIndex].pCell;
if ( nIndex < nCount && nNextThis == nRow )
pDest = pItems[nIndex].pCell;
DBG_ASSERT( 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++;
}
}
}
// sal_True = Zahlformat gesetzt
sal_Bool ScColumn::SetString( SCROW nRow, SCTAB nTabP, const String& rString,
formula::FormulaGrammar::AddressConvention eConv,
SvNumberFormatter* pLangFormatter, bool bDetectNumberFormat )
{
sal_Bool bNumFmtSet = sal_False;
if (VALIDROW(nRow))
{
ScBaseCell* pNewCell = NULL;
sal_Bool bIsLoading = sal_False;
if (rString.Len() > 0)
{
double nVal;
sal_uInt32 nIndex, nOldIndex = 0;
sal_Unicode cFirstChar;
// #i110979# If a different NumberFormatter is passed in (pLangFormatter),
// its formats aren't valid in the document.
// Only use the language / LocaleDataWrapper from pLangFormatter,
// always the document's number formatter for IsNumberFormat.
SvNumberFormatter* pFormatter = 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
&& pFormatter->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
pNewCell = new ScStringCell( rString.Copy(1) );
else
{
sal_Bool bIsText = sal_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 = sal_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 = pFormatter->GetStandardIndex();
}
do
{
if (bIsText)
break;
if (bDetectNumberFormat)
{
if ( pLangFormatter )
{
// for number detection: valid format index for selected language
nIndex = pFormatter->GetStandardIndex( pLangFormatter->GetLanguage() );
}
if (!pFormatter->IsNumberFormat(rString, nIndex, nVal))
break;
if ( pLangFormatter )
{
// convert back to the original language if a built-in format was detected
const SvNumberformat* pOldFormat = pFormatter->GetEntry( nOldIndex );
if ( pOldFormat )
nIndex = pFormatter->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 = sal_False;
const SvNumberformat* pOldFormat = pFormatter->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 == pFormatter->GetStandardFormat(
nOldType, pOldFormat->GetLanguage() ) )
{
bOverwrite = sal_True; // default of these types can be overwritten
}
}
}
if ( !bOverwrite && pFormatter->GetType( nIndex ) == NUMBERFORMAT_LOGICAL )
{
bOverwrite = sal_True; // overwrite anything if boolean was detected
}
if ( bOverwrite )
{
ApplyAttr( nRow, SfxUInt32Item( ATTR_VALUE_FORMAT,
(sal_uInt32) nIndex) );
bNumFmtSet = sal_True;
}
}
}
else
{
// Only check if the string is a regular number.
SvNumberFormatter* pLocaleSource = pLangFormatter ? pLangFormatter : pFormatter;
const LocaleDataWrapper* pLocale = pLocaleSource->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)
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) : sal_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 0 // DR
ScPostIt aCellNote( ScPostIt::UNINITIALIZED );
// Hide visible notes during Filtering.
if(pCell->GetNote(aCellNote) && aCellNote.IsCaptionShown())
{
ScDetectiveFunc( pDocument, nTab ).HideComment( nCol, nRow );
aCellNote.SetShown( false );
pCell->SetNote(aCellNote);
}
#endif
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
sal_Bool ScColumn::GetDataEntries(SCROW nStartRow, TypedScStrCollection& rStrings, sal_Bool bLimit)
{
sal_Bool bFound = sal_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 = sal_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 = sal_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();
}
template<>
void ScColumn::FillDPCacheT( long nDim, SCROW nStartRow, SCROW nEndRow, const boost::function<void(ScDPItemData*)> & rAddLabel, const boost::function<sal_Bool(long,ScDPItemData*, bool)> & rAddData )
{
SCROW nPattenRowStart = -1, nPatternRowEnd = -1;
SvNumberFormatter* pFormatter = pDocument->GetFormatTable();
sal_uLong nNumberFormat = 0;
sal_uLong nNumberFormatType = NUMBERFORMAT_NUMBER;
SCROW nCurRow = nStartRow;
ScDPItemData * pDPItemData = NULL;
if ( pItems )
{
SCSIZE nIndex;
for ( Search( nStartRow, nIndex ) ? void( ) : void(nIndex = nCount); nIndex < nCount && pItems[nIndex].nRow <= nEndRow; ++nIndex, ++nCurRow )
{
for( ; nCurRow < pItems[nIndex].nRow; nCurRow++ )
if( nCurRow == nStartRow )
rAddLabel( new ScDPItemData() );
else
rAddData( nDim, new ScDPItemData(), false);
if( nCurRow > nPatternRowEnd )
if( const ScPatternAttr* pPattern = pAttrArray ? pAttrArray->GetPatternRange( nPattenRowStart, nPatternRowEnd, nCurRow ) : NULL )
nNumberFormatType = pFormatter->GetType( nNumberFormat = pPattern->GetNumberFormat( pFormatter ) );
else
nNumberFormatType = NUMBERFORMAT_NUMBER, nNumberFormat = 0;
if( ScBaseCell* pCell = pItems[nIndex].pCell )
if( pCell->GetCellType() == CELLTYPE_FORMULA && ((ScFormulaCell*)pCell)->GetErrCode() )
{
String str( GetStringFromCell( pCell, nNumberFormat, pFormatter ) );
sal_uInt8 bFlag = ScDPItemData::MK_ERR;
pDPItemData = new ScDPItemData( 0, str, 0.0, bFlag );
}
else if( pCell->HasValueData() )
{
double fVal = GetValueFromCell( pCell );
String str( GetStringFromCell( pCell, nNumberFormat, pFormatter ) );
sal_uInt8 bFlag = ScDPItemData::MK_VAL|ScDPItemData::MK_DATA|(ScDPItemData::MK_DATE * isDateFormat( nNumberFormatType ));
pDPItemData = new ScDPItemData( nNumberFormat, str, fVal, bFlag );
}
else if( !pCell->IsBlank() )
pDPItemData = new ScDPItemData( GetStringFromCell( pCell, nNumberFormat, pFormatter ) );
else
pDPItemData = new ScDPItemData();
else
pDPItemData = new ScDPItemData();
if( nCurRow == nStartRow )
rAddLabel( pDPItemData );
else
rAddData( nDim, pDPItemData, false );
}
}
for( ; nCurRow <= nEndRow; nCurRow++ )
if( nCurRow == nStartRow )
rAddLabel( new ScDPItemData() );
else
rAddData( nDim, new ScDPItemData(), false );
}
void ScColumn::FillDPCache( ScDPTableDataCache * pCache, long nDim, SCROW nStartRow, SCROW nEndRow )
{
FillDPCacheT<boost::function<void(ScDPItemData*)>, boost::function<sal_Bool(long,ScDPItemData*, bool)> >( nDim, nStartRow, nEndRow, boost::bind( &ScDPTableDataCache::AddLabel, pCache, _1 ), boost::bind( &ScDPTableDataCache::AddData, pCache, _1, _2, _3 ) );
}
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, sal_Bool ) 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;
}
sal_Bool ScColumn::HasStringData( SCROW nRow ) const
{
SCSIZE nIndex;
if (Search(nRow, nIndex))
return (pItems[nIndex].pCell)->HasStringData();
return sal_False;
}
sal_Bool ScColumn::HasValueData( SCROW nRow ) const
{
SCSIZE nIndex;
if (Search(nRow, nIndex))
return (pItems[nIndex].pCell)->HasValueData();
return sal_False;
}
sal_Bool ScColumn::HasStringCells( SCROW nStartRow, SCROW nEndRow ) const
{
// sal_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 sal_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 )
{ // more decimals than standard?
sal_uInt16 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;
}