Rev 1221 | Details | Compare with Previous | Last modification | View Log | RSS feed
Rev | Author | Line No. | Line |
---|---|---|---|
671 | 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 | ********************************************************************************************/ |
||
19 | |||
247 | werner | 20 | #ifndef SCRIPTGLOBAL_H |
21 | #define SCRIPTGLOBAL_H |
||
22 | |||
23 | #include <QObject> |
||
827 | werner | 24 | #include <QVariant> |
794 | werner | 25 | #include <QJSValue> |
603 | werner | 26 | |
27 | // Scripting Interface for MapGrid |
||
28 | class MapGrid; // forward |
||
29 | class MapGridWrapper: public QObject |
||
30 | { |
||
31 | Q_OBJECT |
||
767 | werner | 32 | Q_PROPERTY(int valid READ isValid) |
33 | Q_PROPERTY(QString name READ name) |
||
603 | werner | 34 | public: |
35 | MapGridWrapper(QObject *parent=0); |
||
36 | ~MapGridWrapper(); |
||
793 | werner | 37 | static void addToScriptEngine(QJSEngine &engine); |
603 | werner | 38 | MapGrid *map() const { return mMap; } ///< acccess for C++ classes |
39 | bool isValid() const; ///< returns true if map is successfully loaded |
||
40 | QString name() const; |
||
41 | // |
||
42 | |||
43 | public slots: |
||
44 | // query |
||
45 | double area(int id); ///< retrieve the area (m2) of the polygon denoted by 'id' |
||
46 | // actions |
||
47 | void load(QString file_name); |
||
48 | void saveAsImage(QString file); |
||
803 | werner | 49 | void paint(double min_value, double max_value); |
848 | werner | 50 | // active modifications of the map |
51 | void clear(); ///< clears the map (set all values to 0) |
||
980 | werner | 52 | void clearProjectArea(); ///< clear the project area (set to 0), but copy mask with pixels from "outside of project area" (i.e., -1, -2) |
848 | werner | 53 | /// paint a shape on the stand grid with id stand_id |
54 | /// paint_function is a valid expression (paramters: x, y as *metric* coordinates) |
||
55 | /// if wrap_around=true, then the shape is wrapped around the edges of the simulated area (torus) |
||
56 | void createStand(int stand_id, QString paint_function, bool wrap_around); |
||
603 | werner | 57 | |
979 | werner | 58 | /// copy a bit of the source-map 'source' to this map. The |
59 | /// source rectangle is given by coordinates (x1/y1) to (x2/y2). |
||
60 | /// The rectangle will be blitted to the new coordinates destx/desty (moved from x1/y1). |
||
61 | /// id_in: the id of the polygon to copy, id: the id of the pixels in the target |
||
62 | /// return the size (ha) of the valid thing |
||
63 | double copyPolygonFromRect(MapGridWrapper *source, int id_in, int id, double destx, double desty, double x1, double y1, double x2, double y2); |
||
64 | |||
980 | werner | 65 | void createMapIndex(); ///< call after creating stands with copyPolygonFromRect |
66 | |||
603 | werner | 67 | private: |
68 | MapGrid *mMap; |
||
69 | bool mCreated; |
||
70 | |||
71 | }; |
||
72 | |||
73 | /** ScriptGlobal contains a set of |
||
74 | functions and properties that are accessible by JavaScript. |
||
75 | */ |
||
247 | werner | 76 | class Model; |
794 | werner | 77 | class ScriptGlobal : public QObject |
247 | werner | 78 | { |
79 | Q_OBJECT |
||
80 | // read only properties |
||
766 | werner | 81 | Q_PROPERTY(int year READ year) |
82 | Q_PROPERTY(int resourceUnitCount READ resourceUnitCount) |
||
83 | Q_PROPERTY(QString currentDir WRITE setCurrentDir READ currentDir) |
||
850 | werner | 84 | Q_PROPERTY(double worldX READ worldX) |
85 | Q_PROPERTY(double worldY READ worldY) |
||
853 | werner | 86 | Q_PROPERTY(bool qt5 READ qt5) |
979 | werner | 87 | Q_PROPERTY(int msec READ msec) |
1041 | werner | 88 | Q_PROPERTY(QJSValue viewOptions READ viewOptions WRITE setViewOptions) |
247 | werner | 89 | |
90 | |||
91 | public: |
||
92 | ScriptGlobal(QObject *parent=0); |
||
767 | werner | 93 | static void setupGlobalScripting(); |
247 | werner | 94 | // properties accesible by scripts |
853 | werner | 95 | bool qt5() const {return true; } ///< is this the qt5-model? (changes in script object creation) |
979 | werner | 96 | int msec() const; ///< the milliseconds since the start of the day |
247 | werner | 97 | int year() const; ///< current year in the model |
294 | werner | 98 | int resourceUnitCount() const; ///< get number of resource uinit |
247 | werner | 99 | QString currentDir() const { return mCurrentDir; } ///< current execution directory (default is the Script execution directory) |
100 | void setCurrentDir(QString newDir) { mCurrentDir = newDir; } ///< set current working dir |
||
850 | werner | 101 | double worldX(); ///< extent of the world (without buffer) in meters (x-direction) |
102 | double worldY(); ///< extent of the world (without buffer) in meters (y-direction) |
||
767 | werner | 103 | |
104 | // general functions |
||
105 | static void loadScript(const QString &fileName); |
||
106 | static QString executeScript(QString cmd); |
||
1157 | werner | 107 | static QString executeJSFunction(QString function); |
767 | werner | 108 | static QObject *scriptOutput; ///< public "pipe" for script output (is redirected to GUI if available) |
873 | werner | 109 | static QString formattedErrorMessage(const QJSValue &error_value, const QString &sourcecode); |
767 | werner | 110 | |
1041 | werner | 111 | // view options |
112 | /* View options: |
||
113 | * * type: {...} |
||
114 | * * species: bool |
||
115 | * * shade: bool |
||
116 | * |
||
117 | */ |
||
118 | QJSValue viewOptions(); ///< retrieve current viewing options (JS - object) |
||
119 | void setViewOptions(QJSValue opts); ///< set current view options |
||
120 | |||
247 | werner | 121 | public slots: |
294 | werner | 122 | // system stuff |
123 | QVariant setting(QString key); ///< get a value from the global xml-settings (returns undefined if not present) |
||
124 | void set(QString key, QString value); ///< set the value of a setting |
||
793 | werner | 125 | void print(QString message); ///< print the contents of the message to the log |
794 | werner | 126 | void alert(QString message); ///< shows a message box to the user (if in GUI mode) |
127 | void include(QString filename); ///< "include" the given script file and evaluate. The path is relative to the "script" path |
||
294 | werner | 128 | // file stuff |
589 | werner | 129 | QString defaultDirectory(QString dir); ///< get default directory of category 'dir' |
1077 | werner | 130 | QString path(QString filename); ///< get a path relative to the project main folder |
247 | werner | 131 | QString loadTextFile(QString fileName); ///< load content from a text file in a String (@sa CSVFile) |
132 | void saveTextFile(QString fileName, QString content); ///< save string (@p content) to a text file. |
||
133 | bool fileExists(QString fileName); ///< return true if the given file exists. |
||
1180 | werner | 134 | void systemCmd(QString command); ///< execute system command (e.g., copy files) |
294 | werner | 135 | // add trees |
393 | werner | 136 | int addSingleTrees(const int resourceIndex, QString content); ///< add single trees |
137 | int addTrees(const int resourceIndex, QString content); ///< add tree distribution |
||
603 | werner | 138 | int addTreesOnMap(const int standID, QString content); ///< add trees (distribution mode) for stand 'standID' |
600 | werner | 139 | // add saplings |
951 | werner | 140 | int addSaplingsOnMap(MapGridWrapper *map, const int mapID, QString species, int px_per_hectare, double height, int age); |
589 | werner | 141 | // enable/disable outputs |
142 | bool startOutput(QString table_name); ///< starts output 'table_name'. return true if successful |
||
143 | bool stopOutput(QString table_name); ///< stops output 'table_name'. return true if successful |
||
144 | // miscellaneous stuff |
||
634 | werner | 145 | void setViewport(double x, double y, double scale_px_per_m); ///< set the viewport of the main project area view |
589 | werner | 146 | bool screenshot(QString file_name); ///< make a screenshot from the central viewing widget |
716 | werner | 147 | void repaint(); ///< force a repainting of the GUI visualization (if available) |
599 | werner | 148 | bool gridToFile(QString grid_type, QString file_name); ///< create a "ESRI-grid" text file 'grid_type' is one of a fixed list of names, 'file_name' the ouptut file location |
1081 | werner | 149 | |
150 | /// return Javascript grid for given type |
||
151 | QJSValue grid(QString type); |
||
1091 | werner | 152 | /// return a grid with the basal area of the given species (resource unit resolution) |
153 | QJSValue speciesShareGrid(QString species); |
||
154 | /// return a grid (level of resource units) with the result of an expression evaluated in the context of the resource unit. |
||
155 | QJSValue resourceUnitGrid(QString expression); |
||
156 | |||
1064 | werner | 157 | // DOES NOT FULLY WORK |
158 | bool seedMapToFile(QString species, QString file_name); ///< save the "seedmap" (i.e. a grid showing the seed distribution) as ESRI rastser file |
||
716 | werner | 159 | void wait(int milliseconds); ///< wait for 'milliseconds' or (if ms=-1 until a key is pressed) |
675 | werner | 160 | // vegetation snapshots |
161 | bool saveModelSnapshot(QString file_name); |
||
162 | bool loadModelSnapshot(QString file_name); |
||
1213 | werner | 163 | bool saveStandSnapshot(int stand_id, QString file_name); |
164 | bool loadStandSnapshot(int stand_id, QString file_name); |
||
1058 | werner | 165 | // agent-based-model of forest management |
166 | void reloadABE(); |
||
1061 | werner | 167 | |
168 | // UI interface |
||
169 | void setUIshortcuts(QJSValue shortcuts); ///< set a list of JS shortcuts in the UI |
||
1071 | werner | 170 | |
171 | void test_tree_mortality(double thresh, int years, double p_death); |
||
247 | werner | 172 | private: |
793 | werner | 173 | void throwError(const QString &errormessage); |
247 | werner | 174 | QString mCurrentDir; |
175 | Model *mModel; |
||
176 | }; |
||
177 | |||
794 | werner | 178 | /** The ScriptObjectFactory can instantiate objects of other C++ (QObject-based) types. |
179 | * This factory approach is used because the V8 (QJSEngine) does not work with |
||
180 | * the "new" way of creating objects. |
||
181 | */ |
||
182 | class ScriptObjectFactory: public QObject |
||
183 | { |
||
184 | Q_OBJECT |
||
185 | public: |
||
186 | ScriptObjectFactory(QObject *parent=0); |
||
187 | public slots: |
||
188 | QJSValue newCSVFile(QString filename); ///< create a new instance of CSVFile and return it |
||
189 | QJSValue newClimateConverter(); ///< create new instance of ClimateConverter and return it |
||
803 | werner | 190 | QJSValue newMap(); ///< create new instance of Map and return it |
794 | werner | 191 | int stats() {return mObjCreated;} ///< return the number of created objects |
192 | private: |
||
193 | int mObjCreated; |
||
552 | werner | 194 | |
794 | werner | 195 | }; |
196 | |||
247 | werner | 197 | #endif // SCRIPTGLOBAL_H |