Subversion Repositories public iLand

Rev

Rev 1221 | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
1033 werner 1
/********************************************************************************************
2
**    iLand - an individual based forest landscape and disturbance model
3
**    http://iland.boku.ac.at
4
**    Copyright (C) 2009-  Werner Rammer, Rupert Seidl
5
**
6
**    This program is free software: you can redistribute it and/or modify
7
**    it under the terms of the GNU General Public License as published by
8
**    the Free Software Foundation, either version 3 of the License, or
9
**    (at your option) any later version.
10
**
11
**    This program is distributed in the hope that it will be useful,
12
**    but WITHOUT ANY WARRANTY; without even the implied warranty of
13
**    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14
**    GNU General Public License for more details.
15
**
16
**    You should have received a copy of the GNU General Public License
17
**    along with this program.  If not, see <http://www.gnu.org/licenses/>.
18
********************************************************************************************/
990 werner 19
#include "abe_global.h"
20
#include "actplanting.h"
21
 
22
#include "fmstp.h"
23
#include "fmstand.h"
24
#include "fomescript.h"
25
#include "fmtreelist.h"
26
#include "forestmanagementengine.h"
27
 
28
#include "model.h"
29
#include "mapgrid.h"
30
#include "resourceunit.h"
31
#include "resourceunitspecies.h"
1163 werner 32
 
990 werner 33
#include "debugtimer.h"
34
 
35
namespace ABE {
36
 
1095 werner 37
/** @class ActPlanting
38
    @ingroup abe
39
    The ActPlanting class implements artificial regeneration (i.e., planting of trees).
40
 
41
  */
42
 
43
 
990 werner 44
// Planting patterns
45
QVector<QPair<QString, int> > planting_patterns =QVector<QPair<QString, int> >()
46
<< QPair<QString, int>(
47
        "11"\
48
        "11", 2)
49
<< QPair<QString, int>("11111"\
50
                       "11111"\
51
                       "11111"\
52
                       "11111"\
53
                       "11111", 5)
54
<< QPair<QString, int>("1111111111"\
55
                       "1111111111"\
56
                       "1111111111"\
57
                       "1111111111"\
58
                       "1111111111"\
59
                       "1111111111"\
60
                       "1111111111"\
61
                       "1111111111"\
62
                       "1111111111"\
63
                       "1111111111", 10)
64
<< QPair<QString, int>("00110"\
65
                       "11110"\
66
                       "11111"\
67
                       "01111"\
68
                       "00110", 5)
69
<< QPair<QString, int>("0000110000"\
70
                       "0011111100"\
71
                       "0111111110"\
72
                       "0111111110"\
73
                       "1111111111"\
74
                       "1111111111"\
75
                       "0111111110"\
76
                       "0011111110"\
77
                       "0011111100"\
78
                       "0000110000", 10);
79
QStringList planting_pattern_names = QStringList() << "rect2" << "rect10" << "rect20" << "circle5" << "circle10";
80
 
81
QStringList ActPlanting::mAllowedProperties;
82
 
83
 
84
ActPlanting::ActPlanting(FMSTP *parent): Activity(parent)
85
{
86
    //mBaseActivity.setIsScheduled(true); // use the scheduler
87
    //mBaseActivity.setDoSimulate(true); // simulate per default
991 werner 88
    if (mAllowedProperties.isEmpty())
89
        mAllowedProperties = QStringList() << Activity::mAllowedProperties
90
                                           << "species" << "fraction" << "height" << "age" << "clear"
91
                                           << "pattern" << "spacing" << "offset" << "random" << "n";
990 werner 92
 
93
}
94
 
95
void ActPlanting::setup(QJSValue value)
96
{
1208 werner 97
    // check if regeneration is enabled
98
    if (GlobalSettings::instance()->model()->settings().regenerationEnabled == false)
99
        throw IException("Cannot set up planting acitivities when iLand regeneration module is disabled.");
990 werner 100
    Activity::setup(value); // setup base events
101
 
102
    QJSValue items = FMSTP::valueFromJs(value, "items");
103
    mItems.clear();
104
    // iterate over array or over single object
105
    if ((items.isArray() || items.isObject()) && !items.isCallable()) {
106
        QJSValueIterator it(items);
107
        while (it.hasNext()) {
108
            it.next();
109
            if (it.name()==QStringLiteral("length"))
110
                continue;
111
 
112
            mItems.append(SPlantingItem());
113
            SPlantingItem &item = mItems.last();
1063 werner 114
            qDebug() << it.name() << ": " << FomeScript::JStoString(it.value());
990 werner 115
            FMSTP::checkObjectProperties(it.value(), mAllowedProperties, "setup of planting activity:" + name() + "; " + it.name());
116
 
117
            item.setup(it.value());
118
        }
119
    } else {
120
        mItems.append(SPlantingItem());
121
        SPlantingItem &item = mItems.last();
122
        FMSTP::checkObjectProperties(items, mAllowedProperties, "setup of planting activity:" + name());
123
 
124
        item.setup(items);
125
    }
126
    mRequireLoading = false;
127
    for (QVector<SPlantingItem>::const_iterator it=mItems.constBegin(); it!=mItems.constEnd(); ++it) {
128
        if (it->clear == true)
129
            mRequireLoading = true;
130
    }
131
}
132
 
133
bool ActPlanting::execute(FMStand *stand)
134
{
135
    qCDebug(abe) << stand->context() << "execute of planting activity....";
136
    DebugTimer time("ABE:ActPlanting:execute");
137
 
138
    for (int s=0;s<mItems.count();++s) {
1203 werner 139
        mItems[s].run(stand);
990 werner 140
    }
141
 
142
 
143
    return true;
144
}
145
 
146
QStringList ActPlanting::info()
147
{
148
    QStringList lines = Activity::info();
149
 
150
    foreach(const SPlantingItem &item, mItems) {
151
        lines << "-";
152
        lines << QString("species: %1").arg(item.species->id());
153
        lines << QString("fraction: %1").arg(item.fraction);
154
        lines << QString("clear: %1").arg(item.clear);
155
        lines << QString("pattern: %1").arg(item.group_type>-1?planting_pattern_names[item.group_type]:"");
156
        lines << QString("spacing: %1").arg(item.spacing);
157
        lines << QString("offset: %1").arg(item.offset);
158
        lines << QString("random: %1").arg(item.group_random_count>0);
159
        lines << "/-";
160
    }
161
    return lines;
162
}
163
 
164
void ActPlanting::runSinglePlantingItem(FMStand *stand, QJSValue value)
165
{
166
    if (!stand) return;
167
    if (FMSTP::verbose())
168
        qCDebug(abe()) << "run Single Planting Item for Stand" << stand->id();
169
    DebugTimer time("ABE:runSinglePlantingItem");
170
    SPlantingItem item;
171
    item.setup(value);
172
 
1203 werner 173
    item.run(stand);
990 werner 174
 
175
}
176
 
177
 
178
 
179
 
180
bool ActPlanting::SPlantingItem::setup(QJSValue value)
181
{
182
 
183
    QString species_id = FMSTP::valueFromJs(value, "species",QString(), "setup of planting item for planting activity.").toString();
184
    species = GlobalSettings::instance()->model()->speciesSet()->species(species_id);
185
    if (!species)
186
        throw IException(QString("'%1' is not a valid species id for setting up a planting item.").arg(species_id));
187
    fraction = FMSTP::valueFromJs(value, "fraction", "0").toNumber();
188
    height = FMSTP::valueFromJs(value, "height", "0.05").toNumber();
1213 werner 189
    age = FMSTP::valueFromJs(value, "age", "1").toInt();
990 werner 190
    clear = FMSTP::valueFromJs(value, "clear", "false").toBool();
191
 
192
    // pattern
193
    QString group = FMSTP::valueFromJs(value, "pattern", "").toString();
194
    group_type = planting_pattern_names.indexOf(group);
195
    if (!group.isEmpty() && group!="undefined" && group_type==-1)
196
        throw IException(QString("Planting-activity: the pattern '%1' is not valid!").arg(group));
1213 werner 197
    spacing = FMSTP::valueFromJs(value, "spacing", "0").toInt();
198
    offset = FMSTP::valueFromJs(value, "offset", "0").toInt();
990 werner 199
 
200
    bool random = FMSTP::boolValueFromJs(value, "random", false);
201
    if (random)
1213 werner 202
        group_random_count = FMSTP::valueFromJs(value, "n", "0").toInt();
990 werner 203
    else
204
        group_random_count = 0;
205
 
206
    grouped = group_type >= 0;
207
    return true;
208
}
209
 
1203 werner 210
void ActPlanting::SPlantingItem::run(FMStand *stand)
990 werner 211
{
212
    QRectF box = ForestManagementEngine::instance()->standGrid()->boundingBox(stand->id());
213
    const MapGrid *sgrid = ForestManagementEngine::instance()->standGrid();
214
    Model *model = GlobalSettings::instance()->model();
215
    GridRunner<float> runner(model->grid(), box);
216
    if (!grouped) {
1203 werner 217
        // distribute saplings randomly.
218
        // this adds saplings to SaplingCell (only if enough slots are available)
1032 werner 219
        while (runner.next()) {
1203 werner 220
            if (sgrid->standIDFromLIFCoord(runner.currentIndex()) != stand->id())
990 werner 221
                continue;
222
            //
223
            if (drandom() < fraction) {
224
                ResourceUnit *ru = model->ru(runner.currentCoord());
1203 werner 225
                ru->saplingCell(runner.currentIndex())->addSapling(height, age, species->index());
990 werner 226
            }
227
        }
228
    } else {
1203 werner 229
        // grouped saplings
990 werner 230
        const QString &pp = planting_patterns[group_type].first;
231
        int n = planting_patterns[group_type].second;
232
 
233
        if (spacing==0 && group_random_count == 0) {
234
            // pattern based planting (filled)
235
            runner.reset();
236
            while (runner.next()) {
237
                QPoint qp = runner.currentIndex();
1203 werner 238
                if (sgrid->standIDFromLIFCoord(qp) != stand->id())
990 werner 239
                    continue;
1203 werner 240
                // check location in the pre-defined planting patterns
990 werner 241
                int idx = (qp.x()+offset)%n + n*((qp.y()+offset)%n);
242
                if (pp[idx]=='1') {
243
                    ResourceUnit *ru = model->ru(runner.currentCoord());
1203 werner 244
                    SaplingCell *sc = ru->saplingCell(qp);
990 werner 245
 
246
                    if (clear) {
1203 werner 247
                        // clear all sapling trees on the cell
248
                        model->saplings()->clearSaplings(sc,ru,true);
990 werner 249
                    }
1203 werner 250
                    sc->addSapling(height,age,species->index());
990 werner 251
                }
252
            }
253
        } else {
254
            // pattern based (with spacing / offset, random...)
255
            int ispacing = spacing / cPxSize;
256
            QPoint p = model->grid()->indexAt(box.topLeft())-QPoint(offset, offset);
257
            QPoint pstart = p;
258
            QPoint p_end = model->grid()->indexAt(box.bottomRight());
259
            QPoint po;
260
            p.setX(qMax(p.x(),0)); p.setY(qMax(p.y(),0));
261
 
262
            int n_ha = group_random_count * box.width()*box.height()/10000.;
263
            bool do_random = group_random_count>0;
264
 
265
            while( p.x() < p_end.x() && p.y() < p_end.y()) {
266
                if (do_random) {
267
                    // random position!
268
                    if (n_ha--<=0)
269
                        break;
270
                    // select a random position (2m grid index)
271
                    p = model->grid()->indexAt(QPointF( nrandom(box.left(), box.right()), nrandom(box.top(), box.bottom()) ));
272
                }
273
 
274
                // apply the pattern....
275
                for (int y=0;y<n;++y) {
276
                    for (int x=0;x<n;++x) {
277
                        po=p + QPoint(x,y);
1203 werner 278
                        if (sgrid->standIDFromLIFCoord(po) != stand->id())
990 werner 279
                            continue;
280
                        ResourceUnit *ru = model->ru(model->grid()->cellCenterPoint(po));
1203 werner 281
                        SaplingCell *sc = ru->saplingCell(po);
990 werner 282
 
283
                        if (clear) {
284
                            // clear all sapling trees
1203 werner 285
                            model->saplings()->clearSaplings(sc,ru,true);
990 werner 286
                        }
1203 werner 287
                        sc->addSapling(height,age,species->index());
990 werner 288
                    }
289
                }
290
                if (!do_random) {
291
                    // apply offset
292
                    p.rx() += ispacing;
293
                    if (p.x()>= p_end.x()) {
294
                        p.rx() = pstart.x();
295
                        p.ry() += ispacing;
296
                    }
297
                }
298
            }
299
        }
300
    }
301
 
302
 
303
}
304
 
305
 
306
 
307
} // namesapce