Subversion Repositories public iLand

Rev

Rev 1104 | Rev 1217 | Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | RSS feed

/********************************************************************************************
**    iLand - an individual based forest landscape and disturbance model
**    http://iland.boku.ac.at
**    Copyright (C) 2009-  Werner Rammer, Rupert Seidl
**
**    This program 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.
**
**    This program 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 this program.  If not, see <http://www.gnu.org/licenses/>.
********************************************************************************************/

#include "abe_global.h"
#include "global.h"

#include "fmunit.h"

#include "forestmanagementengine.h"
#include "fmstand.h"
#include "scheduler.h"
#include "agent.h"
#include "agenttype.h"
#include "fomescript.h"
#include "fmtreelist.h"

namespace ABE {

/** @class FMUnit
    @ingroup abe
    The FMUnit class encapsulates a forest management unit, comprised of forest stands. Units are the base level at which
    the scheduling works.

  */



void FMUnit::aggregate()
{
    // loop over all stands
    // collect some data....
    double age=0.;
    double volume = 0.;
    double harvest = 0.;
    double totalarea = 0.;
    const QMultiMap<FMUnit*, FMStand*> &stands = ForestManagementEngine::instance()->stands();
    QMultiMap<FMUnit*, FMStand*>::const_iterator it = stands.constFind(this);
    while (it != stands.constEnd() && it.key()==this) {
        const FMStand *s = it.value();
        age += s->age() * s->area();
        volume += s->volume() * s->area();

        totalarea += s->area();
        ++it;
    }
    if (totalarea>0.) {
        age /= totalarea;
        volume /= totalarea;
        harvest /= totalarea;
    }
    qCDebug(abe) << "unit" << id() << "volume (m3/ha)" << volume << "age" << age << "planned harvest: todo";

}

QStringList FMUnit::info() const
{
    return QStringList() << QString("(accumulated) harvest: %1").arg(mRealizedHarvest)
                         << QString("MAI: %1").arg(mMAI)
                         << QString("HDZ: %1").arg(mHDZ)
                         << QString("average age: %1").arg(mMeanAge)
                         << QString("decadal plan: %1").arg(mAnnualHarvestTarget)
                         << QString("current plan: %1").arg(constScheduler()!=0?constScheduler()->harvestTarget():0.);

}

double FMUnit::annualThinningHarvest() const
{
    const QMultiMap<FMUnit*, FMStand*> &stands = ForestManagementEngine::instance()->stands();
    QMultiMap<FMUnit*, FMStand*>::const_iterator it = stands.constFind(const_cast<FMUnit*>(this));
    double harvested=0.;
    while (it != stands.constEnd() && it.key()==this) {
        FMStand *stand = it.value();
        harvested += stand->totalThinningHarvest();
        ++it;
    }
    return harvested;
}

FMUnit::FMUnit(const Agent *agent)
{
    mAgent = agent;
    mScheduler = 0;
    mAnnualHarvestTarget = -1.;
    mRealizedHarvest = 0.;
    mMAI = 0.; mHDZ = 0.; mMeanAge = 0.;
    mTotalArea = 0.; mTotalPlanDeviation = 0.;
    mTotalVolume = 0.;
    mAnnualHarvest = 0.;
    mNumberOfStands = 0;
    mU = 100, mThinningIntensityClass = 2, mSpeciesCompositionIndex = 0;
    mAverageMAI = 0.;


    //if (agent->type()->schedulerOptions().useScheduler)
    // explicit scheduler only for stands/units that include more than one stand
    mScheduler = new Scheduler(this);
}

FMUnit::~FMUnit()
{
    if (mScheduler)
        delete mScheduler;
}

void FMUnit::setId(const QString &id)
{
    mId = id;
}

void FMUnit::resetHarvestCounter()
{
    if (scheduler())
        scheduler()->resetHarvestCounter();
}

void FMUnit::managementPlanUpdate()
{
    const double period_length = 10.;
    // calculate the planned harvest in the next planning period (i.e., 10yrs).
    // this is the sum of planned operations that are already in the scheduler.
    double plan_thinning, plan_final;
    mScheduler->plannedHarvests(plan_final, plan_thinning);
    // the actual harvests of the last planning period
    //double realized = mRealizedHarvest;

    mRealizedHarvest = 0.; // reset
    mRealizedHarvestLastYear = 0.;


    // preparations:
    // MAI-calculation for all stands:
    double total_area = 0.;
    double age = 0.;
    double mai = 0.;
    double hdz = 0.;
    double volume = 0.;
    const QMultiMap<FMUnit*, FMStand*> &stands = ForestManagementEngine::instance()->stands();
    QMultiMap<FMUnit*, FMStand*>::const_iterator it = stands.constFind(this);
    while (it != stands.constEnd() && it.key()==this) {
        FMStand *stand = it.value();
        stand->reload();
        stand->calculateMAI();
        // calculate sustainable total harvest (following Breymann)
        double area = stand->area();
        mai += stand->meanAnnualIncrementTotal() * area; // m3/yr
        age += stand->absoluteAge() * area;
        volume += stand->volume() * area;
        // HDZ: "haubarer" average increment: timber that is ready for final harvest
        if (stand->readyForFinalHarvest())
            hdz += stand->volume() / stand->absoluteAge() * area; //(0.1* stand->U()) * area; // note: changed!!!! was: volume/age * area
        total_area += area;
        ++it;
    }
    // reset
    ForestManagementEngine::instance()->scriptBridge()->treesObj()->setStand(0);
    mTotalArea = total_area;
    if (total_area==0.)
        return;

    mai /= total_area; // m3/ha*yr area weighted average of annual increment
    age /= total_area; // area weighted mean age
    hdz /= total_area; // =sum(Vol/age * share)

    mMAI = mai;
    //mMAI = mMAI * 1.15; // 15% increase, hack WR
    mHDZ = hdz;
    mMeanAge = age;
    mTotalVolume = volume;

    double rotation_length = U();
    double h_tot = mai * 2.*age / rotation_length;  //
    double h_reg = hdz * 2.*age / rotation_length;
    h_reg = h_tot * 0.85; // hack!
    h_reg *= agent()->schedulerOptions().harvestIntensity;
    h_tot *= agent()->schedulerOptions().harvestIntensity;
    double h_thi = qMax(h_tot - h_reg, 0.);

    qCDebug(abe) << "plan-update for unit" << id() << ": h-tot:" << h_tot << "h_reg:" << h_reg << "h_thi:" << h_thi << "of total volume:" << volume;
    double sf = mAgent->useSustainableHarvest();
    // we do not calculate sustainable harvest levels.
    // do a pure bottom up calculation
    double bottom_up_harvest = (plan_final / period_length) / total_area; // m3/ha*yr

    // the sustainable harvest yield is the current yield and some carry over from the last period
    double sustainable_harvest = h_reg;
//    if (mAnnualHarvestTarget>0.) {
//        double delta = realized/(total_area*period_length) - mAnnualHarvestTarget;
//        // if delta > 0: timber removal was too high -> plan less for the current period, and vice versa.
//        sustainable_harvest -= delta;
//    }
    mAnnualHarvestTarget = sustainable_harvest * sf + bottom_up_harvest * (1.-sf);
    mAnnualHarvestTarget = qMax(mAnnualHarvestTarget, 0.);

    mAnnualThinningTarget = (plan_thinning / period_length) / total_area; // m3/ha*yr


    if (scheduler())
        scheduler()->setHarvestTarget(mAnnualHarvestTarget, mAnnualThinningTarget);
}

QMutex _protect_agent_exec;
void FMUnit::runAgent()
{
    QMutexLocker m(&_protect_agent_exec); // avoid parallel execution of agent-code....

    // we need to set an execution context
    FMStand *stand = *ForestManagementEngine::instance()->stands().find(this);
    if (!stand)
        throw IException("Invalid stand in FMUnit::runAgent");

    FomeScript::setExecutionContext(stand, true); // true: add also agent as 'agent'

    QJSValue val;
    QJSValue agent_type = agent()->type()->jsObject();
    if (agent_type.property("run").isCallable()) {
        val = agent_type.property("run").callWithInstance(agent_type);
        qCDebug(abe) << "running agent-function 'run' for unit" <<  id() << ":" << val.toString();
    } else {
       qCDebug(abe) << "function 'run' is not a valid function of agent-type" << agent()->type()->name();
    }

}

void FMUnit::updatePlanOfCurrentYear()
{
    if (!scheduler())
        return;

    if (mTotalArea==0.)
        throw IException("FMUnit:updatePlan: unit area = 0???");

    // compare the harvests of the last year to the plan:
    double harvests = mRealizedHarvest - mRealizedHarvestLastYear;
    mRealizedHarvestLastYear = mRealizedHarvest;
    mAnnualHarvest = harvests;

    // difference in m3/ha
    double delta = harvests/mTotalArea - mAnnualHarvestTarget;
    mTotalPlanDeviation += delta;

    // apply decay function for deviation
    mTotalPlanDeviation *= mAgent->schedulerOptions().deviationDecayRate;

    // relative deviation: >0: too many harvests
    double rel_deviation = mAnnualHarvestTarget? mTotalPlanDeviation / mAnnualHarvestTarget : 0;

    // the current deviation is reduced to 50% in rebounce_yrs years.
    double rebounce_yrs = mAgent->schedulerOptions().scheduleRebounceDuration;

    double new_harvest = mAnnualHarvestTarget * (1. - rel_deviation/rebounce_yrs);

    // limit to minimum/maximum parameter
    new_harvest = qMax(new_harvest, mAgent->schedulerOptions().minScheduleHarvest);
    new_harvest = qMin(new_harvest, mAgent->schedulerOptions().maxScheduleHarvest);
    scheduler()->setHarvestTarget(new_harvest, mAnnualThinningTarget);

}

} // namesapce