LCOV - code coverage report
Current view: top level - sc/source/core/tool - chartpos.cxx (source / functions) Hit Total Coverage
Test: commit e02a6cb2c3e2b23b203b422e4e0680877f232636 Lines: 0 295 0.0 %
Date: 2014-04-14 Functions: 0 14 0.0 %
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 "chartpos.hxx"
      21             : #include "document.hxx"
      22             : #include "rechead.hxx"
      23             : #include <boost/scoped_array.hpp>
      24             : 
      25             : namespace
      26             : {
      27           0 :     bool lcl_hasValueDataButNoDates( ScDocument* pDocument, SCCOL nCol, SCROW nRow, SCTAB nTab )
      28             :     {
      29           0 :         bool bReturn = false;
      30           0 :         if (pDocument->HasValueData( nCol, nRow, nTab ))
      31             :         {
      32             :             //treat dates like text #i25706#
      33           0 :             sal_uInt32 nNumberFormat = pDocument->GetNumberFormat( ScAddress( nCol, nRow, nTab ) );
      34           0 :             short nType = pDocument->GetFormatTable()->GetType(nNumberFormat);
      35           0 :             bool bIsDate = (nType & NUMBERFORMAT_DATE);
      36           0 :             bReturn = !bIsDate;
      37             :         }
      38           0 :         return bReturn;
      39             :     }
      40             : }
      41             : 
      42           0 : ScChartPositioner::ScChartPositioner( ScDocument* pDoc, SCTAB nTab,
      43             :                     SCCOL nStartColP, SCROW nStartRowP, SCCOL nEndColP, SCROW nEndRowP) :
      44             :         pDocument( pDoc ),
      45             :         pPositionMap( NULL ),
      46             :         eGlue( SC_CHARTGLUE_NA ),
      47             :         nStartCol(0),
      48             :         nStartRow(0),
      49             :         bColHeaders( false ),
      50             :         bRowHeaders( false ),
      51           0 :         bDummyUpperLeft( false )
      52             : {
      53           0 :     SetRangeList( ScRange( nStartColP, nStartRowP, nTab, nEndColP, nEndRowP, nTab ) );
      54           0 :     CheckColRowHeaders();
      55           0 : }
      56             : 
      57           0 : ScChartPositioner::ScChartPositioner( ScDocument* pDoc, const ScRangeListRef& rRangeList ) :
      58             :         aRangeListRef( rRangeList ),
      59             :         pDocument( pDoc ),
      60             :         pPositionMap( NULL ),
      61             :         eGlue( SC_CHARTGLUE_NA ),
      62             :         nStartCol(0),
      63             :         nStartRow(0),
      64             :         bColHeaders( false ),
      65             :         bRowHeaders( false ),
      66           0 :         bDummyUpperLeft( false )
      67             : {
      68           0 :     if ( aRangeListRef.Is() )
      69           0 :         CheckColRowHeaders();
      70           0 : }
      71             : 
      72           0 : ScChartPositioner::ScChartPositioner( const ScChartPositioner& rPositioner ) :
      73             :         aRangeListRef( rPositioner.aRangeListRef ),
      74             :         pDocument(rPositioner.pDocument),
      75             :         pPositionMap( NULL ),
      76             :         eGlue(rPositioner.eGlue),
      77             :         nStartCol(rPositioner.nStartCol),
      78             :         nStartRow(rPositioner.nStartRow),
      79             :         bColHeaders(rPositioner.bColHeaders),
      80             :         bRowHeaders(rPositioner.bRowHeaders),
      81           0 :         bDummyUpperLeft( rPositioner.bDummyUpperLeft )
      82             : {
      83           0 : }
      84             : 
      85           0 : ScChartPositioner::~ScChartPositioner()
      86             : {
      87           0 :     delete pPositionMap;
      88           0 : }
      89             : 
      90           0 : bool ScChartPositioner::operator==(const ScChartPositioner& rCmp) const
      91             : {
      92           0 :     return bColHeaders == rCmp.bColHeaders
      93           0 :         && bRowHeaders == rCmp.bRowHeaders
      94           0 :         && *aRangeListRef == *rCmp.aRangeListRef;
      95             : }
      96             : 
      97           0 : void ScChartPositioner::SetRangeList( const ScRange& rRange )
      98             : {
      99           0 :     aRangeListRef = new ScRangeList;
     100           0 :     aRangeListRef->Append( rRange );
     101           0 :     InvalidateGlue();
     102           0 : }
     103             : 
     104           0 : void ScChartPositioner::GlueState()
     105             : {
     106           0 :     if ( eGlue != SC_CHARTGLUE_NA )
     107           0 :         return;
     108           0 :     bDummyUpperLeft = false;
     109             :     ScRange* pR;
     110           0 :     if ( aRangeListRef->size() <= 1 )
     111             :     {
     112           0 :         if (  !aRangeListRef->empty() )
     113             :         {
     114           0 :             pR = aRangeListRef->front();
     115           0 :             if ( pR->aStart.Tab() == pR->aEnd.Tab() )
     116           0 :                 eGlue = SC_CHARTGLUE_NONE;
     117             :             else
     118           0 :                 eGlue = SC_CHARTGLUE_COLS;  // several tables column by column
     119           0 :             nStartCol = pR->aStart.Col();
     120           0 :             nStartRow = pR->aStart.Row();
     121             :         }
     122             :         else
     123             :         {
     124           0 :             InvalidateGlue();
     125           0 :             nStartCol = 0;
     126           0 :             nStartRow = 0;
     127             :         }
     128           0 :         return;
     129             :     }
     130             : 
     131           0 :     pR = aRangeListRef->front();
     132           0 :     nStartCol = pR->aStart.Col();
     133           0 :     nStartRow = pR->aStart.Row();
     134             :     SCCOL nMaxCols, nEndCol;
     135             :     SCROW nMaxRows, nEndRow;
     136           0 :     nMaxCols = nEndCol = 0;
     137           0 :     nMaxRows = nEndRow = 0;
     138             : 
     139             :     // <= so 1 extra pass after last item
     140           0 :     for ( size_t i = 1, nRanges = aRangeListRef->size(); i <= nRanges; ++i )
     141             :     {   // detect spanning/surrounding area etc.
     142             :         SCCOLROW nTmp, n1, n2;
     143           0 :         if ( (n1 = pR->aStart.Col()) < nStartCol ) nStartCol = static_cast<SCCOL>(n1  );
     144           0 :         if ( (n2 = pR->aEnd.Col()  ) > nEndCol   ) nEndCol   = static_cast<SCCOL>(n2  );
     145           0 :         if ( (nTmp = n2 - n1 + 1   ) > nMaxCols  ) nMaxCols  = static_cast<SCCOL>(nTmp);
     146           0 :         if ( (n1 = pR->aStart.Row()) < nStartRow ) nStartRow = static_cast<SCROW>(n1  );
     147           0 :         if ( (n2 = pR->aEnd.Row()  ) > nEndRow   ) nEndRow   = static_cast<SCROW>(n2  );
     148           0 :         if ( (nTmp = n2 - n1 + 1   ) > nMaxRows  ) nMaxRows  = static_cast<SCROW>(nTmp);
     149             : 
     150             :         // in last pass; i = nRanges so don't use at()
     151           0 :         if ( i < nRanges )
     152           0 :             pR = (*aRangeListRef)[i];
     153             :     }
     154           0 :     SCCOL nC = nEndCol - nStartCol + 1;
     155           0 :     if ( nC == 1 )
     156             :     {
     157           0 :         eGlue = SC_CHARTGLUE_ROWS;
     158           0 :         return;
     159             :     }
     160           0 :     SCROW nR = nEndRow - nStartRow + 1;
     161           0 :     if ( nR == 1 )
     162             :     {
     163           0 :         eGlue = SC_CHARTGLUE_COLS;
     164           0 :         return;
     165             :     }
     166           0 :     sal_uLong nCR = (sal_uLong)nC * nR;
     167             : 
     168             :     /*
     169             :     TODO:
     170             :     First do it simple without bit masking. A maximum of 8MB could be allocated
     171             :     this way (256 Cols x 32000 Rows). That could be reduced to 2MB by
     172             :     using 2 Bits per entry, but it is faster this way.
     173             :     Another optimization would be to store only used rows/columns in the array, but
     174             :     would mean another iteration of the RangeList indirect access to the array. */
     175             : 
     176           0 :     const sal_uInt8 nHole = 0;
     177           0 :     const sal_uInt8 nOccu = 1;
     178           0 :     const sal_uInt8 nFree = 2;
     179           0 :     const sal_uInt8 nGlue = 3;
     180             :     sal_uInt8* p;
     181           0 :     boost::scoped_array<sal_uInt8> pA(new sal_uInt8[ nCR ]);
     182           0 :     memset( pA.get(), 0, nCR * sizeof(sal_uInt8) );
     183             : 
     184             :     SCCOL nCol, nCol1, nCol2;
     185             :     SCROW nRow, nRow1, nRow2;
     186           0 :     for ( size_t i = 0, nRanges = aRangeListRef->size(); i < nRanges; ++i )
     187             :     {   // mark selections as used in 2D
     188           0 :         pR = (*aRangeListRef)[i];
     189           0 :         nCol1 = pR->aStart.Col() - nStartCol;
     190           0 :         nCol2 = pR->aEnd.Col() - nStartCol;
     191           0 :         nRow1 = pR->aStart.Row() - nStartRow;
     192           0 :         nRow2 = pR->aEnd.Row() - nStartRow;
     193           0 :         for ( nCol = nCol1; nCol <= nCol2; nCol++ )
     194             :         {
     195           0 :             p = pA.get() + (sal_uLong)nCol * nR + nRow1;
     196           0 :             for ( nRow = nRow1; nRow <= nRow2; nRow++, p++ )
     197           0 :                 *p = nOccu;
     198             :         }
     199             :     }
     200           0 :     sal_Bool bGlue = sal_True;
     201             : 
     202           0 :     sal_Bool bGlueCols = false;
     203           0 :     for ( nCol = 0; bGlue && nCol < nC; nCol++ )
     204             :     {   // iterate columns and try to mark as unused
     205           0 :         p = pA.get() + (sal_uLong)nCol * nR;
     206           0 :         for ( nRow = 0; bGlue && nRow < nR; nRow++, p++ )
     207             :         {
     208           0 :             if ( *p == nOccu )
     209             :             {   // If there's one right in the middle, we can't combine.
     210             :                 // If it were at the edge, we could combine, if in this Column
     211             :                 // in every set line, one is set.
     212           0 :                 if ( nRow > 0 && nCol > 0 )
     213           0 :                     bGlue = false; // nCol==0 can be DummyUpperLeft
     214             :                 else
     215           0 :                     nRow = nR;
     216             :             }
     217             :             else
     218           0 :                 *p = nFree;
     219             :         }
     220           0 :         if ( bGlue && *(p = (pA.get() + ((((sal_uLong)nCol+1) * nR) - 1))) == nFree )
     221             :         {   // mark column as totally unused
     222           0 :             *p = nGlue;
     223           0 :             bGlueCols = sal_True; // one unused column at least
     224             :         }
     225             :     }
     226             : 
     227           0 :     sal_Bool bGlueRows = false;
     228           0 :     for ( nRow = 0; bGlue && nRow < nR; nRow++ )
     229             :     {   // iterate rows and try to mark as unused
     230           0 :         p = pA.get() + nRow;
     231           0 :         for ( nCol = 0; bGlue && nCol < nC; nCol++, p+=nR )
     232             :         {
     233           0 :             if ( *p == nOccu )
     234             :             {
     235           0 :                 if ( nCol > 0 && nRow > 0 )
     236           0 :                     bGlue = false; // nRow==0 can be DummyUpperLeft
     237             :                 else
     238           0 :                     nCol = nC;
     239             :             }
     240             :             else
     241           0 :                 *p = nFree;
     242             :         }
     243           0 :         if ( bGlue && *(p = (pA.get() + ((((sal_uLong)nC-1) * nR) + nRow))) == nFree )
     244             :         {   // mark row as totally unused
     245           0 :             *p = nGlue;
     246           0 :             bGlueRows = sal_True; // one unused row at least
     247             :         }
     248             :     }
     249             : 
     250             :     // If n=1: The upper left corner could be automagically pulled in for labeling
     251           0 :     p = pA.get() + 1;
     252           0 :     for ( sal_uLong n = 1; bGlue && n < nCR; n++, p++ )
     253             :     {   // An untouched field means we could neither reach it through rows nor columns,
     254             :         // thus we can't combine anything
     255           0 :         if ( *p == nHole )
     256           0 :             bGlue = false;
     257             :     }
     258           0 :     if ( bGlue )
     259             :     {
     260           0 :         if ( bGlueCols && bGlueRows )
     261           0 :             eGlue = SC_CHARTGLUE_BOTH;
     262           0 :         else if ( bGlueRows )
     263           0 :             eGlue = SC_CHARTGLUE_ROWS;
     264             :         else
     265           0 :             eGlue = SC_CHARTGLUE_COLS;
     266           0 :         if ( pA[0] != nOccu )
     267           0 :             bDummyUpperLeft = true;
     268             :     }
     269             :     else
     270             :     {
     271           0 :         eGlue = SC_CHARTGLUE_NONE;
     272           0 :     }
     273             : }
     274             : 
     275           0 : void ScChartPositioner::CheckColRowHeaders()
     276             : {
     277             :     SCCOL nCol1, nCol2, iCol;
     278             :     SCROW nRow1, nRow2, iRow;
     279             :     SCTAB nTab1, nTab2;
     280             : 
     281           0 :     sal_Bool bColStrings = sal_True;
     282           0 :     sal_Bool bRowStrings = sal_True;
     283           0 :     GlueState();
     284           0 :     if ( aRangeListRef->size() == 1 )
     285             :     {
     286           0 :         aRangeListRef->front()->GetVars( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 );
     287           0 :         if ( nCol1 > nCol2 || nRow1 > nRow2 )
     288           0 :             bColStrings = bRowStrings = false;
     289             :         else
     290             :         {
     291           0 :             for (iCol=nCol1; iCol<=nCol2 && bColStrings; iCol++)
     292             :             {
     293           0 :                 if (lcl_hasValueDataButNoDates( pDocument, iCol, nRow1, nTab1 ))
     294           0 :                         bColStrings = false;
     295             :             }
     296           0 :             for (iRow=nRow1; iRow<=nRow2 && bRowStrings; iRow++)
     297             :             {
     298           0 :                 if (lcl_hasValueDataButNoDates( pDocument, nCol1, iRow, nTab1 ))
     299           0 :                         bRowStrings = false;
     300             :             }
     301             :         }
     302             :     }
     303             :     else
     304             :     {
     305           0 :         sal_Bool bVert = (eGlue == SC_CHARTGLUE_NONE || eGlue == SC_CHARTGLUE_ROWS);
     306           0 :         for ( size_t i = 0, nRanges = aRangeListRef->size();
     307           0 :               (i < nRanges) && (bColStrings || bRowStrings);
     308             :               ++i
     309             :             )
     310             :         {
     311           0 :             ScRange* pR = (*aRangeListRef)[i];
     312           0 :             pR->GetVars( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 );
     313           0 :             sal_Bool bTopRow = (nRow1 == nStartRow);
     314           0 :             if ( bRowStrings && (bVert || nCol1 == nStartCol) )
     315             :             {   // NONE or ROWS: RowStrings in every selection possible
     316             :                 // COLS or BOTH: only from first column
     317           0 :                 if ( nCol1 <= nCol2 )
     318           0 :                     for (iRow=nRow1; iRow<=nRow2 && bRowStrings; iRow++)
     319             :                     {
     320           0 :                         if (lcl_hasValueDataButNoDates( pDocument, nCol1, iRow, nTab1 ))
     321           0 :                                 bRowStrings = false;
     322             :                     }
     323             :             }
     324           0 :             if ( bColStrings && bTopRow )
     325             :             {   // ColStrings only from first row
     326           0 :                 if ( nRow1 <= nRow2 )
     327           0 :                     for (iCol=nCol1; iCol<=nCol2 && bColStrings; iCol++)
     328             :                     {
     329           0 :                         if (lcl_hasValueDataButNoDates( pDocument, iCol, nRow1, nTab1 ))
     330           0 :                                 bColStrings = false;
     331             :                     }
     332             :             }
     333             :         }
     334             :     }
     335           0 :     bColHeaders = bColStrings;
     336           0 :     bRowHeaders = bRowStrings;
     337           0 : }
     338             : 
     339           0 : const ScChartPositionMap* ScChartPositioner::GetPositionMap()
     340             : {
     341           0 :     CreatePositionMap();
     342           0 :     return pPositionMap;
     343             : }
     344             : 
     345             : 
     346           0 : void ScChartPositioner::CreatePositionMap()
     347             : {
     348           0 :     if ( eGlue == SC_CHARTGLUE_NA && pPositionMap )
     349             :     {
     350           0 :         delete pPositionMap;
     351           0 :         pPositionMap = NULL;
     352             :     }
     353             : 
     354           0 :     if ( pPositionMap )
     355           0 :         return ;
     356             : 
     357           0 :     SCSIZE nColAdd = bRowHeaders ? 1 : 0;
     358           0 :     SCSIZE nRowAdd = bColHeaders ? 1 : 0;
     359             : 
     360             :     SCCOL nCol, nCol1, nCol2;
     361             :     SCROW nRow, nRow1, nRow2;
     362             :     SCTAB nTab, nTab1, nTab2;
     363             : 
     364             : 
     365             :     //  real size (without hidden rows/columns)
     366             : 
     367             : 
     368           0 :     SCSIZE nColCount = 0;
     369           0 :     SCSIZE nRowCount = 0;
     370             : 
     371           0 :     GlueState();
     372             : 
     373           0 :     const sal_Bool bNoGlue = (eGlue == SC_CHARTGLUE_NONE);
     374           0 :     ColumnMap* pCols = new ColumnMap;
     375           0 :     SCROW nNoGlueRow = 0;
     376           0 :     for ( size_t i = 0, nRanges = aRangeListRef->size(); i < nRanges; ++i )
     377             :     {
     378           0 :         ScRange* pR = (*aRangeListRef)[i];
     379           0 :         pR->GetVars( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 );
     380           0 :         for ( nTab = nTab1; nTab <= nTab2; nTab++ )
     381             :         {
     382             :             // nTab in ColKey to allow to have the same col/row in another tabe
     383           0 :             sal_uLong nInsCol = (static_cast<sal_uLong>(nTab) << 16) | (bNoGlue ? 0 :
     384           0 :                     static_cast<sal_uLong>(nCol1));
     385           0 :             for ( nCol = nCol1; nCol <= nCol2; ++nCol, ++nInsCol )
     386             :             {
     387           0 :                 RowMap* pCol = NULL;
     388           0 :                 ColumnMap::const_iterator it = pCols->find( nInsCol );
     389           0 :                 if ( it == pCols->end() )
     390             :                 {
     391           0 :                     pCol = new RowMap;
     392           0 :                     pCols->insert( ColumnMap::value_type( nInsCol, pCol ) );
     393             :                 }
     394             :                 else
     395           0 :                     pCol = it->second;
     396             : 
     397             :                 // in other table a new ColKey already was created,
     398             :                 // the rows must be equal to be filled with Dummy
     399           0 :                 sal_uLong nInsRow = (bNoGlue ? nNoGlueRow : nRow1);
     400           0 :                 for ( nRow = nRow1; nRow <= nRow2; nRow++, nInsRow++ )
     401             :                 {
     402           0 :                     if ( pCol->find( nInsRow ) == pCol->end() )
     403             :                     {
     404           0 :                         pCol->insert( RowMap::value_type( nInsRow, new ScAddress( nCol, nRow, nTab ) ) );
     405             :                     }
     406             :                 }
     407             :             }
     408             :         }
     409             :         // For NoGlue: associated tables will be rendered as ColGlue
     410           0 :         nNoGlueRow += nRow2 - nRow1 + 1;
     411             :     }
     412             : 
     413             :     // count of data
     414           0 :     nColCount = static_cast< SCSIZE >( pCols->size());
     415           0 :     if ( !pCols->empty() )
     416             :     {
     417           0 :         RowMap* pCol = pCols->begin()->second;
     418           0 :         if ( bDummyUpperLeft )
     419           0 :             (*pCol)[ 0 ] = NULL; // Dummy for labeling
     420           0 :         nRowCount = static_cast< SCSIZE >( pCol->size());
     421             :     }
     422             :     else
     423           0 :         nRowCount = 0;
     424           0 :     if ( nColCount > 0 )
     425           0 :         nColCount -= nColAdd;
     426           0 :     if ( nRowCount > 0 )
     427           0 :         nRowCount -= nRowAdd;
     428             : 
     429           0 :     if ( nColCount==0 || nRowCount==0 )
     430             :     {   // create an entry without data
     431             :         RowMap* pCol;
     432           0 :         if ( !pCols->empty() )
     433           0 :             pCol = pCols->begin()->second;
     434             :         else
     435             :         {
     436           0 :             pCol = new RowMap;
     437           0 :             (*pCols)[ 0 ] = pCol;
     438             :         }
     439           0 :         nColCount = 1;
     440           0 :         if ( !pCol->empty() )
     441             :         {   // cannot be if nColCount==0 || nRowCount==0
     442           0 :             ScAddress* pPos = pCol->begin()->second;
     443           0 :             if ( pPos )
     444             :             {
     445           0 :                 sal_uLong nCurrentKey = pCol->begin()->first;
     446           0 :                 delete pPos;
     447           0 :                 (*pCol)[ nCurrentKey ] = NULL;
     448             :             }
     449             :         }
     450             :         else
     451           0 :             (*pCol)[ 0 ] = NULL;
     452           0 :         nRowCount = 1;
     453           0 :         nColAdd = 0;
     454           0 :         nRowAdd = 0;
     455             :     }
     456             :     else
     457             :     {
     458           0 :         if ( bNoGlue )
     459             :         {   // fill gaps with Dummies, first column is master
     460           0 :             RowMap* pFirstCol = pCols->begin()->second;
     461           0 :             sal_uLong nCount = pFirstCol->size();
     462           0 :             RowMap::const_iterator it1 = pFirstCol->begin();
     463           0 :             for ( sal_uLong n = 0; n < nCount; n++, ++it1 )
     464             :             {
     465           0 :                 sal_uLong nKey = it1->first;
     466           0 :                 for (ColumnMap::const_iterator it2 = ++pCols->begin(); it2 != pCols->end(); ++it2 )
     467           0 :                     it2->second->insert( RowMap::value_type( nKey, (ScAddress *)NULL )); // no data
     468             :             }
     469             :         }
     470             :     }
     471             : 
     472             :     pPositionMap = new ScChartPositionMap( static_cast<SCCOL>(nColCount), static_cast<SCROW>(nRowCount),
     473           0 :         static_cast<SCCOL>(nColAdd), static_cast<SCROW>(nRowAdd), *pCols );
     474             : 
     475             :     //  cleanup
     476           0 :     for (ColumnMap::const_iterator it = pCols->begin(); it != pCols->end(); ++it )
     477             :     {   // Only delete tables, not the ScAddress*!
     478           0 :         delete it->second;
     479             :     }
     480           0 :     delete pCols;
     481             : }
     482             : 
     483             : 
     484           0 : ScChartPositionMap::ScChartPositionMap( SCCOL nChartCols, SCROW nChartRows,
     485             :             SCCOL nColAdd, SCROW nRowAdd, ColumnMap& rCols ) :
     486           0 :         ppData( new ScAddress* [ nChartCols * nChartRows ] ),
     487           0 :         ppColHeader( new ScAddress* [ nChartCols ] ),
     488           0 :         ppRowHeader( new ScAddress* [ nChartRows ] ),
     489           0 :         nCount( (sal_uLong) nChartCols * nChartRows ),
     490             :         nColCount( nChartCols ),
     491           0 :         nRowCount( nChartRows )
     492             : {
     493             :     OSL_ENSURE( nColCount && nRowCount, "ScChartPositionMap without dimension" );
     494             : 
     495           0 :     ColumnMap::const_iterator pColIter = rCols.begin();
     496           0 :     RowMap* pCol1 = pColIter->second;
     497           0 :     RowMap::const_iterator pPos1Iter;
     498             : 
     499             :     // row header
     500           0 :     pPos1Iter = pCol1->begin();
     501           0 :     if ( nRowAdd )
     502           0 :         ++pPos1Iter;
     503           0 :     if ( nColAdd )
     504             :     {   // independent
     505           0 :         SCROW nRow = 0;
     506           0 :         for ( ; nRow < nRowCount && pPos1Iter != pCol1->end(); nRow++ )
     507             :         {
     508           0 :             ppRowHeader[ nRow ] = pPos1Iter->second;
     509           0 :             ++pPos1Iter;
     510             :         }
     511           0 :         for ( ; nRow < nRowCount; nRow++ )
     512           0 :             ppRowHeader[ nRow ] = NULL;
     513             :     }
     514             :     else
     515             :     {   // copy
     516           0 :         SCROW nRow = 0;
     517           0 :         for ( ; nRow < nRowCount && pPos1Iter != pCol1->end(); nRow++ )
     518             :         {
     519           0 :             ppRowHeader[ nRow ] = pPos1Iter->second ?
     520           0 :                 new ScAddress( *pPos1Iter->second ) : NULL;
     521           0 :             ++pPos1Iter;
     522             :         }
     523           0 :         for ( ; nRow < nRowCount; nRow++ )
     524           0 :             ppRowHeader[ nRow ] = NULL;
     525             :     }
     526           0 :     if ( nColAdd )
     527             :     {
     528           0 :         ++pColIter;
     529             :     }
     530             : 
     531             :     // data column by column and column-header
     532           0 :     sal_uLong nIndex = 0;
     533           0 :     for ( SCCOL nCol = 0; nCol < nColCount; nCol++ )
     534             :     {
     535           0 :         if ( pColIter != rCols.end() )
     536             :         {
     537           0 :             RowMap* pCol2 = pColIter->second;
     538           0 :             RowMap::const_iterator pPosIter = pCol2->begin();
     539           0 :             if ( pPosIter != pCol2->end() )
     540             :             {
     541           0 :                 if ( nRowAdd )
     542             :                 {
     543           0 :                     ppColHeader[ nCol ] = pPosIter->second; // independent
     544           0 :                     ++pPosIter;
     545             :                 }
     546             :                 else
     547           0 :                     ppColHeader[ nCol ] = pPosIter->second ?
     548           0 :                         new ScAddress( *pPosIter->second ) : NULL;
     549             :             }
     550             : 
     551           0 :             SCROW nRow = 0;
     552           0 :             for ( ; nRow < nRowCount && pPosIter != pCol2->end(); nRow++, nIndex++ )
     553             :             {
     554           0 :                 ppData[ nIndex ] = pPosIter->second;
     555           0 :                 ++pPosIter;
     556             :             }
     557           0 :             for ( ; nRow < nRowCount; nRow++, nIndex++ )
     558           0 :                 ppData[ nIndex ] = NULL;
     559             : 
     560           0 :             ++pColIter;
     561             :         }
     562             :         else
     563             :         {
     564           0 :             ppColHeader[ nCol ] = NULL;
     565           0 :             for ( SCROW nRow = 0; nRow < nRowCount; nRow++, nIndex++ )
     566             :             {
     567           0 :                 ppData[ nIndex ] = NULL;
     568             :             }
     569             :         }
     570             :     }
     571           0 : }
     572             : 
     573             : 
     574           0 : ScChartPositionMap::~ScChartPositionMap()
     575             : {
     576           0 :     for ( sal_uLong nIndex=0; nIndex < nCount; nIndex++ )
     577             :     {
     578           0 :         delete ppData[nIndex];
     579             :     }
     580           0 :     delete [] ppData;
     581             : 
     582             :     SCCOL j;
     583           0 :     for ( j=0; j < nColCount; j++ )
     584             :     {
     585           0 :         delete ppColHeader[j];
     586             :     }
     587           0 :     delete [] ppColHeader;
     588             :     SCROW i;
     589           0 :     for ( i=0; i < nRowCount; i++ )
     590             :     {
     591           0 :         delete ppRowHeader[i];
     592             :     }
     593           0 :     delete [] ppRowHeader;
     594           0 : }
     595             : 
     596             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.10