/* -*- 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/.
*/
#include <memory>
#include <simpleformulacalc.hxx>
#include <document.hxx>
#include <docsh.hxx>
#include <tokenarray.hxx>
#include <interpre.hxx>
#include <compiler.hxx>
#include <sfx2/linkmgr.hxx>
#define DISPLAY_LEN 66
ScSimpleFormulaCalculator::ScSimpleFormulaCalculator( ScDocument& rDoc, const ScAddress& rAddr,
const OUString& rFormula, bool bMatrixFormula, formula::FormulaGrammar::Grammar eGram )
: mnFormatType(SvNumFormatType::ALL)
, mbCalculated(false)
, maAddr(rAddr)
, mrDoc(rDoc)
, maGram(eGram)
, mbMatrixResult(false)
, mbLimitString(false)
, mbMatrixFormula(bMatrixFormula)
{
// compile already here
ScCompiler aComp(mrDoc, maAddr, eGram, true, bMatrixFormula);
mpCode = aComp.CompileString(rFormula);
if(mpCode->GetCodeError() == FormulaError::NONE && mpCode->GetLen())
aComp.CompileTokenArray();
}
ScSimpleFormulaCalculator::~ScSimpleFormulaCalculator()
{
}
void ScSimpleFormulaCalculator::Calculate()
{
if(mbCalculated)
return;
mbCalculated = true;
ScInterpreter aInt(mrDoc.GetFormulaCell( maAddr ), mrDoc, mrDoc.GetNonThreadedContext(), maAddr, *mpCode);
if (mbMatrixFormula)
aInt.AssertFormulaMatrix();
sfx2::LinkManager aNewLinkMgr( mrDoc.GetDocumentShell() );
aInt.SetLinkManager( &aNewLinkMgr );
formula::StackVar aIntType = aInt.Interpret();
if ( aIntType == formula::svMatrixCell )
{
ScCompiler aComp(mrDoc, maAddr, maGram);
OUStringBuffer aStr;
aComp.CreateStringFromToken(aStr, aInt.GetResultToken().get());
mbMatrixResult = true;
if (mbLimitString)
{
const sal_Unicode cCol = ScCompiler::GetNativeSymbol(ocArrayColSep)[0];
const sal_Unicode cRow = ScCompiler::GetNativeSymbol(ocArrayRowSep)[0];
const sal_Int32 n = aStr.getLength();
for (sal_Int32 i = DISPLAY_LEN; i < n; ++i)
{
const sal_Unicode c = aStr[i];
if (c == cCol || c == cRow)
{
aStr.truncate(i+1);
aStr.append("...");
break;
}
}
}
maMatrixFormulaResult = aStr.makeStringAndClear();
}
mnFormatType = aInt.GetRetFormatType();
maResult.SetToken(aInt.GetResultToken().get());
}
bool ScSimpleFormulaCalculator::IsValue()
{
Calculate();
if (mbMatrixResult)
return false;
return maResult.IsValue();
}
bool ScSimpleFormulaCalculator::IsMatrix()
{
Calculate();
return mbMatrixResult;
}
FormulaError ScSimpleFormulaCalculator::GetErrCode()
{
Calculate();
FormulaError nErr = mpCode->GetCodeError();
if (nErr != FormulaError::NONE)
return nErr;
return maResult.GetResultError();
}
double ScSimpleFormulaCalculator::GetValue()
{
Calculate();
if ((mpCode->GetCodeError() == FormulaError::NONE) &&
maResult.GetResultError() == FormulaError::NONE)
return maResult.GetDouble();
return 0.0;
}
svl::SharedString ScSimpleFormulaCalculator::GetString()
{
Calculate();
if (mbMatrixResult)
return svl::SharedString( maMatrixFormulaResult); // string not interned
if ((mpCode->GetCodeError() == FormulaError::NONE) &&
maResult.GetResultError() == FormulaError::NONE)
return maResult.GetString();
return svl::SharedString::getEmptyString();
}
bool ScSimpleFormulaCalculator::HasColRowName() const
{
return formula::FormulaTokenArrayPlainIterator(*mpCode).GetNextColRowName() != nullptr;
}
ScTokenArray* ScSimpleFormulaCalculator::GetCode()
{
return mpCode.get();
}
void ScSimpleFormulaCalculator::SetLimitString(bool bLimitString)
{
mbLimitString = bLimitString;
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
↑ V530 The return value of function 'truncate' is required to be utilized.