TDME2 1.9.121
Terrain2.h
Go to the documentation of this file.
1#pragma once
2
3#include <tdme/tdme.h>
5
6#include <unordered_map>
7#include <unordered_set>
8#include <vector>
9
14#include <tdme/math/Vector2.h>
15#include <tdme/math/Vector3.h>
16
17using std::unordered_map;
18using std::unordered_set;
19using std::vector;
20
27
28/**
29 * Terrain 2 utility
30 * @author Andreas Drewke
31 */
33{
34public:
35 static constexpr float STEP_SIZE { 1.0f };
36 static constexpr float PARTITION_SIZE { 64.0f };
37
38 struct FoliageBrush {
42 };
43
46 float count;
54 float scaleMin;
55 float scaleMax;
56 float heightMin;
57 float heightMax;
58 float slopeMin;
59 float slopeMax;
60 };
61
62private:
63 /**
64 * Get the terrain vertex for given x and z position without providing y component
65 * @param x x
66 * @param z z
67 * @return terrain vertex
68 */
69 inline static void getTerrainVertex(int x, int z, Vector3& vertex) {
70 vertex.set(
71 static_cast<float>(x) * STEP_SIZE,
72 0.0f,
73 static_cast<float>(z) * STEP_SIZE
74 );
75 }
76 /**
77 * Get the terrain vertex for given x and z position
78 * @param terrainHeightVector terrain height vector
79 * @param verticesPerX vertices per x
80 * @param verticesPerZ vertices per z
81 * @param x x
82 * @param z z
83 * @return terrain vertex
84 */
85 inline static bool getTerrainVertex(const vector<float>& terrainHeightVector, int verticesPerX, int verticesPerZ, int x, int z, Vector3& vertex) {
86 vertex.set(
87 static_cast<float>(x) * STEP_SIZE,
88 0.0f,
89 static_cast<float>(z) * STEP_SIZE
90 );
91 if (x < 0 || x >= verticesPerX) return false;
92 if (z < 0 || z >= verticesPerZ) return false;
93 vertex.set(
94 static_cast<float>(x) * STEP_SIZE,
95 terrainHeightVector[z * verticesPerX + x],
96 static_cast<float>(z) * STEP_SIZE
97 );
98 return true;
99 }
100
101 /**
102 * Get the terrain vertex for given x and z position
103 * @param x x
104 * @param z z
105 * @param waterHeight water height
106 * @param vertex vertex
107 */
108 inline static void getWaterVertex(int x, int z, float waterHeight, Vector3& vertex) {
109 vertex.set(
110 static_cast<float>(x) * STEP_SIZE,
111 waterHeight,
112 static_cast<float>(z) * STEP_SIZE
113 );
114 }
115
116 /**
117 * @returns if water includes given position
118 * @param waterPositionSet water position set
119 * @param x x
120 * @param z z
121 */
122 inline static bool hasWaterPosition(const unordered_map<int, unordered_set<int>>& waterPositionSet, int x, int z) {
123 auto waterPositionSetIt = waterPositionSet.find(z);
124 if (waterPositionSetIt == waterPositionSet.end()) return false;
125 auto waterXPositionSetIt = waterPositionSetIt->second.find(x);
126 if (waterXPositionSetIt == waterPositionSetIt->second.end()) return false;
127 return true;
128 }
129
130 /**
131 * Compute terrain vertex normal for given x and z position
132 * @param terrainHeightVector terrain height vector
133 * @param verticesPerX vertices per x
134 * @param verticesPerZ vertices per z
135 * @param x x
136 * @param z z
137 * @return normal for given vertex index
138 */
139 static const Vector3 computeTerrainVertexNormal(const vector<float>& terrainHeightVector, int verticesPerX, int verticesPerZ, int x, int z);
140
141 /**
142 * Determine if water can be generated
143 * @param terrainHeightVector terrain height vector
144 * @param verticesPerX vertices per x
145 * @param verticesPerZ vertices per z
146 * @param x x
147 * @param z z
148 * @param waterHeight water height
149 * @return if water can be generated at given position
150 */
151 inline static bool determineWater(const vector<float>& terrainHeightVector, int verticesPerX, int verticesPerZ, int x, int z, float waterHeight) {
152 if (x < 0 || x >= verticesPerX ||
153 z < 0 || z >= verticesPerZ) return false;
154 auto vertexIdx = z * verticesPerX + x;
155 auto terrainVertexHeight = terrainHeightVector[vertexIdx];
156 if (terrainVertexHeight >= waterHeight) return false;
157 return true;
158 }
159
160 /**
161 * Determine if water can be generated from left to right starting with x and z
162 * @param terrainHeightVector terrain height vector
163 * @param verticesPerX vertices per x
164 * @param verticesPerZ vertices per z
165 * @param x x
166 * @param z z
167 * @param waterHeight water height
168 * @param waterXPositionSet water x position set
169 * @return if water can be generated at given position
170 */
171 inline static void determineWaterXPositionSet(const vector<float>& terrainHeightVector, int verticesPerX, int verticesPerZ, int x, int z, float waterHeight, unordered_set<int>& waterXPositionSet) {
172 auto xMin = -1;
173 auto xMax = +1;
174 while(true == true) {
175 auto terrainHeightVectorX = x + xMin;
176 if (terrainHeightVectorX < 0 || terrainHeightVectorX >= verticesPerX ||
177 z < 0 || z >= verticesPerZ) break;
178 auto vertexIdx = z * verticesPerX + terrainHeightVectorX;
179 auto terrainVertexHeight = terrainHeightVector[vertexIdx];
180 if (terrainVertexHeight >= waterHeight) break;
181 waterXPositionSet.insert(terrainHeightVectorX);
182 xMin--;
183 }
184 while(true == true) {
185 auto terrainHeightVectorX = x + xMax;
186 if (terrainHeightVectorX < 0 || terrainHeightVectorX >= verticesPerX ||
187 z < 0 || z >= verticesPerZ) break;
188 auto vertexIdx = z * verticesPerX + terrainHeightVectorX;
189 auto terrainVertexHeight = terrainHeightVector[vertexIdx];
190 if (terrainVertexHeight >= waterHeight) break;
191 waterXPositionSet.insert(terrainHeightVectorX);
192 xMax++;
193 }
194 if (waterXPositionSet.size() > 2) {
195 waterXPositionSet.insert(x + xMin - 0);
196 waterXPositionSet.insert(x + xMin - 1);
197 waterXPositionSet.insert(x + xMin - 2);
198 waterXPositionSet.insert(x + xMax + 0);
199 waterXPositionSet.insert(x + xMax + 1);
200 waterXPositionSet.insert(x + xMax + 2);
201 }
202 }
203
204public:
215 };
216
217 /**
218 * Create terrain models
219 * @param width width
220 * @param depth depth
221 * @param y float y
222 * @param terrainHeightVector terrain height vector
223 * @param terrainBoundingBox terrain bounding box
224 * @param terrainModels terrain models vector
225 * @param createLODLevels create LOD Levels
226 */
227 static void createTerrainModels(float width, float depth, float y, vector<float>& terrainHeightVector, BoundingBox& terrainBoundingBox, vector<Model*>& terrainModels, bool createLODLevels = false);
228
229 /**
230 * Apply brush to given terrain models
231 * @param terrainBoundingBox terrain bounding box
232 * @param terrainModels terrain models vector
233 * @param terrainHeightVector terrain height vector
234 * @param brushCenterPosition brush center position
235 * @param brushTexture brush texture
236 * @param brushScale brush scale
237 * @param brushStrength brush strength
238 * @param brushOperation brush operation
239 * @param flattenHeight flatten height
240 *
241 */
242 static void applyBrushToTerrainModels(
243 BoundingBox& terrainBoundingBox, // TODO: constness
244 vector<Model*>& terrainModels,
245 vector<float>& terrainHeightVector,
246 const Vector3& brushCenterPosition,
247 Texture* brushTexture,
248 float brushScale,
249 float brushStrength,
250 BrushOperation brushOperation,
251 float flattenHeight = 0.0f
252 );
253
254 /**
255 * Apply ramp brush to given terrain models
256 * @param terrainBoundingBox terrain bounding box
257 * @param terrainModels terrain models vector
258 * @param terrainHeightVector terrain height vector
259 * @param brushCenterPosition brush center position
260 * @param brushTexture brush texture
261 * @param brushRotation brush rotation
262 * @param brushScale brush scale
263 * @param flattenHeightMin minimum flatten height
264 * @param flattenHeightMax maximum flatten height
265 *
266 */
268 BoundingBox& terrainBoundingBox, // TODO: constness
269 vector<Model*>& terrainModels,
270 vector<float>& terrainHeightVector,
271 const Vector3& brushCenterPosition,
272 Texture* brushTexture,
273 float brushRotation,
274 const Vector2& brushScale,
275 float flattenHeightMin,
276 float flattenHeightMax
277 );
278
279 /**
280 * Update foliage after using terrain brush
281 * @param terrainBoundingBox terrain bounding box
282 * @param terrainHeightVector terrain height vector
283 * @param brushCenterPosition brush center position
284 * @param foliageBrush foliage brush
285 * @param foliageMaps foliage maps
286 * @param updateFoliagePartitions update foliage partitions
287 */
288 static void updateFoliageTerrainBrush(
289 BoundingBox& terrainBoundingBox, // TODO: constness
290 vector<float>& terrainHeightVector,
291 const Vector3& brushCenterPosition,
292 const FoliageBrush& foliageBrush,
293 vector<unordered_map<int, vector<Transformations>>>& foliageMaps,
294 unordered_set<int>& updateFoliagePartitions
295 );
296
297 /**
298 * Update foliage after using terrain ramp brush
299 * @param terrainBoundingBox terrain bounding box
300 * @param terrainHeightVector terrain height vector
301 * @param brushCenterPosition brush center position
302 * @param brushTexture brush texture
303 * @param brushRotation brush rotation
304 * @param brushScale brush scale
305 * @param foliageMaps foliage maps
306 * @param updateFoliagePartitions update foliage partitions
307 */
309 BoundingBox& terrainBoundingBox, // TODO: constness
310 vector<float>& terrainHeightVector,
311 const Vector3& brushCenterPosition,
312 Texture* brushTexture,
313 float brushRotation,
314 const Vector2& brushScale,
315 vector<unordered_map<int, vector<Transformations>>>& foliageMaps,
316 unordered_set<int>& updateFoliagePartitions
317 );
318
319 /**
320 * Compute water positions map using a auto fill like algorithm at given brush center position
321 * @param terrainBoundingBox terrain bounding box
322 * @param terrainHeightVector terrain height vector
323 * @param brushCenterPosition brush center position
324 * @param waterHeight waterHeight
325 * @param waterPositionMap water position map
326 */
327 static bool computeWaterPositionMap(
328 BoundingBox& terrainBoundingBox,
329 const vector<float>& terrainHeightVector,
330 const Vector3& brushCenterPosition,
331 float waterHeight,
332 unordered_map<int, unordered_set<int>>& waterPositionMap
333 );
334
335 /**
336 * Compute water reflection environment mapping position
337 * @param waterPositionMap water position map
338 * @param waterHeight water height
339 * @return water reflection environment mapping position
340 *
341 */
343 const unordered_map<int, unordered_set<int>>& waterPositionMap,
344 float waterHeight
345 );
346
347 /**
348 * Create partitioned water models using given water position map
349 * @param terrainBoundingBox terrain bounding box
350 * @param waterPositionMap water position map
351 * @param waterHeight waterHeight
352 * @param waterModelIdx water model index
353 * @param waterModels water models
354 *
355 */
356 static void createWaterModels(
357 BoundingBox& terrainBoundingBox,
358 const unordered_map<int, unordered_set<int>>& waterPositionMap,
359 float waterHeight,
360 int waterModelIdx,
361 vector<Model*>& waterModels
362 );
363
364 /**
365 * Get terrain models height for e.g. flatten
366 * @param terrainBoundingBox terrain bounding box
367 * @param terrainModels terrain models vector
368 * @param terrainHeightVector terrain height vector
369 * @param brushHeight brush height
370 *
371 */
372 static bool getTerrainModelsHeight(
373 BoundingBox& terrainBoundingBox, // TODO: constness
374 vector<Model*>& terrainModels,
375 vector<float>& terrainHeightVector,
376 const Vector3& brushCenterPosition,
377 float& brushHeight
378 );
379
380 /**
381 * Create foliage maps
382 * @param terrainBoundingBox terrain bounding box
383 * @param foliageMaps foliage maps
384 */
385 static void createFoliageMaps(
386 BoundingBox& terrainBoundingBox, // TODO: constness
387 vector<unordered_map<int, vector<Transformations>>>& foliageMaps
388 );
389
390 /**
391 * Create foliage maps
392 * @param terrainWidth terrain width
393 * @param terrainDepth terrain depth
394 * @param foliageMaps foliage maps
395 */
396 static void createFoliageMaps(
397 float terrainWidth,
398 float terrainDepth,
399 vector<unordered_map<int, vector<Transformations>>>& foliageMaps
400 );
401
402 /**
403 * Empty foliage maps
404 * @param foliageMaps foliage maps
405 */
406 static void emptyFoliageMaps(
407 vector<unordered_map<int, vector<Transformations>>>& foliageMaps
408 );
409
410 /**
411 * Apply foliage brush
412 * @param terrainBoundingBox terrain bounding box
413 * @param terrainHeightVector terrain height vector
414 * @param brushCenterPosition brush center position
415 * @param foliageBrush foliage brush
416 * @param foliageBrushPrototypes foliage brush prototypes
417 * @param brushOperation brush operation
418 * @param foliageMaps foliage maps
419 * @param newFoliageMaps new foliage maps
420 * @param prototypeScale prototype scale
421 */
422 static void applyFoliageBrush(
423 BoundingBox& terrainBoundingBox, // TODO: constness
424 vector<float>& terrainHeightVector,
425 const Vector3& brushCenterPosition,
426 const FoliageBrush& foliageBrush,
427 const vector<FoliageBrushPrototype>& foliageBrushPrototypes,
428 BrushOperation brushOperation,
429 vector<unordered_map<int, vector<Transformations>>>& foliageMaps,
430 vector<unordered_map<int, vector<Transformations>>>& newFoliageMaps
431 );
432
433 /**
434 * Apply foliage delete brush
435 * @param terrainBoundingBox terrain bounding box
436 * @param brushCenterPosition brush center position
437 * @param foliageBrush foliage brush
438 * @param foliageBrushPrototypes foliage brush prototypes
439 * @param brushOperation brush operation
440 * @param foliageMaps foliage maps
441 * @param recreateFoliagePartitions recreate foliage partitions
442 */
443 static void applyFoliageDeleteBrush(
444 BoundingBox& terrainBoundingBox, // TODO: constness
445 const Vector3& brushCenterPosition,
446 const FoliageBrush& foliageBrush,
447 const vector<FoliageBrushPrototype>& foliageBrushPrototypes,
448 BrushOperation brushOperation,
449 vector<unordered_map<int, vector<Transformations>>>& foliageMaps,
450 unordered_set<int>& recreateFoliagePartitions
451 );
452
453 /**
454 * Mirror terrain around X axis
455 * @param flipZ flip Z
456 * @param width width
457 * @param depth depth
458 * @param terrainHeightVector terrain height vector
459 * @param waterPositionMapsHeight water position maps heights
460 * @param waterPositionMaps water position maps
461 * @param foliageMaps foliage maps
462 */
463 static void mirrorXAxis(
464 bool flipZ,
465 float width,
466 float depth,
467 vector<float>& terrainHeightVector,
468 unordered_map<int, float>& waterPositionMapsHeight,
469 unordered_map<int, unordered_map<int, unordered_set<int>>>& waterPositionMaps,
470 vector<unordered_map<int, vector<Transformations>>>& foliageMaps
471 );
472
473 /**
474 * Mirror terrain around Z axis
475 * @param flipX flip X
476 * @param width width
477 * @param depth depth
478 * @param terrainHeightVector terrain height vector
479 * @param waterPositionMapsHeight water position maps heights
480 * @param waterPositionMaps water position maps
481 * @param foliageMaps foliage maps
482 */
483 static void mirrorZAxis(
484 bool flipX,
485 float width,
486 float depth,
487 vector<float>& terrainHeightVector,
488 unordered_map<int, float>& waterPositionMapsHeight,
489 unordered_map<int, unordered_map<int, unordered_set<int>>>& waterPositionMaps,
490 vector<unordered_map<int, vector<Transformations>>>& foliageMaps
491 );
492
493};
Transformations which contain scale, rotations and translation.
Representation of a 3d model.
Definition: Model.h:32
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Definition: BoundingBox.h:25
2D vector 2 class
Definition: Vector2.h:19
3D vector 3 class
Definition: Vector3.h:22
Vector3 & set(float x, float y, float z)
Set up vector.
Definition: Vector3.h:73
Terrain 2 utility.
Definition: Terrain2.h:33
static void getWaterVertex(int x, int z, float waterHeight, Vector3 &vertex)
Get the terrain vertex for given x and z position.
Definition: Terrain2.h:108
static void applyRampBrushToTerrainModels(BoundingBox &terrainBoundingBox, vector< Model * > &terrainModels, vector< float > &terrainHeightVector, const Vector3 &brushCenterPosition, Texture *brushTexture, float brushRotation, const Vector2 &brushScale, float flattenHeightMin, float flattenHeightMax)
Apply ramp brush to given terrain models.
Definition: Terrain2.cpp:736
static bool computeWaterPositionMap(BoundingBox &terrainBoundingBox, const vector< float > &terrainHeightVector, const Vector3 &brushCenterPosition, float waterHeight, unordered_map< int, unordered_set< int > > &waterPositionMap)
Compute water positions map using a auto fill like algorithm at given brush center position.
Definition: Terrain2.cpp:1055
static void determineWaterXPositionSet(const vector< float > &terrainHeightVector, int verticesPerX, int verticesPerZ, int x, int z, float waterHeight, unordered_set< int > &waterXPositionSet)
Determine if water can be generated from left to right starting with x and z.
Definition: Terrain2.h:171
static void mirrorXAxis(bool flipZ, float width, float depth, vector< float > &terrainHeightVector, unordered_map< int, float > &waterPositionMapsHeight, unordered_map< int, unordered_map< int, unordered_set< int > > > &waterPositionMaps, vector< unordered_map< int, vector< Transformations > > > &foliageMaps)
Mirror terrain around X axis.
Definition: Terrain2.cpp:2093
static Vector3 computeWaterReflectionEnvironmentMappingPosition(const unordered_map< int, unordered_set< int > > &waterPositionMap, float waterHeight)
Compute water reflection environment mapping position.
Definition: Terrain2.cpp:1157
static constexpr float PARTITION_SIZE
Definition: Terrain2.h:36
static void applyFoliageDeleteBrush(BoundingBox &terrainBoundingBox, const Vector3 &brushCenterPosition, const FoliageBrush &foliageBrush, const vector< FoliageBrushPrototype > &foliageBrushPrototypes, BrushOperation brushOperation, vector< unordered_map< int, vector< Transformations > > > &foliageMaps, unordered_set< int > &recreateFoliagePartitions)
Apply foliage delete brush.
Definition: Terrain2.cpp:1634
static void updateFoliageTerrainRampBrush(BoundingBox &terrainBoundingBox, vector< float > &terrainHeightVector, const Vector3 &brushCenterPosition, Texture *brushTexture, float brushRotation, const Vector2 &brushScale, vector< unordered_map< int, vector< Transformations > > > &foliageMaps, unordered_set< int > &updateFoliagePartitions)
Update foliage after using terrain ramp brush.
Definition: Terrain2.cpp:1922
static bool determineWater(const vector< float > &terrainHeightVector, int verticesPerX, int verticesPerZ, int x, int z, float waterHeight)
Determine if water can be generated.
Definition: Terrain2.h:151
static bool getTerrainVertex(const vector< float > &terrainHeightVector, int verticesPerX, int verticesPerZ, int x, int z, Vector3 &vertex)
Get the terrain vertex for given x and z position.
Definition: Terrain2.h:85
static bool hasWaterPosition(const unordered_map< int, unordered_set< int > > &waterPositionSet, int x, int z)
Definition: Terrain2.h:122
static void createTerrainModels(float width, float depth, float y, vector< float > &terrainHeightVector, BoundingBox &terrainBoundingBox, vector< Model * > &terrainModels, bool createLODLevels=false)
Create terrain models.
Definition: Terrain2.cpp:70
static const Vector3 computeTerrainVertexNormal(const vector< float > &terrainHeightVector, int verticesPerX, int verticesPerZ, int x, int z)
Compute terrain vertex normal for given x and z position.
Definition: Terrain2.cpp:299
static void createWaterModels(BoundingBox &terrainBoundingBox, const unordered_map< int, unordered_set< int > > &waterPositionMap, float waterHeight, int waterModelIdx, vector< Model * > &waterModels)
Create partitioned water models using given water position map.
Definition: Terrain2.cpp:1180
static bool getTerrainModelsHeight(BoundingBox &terrainBoundingBox, vector< Model * > &terrainModels, vector< float > &terrainHeightVector, const Vector3 &brushCenterPosition, float &brushHeight)
Get terrain models height for e.g.
Definition: Terrain2.cpp:1332
static void mirrorZAxis(bool flipX, float width, float depth, vector< float > &terrainHeightVector, unordered_map< int, float > &waterPositionMapsHeight, unordered_map< int, unordered_map< int, unordered_set< int > > > &waterPositionMaps, vector< unordered_map< int, vector< Transformations > > > &foliageMaps)
Mirror terrain around Z axis.
Definition: Terrain2.cpp:2189
static constexpr float STEP_SIZE
Definition: Terrain2.h:35
static void emptyFoliageMaps(vector< unordered_map< int, vector< Transformations > > > &foliageMaps)
Empty foliage maps.
Definition: Terrain2.cpp:1378
static void applyFoliageBrush(BoundingBox &terrainBoundingBox, vector< float > &terrainHeightVector, const Vector3 &brushCenterPosition, const FoliageBrush &foliageBrush, const vector< FoliageBrushPrototype > &foliageBrushPrototypes, BrushOperation brushOperation, vector< unordered_map< int, vector< Transformations > > > &foliageMaps, vector< unordered_map< int, vector< Transformations > > > &newFoliageMaps)
Apply foliage brush.
Definition: Terrain2.cpp:1385
static void updateFoliageTerrainBrush(BoundingBox &terrainBoundingBox, vector< float > &terrainHeightVector, const Vector3 &brushCenterPosition, const FoliageBrush &foliageBrush, vector< unordered_map< int, vector< Transformations > > > &foliageMaps, unordered_set< int > &updateFoliagePartitions)
Update foliage after using terrain brush.
Definition: Terrain2.cpp:1766
static void applyBrushToTerrainModels(BoundingBox &terrainBoundingBox, vector< Model * > &terrainModels, vector< float > &terrainHeightVector, const Vector3 &brushCenterPosition, Texture *brushTexture, float brushScale, float brushStrength, BrushOperation brushOperation, float flattenHeight=0.0f)
Apply brush to given terrain models.
Definition: Terrain2.cpp:395
static void getTerrainVertex(int x, int z, Vector3 &vertex)
Get the terrain vertex for given x and z position without providing y component.
Definition: Terrain2.h:69
static void createFoliageMaps(BoundingBox &terrainBoundingBox, vector< unordered_map< int, vector< Transformations > > > &foliageMaps)
Create foliage maps.
Definition: Terrain2.cpp:1355