You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

168 lines
5.1 KiB

/*
SPDX-FileCopyrightText: 2010 Matteo Agostinelli <agostinelli@gmail.com>
SPDX-License-Identifier: LGPL-2.0-or-later
*/
#include "qalculate_engine.h"
#include <libqalculate/Calculator.h>
#include <libqalculate/ExpressionItem.h>
#include <libqalculate/Function.h>
#include <libqalculate/Prefix.h>
#include <libqalculate/Unit.h>
#include <libqalculate/Variable.h>
#include <QApplication>
#include <QClipboard>
#include <QDebug>
#include <QFile>
#include <KIO/Job>
#include <KLocalizedString>
#include <KProtocolManager>
QAtomicInt QalculateEngine::s_counter;
QalculateEngine::QalculateEngine(QObject *parent)
: QObject(parent)
{
s_counter.ref();
if (!CALCULATOR) {
new Calculator();
CALCULATOR->terminateThreads();
CALCULATOR->loadGlobalDefinitions();
CALCULATOR->loadLocalDefinitions();
CALCULATOR->loadGlobalCurrencies();
CALCULATOR->loadExchangeRates();
}
}
QalculateEngine::~QalculateEngine()
{
if (s_counter.deref()) {
delete CALCULATOR;
CALCULATOR = nullptr;
}
}
void QalculateEngine::updateExchangeRates()
{
QUrl source = QUrl("http://www.ecb.int/stats/eurofxref/eurofxref-daily.xml");
QUrl dest = QUrl::fromLocalFile(QFile::decodeName(CALCULATOR->getExchangeRatesFileName().c_str()));
KIO::Job *getJob = KIO::file_copy(source, dest, -1, KIO::Overwrite | KIO::HideProgressInfo);
connect(getJob, &KJob::result, this, &QalculateEngine::updateResult);
}
void QalculateEngine::updateResult(KJob *job)
{
if (job->error()) {
qDebug() << "The exchange rates could not be updated. The following error has been reported:" << job->errorString();
} else {
// the exchange rates have been successfully updated, now load them
CALCULATOR->loadExchangeRates();
}
}
#if QALCULATE_MAJOR_VERSION > 2 || QALCULATE_MINOR_VERSION > 6
bool has_error()
{
while (CALCULATOR->message()) {
if (CALCULATOR->message()->type() == MESSAGE_ERROR) {
CALCULATOR->clearMessages();
return true;
}
CALCULATOR->nextMessage();
}
return false;
}
bool check_valid_before(const std::string &expression, const EvaluationOptions &search_eo)
{
bool b_valid = false;
if (!b_valid)
b_valid = (expression.find_first_of(OPERATORS NUMBERS PARENTHESISS) != std::string::npos);
if (!b_valid)
b_valid = CALCULATOR->hasToExpression(expression, false, search_eo);
if (!b_valid) {
std::string str = expression;
CALCULATOR->parseSigns(str);
b_valid = (str.find_first_of(OPERATORS NUMBERS PARENTHESISS) != std::string::npos);
if (!b_valid) {
size_t i = str.find_first_of(SPACES);
MathStructure m;
if (!b_valid) {
CALCULATOR->parse(&m, str, search_eo.parse_options);
if (!has_error() && (m.isUnit() || m.isFunction() || (m.isVariable() && (i != std::string::npos || m.variable()->isKnown()))))
b_valid = true;
}
}
}
return b_valid;
}
#endif
QString QalculateEngine::evaluate(const QString &expression, bool *isApproximate)
{
if (expression.isEmpty()) {
return QString();
}
QString input = expression;
// Make sure to use toLocal8Bit, the expression can contain non-latin1 characters
QByteArray ba = input.replace(QChar(0xA3), "GBP").replace(QChar(0xA5), "JPY").replace('$', "USD").replace(QChar(0x20AC), "EUR").toLocal8Bit();
const char *ctext = ba.data();
CALCULATOR->terminateThreads();
EvaluationOptions eo;
eo.auto_post_conversion = POST_CONVERSION_BEST;
eo.keep_zero_units = false;
eo.parse_options.angle_unit = ANGLE_UNIT_RADIANS;
eo.structuring = STRUCTURING_SIMPLIFY;
// suggested in https://github.com/Qalculate/libqalculate/issues/16
// to avoid memory overflow for seemingly innocent calculations (Bug 277011)
eo.approximation = APPROXIMATION_APPROXIMATE;
#if QALCULATE_MAJOR_VERSION > 2 || QALCULATE_MINOR_VERSION > 6
if (!check_valid_before(expression.toStdString(), eo)) {
return QString(); // See https://github.com/Qalculate/libqalculate/issues/442
}
#endif
CALCULATOR->setPrecision(16);
MathStructure result = CALCULATOR->calculate(ctext, eo);
PrintOptions po;
po.number_fraction_format = FRACTION_DECIMAL;
po.indicate_infinite_series = false;
po.use_all_prefixes = false;
po.use_denominator_prefix = true;
po.negative_exponents = false;
po.lower_case_e = true;
po.base_display = BASE_DISPLAY_NORMAL;
#if defined(QALCULATE_MAJOR_VERSION) && defined(QALCULATE_MINOR_VERSION) \
&& (QALCULATE_MAJOR_VERSION > 2 || (QALCULATE_MAJOR_VERSION == 2 && QALCULATE_MINOR_VERSION >= 2))
po.interval_display = INTERVAL_DISPLAY_SIGNIFICANT_DIGITS;
#endif
result.format(po);
m_lastResult = result.print(po).c_str();
if (isApproximate) {
*isApproximate = result.isApproximate();
}
return m_lastResult;
}
void QalculateEngine::copyToClipboard(bool flag)
{
Q_UNUSED(flag);
QApplication::clipboard()->setText(m_lastResult);
}