LCOV - code coverage report
Current view: top level - sc/source/core/tool - rangenam.cxx (source / functions) Hit Total Coverage
Test: commit 10e77ab3ff6f4314137acd6e2702a6e5c1ce1fae Lines: 318 427 74.5 %
Date: 2014-11-03 Functions: 56 63 88.9 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
       2             : /*
       3             :  * This file is part of the LibreOffice project.
       4             :  *
       5             :  * This Source Code Form is subject to the terms of the Mozilla Public
       6             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       7             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       8             :  *
       9             :  * This file incorporates work covered by the following license notice:
      10             :  *
      11             :  *   Licensed to the Apache Software Foundation (ASF) under one or more
      12             :  *   contributor license agreements. See the NOTICE file distributed
      13             :  *   with this work for additional information regarding copyright
      14             :  *   ownership. The ASF licenses this file to you under the Apache
      15             :  *   License, Version 2.0 (the "License"); you may not use this file
      16             :  *   except in compliance with the License. You may obtain a copy of
      17             :  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
      18             :  */
      19             : 
      20             : #include <string.h>
      21             : #include <boost/scoped_ptr.hpp>
      22             : #include <unotools/collatorwrapper.hxx>
      23             : #include <unotools/transliterationwrapper.hxx>
      24             : #include <com/sun/star/sheet/NamedRangeFlag.hpp>
      25             : 
      26             : #include "token.hxx"
      27             : #include "tokenarray.hxx"
      28             : #include "rangenam.hxx"
      29             : #include "global.hxx"
      30             : #include "compiler.hxx"
      31             : #include "rangeutl.hxx"
      32             : #include "rechead.hxx"
      33             : #include "refupdat.hxx"
      34             : #include "document.hxx"
      35             : #include "refupdatecontext.hxx"
      36             : #include <tokenstringcontext.hxx>
      37             : 
      38             : #include <formula/errorcodes.hxx>
      39             : 
      40             : using namespace formula;
      41             : using ::std::pair;
      42             : using ::std::unary_function;
      43             : 
      44             : // ScRangeData
      45             : 
      46         230 : ScRangeData::ScRangeData( ScDocument* pDok,
      47             :                           const OUString& rName,
      48             :                           const OUString& rSymbol,
      49             :                           const ScAddress& rAddress,
      50             :                           RangeType nType,
      51             :                           const FormulaGrammar::Grammar eGrammar ) :
      52             :                 aName       ( rName ),
      53             :                 aUpperName  ( ScGlobal::pCharClass->uppercase( rName ) ),
      54             :                 pCode       ( NULL ),
      55             :                 aPos        ( rAddress ),
      56             :                 eType       ( nType ),
      57             :                 pDoc        ( pDok ),
      58             :                 eTempGrammar( eGrammar ),
      59             :                 nIndex      ( 0 ),
      60             :                 bModified   ( false ),
      61             :                 mnMaxRow    (-1),
      62         230 :                 mnMaxCol    (-1)
      63             : {
      64         230 :     if (!rSymbol.isEmpty())
      65         230 :         CompileRangeData( rSymbol, pDoc->IsImportingXML());
      66             :         // Let the compiler set an error on unknown names for a subsequent
      67             :         // CompileUnresolvedXML().
      68             :     else
      69             :     {
      70             :         // #i63513#/#i65690# don't leave pCode as NULL.
      71             :         // Copy ctor default-constructs pCode if it was NULL, so it's initialized here, too,
      72             :         // to ensure same behavior if unnecessary copying is left out.
      73             : 
      74           0 :         pCode = new ScTokenArray();
      75             :     }
      76         230 : }
      77             : 
      78         344 : ScRangeData::ScRangeData( ScDocument* pDok,
      79             :                           const OUString& rName,
      80             :                           const ScTokenArray& rArr,
      81             :                           const ScAddress& rAddress,
      82             :                           RangeType nType ) :
      83             :                 aName       ( rName ),
      84             :                 aUpperName  ( ScGlobal::pCharClass->uppercase( rName ) ),
      85         344 :                 pCode       ( new ScTokenArray( rArr ) ),
      86             :                 aPos        ( rAddress ),
      87             :                 eType       ( nType ),
      88             :                 pDoc        ( pDok ),
      89             :                 eTempGrammar( FormulaGrammar::GRAM_UNSPECIFIED ),
      90             :                 nIndex      ( 0 ),
      91             :                 bModified   ( false ),
      92             :                 mnMaxRow    (-1),
      93         688 :                 mnMaxCol    (-1)
      94             : {
      95         344 :     InitCode();
      96         344 : }
      97             : 
      98           6 : ScRangeData::ScRangeData( ScDocument* pDok,
      99             :                           const OUString& rName,
     100             :                           const ScAddress& rTarget ) :
     101             :                 aName       ( rName ),
     102             :                 aUpperName  ( ScGlobal::pCharClass->uppercase( rName ) ),
     103           0 :                 pCode       ( new ScTokenArray() ),
     104             :                 aPos        ( rTarget ),
     105             :                 eType       ( RT_NAME ),
     106             :                 pDoc        ( pDok ),
     107             :                 eTempGrammar( FormulaGrammar::GRAM_UNSPECIFIED ),
     108             :                 nIndex      ( 0 ),
     109             :                 bModified   ( false ),
     110             :                 mnMaxRow    (-1),
     111           6 :                 mnMaxCol    (-1)
     112             : {
     113             :     ScSingleRefData aRefData;
     114           6 :     aRefData.InitAddress( rTarget );
     115           6 :     aRefData.SetFlag3D( true );
     116           6 :     pCode->AddSingleReference( aRefData );
     117           6 :     ScCompiler aComp( pDoc, aPos, *pCode );
     118           6 :     aComp.SetGrammar(pDoc->GetGrammar());
     119           6 :     aComp.CompileTokenArray();
     120           6 :     if ( !pCode->GetCodeError() )
     121           6 :         eType |= RT_ABSPOS;
     122           6 : }
     123             : 
     124         848 : ScRangeData::ScRangeData(const ScRangeData& rScRangeData, ScDocument* pDocument) :
     125             :     aName   (rScRangeData.aName),
     126             :     aUpperName  (rScRangeData.aUpperName),
     127         848 :     pCode       (rScRangeData.pCode ? rScRangeData.pCode->Clone() : new ScTokenArray()),   // make real copy (not copy-ctor)
     128             :     aPos        (rScRangeData.aPos),
     129             :     eType       (rScRangeData.eType),
     130             :     pDoc        (pDocument ? pDocument : rScRangeData.pDoc),
     131             :     eTempGrammar(rScRangeData.eTempGrammar),
     132             :     nIndex      (rScRangeData.nIndex),
     133             :     bModified   (rScRangeData.bModified),
     134             :     mnMaxRow    (rScRangeData.mnMaxRow),
     135        1696 :     mnMaxCol    (rScRangeData.mnMaxCol)
     136         848 : {}
     137             : 
     138        2856 : ScRangeData::~ScRangeData()
     139             : {
     140        1428 :     delete pCode;
     141        1428 : }
     142             : 
     143         232 : void ScRangeData::CompileRangeData( const OUString& rSymbol, bool bSetError )
     144             : {
     145         232 :     if (eTempGrammar == FormulaGrammar::GRAM_UNSPECIFIED)
     146             :     {
     147             :         OSL_FAIL( "ScRangeData::CompileRangeData: unspecified grammar");
     148             :         // Anything is almost as bad as this, but we might have the best choice
     149             :         // if not loading documents.
     150           0 :         eTempGrammar = FormulaGrammar::GRAM_NATIVE;
     151             :     }
     152             : 
     153         232 :     ScCompiler aComp( pDoc, aPos );
     154         232 :     aComp.SetGrammar( eTempGrammar);
     155         232 :     if (bSetError)
     156         124 :         aComp.SetExtendedErrorDetection( ScCompiler::EXTENDED_ERROR_DETECTION_NAME_NO_BREAK);
     157         232 :     ScTokenArray* pNewCode = aComp.CompileString( rSymbol );
     158         464 :     boost::scoped_ptr<ScTokenArray> pOldCode( pCode);     // old pCode will be deleted
     159         232 :     pCode = pNewCode;
     160         232 :     if( !pCode->GetCodeError() )
     161             :     {
     162         230 :         pCode->Reset();
     163         230 :         FormulaToken* p = pCode->GetNextReference();
     164         230 :         if( p )
     165             :         {
     166             :             // first token is a reference
     167             :             /* FIXME: wouldn't that need a check if it's exactly one reference? */
     168         180 :             if( p->GetType() == svSingleRef )
     169         118 :                 eType = eType | RT_ABSPOS;
     170             :             else
     171          62 :                 eType = eType | RT_ABSAREA;
     172             :         }
     173             :         // For manual input set an error for an incomplete formula.
     174         230 :         if (!pDoc->IsImportingXML())
     175             :         {
     176         106 :             aComp.CompileTokenArray();
     177         106 :             pCode->DelRPN();
     178             :         }
     179         232 :     }
     180         232 : }
     181             : 
     182         124 : void ScRangeData::CompileUnresolvedXML( sc::CompileFormulaContext& rCxt )
     183             : {
     184         124 :     if (pCode->GetCodeError() == errNoName)
     185             :     {
     186             :         // Reconstruct the symbol/formula and then recompile.
     187           2 :         OUString aSymbol;
     188           2 :         rCxt.setGrammar(eTempGrammar);
     189           4 :         ScCompiler aComp(rCxt, aPos, *pCode);
     190           2 :         aComp.CreateStringFromTokenArray( aSymbol);
     191             :         // Don't let the compiler set an error for unknown names on final
     192             :         // compile, errors are handled by the interpreter thereafter.
     193           4 :         CompileRangeData( aSymbol, false);
     194             :     }
     195         124 : }
     196             : 
     197             : #if DEBUG_FORMULA_COMPILER
     198             : void ScRangeData::Dump() const
     199             : {
     200             :     cout << "-- ScRangeData" << endl;
     201             :     cout << "  name: " << aName << endl;
     202             :     cout << "  ref position: (col=" << aPos.Col() << ", row=" << aPos.Row() << ", sheet=" << aPos.Tab() << ")" << endl;
     203             : 
     204             :     if (pCode)
     205             :         pCode->Dump();
     206             : }
     207             : #endif
     208             : 
     209         338 : void ScRangeData::GuessPosition()
     210             : {
     211             :     // set a position that allows "absoluting" of all relative references
     212             :     // in CalcAbsIfRel without errors
     213             : 
     214             :     OSL_ENSURE(aPos == ScAddress(), "position will go lost now");
     215             : 
     216         338 :     SCsCOL nMinCol = 0;
     217         338 :     SCsROW nMinRow = 0;
     218         338 :     SCsTAB nMinTab = 0;
     219             : 
     220             :     formula::FormulaToken* t;
     221         338 :     pCode->Reset();
     222         884 :     while ( ( t = pCode->GetNextReference() ) != NULL )
     223             :     {
     224         208 :         ScSingleRefData& rRef1 = *t->GetSingleRef();
     225         208 :         if ( rRef1.IsColRel() && rRef1.Col() < nMinCol )
     226           0 :             nMinCol = rRef1.Col();
     227         208 :         if ( rRef1.IsRowRel() && rRef1.Row() < nMinRow )
     228           0 :             nMinRow = rRef1.Row();
     229         208 :         if ( rRef1.IsTabRel() && rRef1.Tab() < nMinTab )
     230           0 :             nMinTab = rRef1.Tab();
     231             : 
     232         208 :         if ( t->GetType() == svDoubleRef )
     233             :         {
     234         112 :             ScSingleRefData& rRef2 = t->GetDoubleRef()->Ref2;
     235         112 :             if ( rRef2.IsColRel() && rRef2.Col() < nMinCol )
     236           0 :                 nMinCol = rRef2.Col();
     237         112 :             if ( rRef2.IsRowRel() && rRef2.Row() < nMinRow )
     238           0 :                 nMinRow = rRef2.Row();
     239         112 :             if ( rRef2.IsTabRel() && rRef2.Tab() < nMinTab )
     240           0 :                 nMinTab = rRef2.Tab();
     241             :         }
     242             :     }
     243             : 
     244         338 :     aPos = ScAddress( (SCCOL)(-nMinCol), (SCROW)(-nMinRow), (SCTAB)(-nMinTab) );
     245         338 : }
     246             : 
     247         110 : void ScRangeData::GetSymbol( OUString& rSymbol, const FormulaGrammar::Grammar eGrammar ) const
     248             : {
     249         110 :     ScCompiler aComp(pDoc, aPos, *pCode);
     250         110 :     aComp.SetGrammar(eGrammar);
     251         110 :     aComp.CreateStringFromTokenArray( rSymbol );
     252         110 : }
     253             : 
     254           0 : void ScRangeData::GetSymbol( OUString& rSymbol, const ScAddress& rPos, const FormulaGrammar::Grammar eGrammar ) const
     255             : {
     256           0 :     OUString aStr;
     257           0 :     ScCompiler aComp(pDoc, rPos, *pCode);
     258           0 :     aComp.SetGrammar(eGrammar);
     259           0 :     aComp.CreateStringFromTokenArray( aStr );
     260           0 :     rSymbol = aStr;
     261           0 : }
     262             : 
     263          42 : void ScRangeData::UpdateSymbol( OUStringBuffer& rBuffer, const ScAddress& rPos,
     264             :                                 const FormulaGrammar::Grammar eGrammar )
     265             : {
     266          42 :     boost::scoped_ptr<ScTokenArray> pTemp( pCode->Clone() );
     267          84 :     ScCompiler aComp( pDoc, rPos, *pTemp.get());
     268          42 :     aComp.SetGrammar(eGrammar);
     269          42 :     aComp.MoveRelWrap(GetMaxCol(), GetMaxRow());
     270          84 :     aComp.CreateStringFromTokenArray( rBuffer );
     271          42 : }
     272             : 
     273          62 : void ScRangeData::UpdateReference( sc::RefUpdateContext& rCxt, SCTAB nLocalTab )
     274             : {
     275          62 :     sc::RefUpdateResult aRes = pCode->AdjustReferenceInName(rCxt, aPos);
     276          62 :     bModified = aRes.mbReferenceModified;
     277          62 :     if (aRes.mbReferenceModified)
     278          32 :         rCxt.maUpdatedNames.setUpdatedName(nLocalTab, nIndex);
     279          62 : }
     280             : 
     281           0 : void ScRangeData::UpdateTranspose( const ScRange& rSource, const ScAddress& rDest )
     282             : {
     283           0 :     bool bChanged = false;
     284             : 
     285             :     formula::FormulaToken* t;
     286           0 :     pCode->Reset();
     287             : 
     288           0 :     while ( ( t = pCode->GetNextReference() ) != NULL )
     289             :     {
     290           0 :         if( t->GetType() != svIndex )
     291             :         {
     292           0 :             SingleDoubleRefModifier aMod( *t );
     293           0 :             ScComplexRefData& rRef = aMod.Ref();
     294           0 :             if (!rRef.Ref1.IsColRel() && !rRef.Ref1.IsRowRel() &&
     295           0 :                     (!rRef.Ref1.IsFlag3D() || !rRef.Ref1.IsTabRel()) &&
     296           0 :                 ( t->GetType() == svSingleRef ||
     297           0 :                 (!rRef.Ref2.IsColRel() && !rRef.Ref2.IsRowRel() &&
     298           0 :                     (!rRef.Ref2.IsFlag3D() || !rRef.Ref2.IsTabRel()))))
     299             :             {
     300           0 :                 ScRange aAbs = rRef.toAbs(aPos);
     301           0 :                 if (ScRefUpdate::UpdateTranspose(pDoc, rSource, rDest, aAbs) != UR_NOTHING)
     302             :                 {
     303           0 :                     rRef.SetRange(aAbs, aPos);
     304           0 :                     bChanged = true;
     305             :                 }
     306           0 :             }
     307             :         }
     308             :     }
     309             : 
     310           0 :     bModified = bChanged;
     311           0 : }
     312             : 
     313           0 : void ScRangeData::UpdateGrow( const ScRange& rArea, SCCOL nGrowX, SCROW nGrowY )
     314             : {
     315           0 :     bool bChanged = false;
     316             : 
     317             :     formula::FormulaToken* t;
     318           0 :     pCode->Reset();
     319             : 
     320           0 :     while ( ( t = pCode->GetNextReference() ) != NULL )
     321             :     {
     322           0 :         if( t->GetType() != svIndex )
     323             :         {
     324           0 :             SingleDoubleRefModifier aMod( *t );
     325           0 :             ScComplexRefData& rRef = aMod.Ref();
     326           0 :             if (!rRef.Ref1.IsColRel() && !rRef.Ref1.IsRowRel() &&
     327           0 :                     (!rRef.Ref1.IsFlag3D() || !rRef.Ref1.IsTabRel()) &&
     328           0 :                 ( t->GetType() == svSingleRef ||
     329           0 :                 (!rRef.Ref2.IsColRel() && !rRef.Ref2.IsRowRel() &&
     330           0 :                     (!rRef.Ref2.IsFlag3D() || !rRef.Ref2.IsTabRel()))))
     331             :             {
     332           0 :                 ScRange aAbs = rRef.toAbs(aPos);
     333           0 :                 if (ScRefUpdate::UpdateGrow(rArea, nGrowX, nGrowY, aAbs) != UR_NOTHING)
     334             :                 {
     335           0 :                     rRef.SetRange(aAbs, aPos);
     336           0 :                     bChanged = true;
     337             :                 }
     338           0 :             }
     339             :         }
     340             :     }
     341             : 
     342           0 :     bModified = bChanged;           // has to be evaluated immediately afterwards
     343           0 : }
     344             : 
     345          30 : bool ScRangeData::operator== (const ScRangeData& rData) const       // for Undo
     346             : {
     347          90 :     if ( nIndex != rData.nIndex ||
     348          60 :          aName  != rData.aName  ||
     349          88 :          aPos   != rData.aPos   ||
     350          30 :          eType  != rData.eType     ) return false;
     351             : 
     352          28 :     sal_uInt16 nLen = pCode->GetLen();
     353          28 :     if ( nLen != rData.pCode->GetLen() ) return false;
     354             : 
     355          28 :     FormulaToken** ppThis = pCode->GetArray();
     356          28 :     FormulaToken** ppOther = rData.pCode->GetArray();
     357             : 
     358          66 :     for ( sal_uInt16 i=0; i<nLen; i++ )
     359          44 :         if ( ppThis[i] != ppOther[i] && !(*ppThis[i] == *ppOther[i]) )
     360           6 :             return false;
     361             : 
     362          22 :     return true;
     363             : }
     364             : 
     365         722 : bool ScRangeData::IsRangeAtBlock( const ScRange& rBlock ) const
     366             : {
     367         722 :     bool bRet = false;
     368         722 :     ScRange aRange;
     369         722 :     if ( IsReference(aRange) )
     370         602 :         bRet = ( rBlock == aRange );
     371         722 :     return bRet;
     372             : }
     373             : 
     374         738 : bool ScRangeData::IsReference( ScRange& rRange ) const
     375             : {
     376         738 :     if ( (eType & ( RT_ABSAREA | RT_REFAREA | RT_ABSPOS )) && pCode )
     377         738 :         return pCode->IsReference(rRange, aPos);
     378             : 
     379           0 :     return false;
     380             : }
     381             : 
     382           0 : bool ScRangeData::IsReference( ScRange& rRange, const ScAddress& rPos ) const
     383             : {
     384           0 :     if ( (eType & ( RT_ABSAREA | RT_REFAREA | RT_ABSPOS ) ) && pCode )
     385           0 :         return pCode->IsReference(rRange, rPos);
     386             : 
     387           0 :     return false;
     388             : }
     389             : 
     390         942 : bool ScRangeData::IsValidReference( ScRange& rRange ) const
     391             : {
     392         942 :     if ( (eType & ( RT_ABSAREA | RT_REFAREA | RT_ABSPOS ) ) && pCode )
     393         942 :         return pCode->IsValidReference(rRange, aPos);
     394             : 
     395           0 :     return false;
     396             : }
     397             : 
     398         120 : void ScRangeData::UpdateInsertTab( sc::RefUpdateInsertTabContext& rCxt, SCTAB nLocalTab )
     399             : {
     400         120 :     sc::RefUpdateResult aRes = pCode->AdjustReferenceOnInsertedTab(rCxt, aPos);
     401         120 :     if (aRes.mbReferenceModified)
     402         102 :         rCxt.maUpdatedNames.setUpdatedName(nLocalTab, nIndex);
     403             : 
     404         120 :     if (rCxt.mnInsertPos <= aPos.Tab())
     405         120 :         aPos.IncTab(rCxt.mnSheets);
     406         120 : }
     407             : 
     408          46 : void ScRangeData::UpdateDeleteTab( sc::RefUpdateDeleteTabContext& rCxt, SCTAB nLocalTab )
     409             : {
     410          46 :     sc::RefUpdateResult aRes = pCode->AdjustReferenceOnDeletedTab(rCxt, aPos);
     411          46 :     if (aRes.mbReferenceModified)
     412          10 :         rCxt.maUpdatedNames.setUpdatedName(nLocalTab, nIndex);
     413             : 
     414          46 :     if (rCxt.mnDeletePos <= aPos.Tab())
     415          10 :         aPos.IncTab(-rCxt.mnSheets);
     416          46 : }
     417             : 
     418           2 : void ScRangeData::UpdateMoveTab( sc::RefUpdateMoveTabContext& rCxt, SCTAB nLocalTab )
     419             : {
     420           2 :     sc::RefUpdateResult aRes = pCode->AdjustReferenceOnMovedTab(rCxt, aPos);
     421           2 :     if (aRes.mbReferenceModified)
     422           2 :         rCxt.maUpdatedNames.setUpdatedName(nLocalTab, nIndex);
     423             : 
     424           2 :     aPos.SetTab(rCxt.getNewTab(aPos.Tab()));
     425           2 : }
     426             : 
     427          16 : void ScRangeData::MakeValidName( OUString& rName )
     428             : {
     429             : 
     430             :     // strip leading invalid characters
     431          16 :     sal_Int32 nPos = 0;
     432          16 :     sal_Int32 nLen = rName.getLength();
     433          32 :     while ( nPos < nLen && !ScCompiler::IsCharFlagAllConventions( rName, nPos, SC_COMPILER_C_NAME) )
     434           0 :         ++nPos;
     435          16 :     if ( nPos>0 )
     436           0 :         rName = rName.copy(nPos);
     437             : 
     438             :     // if the first character is an invalid start character, precede with '_'
     439          16 :     if ( !rName.isEmpty() && !ScCompiler::IsCharFlagAllConventions( rName, 0, SC_COMPILER_C_CHAR_NAME ) )
     440           0 :         rName = "_" + rName;
     441             : 
     442             :     // replace invalid with '_'
     443          16 :     nLen = rName.getLength();
     444          82 :     for (nPos=0; nPos<nLen; nPos++)
     445             :     {
     446          66 :         if ( !ScCompiler::IsCharFlagAllConventions( rName, nPos, SC_COMPILER_C_NAME) )
     447           0 :             rName = rName.replaceAt( nPos, 1, "_" );
     448             :     }
     449             : 
     450             :     // Ensure that the proposed name is not a reference under any convention,
     451             :     // same as in IsNameValid()
     452          16 :     ScAddress aAddr;
     453          16 :     ScRange aRange;
     454         128 :     for (int nConv = FormulaGrammar::CONV_UNSPECIFIED; ++nConv < FormulaGrammar::CONV_LAST; )
     455             :     {
     456          96 :         ScAddress::Details details( static_cast<FormulaGrammar::AddressConvention>( nConv ) );
     457             :         // Don't check Parse on VALID, any partial only VALID may result in
     458             :         // #REF! during compile later!
     459         192 :         while (aRange.Parse( rName, NULL, details) || aAddr.Parse( rName, NULL, details))
     460             :         {
     461             :             //! Range Parse is partially valid also with invalid sheet name,
     462             :             //! Address Parse dito, during compile name would generate a #REF!
     463           0 :             if ( rName.indexOf( '.' ) == -1 )
     464           0 :                 rName = rName.replaceFirst( ".", "_" );
     465             :             else
     466           0 :                 rName = "_" + rName;
     467             :         }
     468             :     }
     469          16 : }
     470             : 
     471          10 : bool ScRangeData::IsNameValid( const OUString& rName, ScDocument* pDoc )
     472             : {
     473             :     /* XXX If changed, sc/source/filter/ftools/ftools.cxx
     474             :      * ScfTools::ConvertToScDefinedName needs to be changed too. */
     475          10 :     sal_Char a('.');
     476          10 :     if (rName.indexOf(a) != -1)
     477           0 :         return false;
     478          10 :     sal_Int32 nPos = 0;
     479          10 :     sal_Int32 nLen = rName.getLength();
     480          10 :     if ( !nLen || !ScCompiler::IsCharFlagAllConventions( rName, nPos++, SC_COMPILER_C_CHAR_NAME ) )
     481           0 :         return false;
     482         114 :     while ( nPos < nLen )
     483             :     {
     484          94 :         if ( !ScCompiler::IsCharFlagAllConventions( rName, nPos++, SC_COMPILER_C_NAME ) )
     485           0 :             return false;
     486             :     }
     487          10 :     ScAddress aAddr;
     488          10 :     ScRange aRange;
     489          80 :     for (int nConv = FormulaGrammar::CONV_UNSPECIFIED; ++nConv < FormulaGrammar::CONV_LAST; )
     490             :     {
     491          60 :         ScAddress::Details details( static_cast<FormulaGrammar::AddressConvention>( nConv ) );
     492             :         // Don't check Parse on VALID, any partial only VALID may result in
     493             :         // #REF! during compile later!
     494          60 :         if (aRange.Parse( rName, pDoc, details) || aAddr.Parse( rName, pDoc, details))
     495           0 :             return false;
     496             :     }
     497          10 :     return true;
     498             : }
     499             : 
     500         752 : SCROW ScRangeData::GetMaxRow() const
     501             : {
     502         752 :     return mnMaxRow >= 0 ? mnMaxRow : MAXROW;
     503             : }
     504             : 
     505         752 : SCCOL ScRangeData::GetMaxCol() const
     506             : {
     507         752 :     return mnMaxCol >= 0 ? mnMaxCol : MAXCOL;
     508             : }
     509             : 
     510         722 : sal_uInt16 ScRangeData::GetErrCode() const
     511             : {
     512         722 :     return pCode ? pCode->GetCodeError() : 0;
     513             : }
     514             : 
     515         806 : bool ScRangeData::HasReferences() const
     516             : {
     517         806 :     return pCode->HasReferences();
     518             : }
     519             : 
     520           2 : sal_uInt32 ScRangeData::GetUnoType() const
     521             : {
     522           2 :     sal_uInt32 nUnoType = 0;
     523           2 :     if ( HasType(RT_CRITERIA) )  nUnoType |= com::sun::star::sheet::NamedRangeFlag::FILTER_CRITERIA;
     524           2 :     if ( HasType(RT_PRINTAREA) ) nUnoType |= com::sun::star::sheet::NamedRangeFlag::PRINT_AREA;
     525           2 :     if ( HasType(RT_COLHEADER) ) nUnoType |= com::sun::star::sheet::NamedRangeFlag::COLUMN_HEADER;
     526           2 :     if ( HasType(RT_ROWHEADER) ) nUnoType |= com::sun::star::sheet::NamedRangeFlag::ROW_HEADER;
     527           2 :     return nUnoType;
     528             : }
     529             : 
     530         326 : void ScRangeData::ValidateTabRefs()
     531             : {
     532             :     //  try to make sure all relative references and the reference position
     533             :     //  are within existing tables, so they can be represented as text
     534             :     //  (if the range of used tables is more than the existing tables,
     535             :     //  the result may still contain invalid tables, because the relative
     536             :     //  references aren't changed so formulas stay the same)
     537             : 
     538             :     //  find range of used tables
     539             : 
     540         326 :     SCTAB nMinTab = aPos.Tab();
     541         326 :     SCTAB nMaxTab = nMinTab;
     542             :     formula::FormulaToken* t;
     543         326 :     pCode->Reset();
     544        1008 :     while ( ( t = pCode->GetNextReference() ) != NULL )
     545             :     {
     546         356 :         ScSingleRefData& rRef1 = *t->GetSingleRef();
     547         356 :         ScAddress aAbs = rRef1.toAbs(aPos);
     548         356 :         if ( rRef1.IsTabRel() && !rRef1.IsTabDeleted() )
     549             :         {
     550          24 :             if (aAbs.Tab() < nMinTab)
     551           0 :                 nMinTab = aAbs.Tab();
     552          24 :             if (aAbs.Tab() > nMaxTab)
     553           0 :                 nMaxTab = aAbs.Tab();
     554             :         }
     555         356 :         if ( t->GetType() == svDoubleRef )
     556             :         {
     557          86 :             ScSingleRefData& rRef2 = t->GetDoubleRef()->Ref2;
     558          86 :             aAbs = rRef2.toAbs(aPos);
     559          86 :             if ( rRef2.IsTabRel() && !rRef2.IsTabDeleted() )
     560             :             {
     561           6 :                 if (aAbs.Tab() < nMinTab)
     562           0 :                     nMinTab = aAbs.Tab();
     563           6 :                 if (aAbs.Tab() > nMaxTab)
     564           0 :                     nMaxTab = aAbs.Tab();
     565             :             }
     566             :         }
     567             :     }
     568             : 
     569         326 :     SCTAB nTabCount = pDoc->GetTableCount();
     570         326 :     if ( nMaxTab >= nTabCount && nMinTab > 0 )
     571             :     {
     572             :         //  move position and relative tab refs
     573             :         //  The formulas that use the name are not changed by this
     574             : 
     575           0 :         SCTAB nMove = nMinTab;
     576           0 :         ScAddress aOldPos = aPos;
     577           0 :         aPos.SetTab( aPos.Tab() - nMove );
     578             : 
     579           0 :         pCode->Reset();
     580           0 :         while ( ( t = pCode->GetNextReference() ) != NULL )
     581             :         {
     582           0 :             switch (t->GetType())
     583             :             {
     584             :                 case svSingleRef:
     585             :                 {
     586           0 :                     ScSingleRefData& rRef = *t->GetSingleRef();
     587           0 :                     if (!rRef.IsTabDeleted())
     588             :                     {
     589           0 :                         ScAddress aAbs = rRef.toAbs(aOldPos);
     590           0 :                         rRef.SetAddress(aAbs, aPos);
     591             :                     }
     592             :                 }
     593           0 :                 break;
     594             :                 case svDoubleRef:
     595             :                 {
     596           0 :                     ScComplexRefData& rRef = *t->GetDoubleRef();
     597           0 :                     if (!rRef.Ref1.IsTabDeleted())
     598             :                     {
     599           0 :                         ScAddress aAbs = rRef.Ref1.toAbs(aOldPos);
     600           0 :                         rRef.Ref1.SetAddress(aAbs, aPos);
     601             :                     }
     602           0 :                     if (!rRef.Ref2.IsTabDeleted())
     603             :                     {
     604           0 :                         ScAddress aAbs = rRef.Ref2.toAbs(aOldPos);
     605           0 :                         rRef.Ref2.SetAddress(aAbs, aPos);
     606             :                     }
     607             :                 }
     608           0 :                 break;
     609             :                 default:
     610             :                     ;
     611             :             }
     612             :         }
     613             :     }
     614         326 : }
     615             : 
     616          24 : void ScRangeData::SetCode( ScTokenArray& rArr )
     617             : {
     618          24 :     boost::scoped_ptr<ScTokenArray> pOldCode( pCode); // old pCode will be deleted
     619          24 :     pCode = new ScTokenArray( rArr );
     620          24 :     InitCode();
     621          24 : }
     622             : 
     623         368 : void ScRangeData::InitCode()
     624             : {
     625         368 :     if( !pCode->GetCodeError() )
     626             :     {
     627         368 :         pCode->Reset();
     628         368 :         FormulaToken* p = pCode->GetNextReference();
     629         368 :         if( p )   // exact one reference at first
     630             :         {
     631         216 :             if( p->GetType() == svSingleRef )
     632         110 :                 eType = eType | RT_ABSPOS;
     633             :             else
     634         106 :                 eType = eType | RT_ABSAREA;
     635             :         }
     636             :     }
     637         368 : }
     638             : 
     639             : extern "C"
     640          60 : int SAL_CALL ScRangeData_QsortNameCompare( const void* p1, const void* p2 )
     641             : {
     642             :     return (int) ScGlobal::GetCollator()->compareString(
     643          60 :             (*(const ScRangeData**)p1)->GetName(),
     644         120 :             (*(const ScRangeData**)p2)->GetName() );
     645             : }
     646             : 
     647           0 : bool operator<(const ScRangeData& left, const ScRangeData& right)
     648             : {
     649           0 :     return left.GetName() < right.GetName();
     650             : }
     651             : 
     652             : namespace {
     653             : 
     654             : /**
     655             :  * Predicate to check if the name references the specified range.
     656             :  */
     657             : class MatchByRange : public unary_function<ScRangeData, bool>
     658             : {
     659             :     const ScRange& mrRange;
     660             : public:
     661         668 :     MatchByRange(const ScRange& rRange) : mrRange(rRange) {}
     662             :     template < typename Pair >
     663         722 :     bool operator() ( Pair const& r) const
     664             :     {
     665         722 :         return r.second->IsRangeAtBlock(mrRange);
     666             :     }
     667             : };
     668             : 
     669             : }
     670             : 
     671        1732 : ScRangeName::ScRangeName() {}
     672             : 
     673         366 : ScRangeName::ScRangeName(const ScRangeName& r) :
     674         366 :     maData(r.maData)
     675             : {
     676             :     // boost::ptr_set clones and deletes, so each collection needs its own
     677             :     // index to data.
     678         366 :     maIndexToData.resize( r.maIndexToData.size(), NULL);
     679         366 :     DataType::const_iterator itr = maData.begin(), itrEnd = maData.end();
     680        1202 :     for (; itr != itrEnd; ++itr)
     681             :     {
     682         836 :         size_t nPos = itr->second->GetIndex() - 1;
     683         836 :         if (nPos >= maIndexToData.size())
     684             :         {
     685             :             OSL_FAIL( "ScRangeName copy-ctor: maIndexToData size doesn't fit");
     686           0 :             maIndexToData.resize(nPos+1, NULL);
     687             :         }
     688         836 :         maIndexToData[nPos] = const_cast<ScRangeData*>(itr->second);
     689             :     }
     690         366 : }
     691             : 
     692         668 : const ScRangeData* ScRangeName::findByRange(const ScRange& rRange) const
     693             : {
     694             :     DataType::const_iterator itr = std::find_if(
     695         668 :         maData.begin(), maData.end(), MatchByRange(rRange));
     696         668 :     return itr == maData.end() ? NULL : itr->second;
     697             : }
     698             : 
     699        6996 : ScRangeData* ScRangeName::findByUpperName(const OUString& rName)
     700             : {
     701        6996 :     DataType::iterator itr = maData.find(rName);
     702        6996 :     return itr == maData.end() ? NULL : itr->second;
     703             : }
     704             : 
     705          22 : const ScRangeData* ScRangeName::findByUpperName(const OUString& rName) const
     706             : {
     707          22 :     DataType::const_iterator itr = maData.find(rName);
     708          22 :     return itr == maData.end() ? NULL : itr->second;
     709             : }
     710             : 
     711         976 : ScRangeData* ScRangeName::findByIndex(sal_uInt16 i) const
     712             : {
     713         976 :     if (!i)
     714             :         // index should never be zero.
     715           0 :         return NULL;
     716             : 
     717         976 :     size_t nPos = i - 1;
     718         976 :     return nPos < maIndexToData.size() ? maIndexToData[nPos] : NULL;
     719             : }
     720             : 
     721         292 : void ScRangeName::UpdateReference(sc::RefUpdateContext& rCxt, SCTAB nLocalTab )
     722             : {
     723         292 :     DataType::iterator itr = maData.begin(), itrEnd = maData.end();
     724         354 :     for (; itr != itrEnd; ++itr)
     725          62 :         itr->second->UpdateReference(rCxt, nLocalTab);
     726         292 : }
     727             : 
     728         270 : void ScRangeName::UpdateInsertTab( sc::RefUpdateInsertTabContext& rCxt, SCTAB nLocalTab )
     729             : {
     730         270 :     DataType::iterator itr = maData.begin(), itrEnd = maData.end();
     731         390 :     for (; itr != itrEnd; ++itr)
     732         120 :         itr->second->UpdateInsertTab(rCxt, nLocalTab);
     733         270 : }
     734             : 
     735         118 : void ScRangeName::UpdateDeleteTab( sc::RefUpdateDeleteTabContext& rCxt, SCTAB nLocalTab )
     736             : {
     737         118 :     DataType::iterator itr = maData.begin(), itrEnd = maData.end();
     738         164 :     for (; itr != itrEnd; ++itr)
     739          46 :         itr->second->UpdateDeleteTab(rCxt, nLocalTab);
     740         118 : }
     741             : 
     742           2 : void ScRangeName::UpdateMoveTab( sc::RefUpdateMoveTabContext& rCxt, SCTAB nLocalTab )
     743             : {
     744           2 :     DataType::iterator itr = maData.begin(), itrEnd = maData.end();
     745           4 :     for (; itr != itrEnd; ++itr)
     746           2 :         itr->second->UpdateMoveTab(rCxt, nLocalTab);
     747           2 : }
     748             : 
     749           0 : void ScRangeName::UpdateTranspose(const ScRange& rSource, const ScAddress& rDest)
     750             : {
     751           0 :     DataType::iterator itr = maData.begin(), itrEnd = maData.end();
     752           0 :     for (; itr != itrEnd; ++itr)
     753           0 :         itr->second->UpdateTranspose(rSource, rDest);
     754           0 : }
     755             : 
     756           0 : void ScRangeName::UpdateGrow(const ScRange& rArea, SCCOL nGrowX, SCROW nGrowY)
     757             : {
     758           0 :     DataType::iterator itr = maData.begin(), itrEnd = maData.end();
     759           0 :     for (; itr != itrEnd; ++itr)
     760           0 :         itr->second->UpdateGrow(rArea, nGrowX, nGrowY);
     761           0 : }
     762             : 
     763         142 : void ScRangeName::CompileUnresolvedXML( sc::CompileFormulaContext& rCxt )
     764             : {
     765         142 :     DataType::iterator itr = maData.begin(), itrEnd = maData.end();
     766         266 :     for (; itr != itrEnd; ++itr)
     767         124 :         itr->second->CompileUnresolvedXML(rCxt);
     768         142 : }
     769             : 
     770         134 : ScRangeName::const_iterator ScRangeName::begin() const
     771             : {
     772         134 :     return maData.begin();
     773             : }
     774             : 
     775         134 : ScRangeName::const_iterator ScRangeName::end() const
     776             : {
     777         134 :     return maData.end();
     778             : }
     779             : 
     780         292 : ScRangeName::iterator ScRangeName::begin()
     781             : {
     782         292 :     return maData.begin();
     783             : }
     784             : 
     785         326 : ScRangeName::iterator ScRangeName::end()
     786             : {
     787         326 :     return maData.end();
     788             : }
     789             : 
     790          10 : size_t ScRangeName::size() const
     791             : {
     792          10 :     return maData.size();
     793             : }
     794             : 
     795        3186 : bool ScRangeName::empty() const
     796             : {
     797        3186 :     return maData.empty();
     798             : }
     799             : 
     800         592 : bool ScRangeName::insert(ScRangeData* p)
     801             : {
     802         592 :     if (!p)
     803           0 :         return false;
     804             : 
     805         592 :     if (!p->GetIndex())
     806             :     {
     807             :         // Assign a new index.  An index must be unique and is never 0.
     808             :         IndexDataType::iterator itr = std::find(
     809         202 :             maIndexToData.begin(), maIndexToData.end(), static_cast<ScRangeData*>(NULL));
     810         202 :         if (itr != maIndexToData.end())
     811             :         {
     812             :             // Empty slot exists.  Re-use it.
     813          10 :             size_t nPos = std::distance(maIndexToData.begin(), itr);
     814          10 :             p->SetIndex(nPos + 1);
     815             :         }
     816             :         else
     817             :             // No empty slot.  Append it to the end.
     818         192 :             p->SetIndex(maIndexToData.size() + 1);
     819             :     }
     820             : 
     821         592 :     OUString aName(p->GetUpperName());
     822         592 :     erase(aName); // ptr_map won't insert it if a duplicate name exists.
     823         592 :     pair<DataType::iterator, bool> r = maData.insert(aName, p);
     824         592 :     if (r.second)
     825             :     {
     826             :         // Data inserted.  Store its index for mapping.
     827         592 :         size_t nPos = p->GetIndex() - 1;
     828         592 :         if (nPos >= maIndexToData.size())
     829         556 :             maIndexToData.resize(nPos+1, NULL);
     830         592 :         maIndexToData[nPos] = p;
     831             :     }
     832         592 :     return r.second;
     833             : }
     834             : 
     835          32 : void ScRangeName::erase(const ScRangeData& r)
     836             : {
     837          32 :     erase(r.GetUpperName());
     838          32 : }
     839             : 
     840         624 : void ScRangeName::erase(const OUString& rName)
     841             : {
     842         624 :     DataType::iterator itr = maData.find(rName);
     843         624 :     if (itr != maData.end())
     844          32 :         erase(itr);
     845         624 : }
     846             : 
     847          32 : void ScRangeName::erase(const iterator& itr)
     848             : {
     849          32 :     sal_uInt16 nIndex = itr->second->GetIndex();
     850          32 :     maData.erase(itr);
     851             :     OSL_ENSURE( 0 < nIndex && nIndex <= maIndexToData.size(), "ScRangeName::erase: bad index");
     852          32 :     if (0 < nIndex && nIndex <= maIndexToData.size())
     853          32 :         maIndexToData[nIndex-1] = NULL;
     854          32 : }
     855             : 
     856          22 : void ScRangeName::clear()
     857             : {
     858          22 :     maData.clear();
     859          22 :     maIndexToData.clear();
     860          22 : }
     861             : 
     862         122 : bool ScRangeName::operator== (const ScRangeName& r) const
     863             : {
     864         122 :     return maData == r.maData;
     865         228 : }
     866             : 
     867             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.10