mirror of
https://github.com/OPM/ResInsight.git
synced 2025-01-08 07:03:25 -06:00
330 lines
10 KiB
C++
330 lines
10 KiB
C++
/////////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright (C) 2017 Statoil ASA
|
|
//
|
|
// ResInsight is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// ResInsight is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
// WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
// FITNESS FOR A PARTICULAR PURPOSE.
|
|
//
|
|
// See the GNU General Public License at <http://www.gnu.org/licenses/gpl.html>
|
|
// for more details.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
#include "RimCalculation.h"
|
|
|
|
#include "expressionparser/ExpressionParser.h"
|
|
|
|
#include "RiaLogging.h"
|
|
#include "RiaSummaryCurveDefinition.h"
|
|
|
|
#include "RimCalculationVariable.h"
|
|
#include "RimSummaryCurve.h"
|
|
|
|
#include "cafPdmUiTextEditor.h"
|
|
|
|
#include <algorithm>
|
|
#include <numeric>
|
|
|
|
|
|
CAF_PDM_SOURCE_INIT(RimCalculation, "RimCalculation");
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
RimCalculation::RimCalculation()
|
|
{
|
|
CAF_PDM_InitObject("RimCalculation", ":/octave.png", "Calculation", "");
|
|
|
|
CAF_PDM_InitFieldNoDefault(&m_description, "Description", "Description", "", "", "");
|
|
m_description.uiCapability()->setUiReadOnly(true);
|
|
|
|
CAF_PDM_InitFieldNoDefault(&m_expression, "Expression", "Expression", "", "", "");
|
|
m_expression.uiCapability()->setUiEditorTypeName(caf::PdmUiTextEditor::uiEditorTypeName());
|
|
|
|
CAF_PDM_InitFieldNoDefault(&m_variables, "Variables", "Variables", "", "", "");
|
|
CAF_PDM_InitFieldNoDefault(&m_calculatedValues, "CalculatedValues", "Calculated Values", "", "", "");
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
void RimCalculation::setDescription(const QString& description)
|
|
{
|
|
m_description = description;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
QString RimCalculation::description() const
|
|
{
|
|
return m_description;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
caf::PdmChildArrayFieldHandle* RimCalculation::variables()
|
|
{
|
|
return &m_variables;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
RimCalculationVariable* RimCalculation::addVariable()
|
|
{
|
|
RimCalculationVariable* v = new RimCalculationVariable;
|
|
m_variables.push_back(v);
|
|
|
|
return v;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
void RimCalculation::deleteVariable(RimCalculationVariable* calcVariable)
|
|
{
|
|
m_variables.removeChildObject(calcVariable);
|
|
|
|
delete calcVariable;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
const std::vector<double>& RimCalculation::values() const
|
|
{
|
|
return m_calculatedValues();
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
caf::PdmFieldHandle* RimCalculation::userDescriptionField()
|
|
{
|
|
return &m_description;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
bool RimCalculation::parseExpression()
|
|
{
|
|
// The expression parser handles only variables in lower case
|
|
QString lowerVariant = m_expression().toLower();
|
|
m_expression = lowerVariant;
|
|
|
|
QString leftHandSideVariableName = RimCalculation::findLeftHandSide(m_expression);
|
|
if (leftHandSideVariableName.isEmpty()) return false;
|
|
|
|
std::vector<QString> variableNames = ExpressionParser::detectReferencedVariables(m_expression);
|
|
|
|
// Remove variables not present in expression
|
|
{
|
|
std::vector<RimCalculationVariable*> toBeDeleted;
|
|
for (RimCalculationVariable* v : m_variables)
|
|
{
|
|
if (std::find(variableNames.begin(), variableNames.end(), v->name()) == variableNames.end())
|
|
{
|
|
toBeDeleted.push_back(v);
|
|
}
|
|
|
|
if (leftHandSideVariableName == v->name())
|
|
{
|
|
toBeDeleted.push_back(v);
|
|
}
|
|
}
|
|
|
|
for (RimCalculationVariable* v : toBeDeleted)
|
|
{
|
|
deleteVariable(v);
|
|
}
|
|
}
|
|
|
|
for (auto variableName : variableNames)
|
|
{
|
|
if (leftHandSideVariableName != variableName)
|
|
{
|
|
if (!findByName(variableName))
|
|
{
|
|
auto v = this->addVariable();
|
|
v->setName(variableName);
|
|
}
|
|
}
|
|
}
|
|
|
|
this->updateConnectedEditors();
|
|
|
|
return true;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
bool RimCalculation::calculate()
|
|
{
|
|
if (!parseExpression()) return false;
|
|
|
|
QString leftHandSideVariableName = RimCalculation::findLeftHandSide(m_expression);
|
|
|
|
ExpressionParser parser;
|
|
|
|
std::vector<std::vector<double>> variableValues;
|
|
variableValues.resize(m_variables.size());
|
|
|
|
size_t itemCount = 0;
|
|
|
|
for (size_t i = 0; i < m_variables.size(); i++)
|
|
{
|
|
RimCalculationVariable* v = m_variables[i];
|
|
|
|
if (!v->summaryCase())
|
|
{
|
|
RiaLogging::error("No summary case defined.");
|
|
|
|
return false;
|
|
}
|
|
|
|
if (!v->summaryAddress())
|
|
{
|
|
RiaLogging::error("No summary address defined.");
|
|
|
|
return false;
|
|
}
|
|
|
|
RimSummaryAddress* sumAdr = v->summaryAddress();
|
|
RiaSummaryCurveDefinition curveDef(v->summaryCase(), v->summaryAddress()->address());
|
|
|
|
std::vector<double>& curveValues = variableValues[i];
|
|
RiaSummaryCurveDefinition::resultValues(curveDef, &curveValues);
|
|
|
|
if (itemCount == 0)
|
|
{
|
|
itemCount = curveValues.size();
|
|
}
|
|
else
|
|
{
|
|
if (itemCount != curveValues.size())
|
|
{
|
|
RiaLogging::error("Not able to evaluate expression varying vector size.");
|
|
|
|
return false;
|
|
}
|
|
}
|
|
|
|
parser.assignVector(v->name(), curveValues);
|
|
}
|
|
|
|
if (itemCount == 0)
|
|
{
|
|
RiaLogging::error("Not able to evaluate expression with no data.");
|
|
|
|
return false;
|
|
}
|
|
|
|
m_calculatedValues.v().resize(itemCount);
|
|
parser.assignVector(leftHandSideVariableName, m_calculatedValues.v());
|
|
|
|
QString errorText;
|
|
bool evaluatedOk = parser.evaluate(m_expression, &errorText);
|
|
|
|
if (evaluatedOk)
|
|
{
|
|
QString txt;
|
|
for (auto v : m_calculatedValues())
|
|
{
|
|
txt += QString::number(v);
|
|
txt += " ";
|
|
}
|
|
|
|
RiaLogging::info(txt);
|
|
}
|
|
else
|
|
{
|
|
RiaLogging::error(errorText);
|
|
}
|
|
|
|
return evaluatedOk;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
/// Find the last assignment using := and interpret the text before the := as LHS
|
|
//--------------------------------------------------------------------------------------------------
|
|
QString RimCalculation::findLeftHandSide(const QString& expresion)
|
|
{
|
|
QString exprWithSpace = expresion;
|
|
exprWithSpace.replace("\n", " ");
|
|
|
|
QStringList words = exprWithSpace.split(" ", QString::SkipEmptyParts);
|
|
|
|
int index = words.lastIndexOf(":=");
|
|
if (index > 0)
|
|
{
|
|
return words[index - 1];
|
|
}
|
|
|
|
return "";
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
RimCalculationVariable* RimCalculation::findByName(const QString& name) const
|
|
{
|
|
for (RimCalculationVariable* v : m_variables)
|
|
{
|
|
if (v->name() == name)
|
|
{
|
|
return v;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
QString RimCalculation::buildCalculationName() const
|
|
{
|
|
QString name = "Default Calculation Name";
|
|
|
|
QString lhs = RimCalculation::findLeftHandSide(m_expression);
|
|
if (!lhs.isEmpty())
|
|
{
|
|
name = lhs;
|
|
|
|
name += " ( ";
|
|
|
|
for (RimCalculationVariable* v : m_variables)
|
|
{
|
|
name += v->summaryAddressDisplayString();
|
|
|
|
if (v != m_variables[m_variables.size() - 1])
|
|
{
|
|
name += ", ";
|
|
}
|
|
}
|
|
|
|
name += " )";
|
|
}
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
///
|
|
//--------------------------------------------------------------------------------------------------
|
|
void RimCalculation::defineUiOrdering(QString uiConfigName, caf::PdmUiOrdering& uiOrdering)
|
|
{
|
|
m_description = buildCalculationName();
|
|
}
|