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 |