/************************************************************************ ** ** @file vformula.cpp ** @author Roman Telezhynskyi ** @date 28 8, 2014 ** ** @brief ** @copyright ** This source code is part of the Valentina project, a pattern making ** program, whose allow create and modeling patterns of clothing. ** Copyright (C) 2013-2015 Valentina project ** All Rights Reserved. ** ** Valentina 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. ** ** Valentina 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 for more details. ** ** You should have received a copy of the GNU General Public License ** along with Valentina. If not, see . ** *************************************************************************/ #include "vformula.h" #include #include #include #include #include "../qmuparser/qmuparsererror.h" #include "../vgeometry/../ifc/ifcdef.h" #include "../vmisc/def.h" #include "../vmisc/vabstractapplication.h" #include "../vmisc/vcommonsettings.h" #include "calculator.h" #include "vcontainer.h" #include "vtranslatevars.h" //VFormula //--------------------------------------------------------------------------------------------------------------------- VFormula::VFormula() : formula(QString()), value(tr("Error")), checkZero(true), data(nullptr), toolId(NULL_ID), postfix(QString()), _error(true), dValue(0) {} //--------------------------------------------------------------------------------------------------------------------- VFormula::VFormula(const QString &formula, const VContainer *container) : formula(qApp->TrVars()->FormulaToUser(formula, qApp->Settings()->GetOsSeparator())), value(tr("Error")), checkZero(true), data(container), toolId(NULL_ID), postfix(QString()), _error(true), dValue(0) {} //--------------------------------------------------------------------------------------------------------------------- VFormula &VFormula::operator=(const VFormula &formula) { if ( &formula == this ) { return *this; } this->formula = formula.formula; this->value = formula.value; this->checkZero = formula.checkZero; this->data = formula.data; this->toolId = formula.toolId; this->postfix = formula.postfix; this->_error = formula._error; this->dValue = formula.dValue; return *this; } //--------------------------------------------------------------------------------------------------------------------- VFormula::VFormula(const VFormula &formula) : formula(formula.formula), value(formula.value), checkZero(formula.checkZero), data(formula.getData()), toolId(formula.toolId), postfix(formula.postfix), _error(formula._error), dValue(formula.dValue) {} //--------------------------------------------------------------------------------------------------------------------- bool VFormula::operator==(const VFormula &formula) const { bool isEqual = false; if (this->formula == formula.GetFormula() && this->value == formula.getStringValue() && this->checkZero == formula.getCheckZero() && this->data == formula.getData() && this->toolId == formula.getToolId() && this->postfix == formula.getPostfix() && this->_error == formula.error() && VFuzzyComparePossibleNulls(this->dValue, formula.getDoubleValue())) { isEqual = true; } return isEqual; } //--------------------------------------------------------------------------------------------------------------------- bool VFormula::operator!=(const VFormula &formula) const { return !VFormula::operator==(formula); } //--------------------------------------------------------------------------------------------------------------------- QString VFormula::GetFormula(FormulaType type) const { if (type == FormulaType::ToUser) { return formula; } else { return qApp->TrVars()->TryFormulaFromUser(formula, qApp->Settings()->GetOsSeparator()); } } //--------------------------------------------------------------------------------------------------------------------- void VFormula::SetFormula(const QString &value, FormulaType type) { if (formula != value) { if (type == FormulaType::ToUser) { formula = qApp->TrVars()->FormulaToUser(value, qApp->Settings()->GetOsSeparator()); } else { formula = value; } } } //--------------------------------------------------------------------------------------------------------------------- QString VFormula::getStringValue() const { return value; } //--------------------------------------------------------------------------------------------------------------------- qreal VFormula::getDoubleValue() const { return dValue; } //--------------------------------------------------------------------------------------------------------------------- bool VFormula::getCheckZero() const { return checkZero; } //--------------------------------------------------------------------------------------------------------------------- void VFormula::setCheckZero(bool value) { if (checkZero != value) { checkZero = value; } } //--------------------------------------------------------------------------------------------------------------------- const VContainer *VFormula::getData() const { return data; } //--------------------------------------------------------------------------------------------------------------------- void VFormula::setData(const VContainer *value) { if (data != value && value != nullptr) { data = value; } } //--------------------------------------------------------------------------------------------------------------------- quint32 VFormula::getToolId() const { return toolId; } //--------------------------------------------------------------------------------------------------------------------- void VFormula::setToolId(const quint32 &value) { toolId = value; } //--------------------------------------------------------------------------------------------------------------------- QString VFormula::getPostfix() const { return postfix; } //--------------------------------------------------------------------------------------------------------------------- void VFormula::setPostfix(const QString &value) { if (postfix != value) { postfix = value; } } //--------------------------------------------------------------------------------------------------------------------- bool VFormula::error() const { return _error; } //--------------------------------------------------------------------------------------------------------------------- int VFormula::FormulaTypeId() { return qMetaTypeId(); } //--------------------------------------------------------------------------------------------------------------------- void VFormula::Eval() { if (data == nullptr) { return; } value = tr("Error"); _error = true; dValue = 0; qreal result = 0; if (not formula.isEmpty()) { try { QScopedPointer cal(new Calculator()); const QString expression = qApp->TrVars()->FormulaFromUser(formula, qApp->Settings()->GetOsSeparator()); result = cal->EvalFormula(data->DataVariables(), expression); } catch (qmu::QmuParserError &e) { qDebug() << "\nMath parser error:\n" << "--------------------------------------\n" << "Message: " << e.GetMsg() << "\n" << "Expression: " << e.GetExpr() << "\n" << "--------------------------------------"; return; } if (not qIsInf(result) && not qIsNaN(result) && not (checkZero && qFuzzyIsNull(result))) { dValue = result; value = qApp->LocaleToString(result) + QLatin1Char(' ') + postfix; _error = false; } } }