LCOV - code coverage report
Current view: top level - sc/source/core/tool - chartpos.cxx (source / functions) Hit Total Coverage
Test: commit 10e77ab3ff6f4314137acd6e2702a6e5c1ce1fae Lines: 221 295 74.9 %
Date: 2014-11-03 Functions: 11 16 68.8 %
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         552 :     bool lcl_hasValueDataButNoDates( ScDocument* pDocument, SCCOL nCol, SCROW nRow, SCTAB nTab )
      28             :     {
      29         552 :         bool bReturn = false;
      30         552 :         if (pDocument->HasValueData( nCol, nRow, nTab ))
      31             :         {
      32             :             //treat dates like text #i25706#
      33          26 :             sal_uInt32 nNumberFormat = pDocument->GetNumberFormat( ScAddress( nCol, nRow, nTab ) );
      34          26 :             short nType = pDocument->GetFormatTable()->GetType(nNumberFormat);
      35          26 :             bool bIsDate = (nType & NUMBERFORMAT_DATE);
      36          26 :             bReturn = !bIsDate;
      37             :         }
      38         552 :         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          52 : 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          52 :         bDummyUpperLeft( false )
      67             : {
      68          52 :     if ( aRangeListRef.Is() )
      69          52 :         CheckColRowHeaders();
      70          52 : }
      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         104 : ScChartPositioner::~ScChartPositioner()
      86             : {
      87          52 :     delete pPositionMap;
      88          52 : }
      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          88 : void ScChartPositioner::GlueState()
     105             : {
     106          88 :     if ( eGlue != SC_CHARTGLUE_NA )
     107          94 :         return;
     108          52 :     bDummyUpperLeft = false;
     109             :     ScRange* pR;
     110          52 :     if ( aRangeListRef->size() <= 1 )
     111             :     {
     112          22 :         if (  !aRangeListRef->empty() )
     113             :         {
     114          22 :             pR = aRangeListRef->front();
     115          22 :             if ( pR->aStart.Tab() == pR->aEnd.Tab() )
     116          22 :                 eGlue = SC_CHARTGLUE_NONE;
     117             :             else
     118           0 :                 eGlue = SC_CHARTGLUE_COLS;  // several tables column by column
     119          22 :             nStartCol = pR->aStart.Col();
     120          22 :             nStartRow = pR->aStart.Row();
     121             :         }
     122             :         else
     123             :         {
     124           0 :             InvalidateGlue();
     125           0 :             nStartCol = 0;
     126           0 :             nStartRow = 0;
     127             :         }
     128          22 :         return;
     129             :     }
     130             : 
     131          30 :     pR = aRangeListRef->front();
     132          30 :     nStartCol = pR->aStart.Col();
     133          30 :     nStartRow = pR->aStart.Row();
     134             :     SCCOL nMaxCols, nEndCol;
     135             :     SCROW nMaxRows, nEndRow;
     136          30 :     nMaxCols = nEndCol = 0;
     137          30 :     nMaxRows = nEndRow = 0;
     138             : 
     139             :     // <= so 1 extra pass after last item
     140         150 :     for ( size_t i = 1, nRanges = aRangeListRef->size(); i <= nRanges; ++i )
     141             :     {   // detect spanning/surrounding area etc.
     142             :         SCCOLROW nTmp, n1, n2;
     143         120 :         if ( (n1 = pR->aStart.Col()) < nStartCol ) nStartCol = static_cast<SCCOL>(n1  );
     144         120 :         if ( (n2 = pR->aEnd.Col()  ) > nEndCol   ) nEndCol   = static_cast<SCCOL>(n2  );
     145         120 :         if ( (nTmp = n2 - n1 + 1   ) > nMaxCols  ) nMaxCols  = static_cast<SCCOL>(nTmp);
     146         120 :         if ( (n1 = pR->aStart.Row()) < nStartRow ) nStartRow = static_cast<SCROW>(n1  );
     147         120 :         if ( (n2 = pR->aEnd.Row()  ) > nEndRow   ) nEndRow   = static_cast<SCROW>(n2  );
     148         120 :         if ( (nTmp = n2 - n1 + 1   ) > nMaxRows  ) nMaxRows  = static_cast<SCROW>(nTmp);
     149             : 
     150             :         // in last pass; i = nRanges so don't use at()
     151         120 :         if ( i < nRanges )
     152          90 :             pR = (*aRangeListRef)[i];
     153             :     }
     154          30 :     SCCOL nC = nEndCol - nStartCol + 1;
     155          30 :     if ( nC == 1 )
     156             :     {
     157           0 :         eGlue = SC_CHARTGLUE_ROWS;
     158           0 :         return;
     159             :     }
     160          30 :     SCROW nR = nEndRow - nStartRow + 1;
     161          30 :     if ( nR == 1 )
     162             :     {
     163           0 :         eGlue = SC_CHARTGLUE_COLS;
     164           0 :         return;
     165             :     }
     166          30 :     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          30 :     const sal_uInt8 nHole = 0;
     177          30 :     const sal_uInt8 nOccu = 1;
     178          30 :     const sal_uInt8 nFree = 2;
     179          30 :     const sal_uInt8 nGlue = 3;
     180             :     sal_uInt8* p;
     181          30 :     boost::scoped_array<sal_uInt8> pA(new sal_uInt8[ nCR ]);
     182          30 :     memset( pA.get(), 0, nCR * sizeof(sal_uInt8) );
     183             : 
     184             :     SCCOL nCol, nCol1, nCol2;
     185             :     SCROW nRow, nRow1, nRow2;
     186         150 :     for ( size_t i = 0, nRanges = aRangeListRef->size(); i < nRanges; ++i )
     187             :     {   // mark selections as used in 2D
     188         120 :         pR = (*aRangeListRef)[i];
     189         120 :         nCol1 = pR->aStart.Col() - nStartCol;
     190         120 :         nCol2 = pR->aEnd.Col() - nStartCol;
     191         120 :         nRow1 = pR->aStart.Row() - nStartRow;
     192         120 :         nRow2 = pR->aEnd.Row() - nStartRow;
     193         360 :         for ( nCol = nCol1; nCol <= nCol2; nCol++ )
     194             :         {
     195         240 :             p = pA.get() + (sal_uLong)nCol * nR + nRow1;
     196         960 :             for ( nRow = nRow1; nRow <= nRow2; nRow++, p++ )
     197         720 :                 *p = nOccu;
     198             :         }
     199             :     }
     200          30 :     bool bGlue = true;
     201             : 
     202          30 :     bool bGlueCols = false;
     203         120 :     for ( nCol = 0; bGlue && nCol < nC; nCol++ )
     204             :     {   // iterate columns and try to mark as unused
     205          90 :         p = pA.get() + (sal_uLong)nCol * nR;
     206         210 :         for ( nRow = 0; bGlue && nRow < nR; nRow++, p++ )
     207             :         {
     208         120 :             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          90 :                 if ( nRow > 0 && nCol > 0 )
     213          30 :                     bGlue = false; // nCol==0 can be DummyUpperLeft
     214             :                 else
     215          60 :                     nRow = nR;
     216             :             }
     217             :             else
     218          30 :                 *p = nFree;
     219             :         }
     220          90 :         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 = true; // one unused column at least
     224             :         }
     225             :     }
     226             : 
     227          30 :     bool bGlueRows = false;
     228          30 :     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 = 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          30 :     p = pA.get() + 1;
     252          30 :     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          30 :     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          30 :         eGlue = SC_CHARTGLUE_NONE;
     272          30 :     }
     273             : }
     274             : 
     275          52 : void ScChartPositioner::CheckColRowHeaders()
     276             : {
     277             :     SCCOL nCol1, nCol2, iCol;
     278             :     SCROW nRow1, nRow2, iRow;
     279             :     SCTAB nTab1, nTab2;
     280             : 
     281          52 :     bool bColStrings = true;
     282          52 :     bool bRowStrings = true;
     283          52 :     GlueState();
     284          52 :     if ( aRangeListRef->size() == 1 )
     285             :     {
     286          22 :         aRangeListRef->front()->GetVars( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 );
     287          22 :         if ( nCol1 > nCol2 || nRow1 > nRow2 )
     288           0 :             bColStrings = bRowStrings = false;
     289             :         else
     290             :         {
     291         110 :             for (iCol=nCol1; iCol<=nCol2 && bColStrings; iCol++)
     292             :             {
     293          88 :                 if (lcl_hasValueDataButNoDates( pDocument, iCol, nRow1, nTab1 ))
     294           0 :                         bColStrings = false;
     295             :             }
     296         132 :             for (iRow=nRow1; iRow<=nRow2 && bRowStrings; iRow++)
     297             :             {
     298         110 :                 if (lcl_hasValueDataButNoDates( pDocument, nCol1, iRow, nTab1 ))
     299           0 :                         bRowStrings = false;
     300             :             }
     301             :         }
     302             :     }
     303             :     else
     304             :     {
     305          30 :         bool bVert = (eGlue == SC_CHARTGLUE_NONE || eGlue == SC_CHARTGLUE_ROWS);
     306         138 :         for ( size_t i = 0, nRanges = aRangeListRef->size();
     307         112 :               (i < nRanges) && (bColStrings || bRowStrings);
     308             :               ++i
     309             :             )
     310             :         {
     311         108 :             ScRange* pR = (*aRangeListRef)[i];
     312         108 :             pR->GetVars( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 );
     313         108 :             bool bTopRow = (nRow1 == nStartRow);
     314         108 :             if ( bRowStrings && (bVert || nCol1 == nStartCol) )
     315             :             {   // NONE or ROWS: RowStrings in every selection possible
     316             :                 // COLS or BOTH: only from first column
     317         108 :                 if ( nCol1 <= nCol2 )
     318         406 :                     for (iRow=nRow1; iRow<=nRow2 && bRowStrings; iRow++)
     319             :                     {
     320         298 :                         if (lcl_hasValueDataButNoDates( pDocument, nCol1, iRow, nTab1 ))
     321          22 :                                 bRowStrings = false;
     322             :                     }
     323             :             }
     324         108 :             if ( bColStrings && bTopRow )
     325             :             {   // ColStrings only from first row
     326          30 :                 if ( nRow1 <= nRow2 )
     327          86 :                     for (iCol=nCol1; iCol<=nCol2 && bColStrings; iCol++)
     328             :                     {
     329          56 :                         if (lcl_hasValueDataButNoDates( pDocument, iCol, nRow1, nTab1 ))
     330           4 :                                 bColStrings = false;
     331             :                     }
     332             :             }
     333             :         }
     334             :     }
     335          52 :     bColHeaders = bColStrings;
     336          52 :     bRowHeaders = bRowStrings;
     337          52 : }
     338             : 
     339         576 : const ScChartPositionMap* ScChartPositioner::GetPositionMap()
     340             : {
     341         576 :     CreatePositionMap();
     342         576 :     return pPositionMap;
     343             : }
     344             : 
     345         576 : void ScChartPositioner::CreatePositionMap()
     346             : {
     347         576 :     if ( eGlue == SC_CHARTGLUE_NA && pPositionMap )
     348             :     {
     349           0 :         delete pPositionMap;
     350           0 :         pPositionMap = NULL;
     351             :     }
     352             : 
     353         576 :     if ( pPositionMap )
     354        1116 :         return ;
     355             : 
     356          36 :     SCSIZE nColAdd = bRowHeaders ? 1 : 0;
     357          36 :     SCSIZE nRowAdd = bColHeaders ? 1 : 0;
     358             : 
     359             :     SCCOL nCol, nCol1, nCol2;
     360             :     SCROW nRow, nRow1, nRow2;
     361             :     SCTAB nTab, nTab1, nTab2;
     362             : 
     363             :     //  real size (without hidden rows/columns)
     364             : 
     365          36 :     SCSIZE nColCount = 0;
     366          36 :     SCSIZE nRowCount = 0;
     367             : 
     368          36 :     GlueState();
     369             : 
     370          36 :     const bool bNoGlue = (eGlue == SC_CHARTGLUE_NONE);
     371          36 :     ColumnMap* pCols = new ColumnMap;
     372          36 :     SCROW nNoGlueRow = 0;
     373         162 :     for ( size_t i = 0, nRanges = aRangeListRef->size(); i < nRanges; ++i )
     374             :     {
     375         126 :         ScRange* pR = (*aRangeListRef)[i];
     376         126 :         pR->GetVars( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 );
     377         252 :         for ( nTab = nTab1; nTab <= nTab2; nTab++ )
     378             :         {
     379             :             // nTab in ColKey to allow to have the same col/row in another tabe
     380         126 :             sal_uLong nInsCol = (static_cast<sal_uLong>(nTab) << 16) | (bNoGlue ? 0 :
     381         126 :                     static_cast<sal_uLong>(nCol1));
     382         390 :             for ( nCol = nCol1; nCol <= nCol2; ++nCol, ++nInsCol )
     383             :             {
     384         264 :                 RowMap* pCol = NULL;
     385         264 :                 ColumnMap::const_iterator it = pCols->find( nInsCol );
     386         264 :                 if ( it == pCols->end() )
     387             :                 {
     388          84 :                     pCol = new RowMap;
     389          84 :                     pCols->insert( ColumnMap::value_type( nInsCol, pCol ) );
     390             :                 }
     391             :                 else
     392         180 :                     pCol = it->second;
     393             : 
     394             :                 // in other table a new ColKey already was created,
     395             :                 // the rows must be equal to be filled with Dummy
     396         264 :                 sal_uLong nInsRow = (bNoGlue ? nNoGlueRow : nRow1);
     397        1104 :                 for ( nRow = nRow1; nRow <= nRow2; nRow++, nInsRow++ )
     398             :                 {
     399         840 :                     if ( pCol->find( nInsRow ) == pCol->end() )
     400             :                     {
     401         840 :                         pCol->insert( RowMap::value_type( nInsRow, new ScAddress( nCol, nRow, nTab ) ) );
     402             :                     }
     403             :                 }
     404             :             }
     405             :         }
     406             :         // For NoGlue: associated tables will be rendered as ColGlue
     407         126 :         nNoGlueRow += nRow2 - nRow1 + 1;
     408             :     }
     409             : 
     410             :     // count of data
     411          36 :     nColCount = static_cast< SCSIZE >( pCols->size());
     412          36 :     if ( !pCols->empty() )
     413             :     {
     414          36 :         RowMap* pCol = pCols->begin()->second;
     415          36 :         if ( bDummyUpperLeft )
     416           0 :             (*pCol)[ 0 ] = NULL; // Dummy for labeling
     417          36 :         nRowCount = static_cast< SCSIZE >( pCol->size());
     418             :     }
     419             :     else
     420           0 :         nRowCount = 0;
     421          36 :     if ( nColCount > 0 )
     422          36 :         nColCount -= nColAdd;
     423          36 :     if ( nRowCount > 0 )
     424          36 :         nRowCount -= nRowAdd;
     425             : 
     426          36 :     if ( nColCount==0 || nRowCount==0 )
     427             :     {   // create an entry without data
     428             :         RowMap* pCol;
     429           0 :         if ( !pCols->empty() )
     430           0 :             pCol = pCols->begin()->second;
     431             :         else
     432             :         {
     433           0 :             pCol = new RowMap;
     434           0 :             (*pCols)[ 0 ] = pCol;
     435             :         }
     436           0 :         nColCount = 1;
     437           0 :         if ( !pCol->empty() )
     438             :         {   // cannot be if nColCount==0 || nRowCount==0
     439           0 :             ScAddress* pPos = pCol->begin()->second;
     440           0 :             if ( pPos )
     441             :             {
     442           0 :                 sal_uLong nCurrentKey = pCol->begin()->first;
     443           0 :                 delete pPos;
     444           0 :                 (*pCol)[ nCurrentKey ] = NULL;
     445             :             }
     446             :         }
     447             :         else
     448           0 :             (*pCol)[ 0 ] = NULL;
     449           0 :         nRowCount = 1;
     450           0 :         nColAdd = 0;
     451           0 :         nRowAdd = 0;
     452             :     }
     453             :     else
     454             :     {
     455          36 :         if ( bNoGlue )
     456             :         {   // fill gaps with Dummies, first column is master
     457          36 :             RowMap* pFirstCol = pCols->begin()->second;
     458          36 :             sal_uLong nCount = pFirstCol->size();
     459          36 :             RowMap::const_iterator it1 = pFirstCol->begin();
     460         426 :             for ( sal_uLong n = 0; n < nCount; n++, ++it1 )
     461             :             {
     462         390 :                 sal_uLong nKey = it1->first;
     463         840 :                 for (ColumnMap::const_iterator it2 = ++pCols->begin(); it2 != pCols->end(); ++it2 )
     464         450 :                     it2->second->insert( RowMap::value_type( nKey, (ScAddress *)NULL )); // no data
     465             :             }
     466             :         }
     467             :     }
     468             : 
     469             :     pPositionMap = new ScChartPositionMap( static_cast<SCCOL>(nColCount), static_cast<SCROW>(nRowCount),
     470          36 :         static_cast<SCCOL>(nColAdd), static_cast<SCROW>(nRowAdd), *pCols );
     471             : 
     472             :     //  cleanup
     473         120 :     for (ColumnMap::const_iterator it = pCols->begin(); it != pCols->end(); ++it )
     474             :     {   // Only delete tables, not the ScAddress*!
     475          84 :         delete it->second;
     476             :     }
     477          36 :     delete pCols;
     478             : }
     479             : 
     480          36 : ScChartPositionMap::ScChartPositionMap( SCCOL nChartCols, SCROW nChartRows,
     481             :             SCCOL nColAdd, SCROW nRowAdd, ColumnMap& rCols ) :
     482          36 :         ppData( new ScAddress* [ nChartCols * nChartRows ] ),
     483          36 :         ppColHeader( new ScAddress* [ nChartCols ] ),
     484          36 :         ppRowHeader( new ScAddress* [ nChartRows ] ),
     485          36 :         nCount( (sal_uLong) nChartCols * nChartRows ),
     486             :         nColCount( nChartCols ),
     487         144 :         nRowCount( nChartRows )
     488             : {
     489             :     OSL_ENSURE( nColCount && nRowCount, "ScChartPositionMap without dimension" );
     490             : 
     491          36 :     ColumnMap::const_iterator pColIter = rCols.begin();
     492          36 :     RowMap* pCol1 = pColIter->second;
     493          36 :     RowMap::const_iterator pPos1Iter;
     494             : 
     495             :     // row header
     496          36 :     pPos1Iter = pCol1->begin();
     497          36 :     if ( nRowAdd )
     498          28 :         ++pPos1Iter;
     499          36 :     if ( nColAdd )
     500             :     {   // independent
     501          28 :         SCROW nRow = 0;
     502         294 :         for ( ; nRow < nRowCount && pPos1Iter != pCol1->end(); nRow++ )
     503             :         {
     504         266 :             ppRowHeader[ nRow ] = pPos1Iter->second;
     505         266 :             ++pPos1Iter;
     506             :         }
     507          28 :         for ( ; nRow < nRowCount; nRow++ )
     508           0 :             ppRowHeader[ nRow ] = NULL;
     509             :     }
     510             :     else
     511             :     {   // copy
     512           8 :         SCROW nRow = 0;
     513         104 :         for ( ; nRow < nRowCount && pPos1Iter != pCol1->end(); nRow++ )
     514             :         {
     515          96 :             ppRowHeader[ nRow ] = pPos1Iter->second ?
     516          96 :                 new ScAddress( *pPos1Iter->second ) : NULL;
     517          96 :             ++pPos1Iter;
     518             :         }
     519           8 :         for ( ; nRow < nRowCount; nRow++ )
     520           0 :             ppRowHeader[ nRow ] = NULL;
     521             :     }
     522          36 :     if ( nColAdd )
     523             :     {
     524          28 :         ++pColIter;
     525             :     }
     526             : 
     527             :     // data column by column and column-header
     528          36 :     sal_uLong nIndex = 0;
     529          92 :     for ( SCCOL nCol = 0; nCol < nColCount; nCol++ )
     530             :     {
     531          56 :         if ( pColIter != rCols.end() )
     532             :         {
     533          56 :             RowMap* pCol2 = pColIter->second;
     534          56 :             RowMap::const_iterator pPosIter = pCol2->begin();
     535          56 :             if ( pPosIter != pCol2->end() )
     536             :             {
     537          56 :                 if ( nRowAdd )
     538             :                 {
     539          40 :                     ppColHeader[ nCol ] = pPosIter->second; // independent
     540          40 :                     ++pPosIter;
     541             :                 }
     542             :                 else
     543          16 :                     ppColHeader[ nCol ] = pPosIter->second ?
     544          16 :                         new ScAddress( *pPosIter->second ) : NULL;
     545             :             }
     546             : 
     547          56 :             SCROW nRow = 0;
     548         562 :             for ( ; nRow < nRowCount && pPosIter != pCol2->end(); nRow++, nIndex++ )
     549             :             {
     550         506 :                 ppData[ nIndex ] = pPosIter->second;
     551         506 :                 ++pPosIter;
     552             :             }
     553          56 :             for ( ; nRow < nRowCount; nRow++, nIndex++ )
     554           0 :                 ppData[ nIndex ] = NULL;
     555             : 
     556          56 :             ++pColIter;
     557             :         }
     558             :         else
     559             :         {
     560           0 :             ppColHeader[ nCol ] = NULL;
     561           0 :             for ( SCROW nRow = 0; nRow < nRowCount; nRow++, nIndex++ )
     562             :             {
     563           0 :                 ppData[ nIndex ] = NULL;
     564             :             }
     565             :         }
     566             :     }
     567          36 : }
     568             : 
     569          36 : ScChartPositionMap::~ScChartPositionMap()
     570             : {
     571         542 :     for ( sal_uLong nIndex=0; nIndex < nCount; nIndex++ )
     572             :     {
     573         506 :         delete ppData[nIndex];
     574             :     }
     575          36 :     delete [] ppData;
     576             : 
     577             :     SCCOL j;
     578          92 :     for ( j=0; j < nColCount; j++ )
     579             :     {
     580          56 :         delete ppColHeader[j];
     581             :     }
     582          36 :     delete [] ppColHeader;
     583             :     SCROW i;
     584         398 :     for ( i=0; i < nRowCount; i++ )
     585             :     {
     586         362 :         delete ppRowHeader[i];
     587             :     }
     588          36 :     delete [] ppRowHeader;
     589         264 : }
     590             : 
     591             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */

Generated by: LCOV version 1.10