(root)/src/output/dynamicstandout.cpp - Rev 207
Rev 189 |
Rev 218 |
Go to most recent revision |
Blame |
Compare with Previous |
Last modification |
View Log
| RSS feed
#include "dynamicstandout.h"
#include "helper.h"
#include "model.h"
#include "resourceunit.h"
#include "species.h"
#include "expressionwrapper.h"
DynamicStandOut::DynamicStandOut()
{
setName("dynamic stand output by species/RU", "dynamicstand");
setDescription("Userdefined outputs for tree aggregates for each stand. see TreeAggregates");
columns() << OutputColumn::year() << OutputColumn::ru() << OutputColumn::species();
// other colums are added during setup...
}
const QStringList aggList = QStringList() << "mean" << "sum" << "min" << "max" << "p25" << "p50" << "p75";
void DynamicStandOut::setup()
{
QString filter = settings().value(".rufilter","");
QString fieldList = settings().value(".columns", "");
if (fieldList.isEmpty())
return;
mRUFilter.setExpression(filter);
// clear columns
columns().erase(columns().begin()+3, columns().end());
mFieldList.clear();
// setup fields
if (!fieldList.isEmpty()) {
QRegExp rx("((?:\\[.+\\]|\\w+)\\.\\w+)");
int pos=0;
QString field;
QStringList fieldParts;
QRegExp field_rx("\\[(.+)\\]\\.(\\w+)");
TreeWrapper tw;
while ((pos = rx.indexIn(fieldList, pos)) != -1) {
pos += rx.matchedLength();
field = rx.cap(1); // field
mFieldList.append(SDynamicField());
// parse field
if (field.count()>0 && field.at(0)!='[') {
// simple expression
fieldParts = field.split(QRegExp("\\W+"), QString::SkipEmptyParts);
mFieldList.back().var_index = tw.variableIndex(fieldParts.first());
} else {
// complex expression
field_rx.indexIn(field);
fieldParts = field_rx.capturedTexts();
fieldParts.pop_front();
mFieldList.back().var_index=-1;
mFieldList.back().expression = fieldParts.first();
}
if (fieldParts.count()!=2)
throw IException(QString("Invalid variable name for dynamic output:") + field);
mFieldList.back().agg_index = aggList.indexOf(fieldParts[1]);
if (mFieldList.back().agg_index==-1)
throw IException(QString("Invalid aggregate expression for dynamic output: %1\nallowed:%2")
.arg(fieldParts[1]).arg(aggList.join(" ")));
QString stripped_field=field;
stripped_field.replace(".", "_");
stripped_field.replace(QRegExp("[\\[\\]\\,\\(\\)<>=!\\s]"), "_");
stripped_field.replace("__", "_");
columns() << OutputColumn(stripped_field, field, OutDouble);
}
}
}
void DynamicStandOut::exec()
{
if (mFieldList.count()==0)
return;
DebugTimer t("dynamic stand output");
Model *m = GlobalSettings::instance()->model();
QVector<double> data; //statistics data
StatData stat; // statistcs helper class
TreeWrapper tw;
RUWrapper ruwrapper;
mRUFilter.setModelObject(&ruwrapper);
Expression custom_expr;
foreach(ResourceUnit *ru, m->ruList()) {
// test filter
if (!mRUFilter.isEmpty()) {
ruwrapper.setResourceUnit(ru);
if (!mRUFilter.execute())
continue;
}
foreach(const ResourceUnitSpecies &rus, ru->ruSpecies()) {
if (rus.constStatistics().count()==0)
continue;
*this << currentYear() << ru->index() << rus.species()->id(); // keys
// dynamic calculations
foreach (const SDynamicField &field, mFieldList) {
if (!field.expression.isEmpty()) {
// setup dynamic dynamic expression if present
custom_expr.setExpression(field.expression);
custom_expr.setModelObject(&tw);
}
data.clear();
foreach(const Tree &tree, ru->trees()) {
if (tree.species()->index()!=rus.species()->index())
continue;
tw.setTree(&tree);
if (field.var_index>=0)
data.push_back(tw.value(field.var_index));
else
data.push_back(custom_expr.execute());
}
// calculate statistics
stat.setData(data);
// aggregate
double value;
switch (field.agg_index) {
case 0: value = stat.mean(); break;
case 1: value = stat.sum(); break;
case 2: value = stat.min(); break;
case 3: value = stat.max(); break;
case 4: value = stat.percentile25(); break;
case 5: value = stat.median(); break;
case 6: value = stat.percentile75(); break;
default: value = 0.; break;
}
// add current value to output
*this << value;
} // foreach (field)
writeRow();
} //foreach species
} // foreach ressource unit
}