/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using ::std::vector; ScMemChart::ScMemChart(SCCOL nCols, SCROW nRows) { nRowCnt = nRows; nColCnt = nCols; pData.reset( new double[nColCnt * nRowCnt] ); memset( pData.get(), 0.0, nColCnt * nRowCnt ); pColText.reset( new OUString[nColCnt] ); pRowText.reset( new OUString[nRowCnt] ); } ScMemChart::~ScMemChart() { } ScChartArray::ScChartArray( ScDocument* pDoc, const ScRangeListRef& rRangeList ) : pDocument( pDoc ), aPositioner(pDoc, rRangeList) {} std::unique_ptr ScChartArray::CreateMemChart() { ScRangeListRef aRangeListRef(GetRangeList()); size_t nCount = aRangeListRef->size(); if ( nCount > 1 ) return CreateMemChartMulti(); else if ( nCount == 1 ) { const ScRange & rR = aRangeListRef->front(); if ( rR.aStart.Tab() != rR.aEnd.Tab() ) return CreateMemChartMulti(); else return CreateMemChartSingle(); } else return CreateMemChartMulti(); // Can handle 0 range better than Single } namespace { double getCellValue( ScDocument& rDoc, const ScAddress& rPos, double fDefault, bool bCalcAsShown ) { double fRet = fDefault; ScRefCellValue aCell(rDoc, rPos); switch (aCell.meType) { case CELLTYPE_VALUE: { fRet = aCell.getValue(); if (bCalcAsShown && fRet != 0.0) { sal_uInt32 nFormat = rDoc.GetNumberFormat(rPos); fRet = rDoc.RoundValueAsShown(fRet, nFormat); } } break; case CELLTYPE_FORMULA: { ScFormulaCell* pFCell = aCell.mpFormula; if (pFCell && pFCell->GetErrCode() == FormulaError::NONE && pFCell->IsValue()) fRet = pFCell->GetValue(); } break; default: ; } return fRet; } } std::unique_ptr ScChartArray::CreateMemChartSingle() { SCSIZE nCol; SCSIZE nRow; // real size (without hidden rows/columns) SCCOL nColAdd = HasRowHeaders() ? 1 : 0; SCROW nRowAdd = HasColHeaders() ? 1 : 0; SCCOL nCol1; SCROW nRow1; SCTAB nTab1; SCCOL nCol2; SCROW nRow2; SCTAB nTab2; ScRangeListRef aRangeListRef(GetRangeList()); aRangeListRef->front().GetVars( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 ); SCCOL nStrCol = nCol1; // remember for labeling SCROW nStrRow = nRow1; // Skip hidden columns. // TODO: make use of last column value once implemented. SCCOL nLastCol = -1; while (pDocument->ColHidden(nCol1, nTab1, nullptr, &nLastCol)) ++nCol1; // Skip hidden rows. SCROW nLastRow = -1; if (pDocument->RowHidden(nRow1, nTab1, nullptr, &nLastRow)) nRow1 = nLastRow + 1; // if everything is hidden then the label remains at the beginning if ( nCol1 <= nCol2 ) { nStrCol = nCol1; nCol1 = sal::static_int_cast( nCol1 + nColAdd ); } if ( nRow1 <= nRow2 ) { nStrRow = nRow1; nRow1 = sal::static_int_cast( nRow1 + nRowAdd ); } SCSIZE nTotalCols = ( nCol1 <= nCol2 ? nCol2 - nCol1 + 1 : 0 ); vector aCols; aCols.reserve(nTotalCols); for (SCSIZE i=0; i(nCol1+i); if (!pDocument->ColHidden(nThisCol, nTab1, nullptr, &nLastCol)) aCols.push_back(nThisCol); } SCSIZE nColCount = aCols.size(); SCSIZE nTotalRows = ( nRow1 <= nRow2 ? nRow2 - nRow1 + 1 : 0 ); vector aRows; aRows.reserve(nTotalRows); if (nRow1 <= nRow2) { // Get all visible rows between nRow1 and nRow2. SCROW nThisRow = nRow1; while (nThisRow <= nRow2) { if (pDocument->RowHidden(nThisRow, nTab1, nullptr, &nLastRow)) nThisRow = nLastRow; else aRows.push_back(nThisRow); ++nThisRow; } } SCSIZE nRowCount = aRows.size(); // May happen at least with more than 32k rows. if (nColCount > SHRT_MAX || nRowCount > SHRT_MAX) { nColCount = 0; nRowCount = 0; } bool bValidData = true; if ( !nColCount ) { bValidData = false; nColCount = 1; aCols.push_back(nStrCol); } if ( !nRowCount ) { bValidData = false; nRowCount = 1; aRows.push_back(nStrRow); } // Data std::unique_ptr pMemChart(new ScMemChart( nColCount, nRowCount )); if ( bValidData ) { bool bCalcAsShown = pDocument->GetDocOptions().IsCalcAsShown(); for (nCol=0; nColSetData(nCol, nRow, nVal); } } } else { // Flag marking data as invalid? for (nCol=0; nColSetData( nCol, nRow, DBL_MIN ); } // Column Header for (nCol=0; nColGetString(aCols[nCol], nStrRow, nTab1); if (aString.isEmpty()) { OUStringBuffer aBuf; aBuf.append(ScResId(STR_COLUMN)); aBuf.append(' '); ScAddress aPos( aCols[ nCol ], 0, 0 ); aBuf.append(aPos.Format(ScRefFlags::COL_VALID)); aString = aBuf.makeStringAndClear(); } pMemChart->SetColText( nCol, aString); } // Row Header for (nRow=0; nRowGetString(nStrCol, aRows[nRow], nTab1); } if (aString.isEmpty()) { OUStringBuffer aBuf; aBuf.append(ScResId(STR_ROW)); aBuf.append(' '); aBuf.append(static_cast(aRows[nRow]+1)); aString = aBuf.makeStringAndClear(); } pMemChart->SetRowText( nRow, aString); } return pMemChart; } std::unique_ptr ScChartArray::CreateMemChartMulti() { SCSIZE nColCount = GetPositionMap()->GetColCount(); SCSIZE nRowCount = GetPositionMap()->GetRowCount(); // May happen at least with more than 32k rows. if (nColCount > SHRT_MAX || nRowCount > SHRT_MAX) { nColCount = 0; nRowCount = 0; } bool bValidData = true; if ( !nColCount ) { bValidData = false; nColCount = 1; } if ( !nRowCount ) { bValidData = false; nRowCount = 1; } // Data std::unique_ptr pMemChart(new ScMemChart( nColCount, nRowCount )); SCSIZE nCol = 0; SCSIZE nRow = 0; bool bCalcAsShown = pDocument->GetDocOptions().IsCalcAsShown(); sal_uLong nIndex = 0; if (bValidData) { for ( nCol = 0; nCol < nColCount; nCol++ ) { for ( nRow = 0; nRow < nRowCount; nRow++, nIndex++ ) { double nVal = DBL_MIN; // Hack for Chart to recognize empty cells const ScAddress* pPos = GetPositionMap()->GetPosition( nIndex ); if (pPos) // otherwise: Gap nVal = getCellValue(*pDocument, *pPos, DBL_MIN, bCalcAsShown); pMemChart->SetData(nCol, nRow, nVal); } } } else { for ( nRow = 0; nRow < nRowCount; nRow++, nIndex++ ) { double nVal = DBL_MIN; // Hack for Chart to recognize empty cells const ScAddress* pPos = GetPositionMap()->GetPosition( nIndex ); if (pPos) // otherwise: Gap nVal = getCellValue(*pDocument, *pPos, DBL_MIN, bCalcAsShown); pMemChart->SetData(nCol, nRow, nVal); } } //TODO: Label when gaps // Column header SCCOL nPosCol = 0; for ( nCol = 0; nCol < nColCount; nCol++ ) { OUString aString; const ScAddress* pPos = GetPositionMap()->GetColHeaderPosition( static_cast(nCol) ); if ( HasColHeaders() && pPos ) aString = pDocument->GetString(pPos->Col(), pPos->Row(), pPos->Tab()); if (aString.isEmpty()) { OUStringBuffer aBuf(ScResId(STR_COLUMN)); aBuf.append(' '); if ( pPos ) nPosCol = pPos->Col() + 1; else nPosCol++; ScAddress aPos( nPosCol - 1, 0, 0 ); aBuf.append(aPos.Format(ScRefFlags::COL_VALID)); aString = aBuf.makeStringAndClear(); } pMemChart->SetColText( nCol, aString); } // Row header SCROW nPosRow = 0; for ( nRow = 0; nRow < nRowCount; nRow++ ) { OUString aString; const ScAddress* pPos = GetPositionMap()->GetRowHeaderPosition( nRow ); if ( HasRowHeaders() && pPos ) aString = pDocument->GetString(pPos->Col(), pPos->Row(), pPos->Tab()); if (aString.isEmpty()) { if ( pPos ) nPosRow = pPos->Row() + 1; else nPosRow++; aString = ScResId(STR_ROW) + " " + OUString::number(static_cast(nPosRow)); } pMemChart->SetRowText( nRow, aString); } return pMemChart; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */