/* -*- 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 "interpre.hxx" #include "scitems.hxx" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "patattr.hxx" #include "global.hxx" #include "document.hxx" #include "dociter.hxx" #include "formulacell.hxx" #include "scmatrix.hxx" #include "docoptio.hxx" #include "globstr.hrc" #include "attrib.hxx" #include "jumpmatrix.hxx" #include "cellkeytranslator.hxx" #include "lookupcache.hxx" #include "rangenam.hxx" #include "rangeutl.hxx" #include "compiler.hxx" #include "externalrefmgr.hxx" #include #include "doubleref.hxx" #include "queryparam.hxx" #include "queryentry.hxx" #include "tokenarray.hxx" #include "compare.hxx" #include #include #include #include #include #include #include #include #include #include static const sal_uInt64 n2power48 = SAL_CONST_UINT64( 281474976710656); // 2^48 IMPL_FIXEDMEMPOOL_NEWDEL( ScTokenStack ) IMPL_FIXEDMEMPOOL_NEWDEL( ScInterpreter ) ScCalcConfig *ScInterpreter::mpGlobalConfig = nullptr; ScTokenStack* ScInterpreter::pGlobalStack = nullptr; bool ScInterpreter::bGlobalStackInUse = false; using namespace formula; using ::std::unique_ptr; void ScInterpreter::ScIfJump() { const short* pJump = pCur->GetJump(); short nJumpCount = pJump[ 0 ]; MatrixDoubleRefToMatrix(); switch ( GetStackType() ) { case svMatrix: { ScMatrixRef pMat = PopMatrix(); if ( !pMat ) PushIllegalParameter(); else { FormulaTokenRef xNew; ScTokenMatrixMap::const_iterator aMapIter; // DoubleError handled by JumpMatrix pMat->SetErrorInterpreter( nullptr); SCSIZE nCols, nRows; pMat->GetDimensions( nCols, nRows ); if ( nCols == 0 || nRows == 0 ) { PushIllegalArgument(); return; } else if (pTokenMatrixMap && ((aMapIter = pTokenMatrixMap->find( pCur)) != pTokenMatrixMap->end())) xNew = (*aMapIter).second; else { ScJumpMatrix* pJumpMat = new ScJumpMatrix( nCols, nRows ); for ( SCSIZE nC=0; nC < nCols; ++nC ) { for ( SCSIZE nR=0; nR < nRows; ++nR ) { double fVal; bool bTrue; bool bIsValue = pMat->IsValue(nC, nR); if (bIsValue) { fVal = pMat->GetDouble(nC, nR); bIsValue = ::rtl::math::isFinite(fVal); bTrue = bIsValue && (fVal != 0.0); if (bTrue) fVal = 1.0; } else { // Treat empty and empty path as 0, but string // as error. bIsValue = (!pMat->IsString(nC, nR) || pMat->IsEmpty(nC, nR)); bTrue = false; fVal = (bIsValue ? 0.0 : CreateDoubleError( errNoValue)); } if ( bTrue ) { // TRUE if( nJumpCount >= 2 ) { // THEN path pJumpMat->SetJump( nC, nR, fVal, pJump[ 1 ], pJump[ nJumpCount ]); } else { // no parameter given for THEN pJumpMat->SetJump( nC, nR, fVal, pJump[ nJumpCount ], pJump[ nJumpCount ]); } } else { // FALSE if( nJumpCount == 3 && bIsValue ) { // ELSE path pJumpMat->SetJump( nC, nR, fVal, pJump[ 2 ], pJump[ nJumpCount ]); } else { // no parameter given for ELSE, // or DoubleError pJumpMat->SetJump( nC, nR, fVal, pJump[ nJumpCount ], pJump[ nJumpCount ]); } } } } xNew = new ScJumpMatrixToken( pJumpMat ); GetTokenMatrixMap().insert( ScTokenMatrixMap::value_type(pCur, xNew)); } if (!xNew.get()) { PushIllegalArgument(); return; } PushTempToken( xNew.get()); // set endpoint of path for main code line aCode.Jump( pJump[ nJumpCount ], pJump[ nJumpCount ] ); } } break; default: { if ( GetBool() ) { // TRUE if( nJumpCount >= 2 ) { // THEN path aCode.Jump( pJump[ 1 ], pJump[ nJumpCount ] ); } else { // no parameter given for THEN nFuncFmtType = css::util::NumberFormat::LOGICAL; PushInt(1); aCode.Jump( pJump[ nJumpCount ], pJump[ nJumpCount ] ); } } else { // FALSE if( nJumpCount == 3 ) { // ELSE path aCode.Jump( pJump[ 2 ], pJump[ nJumpCount ] ); } else { // no parameter given for ELSE nFuncFmtType = css::util::NumberFormat::LOGICAL; PushInt(0); aCode.Jump( pJump[ nJumpCount ], pJump[ nJumpCount ] ); } } } } } /** Store a matrix value in another matrix in the context of that other matrix is the result matrix of a jump matrix. All arguments must be valid and are not checked. */ static void lcl_storeJumpMatResult( const ScMatrix* pMat, ScJumpMatrix* pJumpMat, SCSIZE nC, SCSIZE nR ) { if ( pMat->IsValue( nC, nR ) ) { double fVal = pMat->GetDouble( nC, nR ); pJumpMat->PutResultDouble( fVal, nC, nR ); } else if ( pMat->IsEmpty( nC, nR ) ) { pJumpMat->PutResultEmpty( nC, nR ); } else { pJumpMat->PutResultString(pMat->GetString(nC, nR), nC, nR); } } void ScInterpreter::ScIfError( bool bNAonly ) { const short* pJump = pCur->GetJump(); short nJumpCount = pJump[ 0 ]; if (!sp || nJumpCount != 2) { // Reset nGlobalError here to not propagate the old error, if any. nGlobalError = (sp ? errParameterExpected : errUnknownStackVariable); PushError( nGlobalError); aCode.Jump( pJump[ nJumpCount ], pJump[ nJumpCount ] ); return; } FormulaTokenRef xToken( pStack[ sp - 1 ] ); bool bError = false; sal_uInt16 nOldGlobalError = nGlobalError; nGlobalError = 0; MatrixDoubleRefToMatrix(); switch (GetStackType()) { default: Pop(); // Act on implicitly propagated error, if any. if (nOldGlobalError) nGlobalError = nOldGlobalError; if (nGlobalError) bError = true; break; case svError: PopError(); bError = true; break; case svDoubleRef: case svSingleRef: { ScAddress aAdr; if (!PopDoubleRefOrSingleRef( aAdr)) bError = true; else { ScRefCellValue aCell(*pDok, aAdr); nGlobalError = GetCellErrCode(aCell); if (nGlobalError) bError = true; } } break; case svExternalSingleRef: case svExternalDoubleRef: { double fVal; svl::SharedString aStr; // Handles also existing jump matrix case and sets error on // elements. GetDoubleOrStringFromMatrix( fVal, aStr); if (nGlobalError) bError = true; } break; case svMatrix: { const ScMatrixRef pMat = PopMatrix(); if (!pMat || (nGlobalError && (!bNAonly || nGlobalError == NOTAVAILABLE))) { bError = true; break; // switch } // If the matrix has no queried error at all we can simply use // it as result and don't need to bother with jump matrix. SCSIZE nErrorCol = ::std::numeric_limits::max(), nErrorRow = ::std::numeric_limits::max(); SCSIZE nCols, nRows; pMat->GetDimensions( nCols, nRows ); if (nCols == 0 || nRows == 0) { bError = true; break; // switch } for (SCSIZE nC=0; nC < nCols && !bError; ++nC) { for (SCSIZE nR=0; nR < nRows && !bError; ++nR) { sal_uInt16 nErr = pMat->GetError( nC, nR ); if (nErr && (!bNAonly || nErr == NOTAVAILABLE)) { bError = true; nErrorCol = nC; nErrorRow = nR; } } } if (!bError) break; // switch, we're done and have the result FormulaTokenRef xNew; ScTokenMatrixMap::const_iterator aMapIter; if (pTokenMatrixMap && ((aMapIter = pTokenMatrixMap->find( pCur)) != pTokenMatrixMap->end())) { xNew = (*aMapIter).second; } else { const ScMatrix* pMatPtr = pMat.get(); ScJumpMatrix* pJumpMat = new ScJumpMatrix( nCols, nRows ); // Init all jumps to no error to save single calls. Error // is the exceptional condition. const double fFlagResult = CreateDoubleError( errJumpMatHasResult); pJumpMat->SetAllJumps( fFlagResult, pJump[ nJumpCount ], pJump[ nJumpCount ] ); // Up to first error position simply store results, no need // to evaluate error conditions again. SCSIZE nC = 0, nR = 0; for ( ; nC < nCols && (nC != nErrorCol || nR != nErrorRow); /*nop*/ ) { for (nR = 0 ; nR < nRows && (nC != nErrorCol || nR != nErrorRow); ++nR) { lcl_storeJumpMatResult(pMatPtr, pJumpMat, nC, nR); } if (nC != nErrorCol && nR != nErrorRow) ++nC; } // Now the mixed cases. for ( ; nC < nCols; ++nC) { for ( ; nR < nRows; ++nR) { sal_uInt16 nErr = pMat->GetError( nC, nR ); if (nErr && (!bNAonly || nErr == NOTAVAILABLE)) { // TRUE, THEN path pJumpMat->SetJump( nC, nR, 1.0, pJump[ 1 ], pJump[ nJumpCount ] ); } else { // FALSE, EMPTY path, store result instead lcl_storeJumpMatResult(pMatPtr, pJumpMat, nC, nR); } } nR = 0; } xNew = new ScJumpMatrixToken( pJumpMat ); GetTokenMatrixMap().insert( ScTokenMatrixMap::value_type( pCur, xNew )); } nGlobalError = nOldGlobalError; PushTempToken( xNew.get() ); // set endpoint of path for main code line aCode.Jump( pJump[ nJumpCount ], pJump[ nJumpCount ] ); return; } break; } if (bError && (!bNAonly || nGlobalError == NOTAVAILABLE)) { // error, calculate 2nd argument nGlobalError = 0; aCode.Jump( pJump[ 1 ], pJump[ nJumpCount ] ); } else { // no error, push 1st argument and continue nGlobalError = nOldGlobalError; PushTempToken( xToken.get()); aCode.Jump( pJump[ nJumpCount ], pJump[ nJumpCount ] ); } } void ScInterpreter::ScChooseJump() { // We have to set a jump, if there was none chosen because of an error set // it to endpoint. bool bHaveJump = false; const short* pJump = pCur->GetJump(); short nJumpCount = pJump[ 0 ]; MatrixDoubleRefToMatrix(); switch ( GetStackType() ) { case svMatrix: { ScMatrixRef pMat = PopMatrix(); if ( !pMat ) PushIllegalParameter(); else { FormulaTokenRef xNew; ScTokenMatrixMap::const_iterator aMapIter; // DoubleError handled by JumpMatrix pMat->SetErrorInterpreter( nullptr); SCSIZE nCols, nRows; pMat->GetDimensions( nCols, nRows ); if ( nCols == 0 || nRows == 0 ) PushIllegalParameter(); else if (pTokenMatrixMap && ((aMapIter = pTokenMatrixMap->find( pCur)) != pTokenMatrixMap->end())) xNew = (*aMapIter).second; else { ScJumpMatrix* pJumpMat = new ScJumpMatrix( nCols, nRows ); for ( SCSIZE nC=0; nC < nCols; ++nC ) { for ( SCSIZE nR=0; nR < nRows; ++nR ) { double fVal; bool bIsValue = pMat->IsValue(nC, nR); if ( bIsValue ) { fVal = pMat->GetDouble(nC, nR); bIsValue = ::rtl::math::isFinite( fVal ); if ( bIsValue ) { fVal = ::rtl::math::approxFloor( fVal); if ( (fVal < 1) || (fVal >= nJumpCount)) { bIsValue = false; fVal = CreateDoubleError( errIllegalArgument); } } } else { fVal = CreateDoubleError( errNoValue); } if ( bIsValue ) { pJumpMat->SetJump( nC, nR, fVal, pJump[ (short)fVal ], pJump[ nJumpCount ]); } else { pJumpMat->SetJump( nC, nR, fVal, pJump[ nJumpCount ], pJump[ nJumpCount ]); } } } xNew = new ScJumpMatrixToken( pJumpMat ); GetTokenMatrixMap().insert( ScTokenMatrixMap::value_type( pCur, xNew)); } if (xNew.get()) { PushTempToken( xNew.get()); // set endpoint of path for main code line aCode.Jump( pJump[ nJumpCount ], pJump[ nJumpCount ] ); bHaveJump = true; } } } break; default: { sal_Int16 nJumpIndex = GetInt16(); if (!nGlobalError && (nJumpIndex >= 1) && (nJumpIndex < nJumpCount)) { aCode.Jump( pJump[ (short) nJumpIndex ], pJump[ nJumpCount ] ); bHaveJump = true; } else PushIllegalArgument(); } } if (!bHaveJump) aCode.Jump( pJump[ nJumpCount ], pJump[ nJumpCount ] ); } static void lcl_AdjustJumpMatrix( ScJumpMatrix* pJumpM, SCSIZE nParmCols, SCSIZE nParmRows ) { SCSIZE nJumpCols, nJumpRows; SCSIZE nResCols, nResRows; SCSIZE nAdjustCols, nAdjustRows; pJumpM->GetDimensions( nJumpCols, nJumpRows ); pJumpM->GetResMatDimensions( nResCols, nResRows ); if (( nJumpCols == 1 && nParmCols > nResCols ) || ( nJumpRows == 1 && nParmRows > nResRows )) { if ( nJumpCols == 1 && nJumpRows == 1 ) { nAdjustCols = nParmCols > nResCols ? nParmCols : nResCols; nAdjustRows = nParmRows > nResRows ? nParmRows : nResRows; } else if ( nJumpCols == 1 ) { nAdjustCols = nParmCols; nAdjustRows = nResRows; } else { nAdjustCols = nResCols; nAdjustRows = nParmRows; } pJumpM->SetNewResMat( nAdjustCols, nAdjustRows ); } } bool ScInterpreter::JumpMatrix( short nStackLevel ) { pJumpMatrix = pStack[sp-nStackLevel]->GetJumpMatrix(); bool bHasResMat = pJumpMatrix->HasResultMatrix(); SCSIZE nC, nR; if ( nStackLevel == 2 ) { if ( aCode.HasStacked() ) aCode.Pop(); // pop what Jump() pushed else { OSL_FAIL( "ScInterpreter::JumpMatrix: pop goes the weasel" ); } if ( !bHasResMat ) { Pop(); SetError( errUnknownStackVariable ); } else { pJumpMatrix->GetPos( nC, nR ); switch ( GetStackType() ) { case svDouble: { double fVal = GetDouble(); if ( nGlobalError ) { fVal = CreateDoubleError( nGlobalError ); nGlobalError = 0; } pJumpMatrix->PutResultDouble( fVal, nC, nR ); } break; case svString: { svl::SharedString aStr = GetString(); if ( nGlobalError ) { pJumpMatrix->PutResultDouble( CreateDoubleError( nGlobalError), nC, nR); nGlobalError = 0; } else pJumpMatrix->PutResultString(aStr, nC, nR); } break; case svSingleRef: { ScAddress aAdr; PopSingleRef( aAdr ); if ( nGlobalError ) { pJumpMatrix->PutResultDouble( CreateDoubleError( nGlobalError), nC, nR); nGlobalError = 0; } else { ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasEmptyValue()) pJumpMatrix->PutResultEmpty( nC, nR ); else if (aCell.hasNumeric()) { double fVal = GetCellValue(aAdr, aCell); if ( nGlobalError ) { fVal = CreateDoubleError( nGlobalError); nGlobalError = 0; } pJumpMatrix->PutResultDouble( fVal, nC, nR ); } else { svl::SharedString aStr; GetCellString(aStr, aCell); if ( nGlobalError ) { pJumpMatrix->PutResultDouble( CreateDoubleError( nGlobalError), nC, nR); nGlobalError = 0; } else pJumpMatrix->PutResultString(aStr, nC, nR); } } } break; case svDoubleRef: { // upper left plus offset within matrix double fVal; ScRange aRange; PopDoubleRef( aRange ); if ( nGlobalError ) { fVal = CreateDoubleError( nGlobalError ); nGlobalError = 0; pJumpMatrix->PutResultDouble( fVal, nC, nR ); } else { // Do not modify the original range because we use it // to adjust the size of the result matrix if necessary. ScAddress aAdr( aRange.aStart); sal_uLong nCol = (sal_uLong)aAdr.Col() + nC; sal_uLong nRow = (sal_uLong)aAdr.Row() + nR; if ((nCol > static_cast(aRange.aEnd.Col()) && aRange.aEnd.Col() != aRange.aStart.Col()) || (nRow > static_cast(aRange.aEnd.Row()) && aRange.aEnd.Row() != aRange.aStart.Row())) { fVal = CreateDoubleError( NOTAVAILABLE ); pJumpMatrix->PutResultDouble( fVal, nC, nR ); } else { // Replicate column and/or row of a vector if it is // one. Note that this could be a range reference // that in fact consists of only one cell, e.g. A1:A1 if (aRange.aEnd.Col() == aRange.aStart.Col()) nCol = aRange.aStart.Col(); if (aRange.aEnd.Row() == aRange.aStart.Row()) nRow = aRange.aStart.Row(); aAdr.SetCol( static_cast(nCol) ); aAdr.SetRow( static_cast(nRow) ); ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasEmptyValue()) pJumpMatrix->PutResultEmpty( nC, nR ); else if (aCell.hasNumeric()) { double fCellVal = GetCellValue(aAdr, aCell); if ( nGlobalError ) { fCellVal = CreateDoubleError( nGlobalError); nGlobalError = 0; } pJumpMatrix->PutResultDouble( fCellVal, nC, nR ); } else { svl::SharedString aStr; GetCellString(aStr, aCell); if ( nGlobalError ) { pJumpMatrix->PutResultDouble( CreateDoubleError( nGlobalError), nC, nR); nGlobalError = 0; } else pJumpMatrix->PutResultString(aStr, nC, nR); } } SCSIZE nParmCols = aRange.aEnd.Col() - aRange.aStart.Col() + 1; SCSIZE nParmRows = aRange.aEnd.Row() - aRange.aStart.Row() + 1; lcl_AdjustJumpMatrix( pJumpMatrix, nParmCols, nParmRows ); } } break; case svMatrix: { // match matrix offsets double fVal; ScMatrixRef pMat = PopMatrix(); if ( nGlobalError ) { fVal = CreateDoubleError( nGlobalError ); nGlobalError = 0; pJumpMatrix->PutResultDouble( fVal, nC, nR ); } else if ( !pMat ) { fVal = CreateDoubleError( errUnknownVariable ); pJumpMatrix->PutResultDouble( fVal, nC, nR ); } else { SCSIZE nCols, nRows; pMat->GetDimensions( nCols, nRows ); if ((nCols <= nC && nCols != 1) || (nRows <= nR && nRows != 1)) { fVal = CreateDoubleError( NOTAVAILABLE ); pJumpMatrix->PutResultDouble( fVal, nC, nR ); } else { lcl_storeJumpMatResult(pMat.get(), pJumpMatrix, nC, nR); } lcl_AdjustJumpMatrix( pJumpMatrix, nCols, nRows ); } } break; case svError: { PopError(); double fVal = CreateDoubleError( nGlobalError); nGlobalError = 0; pJumpMatrix->PutResultDouble( fVal, nC, nR ); } break; default: { Pop(); double fVal = CreateDoubleError( errIllegalArgument); pJumpMatrix->PutResultDouble( fVal, nC, nR ); } } } } bool bCont = pJumpMatrix->Next( nC, nR ); if ( bCont ) { double fBool; short nStart, nNext, nStop; pJumpMatrix->GetJump( nC, nR, fBool, nStart, nNext, nStop ); while ( bCont && nStart == nNext ) { // push all results that have no jump path if ( bHasResMat && (GetDoubleErrorValue( fBool) != errJumpMatHasResult) ) { // a false without path results in an empty path value if ( fBool == 0.0 ) pJumpMatrix->PutResultEmptyPath( nC, nR ); else pJumpMatrix->PutResultDouble( fBool, nC, nR ); } bCont = pJumpMatrix->Next( nC, nR ); if ( bCont ) pJumpMatrix->GetJump( nC, nR, fBool, nStart, nNext, nStop ); } if ( bCont && nStart != nNext ) { const ScTokenVec* pParams = pJumpMatrix->GetJumpParameters(); if ( pParams ) { for ( ScTokenVec::const_iterator i = pParams->begin(); i != pParams->end(); ++i ) { // This is not the current state of the interpreter, so // push without error, and elements' errors are coded into // double. PushWithoutError( *(*i)); } } aCode.Jump( nStart, nNext, nStop ); } } if ( !bCont ) { // we're done with it, throw away jump matrix, keep result ScMatrix* pResMat = pJumpMatrix->GetResultMatrix(); pJumpMatrix = nullptr; Pop(); PushMatrix( pResMat ); // Remove jump matrix from map and remember result matrix in case it // could be reused in another path of the same condition. if (pTokenMatrixMap) { pTokenMatrixMap->erase( pCur); pTokenMatrixMap->insert( ScTokenMatrixMap::value_type( pCur, pStack[sp-1])); } return true; } return false; } double ScInterpreter::Compare( ScQueryOp eOp ) { sc::Compare aComp; aComp.meOp = eOp; aComp.mbIgnoreCase = pDok->GetDocOptions().IsIgnoreCase(); for( short i = 1; i >= 0; i-- ) { sc::Compare::Cell& rCell = aComp.maCells[i]; switch ( GetRawStackType() ) { case svEmptyCell: Pop(); rCell.mbEmpty = true; break; case svMissing: case svDouble: rCell.mfValue = GetDouble(); rCell.mbValue = true; break; case svString: rCell.maStr = GetString(); rCell.mbValue = false; break; case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) break; ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasEmptyValue()) rCell.mbEmpty = true; else if (aCell.hasString()) { svl::SharedString aStr; GetCellString(aStr, aCell); rCell.maStr = aStr; rCell.mbValue = false; } else { rCell.mfValue = GetCellValue(aAdr, aCell); rCell.mbValue = true; } } break; case svExternalSingleRef: { ScMatrixRef pMat = GetMatrix(); if (!pMat) { SetError( errIllegalParameter); break; } SCSIZE nC, nR; pMat->GetDimensions(nC, nR); if (!nC || !nR) { SetError( errIllegalParameter); break; } if (pMat->IsEmpty(0, 0)) rCell.mbEmpty = true; else if (pMat->IsString(0, 0)) { rCell.maStr = pMat->GetString(0, 0); rCell.mbValue = false; } else { rCell.mfValue = pMat->GetDouble(0, 0); rCell.mbValue = true; } } break; case svExternalDoubleRef: // TODO: Find out how to handle this... // Xcl generates a position dependent intersection using // col/row, as it seems to do for all range references, not // only in compare context. We'd need a general implementation // for that behavior similar to svDoubleRef in scalar and array // mode. Which also means we'd have to change all places where // it currently is handled along with svMatrix. default: SetError( errIllegalParameter); break; } } if( nGlobalError ) return 0; nCurFmtType = nFuncFmtType = css::util::NumberFormat::LOGICAL; return sc::CompareFunc(aComp); } sc::RangeMatrix ScInterpreter::CompareMat( ScQueryOp eOp, sc::CompareOptions* pOptions ) { sc::Compare aComp; aComp.meOp = eOp; aComp.mbIgnoreCase = pDok->GetDocOptions().IsIgnoreCase(); sc::RangeMatrix aMat[2]; ScAddress aAdr; for( short i = 1; i >= 0; i-- ) { sc::Compare::Cell& rCell = aComp.maCells[i]; switch (GetRawStackType()) { case svEmptyCell: Pop(); rCell.mbEmpty = true; break; case svMissing: case svDouble: rCell.mfValue = GetDouble(); rCell.mbValue = true; break; case svString: rCell.maStr = GetString(); rCell.mbValue = false; break; case svSingleRef: { PopSingleRef( aAdr ); ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasEmptyValue()) rCell.mbEmpty = true; else if (aCell.hasString()) { svl::SharedString aStr; GetCellString(aStr, aCell); rCell.maStr = aStr; rCell.mbValue = false; } else { rCell.mfValue = GetCellValue(aAdr, aCell); rCell.mbValue = true; } } break; case svExternalSingleRef: case svExternalDoubleRef: case svDoubleRef: case svMatrix: aMat[i] = GetRangeMatrix(); if (!aMat[i].mpMat) SetError( errIllegalParameter); else aMat[i].mpMat->SetErrorInterpreter(nullptr); // errors are transported as DoubleError inside matrix break; default: SetError( errIllegalParameter); break; } } sc::RangeMatrix aRes; if (nGlobalError) { nCurFmtType = nFuncFmtType = css::util::NumberFormat::LOGICAL; return aRes; } if (aMat[0].mpMat && aMat[1].mpMat) { SCSIZE nC0, nC1; SCSIZE nR0, nR1; aMat[0].mpMat->GetDimensions(nC0, nR0); aMat[1].mpMat->GetDimensions(nC1, nR1); SCSIZE nC = std::max( nC0, nC1 ); SCSIZE nR = std::max( nR0, nR1 ); aRes.mpMat = GetNewMat( nC, nR); if (!aRes.mpMat) return aRes; for ( SCSIZE j=0; jValidColRowOrReplicated(nCol, nRow) && aMat[1].mpMat->ValidColRowOrReplicated(nCol, nRow)) { for ( short i=1; i>=0; i-- ) { sc::Compare::Cell& rCell = aComp.maCells[i]; if (aMat[i].mpMat->IsString(j, k)) { rCell.mbValue = false; rCell.maStr = aMat[i].mpMat->GetString(j, k); rCell.mbEmpty = aMat[i].mpMat->IsEmpty(j, k); } else { rCell.mbValue = true; rCell.mfValue = aMat[i].mpMat->GetDouble(j, k); rCell.mbEmpty = false; } } aRes.mpMat->PutDouble( sc::CompareFunc( aComp, pOptions), j, k); } else aRes.mpMat->PutError( errNoValue, j, k); } } switch (eOp) { case SC_EQUAL: aRes.mpMat->CompareEqual(); break; case SC_LESS: aRes.mpMat->CompareLess(); break; case SC_GREATER: aRes.mpMat->CompareGreater(); break; case SC_LESS_EQUAL: aRes.mpMat->CompareLessEqual(); break; case SC_GREATER_EQUAL: aRes.mpMat->CompareGreaterEqual(); break; case SC_NOT_EQUAL: aRes.mpMat->CompareNotEqual(); break; default: OSL_TRACE( "ScInterpreter::QueryMat: unhandled comparison operator: %d", (int)eOp); aRes.mpMat.reset(); return aRes; } } else if (aMat[0].mpMat || aMat[1].mpMat) { size_t i = ( aMat[0].mpMat ? 0 : 1); aRes.mnCol1 = aMat[i].mnCol1; aRes.mnRow1 = aMat[i].mnRow1; aRes.mnTab1 = aMat[i].mnTab1; aRes.mnCol2 = aMat[i].mnCol2; aRes.mnRow2 = aMat[i].mnRow2; aRes.mnTab2 = aMat[i].mnTab2; ScMatrix& rMat = *aMat[i].mpMat; aRes.mpMat = rMat.CompareMatrix(aComp, i, pOptions); if (!aRes.mpMat) return aRes; } nCurFmtType = nFuncFmtType = css::util::NumberFormat::LOGICAL; return aRes; } ScMatrixRef ScInterpreter::QueryMat( const ScMatrixRef& pMat, sc::CompareOptions& rOptions ) { short nSaveCurFmtType = nCurFmtType; short nSaveFuncFmtType = nFuncFmtType; PushMatrix( pMat); const ScQueryEntry::Item& rItem = rOptions.aQueryEntry.GetQueryItem(); if (rItem.meType == ScQueryEntry::ByString) PushString(rItem.maString.getString()); else PushDouble(rItem.mfVal); ScMatrixRef pResultMatrix = CompareMat(rOptions.aQueryEntry.eOp, &rOptions).mpMat; nCurFmtType = nSaveCurFmtType; nFuncFmtType = nSaveFuncFmtType; if (nGlobalError || !pResultMatrix) { SetError( errIllegalParameter); return pResultMatrix; } return pResultMatrix; } void ScInterpreter::ScEqual() { if ( GetStackType(1) == svMatrix || GetStackType(2) == svMatrix ) { sc::RangeMatrix aMat = CompareMat(SC_EQUAL); if (!aMat.mpMat) { PushIllegalParameter(); return; } PushMatrix(aMat); } else PushInt( int(Compare( SC_EQUAL) == 0) ); } void ScInterpreter::ScNotEqual() { if ( GetStackType(1) == svMatrix || GetStackType(2) == svMatrix ) { sc::RangeMatrix aMat = CompareMat(SC_NOT_EQUAL); if (!aMat.mpMat) { PushIllegalParameter(); return; } PushMatrix(aMat); } else PushInt( int(Compare( SC_NOT_EQUAL) != 0) ); } void ScInterpreter::ScLess() { if ( GetStackType(1) == svMatrix || GetStackType(2) == svMatrix ) { sc::RangeMatrix aMat = CompareMat(SC_LESS); if (!aMat.mpMat) { PushIllegalParameter(); return; } PushMatrix(aMat); } else PushInt( int(Compare( SC_LESS) < 0) ); } void ScInterpreter::ScGreater() { if ( GetStackType(1) == svMatrix || GetStackType(2) == svMatrix ) { sc::RangeMatrix aMat = CompareMat(SC_GREATER); if (!aMat.mpMat) { PushIllegalParameter(); return; } PushMatrix(aMat); } else PushInt( int(Compare( SC_GREATER) > 0) ); } void ScInterpreter::ScLessEqual() { if ( GetStackType(1) == svMatrix || GetStackType(2) == svMatrix ) { sc::RangeMatrix aMat = CompareMat(SC_LESS_EQUAL); if (!aMat.mpMat) { PushIllegalParameter(); return; } PushMatrix(aMat); } else PushInt( int(Compare( SC_LESS_EQUAL) <= 0) ); } void ScInterpreter::ScGreaterEqual() { if ( GetStackType(1) == svMatrix || GetStackType(2) == svMatrix ) { sc::RangeMatrix aMat = CompareMat(SC_GREATER_EQUAL); if (!aMat.mpMat) { PushIllegalParameter(); return; } PushMatrix(aMat); } else PushInt( int(Compare( SC_GREATER_EQUAL) >= 0) ); } void ScInterpreter::ScAnd() { nFuncFmtType = css::util::NumberFormat::LOGICAL; short nParamCount = GetByte(); if ( MustHaveParamCountMin( nParamCount, 1 ) ) { bool bHaveValue = false; bool bRes = true; size_t nRefInList = 0; while( nParamCount-- > 0) { if ( !nGlobalError ) { switch ( GetStackType() ) { case svDouble : bHaveValue = true; bRes &= ( PopDouble() != 0.0 ); break; case svString : Pop(); SetError( errNoValue ); break; case svSingleRef : { ScAddress aAdr; PopSingleRef( aAdr ); if ( !nGlobalError ) { ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasNumeric()) { bHaveValue = true; bRes &= ( GetCellValue(aAdr, aCell) != 0.0 ); } // else: Xcl raises no error here } } break; case svDoubleRef: case svRefList: { ScRange aRange; PopDoubleRef( aRange, nParamCount, nRefInList); if ( !nGlobalError ) { double fVal; sal_uInt16 nErr = 0; ScValueIterator aValIter( pDok, aRange ); if ( aValIter.GetFirst( fVal, nErr ) && nErr == 0 ) { bHaveValue = true; do { bRes &= ( fVal != 0.0 ); } while ( (nErr == 0) && aValIter.GetNext( fVal, nErr ) ); } SetError( nErr ); } } break; case svExternalSingleRef: case svExternalDoubleRef: case svMatrix: { ScMatrixRef pMat = GetMatrix(); if ( pMat ) { bHaveValue = true; double fVal = pMat->And(); sal_uInt16 nErr = GetDoubleErrorValue( fVal ); if ( nErr ) { SetError( nErr ); bRes = false; } else bRes &= (fVal != 0.0); } // else: GetMatrix did set errIllegalParameter } break; default: Pop(); SetError( errIllegalParameter); } } else Pop(); } if ( bHaveValue ) PushInt( int(bRes) ); else PushNoValue(); } } void ScInterpreter::ScOr() { nFuncFmtType = css::util::NumberFormat::LOGICAL; short nParamCount = GetByte(); if ( MustHaveParamCountMin( nParamCount, 1 ) ) { bool bHaveValue = false; bool bRes = false; size_t nRefInList = 0; while( nParamCount-- > 0) { if ( !nGlobalError ) { switch ( GetStackType() ) { case svDouble : bHaveValue = true; bRes |= ( PopDouble() != 0.0 ); break; case svString : Pop(); SetError( errNoValue ); break; case svSingleRef : { ScAddress aAdr; PopSingleRef( aAdr ); if ( !nGlobalError ) { ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasNumeric()) { bHaveValue = true; bRes |= ( GetCellValue(aAdr, aCell) != 0.0 ); } // else: Xcl raises no error here } } break; case svDoubleRef: case svRefList: { ScRange aRange; PopDoubleRef( aRange, nParamCount, nRefInList); if ( !nGlobalError ) { double fVal; sal_uInt16 nErr = 0; ScValueIterator aValIter( pDok, aRange ); if ( aValIter.GetFirst( fVal, nErr ) ) { bHaveValue = true; do { bRes |= ( fVal != 0.0 ); } while ( (nErr == 0) && aValIter.GetNext( fVal, nErr ) ); } SetError( nErr ); } } break; case svExternalSingleRef: case svExternalDoubleRef: case svMatrix: { bHaveValue = true; ScMatrixRef pMat = GetMatrix(); if ( pMat ) { bHaveValue = true; double fVal = pMat->Or(); sal_uInt16 nErr = GetDoubleErrorValue( fVal ); if ( nErr ) { SetError( nErr ); bRes = false; } else bRes |= (fVal != 0.0); } // else: GetMatrix did set errIllegalParameter } break; default: Pop(); SetError( errIllegalParameter); } } else Pop(); } if ( bHaveValue ) PushInt( int(bRes) ); else PushNoValue(); } } void ScInterpreter::ScXor() { nFuncFmtType = css::util::NumberFormat::LOGICAL; short nParamCount = GetByte(); if ( MustHaveParamCountMin( nParamCount, 1 ) ) { bool bHaveValue = false; bool bRes = false; size_t nRefInList = 0; while( nParamCount-- > 0) { if ( !nGlobalError ) { switch ( GetStackType() ) { case svDouble : bHaveValue = true; bRes ^= ( PopDouble() != 0.0 ); break; case svString : Pop(); SetError( errNoValue ); break; case svSingleRef : { ScAddress aAdr; PopSingleRef( aAdr ); if ( !nGlobalError ) { ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasNumeric()) { bHaveValue = true; bRes ^= ( GetCellValue(aAdr, aCell) != 0.0 ); } /* TODO: set error? Excel doesn't have XOR, but * doesn't set an error in this case for AND and * OR. */ } } break; case svDoubleRef: case svRefList: { ScRange aRange; PopDoubleRef( aRange, nParamCount, nRefInList); if ( !nGlobalError ) { double fVal; sal_uInt16 nErr = 0; ScValueIterator aValIter( pDok, aRange ); if ( aValIter.GetFirst( fVal, nErr ) ) { bHaveValue = true; do { bRes ^= ( fVal != 0.0 ); } while ( (nErr == 0) && aValIter.GetNext( fVal, nErr ) ); } SetError( nErr ); } } break; case svExternalSingleRef: case svExternalDoubleRef: case svMatrix: { bHaveValue = true; ScMatrixRef pMat = GetMatrix(); if ( pMat ) { bHaveValue = true; double fVal = pMat->Xor(); sal_uInt16 nErr = GetDoubleErrorValue( fVal ); if ( nErr ) { SetError( nErr ); bRes = false; } else bRes ^= ( fVal != 0.0 ); } // else: GetMatrix did set errIllegalParameter } break; default: Pop(); SetError( errIllegalParameter); } } else Pop(); } if ( bHaveValue ) PushInt( int(bRes) ); else PushNoValue(); } } void ScInterpreter::ScNeg() { // Simple negation doesn't change current format type to number, keep // current type. nFuncFmtType = nCurFmtType; switch ( GetStackType() ) { case svMatrix : { ScMatrixRef pMat = GetMatrix(); if ( !pMat ) PushIllegalParameter(); else { SCSIZE nC, nR; pMat->GetDimensions( nC, nR ); ScMatrixRef pResMat = GetNewMat( nC, nR); if ( !pResMat ) PushIllegalArgument(); else { pMat->NegOp( *pResMat); PushMatrix( pResMat ); } } } break; default: PushDouble( -GetDouble() ); } } void ScInterpreter::ScPercentSign() { nFuncFmtType = css::util::NumberFormat::PERCENT; const FormulaToken* pSaveCur = pCur; sal_uInt8 nSavePar = cPar; PushInt( 100 ); cPar = 2; FormulaByteToken aDivOp( ocDiv, cPar ); pCur = &aDivOp; ScDiv(); pCur = pSaveCur; cPar = nSavePar; } void ScInterpreter::ScNot() { nFuncFmtType = css::util::NumberFormat::LOGICAL; switch ( GetStackType() ) { case svMatrix : { ScMatrixRef pMat = GetMatrix(); if ( !pMat ) PushIllegalParameter(); else { SCSIZE nC, nR; pMat->GetDimensions( nC, nR ); ScMatrixRef pResMat = GetNewMat( nC, nR); if ( !pResMat ) PushIllegalArgument(); else { pMat->NotOp( *pResMat); PushMatrix( pResMat ); } } } break; default: PushInt( int(GetDouble() == 0.0) ); } } void ScInterpreter::ScBitAnd() { if ( !MustHaveParamCount( GetByte(), 2 ) ) return; double num1 = ::rtl::math::approxFloor( GetDouble()); double num2 = ::rtl::math::approxFloor( GetDouble()); if ( (num1 >= n2power48) || (num1 < 0) || (num2 >= n2power48) || (num2 < 0)) PushIllegalArgument(); else PushDouble ((sal_uInt64) num1 & (sal_uInt64) num2); } void ScInterpreter::ScBitOr() { if ( !MustHaveParamCount( GetByte(), 2 ) ) return; double num1 = ::rtl::math::approxFloor( GetDouble()); double num2 = ::rtl::math::approxFloor( GetDouble()); if ( (num1 >= n2power48) || (num1 < 0) || (num2 >= n2power48) || (num2 < 0)) PushIllegalArgument(); else PushDouble ((sal_uInt64) num1 | (sal_uInt64) num2); } void ScInterpreter::ScBitXor() { if ( !MustHaveParamCount( GetByte(), 2 ) ) return; double num1 = ::rtl::math::approxFloor( GetDouble()); double num2 = ::rtl::math::approxFloor( GetDouble()); if ( (num1 >= n2power48) || (num1 < 0) || (num2 >= n2power48) || (num2 < 0)) PushIllegalArgument(); else PushDouble ((sal_uInt64) num1 ^ (sal_uInt64) num2); } void ScInterpreter::ScBitLshift() { if ( !MustHaveParamCount( GetByte(), 2 ) ) return; double fShift = ::rtl::math::approxFloor( GetDouble()); double num = ::rtl::math::approxFloor( GetDouble()); if ((num >= n2power48) || (num < 0)) PushIllegalArgument(); else { double fRes; if (fShift < 0) fRes = ::rtl::math::approxFloor( num / pow( 2.0, -fShift)); else if (fShift == 0) fRes = num; else fRes = num * pow( 2.0, fShift); PushDouble( fRes); } } void ScInterpreter::ScBitRshift() { if ( !MustHaveParamCount( GetByte(), 2 ) ) return; double fShift = ::rtl::math::approxFloor( GetDouble()); double num = ::rtl::math::approxFloor( GetDouble()); if ((num >= n2power48) || (num < 0)) PushIllegalArgument(); else { double fRes; if (fShift < 0) fRes = num * pow( 2.0, -fShift); else if (fShift == 0) fRes = num; else fRes = ::rtl::math::approxFloor( num / pow( 2.0, fShift)); PushDouble( fRes); } } void ScInterpreter::ScPi() { PushDouble(F_PI); } void ScInterpreter::ScRandom() { PushDouble(::comphelper::rng::uniform_real_distribution()); } void ScInterpreter::ScTrue() { nFuncFmtType = css::util::NumberFormat::LOGICAL; PushInt(1); } void ScInterpreter::ScFalse() { nFuncFmtType = css::util::NumberFormat::LOGICAL; PushInt(0); } void ScInterpreter::ScDeg() { PushDouble((GetDouble() / F_PI) * 180.0); } void ScInterpreter::ScRad() { PushDouble(GetDouble() * (F_PI / 180)); } void ScInterpreter::ScSin() { PushDouble(::rtl::math::sin(GetDouble())); } void ScInterpreter::ScCos() { PushDouble(::rtl::math::cos(GetDouble())); } void ScInterpreter::ScTan() { PushDouble(::rtl::math::tan(GetDouble())); } void ScInterpreter::ScCot() { PushDouble(1.0 / ::rtl::math::tan(GetDouble())); } void ScInterpreter::ScArcSin() { PushDouble(asin(GetDouble())); } void ScInterpreter::ScArcCos() { PushDouble(acos(GetDouble())); } void ScInterpreter::ScArcTan() { PushDouble(atan(GetDouble())); } void ScInterpreter::ScArcCot() { PushDouble((F_PI2) - atan(GetDouble())); } void ScInterpreter::ScSinHyp() { PushDouble(sinh(GetDouble())); } void ScInterpreter::ScCosHyp() { PushDouble(cosh(GetDouble())); } void ScInterpreter::ScTanHyp() { PushDouble(tanh(GetDouble())); } void ScInterpreter::ScCotHyp() { PushDouble(1.0 / tanh(GetDouble())); } void ScInterpreter::ScArcSinHyp() { PushDouble( ::rtl::math::asinh( GetDouble())); } void ScInterpreter::ScArcCosHyp() { double fVal = GetDouble(); if (fVal < 1.0) PushIllegalArgument(); else PushDouble( ::rtl::math::acosh( fVal)); } void ScInterpreter::ScArcTanHyp() { double fVal = GetDouble(); if (fabs(fVal) >= 1.0) PushIllegalArgument(); else PushDouble( ::rtl::math::atanh( fVal)); } void ScInterpreter::ScArcCotHyp() { double nVal = GetDouble(); if (fabs(nVal) <= 1.0) PushIllegalArgument(); else PushDouble(0.5 * log((nVal + 1.0) / (nVal - 1.0))); } void ScInterpreter::ScCosecant() { PushDouble(1.0 / ::rtl::math::sin(GetDouble())); } void ScInterpreter::ScSecant() { PushDouble(1.0 / ::rtl::math::cos(GetDouble())); } void ScInterpreter::ScCosecantHyp() { PushDouble(1.0 / sinh(GetDouble())); } void ScInterpreter::ScSecantHyp() { PushDouble(1.0 / cosh(GetDouble())); } void ScInterpreter::ScExp() { PushDouble(exp(GetDouble())); } void ScInterpreter::ScSqrt() { double fVal = GetDouble(); if (fVal >= 0.0) PushDouble(sqrt(fVal)); else PushIllegalArgument(); } void ScInterpreter::ScIsEmpty() { short nRes = 0; nFuncFmtType = css::util::NumberFormat::LOGICAL; switch ( GetRawStackType() ) { case svEmptyCell: { FormulaTokenRef p = PopToken(); if (!static_cast(p.get())->IsInherited()) nRes = 1; } break; case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) break; // NOTE: this differs from COUNTBLANK() ScCountEmptyCells() that // may treat ="" in the referenced cell as blank for Excel // interoperability. ScRefCellValue aCell(*pDok, aAdr); if (aCell.meType == CELLTYPE_NONE) nRes = 1; } break; case svExternalSingleRef: case svExternalDoubleRef: case svMatrix: { ScMatrixRef pMat = GetMatrix(); if ( !pMat ) ; // nothing else if ( !pJumpMatrix ) nRes = pMat->IsEmptyCell( 0, 0) ? 1 : 0; else { SCSIZE nCols, nRows, nC, nR; pMat->GetDimensions( nCols, nRows); pJumpMatrix->GetPos( nC, nR); if ( nC < nCols && nR < nRows ) nRes = pMat->IsEmptyCell( nC, nR) ? 1 : 0; // else: false, not empty (which is what Xcl does) } } break; default: Pop(); } nGlobalError = 0; PushInt( nRes ); } bool ScInterpreter::IsString() { nFuncFmtType = css::util::NumberFormat::LOGICAL; bool bRes = false; switch ( GetRawStackType() ) { case svString: Pop(); bRes = true; break; case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) break; ScRefCellValue aCell(*pDok, aAdr); if (GetCellErrCode(aCell) == 0) { switch (aCell.meType) { case CELLTYPE_STRING : case CELLTYPE_EDIT : bRes = true; break; case CELLTYPE_FORMULA : bRes = (!aCell.mpFormula->IsValue() && !aCell.mpFormula->IsEmpty()); break; default: ; // nothing } } } break; case svExternalSingleRef: { ScExternalRefCache::TokenRef pToken; PopExternalSingleRef(pToken); if (!nGlobalError && pToken->GetType() == svString) bRes = true; } break; case svExternalDoubleRef: case svMatrix: { ScMatrixRef pMat = GetMatrix(); if ( !pMat ) ; // nothing else if ( !pJumpMatrix ) bRes = pMat->IsString(0, 0) && !pMat->IsEmpty(0, 0); else { SCSIZE nCols, nRows, nC, nR; pMat->GetDimensions( nCols, nRows); pJumpMatrix->GetPos( nC, nR); if ( nC < nCols && nR < nRows ) bRes = pMat->IsString( nC, nR) && !pMat->IsEmpty( nC, nR); } } break; default: Pop(); } nGlobalError = 0; return bRes; } void ScInterpreter::ScIsString() { PushInt( int(IsString()) ); } void ScInterpreter::ScIsNonString() { PushInt( int(!IsString()) ); } void ScInterpreter::ScIsLogical() { bool bRes = false; switch ( GetStackType() ) { case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) break; ScRefCellValue aCell(*pDok, aAdr); if (GetCellErrCode(aCell) == 0) { if (aCell.hasNumeric()) { sal_uLong nFormat = GetCellNumberFormat(aAdr, aCell); bRes = (pFormatter->GetType(nFormat) == css::util::NumberFormat::LOGICAL); } } } break; case svMatrix: { double fVal; svl::SharedString aStr; ScMatValType nMatValType = GetDoubleOrStringFromMatrix( fVal, aStr); bRes = (nMatValType == SC_MATVAL_BOOLEAN); } break; default: PopError(); if ( !nGlobalError ) bRes = ( nCurFmtType == css::util::NumberFormat::LOGICAL ); } nCurFmtType = nFuncFmtType = css::util::NumberFormat::LOGICAL; nGlobalError = 0; PushInt( int(bRes) ); } void ScInterpreter::ScType() { short nType = 0; switch ( GetStackType() ) { case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) break; ScRefCellValue aCell(*pDok, aAdr); if (GetCellErrCode(aCell) == 0) { switch (aCell.meType) { // NOTE: this is Xcl nonsense! case CELLTYPE_STRING : case CELLTYPE_EDIT : nType = 2; break; case CELLTYPE_VALUE : { sal_uLong nFormat = GetCellNumberFormat(aAdr, aCell); if (pFormatter->GetType(nFormat) == css::util::NumberFormat::LOGICAL) nType = 4; else nType = 1; } break; case CELLTYPE_FORMULA : nType = 8; break; default: PushIllegalArgument(); } } else nType = 16; } break; case svString: PopError(); if ( nGlobalError ) { nType = 16; nGlobalError = 0; } else nType = 2; break; case svMatrix: PopMatrix(); if ( nGlobalError ) { nType = 16; nGlobalError = 0; } else nType = 64; // we could return the type of one element if in JumpMatrix or // ForceArray mode, but Xcl doesn't ... break; default: PopError(); if ( nGlobalError ) { nType = 16; nGlobalError = 0; } else nType = 1; } PushInt( nType ); } static inline bool lcl_FormatHasNegColor( const SvNumberformat* pFormat ) { return pFormat && pFormat->GetColor( 1 ); } static inline bool lcl_FormatHasOpenPar( const SvNumberformat* pFormat ) { return pFormat && (pFormat->GetFormatstring().indexOf('(') != -1); } namespace { void getFormatString(SvNumberFormatter* pFormatter, sal_uLong nFormat, OUString& rFmtStr) { bool bAppendPrec = true; sal_uInt16 nPrec, nLeading; bool bThousand, bIsRed; pFormatter->GetFormatSpecialInfo( nFormat, bThousand, bIsRed, nPrec, nLeading ); switch( pFormatter->GetType( nFormat ) ) { case css::util::NumberFormat::NUMBER: if(bThousand) rFmtStr = ","; else rFmtStr = "F"; break; case css::util::NumberFormat::CURRENCY: rFmtStr = "C"; break; case css::util::NumberFormat::SCIENTIFIC: rFmtStr = "S"; break; case css::util::NumberFormat::PERCENT: rFmtStr = "P"; break; default: { bAppendPrec = false; switch( pFormatter->GetIndexTableOffset( nFormat ) ) { case NF_DATE_SYSTEM_SHORT: case NF_DATE_SYS_DMMMYY: case NF_DATE_SYS_DDMMYY: case NF_DATE_SYS_DDMMYYYY: case NF_DATE_SYS_DMMMYYYY: case NF_DATE_DIN_DMMMYYYY: case NF_DATE_SYS_DMMMMYYYY: case NF_DATE_DIN_DMMMMYYYY: rFmtStr = "D1"; break; case NF_DATE_SYS_DDMMM: rFmtStr = "D2"; break; case NF_DATE_SYS_MMYY: rFmtStr = "D3"; break; case NF_DATETIME_SYSTEM_SHORT_HHMM: case NF_DATETIME_SYS_DDMMYYYY_HHMMSS: rFmtStr = "D4"; break; case NF_DATE_DIN_MMDD: rFmtStr = "D5"; break; case NF_TIME_HHMMSSAMPM: rFmtStr = "D6"; break; case NF_TIME_HHMMAMPM: rFmtStr = "D7"; break; case NF_TIME_HHMMSS: rFmtStr = "D8"; break; case NF_TIME_HHMM: rFmtStr = "D9"; break; default: rFmtStr = "G"; } } } if( bAppendPrec ) rFmtStr += OUString::number(nPrec); const SvNumberformat* pFormat = pFormatter->GetEntry( nFormat ); if( lcl_FormatHasNegColor( pFormat ) ) rFmtStr += "-"; if( lcl_FormatHasOpenPar( pFormat ) ) rFmtStr += "()"; } } void ScInterpreter::ScCell() { // ATTRIBUTE ; [REF] sal_uInt8 nParamCount = GetByte(); if( MustHaveParamCount( nParamCount, 1, 2 ) ) { ScAddress aCellPos( aPos ); bool bError = false; if( nParamCount == 2 ) { switch (GetStackType()) { case svExternalSingleRef: case svExternalDoubleRef: { // Let's handle external reference separately... ScCellExternal(); return; } default: ; } bError = !PopDoubleRefOrSingleRef( aCellPos ); } OUString aInfoType = GetString().getString(); if( bError || nGlobalError ) PushIllegalParameter(); else { ScRefCellValue aCell(*pDok, aCellPos); ScCellKeywordTranslator::transKeyword(aInfoType, ScGlobal::GetLocale(), ocCell); // *** ADDRESS INFO *** if( aInfoType == "COL" ) { // column number (1-based) PushInt( aCellPos.Col() + 1 ); } else if( aInfoType == "ROW" ) { // row number (1-based) PushInt( aCellPos.Row() + 1 ); } else if( aInfoType == "SHEET" ) { // table number (1-based) PushInt( aCellPos.Tab() + 1 ); } else if( aInfoType == "ADDRESS" ) { // address formatted as [['FILENAME'#]$TABLE.]$COL$ROW ScRefFlags nFlags = (aCellPos.Tab() == aPos.Tab()) ? (ScRefFlags::ADDR_ABS) : (ScRefFlags::ADDR_ABS_3D); OUString aStr(aCellPos.Format(nFlags, pDok, pDok->GetAddressConvention())); PushString(aStr); } else if( aInfoType == "FILENAME" ) { // file name and table name: 'FILENAME'#$TABLE SCTAB nTab = aCellPos.Tab(); OUString aFuncResult; if( nTab < pDok->GetTableCount() ) { if( pDok->GetLinkMode( nTab ) == ScLinkMode::VALUE ) pDok->GetName( nTab, aFuncResult ); else { SfxObjectShell* pShell = pDok->GetDocumentShell(); if( pShell && pShell->GetMedium() ) { OUStringBuffer aBuf; aBuf.append('\''); const INetURLObject& rURLObj = pShell->GetMedium()->GetURLObject(); aBuf.append(rURLObj.GetMainURL(INetURLObject::DECODE_UNAMBIGUOUS)); aBuf.append("'#$"); OUString aTabName; pDok->GetName( nTab, aTabName ); aBuf.append(aTabName); aFuncResult = aBuf.makeStringAndClear(); } } } PushString( aFuncResult ); } else if( aInfoType == "COORD" ) { // address, lotus 1-2-3 formatted: $TABLE:$COL$ROW // Yes, passing tab as col is intentional! OUStringBuffer aFuncResult; OUString aCellStr = ScAddress( static_cast(aCellPos.Tab()), 0, 0 ).Format( (ScRefFlags::COL_ABS|ScRefFlags::COL_VALID), nullptr, pDok->GetAddressConvention() ); aFuncResult.append(aCellStr); aFuncResult.append(':'); aCellStr = aCellPos.Format((ScRefFlags::COL_ABS|ScRefFlags::COL_VALID|ScRefFlags::ROW_ABS|ScRefFlags::ROW_VALID), nullptr, pDok->GetAddressConvention()); aFuncResult.append(aCellStr); PushString( aFuncResult.makeStringAndClear() ); } // *** CELL PROPERTIES *** else if( aInfoType == "CONTENTS" ) { // contents of the cell, no formatting if (aCell.hasString()) { svl::SharedString aStr; GetCellString(aStr, aCell); PushString( aStr ); } else PushDouble(GetCellValue(aCellPos, aCell)); } else if( aInfoType == "TYPE" ) { // b = blank; l = string (label); v = otherwise (value) sal_Unicode c; if (aCell.hasString()) c = 'l'; else c = aCell.hasNumeric() ? 'v' : 'b'; PushString( OUString(c) ); } else if( aInfoType == "WIDTH" ) { // column width (rounded off as count of zero characters in standard font and size) Printer* pPrinter = pDok->GetPrinter(); MapMode aOldMode( pPrinter->GetMapMode() ); vcl::Font aOldFont( pPrinter->GetFont() ); vcl::Font aDefFont; pPrinter->SetMapMode( MAP_TWIP ); // font color doesn't matter here pDok->GetDefPattern()->GetFont( aDefFont, SC_AUTOCOL_BLACK, pPrinter ); pPrinter->SetFont( aDefFont ); long nZeroWidth = pPrinter->GetTextWidth( OUString( '0' ) ); pPrinter->SetFont( aOldFont ); pPrinter->SetMapMode( aOldMode ); int nZeroCount = (int)(pDok->GetColWidth( aCellPos.Col(), aCellPos.Tab() ) / nZeroWidth); PushInt( nZeroCount ); } else if( aInfoType == "PREFIX" ) { // ' = left; " = right; ^ = centered sal_Unicode c = 0; if (aCell.hasString()) { const SvxHorJustifyItem* pJustAttr = static_cast( pDok->GetAttr( aCellPos.Col(), aCellPos.Row(), aCellPos.Tab(), ATTR_HOR_JUSTIFY )); switch( pJustAttr->GetValue() ) { case SVX_HOR_JUSTIFY_STANDARD: case SVX_HOR_JUSTIFY_LEFT: case SVX_HOR_JUSTIFY_BLOCK: c = '\''; break; case SVX_HOR_JUSTIFY_CENTER: c = '^'; break; case SVX_HOR_JUSTIFY_RIGHT: c = '"'; break; case SVX_HOR_JUSTIFY_REPEAT: c = '\\'; break; } } PushString( OUString(c) ); } else if( aInfoType == "PROTECT" ) { // 1 = cell locked const ScProtectionAttr* pProtAttr = static_cast( pDok->GetAttr( aCellPos.Col(), aCellPos.Row(), aCellPos.Tab(), ATTR_PROTECTION )); PushInt( pProtAttr->GetProtection() ? 1 : 0 ); } // *** FORMATTING *** else if( aInfoType == "FORMAT" ) { // specific format code for standard formats OUString aFuncResult; sal_uLong nFormat = pDok->GetNumberFormat( aCellPos ); getFormatString(pFormatter, nFormat, aFuncResult); PushString( aFuncResult ); } else if( aInfoType == "COLOR" ) { // 1 = negative values are colored, otherwise 0 const SvNumberformat* pFormat = pFormatter->GetEntry( pDok->GetNumberFormat( aCellPos ) ); PushInt( lcl_FormatHasNegColor( pFormat ) ? 1 : 0 ); } else if( aInfoType == "PARENTHESES" ) { // 1 = format string contains a '(' character, otherwise 0 const SvNumberformat* pFormat = pFormatter->GetEntry( pDok->GetNumberFormat( aCellPos ) ); PushInt( lcl_FormatHasOpenPar( pFormat ) ? 1 : 0 ); } else PushIllegalArgument(); } } } void ScInterpreter::ScCellExternal() { sal_uInt16 nFileId; OUString aTabName; ScSingleRefData aRef; ScExternalRefCache::TokenRef pToken; ScExternalRefCache::CellFormat aFmt; PopExternalSingleRef(nFileId, aTabName, aRef, pToken, &aFmt); if (nGlobalError) { PushError( nGlobalError); return; } OUString aInfoType = GetString().getString(); if (nGlobalError) { PushError( nGlobalError); return; } SCCOL nCol; SCROW nRow; SCTAB nTab; aRef.SetAbsTab(0); // external ref has a tab index of -1, which SingleRefToVars() don't like. SingleRefToVars(aRef, nCol, nRow, nTab); if (nGlobalError) { PushIllegalParameter(); return; } aRef.SetAbsTab(-1); // revert the value. ScCellKeywordTranslator::transKeyword(aInfoType, ScGlobal::GetLocale(), ocCell); ScExternalRefManager* pRefMgr = pDok->GetExternalRefManager(); if ( aInfoType == "COL" ) PushInt(nCol + 1); else if ( aInfoType == "ROW" ) PushInt(nRow + 1); else if ( aInfoType == "SHEET" ) { // For SHEET, No idea what number we should set, but let's always set // 1 if the external sheet exists, no matter what sheet. Excel does // the same. if (pRefMgr->getCacheTable(nFileId, aTabName, false).get()) PushInt(1); else SetError(errNoName); } else if ( aInfoType == "ADDRESS" ) { // ODF 1.2 says we need to always display address using the ODF A1 grammar. ScTokenArray aArray; aArray.AddExternalSingleReference(nFileId, svl::SharedString( aTabName), aRef); // string not interned ScCompiler aComp(pDok, aPos, aArray); aComp.SetGrammar(formula::FormulaGrammar::GRAM_ODFF_A1); OUString aStr; aComp.CreateStringFromTokenArray(aStr); PushString(aStr); } else if ( aInfoType == "FILENAME" ) { // 'file URI'#$SheetName const OUString* p = pRefMgr->getExternalFileName(nFileId); if (!p) { // In theory this should never happen... SetError(errNoName); return; } OUStringBuffer aBuf; aBuf.append('\''); aBuf.append(*p); aBuf.append("'#$"); aBuf.append(aTabName); PushString(aBuf.makeStringAndClear()); } else if ( aInfoType == "CONTENTS" ) { switch (pToken->GetType()) { case svString: PushString(pToken->GetString()); break; case svDouble: PushString(OUString::number(pToken->GetDouble())); break; case svError: PushString(ScGlobal::GetErrorString(pToken->GetError())); break; default: PushString(ScGlobal::GetEmptyOUString()); } } else if ( aInfoType == "TYPE" ) { sal_Unicode c = 'v'; switch (pToken->GetType()) { case svString: c = 'l'; break; case svEmptyCell: c = 'b'; break; default: ; } PushString(OUString(c)); } else if ( aInfoType == "FORMAT" ) { OUString aFmtStr; sal_uLong nFmt = aFmt.mbIsSet ? aFmt.mnIndex : 0; getFormatString(pFormatter, nFmt, aFmtStr); PushString(aFmtStr); } else if ( aInfoType == "COLOR" ) { // 1 = negative values are colored, otherwise 0 int nVal = 0; if (aFmt.mbIsSet) { const SvNumberformat* pFormat = pFormatter->GetEntry(aFmt.mnIndex); nVal = lcl_FormatHasNegColor(pFormat) ? 1 : 0; } PushInt(nVal); } else if ( aInfoType == "PARENTHESES" ) { // 1 = format string contains a '(' character, otherwise 0 int nVal = 0; if (aFmt.mbIsSet) { const SvNumberformat* pFormat = pFormatter->GetEntry(aFmt.mnIndex); nVal = lcl_FormatHasOpenPar(pFormat) ? 1 : 0; } PushInt(nVal); } else PushIllegalParameter(); } void ScInterpreter::ScIsRef() { nFuncFmtType = css::util::NumberFormat::LOGICAL; bool bRes = false; switch ( GetStackType() ) { case svSingleRef : { ScAddress aAdr; PopSingleRef( aAdr ); if ( !nGlobalError ) bRes = true; } break; case svDoubleRef : { ScRange aRange; PopDoubleRef( aRange ); if ( !nGlobalError ) bRes = true; } break; case svRefList : { FormulaTokenRef x = PopToken(); if ( !nGlobalError ) bRes = !x.get()->GetRefList()->empty(); } break; case svExternalSingleRef: { ScExternalRefCache::TokenRef pToken; PopExternalSingleRef(pToken); if (!nGlobalError) bRes = true; } break; case svExternalDoubleRef: { ScExternalRefCache::TokenArrayRef pArray; PopExternalDoubleRef(pArray); if (!nGlobalError) bRes = true; } break; default: Pop(); } nGlobalError = 0; PushInt( int(bRes) ); } void ScInterpreter::ScIsValue() { nFuncFmtType = css::util::NumberFormat::LOGICAL; bool bRes = false; switch ( GetRawStackType() ) { case svDouble: Pop(); bRes = true; break; case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) break; ScRefCellValue aCell(*pDok, aAdr); if (GetCellErrCode(aCell) == 0) { switch (aCell.meType) { case CELLTYPE_VALUE : bRes = true; break; case CELLTYPE_FORMULA : bRes = (aCell.mpFormula->IsValue() && !aCell.mpFormula->IsEmpty()); break; default: ; // nothing } } } break; case svExternalSingleRef: { ScExternalRefCache::TokenRef pToken; PopExternalSingleRef(pToken); if (!nGlobalError && pToken->GetType() == svDouble) bRes = true; } break; case svExternalDoubleRef: case svMatrix: { ScMatrixRef pMat = GetMatrix(); if ( !pMat ) ; // nothing else if ( !pJumpMatrix ) { if (pMat->GetErrorIfNotString( 0, 0) == 0) bRes = pMat->IsValue( 0, 0); } else { SCSIZE nCols, nRows, nC, nR; pMat->GetDimensions( nCols, nRows); pJumpMatrix->GetPos( nC, nR); if ( nC < nCols && nR < nRows ) if (pMat->GetErrorIfNotString( nC, nR) == 0) bRes = pMat->IsValue( nC, nR); } } break; default: Pop(); } nGlobalError = 0; PushInt( int(bRes) ); } void ScInterpreter::ScIsFormula() { nFuncFmtType = css::util::NumberFormat::LOGICAL; bool bRes = false; switch ( GetStackType() ) { case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) break; bRes = (pDok->GetCellType(aAdr) == CELLTYPE_FORMULA); } break; default: Pop(); } nGlobalError = 0; PushInt( int(bRes) ); } void ScInterpreter::ScFormula() { OUString aFormula; switch ( GetStackType() ) { case svDoubleRef : if (bMatrixFormula) { SCCOL nCol1, nCol2; SCROW nRow1, nRow2; SCTAB nTab1, nTab2; PopDoubleRef( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); if (nGlobalError) break; if (nTab1 != nTab2) { SetError( errIllegalArgument); break; } ScMatrixRef pResMat = GetNewMat( nCol2 - nCol1 + 1, nRow2 - nRow1 + 1, true); if (!pResMat) break; /* TODO: use a column iterator instead? */ SCSIZE i=0, j=0; ScAddress aAdr(0,0,nTab1); for (SCCOL nCol = nCol1; nCol <= nCol2; ++nCol) { aAdr.SetCol(nCol); for (SCROW nRow = nRow1; nRow <= nRow2; ++nRow) { aAdr.SetRow(nRow); ScRefCellValue aCell(*pDok, aAdr); switch (aCell.meType) { case CELLTYPE_FORMULA : aCell.mpFormula->GetFormula(aFormula); pResMat->PutString( mrStrPool.intern( aFormula), i,j); break; default: pResMat->PutError( NOTAVAILABLE, i,j); } ++j; } ++i; j = 0; } PushMatrix( pResMat); return; } SAL_FALLTHROUGH; case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) break; ScRefCellValue aCell(*pDok, aAdr); switch (aCell.meType) { case CELLTYPE_FORMULA : aCell.mpFormula->GetFormula(aFormula); break; default: SetError( NOTAVAILABLE ); } } break; default: Pop(); SetError( NOTAVAILABLE ); } PushString( aFormula ); } void ScInterpreter::ScIsNV() { nFuncFmtType = css::util::NumberFormat::LOGICAL; bool bRes = false; switch ( GetStackType() ) { case svDoubleRef : case svSingleRef : { ScAddress aAdr; bool bOk = PopDoubleRefOrSingleRef( aAdr ); if ( nGlobalError == NOTAVAILABLE ) bRes = true; else if (bOk) { ScRefCellValue aCell(*pDok, aAdr); sal_uInt16 nErr = GetCellErrCode(aCell); bRes = (nErr == NOTAVAILABLE); } } break; case svExternalSingleRef: { ScExternalRefCache::TokenRef pToken; PopExternalSingleRef(pToken); if (nGlobalError == NOTAVAILABLE || (pToken && pToken->GetType() == svError && pToken->GetError() == NOTAVAILABLE)) bRes = true; } break; case svExternalDoubleRef: case svMatrix: { ScMatrixRef pMat = GetMatrix(); if ( !pMat ) ; // nothing else if ( !pJumpMatrix ) bRes = (pMat->GetErrorIfNotString( 0, 0) == NOTAVAILABLE); else { SCSIZE nCols, nRows, nC, nR; pMat->GetDimensions( nCols, nRows); pJumpMatrix->GetPos( nC, nR); if ( nC < nCols && nR < nRows ) bRes = (pMat->GetErrorIfNotString( nC, nR) == NOTAVAILABLE); } } break; default: PopError(); if ( nGlobalError == NOTAVAILABLE ) bRes = true; } nGlobalError = 0; PushInt( int(bRes) ); } void ScInterpreter::ScIsErr() { nFuncFmtType = css::util::NumberFormat::LOGICAL; bool bRes = false; switch ( GetStackType() ) { case svDoubleRef : case svSingleRef : { ScAddress aAdr; bool bOk = PopDoubleRefOrSingleRef( aAdr ); if ( !bOk || (nGlobalError && nGlobalError != NOTAVAILABLE) ) bRes = true; else { ScRefCellValue aCell(*pDok, aAdr); sal_uInt16 nErr = GetCellErrCode(aCell); bRes = (nErr && nErr != NOTAVAILABLE); } } break; case svExternalSingleRef: { ScExternalRefCache::TokenRef pToken; PopExternalSingleRef(pToken); if ((nGlobalError && nGlobalError != NOTAVAILABLE) || !pToken || (pToken->GetType() == svError && pToken->GetError() != NOTAVAILABLE)) bRes = true; } break; case svExternalDoubleRef: case svMatrix: { ScMatrixRef pMat = GetMatrix(); if ( nGlobalError || !pMat ) bRes = ((nGlobalError && nGlobalError != NOTAVAILABLE) || !pMat); else if ( !pJumpMatrix ) { sal_uInt16 nErr = pMat->GetErrorIfNotString( 0, 0); bRes = (nErr && nErr != NOTAVAILABLE); } else { SCSIZE nCols, nRows, nC, nR; pMat->GetDimensions( nCols, nRows); pJumpMatrix->GetPos( nC, nR); if ( nC < nCols && nR < nRows ) { sal_uInt16 nErr = pMat->GetErrorIfNotString( nC, nR); bRes = (nErr && nErr != NOTAVAILABLE); } } } break; default: PopError(); if ( nGlobalError && nGlobalError != NOTAVAILABLE ) bRes = true; } nGlobalError = 0; PushInt( int(bRes) ); } void ScInterpreter::ScIsError() { nFuncFmtType = css::util::NumberFormat::LOGICAL; bool bRes = false; switch ( GetStackType() ) { case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) { bRes = true; break; } if ( nGlobalError ) bRes = true; else { ScRefCellValue aCell(*pDok, aAdr); bRes = (GetCellErrCode(aCell) != 0); } } break; case svExternalSingleRef: { ScExternalRefCache::TokenRef pToken; PopExternalSingleRef(pToken); if (nGlobalError || pToken->GetType() == svError) bRes = true; } break; case svExternalDoubleRef: case svMatrix: { ScMatrixRef pMat = GetMatrix(); if ( nGlobalError || !pMat ) bRes = true; else if ( !pJumpMatrix ) bRes = (pMat->GetErrorIfNotString( 0, 0) != 0); else { SCSIZE nCols, nRows, nC, nR; pMat->GetDimensions( nCols, nRows); pJumpMatrix->GetPos( nC, nR); if ( nC < nCols && nR < nRows ) bRes = (pMat->GetErrorIfNotString( nC, nR) != 0); } } break; default: PopError(); if ( nGlobalError ) bRes = true; } nGlobalError = 0; PushInt( int(bRes) ); } bool ScInterpreter::IsEven() { nFuncFmtType = css::util::NumberFormat::LOGICAL; bool bRes = false; double fVal = 0.0; switch ( GetStackType() ) { case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) break; ScRefCellValue aCell(*pDok, aAdr); sal_uInt16 nErr = GetCellErrCode(aCell); if (nErr != 0) SetError(nErr); else { switch (aCell.meType) { case CELLTYPE_VALUE : fVal = GetCellValue(aAdr, aCell); bRes = true; break; case CELLTYPE_FORMULA : if (aCell.mpFormula->IsValue()) { fVal = GetCellValue(aAdr, aCell); bRes = true; } break; default: ; // nothing } } } break; case svDouble: { fVal = PopDouble(); bRes = true; } break; case svExternalSingleRef: { ScExternalRefCache::TokenRef pToken; PopExternalSingleRef(pToken); if (!nGlobalError && pToken->GetType() == svDouble) { fVal = pToken->GetDouble(); bRes = true; } } break; case svExternalDoubleRef: case svMatrix: { ScMatrixRef pMat = GetMatrix(); if ( !pMat ) ; // nothing else if ( !pJumpMatrix ) { bRes = pMat->IsValue( 0, 0); if ( bRes ) fVal = pMat->GetDouble( 0, 0); } else { SCSIZE nCols, nRows, nC, nR; pMat->GetDimensions( nCols, nRows); pJumpMatrix->GetPos( nC, nR); if ( nC < nCols && nR < nRows ) { bRes = pMat->IsValue( nC, nR); if ( bRes ) fVal = pMat->GetDouble( nC, nR); } else SetError( errNoValue); } } break; default: ; // nothing } if ( !bRes ) SetError( errIllegalParameter); else bRes = ( fmod( ::rtl::math::approxFloor( fabs( fVal ) ), 2.0 ) < 0.5 ); return bRes; } void ScInterpreter::ScIsEven() { PushInt( int(IsEven()) ); } void ScInterpreter::ScIsOdd() { PushInt( int(!IsEven()) ); } void ScInterpreter::ScN() { sal_uInt16 nErr = nGlobalError; nGlobalError = 0; // Temporarily override the ConvertStringToValue() error for // GetCellValue() / GetCellValueOrZero() sal_uInt16 nSErr = mnStringNoValueError; mnStringNoValueError = errCellNoValue; double fVal = GetDouble(); mnStringNoValueError = nSErr; if (nErr) nGlobalError = nErr; // preserve previous error if any else if (nGlobalError == errCellNoValue) nGlobalError = 0; // reset temporary detection error PushDouble(fVal); } void ScInterpreter::ScTrim() { // Doesn't only trim but also removes duplicated blanks within! OUString aVal = comphelper::string::strip(GetString().getString(), ' '); OUStringBuffer aStr; const sal_Unicode* p = aVal.getStr(); const sal_Unicode* const pEnd = p + aVal.getLength(); while ( p < pEnd ) { if ( *p != ' ' || p[-1] != ' ' ) // first can't be ' ', so -1 is fine aStr.append(*p); p++; } PushString(aStr.makeStringAndClear()); } void ScInterpreter::ScUpper() { OUString aString = ScGlobal::pCharClass->uppercase(GetString().getString()); PushString(aString); } void ScInterpreter::ScProper() { //2do: what to do with I18N-CJK ?!? OUStringBuffer aStr(GetString().getString()); const sal_Int32 nLen = aStr.getLength(); if ( nLen > 0 ) { OUString aUpr(ScGlobal::pCharClass->uppercase(aStr.toString())); OUString aLwr(ScGlobal::pCharClass->lowercase(aStr.toString())); aStr[0] = aUpr[0]; sal_Int32 nPos = 1; while( nPos < nLen ) { OUString aTmpStr( aStr[nPos-1] ); if ( !ScGlobal::pCharClass->isLetter( aTmpStr, 0 ) ) aStr[nPos] = aUpr[nPos]; else aStr[nPos] = aLwr[nPos]; ++nPos; } } PushString(aStr.makeStringAndClear()); } void ScInterpreter::ScLower() { OUString aString = ScGlobal::pCharClass->lowercase(GetString().getString()); PushString(aString); } void ScInterpreter::ScLen() { PushDouble(GetString().getLength()); } void ScInterpreter::ScT() { switch ( GetStackType() ) { case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) { PushInt(0); return ; } bool bValue = false; ScRefCellValue aCell(*pDok, aAdr); if (GetCellErrCode(aCell) == 0) { switch (aCell.meType) { case CELLTYPE_VALUE : bValue = true; break; case CELLTYPE_FORMULA : bValue = aCell.mpFormula->IsValue(); break; default: ; // nothing } } if ( bValue ) PushString(EMPTY_OUSTRING); else { // like GetString() svl::SharedString aStr; GetCellString(aStr, aCell); PushString(aStr); } } break; case svMatrix: case svExternalSingleRef: case svExternalDoubleRef: { double fVal; svl::SharedString aStr; ScMatValType nMatValType = GetDoubleOrStringFromMatrix( fVal, aStr); if (ScMatrix::IsValueType( nMatValType)) PushString(svl::SharedString::getEmptyString()); else PushString( aStr); } break; case svDouble : { PopError(); PushString( EMPTY_OUSTRING ); } break; case svString : ; // leave on stack break; default : PushError( errUnknownOpCode); } } void ScInterpreter::ScValue() { OUString aInputString; double fVal; switch ( GetRawStackType() ) { case svMissing: case svEmptyCell: Pop(); PushInt(0); return; case svDouble: return; // leave on stack case svSingleRef: case svDoubleRef: { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) { PushInt(0); return; } ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasString()) { svl::SharedString aSS; GetCellString(aSS, aCell); aInputString = aSS.getString(); } else if (aCell.hasNumeric()) { PushDouble( GetCellValue(aAdr, aCell) ); return; } else { PushDouble(0.0); return; } } break; case svMatrix: { svl::SharedString aSS; ScMatValType nType = GetDoubleOrStringFromMatrix( fVal, aSS); aInputString = aSS.getString(); switch (nType) { case SC_MATVAL_EMPTY: fVal = 0.0; SAL_FALLTHROUGH; case SC_MATVAL_VALUE: case SC_MATVAL_BOOLEAN: PushDouble( fVal); return; case SC_MATVAL_STRING: // evaluated below break; default: PushIllegalArgument(); } } break; default: aInputString = GetString().getString(); break; } sal_uInt32 nFIndex = 0; // 0 for default locale if (pFormatter->IsNumberFormat(aInputString, nFIndex, fVal)) PushDouble(fVal); else PushIllegalArgument(); } // fdo#57180 void ScInterpreter::ScNumberValue() { sal_uInt8 nParamCount = GetByte(); if ( !MustHaveParamCount( nParamCount, 1, 3 ) ) return; OUString aInputString; OUString aDecimalSeparator, aGroupSeparator; sal_Unicode cDecimalSeparator = 0; if ( nParamCount == 3 ) aGroupSeparator = GetString().getString(); if ( nParamCount >= 2 ) { aDecimalSeparator = GetString().getString(); if ( aDecimalSeparator.getLength() == 1 ) cDecimalSeparator = aDecimalSeparator[ 0 ]; else { PushIllegalArgument(); //if given, separator length must be 1 return; } } if ( cDecimalSeparator && aGroupSeparator.indexOf( cDecimalSeparator ) != -1 ) { PushIllegalArgument(); //decimal separator cannot appear in group separator return; } switch (GetStackType()) { case svDouble: return; // leave on stack default: aInputString = GetString().getString(); } if ( nGlobalError ) { PushError( nGlobalError ); return; } if ( aInputString.isEmpty() ) { if ( maCalcConfig.mbEmptyStringAsZero ) PushDouble( 0.0 ); else PushNoValue(); return; } sal_Int32 nDecSep = aInputString.indexOf( cDecimalSeparator ); if ( nDecSep != 0 ) { OUString aTemporary( nDecSep >= 0 ? aInputString.copy( 0, nDecSep ) : aInputString ); sal_Int32 nIndex = 0; while (nIndex < aGroupSeparator.getLength()) { sal_uInt32 nChar = aGroupSeparator.iterateCodePoints( &nIndex ); aTemporary = aTemporary.replaceAll( OUString( &nChar, 1 ), "" ); } if ( nDecSep >= 0 ) aInputString = aTemporary + aInputString.copy( nDecSep ); else aInputString = aTemporary; } for ( sal_Int32 i = aInputString.getLength(); --i >= 0; ) { sal_Unicode c = aInputString[ i ]; if ( c == 0x0020 || c == 0x0009 || c == 0x000A || c == 0x000D ) aInputString = aInputString.replaceAt( i, 1, "" ); // remove spaces etc. } sal_Int32 nPercentCount = 0; for ( sal_Int32 i = aInputString.getLength() - 1; i >= 0 && aInputString[ i ] == 0x0025; i-- ) { aInputString = aInputString.replaceAt( i, 1, "" ); // remove and count trailing '%' nPercentCount++; } rtl_math_ConversionStatus eStatus; sal_Int32 nParseEnd; double fVal = ::rtl::math::stringToDouble( aInputString, cDecimalSeparator, 0, &eStatus, &nParseEnd ); if ( eStatus == rtl_math_ConversionStatus_Ok && nParseEnd == aInputString.getLength() ) { if (nPercentCount) fVal *= pow( 10.0, -(nPercentCount * 2)); // process '%' from input string PushDouble(fVal); return; } PushNoValue(); } //2do: this should be a proper unicode string method static inline bool lcl_ScInterpreter_IsPrintable( sal_Unicode c ) { return 0x20 <= c && c != 0x7f; } void ScInterpreter::ScClean() { OUString aStr = GetString().getString(); for ( sal_Int32 i = 0; i < aStr.getLength(); i++ ) { if ( !lcl_ScInterpreter_IsPrintable( aStr[i] ) ) aStr = aStr.replaceAt(i,1,""); } PushString(aStr); } void ScInterpreter::ScCode() { //2do: make it full range unicode? OUString aStr = GetString().getString(); if (aStr.isEmpty()) PushInt(0); else { //"classic" ByteString conversion flags const sal_uInt32 convertFlags = RTL_UNICODETOTEXT_FLAGS_NONSPACING_IGNORE | RTL_UNICODETOTEXT_FLAGS_CONTROL_IGNORE | RTL_UNICODETOTEXT_FLAGS_FLUSH | RTL_UNICODETOTEXT_FLAGS_UNDEFINED_DEFAULT | RTL_UNICODETOTEXT_FLAGS_INVALID_DEFAULT | RTL_UNICODETOTEXT_FLAGS_UNDEFINED_REPLACE; PushInt( (unsigned char) OUStringToOString(OUString(aStr[0]), osl_getThreadTextEncoding(), convertFlags).toChar() ); } } void ScInterpreter::ScChar() { //2do: make it full range unicode? double fVal = GetDouble(); if (fVal < 0.0 || fVal >= 256.0) PushIllegalArgument(); else { //"classic" ByteString conversion flags const sal_uInt32 convertFlags = RTL_TEXTTOUNICODE_FLAGS_UNDEFINED_DEFAULT | RTL_TEXTTOUNICODE_FLAGS_MBUNDEFINED_DEFAULT | RTL_TEXTTOUNICODE_FLAGS_INVALID_DEFAULT; sal_Char cEncodedChar = static_cast(fVal); OUString aStr(&cEncodedChar, 1, osl_getThreadTextEncoding(), convertFlags); PushString(aStr); } } /* #i70213# fullwidth/halfwidth conversion provided by * Takashi Nakamoto * erAck: added Excel compatibility conversions as seen in issue's test case. */ static OUString lcl_convertIntoHalfWidth( const OUString & rStr ) { static bool bFirstASCCall = true; static utl::TransliterationWrapper aTrans( ::comphelper::getProcessComponentContext(), 0 ); if( bFirstASCCall ) { aTrans.loadModuleByImplName( "FULLWIDTH_HALFWIDTH_LIKE_ASC", LANGUAGE_SYSTEM ); bFirstASCCall = false; } return aTrans.transliterate( rStr, 0, sal_uInt16( rStr.getLength() ), nullptr ); } static OUString lcl_convertIntoFullWidth( const OUString & rStr ) { static bool bFirstJISCall = true; static utl::TransliterationWrapper aTrans( ::comphelper::getProcessComponentContext(), 0 ); if( bFirstJISCall ) { aTrans.loadModuleByImplName( "HALFWIDTH_FULLWIDTH_LIKE_JIS", LANGUAGE_SYSTEM ); bFirstJISCall = false; } return aTrans.transliterate( rStr, 0, sal_uInt16( rStr.getLength() ), nullptr ); } /* ODFF: * Summary: Converts half-width to full-width ASCII and katakana characters. * Semantics: Conversion is done for half-width ASCII and katakana characters, * other characters are simply copied from T to the result. This is the * complementary function to ASC. * For references regarding halfwidth and fullwidth characters see * http://www.unicode.org/reports/tr11/ * http://www.unicode.org/charts/charindex2.html#H * http://www.unicode.org/charts/charindex2.html#F */ void ScInterpreter::ScJis() { if (MustHaveParamCount( GetByte(), 1)) PushString( lcl_convertIntoFullWidth( GetString().getString())); } /* ODFF: * Summary: Converts full-width to half-width ASCII and katakana characters. * Semantics: Conversion is done for full-width ASCII and katakana characters, * other characters are simply copied from T to the result. This is the * complementary function to JIS. */ void ScInterpreter::ScAsc() { if (MustHaveParamCount( GetByte(), 1)) PushString( lcl_convertIntoHalfWidth( GetString().getString())); } void ScInterpreter::ScUnicode() { if ( MustHaveParamCount( GetByte(), 1 ) ) { OUString aStr = GetString().getString(); if (aStr.isEmpty()) PushIllegalParameter(); else { sal_Int32 i = 0; PushDouble(aStr.iterateCodePoints(&i)); } } } void ScInterpreter::ScUnichar() { if ( MustHaveParamCount( GetByte(), 1 ) ) { sal_uInt32 nCodePoint = GetUInt32(); if (nGlobalError || !rtl::isUnicodeCodePoint(nCodePoint)) PushIllegalArgument(); else { OUString aStr( &nCodePoint, 1 ); PushString( aStr ); } } } void ScInterpreter::ScMin( bool bTextAsZero ) { short nParamCount = GetByte(); if (!MustHaveParamCountMin( nParamCount, 1)) return; double nMin = ::std::numeric_limits::max(); double nVal = 0.0; ScAddress aAdr; ScRange aRange; size_t nRefInList = 0; while (nParamCount-- > 0) { switch (GetStackType()) { case svDouble : { nVal = GetDouble(); if (nMin > nVal) nMin = nVal; nFuncFmtType = css::util::NumberFormat::NUMBER; } break; case svSingleRef : { PopSingleRef( aAdr ); ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasNumeric()) { nVal = GetCellValue(aAdr, aCell); CurFmtToFuncFmt(); if (nMin > nVal) nMin = nVal; } else if (bTextAsZero && aCell.hasString()) { if ( nMin > 0.0 ) nMin = 0.0; } } break; case svDoubleRef : case svRefList : { sal_uInt16 nErr = 0; PopDoubleRef( aRange, nParamCount, nRefInList); ScValueIterator aValIter( pDok, aRange, mnSubTotalFlags, bTextAsZero ); if (aValIter.GetFirst(nVal, nErr)) { if (nMin > nVal) nMin = nVal; aValIter.GetCurNumFmtInfo( nFuncFmtType, nFuncFmtIndex ); while ((nErr == 0) && aValIter.GetNext(nVal, nErr)) { if (nMin > nVal) nMin = nVal; } SetError(nErr); } } break; case svMatrix : case svExternalSingleRef: case svExternalDoubleRef: { ScMatrixRef pMat = GetMatrix(); if (pMat) { nFuncFmtType = css::util::NumberFormat::NUMBER; nVal = pMat->GetMinValue(bTextAsZero); if (nMin > nVal) nMin = nVal; } } break; case svString : { Pop(); if ( bTextAsZero ) { if ( nMin > 0.0 ) nMin = 0.0; } else SetError(errIllegalParameter); } break; default : Pop(); SetError(errIllegalParameter); } } if ( nVal < nMin ) PushDouble(0.0); else PushDouble(nMin); } void ScInterpreter::ScMax( bool bTextAsZero ) { short nParamCount = GetByte(); if (!MustHaveParamCountMin( nParamCount, 1)) return; double nMax = -(::std::numeric_limits::max()); double nVal = 0.0; ScAddress aAdr; ScRange aRange; size_t nRefInList = 0; while (nParamCount-- > 0) { switch (GetStackType()) { case svDouble : { nVal = GetDouble(); if (nMax < nVal) nMax = nVal; nFuncFmtType = css::util::NumberFormat::NUMBER; } break; case svSingleRef : { PopSingleRef( aAdr ); ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasNumeric()) { nVal = GetCellValue(aAdr, aCell); CurFmtToFuncFmt(); if (nMax < nVal) nMax = nVal; } else if (bTextAsZero && aCell.hasString()) { if ( nMax < 0.0 ) nMax = 0.0; } } break; case svDoubleRef : case svRefList : { sal_uInt16 nErr = 0; PopDoubleRef( aRange, nParamCount, nRefInList); ScValueIterator aValIter( pDok, aRange, mnSubTotalFlags, bTextAsZero ); if (aValIter.GetFirst(nVal, nErr)) { if (nMax < nVal) nMax = nVal; aValIter.GetCurNumFmtInfo( nFuncFmtType, nFuncFmtIndex ); while ((nErr == 0) && aValIter.GetNext(nVal, nErr)) { if (nMax < nVal) nMax = nVal; } SetError(nErr); } } break; case svMatrix : case svExternalSingleRef: case svExternalDoubleRef: { ScMatrixRef pMat = GetMatrix(); if (pMat) { nFuncFmtType = css::util::NumberFormat::NUMBER; nVal = pMat->GetMaxValue(bTextAsZero); if (nMax < nVal) nMax = nVal; } } break; case svString : { Pop(); if ( bTextAsZero ) { if ( nMax < 0.0 ) nMax = 0.0; } else SetError(errIllegalParameter); } break; default : Pop(); SetError(errIllegalParameter); } } if ( nVal > nMax ) PushDouble(0.0); else PushDouble(nMax); } void ScInterpreter::GetStVarParams( double& rVal, double& rValCount, bool bTextAsZero ) { short nParamCount = GetByte(); std::vector values; double fSum = 0.0; double vSum = 0.0; double vMean = 0.0; double fVal = 0.0; rValCount = 0.0; ScAddress aAdr; ScRange aRange; size_t nRefInList = 0; while (nParamCount-- > 0) { switch (GetStackType()) { case svDouble : { fVal = GetDouble(); if ( nGlobalError ) return; values.push_back(fVal); fSum += fVal; rValCount++; } break; case svSingleRef : { PopSingleRef( aAdr ); ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasNumeric()) { fVal = GetCellValue(aAdr, aCell); if ( nGlobalError ) return; values.push_back(fVal); fSum += fVal; rValCount++; } else if (bTextAsZero && aCell.hasString()) { values.push_back(0.0); rValCount++; } } break; case svDoubleRef : case svRefList : { sal_uInt16 nErr = 0; PopDoubleRef( aRange, nParamCount, nRefInList); ScValueIterator aValIter( pDok, aRange, mnSubTotalFlags, bTextAsZero ); if (aValIter.GetFirst(fVal, nErr)) { do { values.push_back(fVal); fSum += fVal; rValCount++; } while ((nErr == 0) && aValIter.GetNext(fVal, nErr)); } if ( nErr ) { SetError(nErr); return; } } break; case svMatrix : { ScMatrixRef pMat = PopMatrix(); if (pMat) { SCSIZE nC, nR; pMat->GetDimensions(nC, nR); for (SCSIZE nMatCol = 0; nMatCol < nC; nMatCol++) { for (SCSIZE nMatRow = 0; nMatRow < nR; nMatRow++) { if (!pMat->IsString(nMatCol,nMatRow)) { fVal= pMat->GetDouble(nMatCol,nMatRow); values.push_back(fVal); fSum += fVal; rValCount++; } else if ( bTextAsZero ) { values.push_back(0.0); rValCount++; } } } } } break; case svString : { Pop(); if ( bTextAsZero ) { values.push_back(0.0); rValCount++; } else SetError(errIllegalParameter); } break; default : Pop(); SetError(errIllegalParameter); } } ::std::vector::size_type n = values.size(); vMean = fSum / n; for (::std::vector::size_type i = 0; i < n; i++) vSum += ::rtl::math::approxSub( values[i], vMean) * ::rtl::math::approxSub( values[i], vMean); rVal = vSum; } void ScInterpreter::ScVar( bool bTextAsZero ) { double nVal; double nValCount; GetStVarParams( nVal, nValCount, bTextAsZero ); if (nValCount <= 1.0) PushError( errDivisionByZero ); else PushDouble( nVal / (nValCount - 1.0)); } void ScInterpreter::ScVarP( bool bTextAsZero ) { double nVal; double nValCount; GetStVarParams( nVal, nValCount, bTextAsZero ); PushDouble( div( nVal, nValCount)); } void ScInterpreter::ScStDev( bool bTextAsZero ) { double nVal; double nValCount; GetStVarParams( nVal, nValCount, bTextAsZero ); if (nValCount <= 1.0) PushError( errDivisionByZero ); else PushDouble( sqrt( nVal / (nValCount - 1.0))); } void ScInterpreter::ScStDevP( bool bTextAsZero ) { double nVal; double nValCount; GetStVarParams( nVal, nValCount, bTextAsZero ); if (nValCount == 0.0) PushError( errDivisionByZero ); else PushDouble( sqrt( nVal / nValCount)); /* this was: PushDouble( sqrt( div( nVal, nValCount))); * * Besides that the special NAN gets lost in the call through sqrt(), * unxlngi6.pro then looped back and forth somewhere between div() and * ::rtl::math::setNan(). Tests showed that * * sqrt( div( 1, 0)); * * produced a loop, but * * double f1 = div( 1, 0); * sqrt( f1 ); * * was fine. There seems to be some compiler optimization problem. It does * not occur when compiled with debug=t. */ } void ScInterpreter::ScColumns() { sal_uInt8 nParamCount = GetByte(); sal_uLong nVal = 0; SCCOL nCol1; SCROW nRow1; SCTAB nTab1; SCCOL nCol2; SCROW nRow2; SCTAB nTab2; while (nParamCount-- > 0) { switch ( GetStackType() ) { case svSingleRef: PopError(); nVal++; break; case svDoubleRef: PopDoubleRef(nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); nVal += static_cast(nTab2 - nTab1 + 1) * static_cast(nCol2 - nCol1 + 1); break; case svMatrix: { ScMatrixRef pMat = PopMatrix(); if (pMat) { SCSIZE nC, nR; pMat->GetDimensions(nC, nR); nVal += nC; } } break; case svExternalSingleRef: PopError(); nVal++; break; case svExternalDoubleRef: { sal_uInt16 nFileId; OUString aTabName; ScComplexRefData aRef; PopExternalDoubleRef( nFileId, aTabName, aRef); ScRange aAbs = aRef.toAbs(aPos); nVal += static_cast(aAbs.aEnd.Tab() - aAbs.aStart.Tab() + 1) * static_cast(aAbs.aEnd.Col() - aAbs.aStart.Col() + 1); } break; default: PopError(); SetError(errIllegalParameter); } } PushDouble((double)nVal); } void ScInterpreter::ScRows() { sal_uInt8 nParamCount = GetByte(); sal_uLong nVal = 0; SCCOL nCol1; SCROW nRow1; SCTAB nTab1; SCCOL nCol2; SCROW nRow2; SCTAB nTab2; while (nParamCount-- > 0) { switch ( GetStackType() ) { case svSingleRef: PopError(); nVal++; break; case svDoubleRef: PopDoubleRef(nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); nVal += static_cast(nTab2 - nTab1 + 1) * static_cast(nRow2 - nRow1 + 1); break; case svMatrix: { ScMatrixRef pMat = PopMatrix(); if (pMat) { SCSIZE nC, nR; pMat->GetDimensions(nC, nR); nVal += nR; } } break; case svExternalSingleRef: PopError(); nVal++; break; case svExternalDoubleRef: { sal_uInt16 nFileId; OUString aTabName; ScComplexRefData aRef; PopExternalDoubleRef( nFileId, aTabName, aRef); ScRange aAbs = aRef.toAbs(aPos); nVal += static_cast(aAbs.aEnd.Tab() - aAbs.aStart.Tab() + 1) * static_cast(aAbs.aEnd.Row() - aAbs.aStart.Row() + 1); } break; default: PopError(); SetError(errIllegalParameter); } } PushDouble((double)nVal); } void ScInterpreter::ScSheets() { sal_uInt8 nParamCount = GetByte(); sal_uLong nVal; if ( nParamCount == 0 ) nVal = pDok->GetTableCount(); else { nVal = 0; SCCOL nCol1; SCROW nRow1; SCTAB nTab1; SCCOL nCol2; SCROW nRow2; SCTAB nTab2; while (nParamCount-- > 0) { switch ( GetStackType() ) { case svSingleRef: PopError(); nVal++; break; case svDoubleRef: PopDoubleRef(nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); nVal += static_cast(nTab2 - nTab1 + 1); break; case svMatrix: PopError(); nVal++; break; case svExternalSingleRef: PopError(); nVal++; break; case svExternalDoubleRef: { sal_uInt16 nFileId; OUString aTabName; ScComplexRefData aRef; PopExternalDoubleRef( nFileId, aTabName, aRef); ScRange aAbs = aRef.toAbs(aPos); nVal += static_cast(aAbs.aEnd.Tab() - aAbs.aStart.Tab() + 1); } break; default: PopError(); SetError( errIllegalParameter ); } } } PushDouble( (double) nVal ); } void ScInterpreter::ScColumn() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 0, 1 ) ) { double nVal = 0; if (nParamCount == 0) { nVal = aPos.Col() + 1; if (bMatrixFormula) { SCCOL nCols; SCROW nRows; pMyFormulaCell->GetMatColsRows( nCols, nRows); if (nCols == 0) { // Happens if called via ScViewFunc::EnterMatrix() // ScFormulaCell::GetResultDimensions() as of course a // matrix result is not available yet. nCols = 1; } ScMatrixRef pResMat = GetNewMat( static_cast(nCols), 1); if (pResMat) { for (SCCOL i=0; i < nCols; ++i) pResMat->PutDouble( nVal + i, static_cast(i), 0); PushMatrix( pResMat); return; } } } else { switch ( GetStackType() ) { case svSingleRef : { SCCOL nCol1; SCROW nRow1; SCTAB nTab1; PopSingleRef( nCol1, nRow1, nTab1 ); nVal = (double) (nCol1 + 1); } break; case svDoubleRef : { SCCOL nCol1; SCROW nRow1; SCTAB nTab1; SCCOL nCol2; SCROW nRow2; SCTAB nTab2; PopDoubleRef( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 ); if (nCol2 > nCol1) { ScMatrixRef pResMat = GetNewMat( static_cast(nCol2-nCol1+1), 1); if (pResMat) { for (SCCOL i = nCol1; i <= nCol2; i++) pResMat->PutDouble((double)(i+1), static_cast(i-nCol1), 0); PushMatrix(pResMat); return; } else nVal = 0.0; } else nVal = (double) (nCol1 + 1); } break; default: SetError( errIllegalParameter ); nVal = 0.0; } } PushDouble( nVal ); } } void ScInterpreter::ScRow() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 0, 1 ) ) { double nVal = 0; if (nParamCount == 0) { nVal = aPos.Row() + 1; if (bMatrixFormula) { SCCOL nCols; SCROW nRows; pMyFormulaCell->GetMatColsRows( nCols, nRows); if (nRows == 0) { // Happens if called via ScViewFunc::EnterMatrix() // ScFormulaCell::GetResultDimensions() as of course a // matrix result is not available yet. nRows = 1; } ScMatrixRef pResMat = GetNewMat( 1, static_cast(nRows)); if (pResMat) { for (SCROW i=0; i < nRows; i++) pResMat->PutDouble( nVal + i, 0, static_cast(i)); PushMatrix( pResMat); return; } } } else { switch ( GetStackType() ) { case svSingleRef : { SCCOL nCol1; SCROW nRow1; SCTAB nTab1; PopSingleRef( nCol1, nRow1, nTab1 ); nVal = (double) (nRow1 + 1); } break; case svDoubleRef : { SCCOL nCol1; SCROW nRow1; SCTAB nTab1; SCCOL nCol2; SCROW nRow2; SCTAB nTab2; PopDoubleRef( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 ); if (nRow2 > nRow1) { ScMatrixRef pResMat = GetNewMat( 1, static_cast(nRow2-nRow1+1)); if (pResMat) { for (SCROW i = nRow1; i <= nRow2; i++) pResMat->PutDouble((double)(i+1), 0, static_cast(i-nRow1)); PushMatrix(pResMat); return; } else nVal = 0.0; } else nVal = (double) (nRow1 + 1); } break; default: SetError( errIllegalParameter ); nVal = 0.0; } } PushDouble( nVal ); } } void ScInterpreter::ScSheet() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 0, 1 ) ) { SCTAB nVal = 0; if ( nParamCount == 0 ) nVal = aPos.Tab() + 1; else { switch ( GetStackType() ) { case svString : { svl::SharedString aStr = PopString(); if ( pDok->GetTable(aStr.getString(), nVal)) ++nVal; else SetError( errIllegalArgument ); } break; case svSingleRef : { SCCOL nCol1; SCROW nRow1; SCTAB nTab1; PopSingleRef( nCol1, nRow1, nTab1 ); nVal = nTab1 + 1; } break; case svDoubleRef : { SCCOL nCol1; SCROW nRow1; SCTAB nTab1; SCCOL nCol2; SCROW nRow2; SCTAB nTab2; PopDoubleRef( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 ); nVal = nTab1 + 1; } break; default: SetError( errIllegalParameter ); } if ( nGlobalError ) nVal = 0; } PushDouble( (double) nVal ); } } namespace { class VectorMatrixAccessor { public: VectorMatrixAccessor(const ScMatrix& rMat, bool bColVec) : mrMat(rMat), mbColVec(bColVec) {} bool IsEmpty(SCSIZE i) const { return mbColVec ? mrMat.IsEmpty(0, i) : mrMat.IsEmpty(i, 0); } bool IsEmptyPath(SCSIZE i) const { return mbColVec ? mrMat.IsEmptyPath(0, i) : mrMat.IsEmptyPath(i, 0); } bool IsValue(SCSIZE i) const { return mbColVec ? mrMat.IsValue(0, i) : mrMat.IsValue(i, 0); } bool IsString(SCSIZE i) const { return mbColVec ? mrMat.IsString(0, i) : mrMat.IsString(i, 0); } double GetDouble(SCSIZE i) const { return mbColVec ? mrMat.GetDouble(0, i) : mrMat.GetDouble(i, 0); } OUString GetString(SCSIZE i) const { return mbColVec ? mrMat.GetString(0, i).getString() : mrMat.GetString(i, 0).getString(); } SCSIZE GetElementCount() const { SCSIZE nC, nR; mrMat.GetDimensions(nC, nR); return mbColVec ? nR : nC; } private: const ScMatrix& mrMat; bool mbColVec; }; /** returns -1 when the matrix value is smaller than the query value, 0 when they are equal, and 1 when the matrix value is larger than the query value. */ sal_Int32 lcl_CompareMatrix2Query( SCSIZE i, const VectorMatrixAccessor& rMat, const ScQueryEntry& rEntry) { if (rMat.IsEmpty(i)) { /* TODO: in case we introduced query for real empty this would have to * be changed! */ return -1; // empty always less than anything else } /* FIXME: what is an empty path (result of IF(false;true_path) in * comparisons? */ bool bByString = rEntry.GetQueryItem().meType == ScQueryEntry::ByString; if (rMat.IsValue(i)) { if (bByString) return -1; // numeric always less than string const double nVal1 = rMat.GetDouble(i); const double nVal2 = rEntry.GetQueryItem().mfVal; if (nVal1 == nVal2) return 0; return nVal1 < nVal2 ? -1 : 1; } if (!bByString) return 1; // string always greater than numeric OUString aStr1 = rMat.GetString(i); OUString aStr2 = rEntry.GetQueryItem().maString.getString(); return ScGlobal::GetCollator()->compareString(aStr1, aStr2); // case-insensitive } /** returns the last item with the identical value as the original item value. */ void lcl_GetLastMatch( SCSIZE& rIndex, const VectorMatrixAccessor& rMat, SCSIZE nMatCount, bool bReverse) { if (rMat.IsValue(rIndex)) { double nVal = rMat.GetDouble(rIndex); if (bReverse) while (rIndex > 0 && rMat.IsValue(rIndex-1) && nVal == rMat.GetDouble(rIndex-1)) --rIndex; else while (rIndex < nMatCount-1 && rMat.IsValue(rIndex+1) && nVal == rMat.GetDouble(rIndex+1)) ++rIndex; } // Order of IsEmptyPath, IsEmpty, IsString is significant! else if (rMat.IsEmptyPath(rIndex)) { if (bReverse) while (rIndex > 0 && rMat.IsEmptyPath(rIndex-1)) --rIndex; else while (rIndex < nMatCount-1 && rMat.IsEmptyPath(rIndex+1)) ++rIndex; } else if (rMat.IsEmpty(rIndex)) { if (bReverse) while (rIndex > 0 && rMat.IsEmpty(rIndex-1)) --rIndex; else while (rIndex < nMatCount-1 && rMat.IsEmpty(rIndex+1)) ++rIndex; } else if (rMat.IsString(rIndex)) { OUString aStr( rMat.GetString(rIndex)); if (bReverse) while (rIndex > 0 && rMat.IsString(rIndex-1) && aStr == rMat.GetString(rIndex-1)) --rIndex; else while (rIndex < nMatCount-1 && rMat.IsString(rIndex+1) && aStr == rMat.GetString(rIndex+1)) ++rIndex; } else { OSL_FAIL("lcl_GetLastMatch: unhandled matrix type"); } } } void ScInterpreter::ScMatch() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 2, 3 ) ) { double fTyp; if (nParamCount == 3) fTyp = GetDouble(); else fTyp = 1.0; SCCOL nCol1 = 0; SCROW nRow1 = 0; SCTAB nTab1 = 0; SCCOL nCol2 = 0; SCROW nRow2 = 0; ScMatrixRef pMatSrc = nullptr; switch (GetStackType()) { case svSingleRef: PopSingleRef( nCol1, nRow1, nTab1); nCol2 = nCol1; nRow2 = nRow1; break; case svDoubleRef: { SCTAB nTab2 = 0; PopDoubleRef(nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); if (nTab1 != nTab2 || (nCol1 != nCol2 && nRow1 != nRow2)) { PushIllegalParameter(); return; } } break; case svMatrix: case svExternalDoubleRef: { if (GetStackType() == svMatrix) pMatSrc = PopMatrix(); else PopExternalDoubleRef(pMatSrc); if (!pMatSrc) { PushIllegalParameter(); return; } } break; default: PushIllegalParameter(); return; } if (nGlobalError == 0) { double fVal; ScQueryParam rParam; rParam.nCol1 = nCol1; rParam.nRow1 = nRow1; rParam.nCol2 = nCol2; rParam.nTab = nTab1; ScQueryEntry& rEntry = rParam.GetEntry(0); ScQueryEntry::Item& rItem = rEntry.GetQueryItem(); rEntry.bDoQuery = true; if (fTyp < 0.0) rEntry.eOp = SC_GREATER_EQUAL; else if (fTyp > 0.0) rEntry.eOp = SC_LESS_EQUAL; switch ( GetStackType() ) { case svDouble: { fVal = GetDouble(); rItem.mfVal = fVal; rItem.meType = ScQueryEntry::ByValue; } break; case svString: { rItem.meType = ScQueryEntry::ByString; rItem.maString = GetString(); } break; case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) { PushInt(0); return ; } ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasNumeric()) { fVal = GetCellValue(aAdr, aCell); rItem.meType = ScQueryEntry::ByValue; rItem.mfVal = fVal; } else { GetCellString(rItem.maString, aCell); rItem.meType = ScQueryEntry::ByString; } } break; case svExternalSingleRef: { ScExternalRefCache::TokenRef pToken; PopExternalSingleRef(pToken); if (nGlobalError) { PushError( nGlobalError); return; } if (pToken->GetType() == svDouble) { rItem.meType = ScQueryEntry::ByValue; rItem.mfVal = pToken->GetDouble(); } else { rItem.meType = ScQueryEntry::ByString; rItem.maString = pToken->GetString(); } } break; case svExternalDoubleRef: case svMatrix : { svl::SharedString aStr; ScMatValType nType = GetDoubleOrStringFromMatrix( rItem.mfVal, aStr); rItem.maString = aStr; rItem.meType = ScMatrix::IsNonValueType(nType) ? ScQueryEntry::ByString : ScQueryEntry::ByValue; } break; default: { PushIllegalParameter(); return; } } if (rItem.meType == ScQueryEntry::ByString) { bool bIsVBAMode = pDok->IsInVBAMode(); if ( bIsVBAMode ) rParam.eSearchType = utl::SearchParam::SRCH_WILDCARD; else rParam.eSearchType = DetectSearchType(rEntry.GetQueryItem().maString.getString(), pDok); } if (pMatSrc) // The source data is matrix array. { SCSIZE nC, nR; pMatSrc->GetDimensions( nC, nR); if (nC > 1 && nR > 1) { // The source matrix must be a vector. PushIllegalParameter(); return; } SCSIZE nMatCount = (nC == 1) ? nR : nC; VectorMatrixAccessor aMatAcc(*pMatSrc, nC == 1); // simple serial search for equality mode (source data doesn't // need to be sorted). if (rEntry.eOp == SC_EQUAL) { for (SCSIZE i = 0; i < nMatCount; ++i) { if (lcl_CompareMatrix2Query( i, aMatAcc, rEntry) == 0) { PushDouble(i+1); // found ! return; } } PushNA(); // not found return; } // binary search for non-equality mode (the source data is // assumed to be sorted). bool bAscOrder = (rEntry.eOp == SC_LESS_EQUAL); SCSIZE nFirst = 0, nLast = nMatCount-1, nHitIndex = 0; for (SCSIZE nLen = nLast-nFirst; nLen > 0; nLen = nLast-nFirst) { SCSIZE nMid = nFirst + nLen/2; sal_Int32 nCmp = lcl_CompareMatrix2Query( nMid, aMatAcc, rEntry); if (nCmp == 0) { // exact match. find the last item with the same value. lcl_GetLastMatch( nMid, aMatAcc, nMatCount, !bAscOrder); PushDouble( nMid+1); return; } if (nLen == 1) // first and last items are next to each other. { if (nCmp < 0) nHitIndex = bAscOrder ? nLast : nFirst; else nHitIndex = bAscOrder ? nFirst : nLast; break; } if (nCmp < 0) { if (bAscOrder) nFirst = nMid; else nLast = nMid; } else { if (bAscOrder) nLast = nMid; else nFirst = nMid; } } if (nHitIndex == nMatCount-1) // last item { sal_Int32 nCmp = lcl_CompareMatrix2Query( nHitIndex, aMatAcc, rEntry); if ((bAscOrder && nCmp <= 0) || (!bAscOrder && nCmp >= 0)) { // either the last item is an exact match or the real // hit is beyond the last item. PushDouble( nHitIndex+1); return; } } if (nHitIndex > 0) // valid hit must be 2nd item or higher { PushDouble( nHitIndex); // non-exact match return; } PushNA(); return; } SCCOLROW nDelta = 0; if (nCol1 == nCol2) { // search row in column rParam.nRow2 = nRow2; rEntry.nField = nCol1; ScAddress aResultPos( nCol1, nRow1, nTab1); if (!LookupQueryWithCache( aResultPos, rParam)) { PushNA(); return; } nDelta = aResultPos.Row() - nRow1; } else { // search column in row SCCOL nC; rParam.bByRow = false; rParam.nRow2 = nRow1; rEntry.nField = nCol1; ScQueryCellIterator aCellIter(pDok, nTab1, rParam, false); // Advance Entry.nField in Iterator if column changed aCellIter.SetAdvanceQueryParamEntryField( true ); if (fTyp == 0.0) { // EQUAL if ( aCellIter.GetFirst() ) nC = aCellIter.GetCol(); else { PushNA(); return; } } else { // <= or >= SCROW nR; if ( !aCellIter.FindEqualOrSortedLastInRange( nC, nR ) ) { PushNA(); return; } } nDelta = nC - nCol1; } PushDouble((double) (nDelta + 1)); } else PushIllegalParameter(); } } namespace { bool isCellContentEmpty( const ScRefCellValue& rCell ) { switch (rCell.meType) { case CELLTYPE_VALUE: case CELLTYPE_STRING: case CELLTYPE_EDIT: return false; case CELLTYPE_FORMULA: { // NOTE: Excel treats ="" in a referenced cell as blank in // COUNTBLANK() but not in ISBLANK(), which is inconsistent. // COUNTBLANK() tests the (display) result whereas ISBLANK() tests // the cell content. // ODFF allows both for COUNTBLANK(). // OOo and LibreOffice prior to 4.4 did not treat ="" as blank in // COUNTBLANK(), we now do for Excel interoperability. /* TODO: introduce yet another compatibility option? */ sc::FormulaResultValue aRes = rCell.mpFormula->GetResult(); if (aRes.meType != sc::FormulaResultValue::String) return false; if (!aRes.maString.isEmpty()) return false; } break; default: ; } return true; } } void ScInterpreter::ScCountEmptyCells() { if ( MustHaveParamCount( GetByte(), 1 ) ) { sal_uLong nMaxCount = 0, nCount = 0; switch (GetStackType()) { case svSingleRef : { nMaxCount = 1; ScAddress aAdr; PopSingleRef( aAdr ); ScRefCellValue aCell(*pDok, aAdr); if (!isCellContentEmpty(aCell)) nCount = 1; } break; case svDoubleRef : case svRefList : { ScRange aRange; short nParam = 1; size_t nRefInList = 0; while (nParam-- > 0) { PopDoubleRef( aRange, nParam, nRefInList); nMaxCount += static_cast(aRange.aEnd.Row() - aRange.aStart.Row() + 1) * static_cast(aRange.aEnd.Col() - aRange.aStart.Col() + 1) * static_cast(aRange.aEnd.Tab() - aRange.aStart.Tab() + 1); ScCellIterator aIter( pDok, aRange, mnSubTotalFlags); for (bool bHas = aIter.first(); bHas; bHas = aIter.next()) { const ScRefCellValue& rCell = aIter.getRefCellValue(); if (!isCellContentEmpty(rCell)) ++nCount; } } } break; default : SetError(errIllegalParameter); break; } PushDouble(nMaxCount - nCount); } } double ScInterpreter::IterateParametersIf( ScIterFuncIf eFunc ) { sal_uInt8 nParamCount = GetByte(); if ( !MustHaveParamCount( nParamCount, 2, 3 ) ) return 0; SCCOL nCol3 = 0; SCROW nRow3 = 0; SCTAB nTab3 = 0; ScMatrixRef pSumExtraMatrix; bool bSumExtraRange = (nParamCount == 3); if (bSumExtraRange) { // Save only the upperleft cell in case of cell range. The geometry // of the 3rd parameter is taken from the 1st parameter. switch ( GetStackType() ) { case svDoubleRef : { SCCOL nColJunk = 0; SCROW nRowJunk = 0; SCTAB nTabJunk = 0; PopDoubleRef( nCol3, nRow3, nTab3, nColJunk, nRowJunk, nTabJunk ); if ( nTabJunk != nTab3 ) { SetError( errIllegalParameter); return 0; } } break; case svSingleRef : PopSingleRef( nCol3, nRow3, nTab3 ); break; case svMatrix: pSumExtraMatrix = PopMatrix(); // nCol3, nRow3, nTab3 remain 0 break; case svExternalSingleRef: { pSumExtraMatrix = GetNewMat(1,1); ScExternalRefCache::TokenRef pToken; PopExternalSingleRef(pToken); if (nGlobalError) return 0; if (pToken->GetType() == svDouble) pSumExtraMatrix->PutDouble(pToken->GetDouble(), 0, 0); else pSumExtraMatrix->PutString(pToken->GetString(), 0, 0); } break; case svExternalDoubleRef: PopExternalDoubleRef(pSumExtraMatrix); break; default: SetError( errIllegalParameter); return 0; } } svl::SharedString aString; double fVal = 0.0; bool bIsString = true; switch ( GetStackType() ) { case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) return 0; ScRefCellValue aCell(*pDok, aAdr); switch (aCell.meType) { case CELLTYPE_VALUE : fVal = GetCellValue(aAdr, aCell); bIsString = false; break; case CELLTYPE_FORMULA : if (aCell.mpFormula->IsValue()) { fVal = GetCellValue(aAdr, aCell); bIsString = false; } else GetCellString(aString, aCell); break; case CELLTYPE_STRING : case CELLTYPE_EDIT : GetCellString(aString, aCell); break; default: fVal = 0.0; bIsString = false; } } break; case svString: aString = GetString(); break; case svMatrix : case svExternalDoubleRef: { ScMatValType nType = GetDoubleOrStringFromMatrix( fVal, aString); bIsString = ScMatrix::IsNonValueType( nType); } break; case svExternalSingleRef: { ScExternalRefCache::TokenRef pToken; PopExternalSingleRef(pToken); if (!nGlobalError) { if (pToken->GetType() == svDouble) { fVal = pToken->GetDouble(); bIsString = false; } else aString = pToken->GetString(); } } break; default: { fVal = GetDouble(); bIsString = false; } } double fSum = 0.0; double fMem = 0.0; double fRes = 0.0; double fCount = 0.0; bool bNull = true; short nParam = 1; size_t nRefInList = 0; while (nParam-- > 0) { SCCOL nCol1 = 0; SCROW nRow1 = 0; SCTAB nTab1 = 0; SCCOL nCol2 = 0; SCROW nRow2 = 0; SCTAB nTab2 = 0; ScMatrixRef pQueryMatrix; switch ( GetStackType() ) { case svRefList : if (bSumExtraRange) { SetError( errIllegalParameter); } else { ScRange aRange; PopDoubleRef( aRange, nParam, nRefInList); aRange.GetVars( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); } break; case svDoubleRef : PopDoubleRef( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 ); break; case svSingleRef : PopSingleRef( nCol1, nRow1, nTab1 ); nCol2 = nCol1; nRow2 = nRow1; nTab2 = nTab1; break; case svMatrix: case svExternalSingleRef: case svExternalDoubleRef: { pQueryMatrix = GetMatrix(); if (!pQueryMatrix) { SetError( errIllegalParameter); return 0; } nCol1 = 0; nRow1 = 0; nTab1 = 0; SCSIZE nC, nR; pQueryMatrix->GetDimensions( nC, nR); nCol2 = static_cast(nC - 1); nRow2 = static_cast(nR - 1); nTab2 = 0; } break; default: SetError( errIllegalParameter); } if ( nTab1 != nTab2 ) { SetError( errIllegalParameter); } if (bSumExtraRange) { // Take the range geometry of the 1st parameter and apply it to // the 3rd. If parts of the resulting range would point outside // the sheet, don't complain but silently ignore and simply cut // them away, this is what Xcl does :-/ // For the cut-away part we also don't need to determine the // criteria match, so shrink the source range accordingly, // instead of the result range. SCCOL nColDelta = nCol2 - nCol1; SCROW nRowDelta = nRow2 - nRow1; SCCOL nMaxCol; SCROW nMaxRow; if (pSumExtraMatrix) { SCSIZE nC, nR; pSumExtraMatrix->GetDimensions( nC, nR); nMaxCol = static_cast(nC - 1); nMaxRow = static_cast(nR - 1); } else { nMaxCol = MAXCOL; nMaxRow = MAXROW; } if (nCol3 + nColDelta > nMaxCol) { SCCOL nNewDelta = nMaxCol - nCol3; nCol2 = nCol1 + nNewDelta; } if (nRow3 + nRowDelta > nMaxRow) { SCROW nNewDelta = nMaxRow - nRow3; nRow2 = nRow1 + nNewDelta; } } else { nCol3 = nCol1; nRow3 = nRow1; nTab3 = nTab1; } if (nGlobalError == 0) { ScQueryParam rParam; rParam.nRow1 = nRow1; rParam.nRow2 = nRow2; ScQueryEntry& rEntry = rParam.GetEntry(0); ScQueryEntry::Item& rItem = rEntry.GetQueryItem(); rEntry.bDoQuery = true; if (!bIsString) { rItem.meType = ScQueryEntry::ByValue; rItem.mfVal = fVal; rEntry.eOp = SC_EQUAL; } else { rParam.FillInExcelSyntax(pDok->GetSharedStringPool(), aString.getString(), 0, pFormatter); if (rItem.meType == ScQueryEntry::ByString) rParam.eSearchType = DetectSearchType(rItem.maString.getString(), pDok); } ScAddress aAdr; aAdr.SetTab( nTab3 ); rParam.nCol1 = nCol1; rParam.nCol2 = nCol2; rEntry.nField = nCol1; SCsCOL nColDiff = nCol3 - nCol1; SCsROW nRowDiff = nRow3 - nRow1; if (pQueryMatrix) { // Never case-sensitive. sc::CompareOptions aOptions( pDok, rEntry, rParam.eSearchType); ScMatrixRef pResultMatrix = QueryMat( pQueryMatrix, aOptions); if (nGlobalError || !pResultMatrix) { SetError( errIllegalParameter); } if (pSumExtraMatrix) { for (SCCOL nCol = nCol1; nCol <= nCol2; ++nCol) { for (SCROW nRow = nRow1; nRow <= nRow2; ++nRow) { if (pResultMatrix->IsValue( nCol, nRow) && pResultMatrix->GetDouble( nCol, nRow)) { SCSIZE nC = nCol + nColDiff; SCSIZE nR = nRow + nRowDiff; if (pSumExtraMatrix->IsValue( nC, nR)) { fVal = pSumExtraMatrix->GetDouble( nC, nR); ++fCount; if ( bNull && fVal != 0.0 ) { bNull = false; fMem = fVal; } else fSum += fVal; } } } } } else { for (SCCOL nCol = nCol1; nCol <= nCol2; ++nCol) { for (SCROW nRow = nRow1; nRow <= nRow2; ++nRow) { if (pResultMatrix->GetDouble( nCol, nRow)) { aAdr.SetCol( nCol + nColDiff); aAdr.SetRow( nRow + nRowDiff); ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasNumeric()) { fVal = GetCellValue(aAdr, aCell); ++fCount; if ( bNull && fVal != 0.0 ) { bNull = false; fMem = fVal; } else fSum += fVal; } } } } } } else { ScQueryCellIterator aCellIter(pDok, nTab1, rParam, false); // Increment Entry.nField in iterator when switching to next column. aCellIter.SetAdvanceQueryParamEntryField( true ); if ( aCellIter.GetFirst() ) { if (pSumExtraMatrix) { do { SCSIZE nC = aCellIter.GetCol() + nColDiff; SCSIZE nR = aCellIter.GetRow() + nRowDiff; if (pSumExtraMatrix->IsValue( nC, nR)) { fVal = pSumExtraMatrix->GetDouble( nC, nR); ++fCount; if ( bNull && fVal != 0.0 ) { bNull = false; fMem = fVal; } else fSum += fVal; } } while ( aCellIter.GetNext() ); } else { do { aAdr.SetCol( aCellIter.GetCol() + nColDiff); aAdr.SetRow( aCellIter.GetRow() + nRowDiff); ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasNumeric()) { fVal = GetCellValue(aAdr, aCell); ++fCount; if ( bNull && fVal != 0.0 ) { bNull = false; fMem = fVal; } else fSum += fVal; } } while ( aCellIter.GetNext() ); } } } } else { SetError( errIllegalParameter); return 0; } } switch( eFunc ) { case ifSUMIF: fRes = ::rtl::math::approxAdd( fSum, fMem ); break; case ifAVERAGEIF: fRes = div( ::rtl::math::approxAdd( fSum, fMem ), fCount); break; } return fRes; } void ScInterpreter::ScSumIf() { PushDouble( IterateParametersIf( ifSUMIF)); } void ScInterpreter::ScAverageIf() { PushDouble( IterateParametersIf( ifAVERAGEIF)); } void ScInterpreter::ScCountIf() { if ( MustHaveParamCount( GetByte(), 2 ) ) { svl::SharedString aString; double fVal = 0.0; bool bIsString = true; switch ( GetStackType() ) { case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) { PushInt(0); return ; } ScRefCellValue aCell(*pDok, aAdr); switch (aCell.meType) { case CELLTYPE_VALUE : fVal = GetCellValue(aAdr, aCell); bIsString = false; break; case CELLTYPE_FORMULA : if (aCell.mpFormula->IsValue()) { fVal = GetCellValue(aAdr, aCell); bIsString = false; } else GetCellString(aString, aCell); break; case CELLTYPE_STRING : case CELLTYPE_EDIT : GetCellString(aString, aCell); break; default: fVal = 0.0; bIsString = false; } } break; case svMatrix: case svExternalSingleRef: case svExternalDoubleRef: { ScMatValType nType = GetDoubleOrStringFromMatrix(fVal, aString); bIsString = ScMatrix::IsNonValueType( nType); } break; case svString: aString = GetString(); break; default: { fVal = GetDouble(); bIsString = false; } } double fCount = 0.0; short nParam = 1; size_t nRefInList = 0; while (nParam-- > 0) { SCCOL nCol1 = 0; SCROW nRow1 = 0; SCTAB nTab1 = 0; SCCOL nCol2 = 0; SCROW nRow2 = 0; SCTAB nTab2 = 0; ScMatrixRef pQueryMatrix; switch ( GetStackType() ) { case svDoubleRef : case svRefList : { ScRange aRange; PopDoubleRef( aRange, nParam, nRefInList); aRange.GetVars( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); } break; case svSingleRef : PopSingleRef( nCol1, nRow1, nTab1 ); nCol2 = nCol1; nRow2 = nRow1; nTab2 = nTab1; break; case svMatrix: case svExternalSingleRef: case svExternalDoubleRef: { pQueryMatrix = GetMatrix(); if (!pQueryMatrix) { PushIllegalParameter(); return; } nCol1 = 0; nRow1 = 0; nTab1 = 0; SCSIZE nC, nR; pQueryMatrix->GetDimensions( nC, nR); nCol2 = static_cast(nC - 1); nRow2 = static_cast(nR - 1); nTab2 = 0; } break; default: PushIllegalParameter(); return ; } if ( nTab1 != nTab2 ) { PushIllegalParameter(); return; } if (nCol1 > nCol2) { PushIllegalParameter(); return; } if (nGlobalError == 0) { ScQueryParam rParam; rParam.nRow1 = nRow1; rParam.nRow2 = nRow2; ScQueryEntry& rEntry = rParam.GetEntry(0); ScQueryEntry::Item& rItem = rEntry.GetQueryItem(); rEntry.bDoQuery = true; if (!bIsString) { rItem.meType = ScQueryEntry::ByValue; rItem.mfVal = fVal; rEntry.eOp = SC_EQUAL; } else { rParam.FillInExcelSyntax(pDok->GetSharedStringPool(), aString.getString(), 0, pFormatter); if (rItem.meType == ScQueryEntry::ByString) rParam.eSearchType = DetectSearchType(rItem.maString.getString(), pDok); } rParam.nCol1 = nCol1; rParam.nCol2 = nCol2; rEntry.nField = nCol1; if (pQueryMatrix) { // Never case-sensitive. sc::CompareOptions aOptions( pDok, rEntry, rParam.eSearchType); ScMatrixRef pResultMatrix = QueryMat( pQueryMatrix, aOptions); if (nGlobalError || !pResultMatrix) { PushIllegalParameter(); return; } SCSIZE nSize = pResultMatrix->GetElementCount(); for (SCSIZE nIndex = 0; nIndex < nSize; ++nIndex) { if (pResultMatrix->IsValue( nIndex) && pResultMatrix->GetDouble( nIndex)) ++fCount; } } else { ScQueryCellIterator aCellIter(pDok, nTab1, rParam, false); // Keep Entry.nField in iterator on column change aCellIter.SetAdvanceQueryParamEntryField( true ); if ( aCellIter.GetFirst() ) { do { fCount++; } while ( aCellIter.GetNext() ); } } } else { PushIllegalParameter(); return; } } PushDouble(fCount); } } double ScInterpreter::IterateParametersIfs( ScIterFuncIfs eFunc ) { sal_uInt8 nParamCount = GetByte(); sal_uInt8 nQueryCount = nParamCount / 2; bool bCheck; if ( eFunc == ifCOUNTIFS ) bCheck = (nParamCount >= 2) && (nParamCount % 2 == 0); else bCheck = (nParamCount >= 3) && (nParamCount % 2 == 1); if ( !bCheck ) { SetError( errParameterExpected); } else { std::vector aResArray; size_t nRowSize = 0; size_t nColSize = 0; double fVal = 0.0; double fSum = 0.0; double fMem = 0.0; double fRes = 0.0; double fCount = 0.0; double fMin = std::numeric_limits::max(); double fMax = std::numeric_limits::min(); short nParam = 1; size_t nRefInList = 0; SCCOL nDimensionCols = 0; SCROW nDimensionRows = 0; while (nParamCount > 1 && !nGlobalError) { // take criteria svl::SharedString aString; fVal = 0.0; bool bIsString = true; switch ( GetStackType() ) { case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) return 0; ScRefCellValue aCell(*pDok, aAdr); switch (aCell.meType) { case CELLTYPE_VALUE : fVal = GetCellValue(aAdr, aCell); bIsString = false; break; case CELLTYPE_FORMULA : if (aCell.mpFormula->IsValue()) { fVal = GetCellValue(aAdr, aCell); bIsString = false; } else GetCellString(aString, aCell); break; case CELLTYPE_STRING : case CELLTYPE_EDIT : GetCellString(aString, aCell); break; default: fVal = 0.0; bIsString = false; } } break; case svString: aString = GetString(); break; case svMatrix : case svExternalDoubleRef: { ScMatValType nType = GetDoubleOrStringFromMatrix( fVal, aString); bIsString = ScMatrix::IsNonValueType( nType); } break; case svExternalSingleRef: { ScExternalRefCache::TokenRef pToken; PopExternalSingleRef(pToken); if (!nGlobalError) { if (pToken->GetType() == svDouble) { fVal = pToken->GetDouble(); bIsString = false; } else aString = pToken->GetString(); } } break; default: { fVal = GetDouble(); bIsString = false; } } if (nGlobalError) return 0; // and bail out, no need to evaluate other arguments // take range nParam = 1; nRefInList = 0; SCCOL nCol1 = 0; SCROW nRow1 = 0; SCTAB nTab1 = 0; SCCOL nCol2 = 0; SCROW nRow2 = 0; SCTAB nTab2 = 0; ScMatrixRef pQueryMatrix; switch ( GetStackType() ) { case svRefList : { ScRange aRange; PopDoubleRef( aRange, nParam, nRefInList); aRange.GetVars( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); } break; case svDoubleRef : PopDoubleRef( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2 ); break; case svSingleRef : PopSingleRef( nCol1, nRow1, nTab1 ); nCol2 = nCol1; nRow2 = nRow1; nTab2 = nTab1; break; case svMatrix: case svExternalSingleRef: case svExternalDoubleRef: { pQueryMatrix = PopMatrix(); if (!pQueryMatrix) { SetError( errIllegalParameter); return 0; } nCol1 = 0; nRow1 = 0; nTab1 = 0; SCSIZE nC, nR; pQueryMatrix->GetDimensions( nC, nR); nCol2 = static_cast(nC - 1); nRow2 = static_cast(nR - 1); nTab2 = 0; } break; default: SetError( errIllegalParameter); return 0; } if ( nTab1 != nTab2 ) { SetError( errIllegalArgument); return 0; } // All reference ranges must be of same dimension and size. if (!nDimensionCols) nDimensionCols = nCol2 - nCol1 + 1; if (!nDimensionRows) nDimensionRows = nRow2 - nRow1 + 1; if ((nDimensionCols != (nCol2 - nCol1 + 1)) || (nDimensionRows != (nRow2 - nRow1 + 1))) { SetError ( errIllegalArgument); return 0; } // recalculate matrix values if (nGlobalError) return 0; // initialize temporary result matrix if (aResArray.empty()) { nColSize = nCol2 - nCol1 + 1; nRowSize = nRow2 - nRow1 + 1; aResArray.resize(nColSize*nRowSize, 0); } ScQueryParam rParam; rParam.nRow1 = nRow1; rParam.nRow2 = nRow2; ScQueryEntry& rEntry = rParam.GetEntry(0); ScQueryEntry::Item& rItem = rEntry.GetQueryItem(); rEntry.bDoQuery = true; if (!bIsString) { rItem.meType = ScQueryEntry::ByValue; rItem.mfVal = fVal; rEntry.eOp = SC_EQUAL; } else { rParam.FillInExcelSyntax(pDok->GetSharedStringPool(), aString.getString(), 0, pFormatter); if (rItem.meType == ScQueryEntry::ByString) rParam.eSearchType = DetectSearchType(rItem.maString.getString(), pDok); } ScAddress aAdr; aAdr.SetTab( nTab1 ); rParam.nCol1 = nCol1; rParam.nCol2 = nCol2; rEntry.nField = nCol1; SCsCOL nColDiff = -nCol1; SCsROW nRowDiff = -nRow1; if (pQueryMatrix) { // Never case-sensitive. sc::CompareOptions aOptions( pDok, rEntry, rParam.eSearchType); ScMatrixRef pResultMatrix = QueryMat( pQueryMatrix, aOptions); if (nGlobalError || !pResultMatrix) { SetError( errIllegalParameter); return 0; } // result matrix is filled with boolean values. std::vector aResValues; pResultMatrix->GetDoubleArray(aResValues); if (aResArray.size() != aResValues.size()) { SetError( errIllegalParameter); return 0; } std::vector::iterator itRes = aResArray.begin(), itResEnd = aResArray.end(); std::vector::const_iterator itThisRes = aResValues.begin(); for (; itRes != itResEnd; ++itRes, ++itThisRes) *itRes += *itThisRes; } else { ScQueryCellIterator aCellIter(pDok, nTab1, rParam, false); // Increment Entry.nField in iterator when switching to next column. aCellIter.SetAdvanceQueryParamEntryField( true ); if ( aCellIter.GetFirst() ) { do { size_t nC = aCellIter.GetCol() + nColDiff; size_t nR = aCellIter.GetRow() + nRowDiff; ++aResArray[nC*nRowSize+nR]; } while ( aCellIter.GetNext() ); } } nParamCount -= 2; } if (nGlobalError) return 0; // bail out // main range - only for AVERAGEIFS, SUMIFS, MINIFS and MAXIFS if (nParamCount == 1) { nParam = 1; nRefInList = 0; bool bNull = true; SCCOL nMainCol1 = 0; SCROW nMainRow1 = 0; SCTAB nMainTab1 = 0; SCCOL nMainCol2 = 0; SCROW nMainRow2 = 0; SCTAB nMainTab2 = 0; ScMatrixRef pMainMatrix; switch ( GetStackType() ) { case svRefList : { ScRange aRange; PopDoubleRef( aRange, nParam, nRefInList); aRange.GetVars( nMainCol1, nMainRow1, nMainTab1, nMainCol2, nMainRow2, nMainTab2); } break; case svDoubleRef : PopDoubleRef( nMainCol1, nMainRow1, nMainTab1, nMainCol2, nMainRow2, nMainTab2 ); break; case svSingleRef : PopSingleRef( nMainCol1, nMainRow1, nMainTab1 ); nMainCol2 = nMainCol1; nMainRow2 = nMainRow1; nMainTab2 = nMainTab1; break; case svMatrix: case svExternalSingleRef: case svExternalDoubleRef: { pMainMatrix = PopMatrix(); if (!pMainMatrix) { SetError( errIllegalParameter); return 0; } nMainCol1 = 0; nMainRow1 = 0; nMainTab1 = 0; SCSIZE nC, nR; pMainMatrix->GetDimensions( nC, nR); nMainCol2 = static_cast(nC - 1); nMainRow2 = static_cast(nR - 1); nMainTab2 = 0; } break; default: SetError( errIllegalParameter); return 0; } if ( nMainTab1 != nMainTab2 ) { SetError( errIllegalArgument); return 0; } // All reference ranges must be of same dimension and size. if ((nDimensionCols != (nMainCol2 - nMainCol1 + 1)) || (nDimensionRows != (nMainRow2 - nMainRow1 + 1))) { SetError ( errIllegalArgument); return 0; } if (nGlobalError) return 0; // bail out // end-result calculation ScAddress aAdr; aAdr.SetTab( nMainTab1 ); if (pMainMatrix) { std::vector aMainValues; pMainMatrix->GetDoubleArray(aMainValues, false); // Map empty values to NaN's. if (aResArray.size() != aMainValues.size()) { SetError( errIllegalArgument); return 0; } std::vector::const_iterator itRes = aResArray.begin(), itResEnd = aResArray.end(); std::vector::const_iterator itMain = aMainValues.begin(); for (; itRes != itResEnd; ++itRes, ++itMain) { if (*itRes != nQueryCount) continue; fVal = *itMain; if (GetDoubleErrorValue(fVal) == errElementNaN) continue; ++fCount; if (bNull && fVal != 0.0) { bNull = false; fMem = fVal; } else fSum += fVal; if ( fMin > fVal ) fMin = fVal; if ( fMax < fVal ) fMax = fVal; } } else { std::vector::const_iterator itRes = aResArray.begin(); for (size_t nCol = 0; nCol < nColSize; ++nCol) { for (size_t nRow = 0; nRow < nRowSize; ++nRow, ++itRes) { if (*itRes == nQueryCount) { aAdr.SetCol( static_cast(nCol) + nMainCol1); aAdr.SetRow( static_cast(nRow) + nMainRow1); ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasNumeric()) { fVal = GetCellValue(aAdr, aCell); ++fCount; if ( bNull && fVal != 0.0 ) { bNull = false; fMem = fVal; } else fSum += fVal; if ( fMin > fVal ) fMin = fVal; if ( fMax < fVal ) fMax = fVal; } } } } } } else { std::vector::const_iterator itRes = aResArray.begin(), itResEnd = aResArray.end(); for (; itRes != itResEnd; ++itRes) if (*itRes == nQueryCount) ++fCount; } switch( eFunc ) { case ifSUMIFS: fRes = ::rtl::math::approxAdd( fSum, fMem ); break; case ifAVERAGEIFS: fRes = div( ::rtl::math::approxAdd( fSum, fMem ), fCount); break; case ifCOUNTIFS: fRes = fCount; break; case ifMINIFS: fRes = ( fMin < std::numeric_limits::max() ? fMin : 0 ); break; case ifMAXIFS: fRes = ( fMax > std::numeric_limits::min() ? fMax : 0 ); break; default: ; // nothing } return fRes; } return 0; } void ScInterpreter::ScSumIfs() { PushDouble( IterateParametersIfs( ifSUMIFS)); } void ScInterpreter::ScAverageIfs() { PushDouble( IterateParametersIfs( ifAVERAGEIFS)); } void ScInterpreter::ScCountIfs() { PushDouble( IterateParametersIfs( ifCOUNTIFS)); } void ScInterpreter::ScMinIfs_MS() { PushDouble( IterateParametersIfs( ifMINIFS ) ); } void ScInterpreter::ScMaxIfs_MS() { PushDouble( IterateParametersIfs( ifMAXIFS ) ); } void ScInterpreter::ScLookup() { sal_uInt8 nParamCount = GetByte(); if ( !MustHaveParamCount( nParamCount, 2, 3 ) ) return ; ScMatrixRef pDataMat = nullptr, pResMat = nullptr; SCCOL nCol1 = 0, nCol2 = 0, nResCol1 = 0, nResCol2 = 0; SCROW nRow1 = 0, nRow2 = 0, nResRow1 = 0, nResRow2 = 0; SCTAB nTab1 = 0, nResTab = 0; SCSIZE nLenMajor = 0; // length of major direction bool bVertical = true; // whether to lookup vertically or horizontally // The third parameter, result array, for double, string and single reference. double fResVal = 0.0; svl::SharedString aResStr; ScAddress aResAdr; StackVar eResArrayType = svUnknown; if (nParamCount == 3) { eResArrayType = GetStackType(); switch (eResArrayType) { case svDoubleRef: { SCTAB nTabJunk; PopDoubleRef(nResCol1, nResRow1, nResTab, nResCol2, nResRow2, nTabJunk); if (nResTab != nTabJunk || ((nResRow2 - nResRow1) > 0 && (nResCol2 - nResCol1) > 0)) { // The result array must be a vector. PushIllegalParameter(); return; } } break; case svMatrix: case svExternalSingleRef: case svExternalDoubleRef: { pResMat = GetMatrix(); if (!pResMat) { PushIllegalParameter(); return; } SCSIZE nC, nR; pResMat->GetDimensions(nC, nR); if (nC != 1 && nR != 1) { // Result matrix must be a vector. PushIllegalParameter(); return; } } break; case svDouble: fResVal = GetDouble(); break; case svString: aResStr = GetString(); break; case svSingleRef: PopSingleRef( aResAdr ); break; default: PushIllegalParameter(); return; } } // For double, string and single reference. double fDataVal = 0.0; svl::SharedString aDataStr; ScAddress aDataAdr; bool bValueData = false; // Get the data-result range and also determine whether this is vertical // lookup or horizontal lookup. StackVar eDataArrayType = GetStackType(); switch (eDataArrayType) { case svDoubleRef: { SCTAB nTabJunk; PopDoubleRef(nCol1, nRow1, nTab1, nCol2, nRow2, nTabJunk); if (nTab1 != nTabJunk) { PushIllegalParameter(); return; } bVertical = (nRow2 - nRow1) >= (nCol2 - nCol1); nLenMajor = bVertical ? nRow2 - nRow1 + 1 : nCol2 - nCol1 + 1; } break; case svMatrix: case svExternalSingleRef: case svExternalDoubleRef: { pDataMat = GetMatrix(); if (!pDataMat) { PushIllegalParameter(); return; } SCSIZE nC, nR; pDataMat->GetDimensions(nC, nR); bVertical = (nR >= nC); nLenMajor = bVertical ? nR : nC; } break; case svDouble: { fDataVal = GetDouble(); bValueData = true; } break; case svString: { aDataStr = GetString(); } break; case svSingleRef: { PopSingleRef( aDataAdr ); ScRefCellValue aCell(*pDok, aDataAdr); if (aCell.hasEmptyValue()) { // Empty cells aren't found anywhere, bail out early. SetError( NOTAVAILABLE); } else if (aCell.hasNumeric()) { fDataVal = GetCellValue(aDataAdr, aCell); bValueData = true; } else GetCellString(aDataStr, aCell); } break; default: SetError( errIllegalParameter); } if (nGlobalError) { PushError( nGlobalError); return; } // Get the lookup value. ScQueryParam aParam; ScQueryEntry& rEntry = aParam.GetEntry(0); if ( !FillEntry(rEntry) ) return; if ( eDataArrayType == svDouble || eDataArrayType == svString || eDataArrayType == svSingleRef ) { // Delta position for a single value is always 0. // Found if data <= query, but not if query is string and found data is // numeric or vice versa. This is how Excel does it but doesn't // document it. bool bFound = false; ScQueryEntry::Item& rItem = rEntry.GetQueryItem(); if ( bValueData ) { if (rItem.meType == ScQueryEntry::ByString) bFound = false; else bFound = (fDataVal <= rItem.mfVal); } else { if (rItem.meType != ScQueryEntry::ByString) bFound = false; else bFound = (ScGlobal::GetCollator()->compareString(aDataStr.getString(), rItem.maString.getString()) <= 0); } if (!bFound) { PushNA(); return; } if (pResMat) { if (pResMat->IsValue( 0, 0 )) PushDouble(pResMat->GetDouble( 0, 0 )); else PushString(pResMat->GetString(0, 0)); } else if (nParamCount == 3) { switch (eResArrayType) { case svDouble: PushDouble( fResVal ); break; case svString: PushString( aResStr ); break; case svDoubleRef: aResAdr.Set( nResCol1, nResRow1, nResTab); SAL_FALLTHROUGH; case svSingleRef: PushCellResultToken( true, aResAdr, nullptr, nullptr); break; default: OSL_FAIL( "ScInterpreter::ScLookup: unhandled eResArrayType, single value data"); } } else { switch (eDataArrayType) { case svDouble: PushDouble( fDataVal ); break; case svString: PushString( aDataStr ); break; case svSingleRef: PushCellResultToken( true, aDataAdr, nullptr, nullptr); break; default: OSL_FAIL( "ScInterpreter::ScLookup: unhandled eDataArrayType, single value data"); } } return; } // Now, perform the search to compute the delta position (nDelta). if (pDataMat) { // Data array is given as a matrix. rEntry.bDoQuery = true; rEntry.eOp = SC_LESS_EQUAL; bool bFound = false; SCSIZE nC, nR; pDataMat->GetDimensions(nC, nR); // In case of non-vector matrix, only search the first row or column. ScMatrixRef pDataMat2; if (bVertical) { ScMatrixRef pTempMat = GetNewMat(1, nR); for (SCSIZE i = 0; i < nR; ++i) if (pDataMat->IsValue(0, i)) pTempMat->PutDouble(pDataMat->GetDouble(0, i), 0, i); else pTempMat->PutString(pDataMat->GetString(0, i), 0, i); pDataMat2 = pTempMat; } else { ScMatrixRef pTempMat = GetNewMat(nC, 1); for (SCSIZE i = 0; i < nC; ++i) if (pDataMat->IsValue(i, 0)) pTempMat->PutDouble(pDataMat->GetDouble(i, 0), i, 0); else pTempMat->PutString(pDataMat->GetString(i, 0), i, 0); pDataMat2 = pTempMat; } VectorMatrixAccessor aMatAcc2(*pDataMat2, bVertical); // binary search for non-equality mode (the source data is // assumed to be sorted in ascending order). SCCOLROW nDelta = -1; SCSIZE nFirst = 0, nLast = nLenMajor-1; //, nHitIndex = 0; for (SCSIZE nLen = nLast-nFirst; nLen > 0; nLen = nLast-nFirst) { SCSIZE nMid = nFirst + nLen/2; sal_Int32 nCmp = lcl_CompareMatrix2Query( nMid, aMatAcc2, rEntry); if (nCmp == 0) { // exact match. find the last item with the same value. lcl_GetLastMatch( nMid, aMatAcc2, nLenMajor, false); nDelta = nMid; bFound = true; break; } if (nLen == 1) // first and last items are next to each other. { nDelta = nCmp < 0 ? nLast - 1 : nFirst - 1; // If already the 1st item is greater there's nothing found. bFound = (nDelta >= 0); break; } if (nCmp < 0) nFirst = nMid; else nLast = nMid; } if (nDelta == static_cast(nLenMajor-2)) // last item { sal_Int32 nCmp = lcl_CompareMatrix2Query(nDelta+1, aMatAcc2, rEntry); if (nCmp <= 0) { // either the last item is an exact match or the real // hit is beyond the last item. nDelta += 1; bFound = true; } } else if (nDelta > 0) // valid hit must be 2nd item or higher { // non-exact match bFound = true; } // With 0-9 < A-Z, if query is numeric and data found is string, or // vice versa, the (yet another undocumented) Excel behavior is to // return #N/A instead. if (bFound) { VectorMatrixAccessor aMatAcc(*pDataMat, bVertical); SCCOLROW i = nDelta; SCSIZE n = aMatAcc.GetElementCount(); if (static_cast(i) >= n) i = static_cast(n); bool bByString = rEntry.GetQueryItem().meType == ScQueryEntry::ByString; if (bByString == aMatAcc.IsValue(i)) bFound = false; } if (!bFound) { PushNA(); return; } // Now that we've found the delta, push the result back to the cell. if (pResMat) { VectorMatrixAccessor aResMatAcc(*pResMat, bVertical); // result array is matrix. if (static_cast(nDelta) >= aResMatAcc.GetElementCount()) { PushNA(); return; } if (aResMatAcc.IsValue(nDelta)) PushDouble(aResMatAcc.GetDouble(nDelta)); else PushString(aResMatAcc.GetString(nDelta)); } else if (nParamCount == 3) { // result array is cell range. ScAddress aAdr; aAdr.SetTab(nResTab); bool bResVertical = (nResRow2 - nResRow1) > 0; if (bResVertical) { SCROW nTempRow = static_cast(nResRow1 + nDelta); if (nTempRow > MAXROW) { PushDouble(0); return; } aAdr.SetCol(nResCol1); aAdr.SetRow(nTempRow); } else { SCCOL nTempCol = static_cast(nResCol1 + nDelta); if (nTempCol > MAXCOL) { PushDouble(0); return; } aAdr.SetCol(nTempCol); aAdr.SetRow(nResRow1); } PushCellResultToken(true, aAdr, nullptr, nullptr); } else { // no result array. Use the data array to get the final value from. if (bVertical) { if (pDataMat->IsValue(nC-1, nDelta)) PushDouble(pDataMat->GetDouble(nC-1, nDelta)); else PushString(pDataMat->GetString(nC-1, nDelta)); } else { if (pDataMat->IsValue(nDelta, nR-1)) PushDouble(pDataMat->GetDouble(nDelta, nR-1)); else PushString(pDataMat->GetString(nDelta, nR-1)); } } return; } // Perform cell range search. aParam.nCol1 = nCol1; aParam.nRow1 = nRow1; aParam.nCol2 = bVertical ? nCol1 : nCol2; aParam.nRow2 = bVertical ? nRow2 : nRow1; aParam.bByRow = bVertical; rEntry.bDoQuery = true; rEntry.eOp = SC_LESS_EQUAL; rEntry.nField = nCol1; ScQueryEntry::Item& rItem = rEntry.GetQueryItem(); if (rItem.meType == ScQueryEntry::ByString) aParam.eSearchType = DetectSearchType(rItem.maString.getString(), pDok); ScQueryCellIterator aCellIter(pDok, nTab1, aParam, false); SCCOL nC; SCROW nR; // Advance Entry.nField in iterator upon switching columns if // lookup in row. aCellIter.SetAdvanceQueryParamEntryField(!bVertical); if ( !aCellIter.FindEqualOrSortedLastInRange(nC, nR) ) { PushNA(); return; } SCCOLROW nDelta = bVertical ? static_cast(nR-nRow1) : static_cast(nC-nCol1); if (pResMat) { VectorMatrixAccessor aResMatAcc(*pResMat, bVertical); // Use the matrix result array. if (aResMatAcc.IsValue(nDelta)) PushDouble(aResMatAcc.GetDouble(nDelta)); else PushString(aResMatAcc.GetString(nDelta)); } else if (nParamCount == 3) { switch (eResArrayType) { case svDoubleRef: { // Use the result array vector. Note that the result array is assumed // to be a vector (i.e. 1-dimensional array). ScAddress aAdr; aAdr.SetTab(nResTab); bool bResVertical = (nResRow2 - nResRow1) > 0; if (bResVertical) { SCROW nTempRow = static_cast(nResRow1 + nDelta); if (nTempRow > MAXROW) { PushDouble(0); return; } aAdr.SetCol(nResCol1); aAdr.SetRow(nTempRow); } else { SCCOL nTempCol = static_cast(nResCol1 + nDelta); if (nTempCol > MAXCOL) { PushDouble(0); return; } aAdr.SetCol(nTempCol); aAdr.SetRow(nResRow1); } PushCellResultToken( true, aAdr, nullptr, nullptr); } break; case svDouble: case svString: case svSingleRef: { if (nDelta != 0) PushNA(); else { switch (eResArrayType) { case svDouble: PushDouble( fResVal ); break; case svString: PushString( aResStr ); break; case svSingleRef: PushCellResultToken( true, aResAdr, nullptr, nullptr); break; default: ; // nothing } } } break; default: OSL_FAIL( "ScInterpreter::ScLookup: unhandled eResArrayType, range search"); } } else { // Regardless of whether or not the result array exists, the last // array is always used as the "result" array. ScAddress aAdr; aAdr.SetTab(nTab1); if (bVertical) { SCROW nTempRow = static_cast(nRow1 + nDelta); if (nTempRow > MAXROW) { PushDouble(0); return; } aAdr.SetCol(nCol2); aAdr.SetRow(nTempRow); } else { SCCOL nTempCol = static_cast(nCol1 + nDelta); if (nTempCol > MAXCOL) { PushDouble(0); return; } aAdr.SetCol(nTempCol); aAdr.SetRow(nRow2); } PushCellResultToken(true, aAdr, nullptr, nullptr); } } void ScInterpreter::ScHLookup() { CalculateLookup(true); } void ScInterpreter::CalculateLookup(bool bHLookup) { sal_uInt8 nParamCount = GetByte(); if (!MustHaveParamCount(nParamCount, 3, 4)) return; // Optional 4th argument to declare whether or not the range is sorted. bool bSorted = true; if (nParamCount == 4) bSorted = GetBool(); // Index of column to search. double fIndex = ::rtl::math::approxFloor( GetDouble() ) - 1.0; ScMatrixRef pMat = nullptr; SCSIZE nC = 0, nR = 0; SCCOL nCol1 = 0; SCROW nRow1 = 0; SCTAB nTab1 = 0; SCCOL nCol2 = 0; SCROW nRow2 = 0; SCTAB nTab2; StackVar eType = GetStackType(); if (eType == svDoubleRef) { PopDoubleRef(nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); if (nTab1 != nTab2) { PushIllegalParameter(); return; } } else if (eType == svSingleRef) { PopSingleRef(nCol1, nRow1, nTab1); nCol2 = nCol1; nRow2 = nRow1; } else if (eType == svMatrix || eType == svExternalDoubleRef || eType == svExternalSingleRef) { pMat = GetMatrix(); if (pMat) pMat->GetDimensions(nC, nR); else { PushIllegalParameter(); return; } } else { PushIllegalParameter(); return; } if ( fIndex < 0.0 || (bHLookup ? (pMat ? (fIndex >= nR) : (fIndex+nRow1 > nRow2)) : (pMat ? (fIndex >= nC) : (fIndex+nCol1 > nCol2)) ) ) { PushIllegalArgument(); return; } SCROW nZIndex = static_cast(fIndex); SCCOL nSpIndex = static_cast(fIndex); if (!pMat) { nZIndex += nRow1; // Wertzeile nSpIndex = sal::static_int_cast( nSpIndex + nCol1 ); // value column } if (nGlobalError) { PushIllegalParameter(); return; } ScQueryParam aParam; aParam.nCol1 = nCol1; aParam.nRow1 = nRow1; if ( bHLookup ) { aParam.nCol2 = nCol2; aParam.nRow2 = nRow1; // nur in der ersten Zeile suchen aParam.bByRow = false; } else { aParam.nCol2 = nCol1; // nur in der ersten Spalte suchen aParam.nRow2 = nRow2; aParam.nTab = nTab1; } ScQueryEntry& rEntry = aParam.GetEntry(0); rEntry.bDoQuery = true; if ( bSorted ) rEntry.eOp = SC_LESS_EQUAL; if ( !FillEntry(rEntry) ) return; ScQueryEntry::Item& rItem = rEntry.GetQueryItem(); if (rItem.meType == ScQueryEntry::ByString) aParam.eSearchType = DetectSearchType(rItem.maString.getString(), pDok); if (pMat) { SCSIZE nMatCount = bHLookup ? nC : nR; SCSIZE nDelta = SCSIZE_MAX; if (rItem.meType == ScQueryEntry::ByString) { //!!!!!!! //TODO: enable regex on matrix strings //!!!!!!! svl::SharedString aParamStr = rItem.maString; if ( bSorted ) { static CollatorWrapper* pCollator = ScGlobal::GetCollator(); for (SCSIZE i = 0; i < nMatCount; i++) { if (bHLookup ? pMat->IsString(i, 0) : pMat->IsString(0, i)) { sal_Int32 nRes = pCollator->compareString( bHLookup ? pMat->GetString(i,0).getString() : pMat->GetString(0,i).getString(), aParamStr.getString()); if (nRes <= 0) nDelta = i; else if (i>0) // #i2168# ignore first mismatch i = nMatCount+1; } else nDelta = i; } } else { if (bHLookup) { for (SCSIZE i = 0; i < nMatCount; i++) { if (pMat->IsString(i, 0)) { if (pMat->GetString(i,0).getDataIgnoreCase() == aParamStr.getDataIgnoreCase()) { nDelta = i; i = nMatCount + 1; } } } } else { nDelta = pMat->MatchStringInColumns(aParamStr, 0, 0); } } } else { if ( bSorted ) { // #i2168# ignore strings for (SCSIZE i = 0; i < nMatCount; i++) { if (!(bHLookup ? pMat->IsString(i, 0) : pMat->IsString(0, i))) { if ((bHLookup ? pMat->GetDouble(i,0) : pMat->GetDouble(0,i)) <= rItem.mfVal) nDelta = i; else i = nMatCount+1; } } } else { if (bHLookup) { for (SCSIZE i = 0; i < nMatCount; i++) { if (! pMat->IsString(i, 0) ) { if ( pMat->GetDouble(i,0) == rItem.mfVal) { nDelta = i; i = nMatCount + 1; } } } } else { nDelta = pMat->MatchDoubleInColumns(rItem.mfVal, 0, 0); } } } if ( nDelta != SCSIZE_MAX ) { SCSIZE nX = static_cast(nSpIndex); SCSIZE nY = nDelta; if ( bHLookup ) { nX = nDelta; nY = static_cast(nZIndex); } if ( pMat->IsString( nX, nY) ) PushString(pMat->GetString( nX,nY).getString()); else PushDouble(pMat->GetDouble( nX,nY)); } else PushNA(); } else { rEntry.nField = nCol1; bool bFound = false; SCCOL nCol = 0; SCROW nRow = 0; if ( bSorted ) rEntry.eOp = SC_LESS_EQUAL; if ( bHLookup ) { ScQueryCellIterator aCellIter(pDok, nTab1, aParam, false); // advance Entry.nField in Iterator upon switching columns aCellIter.SetAdvanceQueryParamEntryField( true ); if ( bSorted ) { SCROW nRow1_temp; bFound = aCellIter.FindEqualOrSortedLastInRange( nCol, nRow1_temp ); } else if ( aCellIter.GetFirst() ) { bFound = true; nCol = aCellIter.GetCol(); } nRow = nZIndex; } else { ScAddress aResultPos( nCol1, nRow1, nTab1); bFound = LookupQueryWithCache( aResultPos, aParam); nRow = aResultPos.Row(); nCol = nSpIndex; } if ( bFound ) { ScAddress aAdr( nCol, nRow, nTab1 ); PushCellResultToken( true, aAdr, nullptr, nullptr); } else PushNA(); } } bool ScInterpreter::FillEntry(ScQueryEntry& rEntry) { ScQueryEntry::Item& rItem = rEntry.GetQueryItem(); switch ( GetStackType() ) { case svDouble: { rItem.meType = ScQueryEntry::ByValue; rItem.mfVal = GetDouble(); } break; case svString: { rItem.meType = ScQueryEntry::ByString; rItem.maString = GetString(); } break; case svDoubleRef : case svSingleRef : { ScAddress aAdr; if ( !PopDoubleRefOrSingleRef( aAdr ) ) { PushInt(0); return false; } ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasNumeric()) { rItem.meType = ScQueryEntry::ByValue; rItem.mfVal = GetCellValue(aAdr, aCell); } else { GetCellString(rItem.maString, aCell); rItem.meType = ScQueryEntry::ByString; } } break; case svMatrix : { svl::SharedString aStr; const ScMatValType nType = GetDoubleOrStringFromMatrix(rItem.mfVal, aStr); rItem.maString = aStr; rItem.meType = ScMatrix::IsNonValueType(nType) ? ScQueryEntry::ByString : ScQueryEntry::ByValue; } break; default: { PushIllegalParameter(); return false; } } // switch ( GetStackType() ) return true; } void ScInterpreter::ScVLookup() { CalculateLookup(false); } void ScInterpreter::ScSubTotal() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCountMin( nParamCount, 2 ) ) { // We must fish the 1st parameter deep from the stack! And push it on top. const FormulaToken* p = pStack[ sp - nParamCount ]; PushTempToken( *p ); /* TODO: use FormulaTokenRef instead */ sal_Int32 nFunc = GetInt32(); mnSubTotalFlags |= SUBTOTAL_IGN_NESTED_ST_AG | SUBTOTAL_IGN_FILTERED; if (nFunc > 100) { // For opcodes 101 through 111, we need to skip hidden cells. // Other than that these opcodes are identical to 1 through 11. mnSubTotalFlags |= SUBTOTAL_IGN_HIDDEN; nFunc -= 100; } if ( nGlobalError || nFunc < 1 || nFunc > 11 ) PushIllegalArgument(); // simulate return on stack, not SetError(...) else { cPar = nParamCount - 1; switch( nFunc ) { case SUBTOTAL_FUNC_AVE : ScAverage(); break; case SUBTOTAL_FUNC_CNT : ScCount(); break; case SUBTOTAL_FUNC_CNT2 : ScCount2(); break; case SUBTOTAL_FUNC_MAX : ScMax(); break; case SUBTOTAL_FUNC_MIN : ScMin(); break; case SUBTOTAL_FUNC_PROD : ScProduct(); break; case SUBTOTAL_FUNC_STD : ScStDev(); break; case SUBTOTAL_FUNC_STDP : ScStDevP(); break; case SUBTOTAL_FUNC_SUM : ScSum(); break; case SUBTOTAL_FUNC_VAR : ScVar(); break; case SUBTOTAL_FUNC_VARP : ScVarP(); break; default : PushIllegalArgument(); break; } } mnSubTotalFlags = 0x00; // Get rid of the 1st (fished) parameter. double nVal = GetDouble(); Pop(); PushDouble( nVal ); } } void ScInterpreter::ScAggregate() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCountMin( nParamCount, 3 ) ) { // fish the 1st parameter from the stack and push it on top. const FormulaToken* p = pStack[ sp - nParamCount ]; PushTempToken( *p ); /* TODO: use FormulaTokenRef instead */ sal_Int32 nFunc = GetInt32(); // fish the 2nd parameter from the stack and push it on top. const FormulaToken* p2 = pStack[ sp - ( nParamCount - 1 ) ]; PushTempToken( *p2 ); /* TODO: use FormulaTokenRef instead */ sal_Int32 nOption = GetInt32(); if ( nGlobalError || nFunc < 1 || nFunc > 19 ) PushIllegalArgument(); else { switch ( nOption) { case 0 : // ignore nested SUBTOTAL and AGGREGATE functions mnSubTotalFlags = SUBTOTAL_IGN_NESTED_ST_AG; break; case 1 : // ignore hidden rows, nested SUBTOTAL and AGGREGATE functions mnSubTotalFlags = SUBTOTAL_IGN_HIDDEN | SUBTOTAL_IGN_NESTED_ST_AG; break; case 2 : // ignore error values, nested SUBTOTAL and AGGREGATE functions mnSubTotalFlags = SUBTOTAL_IGN_ERR_VAL | SUBTOTAL_IGN_NESTED_ST_AG; break; case 3 : // ignore hidden rows, error values, nested SUBTOTAL and AGGREGATE functions mnSubTotalFlags = SUBTOTAL_IGN_HIDDEN | SUBTOTAL_IGN_ERR_VAL | SUBTOTAL_IGN_NESTED_ST_AG; break; case 4 : // ignore nothing mnSubTotalFlags = 0x00; break; case 5 : // ignore hidden rows mnSubTotalFlags = SUBTOTAL_IGN_HIDDEN ; break; case 6 : // ignore error values mnSubTotalFlags = SUBTOTAL_IGN_ERR_VAL ; break; case 7 : // ignore hidden rows and error values mnSubTotalFlags = SUBTOTAL_IGN_HIDDEN | SUBTOTAL_IGN_ERR_VAL ; break; default : PushIllegalArgument(); return; } cPar = nParamCount - 2; switch ( nFunc ) { case SUBTOTAL_FUNC_AVE : ScAverage(); break; case SUBTOTAL_FUNC_CNT : ScCount(); break; case SUBTOTAL_FUNC_CNT2 : ScCount2(); break; case SUBTOTAL_FUNC_MAX : ScMax(); break; case SUBTOTAL_FUNC_MIN : ScMin(); break; case SUBTOTAL_FUNC_PROD : ScProduct(); break; case SUBTOTAL_FUNC_STD : ScStDev(); break; case SUBTOTAL_FUNC_STDP : ScStDevP(); break; case SUBTOTAL_FUNC_SUM : ScSum(); break; case SUBTOTAL_FUNC_VAR : ScVar(); break; case SUBTOTAL_FUNC_VARP : ScVarP(); break; case AGGREGATE_FUNC_MEDIAN : ScMedian(); break; case AGGREGATE_FUNC_MODSNGL : ScModalValue(); break; case AGGREGATE_FUNC_LARGE : ScLarge(); break; case AGGREGATE_FUNC_SMALL : ScSmall(); break; case AGGREGATE_FUNC_PERCINC : ScPercentile( true ); break; case AGGREGATE_FUNC_QRTINC : ScQuartile( true ); break; case AGGREGATE_FUNC_PERCEXC : ScPercentile( false ); break; case AGGREGATE_FUNC_QRTEXC : ScQuartile( false ); break; default : PushIllegalArgument(); break; } mnSubTotalFlags = 0x00; } double nVal = GetDouble(); // Get rid of the 1st and 2nd (fished) parameters. Pop(); Pop(); PushDouble( nVal ); } } std::unique_ptr ScInterpreter::GetDBParams( bool& rMissingField ) { bool bAllowMissingField = false; if ( rMissingField ) { bAllowMissingField = true; rMissingField = false; } if ( GetByte() == 3 ) { // First, get the query criteria range. ::std::unique_ptr pQueryRef( PopDBDoubleRef() ); if (!pQueryRef.get()) return nullptr; bool bByVal = true; double nVal = 0.0; svl::SharedString aStr; ScRange aMissingRange; bool bRangeFake = false; switch (GetStackType()) { case svDouble : nVal = ::rtl::math::approxFloor( GetDouble() ); if ( bAllowMissingField && nVal == 0.0 ) rMissingField = true; // fake missing parameter break; case svString : bByVal = false; aStr = GetString(); break; case svSingleRef : { ScAddress aAdr; PopSingleRef( aAdr ); ScRefCellValue aCell(*pDok, aAdr); if (aCell.hasNumeric()) nVal = GetCellValue(aAdr, aCell); else { bByVal = false; GetCellString(aStr, aCell); } } break; case svDoubleRef : if ( bAllowMissingField ) { // fake missing parameter for old SO compatibility bRangeFake = true; PopDoubleRef( aMissingRange ); } else { PopError(); SetError( errIllegalParameter ); } break; case svMissing : PopError(); if ( bAllowMissingField ) rMissingField = true; else SetError( errIllegalParameter ); break; default: PopError(); SetError( errIllegalParameter ); } if (nGlobalError) return nullptr; unique_ptr pDBRef( PopDBDoubleRef() ); if (nGlobalError || !pDBRef.get()) return nullptr; if ( bRangeFake ) { // range parameter must match entire database range if (pDBRef->isRangeEqual(aMissingRange)) rMissingField = true; else SetError( errIllegalParameter ); } if (nGlobalError) return nullptr; SCCOL nField = pDBRef->getFirstFieldColumn(); if (rMissingField) ; // special case else if (bByVal) nField = pDBRef->findFieldColumn(static_cast(nVal)); else { sal_uInt16 nErr = 0; nField = pDBRef->findFieldColumn(aStr.getString(), &nErr); SetError(nErr); } if (!ValidCol(nField)) return nullptr; unique_ptr pParam( pDBRef->createQueryParam(pQueryRef.get()) ); if (pParam.get()) { // An allowed missing field parameter sets the result field // to any of the query fields, just to be able to return // some cell from the iterator. if ( rMissingField ) nField = static_cast(pParam->GetEntry(0).nField); pParam->mnField = nField; SCSIZE nCount = pParam->GetEntryCount(); for ( SCSIZE i=0; i < nCount; i++ ) { ScQueryEntry& rEntry = pParam->GetEntry(i); if (!rEntry.bDoQuery) break; ScQueryEntry::Item& rItem = rEntry.GetQueryItem(); sal_uInt32 nIndex = 0; OUString aQueryStr = rItem.maString.getString(); bool bNumber = pFormatter->IsNumberFormat( aQueryStr, nIndex, rItem.mfVal); rItem.meType = bNumber ? ScQueryEntry::ByValue : ScQueryEntry::ByString; if (!bNumber && pParam->eSearchType == utl::SearchParam::SRCH_NORMAL) pParam->eSearchType = DetectSearchType(aQueryStr, pDok); } return pParam; } } return nullptr; } void ScInterpreter::DBIterator( ScIterFunc eFunc ) { double nErg = 0.0; double fMem = 0.0; sal_uLong nCount = 0; bool bMissingField = false; unique_ptr pQueryParam( GetDBParams(bMissingField) ); if (pQueryParam.get()) { if (!pQueryParam->IsValidFieldIndex()) { SetError(errNoValue); return; } ScDBQueryDataIterator aValIter(pDok, pQueryParam.release()); ScDBQueryDataIterator::Value aValue; if ( aValIter.GetFirst(aValue) && !aValue.mnError ) { switch( eFunc ) { case ifPRODUCT: nErg = 1; break; case ifMAX: nErg = -MAXDOUBLE; break; case ifMIN: nErg = MAXDOUBLE; break; default: ; // nothing } bool bNull = true; do { nCount++; switch( eFunc ) { case ifAVERAGE: case ifSUM: if ( bNull && aValue.mfValue != 0.0 ) { bNull = false; fMem = aValue.mfValue; } else nErg += aValue.mfValue; break; case ifSUMSQ: nErg += aValue.mfValue * aValue.mfValue; break; case ifPRODUCT: nErg *= aValue.mfValue; break; case ifMAX: if( aValue.mfValue > nErg ) nErg = aValue.mfValue; break; case ifMIN: if( aValue.mfValue < nErg ) nErg = aValue.mfValue; break; default: ; // nothing } } while ( aValIter.GetNext(aValue) && !aValue.mnError ); } SetError(aValue.mnError); } else SetError( errIllegalParameter); switch( eFunc ) { case ifCOUNT: nErg = nCount; break; case ifSUM: nErg = ::rtl::math::approxAdd( nErg, fMem ); break; case ifAVERAGE: nErg = div(::rtl::math::approxAdd(nErg, fMem), nCount); break; default: ; // nothing } PushDouble( nErg ); } void ScInterpreter::ScDBSum() { DBIterator( ifSUM ); } void ScInterpreter::ScDBCount() { bool bMissingField = true; unique_ptr pQueryParam( GetDBParams(bMissingField) ); if (pQueryParam.get()) { sal_uLong nCount = 0; if ( bMissingField && pQueryParam->GetType() == ScDBQueryParamBase::INTERNAL ) { // count all matching records // TODO: currently the QueryIterators only return cell pointers of // existing cells, so if a query matches an empty cell there's // nothing returned, and therefore not counted! // Since this has ever been the case and this code here only came // into existence to fix #i6899 and it never worked before we'll // have to live with it until we reimplement the iterators to also // return empty cells, which would mean to adapt all callers of // iterators. ScDBQueryParamInternal* p = static_cast(pQueryParam.get()); p->nCol2 = p->nCol1; // Don't forget to select only one column. SCTAB nTab = p->nTab; // ScQueryCellIterator doesn't make use of ScDBQueryParamBase::mnField, // so the source range has to be restricted, like before the introduction // of ScDBQueryParamBase. p->nCol1 = p->nCol2 = p->mnField; ScQueryCellIterator aCellIter( pDok, nTab, *p); if ( aCellIter.GetFirst() ) { do { nCount++; } while ( aCellIter.GetNext() ); } } else { // count only matching records with a value in the "result" field if (!pQueryParam->IsValidFieldIndex()) { SetError(errNoValue); return; } ScDBQueryDataIterator aValIter( pDok, pQueryParam.release()); ScDBQueryDataIterator::Value aValue; if ( aValIter.GetFirst(aValue) && !aValue.mnError ) { do { nCount++; } while ( aValIter.GetNext(aValue) && !aValue.mnError ); } SetError(aValue.mnError); } PushDouble( nCount ); } else PushIllegalParameter(); } void ScInterpreter::ScDBCount2() { bool bMissingField = true; unique_ptr pQueryParam( GetDBParams(bMissingField) ); if (pQueryParam.get()) { if (!pQueryParam->IsValidFieldIndex()) { SetError(errNoValue); return; } sal_uLong nCount = 0; pQueryParam->mbSkipString = false; ScDBQueryDataIterator aValIter( pDok, pQueryParam.release()); ScDBQueryDataIterator::Value aValue; if ( aValIter.GetFirst(aValue) && !aValue.mnError ) { do { nCount++; } while ( aValIter.GetNext(aValue) && !aValue.mnError ); } SetError(aValue.mnError); PushDouble( nCount ); } else PushIllegalParameter(); } void ScInterpreter::ScDBAverage() { DBIterator( ifAVERAGE ); } void ScInterpreter::ScDBMax() { DBIterator( ifMAX ); } void ScInterpreter::ScDBMin() { DBIterator( ifMIN ); } void ScInterpreter::ScDBProduct() { DBIterator( ifPRODUCT ); } void ScInterpreter::GetDBStVarParams( double& rVal, double& rValCount ) { std::vector values; double vSum = 0.0; double vMean = 0.0; rValCount = 0.0; double fSum = 0.0; bool bMissingField = false; unique_ptr pQueryParam( GetDBParams(bMissingField) ); if (pQueryParam.get()) { if (!pQueryParam->IsValidFieldIndex()) { SetError(errNoValue); return; } ScDBQueryDataIterator aValIter(pDok, pQueryParam.release()); ScDBQueryDataIterator::Value aValue; if (aValIter.GetFirst(aValue) && !aValue.mnError) { do { rValCount++; values.push_back(aValue.mfValue); fSum += aValue.mfValue; } while ((aValue.mnError == 0) && aValIter.GetNext(aValue)); } SetError(aValue.mnError); } else SetError( errIllegalParameter); vMean = fSum / values.size(); for (double v : values) vSum += (v - vMean) * (v - vMean); rVal = vSum; } void ScInterpreter::ScDBStdDev() { double fVal, fCount; GetDBStVarParams( fVal, fCount ); PushDouble( sqrt(fVal/(fCount-1))); } void ScInterpreter::ScDBStdDevP() { double fVal, fCount; GetDBStVarParams( fVal, fCount ); PushDouble( sqrt(fVal/fCount)); } void ScInterpreter::ScDBVar() { double fVal, fCount; GetDBStVarParams( fVal, fCount ); PushDouble(fVal/(fCount-1)); } void ScInterpreter::ScDBVarP() { double fVal, fCount; GetDBStVarParams( fVal, fCount ); PushDouble(fVal/fCount); } void ScInterpreter::ScIndirect() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 1, 2 ) ) { // Reference address syntax for INDIRECT is configurable. FormulaGrammar::AddressConvention eConv = maCalcConfig.meStringRefAddressSyntax; if (eConv == FormulaGrammar::CONV_UNSPECIFIED) // Use the current address syntax if unspecified. eConv = pDok->GetAddressConvention(); // either CONV_A1_XL_A1 was explicitly configured, or it wasn't possible // to determine which syntax to use during doc import bool bTryXlA1 = (eConv == FormulaGrammar::CONV_A1_XL_A1); if (nParamCount == 2 && 0.0 == ::rtl::math::approxFloor( GetDouble())) { // Overwrite the config and try Excel R1C1. eConv = FormulaGrammar::CONV_XL_R1C1; bTryXlA1 = false; } const ScAddress::Details aDetails( bTryXlA1 ? FormulaGrammar::CONV_OOO : eConv, aPos ); const ScAddress::Details aDetailsXlA1( FormulaGrammar::CONV_XL_A1, aPos ); SCTAB nTab = aPos.Tab(); OUString sRefStr = GetString().getString(); ScRefAddress aRefAd, aRefAd2; ScAddress::ExternalInfo aExtInfo; if ( ConvertDoubleRef(pDok, sRefStr, nTab, aRefAd, aRefAd2, aDetails, &aExtInfo) || ( bTryXlA1 && ConvertDoubleRef(pDok, sRefStr, nTab, aRefAd, aRefAd2, aDetailsXlA1, &aExtInfo) ) ) { if (aExtInfo.mbExternal) { PushExternalDoubleRef( aExtInfo.mnFileId, aExtInfo.maTabName, aRefAd.Col(), aRefAd.Row(), aRefAd.Tab(), aRefAd2.Col(), aRefAd2.Row(), aRefAd2.Tab()); } else PushDoubleRef( aRefAd, aRefAd2); } else if ( ConvertSingleRef(pDok, sRefStr, nTab, aRefAd, aDetails, &aExtInfo) || ( bTryXlA1 && ConvertSingleRef (pDok, sRefStr, nTab, aRefAd, aDetailsXlA1, &aExtInfo) ) ) { if (aExtInfo.mbExternal) { PushExternalSingleRef( aExtInfo.mnFileId, aExtInfo.maTabName, aRefAd.Col(), aRefAd.Row(), aRefAd.Tab()); } else PushSingleRef( aRefAd); } else { do { ScRangeData* pData = ScRangeStringConverter::GetRangeDataFromString(sRefStr, nTab, pDok); if (!pData) break; // We need this in order to obtain a good range. pData->ValidateTabRefs(); ScRange aRange; // This is the usual way to treat named ranges containing // relative references. if (!pData->IsReference( aRange, aPos)) break; if (aRange.aStart == aRange.aEnd) PushSingleRef( aRange.aStart.Col(), aRange.aStart.Row(), aRange.aStart.Tab()); else PushDoubleRef( aRange.aStart.Col(), aRange.aStart.Row(), aRange.aStart.Tab(), aRange.aEnd.Col(), aRange.aEnd.Row(), aRange.aEnd.Tab()); // success! return; } while (false); do { OUString aName( ScGlobal::pCharClass->uppercase( sRefStr)); ScDBCollection::NamedDBs& rDBs = pDok->GetDBCollection()->getNamedDBs(); const ScDBData* pData = rDBs.findByUpperName( aName); if (!pData) break; ScRange aRange; pData->GetArea( aRange); // In Excel, specifying a table name without [] resolves to the // same as with [], a range that excludes header and totals // rows and contains only data rows. Do the same. if (pData->HasHeader()) aRange.aStart.IncRow(); if (pData->HasTotals()) aRange.aEnd.IncRow(-1); if (aRange.aStart.Row() > aRange.aEnd.Row()) break; if (aRange.aStart == aRange.aEnd) PushSingleRef( aRange.aStart.Col(), aRange.aStart.Row(), aRange.aStart.Tab()); else PushDoubleRef( aRange.aStart.Col(), aRange.aStart.Row(), aRange.aStart.Tab(), aRange.aEnd.Col(), aRange.aEnd.Row(), aRange.aEnd.Tab()); // success! return; } while (false); // It may be even a TableRef. // Anything else that resolves to one reference could be added // here, but we don't want to compile every arbitrary string. This // is already nasty enough.. sal_Int32 nIndex = 0; if ((nIndex = sRefStr.indexOf('[')) >= 0 && sRefStr.indexOf(']',nIndex+1) > nIndex) { do { ScCompiler aComp( pDok, aPos); aComp.SetGrammar( pDok->GetGrammar()); aComp.SetRefConvention( eConv); // must be after grammar std::unique_ptr pArr( aComp.CompileString( sRefStr)); // Whatever.. use only the specific case. if (!pArr->HasOpCode( ocTableRef)) break; aComp.CompileTokenArray(); // A syntactically valid reference will generate exactly // one RPN token, a reference or error. Discard everything // else as error. if (pArr->GetCodeLen() != 1) break; ScTokenRef xTok( pArr->FirstRPN()); if (!xTok) break; switch (xTok->GetType()) { case svSingleRef: case svDoubleRef: case svError: PushTempToken( xTok.get()); // success! return; default: ; // nothing } } while (false); } PushError( errNoRef); } } } void ScInterpreter::ScAddressFunc() { OUString sTabStr; sal_uInt8 nParamCount = GetByte(); if( !MustHaveParamCount( nParamCount, 2, 5 ) ) return; if( nParamCount >= 5 ) sTabStr = GetString().getString(); FormulaGrammar::AddressConvention eConv = FormulaGrammar::CONV_OOO; // default if( nParamCount >= 4 && 0.0 == ::rtl::math::approxFloor( GetDoubleWithDefault( 1.0))) eConv = FormulaGrammar::CONV_XL_R1C1; ScRefFlags nFlags = ScRefFlags::COL_ABS | ScRefFlags::ROW_ABS; // default if( nParamCount >= 3 ) { sal_Int32 n = GetInt32WithDefault(1); switch ( n ) { default : PushNoValue(); return; case 5: case 1 : break; // default case 6: case 2 : nFlags = ScRefFlags::ROW_ABS; break; case 7: case 3 : nFlags = ScRefFlags::COL_ABS; break; case 8: case 4 : nFlags = ScRefFlags::ZERO; break; // both relative } } nFlags |= ScRefFlags::VALID | ScRefFlags::ROW_VALID | ScRefFlags::COL_VALID; SCCOL nCol = (SCCOL) GetInt16(); SCROW nRow = (SCROW) GetInt32(); if( eConv == FormulaGrammar::CONV_XL_R1C1 ) { // YUCK! The XL interface actually treats rel R1C1 refs differently // than A1 if( !(nFlags & ScRefFlags::COL_ABS) ) nCol += aPos.Col() + 1; if( !(nFlags & ScRefFlags::ROW_ABS) ) nRow += aPos.Row() + 1; } --nCol; --nRow; if (nGlobalError || !ValidCol( nCol) || !ValidRow( nRow)) { PushIllegalArgument(); return; } const ScAddress::Details aDetails( eConv, aPos ); const ScAddress aAdr( nCol, nRow, 0); OUString aRefStr(aAdr.Format(nFlags, pDok, aDetails)); if( nParamCount >= 5 && !sTabStr.isEmpty() ) { OUString aDoc; if (eConv == FormulaGrammar::CONV_OOO) { // Isolate Tab from 'Doc'#Tab sal_Int32 nPos = ScCompiler::GetDocTabPos( sTabStr); if (nPos != -1) { if (sTabStr[nPos+1] == '$') ++nPos; // also split 'Doc'#$Tab aDoc = sTabStr.copy( 0, nPos+1); sTabStr = sTabStr.copy( nPos+1); } } /* TODO: yet unsupported external reference in CONV_XL_R1C1 syntax may * need some extra handling to isolate Tab from Doc. */ if (sTabStr[0] != '\'' || !sTabStr.endsWith("'")) ScCompiler::CheckTabQuotes( sTabStr, eConv); if (!aDoc.isEmpty()) sTabStr = aDoc + sTabStr; sTabStr += eConv == FormulaGrammar::CONV_XL_R1C1 ? OUString("!") : OUString("."); sTabStr += aRefStr; PushString( sTabStr ); } else PushString( aRefStr ); } void ScInterpreter::ScOffset() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 3, 5 ) ) { sal_Int32 nColNew = -1, nRowNew = -1, nColPlus, nRowPlus; if (nParamCount == 5) nColNew = GetInt32(); if (nParamCount >= 4) nRowNew = GetInt32WithDefault(-1); nColPlus = GetInt32(); nRowPlus = GetInt32(); if (nGlobalError) { PushError( nGlobalError); return; } SCCOL nCol1(0); SCROW nRow1(0); SCTAB nTab1(0); SCCOL nCol2(0); SCROW nRow2(0); SCTAB nTab2(0); if (nColNew == 0 || nRowNew == 0) { PushIllegalArgument(); return; } switch (GetStackType()) { case svSingleRef: { PopSingleRef(nCol1, nRow1, nTab1); if (nParamCount == 3 || (nColNew < 0 && nRowNew < 0)) { nCol1 = (SCCOL)((long) nCol1 + nColPlus); nRow1 = (SCROW)((long) nRow1 + nRowPlus); if (!ValidCol(nCol1) || !ValidRow(nRow1)) PushIllegalArgument(); else PushSingleRef(nCol1, nRow1, nTab1); } else { if (nColNew < 0) nColNew = 1; if (nRowNew < 0) nRowNew = 1; nCol1 = (SCCOL)((long)nCol1+nColPlus); nRow1 = (SCROW)((long)nRow1+nRowPlus); nCol2 = (SCCOL)((long)nCol1+nColNew-1); nRow2 = (SCROW)((long)nRow1+nRowNew-1); if (!ValidCol(nCol1) || !ValidRow(nRow1) || !ValidCol(nCol2) || !ValidRow(nRow2)) PushIllegalArgument(); else PushDoubleRef(nCol1, nRow1, nTab1, nCol2, nRow2, nTab1); } break; } case svExternalSingleRef: { sal_uInt16 nFileId; OUString aTabName; ScSingleRefData aRef; PopExternalSingleRef(nFileId, aTabName, aRef); ScAddress aAbsRef = aRef.toAbs(aPos); nCol1 = aAbsRef.Col(); nRow1 = aAbsRef.Row(); nTab1 = aAbsRef.Tab(); if (nParamCount == 3 || (nColNew < 0 && nRowNew < 0)) { nCol1 = (SCCOL)((long) nCol1 + nColPlus); nRow1 = (SCROW)((long) nRow1 + nRowPlus); if (!ValidCol(nCol1) || !ValidRow(nRow1)) PushIllegalArgument(); else PushExternalSingleRef(nFileId, aTabName, nCol1, nRow1, nTab1); } else { if (nColNew < 0) nColNew = 1; if (nRowNew < 0) nRowNew = 1; nCol1 = (SCCOL)((long)nCol1+nColPlus); nRow1 = (SCROW)((long)nRow1+nRowPlus); nCol2 = (SCCOL)((long)nCol1+nColNew-1); nRow2 = (SCROW)((long)nRow1+nRowNew-1); nTab2 = nTab1; if (!ValidCol(nCol1) || !ValidRow(nRow1) || !ValidCol(nCol2) || !ValidRow(nRow2)) PushIllegalArgument(); else PushExternalDoubleRef(nFileId, aTabName, nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); } break; } case svDoubleRef: { PopDoubleRef(nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); if (nColNew < 0) nColNew = nCol2 - nCol1 + 1; if (nRowNew < 0) nRowNew = nRow2 - nRow1 + 1; nCol1 = (SCCOL)((long)nCol1+nColPlus); nRow1 = (SCROW)((long)nRow1+nRowPlus); nCol2 = (SCCOL)((long)nCol1+nColNew-1); nRow2 = (SCROW)((long)nRow1+nRowNew-1); if (!ValidCol(nCol1) || !ValidRow(nRow1) || !ValidCol(nCol2) || !ValidRow(nRow2) || nTab1 != nTab2) PushIllegalArgument(); else PushDoubleRef(nCol1, nRow1, nTab1, nCol2, nRow2, nTab1); break; } case svExternalDoubleRef: { sal_uInt16 nFileId; OUString aTabName; ScComplexRefData aRef; PopExternalDoubleRef(nFileId, aTabName, aRef); ScRange aAbs = aRef.toAbs(aPos); nCol1 = aAbs.aStart.Col(); nRow1 = aAbs.aStart.Row(); nTab1 = aAbs.aStart.Tab(); nCol2 = aAbs.aEnd.Col(); nRow2 = aAbs.aEnd.Row(); nTab2 = aAbs.aEnd.Tab(); if (nColNew < 0) nColNew = nCol2 - nCol1 + 1; if (nRowNew < 0) nRowNew = nRow2 - nRow1 + 1; nCol1 = (SCCOL)((long)nCol1+nColPlus); nRow1 = (SCROW)((long)nRow1+nRowPlus); nCol2 = (SCCOL)((long)nCol1+nColNew-1); nRow2 = (SCROW)((long)nRow1+nRowNew-1); if (!ValidCol(nCol1) || !ValidRow(nRow1) || !ValidCol(nCol2) || !ValidRow(nRow2) || nTab1 != nTab2) PushIllegalArgument(); else PushExternalDoubleRef(nFileId, aTabName, nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); break; } default: PushIllegalParameter(); break; } // end switch } } void ScInterpreter::ScIndex() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 1, 4 ) ) { sal_uInt32 nArea; size_t nAreaCount; SCCOL nCol; SCROW nRow; if (nParamCount == 4) nArea = GetUInt32(); else nArea = 1; if (nParamCount >= 3) nCol = (SCCOL) GetInt16(); else nCol = 0; if (nParamCount >= 2) nRow = (SCROW) GetInt32(); else nRow = 0; if (GetStackType() == svRefList) nAreaCount = (sp ? pStack[sp-1]->GetRefList()->size() : 0); else nAreaCount = 1; // one reference or array or whatever if (nGlobalError || nAreaCount == 0 || (size_t)nArea > nAreaCount) { PushError( errNoRef); return; } else if (nArea < 1 || nCol < 0 || nRow < 0) { PushIllegalArgument(); return; } switch (GetStackType()) { case svMatrix: case svExternalSingleRef: case svExternalDoubleRef: { if (nArea != 1) SetError(errIllegalArgument); sal_uInt16 nOldSp = sp; ScMatrixRef pMat = GetMatrix(); if (pMat) { SCSIZE nC, nR; pMat->GetDimensions(nC, nR); // Access one element of a vector independent of col/row // orientation? bool bVector = ((nCol == 0 || nRow == 0) && (nC == 1 || nR == 1)); SCSIZE nElement = ::std::max( static_cast(nCol), static_cast(nRow)); if (nC == 0 || nR == 0 || (!bVector && (static_cast(nCol) > nC || static_cast(nRow) > nR)) || (bVector && nElement > nC * nR)) PushIllegalArgument(); else if (nCol == 0 && nRow == 0) sp = nOldSp; else if (bVector) { --nElement; if (pMat->IsString( nElement)) PushString( pMat->GetString(nElement).getString()); else PushDouble( pMat->GetDouble( nElement)); } else if (nCol == 0) { ScMatrixRef pResMat = GetNewMat(nC, 1); if (pResMat) { SCSIZE nRowMinus1 = static_cast(nRow - 1); for (SCSIZE i = 0; i < nC; i++) if (!pMat->IsString(i, nRowMinus1)) pResMat->PutDouble(pMat->GetDouble(i, nRowMinus1), i, 0); else pResMat->PutString(pMat->GetString(i, nRowMinus1), i, 0); PushMatrix(pResMat); } else PushIllegalArgument(); } else if (nRow == 0) { ScMatrixRef pResMat = GetNewMat(1, nR); if (pResMat) { SCSIZE nColMinus1 = static_cast(nCol - 1); for (SCSIZE i = 0; i < nR; i++) if (!pMat->IsString(nColMinus1, i)) pResMat->PutDouble(pMat->GetDouble(nColMinus1, i), i); else pResMat->PutString(pMat->GetString(nColMinus1, i), i); PushMatrix(pResMat); } else PushIllegalArgument(); } else { if (!pMat->IsString( static_cast(nCol-1), static_cast(nRow-1))) PushDouble( pMat->GetDouble( static_cast(nCol-1), static_cast(nRow-1))); else PushString( pMat->GetString( static_cast(nCol-1), static_cast(nRow-1)).getString()); } } } break; case svSingleRef: { SCCOL nCol1 = 0; SCROW nRow1 = 0; SCTAB nTab1 = 0; PopSingleRef( nCol1, nRow1, nTab1); if (nCol > 1 || nRow > 1) PushIllegalArgument(); else PushSingleRef( nCol1, nRow1, nTab1); } break; case svDoubleRef: case svRefList: { SCCOL nCol1 = 0; SCROW nRow1 = 0; SCTAB nTab1 = 0; SCCOL nCol2 = 0; SCROW nRow2 = 0; SCTAB nTab2 = 0; bool bRowArray = false; if (GetStackType() == svRefList) { FormulaTokenRef xRef = PopToken(); if (nGlobalError || !xRef) { PushIllegalParameter(); return; } ScRange aRange( ScAddress::UNINITIALIZED); DoubleRefToRange( (*(xRef.get()->GetRefList()))[nArea-1], aRange); aRange.GetVars( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); if ( nParamCount == 2 && nRow1 == nRow2 ) bRowArray = true; } else { PopDoubleRef( nCol1, nRow1, nTab1, nCol2, nRow2, nTab2); if ( nParamCount == 2 && nRow1 == nRow2 ) bRowArray = true; } if ( nTab1 != nTab2 || (nCol > 0 && nCol1+nCol-1 > nCol2) || (nRow > 0 && nRow1+nRow-1 > nRow2 && !bRowArray ) || ( nRow > nCol2 - nCol1 + 1 && bRowArray )) PushIllegalArgument(); else if (nCol == 0 && nRow == 0) { if ( nCol1 == nCol2 && nRow1 == nRow2 ) PushSingleRef( nCol1, nRow1, nTab1 ); else PushDoubleRef( nCol1, nRow1, nTab1, nCol2, nRow2, nTab1 ); } else if (nRow == 0) { if ( nRow1 == nRow2 ) PushSingleRef( nCol1+nCol-1, nRow1, nTab1 ); else PushDoubleRef( nCol1+nCol-1, nRow1, nTab1, nCol1+nCol-1, nRow2, nTab1 ); } else if (nCol == 0) { if ( nCol1 == nCol2 ) PushSingleRef( nCol1, nRow1+nRow-1, nTab1 ); else if ( bRowArray ) { nCol =(SCCOL) nRow; nRow = 1; PushSingleRef( nCol1+nCol-1, nRow1+nRow-1, nTab1); } else PushDoubleRef( nCol1, nRow1+nRow-1, nTab1, nCol2, nRow1+nRow-1, nTab1); } else PushSingleRef( nCol1+nCol-1, nRow1+nRow-1, nTab1); } break; default: PushIllegalParameter(); } } } void ScInterpreter::ScMultiArea() { // Legacy support, convert to RefList sal_uInt8 nParamCount = GetByte(); if (MustHaveParamCountMin( nParamCount, 1)) { while (!nGlobalError && nParamCount-- > 1) { ScUnionFunc(); } } } void ScInterpreter::ScAreas() { sal_uInt8 nParamCount = GetByte(); if (MustHaveParamCount( nParamCount, 1)) { size_t nCount = 0; switch (GetStackType()) { case svSingleRef: { FormulaTokenRef xT = PopToken(); ValidateRef( *xT.get()->GetSingleRef()); ++nCount; } break; case svDoubleRef: { FormulaTokenRef xT = PopToken(); ValidateRef( *xT.get()->GetDoubleRef()); ++nCount; } break; case svRefList: { FormulaTokenRef xT = PopToken(); ValidateRef( *(xT.get()->GetRefList())); nCount += xT.get()->GetRefList()->size(); } break; default: SetError( errIllegalParameter); } PushDouble( double(nCount)); } } void ScInterpreter::ScCurrency() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 1, 2 ) ) { OUString aStr; double fDec; if (nParamCount == 2) { fDec = ::rtl::math::approxFloor(GetDouble()); if (fDec < -15.0 || fDec > 15.0) { PushIllegalArgument(); return; } } else fDec = 2.0; double fVal = GetDouble(); double fFac; if ( fDec != 0.0 ) fFac = pow( (double)10, fDec ); else fFac = 1.0; if (fVal < 0.0) fVal = ceil(fVal*fFac-0.5)/fFac; else fVal = floor(fVal*fFac+0.5)/fFac; Color* pColor = nullptr; if ( fDec < 0.0 ) fDec = 0.0; sal_uLong nIndex = pFormatter->GetStandardFormat( css::util::NumberFormat::CURRENCY, ScGlobal::eLnge); if ( (sal_uInt16) fDec != pFormatter->GetFormatPrecision( nIndex ) ) { OUString sFormatString = pFormatter->GenerateFormat( nIndex, ScGlobal::eLnge, true, // mit Tausenderpunkt false, // nicht rot (sal_uInt16) fDec);// Nachkommastellen if (!pFormatter->GetPreviewString(sFormatString, fVal, aStr, &pColor, ScGlobal::eLnge)) SetError(errIllegalArgument); } else { pFormatter->GetOutputString(fVal, nIndex, aStr, &pColor); } PushString(aStr); } } void ScInterpreter::ScReplace() { if ( MustHaveParamCount( GetByte(), 4 ) ) { OUString aNewStr = GetString().getString(); double fCount = GetStringPositionArgument(); double fPos = GetStringPositionArgument(); OUString aOldStr = GetString().getString(); if (fPos < 1.0 || fCount < 0.0) PushIllegalArgument(); else { sal_Int32 nCount = static_cast(fCount); sal_Int32 nPos = static_cast(fPos); sal_Int32 nLen = aOldStr.getLength(); if (nPos > nLen + 1) nPos = nLen + 1; if (nCount > nLen - nPos + 1) nCount = nLen - nPos + 1; aOldStr = aOldStr.replaceAt( nPos-1, nCount, "" ); if ( CheckStringResultLen( aOldStr, aNewStr ) ) aOldStr = aOldStr.replaceAt( nPos-1, 0, aNewStr ); PushString( aOldStr ); } } } void ScInterpreter::ScFixed() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 1, 3 ) ) { OUString aStr; double fDec; bool bThousand; if (nParamCount == 3) bThousand = !GetBool(); // Param TRUE: keine Tausenderpunkte else bThousand = true; if (nParamCount >= 2) { fDec = ::rtl::math::approxFloor(GetDoubleWithDefault( 2.0 )); if (fDec < -15.0 || fDec > 15.0) { PushIllegalArgument(); return; } } else fDec = 2.0; double fVal = GetDouble(); double fFac; if ( fDec != 0.0 ) fFac = pow( (double)10, fDec ); else fFac = 1.0; if (fVal < 0.0) fVal = ceil(fVal*fFac-0.5)/fFac; else fVal = floor(fVal*fFac+0.5)/fFac; Color* pColor = nullptr; if (fDec < 0.0) fDec = 0.0; sal_uLong nIndex = pFormatter->GetStandardFormat( css::util::NumberFormat::NUMBER, ScGlobal::eLnge); OUString sFormatString = pFormatter->GenerateFormat( nIndex, ScGlobal::eLnge, bThousand, // mit Tausenderpunkt false, // nicht rot (sal_uInt16) fDec);// Nachkommastellen if (!pFormatter->GetPreviewString(sFormatString, fVal, aStr, &pColor, ScGlobal::eLnge)) PushIllegalArgument(); else PushString(aStr); } } void ScInterpreter::ScFind() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 2, 3 ) ) { sal_Int32 nAnz; if (nParamCount == 3) nAnz = GetDouble(); else nAnz = 1; OUString sStr = GetString().getString(); if (nAnz < 1 || nAnz > sStr.getLength()) PushNoValue(); else { sal_Int32 nPos = sStr.indexOf(GetString().getString(), nAnz - 1); if (nPos == -1) PushNoValue(); else PushDouble((double)(nPos + 1)); } } } void ScInterpreter::ScExact() { nFuncFmtType = css::util::NumberFormat::LOGICAL; if ( MustHaveParamCount( GetByte(), 2 ) ) { svl::SharedString s1 = GetString(); svl::SharedString s2 = GetString(); PushInt( int(s1.getData() == s2.getData()) ); } } void ScInterpreter::ScLeft() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 1, 2 ) ) { sal_Int32 n; if (nParamCount == 2) { double nVal = GetStringPositionArgument(); if (nVal < 0.0) { PushIllegalArgument(); return ; } else n = (sal_Int32) nVal; } else n = 1; OUString aStr = GetString().getString(); n = std::min(n, aStr.getLength()); aStr = aStr.copy( 0, n ); PushString( aStr ); } } typedef struct { UBlockCode from; UBlockCode to; } UBlockScript; static const UBlockScript scriptList[] = { {UBLOCK_HANGUL_JAMO, UBLOCK_HANGUL_JAMO}, {UBLOCK_CJK_RADICALS_SUPPLEMENT, UBLOCK_HANGUL_SYLLABLES}, {UBLOCK_CJK_COMPATIBILITY_IDEOGRAPHS,UBLOCK_CJK_RADICALS_SUPPLEMENT }, {UBLOCK_IDEOGRAPHIC_DESCRIPTION_CHARACTERS,UBLOCK_CJK_COMPATIBILITY_IDEOGRAPHS}, {UBLOCK_CJK_COMPATIBILITY_FORMS, UBLOCK_CJK_COMPATIBILITY_FORMS}, {UBLOCK_HALFWIDTH_AND_FULLWIDTH_FORMS, UBLOCK_HALFWIDTH_AND_FULLWIDTH_FORMS}, {UBLOCK_CJK_UNIFIED_IDEOGRAPHS_EXTENSION_B, UBLOCK_CJK_COMPATIBILITY_IDEOGRAPHS_SUPPLEMENT}, {UBLOCK_CJK_STROKES, UBLOCK_CJK_STROKES} }; #define scriptListCount sizeof (scriptList) / sizeof (UBlockScript) bool SAL_CALL lcl_getScriptClass(sal_uInt32 currentChar) { // for the locale of ja-JP, character U+0x005c and U+0x20ac should be ScriptType::Asian if( (currentChar == 0x005c || currentChar == 0x20ac) && (MsLangId::getSystemLanguage() == LANGUAGE_JAPANESE) ) return true; sal_uInt16 i; static bool bRet = false; UBlockCode block = (UBlockCode)ublock_getCode((sal_uInt32)currentChar); for ( i = 0; i < scriptListCount; i++) { if (block <= scriptList[i].to) break; } bRet = (i < scriptListCount && block >= scriptList[i].from); return bRet; } bool IsDBCS(sal_Unicode ch) { return lcl_getScriptClass(ch); } sal_Int32 getLengthB(const OUString &str) { if(str.isEmpty()) return 0; sal_Int32 index = 0; sal_Int32 length = 0; while(index < str.getLength()){ if(IsDBCS(str[index])) length += 2; else length++; index++; } return length; } void ScInterpreter::ScLenB() { PushDouble( getLengthB(GetString().getString()) ); } OUString lcl_RightB(const OUString &rStr, sal_Int32 n) { if( n < getLengthB(rStr) ) { OUStringBuffer aBuf(rStr); sal_Int32 index = aBuf.getLength(); while(index-- >= 0) { if(0 == n) { aBuf.remove( 0, index + 1); break; } if(-1 == n) { aBuf.remove( 0, index + 2 ); aBuf.insert( 0, " "); break; } if(IsDBCS(aBuf[index])) n -= 2; else n--; } return aBuf.makeStringAndClear(); } return rStr; } void ScInterpreter::ScRightB() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 1, 2 ) ) { sal_Int32 n; if (nParamCount == 2) { double nVal = GetStringPositionArgument(); if ( nVal < 0.0 ) { PushIllegalArgument(); return ; } else n = (sal_Int32) nVal; } else n = 1; OUString aStr(lcl_RightB(GetString().getString(), n)); PushString( aStr ); } } OUString lcl_LeftB(const OUString &rStr, sal_Int32 n) { if( n < getLengthB(rStr) ) { OUStringBuffer aBuf(rStr); sal_Int32 index = -1; while(index++ < aBuf.getLength()) { if(0 == n) { aBuf.truncate(index); break; } if(-1 == n) { aBuf.truncate( index - 1 ); aBuf.append(" "); break; } if(IsDBCS(aBuf[index])) n -= 2; else n--; } return aBuf.makeStringAndClear(); } return rStr; } void ScInterpreter::ScLeftB() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 1, 2 ) ) { sal_Int32 n; if (nParamCount == 2) { double nVal = GetStringPositionArgument(); if ( nVal < 0.0 ) { PushIllegalArgument(); return ; } else n = (sal_Int32) nVal; } else n = 1; OUString aStr(lcl_LeftB(GetString().getString(), n)); PushString( aStr ); } } void ScInterpreter::ScMidB() { if ( MustHaveParamCount( GetByte(), 3 ) ) { double fAnz = GetStringPositionArgument(); double fAnfang = GetStringPositionArgument(); OUString aStr = GetString().getString(); if (fAnfang < 1.0 || fAnz < 0.0) PushIllegalArgument(); else { aStr = lcl_LeftB(aStr, (sal_Int32)fAnfang + (sal_Int32)fAnz - 1); sal_Int32 nCnt = getLengthB(aStr) - (sal_Int32)fAnfang + 1; aStr = lcl_RightB(aStr, nCnt>0 ? nCnt:0); PushString(aStr); } } } void ScInterpreter::ScRight() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 1, 2 ) ) { sal_Int32 n; if (nParamCount == 2) { double nVal = GetStringPositionArgument(); if (nVal < 0.0) { PushIllegalArgument(); return ; } else n = (sal_Int32) nVal; } else n = 1; OUString aStr = GetString().getString(); if( n < aStr.getLength() ) aStr = aStr.copy( aStr.getLength() - n ); PushString( aStr ); } } void ScInterpreter::ScSearch() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 2, 3 ) ) { double fAnz; if (nParamCount == 3) { // This should use GetStringPositionArgument() but old versions up // to LibreOffice 4.2.5 allowed and ignored 0 and negative values. // It is unnecessary to break existing documents that "rely" on // that behavior. Though ODFF constrains Start to be >=1. /* TODO: fix this and possibly break those broken documents? */ fAnz = rtl::math::approxFloor( GetDouble()); if (fAnz < 1.0) fAnz = 1.0; else if (!CheckStringPositionArgument( fAnz)) { PushIllegalArgument(); return; } } else fAnz = 1.0; OUString sStr = GetString().getString(); OUString SearchStr = GetString().getString(); sal_Int32 nPos = fAnz - 1; sal_Int32 nEndPos = sStr.getLength(); if( nPos >= nEndPos ) PushNoValue(); else { utl::SearchParam::SearchType eSearchType = DetectSearchType( SearchStr, pDok ); utl::SearchParam sPar(SearchStr, eSearchType, false, '~', false); utl::TextSearch sT( sPar, *ScGlobal::pCharClass ); bool bBool = sT.SearchForward(sStr, &nPos, &nEndPos); if (!bBool) PushNoValue(); else PushDouble((double)(nPos) + 1); } } } void ScInterpreter::ScMid() { if ( MustHaveParamCount( GetByte(), 3 ) ) { double fAnz = GetStringPositionArgument(); double fAnfang = GetStringPositionArgument(); OUString aStr = GetString().getString(); if (fAnfang < 1.0 || fAnz < 0.0) PushIllegalArgument(); else { sal_Int32 nCharacters = std::min(static_cast(fAnz), aStr.getLength() - fAnfang + 1); OUString sRes; if (nCharacters > 0) sRes = aStr.copy(static_cast(fAnfang-1), nCharacters); PushString(sRes); } } } void ScInterpreter::ScText() { if ( MustHaveParamCount( GetByte(), 2 ) ) { OUString sFormatString = GetString().getString(); svl::SharedString aStr; bool bString = false; double fVal = 0.0; switch (GetStackType()) { case svError: PopError(); break; case svDouble: fVal = PopDouble(); break; default: { FormulaTokenRef xTok( PopToken()); if (!nGlobalError) { PushTempToken( xTok.get()); // Temporarily override the ConvertStringToValue() // error for GetCellValue() / GetCellValueOrZero() sal_uInt16 nSErr = mnStringNoValueError; mnStringNoValueError = errNotNumericString; fVal = GetDouble(); mnStringNoValueError = nSErr; if (nGlobalError == errNotNumericString) { // Not numeric. nGlobalError = 0; PushTempToken( xTok.get()); aStr = GetString(); bString = true; } } } } if (nGlobalError) PushError( nGlobalError); else { OUString aResult; Color* pColor = nullptr; LanguageType eCellLang; const ScPatternAttr* pPattern = pDok->GetPattern( aPos.Col(), aPos.Row(), aPos.Tab() ); if ( pPattern ) eCellLang = static_cast( pPattern->GetItem( ATTR_LANGUAGE_FORMAT )).GetValue(); else eCellLang = ScGlobal::eLnge; if (bString) { if (!pFormatter->GetPreviewString( sFormatString, aStr.getString(), aResult, &pColor, eCellLang)) PushIllegalArgument(); else PushString( aResult); } else { if (!pFormatter->GetPreviewStringGuess( sFormatString, fVal, aResult, &pColor, eCellLang)) PushIllegalArgument(); else PushString( aResult); } } } } void ScInterpreter::ScSubstitute() { sal_uInt8 nParamCount = GetByte(); if ( MustHaveParamCount( nParamCount, 3, 4 ) ) { sal_Int32 nAnz; if (nParamCount == 4) { double fAnz = GetStringPositionArgument(); if( fAnz < 1 ) { PushIllegalArgument(); return; } else nAnz = (sal_Int32) fAnz; } else nAnz = 0; OUString sNewStr = GetString().getString(); OUString sOldStr = GetString().getString(); OUString sStr = GetString().getString(); sal_Int32 nPos = 0; sal_Int32 nCount = 0; sal_Int32 nNewLen = sNewStr.getLength(); sal_Int32 nOldLen = sOldStr.getLength(); while( true ) { nPos = sStr.indexOf( sOldStr, nPos ); if (nPos != -1) { nCount++; if( !nAnz || nCount == nAnz ) { sStr = sStr.replaceAt(nPos,nOldLen, ""); if ( CheckStringResultLen( sStr, sNewStr ) ) { sStr = sStr.replaceAt(nPos, 0, sNewStr); nPos = sal::static_int_cast( nPos + nNewLen ); } else break; } else nPos++; } else break; } PushString( sStr ); } } void ScInterpreter::ScRept() { if ( MustHaveParamCount( GetByte(), 2 ) ) { double fAnz = GetStringPositionArgument(); OUString aStr = GetString().getString(); if ( fAnz < 0.0 ) PushIllegalArgument(); else if ( fAnz * aStr.getLength() > SAL_MAX_UINT16 ) { PushError( errStringOverflow ); } else if ( fAnz == 0.0 ) PushString( EMPTY_OUSTRING ); else { const sal_Int32 nLen = aStr.getLength(); sal_Int32 n = (sal_Int32) fAnz; OUStringBuffer aRes(n*nLen); while( n-- ) aRes.append(aStr); PushString( aRes.makeStringAndClear() ); } } } void ScInterpreter::ScConcat() { sal_uInt8 nParamCount = GetByte(); OUString aRes; while( nParamCount-- > 0) { OUString aStr = GetString().getString(); aRes = aStr + aRes; } PushString( aRes ); } sal_uInt16 ScInterpreter::GetErrorType() { sal_uInt16 nErr; sal_uInt16 nOldError = nGlobalError; nGlobalError = 0; switch ( GetStackType() ) { case svRefList : { FormulaTokenRef x = PopToken(); if (nGlobalError) nErr = nGlobalError; else { const ScRefList* pRefList = x.get()->GetRefList(); size_t n = pRefList->size(); if (!n) nErr = errNoRef; else if (n > 1) nErr = errNoValue; else { ScRange aRange; DoubleRefToRange( (*pRefList)[0], aRange); if (nGlobalError) nErr = nGlobalError; else { ScAddress aAdr; if ( DoubleRefToPosSingleRef( aRange, aAdr ) ) nErr = pDok->GetErrCode( aAdr ); else nErr = nGlobalError; } } } } break; case svDoubleRef : { ScRange aRange; PopDoubleRef( aRange ); if ( nGlobalError ) nErr = nGlobalError; else { ScAddress aAdr; if ( DoubleRefToPosSingleRef( aRange, aAdr ) ) nErr = pDok->GetErrCode( aAdr ); else nErr = nGlobalError; } } break; case svSingleRef : { ScAddress aAdr; PopSingleRef( aAdr ); if ( nGlobalError ) nErr = nGlobalError; else nErr = pDok->GetErrCode( aAdr ); } break; default: PopError(); nErr = nGlobalError; } nGlobalError = nOldError; return nErr; } void ScInterpreter::ScErrorType() { sal_uInt16 nErr = GetErrorType(); if ( nErr ) { nGlobalError = 0; PushDouble( nErr ); } else { PushNA(); } } void ScInterpreter::ScErrorType_ODF() { sal_uInt16 nErr = GetErrorType(); sal_uInt16 nErrType; switch ( nErr ) { case errParameterExpected : // #NULL! nErrType = 1; break; case errDivisionByZero : // #DIV/0! nErrType = 2; break; case errNoValue : // #VALUE! nErrType = 3; break; case errNoRef : // #REF! nErrType = 4; break; case errNoName : // #NAME? nErrType = 5; break; case errIllegalFPOperation : // #NUM! nErrType = 6; break; case NOTAVAILABLE : // #N/A nErrType = 7; break; /* #GETTING_DATA is a message that can appear in Excel when a large or complex worksheet is being calculated. In Excel 2007 and newer, operations are grouped so more complicated cells may finish after earlier ones do. While the calculations are still processing, the unfinished cells may display #GETTING_DATA. Because the message is temporary and disappears when the calculations complete, this isn’t a true error. No calc error code known (yet). case : // GETTING_DATA nErrType = 8; break; */ default : nErrType = 0; break; } if ( nErrType ) { nGlobalError = 0; PushDouble( nErrType ); } else PushNA(); } bool ScInterpreter::MayBeRegExp( const OUString& rStr, const ScDocument* pDoc, bool bIgnoreWildcards ) { if ( pDoc && !pDoc->GetDocOptions().IsFormulaRegexEnabled() ) return false; if ( rStr.isEmpty() || (rStr.getLength() == 1 && !rStr.startsWith(".")) ) return false; // single meta characters can not be a regexp // First two characters are wildcard '?' and '*' characters. static const sal_Unicode cre[] = { '?','*','+','.','[',']','^','$','\\','<','>','(',')','|', 0 }; const sal_Unicode* const pre = bIgnoreWildcards ? cre + 2 : cre; const sal_Unicode* p1 = rStr.getStr(); sal_Unicode c1; while ( ( c1 = *p1++ ) != 0 ) { const sal_Unicode* p2 = pre; while ( *p2 ) { if ( c1 == *p2++ ) return true; } } return false; } bool ScInterpreter::MayBeWildcard( const OUString& rStr, const ScDocument* pDoc ) { if ( pDoc && !pDoc->GetDocOptions().IsFormulaWildcardsEnabled() ) return false; // Wildcards with '~' escape, if there are no wildcards then an escaped // character does not make sense, but it modifies the search pattern in an // Excel compatible wildcard search.. static const sal_Unicode cw[] = { '*','?','~', 0 }; const sal_Unicode* p1 = rStr.getStr(); sal_Unicode c1; while ( ( c1 = *p1++ ) != 0 ) { const sal_Unicode* p2 = cw; while ( *p2 ) { if ( c1 == *p2++ ) return true; } } return false; } utl::SearchParam::SearchType ScInterpreter::DetectSearchType( const OUString& rStr, const ScDocument* pDoc ) { if (pDoc) { if (pDoc->GetDocOptions().IsFormulaWildcardsEnabled()) return MayBeWildcard( rStr, nullptr) ? utl::SearchParam::SRCH_WILDCARD : utl::SearchParam::SRCH_NORMAL; if (pDoc->GetDocOptions().IsFormulaRegexEnabled()) return MayBeRegExp( rStr, nullptr) ? utl::SearchParam::SRCH_REGEXP : utl::SearchParam::SRCH_NORMAL; } else { /* TODO: obtain the global config for this rare case? */ if (MayBeRegExp( rStr, nullptr, true)) return utl::SearchParam::SRCH_REGEXP; if (MayBeWildcard( rStr, nullptr)) return utl::SearchParam::SRCH_WILDCARD; } return utl::SearchParam::SRCH_NORMAL; } static bool lcl_LookupQuery( ScAddress & o_rResultPos, ScDocument * pDoc, const ScQueryParam & rParam, const ScQueryEntry & rEntry ) { bool bFound = false; ScQueryCellIterator aCellIter( pDoc, rParam.nTab, rParam, false); if (rEntry.eOp != SC_EQUAL) { // range lookup <= or >= SCCOL nCol; SCROW nRow; bFound = aCellIter.FindEqualOrSortedLastInRange( nCol, nRow); if (bFound) { o_rResultPos.SetCol( nCol); o_rResultPos.SetRow( nRow); } } else if (aCellIter.GetFirst()) { // EQUAL bFound = true; o_rResultPos.SetCol( aCellIter.GetCol()); o_rResultPos.SetRow( aCellIter.GetRow()); } return bFound; } bool ScInterpreter::LookupQueryWithCache( ScAddress & o_rResultPos, const ScQueryParam & rParam ) const { bool bFound = false; const ScQueryEntry& rEntry = rParam.GetEntry(0); bool bColumnsMatch = (rParam.nCol1 == rEntry.nField); OSL_ENSURE( bColumnsMatch, "ScInterpreter::LookupQueryWithCache: columns don't match"); // At least all volatile functions that generate indirect references have // to force non-cached lookup. /* TODO: We could further classify volatile functions into reference * generating and not reference generating functions to have to force less * direct lookups here. We could even further attribute volatility per * parameter so it would affect only the lookup range parameter. */ if (!bColumnsMatch || GetVolatileType() != NOT_VOLATILE) bFound = lcl_LookupQuery( o_rResultPos, pDok, rParam, rEntry); else { ScRange aLookupRange( rParam.nCol1, rParam.nRow1, rParam.nTab, rParam.nCol2, rParam.nRow2, rParam.nTab); ScLookupCache& rCache = pDok->GetLookupCache( aLookupRange); ScLookupCache::QueryCriteria aCriteria( rEntry); ScLookupCache::Result eCacheResult = rCache.lookup( o_rResultPos, aCriteria, aPos); switch (eCacheResult) { case ScLookupCache::NOT_CACHED : case ScLookupCache::CRITERIA_DIFFERENT : bFound = lcl_LookupQuery( o_rResultPos, pDok, rParam, rEntry); if (eCacheResult == ScLookupCache::NOT_CACHED) rCache.insert( o_rResultPos, aCriteria, aPos, bFound); break; case ScLookupCache::FOUND : bFound = true; break; case ScLookupCache::NOT_AVAILABLE : ; // nothing, bFound remains FALSE break; } } return bFound; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */