TDME2 1.9.121
Terrain2.cpp
Go to the documentation of this file.
2
3#include <array>
4#include <map>
5#include <set>
6#include <string>
7#include <unordered_map>
8#include <unordered_set>
9#include <vector>
10
11#include <tdme/tdme.h>
27#include <tdme/math/Math.h>
29#include <tdme/math/Vector2.h>
30#include <tdme/math/Vector3.h>
35
36using std::array;
37using std::map;
38using std::set;
39using std::string;
40using std::to_string;
41using std::unordered_map;
42using std::unordered_set;
43using std::vector;
44
46
69
70void Terrain2::createTerrainModels(float width, float depth, float y, vector<float>& terrainHeightVector, BoundingBox& terrainBoundingBox, vector<Model*>& terrainModels, bool createLODLevels)
71{
72 vector<unordered_map<int, int>> partitionTerrainTriangles;
73 vector<vector<Vector3>> partitionTerrainVertices;
74 vector<vector<Vector3>> partitionTerrainNormals;
75 vector<vector<array<int, 6>>> partitionTerrainFaces;
76 vector<vector<int32_t>> partitionLod1Indices;
77 vector<vector<int32_t>> partitionLod2Indices;
78 vector<vector<int32_t>> partitionLod3Indices;
79 auto partitionsX = static_cast<int>(Math::ceil(width / PARTITION_SIZE));
80 auto partitionsZ = static_cast<int>(Math::ceil(depth / PARTITION_SIZE));
81 auto partitionCount = partitionsX * partitionsZ;
82 partitionTerrainTriangles.resize(partitionCount);
83 partitionTerrainVertices.resize(partitionCount);
84 partitionTerrainNormals.resize(partitionCount);
85 partitionTerrainFaces.resize(partitionCount);
86 if (terrainHeightVector.empty() == true) {
87 for (float z = 0.0f; z < depth; z+= STEP_SIZE) {
88 for (float x = 0.0f; x < width; x+= STEP_SIZE) {
89 terrainHeightVector.push_back(y);
90 }
91 }
92 }
93 auto terrainHeightVectorVerticesPerX = static_cast<int>(Math::ceil(width / STEP_SIZE));
94 auto terreinHeightVectorVerticesPerZ = static_cast<int>(Math::ceil(depth / STEP_SIZE));
95 for (float z = 0.0f; z < depth; z+= STEP_SIZE) {
96 for (float x = 0.0f; x < width; x+= STEP_SIZE) {
97
98 auto partitionX = static_cast<int>(x / PARTITION_SIZE);
99 auto partitionZ = static_cast<int>(z / PARTITION_SIZE);
100 auto partitionIdx = partitionZ * partitionsX + partitionX;
101
102 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
103 auto& terrainNormals = partitionTerrainNormals[partitionIdx];
104 auto& terrainFaces = partitionTerrainFaces[partitionIdx];
105 auto& terrainTriangles = partitionTerrainTriangles[partitionIdx];
106
107 int normalIdx = terrainNormals.size();
108 int vertexIdx = terrainVertices.size();
109
110 auto terrainHeightVectorX = static_cast<int>(x / STEP_SIZE);
111 auto terrainHeightVectorZ = static_cast<int>(z / STEP_SIZE);
112
113 Vector3 topVertex;
114 Vector3 topLeftVertex;
115 Vector3 leftVertex;
116 Vector3 vertex;
117
118 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX, terrainHeightVectorZ - 1, topVertex);
119 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX - 1, terrainHeightVectorZ - 1, topLeftVertex);
120 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX - 1, terrainHeightVectorZ, leftVertex);
121 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX, terrainHeightVectorZ, vertex);
122
123 terrainVertices.push_back(topVertex);
124 terrainVertices.push_back(topLeftVertex);
125 terrainVertices.push_back(leftVertex);
126 terrainVertices.push_back(vertex);
127
128 auto topNormal = computeTerrainVertexNormal(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX, terrainHeightVectorZ - 1);
129 auto topLeftNormal = computeTerrainVertexNormal(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX - 1, terrainHeightVectorZ - 1);
130 auto leftNormal = computeTerrainVertexNormal(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX - 1, terrainHeightVectorZ);
131 auto normal = computeTerrainVertexNormal(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX, terrainHeightVectorZ);
132
133 terrainNormals.push_back(topNormal);
134 terrainNormals.push_back(topLeftNormal);
135 terrainNormals.push_back(leftNormal);
136 terrainNormals.push_back(normal);
137
138 terrainFaces.push_back(
139 {
140 vertexIdx + 0,
141 vertexIdx + 1,
142 vertexIdx + 2,
143 normalIdx + 0,
144 normalIdx + 1,
145 normalIdx + 2
146 }
147 );
148 terrainFaces.push_back(
149 {
150 vertexIdx + 2,
151 vertexIdx + 3,
152 vertexIdx + 0,
153 normalIdx + 2,
154 normalIdx + 3,
155 normalIdx + 0
156 }
157 );
158 terrainTriangles[terrainHeightVectorZ]+= 2;
159 }
160 }
161 auto partitionIdx = 0;
162 for (auto partitionIdx = 0; partitionIdx < partitionCount; partitionIdx++) {
163 if (partitionTerrainFaces[partitionIdx].empty() == true) continue;
164 auto modelId = "terrain." + to_string(partitionIdx);
165 auto terrainModel = new Model(modelId, modelId, UpVector::Y_UP, RotationOrder::ZYX, nullptr);
166 auto terrainMaterial = new Material("terrain");
167 terrainMaterial->setSpecularMaterialProperties(new SpecularMaterialProperties());
168 // TODO: Fix me! The textures seem to be much too dark
169 terrainMaterial->getSpecularMaterialProperties()->setAmbientColor(Color4(2.0f, 2.0f, 2.0f, 0.0f));
170 terrainMaterial->getSpecularMaterialProperties()->setDiffuseColor(Color4(1.0f, 1.0f, 1.0f, 1.0f));
171 terrainMaterial->getSpecularMaterialProperties()->setSpecularColor(Color4(0.0f, 0.0f, 0.0f, 0.0f));
172 terrainModel->getMaterials()[terrainMaterial->getId()] = terrainMaterial;
173 auto terrainNode = new Node(terrainModel, nullptr, "terrain", "terrain");
174 FacesEntity nodeFacesEntityTerrain(terrainNode, "terrain.facesentity");
175 nodeFacesEntityTerrain.setMaterial(terrainMaterial);
176 vector<FacesEntity> nodeFacesEntities;
177 vector<Face> nodeFaces;
178 auto trianglesPerX = partitionTerrainTriangles[partitionIdx].begin()->second;
179 auto trianglesPerZ = partitionTerrainTriangles[partitionIdx].size();
180 for (auto& faceIndices: partitionTerrainFaces[partitionIdx]) {
181 nodeFaces.push_back(
182 Face(
183 terrainNode,
184 faceIndices[0],
185 faceIndices[1],
186 faceIndices[2],
187 faceIndices[3],
188 faceIndices[4],
189 faceIndices[5]
190 )
191 );
192 };
193
194 // create LOD levels?
195 if (createLODLevels == true) {
196 // lod1
197 {
198 vector<int32_t> lod1Indices;
199 auto& facesIndices = partitionTerrainFaces[partitionIdx];
200 auto finishedZ = false;
201 for (auto z = 0; finishedZ == false && z < trianglesPerZ; z+= 4) {
202 if (z > trianglesPerZ - 4) {
203 z = trianglesPerZ - 4;
204 finishedZ = true;
205 }
206 auto finishedX = false;
207 for (auto x = 0; finishedX == false && x < trianglesPerX; x+= 8) {
208 if (x > trianglesPerX - 8) {
209 x = trianglesPerX - 8;
210 finishedX = true;
211 }
212 lod1Indices.push_back(facesIndices[z * trianglesPerX + x + 6][0]); // top
213 lod1Indices.push_back(facesIndices[z * trianglesPerX + x][1]); // top left
214 lod1Indices.push_back(facesIndices[(z + 3) * trianglesPerX + x][2]); // left
215 lod1Indices.push_back(facesIndices[(z + 3) * trianglesPerX + x][2]); // left
216 lod1Indices.push_back(facesIndices[(z + 3) * trianglesPerX + x + 7][1]); // vertex
217 lod1Indices.push_back(facesIndices[z * trianglesPerX + x + 6][0]); // top
218 }
219 }
220 nodeFacesEntityTerrain.setLOD1Indices(lod1Indices);
221 nodeFacesEntityTerrain.setLOD1Distance(64.0f);
222 }
223
224 // lod2
225 {
226 vector<int32_t> lod2Indices;
227 auto& facesIndices = partitionTerrainFaces[partitionIdx];
228 auto finishedZ = false;
229 for (auto z = 0; finishedZ == false && z < trianglesPerZ; z+= 8) {
230 if (z > trianglesPerZ - 8) {
231 z = trianglesPerZ - 8;
232 finishedZ = true;
233 }
234 auto finishedX = false;
235 for (auto x = 0; finishedX == false && x < trianglesPerX; x+= 16) {
236 if (x > trianglesPerX - 16) {
237 x = trianglesPerX - 16;
238 finishedX = true;
239 }
240 lod2Indices.push_back(facesIndices[z * trianglesPerX + x + 14][0]); // top
241 lod2Indices.push_back(facesIndices[z * trianglesPerX + x][1]); // top left
242 lod2Indices.push_back(facesIndices[(z + 7) * trianglesPerX + x][2]); // left
243 lod2Indices.push_back(facesIndices[(z + 7) * trianglesPerX + x][2]); // left
244 lod2Indices.push_back(facesIndices[(z + 7) * trianglesPerX + x + 15][1]); // vertex
245 lod2Indices.push_back(facesIndices[z * trianglesPerX + x + 14][0]); // top
246 }
247 }
248 nodeFacesEntityTerrain.setLOD2Indices(lod2Indices);
249 nodeFacesEntityTerrain.setLOD2Distance(128.0f);
250 }
251
252 // lod3
253 {
254 vector<int32_t> lod3Indices;
255 auto& facesIndices = partitionTerrainFaces[partitionIdx];
256 auto finishedZ = false;
257 for (auto z = 0; finishedZ == false && z < trianglesPerZ; z+= 16) {
258 if (z > trianglesPerZ - 16) {
259 z = trianglesPerZ - 16;
260 finishedZ = true;
261 }
262 auto finishedX = false;
263 for (auto x = 0; finishedX == false && x < trianglesPerX; x+= 32) {
264 if (x > trianglesPerX - 32) {
265 x = trianglesPerX - 32;
266 finishedX = true;
267 }
268 lod3Indices.push_back(facesIndices[z * trianglesPerX + x + 30][0]); // top
269 lod3Indices.push_back(facesIndices[z * trianglesPerX + x][1]); // top left
270 lod3Indices.push_back(facesIndices[(z + 15) * trianglesPerX + x][2]); // left
271 lod3Indices.push_back(facesIndices[(z + 15) * trianglesPerX + x][2]); // left
272 lod3Indices.push_back(facesIndices[(z + 15) * trianglesPerX + x + 31][1]); // vertex
273 lod3Indices.push_back(facesIndices[z * trianglesPerX + x + 30][0]); // top
274 }
275 }
276 nodeFacesEntityTerrain.setLOD3Indices(lod3Indices);
277 nodeFacesEntityTerrain.setLOD3Distance(192.0f);
278 }
279 }
280
281 nodeFacesEntityTerrain.setFaces(nodeFaces);
282 nodeFacesEntities.push_back(nodeFacesEntityTerrain);
283 terrainNode->setVertices(partitionTerrainVertices[partitionIdx]);
284 terrainNode->setNormals(partitionTerrainNormals[partitionIdx]);
285 terrainNode->setFacesEntities(nodeFacesEntities);
286 terrainModel->getNodes()[terrainNode->getId()] = terrainNode;
287 terrainModel->getSubNodes()[terrainNode->getId()] = terrainNode;
288 terrainModel->invalidateBoundingBox();
289 if (partitionIdx == 0) {
290 terrainBoundingBox = *terrainModel->getBoundingBox();
291 } else {
292 terrainBoundingBox.extend(terrainModel->getBoundingBox());
293 }
294 ModelTools::createDefaultAnimation(terrainModel, 1);
295 terrainModels.push_back(terrainModel);
296 }
297}
298
299inline const Vector3 Terrain2::computeTerrainVertexNormal(const vector<float>& terrainHeightVector, int verticesPerX, int verticesPerZ, int x, int z) {
300 Vector3 vertexNormal;
301
302 Vector3 vertex;
303 auto haveVertex = getTerrainVertex(terrainHeightVector, verticesPerX, verticesPerZ, x, z - 1, vertex);
304 if (haveVertex == false) return Vector3(0.0f, 1.0f, 0.0f);
305
306 Vector3 topVertex;
307 Vector3 topLeftVertex;
308 Vector3 leftVertex;
309 Vector3 bottomVertex;
310 Vector3 rightVertex;
311 Vector3 bottomRightVertex;
312
313 auto haveTopVertex = getTerrainVertex(terrainHeightVector, verticesPerX, verticesPerZ, x, z - 1, topVertex);
314 auto haveTopLeftVertex = getTerrainVertex(terrainHeightVector, verticesPerX, verticesPerZ, x - 1, z - 1, topLeftVertex);
315 auto haveLeftVertex = getTerrainVertex(terrainHeightVector, verticesPerX, verticesPerZ, x - 1, z, leftVertex);
316 auto haveBottomVertex = getTerrainVertex(terrainHeightVector, verticesPerX, verticesPerZ, x, z + 1, bottomVertex);
317 auto haveRightVertex = getTerrainVertex(terrainHeightVector, verticesPerX, verticesPerZ, x + 1, z, rightVertex);
318 auto haveBottomRightVertex = getTerrainVertex(terrainHeightVector, verticesPerX, verticesPerZ, x + 1, z + 1, bottomRightVertex);
319
320 Vector3 triangleNormal;
321 int normalCount = 0;
322 if (haveTopVertex == true && haveTopLeftVertex == true) {
323 triangleNormal = ModelTools::computeNormal(
324 {
325 topVertex,
326 topLeftVertex,
327 vertex
328 }
329 );
330 vertexNormal.add(triangleNormal);
331 normalCount++;
332 }
333 if (haveTopLeftVertex == true && haveLeftVertex == true) {
334 triangleNormal = ModelTools::computeNormal(
335 {
336 topLeftVertex,
337 leftVertex,
338 vertex
339 }
340 );
341 vertexNormal.add(triangleNormal);
342 normalCount++;
343 }
344 if (haveLeftVertex == true && haveBottomVertex == true) {
345 triangleNormal = ModelTools::computeNormal(
346 {
347 leftVertex,
348 bottomVertex,
349 vertex
350 }
351 );
352 vertexNormal.add(triangleNormal);
353 normalCount++;
354 }
355 if (haveBottomVertex == true && haveBottomRightVertex == true) {
356 triangleNormal = ModelTools::computeNormal(
357 {
358 bottomVertex,
359 bottomRightVertex,
360 vertex
361 }
362 );
363 vertexNormal.add(triangleNormal);
364 normalCount++;
365 }
366 if (haveBottomRightVertex == true && haveRightVertex == true) {
367 triangleNormal = ModelTools::computeNormal(
368 {
369 bottomRightVertex,
370 rightVertex,
371 vertex
372 }
373 );
374 vertexNormal.add(triangleNormal);
375 normalCount++;
376 }
377 if (haveRightVertex == true && haveTopVertex == true) {
378 triangleNormal = ModelTools::computeNormal(
379 {
380 rightVertex,
381 topVertex,
382 vertex
383 }
384 );
385 vertexNormal.add(triangleNormal);
386 normalCount++;
387 }
388 if (normalCount > 0) {
389 return vertexNormal.normalize();
390 }
391 Console::println("Terrain2::computeTerrainVertexNormal(): no vertex normal available: normal count == 0");
392 return vertexNormal.set(0.0f, 1.0f, 0.0f);
393}
394
396 BoundingBox& terrainBoundingBox,
397 vector<Model*>& terrainModels,
398 vector<float>& terrainHeightVector,
399 const Vector3& brushCenterPosition,
400 Texture* brushTexture,
401 float brushScale,
402 float brushStrength,
403 BrushOperation brushOperation,
404 float brushHeight
405) {
406 // check if we have a texture
407 if (brushTexture == nullptr) return;
408 // check if we have a model
409 if (terrainModels.empty() == true) return;
410
411 // apply brush
412 vector<vector<Vector3>> partitionTerrainVertices;
413 vector<vector<Vector3>> partitionTerrainNormals;
414 partitionTerrainVertices.resize(terrainModels.size());
415 partitionTerrainNormals.resize(terrainModels.size());
416 auto partitionsX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / PARTITION_SIZE));
417 auto terrainHeightVectorVerticesPerX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / STEP_SIZE));
418 auto terreinHeightVectorVerticesPerZ = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getZ() / STEP_SIZE));
419
420 // water
421 if (brushOperation == BRUSHOPERATION_WATER_ADD) return;
422
423 // other operations
424 auto textureData = brushTexture->getTextureData();
425 auto textureWidth = brushTexture->getTextureWidth();
426 auto textureHeight = brushTexture->getTextureHeight();
427 auto textureBytePerPixel = brushTexture->getDepth() == 32?4:3;
428 for (auto z = 0.0f; z < textureHeight * brushScale; z+= STEP_SIZE) {
429 auto brushPosition =
430 brushCenterPosition.
431 clone().
432 sub(
433 Vector3(
434 (static_cast<float>(textureWidth) * brushScale) / 2.0f,
435 0.0f,
436 (static_cast<float>(textureHeight) * brushScale) / 2.0f
437 )
438 ).
439 add(
440 Vector3(
441 0.0f,
442 0.0f,
443 z
444 )
445 );
446 for (auto x = 0.0f; x < textureWidth * brushScale; x+= STEP_SIZE) {
447 auto textureX = static_cast<int>(x / brushScale);
448 auto textureY = static_cast<int>(z / brushScale);
449 auto red = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 0);
450 auto green = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 1);
451 auto blue = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 2);
452 auto alpha = textureBytePerPixel == 3?255:textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 3);
453 auto appliedStrength = (static_cast<float>(red) + static_cast<float>(green) + static_cast<float>(blue)) / (255.0f * 3.0f) * brushStrength;
454 auto terrainHeightVectorX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / STEP_SIZE);
455 auto terrainHeightVectorZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / STEP_SIZE);
456 if (terrainHeightVectorX < 0 || terrainHeightVectorX >= terrainHeightVectorVerticesPerX ||
457 terrainHeightVectorZ < 0 || terrainHeightVectorZ >= terreinHeightVectorVerticesPerZ) {
458 brushPosition.add(
459 Vector3(
460 STEP_SIZE,
461 0.0f,
462 0.0f
463 )
464 );
465 continue;
466 }
467 auto vertexIdx = terrainHeightVectorZ * terrainHeightVectorVerticesPerX + terrainHeightVectorX;
468 auto terrainVertexHeight = terrainHeightVector[vertexIdx];
469 switch(brushOperation) {
471 terrainVertexHeight+= appliedStrength;
472 break;
474 terrainVertexHeight-= appliedStrength;
475 break;
477 terrainVertexHeight = terrainVertexHeight * (1.0f - Math::clamp(appliedStrength, 0.0f, 1.0f)) + brushHeight * Math::clamp(appliedStrength, 0.0f, 1.0f);
478 break;
480 terrainVertexHeight = terrainVertexHeight * (1.0f - Math::clamp(appliedStrength, 0.0f, 1.0f)) + 0.0f * Math::clamp(appliedStrength, 0.0f, 1.0f);
481 break;
483 auto terrainVertexHeightNeighbours = 0.0f;
484 auto terrainVertexHeightNeighbourCount = 0;
485 Vector3 topVertex;
486 Vector3 leftVertex;
487 Vector3 bottomVertex;
488 Vector3 rightVertex;
489 auto haveTopVertex = getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX, terrainHeightVectorZ - 1, topVertex);
490 auto haveLeftVertex = getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX - 1, terrainHeightVectorZ, leftVertex);
491 auto haveBottomVertex = getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX, terrainHeightVectorZ + 1, bottomVertex);
492 auto haveRightVertex = getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX + 1, terrainHeightVectorZ, rightVertex);
493 if (haveTopVertex == true) {
494 terrainVertexHeightNeighbourCount++;
495 terrainVertexHeightNeighbours+= topVertex[1];
496 }
497 if (haveLeftVertex == true) {
498 terrainVertexHeightNeighbourCount++;
499 terrainVertexHeightNeighbours+= leftVertex[1];
500 }
501 if (haveBottomVertex == true) {
502 terrainVertexHeightNeighbourCount++;
503 terrainVertexHeightNeighbours+= bottomVertex[1];
504 }
505 if (haveRightVertex == true) {
506 terrainVertexHeightNeighbourCount++;
507 terrainVertexHeightNeighbours+= rightVertex[1];
508 }
509 if (terrainVertexHeightNeighbourCount > 0) {
510 auto terrainVertexHeightSmoothed = terrainVertexHeightNeighbours / static_cast<float>(terrainVertexHeightNeighbourCount);
511 terrainVertexHeight = terrainVertexHeight * (1.0f - Math::clamp(appliedStrength, 0.0f, 1.0f)) + terrainVertexHeightSmoothed * Math::clamp(appliedStrength, 0.0f, 1.0f);
512 }
513 break;
514 }
515 terrainHeightVector[vertexIdx] = terrainVertexHeight;
516
517 // original
518 {
519 auto _brushPosition = brushPosition;
520 auto partitionX = static_cast<int>((_brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
521 auto partitionZ = static_cast<int>((_brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
522 auto partitionIdx = partitionZ * partitionsX + partitionX;
523 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:nullptr;
524 auto terrainNode = terrainModel != nullptr?terrainModel->getNodeById("terrain"):nullptr;
525
526 auto terrainModelX = terrainNode != nullptr?static_cast<int>((_brushPosition.getX() - terrainModel->getBoundingBox()->getMin().getX()) / STEP_SIZE):-1;
527 auto terrainModelZ = terrainNode != nullptr?static_cast<int>((_brushPosition.getZ() - terrainModel->getBoundingBox()->getMin().getZ()) / STEP_SIZE):-1;
528 auto terrainModelVerticesPerZ = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getZ() / STEP_SIZE)):-1;
529 auto terrainModelVerticesPerX = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getX() / STEP_SIZE)):-1;
530
531 if (terrainNode != nullptr &&
532 terrainModelX >= 0 &&
533 terrainModelX < terrainModelVerticesPerX &&
534 terrainModelZ >= 0 &&
535 terrainModelZ < terrainModelVerticesPerZ) {
536 if (partitionTerrainVertices[partitionIdx].empty() == true) {
537 partitionTerrainVertices[partitionIdx] = terrainNode->getVertices();
538 }
539 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
540 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
541 terrainVertices[vertexIdx + 3][1] = terrainVertexHeight;
542 }
543 }
544
545 // top
546 {
547 auto _brushPosition = brushPosition.clone().sub(Vector3(0.0f, 0.0f, -STEP_SIZE));
548 auto partitionX = static_cast<int>((_brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
549 auto partitionZ = static_cast<int>((_brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
550 auto partitionIdx = partitionZ * partitionsX + partitionX;
551 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:nullptr;
552 auto terrainNode = terrainModel != nullptr?terrainModel->getNodeById("terrain"):nullptr;
553
554 auto terrainModelX = terrainNode != nullptr?static_cast<int>((_brushPosition.getX() - terrainModel->getBoundingBox()->getMin().getX()) / STEP_SIZE):-1;
555 auto terrainModelZ = terrainNode != nullptr?static_cast<int>((_brushPosition.getZ() - terrainModel->getBoundingBox()->getMin().getZ()) / STEP_SIZE):-1;
556 auto terrainModelVerticesPerZ = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getZ() / STEP_SIZE)):-1;
557 auto terrainModelVerticesPerX = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getX() / STEP_SIZE)):-1;
558
559 if (terrainNode != nullptr &&
560 terrainModelX >= 0 &&
561 terrainModelX < terrainModelVerticesPerX &&
562 terrainModelZ >= 0 &&
563 terrainModelZ < terrainModelVerticesPerZ) {
564 if (partitionTerrainVertices[partitionIdx].empty() == true) {
565 partitionTerrainVertices[partitionIdx] = terrainNode->getVertices();
566 }
567 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
568 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
569 terrainVertices[vertexIdx + 0][1] = terrainVertexHeight;
570 }
571 }
572
573 // top, left
574 {
575 auto _brushPosition = brushPosition.clone().sub(Vector3(-STEP_SIZE, 0.0f, -STEP_SIZE));
576 auto partitionX = static_cast<int>((_brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
577 auto partitionZ = static_cast<int>((_brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
578 auto partitionIdx = partitionZ * partitionsX + partitionX;
579 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:nullptr;
580 auto terrainNode = terrainModel != nullptr?terrainModel->getNodeById("terrain"):nullptr;
581
582 auto terrainModelX = terrainNode != nullptr?static_cast<int>((_brushPosition.getX() - terrainModel->getBoundingBox()->getMin().getX()) / STEP_SIZE):-1;
583 auto terrainModelZ = terrainNode != nullptr?static_cast<int>((_brushPosition.getZ() - terrainModel->getBoundingBox()->getMin().getZ()) / STEP_SIZE):-1;
584 auto terrainModelVerticesPerZ = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getZ() / STEP_SIZE)):-1;
585 auto terrainModelVerticesPerX = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getX() / STEP_SIZE)):-1;
586
587 if (terrainNode != nullptr &&
588 terrainModelX >= 0 &&
589 terrainModelX < terrainModelVerticesPerX &&
590 terrainModelZ >= 0 &&
591 terrainModelZ < terrainModelVerticesPerZ) {
592 if (partitionTerrainVertices[partitionIdx].empty() == true) {
593 partitionTerrainVertices[partitionIdx] = terrainNode->getVertices();
594 }
595 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
596 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
597 terrainVertices[vertexIdx + 1][1] = terrainVertexHeight;
598 }
599 }
600
601 // left
602 {
603 auto _brushPosition = brushPosition.clone().sub(Vector3(-STEP_SIZE, 0.0f, 0.0f));
604 auto partitionX = static_cast<int>((_brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
605 auto partitionZ = static_cast<int>((_brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
606 auto partitionIdx = partitionZ * partitionsX + partitionX;
607 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:nullptr;
608 auto terrainNode = terrainModel != nullptr?terrainModel->getNodeById("terrain"):nullptr;
609
610 auto terrainModelX = terrainNode != nullptr?static_cast<int>((_brushPosition.getX() - terrainModel->getBoundingBox()->getMin().getX()) / STEP_SIZE):-1;
611 auto terrainModelZ = terrainNode != nullptr?static_cast<int>((_brushPosition.getZ() - terrainModel->getBoundingBox()->getMin().getZ()) / STEP_SIZE):-1;
612 auto terrainModelVerticesPerZ = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getZ() / STEP_SIZE)):-1;
613 auto terrainModelVerticesPerX = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getX() / STEP_SIZE)):-1;
614
615 if (terrainNode != nullptr &&
616 terrainModelX >= 0 &&
617 terrainModelX < terrainModelVerticesPerX &&
618 terrainModelZ >= 0 &&
619 terrainModelZ < terrainModelVerticesPerZ) {
620 if (partitionTerrainVertices[partitionIdx].empty() == true) {
621 partitionTerrainVertices[partitionIdx] = terrainNode->getVertices();
622 }
623 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
624 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
625 terrainVertices[vertexIdx + 2][1] = terrainVertexHeight;
626 }
627 }
628
629 //
630 brushPosition.add(
631 Vector3(
632 STEP_SIZE,
633 0.0f,
634 0.0f
635 )
636 );
637 }
638 }
639
640 // normals
641 for (auto z = -STEP_SIZE * 8.0f; z < textureHeight * brushScale + STEP_SIZE * 16.0f; z+= STEP_SIZE) {
642 auto brushPosition =
643 brushCenterPosition.
644 clone().
645 sub(
646 Vector3(
647 (static_cast<float>(textureWidth) * brushScale) / 2.0f,
648 0.0f,
649 (static_cast<float>(textureHeight) * brushScale) / 2.0f
650 )
651 ).
652 add(
653 Vector3(
654 -STEP_SIZE * 8.0f,
655 0.0f,
656 z
657 )
658 );
659 for (auto x = -STEP_SIZE * 8.0f; x < textureWidth * brushScale + STEP_SIZE * 16.0f; x+= STEP_SIZE) {
660 auto partitionX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
661 auto partitionZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
662 auto partitionIdx = partitionZ * partitionsX + partitionX;
663 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:nullptr;
664 auto terrainNode = terrainModel != nullptr?terrainModel->getNodeById("terrain"):nullptr;
665
666 auto terrainHeightVectorX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / STEP_SIZE) + 1;
667 auto terrainHeightVectorZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / STEP_SIZE) + 1;
668 auto terrainModelX = terrainNode != nullptr?static_cast<int>((brushPosition.getX() - terrainModel->getBoundingBox()->getMin().getX()) / STEP_SIZE):-1;
669 auto terrainModelZ = terrainNode != nullptr?static_cast<int>((brushPosition.getZ() - terrainModel->getBoundingBox()->getMin().getZ()) / STEP_SIZE):-1;
670 auto terrainModelVerticesPerZ = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getZ() / STEP_SIZE)):-1;
671 auto terrainModelVerticesPerX = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getX() / STEP_SIZE)):-1;
672
673 if (terrainNode != nullptr &&
674 terrainModelX >= 0 &&
675 terrainModelX < terrainModelVerticesPerX &&
676 terrainModelZ >= 0 &&
677 terrainModelZ < terrainModelVerticesPerZ) {
678 if (partitionTerrainNormals[partitionIdx].empty() == true) {
679 partitionTerrainNormals[partitionIdx] = terrainNode->getNormals();
680 }
681 auto& terrainNormals = partitionTerrainNormals[partitionIdx];
682
683 auto topNormal = computeTerrainVertexNormal(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX, terrainHeightVectorZ - 1);
684 auto topLeftNormal = computeTerrainVertexNormal(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX - 1, terrainHeightVectorZ - 1);
685 auto leftNormal = computeTerrainVertexNormal(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX - 1, terrainHeightVectorZ);
686 auto normal = computeTerrainVertexNormal(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX, terrainHeightVectorZ);
687 auto normalIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
688
689 terrainNormals[normalIdx + 0] = topNormal;
690 terrainNormals[normalIdx + 1] = topLeftNormal;
691 terrainNormals[normalIdx + 2] = leftNormal;
692 terrainNormals[normalIdx + 3] = normal;
693 }
694
695 //
696 brushPosition.add(
697 Vector3(
698 STEP_SIZE,
699 0.0f,
700 0.0f
701 )
702 );
703 }
704 }
705
706 // set terrain model vertices
707 {
708 auto partitionIdx = 0;
709 for (auto& terrainVertices: partitionTerrainVertices) {
710 if (terrainVertices.empty() == false) {
711 auto terrainNode = terrainModels[partitionIdx]->getNodeById("terrain");
712 if (terrainNode != nullptr) {
713 terrainNode->setVertices(terrainVertices);
714 terrainModels[partitionIdx]->invalidateBoundingBox();
715 }
716 }
717 partitionIdx++;
718 }
719 }
720 // set terrain model normals
721 {
722 auto partitionIdx = 0;
723 for (auto& terrainNormals: partitionTerrainNormals) {
724 if (terrainNormals.empty() == false) {
725 auto terrainNode = terrainModels[partitionIdx]->getNodeById("terrain");
726 if (terrainNode != nullptr) {
727 terrainNode->setNormals(terrainNormals);
728 terrainModels[partitionIdx]->invalidateBoundingBox();
729 }
730 }
731 partitionIdx++;
732 }
733 }
734}
735
737 BoundingBox& terrainBoundingBox, // TODO: constness
738 vector<Model*>& terrainModels,
739 vector<float>& terrainHeightVector,
740 const Vector3& brushCenterPosition,
741 Texture* brushTexture,
742 float brushRotation,
743 const Vector2& brushScale,
744 float heightMin,
745 float heightMax
746) {
747 // check if we have a texture
748 if (brushTexture == nullptr) return;
749 // check if we have a model
750 if (terrainModels.empty() == true) return;
751
752 // apply brush
753 vector<vector<Vector3>> partitionTerrainVertices;
754 vector<vector<Vector3>> partitionTerrainNormals;
755 partitionTerrainVertices.resize(terrainModels.size());
756 partitionTerrainNormals.resize(terrainModels.size());
757 auto partitionsX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / PARTITION_SIZE));
758 auto terrainHeightVectorVerticesPerX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / STEP_SIZE));
759 auto terreinHeightVectorVerticesPerZ = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getZ() / STEP_SIZE));
760
761 // texture
762 auto textureData = brushTexture->getTextureData();
763 auto textureWidth = brushTexture->getTextureWidth();
764 auto textureHeight = brushTexture->getTextureHeight();
765 auto textureBytePerPixel = brushTexture->getDepth() == 32?4:3;
766
767 // brush texture matrix
768 Matrix2D3x3 brushTextureMatrix;
769 brushTextureMatrix.identity();
770 brushTextureMatrix.translate(Vector2(static_cast<float>(textureWidth) / 2.0f, static_cast<float>(textureHeight) / 2.0f));
771 brushTextureMatrix.multiply((Matrix2D3x3()).identity().scale(Vector2(1.0f / brushScale.getX(), 1.0f / brushScale.getY())));
772 brushTextureMatrix.multiply((Matrix2D3x3()).identity().rotate(brushRotation));
773 auto brushScaleMax = Math::max(brushScale.getX(), brushScale.getY());
774
775 //
776 for (auto z = -textureHeight * brushScaleMax * 2.0f; z < textureHeight * brushScaleMax * 2.0f; z+= STEP_SIZE) {
777 auto brushPosition =
778 brushCenterPosition.
779 clone().
780 sub(
781 Vector3(
782 static_cast<float>(textureWidth) * brushScaleMax * 2.0f,
783 0.0f,
784 0.0f
785 )
786 ).
787 add(
788 Vector3(
789 0.0f,
790 0.0f,
791 z
792 )
793 );
794 for (auto x = -textureWidth * brushScaleMax * 2.0f; x < textureWidth * brushScaleMax * 2.0f; x+= STEP_SIZE) {
795 auto texturePositionUntransformed = Vector2(x, z);
796 auto texturePosition = brushTextureMatrix.multiply(texturePositionUntransformed);
797 auto textureX = static_cast<int>(texturePosition.getX());
798 auto textureY = static_cast<int>(texturePosition.getY());
799 if (textureX < 0 || textureX >= textureWidth ||
800 textureY < 0 || textureY >= textureHeight) {
801 brushPosition.add(
802 Vector3(
803 STEP_SIZE,
804 0.0f,
805 0.0f
806 )
807 );
808 continue;
809 }
810 auto red = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 0);
811 auto green = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 1);
812 auto blue = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 2);
813 auto alpha = textureBytePerPixel == 3?255:textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 3);
814 auto height = ((static_cast<float>(red) + static_cast<float>(green) + static_cast<float>(blue)) / (255.0f * 3.0f) * (heightMax - heightMin)) + heightMin;
815 auto terrainHeightVectorX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / STEP_SIZE);
816 auto terrainHeightVectorZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / STEP_SIZE);
817 if (terrainHeightVectorX < 0 || terrainHeightVectorX >= terrainHeightVectorVerticesPerX ||
818 terrainHeightVectorZ < 0 || terrainHeightVectorZ >= terreinHeightVectorVerticesPerZ) {
819 brushPosition.add(
820 Vector3(
821 STEP_SIZE,
822 0.0f,
823 0.0f
824 )
825 );
826 continue;
827 }
828 auto terrainVertexHeight = 0.0f;
829 {
830 auto vertexIdx = terrainHeightVectorZ * terrainHeightVectorVerticesPerX + terrainHeightVectorX;
831 terrainVertexHeight = terrainHeightVector[vertexIdx];
832 terrainVertexHeight = height > terrainVertexHeight?height:terrainVertexHeight;
833 terrainHeightVector[vertexIdx] = terrainVertexHeight;
834 }
835
836 // original
837 {
838 auto _brushPosition = brushPosition;
839 auto partitionX = static_cast<int>((_brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
840 auto partitionZ = static_cast<int>((_brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
841 auto partitionIdx = partitionZ * partitionsX + partitionX;
842 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:nullptr;
843 auto terrainNode = terrainModel != nullptr?terrainModel->getNodeById("terrain"):nullptr;
844
845 auto terrainModelX = terrainNode != nullptr?static_cast<int>((_brushPosition.getX() - terrainModel->getBoundingBox()->getMin().getX()) / STEP_SIZE):-1;
846 auto terrainModelZ = terrainNode != nullptr?static_cast<int>((_brushPosition.getZ() - terrainModel->getBoundingBox()->getMin().getZ()) / STEP_SIZE):-1;
847 auto terrainModelVerticesPerZ = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getZ() / STEP_SIZE)):-1;
848 auto terrainModelVerticesPerX = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getX() / STEP_SIZE)):-1;
849
850 if (terrainNode != nullptr &&
851 terrainModelX >= 0 &&
852 terrainModelX < terrainModelVerticesPerX &&
853 terrainModelZ >= 0 &&
854 terrainModelZ < terrainModelVerticesPerZ) {
855 if (partitionTerrainVertices[partitionIdx].empty() == true) {
856 partitionTerrainVertices[partitionIdx] = terrainNode->getVertices();
857 }
858 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
859 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
860 terrainVertices[vertexIdx + 3][1] = terrainVertexHeight;
861 }
862 }
863
864 // top
865 {
866 auto _brushPosition = brushPosition.clone().sub(Vector3(0.0f, 0.0f, -STEP_SIZE));
867 auto partitionX = static_cast<int>((_brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
868 auto partitionZ = static_cast<int>((_brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
869 auto partitionIdx = partitionZ * partitionsX + partitionX;
870 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:nullptr;
871 auto terrainNode = terrainModel != nullptr?terrainModel->getNodeById("terrain"):nullptr;
872
873 auto terrainModelX = terrainNode != nullptr?static_cast<int>((_brushPosition.getX() - terrainModel->getBoundingBox()->getMin().getX()) / STEP_SIZE):-1;
874 auto terrainModelZ = terrainNode != nullptr?static_cast<int>((_brushPosition.getZ() - terrainModel->getBoundingBox()->getMin().getZ()) / STEP_SIZE):-1;
875 auto terrainModelVerticesPerZ = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getZ() / STEP_SIZE)):-1;
876 auto terrainModelVerticesPerX = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getX() / STEP_SIZE)):-1;
877
878 if (terrainNode != nullptr &&
879 terrainModelX >= 0 &&
880 terrainModelX < terrainModelVerticesPerX &&
881 terrainModelZ >= 0 &&
882 terrainModelZ < terrainModelVerticesPerZ) {
883 if (partitionTerrainVertices[partitionIdx].empty() == true) {
884 partitionTerrainVertices[partitionIdx] = terrainNode->getVertices();
885 }
886 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
887 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
888 terrainVertices[vertexIdx + 0][1] = terrainVertexHeight;
889 }
890 }
891
892 // top, left
893 {
894 auto _brushPosition = brushPosition.clone().sub(Vector3(-STEP_SIZE, 0.0f, -STEP_SIZE));
895 auto partitionX = static_cast<int>((_brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
896 auto partitionZ = static_cast<int>((_brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
897 auto partitionIdx = partitionZ * partitionsX + partitionX;
898 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:nullptr;
899 auto terrainNode = terrainModel != nullptr?terrainModel->getNodeById("terrain"):nullptr;
900
901 auto terrainModelX = terrainNode != nullptr?static_cast<int>((_brushPosition.getX() - terrainModel->getBoundingBox()->getMin().getX()) / STEP_SIZE):-1;
902 auto terrainModelZ = terrainNode != nullptr?static_cast<int>((_brushPosition.getZ() - terrainModel->getBoundingBox()->getMin().getZ()) / STEP_SIZE):-1;
903 auto terrainModelVerticesPerZ = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getZ() / STEP_SIZE)):-1;
904 auto terrainModelVerticesPerX = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getX() / STEP_SIZE)):-1;
905
906 if (terrainNode != nullptr &&
907 terrainModelX >= 0 &&
908 terrainModelX < terrainModelVerticesPerX &&
909 terrainModelZ >= 0 &&
910 terrainModelZ < terrainModelVerticesPerZ) {
911 if (partitionTerrainVertices[partitionIdx].empty() == true) {
912 partitionTerrainVertices[partitionIdx] = terrainNode->getVertices();
913 }
914 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
915 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
916 terrainVertices[vertexIdx + 1][1] = terrainVertexHeight;
917 }
918 }
919
920 // left
921 {
922 auto _brushPosition = brushPosition.clone().sub(Vector3(-STEP_SIZE, 0.0f, 0.0f));
923 auto partitionX = static_cast<int>((_brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
924 auto partitionZ = static_cast<int>((_brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
925 auto partitionIdx = partitionZ * partitionsX + partitionX;
926 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:nullptr;
927 auto terrainNode = terrainModel != nullptr?terrainModel->getNodeById("terrain"):nullptr;
928
929 auto terrainModelX = terrainNode != nullptr?static_cast<int>((_brushPosition.getX() - terrainModel->getBoundingBox()->getMin().getX()) / STEP_SIZE):-1;
930 auto terrainModelZ = terrainNode != nullptr?static_cast<int>((_brushPosition.getZ() - terrainModel->getBoundingBox()->getMin().getZ()) / STEP_SIZE):-1;
931 auto terrainModelVerticesPerZ = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getZ() / STEP_SIZE)):-1;
932 auto terrainModelVerticesPerX = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getX() / STEP_SIZE)):-1;
933
934 if (terrainNode != nullptr &&
935 terrainModelX >= 0 &&
936 terrainModelX < terrainModelVerticesPerX &&
937 terrainModelZ >= 0 &&
938 terrainModelZ < terrainModelVerticesPerZ) {
939 if (partitionTerrainVertices[partitionIdx].empty() == true) {
940 partitionTerrainVertices[partitionIdx] = terrainNode->getVertices();
941 }
942 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
943 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
944 terrainVertices[vertexIdx + 2][1] = terrainVertexHeight;
945 }
946 }
947
948 //
949 brushPosition.add(
950 Vector3(
951 STEP_SIZE,
952 0.0f,
953 0.0f
954 )
955 );
956 }
957 }
958
959 // normals
960 for (auto z = -textureHeight * brushScaleMax * 2.0f; z < textureHeight * brushScaleMax * 2.0f; z+= STEP_SIZE) {
961 auto brushPosition =
962 brushCenterPosition.
963 clone().
964 sub(
965 Vector3(
966 static_cast<float>(textureWidth) * brushScaleMax * 2.0f,
967 0.0f,
968 0.0f
969 )
970 ).
971 add(
972 Vector3(
973 0.0f,
974 0.0f,
975 z
976 )
977 );
978 for (auto x = -textureWidth * brushScaleMax * 2.0f; x < textureWidth * brushScaleMax * 2.0f; x+= STEP_SIZE) {
979 auto partitionX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
980 auto partitionZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
981 auto partitionIdx = partitionZ * partitionsX + partitionX;
982 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:nullptr;
983 auto terrainNode = terrainModel != nullptr?terrainModel->getNodeById("terrain"):nullptr;
984
985 auto terrainHeightVectorX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / STEP_SIZE) + 1;
986 auto terrainHeightVectorZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / STEP_SIZE) + 1;
987 auto terrainModelX = terrainNode != nullptr?static_cast<int>((brushPosition.getX() - terrainModel->getBoundingBox()->getMin().getX()) / STEP_SIZE):-1;
988 auto terrainModelZ = terrainNode != nullptr?static_cast<int>((brushPosition.getZ() - terrainModel->getBoundingBox()->getMin().getZ()) / STEP_SIZE):-1;
989 auto terrainModelVerticesPerZ = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getZ() / STEP_SIZE)):-1;
990 auto terrainModelVerticesPerX = terrainNode != nullptr?static_cast<int>(Math::ceil(terrainModel->getBoundingBox()->getDimensions().getX() / STEP_SIZE)):-1;
991
992 if (terrainNode != nullptr &&
993 terrainModelX >= 0 &&
994 terrainModelX < terrainModelVerticesPerX &&
995 terrainModelZ >= 0 &&
996 terrainModelZ < terrainModelVerticesPerZ) {
997 if (partitionTerrainNormals[partitionIdx].empty() == true) {
998 partitionTerrainNormals[partitionIdx] = terrainNode->getNormals();
999 }
1000 auto& terrainNormals = partitionTerrainNormals[partitionIdx];
1001
1002 auto topNormal = computeTerrainVertexNormal(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX, terrainHeightVectorZ - 1);
1003 auto topLeftNormal = computeTerrainVertexNormal(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX - 1, terrainHeightVectorZ - 1);
1004 auto leftNormal = computeTerrainVertexNormal(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX - 1, terrainHeightVectorZ);
1005 auto normal = computeTerrainVertexNormal(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorX, terrainHeightVectorZ);
1006 auto normalIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
1007
1008 terrainNormals[normalIdx + 0] = topNormal;
1009 terrainNormals[normalIdx + 1] = topLeftNormal;
1010 terrainNormals[normalIdx + 2] = leftNormal;
1011 terrainNormals[normalIdx + 3] = normal;
1012 }
1013
1014 //
1015 brushPosition.add(
1016 Vector3(
1017 STEP_SIZE,
1018 0.0f,
1019 0.0f
1020 )
1021 );
1022 }
1023 }
1024
1025 // set terrain model vertices
1026 {
1027 auto partitionIdx = 0;
1028 for (auto& terrainVertices: partitionTerrainVertices) {
1029 if (terrainVertices.empty() == false) {
1030 auto terrainNode = terrainModels[partitionIdx]->getNodeById("terrain");
1031 if (terrainNode != nullptr) {
1032 terrainNode->setVertices(terrainVertices);
1033 terrainModels[partitionIdx]->invalidateBoundingBox();
1034 }
1035 }
1036 partitionIdx++;
1037 }
1038 }
1039 // set terrain model normals
1040 {
1041 auto partitionIdx = 0;
1042 for (auto& terrainNormals: partitionTerrainNormals) {
1043 if (terrainNormals.empty() == false) {
1044 auto terrainNode = terrainModels[partitionIdx]->getNodeById("terrain");
1045 if (terrainNode != nullptr) {
1046 terrainNode->setNormals(terrainNormals);
1047 terrainModels[partitionIdx]->invalidateBoundingBox();
1048 }
1049 }
1050 partitionIdx++;
1051 }
1052 }
1053}
1054
1055bool Terrain2::computeWaterPositionMap(BoundingBox& terrainBoundingBox, const vector<float>& terrainHeightVector, const Vector3& brushCenterPosition, float waterHeight, unordered_map<int, unordered_set<int>>& waterPositionMap) {
1056 auto terrainHeightVectorVerticesPerX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / STEP_SIZE));
1057 auto terreinHeightVectorVerticesPerZ = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getZ() / STEP_SIZE));
1058
1059 auto brushPosition = brushCenterPosition;
1060 auto terrainHeightVectorXCenter = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / STEP_SIZE);
1061 auto terrainHeightVectorZCenter = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / STEP_SIZE);
1062
1063 //
1064 waterPositionMap[terrainHeightVectorZCenter].insert(terrainHeightVectorXCenter);
1065
1066 //
1067 Console::println("Terrain2::determineWaterPositionSet: " + to_string(terrainHeightVectorXCenter) + " / " + to_string(terrainHeightVectorZCenter) + " @ " + to_string(waterHeight));
1068
1069 //
1070 determineWaterXPositionSet(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorXCenter, terrainHeightVectorZCenter, waterHeight, waterPositionMap[terrainHeightVectorZCenter]);
1071
1072 //
1073 {
1074 auto zLast = 0;
1075 auto zMin = -1;
1076 while (true == true) {
1077 auto terrainHeightVectorZLast = terrainHeightVectorZCenter + zLast;
1078 auto terrainHeightVectorZ = terrainHeightVectorZCenter + zMin;
1079 for (auto zLastWaterXPosition: waterPositionMap[terrainHeightVectorZLast]) {
1080 if (determineWater(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, zLastWaterXPosition, terrainHeightVectorZ, waterHeight) == true) {
1081 determineWaterXPositionSet(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, zLastWaterXPosition, terrainHeightVectorZ, waterHeight, waterPositionMap[terrainHeightVectorZ]);
1082 }
1083 }
1084 if (waterPositionMap[terrainHeightVectorZ].empty() == true) break;
1085 zLast = zMin;
1086 zMin--;
1087 }
1088 }
1089
1090 //
1091 {
1092 auto zLast = 0;
1093 auto zMax = 1;
1094 while (true == true) {
1095 auto terrainHeightVectorZLast = terrainHeightVectorZCenter + zLast;
1096 auto terrainHeightVectorZ = terrainHeightVectorZCenter + zMax;
1097 for (auto zLastWaterXPosition: waterPositionMap[terrainHeightVectorZLast]) {
1098 if (determineWater(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, zLastWaterXPosition, terrainHeightVectorZ, waterHeight) == true) {
1099 determineWaterXPositionSet(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, zLastWaterXPosition, terrainHeightVectorZ, waterHeight, waterPositionMap[terrainHeightVectorZ]);
1100 }
1101 }
1102 if (waterPositionMap[terrainHeightVectorZ].empty() == true) break;
1103 zLast = zMax;
1104 zMax++;
1105 }
1106 }
1107
1108 {
1109 auto waterPositionMapCopy = waterPositionMap;
1110 auto zMin = waterPositionMap.begin()->first;
1111 auto zMax = waterPositionMap.begin()->first;
1112 do {
1113 waterPositionMapCopy = waterPositionMap;
1114 for (auto& waterPositionMapIt: waterPositionMap) {
1115 auto z = waterPositionMapIt.first;
1116 for (auto x: waterPositionMapIt.second) {
1117 if (hasWaterPosition(waterPositionMap, x, z - 1) == false &&
1118 determineWater(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, x, z - 1, waterHeight) == true) {
1120 terrainBoundingBox,
1121 terrainHeightVector,
1122 Vector3(
1123 terrainBoundingBox.getMin().getX() + x * STEP_SIZE,
1124 waterHeight,
1125 terrainBoundingBox.getMin().getZ() + ((z - 1) * STEP_SIZE)
1126 ),
1127 waterHeight,
1128 waterPositionMap
1129 );
1130 } else
1131 if (hasWaterPosition(waterPositionMap, x, z + 1) == false &&
1132 determineWater(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, x, z + 1, waterHeight) == true) {
1134 terrainBoundingBox,
1135 terrainHeightVector,
1136 Vector3(
1137 terrainBoundingBox.getMin().getX() + x * STEP_SIZE,
1138 waterHeight,
1139 terrainBoundingBox.getMin().getZ() + ((z + 1) * STEP_SIZE)
1140 ),
1141 waterHeight,
1142 waterPositionMap
1143 );
1144 }
1145 }
1146 }
1147 } while (waterPositionMapCopy.size() != waterPositionMap.size());
1148 waterPositionMap[zMax] = waterPositionMap[zMax - 1];
1149 }
1150
1151 //
1152 auto haveWaterPositionSet = waterPositionMap.empty() == false;
1153 Console::println("Terrain2::determineWaterPositionSet: Have water position set: " + to_string(haveWaterPositionSet));
1154 return haveWaterPositionSet;
1155}
1156
1157Vector3 Terrain2::computeWaterReflectionEnvironmentMappingPosition(const unordered_map<int, unordered_set<int>>& waterPositionMap, float waterHeight) {
1158 // determine reflection environment mapping position
1159 auto zMin = Integer::MAX_VALUE;
1160 auto zMax = Integer::MIN_VALUE;
1161 auto xMin = Integer::MAX_VALUE;
1162 auto xMax = Integer::MIN_VALUE;
1163 for (auto& mIt: waterPositionMap) {
1164 auto z = mIt.first;
1165 if (z < zMin) zMin = z;
1166 if (z > zMax) zMax = z;
1167 for (auto x: mIt.second) {
1168 if (x < xMin) xMin = x;
1169 if (x > xMax) xMax = x;
1170 }
1171 }
1172
1173 return Vector3(
1174 (static_cast<float>(xMin + xMax) / 2.0f) * STEP_SIZE,
1175 waterHeight + 2.0f,
1176 (static_cast<float>(zMin + zMax) / 2.0f) * STEP_SIZE
1177 );
1178}
1179
1181 BoundingBox& terrainBoundingBox,
1182 const unordered_map<int, unordered_set<int>>& waterPositionMap,
1183 float waterHeight,
1184 int waterModelIdx,
1185 vector<Model*>& waterModels
1186) {
1187 auto width = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX()));
1188 auto depth = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getZ()));
1189 auto partitionsX = static_cast<int>(Math::ceil(width / PARTITION_SIZE));
1190 auto partitionsZ = static_cast<int>(Math::ceil(depth / PARTITION_SIZE));
1191 auto partitionCount = partitionsX * partitionsZ;
1192 vector<vector<Vector3>> partitionTerrainVertices;
1193 vector<vector<Vector3>> partitionTerrainNormals;
1194 vector<vector<array<int, 6>>> partitionWaterFaces;
1195 partitionTerrainVertices.resize(partitionCount);
1196 partitionTerrainNormals.resize(partitionCount);
1197 partitionWaterFaces.resize(partitionCount);
1198 for (float z = 0.0f; z < depth; z+= STEP_SIZE) {
1199 for (float x = 0.0f; x < width; x+= STEP_SIZE) {
1200 auto terrainHeightVectorX = static_cast<int>(x / STEP_SIZE);
1201 auto terrainHeightVectorZ = static_cast<int>(z / STEP_SIZE);
1202
1203 Vector3 topVertex;
1204 Vector3 topLeftVertex;
1205 Vector3 leftVertex;
1206 Vector3 vertex;
1207
1208 auto hasTopLeft = hasWaterPosition(waterPositionMap, terrainHeightVectorX - 1, terrainHeightVectorZ - 1);
1209 auto hasTop = hasWaterPosition(waterPositionMap, terrainHeightVectorX, terrainHeightVectorZ - 1);
1210 auto hasLeft = hasWaterPosition(waterPositionMap, terrainHeightVectorX - 1, terrainHeightVectorZ);
1211 auto hasOrigin = hasWaterPosition(waterPositionMap, terrainHeightVectorX, terrainHeightVectorZ);
1212
1213 auto haveVertexCount = 0;
1214 if (hasTop == true) haveVertexCount++;
1215 if (hasTopLeft == true) haveVertexCount++;
1216 if (hasLeft == true) haveVertexCount++;
1217 if (hasOrigin == true) haveVertexCount++;
1218 if (haveVertexCount < 3) continue;
1219
1220 auto partitionX = static_cast<int>(x / PARTITION_SIZE);
1221 auto partitionZ = static_cast<int>(z / PARTITION_SIZE);
1222 auto partitionIdx = partitionZ * partitionsX + partitionX;
1223
1224 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
1225 auto& terrainNormals = partitionTerrainNormals[partitionIdx];
1226 auto& terrainFaces = partitionWaterFaces[partitionIdx];
1227
1228 int normalIdx = terrainNormals.size();
1229 int vertexIdx = terrainVertices.size();
1230
1231 getWaterVertex(terrainHeightVectorX, terrainHeightVectorZ - 1, waterHeight, topVertex);
1232 getWaterVertex(terrainHeightVectorX - 1, terrainHeightVectorZ - 1, waterHeight, topLeftVertex);
1233 getWaterVertex(terrainHeightVectorX - 1, terrainHeightVectorZ, waterHeight, leftVertex);
1234 getWaterVertex(terrainHeightVectorX, terrainHeightVectorZ, waterHeight, vertex);
1235
1236 if (hasTop == true) terrainVertices.push_back(topVertex);
1237 if (hasTopLeft == true) terrainVertices.push_back(topLeftVertex);
1238 if (hasLeft == true) terrainVertices.push_back(leftVertex);
1239 if (hasOrigin == true) terrainVertices.push_back(vertex);
1240
1241 auto normal = Vector3(0.0f, 1.0f, 0.0f);
1242 auto topNormal = Vector3(0.0f, 1.0f, 0.0f);
1243 auto topLeftNormal = Vector3(0.0f, 1.0f, 0.0f);
1244 auto leftNormal = Vector3(0.0f, 1.0f, 0.0f);
1245
1246 if (hasTop == true) terrainNormals.push_back(topNormal);
1247 if (hasTopLeft == true) terrainNormals.push_back(topLeftNormal);
1248 if (hasLeft == true) terrainNormals.push_back(leftNormal);
1249 if (hasOrigin == true) terrainNormals.push_back(normal);
1250
1251 if (hasTopLeft == false ||
1252 hasTop == false ||
1253 hasLeft == false ||
1254 hasOrigin == false) {
1255 terrainFaces.push_back(
1256 {
1257 vertexIdx + 0,
1258 vertexIdx + 1,
1259 vertexIdx + 2,
1260 normalIdx + 0,
1261 normalIdx + 1,
1262 normalIdx + 2
1263 }
1264 );
1265 } else {
1266 terrainFaces.push_back(
1267 {
1268 vertexIdx + 0,
1269 vertexIdx + 1,
1270 vertexIdx + 2,
1271 normalIdx + 0,
1272 normalIdx + 1,
1273 normalIdx + 2
1274 }
1275 );
1276 terrainFaces.push_back(
1277 {
1278 vertexIdx + 2,
1279 vertexIdx + 3,
1280 vertexIdx + 0,
1281 normalIdx + 2,
1282 normalIdx + 3,
1283 normalIdx + 0
1284 }
1285 );
1286 }
1287 }
1288 }
1289 for (auto partitionIdx = 0; partitionIdx < partitionCount; partitionIdx++) {
1290 if (partitionWaterFaces[partitionIdx].empty() == true) continue;
1291 auto modelId = "water." + to_string(waterModelIdx) + "." + to_string(partitionIdx);
1292 auto waterModel = new Model(modelId, modelId, UpVector::Y_UP, RotationOrder::ZYX, nullptr);
1293 auto waterMaterial = new Material("water");
1294 waterMaterial->setDoubleSided(true);
1295 waterMaterial->setSpecularMaterialProperties(new SpecularMaterialProperties());
1296 waterMaterial->getSpecularMaterialProperties()->setAmbientColor(Color4(0.022f, 0.13f, 0.56f, 1.0f));
1297 waterMaterial->getSpecularMaterialProperties()->setDiffuseColor(Color4(0.026f, 0.15f, 0.64f, 1.0f));
1298 waterMaterial->getSpecularMaterialProperties()->setSpecularColor(Color4(1.0f, 1.0f, 1.0f, 1.0f));
1299 waterMaterial->getSpecularMaterialProperties()->setShininess(100.0f);
1300 waterModel->getMaterials()[waterMaterial->getId()] = waterMaterial;
1301 auto waterNode = new Node(waterModel, nullptr, "water", "water");
1302 FacesEntity nodeFacesEntityWater(waterNode, "water.facesentity");
1303 nodeFacesEntityWater.setMaterial(waterMaterial);
1304 vector<FacesEntity> nodeFacesEntities;
1305 vector<Face> nodeFaces;
1306 for (auto faceIndices: partitionWaterFaces[partitionIdx]) {
1307 nodeFaces.push_back(
1308 Face(
1309 waterNode,
1310 faceIndices[0],
1311 faceIndices[1],
1312 faceIndices[2],
1313 faceIndices[3],
1314 faceIndices[4],
1315 faceIndices[5]
1316 )
1317 );
1318 };
1319 nodeFacesEntityWater.setFaces(nodeFaces);
1320 nodeFacesEntities.push_back(nodeFacesEntityWater);
1321 waterNode->setVertices(partitionTerrainVertices[partitionIdx]);
1322 waterNode->setNormals(partitionTerrainNormals[partitionIdx]);
1323 waterNode->setFacesEntities(nodeFacesEntities);
1324 waterModel->getNodes()[waterNode->getId()] = waterNode;
1325 waterModel->getSubNodes()[waterNode->getId()] = waterNode;
1328 waterModels.push_back(waterModel);
1329 }
1330}
1331
1333 BoundingBox& terrainBoundingBox,
1334 vector<Model*>& terrainModels,
1335 vector<float>& terrainHeightVector,
1336 const Vector3& brushCenterPosition,
1337 float& brushHeight
1338) {
1339 // check if we have a model
1340 if (terrainModels.empty() == true) return false;
1341
1342 // get height at brush position
1343 auto terrainHeightVectorVerticesPerX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / STEP_SIZE));
1344 auto terreinHeightVectorVerticesPerZ = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getZ() / STEP_SIZE));
1345 auto terrainHeightVectorX = static_cast<int>((brushCenterPosition.getX() - terrainBoundingBox.getMin().getX()) / STEP_SIZE);
1346 auto terrainHeightVectorZ = static_cast<int>((brushCenterPosition.getZ() - terrainBoundingBox.getMin().getZ()) / STEP_SIZE);
1347 if (terrainHeightVectorX < 0 || terrainHeightVectorX >= terreinHeightVectorVerticesPerZ ||
1348 terrainHeightVectorZ < 0 || terrainHeightVectorZ >= terreinHeightVectorVerticesPerZ) return false;
1349 brushHeight = terrainHeightVector[terrainHeightVectorZ * terrainHeightVectorVerticesPerX + terrainHeightVectorX];
1350
1351 //
1352 return true;
1353}
1354
1356 BoundingBox& terrainBoundingBox,
1357 vector<unordered_map<int, vector<Transformations>>>& foliageMaps
1358) {
1359 //
1360 auto width = terrainBoundingBox.getDimensions().getX();
1361 auto depth = terrainBoundingBox.getDimensions().getZ();
1362 createFoliageMaps(width, depth, foliageMaps);
1363}
1364
1366 float terrainWidth,
1367 float terrainDepth,
1368 vector<unordered_map<int, vector<Transformations>>>& foliageMaps
1369) {
1370 //
1371 auto partitionsX = static_cast<int>(Math::ceil(terrainWidth / PARTITION_SIZE));
1372 auto partitionsZ = static_cast<int>(Math::ceil(terrainDepth / PARTITION_SIZE));
1373 auto partitionCount = partitionsX * partitionsZ;
1374 foliageMaps.resize(partitionCount);
1375 for (auto& foliageMap: foliageMaps) foliageMap.clear();
1376}
1377
1379 vector<unordered_map<int, vector<Transformations>>>& foliageMaps
1380) {
1381 //
1382 for (auto& foliageMap: foliageMaps) foliageMap.clear();
1383}
1384
1386 BoundingBox& terrainBoundingBox,
1387 vector<float>& terrainHeightVector,
1388 const Vector3& brushCenterPosition,
1389 const FoliageBrush& foliageBrush,
1390 const vector<FoliageBrushPrototype>& foliageBrushPrototypes,
1391 BrushOperation brushOperation,
1392 vector<unordered_map<int, vector<Transformations>>>& foliageMaps,
1393 vector<unordered_map<int, vector<Transformations>>>& newFoliageMaps
1394) {
1395 // check if we have a texture
1396 if (foliageBrush.brushTexture == nullptr) return;
1397
1398 // apply brush
1399 auto partitionsX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / PARTITION_SIZE));
1400 auto terrainHeightVectorVerticesPerX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / STEP_SIZE));
1401 auto terreinHeightVectorVerticesPerZ = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getZ() / STEP_SIZE));
1402
1403 // other operations
1404 auto textureData = foliageBrush.brushTexture->getTextureData();
1405 auto textureWidth = foliageBrush.brushTexture->getTextureWidth();
1406 auto textureHeight = foliageBrush.brushTexture->getTextureHeight();
1407 auto textureBytePerPixel = foliageBrush.brushTexture->getDepth() == 32?4:3;
1408
1409 //
1410 vector<unordered_map<int, float>> brushMapCountMapTemplate;
1411 auto brushMapCountMapWidth = static_cast<int>(textureWidth * foliageBrush.brushScale);
1412 auto brushMapCountMapDepth = static_cast<int>(textureHeight * foliageBrush.brushScale);
1413 for (auto z = 0.0f; z < textureHeight * foliageBrush.brushScale; z+= 1.0f) {
1414 for (auto x = 0.0f; x < textureWidth * foliageBrush.brushScale; x+= 1.0f) {
1415 auto textureX = static_cast<int>(x / foliageBrush.brushScale);
1416 auto textureY = static_cast<int>(z / foliageBrush.brushScale);
1417 auto red = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 0);
1418 auto green = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 1);
1419 auto blue = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 2);
1420 auto alpha = textureBytePerPixel == 3?255:textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 3);
1421 auto appliedDensity = (static_cast<float>(red) + static_cast<float>(green) + static_cast<float>(blue)) / (255.0f * 3.0f) * foliageBrush.brushDensity;
1422 unordered_map<int, float> brushMapCountMapEntity;
1423 for (auto& foliageBrushPrototype: foliageBrushPrototypes) {
1424 if (foliageBrushPrototype.prototypeId == -1) continue;
1425 auto foliageCount = foliageBrushPrototype.count;
1426 brushMapCountMapEntity[foliageBrushPrototype.prototypeId] = foliageCount * appliedDensity;
1427 }
1428 brushMapCountMapTemplate.push_back(brushMapCountMapEntity);
1429 }
1430 }
1431
1432 //
1433 auto brushMapCountMap = brushMapCountMapTemplate;
1434 for (auto& foliageBrushPrototype: foliageBrushPrototypes) {
1435 if (foliageBrushPrototype.prototypeId == -1) continue;
1436 float totalCount = 0.0f;
1437 for (auto i = 0; i < brushMapCountMap.size(); i++) {
1438 auto& brushMapCountMapEntity = brushMapCountMap[i];
1439 auto count = brushMapCountMapEntity[foliageBrushPrototype.prototypeId];
1440 auto countFloor = Math::floor(count);
1441 totalCount+= count - countFloor;
1442 brushMapCountMapEntity[foliageBrushPrototype.prototypeId] = countFloor;
1443 auto totalCountFloor = Math::floor(totalCount);
1444 if (totalCount >= 1.0f) brushMapCountMapEntity[foliageBrushPrototype.prototypeId]+= totalCountFloor;
1445 totalCount-= totalCountFloor;
1446 }
1447 }
1448
1449 // randomize
1450 unordered_map<int, unordered_map<int, vector<int>>> brushMapIdxPerDensityPerPrototype;
1451 for (auto& foliageBrushPrototype: foliageBrushPrototypes) {
1452 for (auto i = 0; i < brushMapCountMapTemplate.size(); i++) {
1453 auto brushMapPrototypeCount = brushMapCountMapTemplate[i][foliageBrushPrototype.prototypeId];
1454 brushMapIdxPerDensityPerPrototype[foliageBrushPrototype.prototypeId][static_cast<int>(brushMapPrototypeCount * 1000.0f)].push_back(i);
1455 }
1456 }
1457 for (auto& foliageBrushPrototype: foliageBrushPrototypes) {
1458 for (auto i = 0; i < brushMapCountMap.size(); i++) {
1459 auto brushMapPrototypeCountMapEntityITemplate = brushMapCountMapTemplate[i][foliageBrushPrototype.prototypeId];
1460 auto brushMapPrototypeCountMapEntityI = brushMapCountMap[i][foliageBrushPrototype.prototypeId];
1461 auto& brushMapIdxPerDensityPerPrototypeVector = brushMapIdxPerDensityPerPrototype[foliageBrushPrototype.prototypeId][static_cast<int>(brushMapPrototypeCountMapEntityITemplate * 1000.0f)];
1462 auto j = brushMapIdxPerDensityPerPrototypeVector[static_cast<int>(brushMapIdxPerDensityPerPrototypeVector.size() - 1) * Math::random()];
1463 auto brushMapPrototypeCountMapEntityJ = brushMapCountMap[j][foliageBrushPrototype.prototypeId];
1464 brushMapCountMap[j][foliageBrushPrototype.prototypeId] = brushMapPrototypeCountMapEntityI;
1465 brushMapCountMap[i][foliageBrushPrototype.prototypeId] = brushMapPrototypeCountMapEntityJ;
1466 }
1467 }
1468
1469 //
1470 for (auto z = 0.0f; z < textureHeight * foliageBrush.brushScale; z+= 1.0f) {
1471 auto brushPosition =
1472 brushCenterPosition.
1473 clone().
1474 sub(
1475 Vector3(
1476 (static_cast<float>(textureWidth) * foliageBrush.brushScale) / 2.0f,
1477 0.0f,
1478 ((static_cast<float>(textureHeight) * foliageBrush.brushScale) / 2.0f)
1479 )
1480 ).
1481 add(
1482 Vector3(
1483 0.0f,
1484 0.0f,
1485 z
1486 )
1487 );
1488 for (auto x = 0.0f; x < textureWidth * foliageBrush.brushScale; x+= 1.0f) {
1489 auto brushMapCountMapX = static_cast<int>(x);
1490 auto brushMapCountMapZ = static_cast<int>(z);
1491 auto brushMapCountMapEntity = brushMapCountMap[brushMapCountMapZ * brushMapCountMapWidth + brushMapCountMapX];
1492 auto terrainHeightVectorX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / STEP_SIZE);
1493 auto terrainHeightVectorZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / STEP_SIZE);
1494 if (brushPosition.getX() < 0.0f || brushPosition.getZ() < 0.0f ||
1495 terrainHeightVectorX < 0 || terrainHeightVectorX >= terrainHeightVectorVerticesPerX ||
1496 terrainHeightVectorZ < 0 || terrainHeightVectorZ >= terreinHeightVectorVerticesPerZ) {
1497 //
1498 brushPosition.add(
1499 Vector3(
1500 STEP_SIZE,
1501 0.0f,
1502 0.0f
1503 )
1504 );
1505 continue;
1506 }
1507
1508 //
1509 switch(brushOperation) {
1510 case BRUSHOPERATION_ADD:
1511 for (auto& brushMapCountMapEntityIt: brushMapCountMapEntity) {
1512 auto prototypeId = brushMapCountMapEntityIt.first;
1513 if (prototypeId == -1) continue;
1514 auto prototypeCount = brushMapCountMapEntityIt.second;
1515
1516 auto prototypeIdx = -1;
1517 for (auto i = 0; i < foliageBrushPrototypes.size(); i++) {
1518 if (foliageBrushPrototypes[i].prototypeId == prototypeId) prototypeIdx = i;
1519 }
1520 if (prototypeIdx == -1) continue;
1521
1522 //
1523 for (auto i = 0; i < static_cast<int>(prototypeCount); i++) {
1524 auto prototypeScale = foliageBrushPrototypes[prototypeIdx].scaleMin + ((foliageBrushPrototypes[prototypeIdx].scaleMax - foliageBrushPrototypes[prototypeIdx].scaleMin) * Math::random());
1525
1526 //
1527 Vector3 translation(
1528 Math::floor(brushPosition.getX()) + Math::random(),
1529 0.0f,
1530 Math::floor(brushPosition.getZ()) + Math::random()
1531 );
1532
1533 //
1534 auto partitionX = static_cast<int>((translation.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
1535 auto partitionZ = static_cast<int>((translation.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
1536 auto partitionIdx = partitionZ * partitionsX + partitionX;
1537
1538 //
1539 auto haveContact = false;
1540 Vector3 contact;
1541 Vector3 normal;
1542 float height;
1543 for (int _z = -1; _z < 2; _z++)
1544 for (int _x = -1; _x < 2; _x++) {
1545 Vector3 topVertex;
1546 Vector3 topLeftVertex;
1547 Vector3 leftVertex;
1548 Vector3 vertex;
1549
1550 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, _x + terrainHeightVectorX, _z + terrainHeightVectorZ - 1, topVertex);
1551 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, _x + terrainHeightVectorX - 1, _z + terrainHeightVectorZ - 1, topLeftVertex);
1552 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, _x + terrainHeightVectorX - 1, _z + terrainHeightVectorZ, leftVertex);
1553 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, _x + terrainHeightVectorX, _z + terrainHeightVectorZ, vertex);
1554
1555 if (LineSegment::doesLineSegmentCollideWithTriangle(topVertex, topLeftVertex, leftVertex, translation.clone().setY(-10000.0f), translation.clone().setY(+10000.0f), contact) == true) {
1556 haveContact = true;
1557 normal = ModelTools::computeNormal({topVertex, topLeftVertex, leftVertex});
1558 height = (topVertex.getY() + topLeftVertex.getY() + leftVertex.getY()) / 3.0f;
1559 break;
1560 } else
1561 if (LineSegment::doesLineSegmentCollideWithTriangle(leftVertex, vertex, topVertex, translation.clone().setY(-10000.0f), translation.clone().setY(+10000.0f), contact) == true) {
1562 haveContact = true;
1563 normal = ModelTools::computeNormal({leftVertex, vertex, topVertex});
1564 height = (leftVertex.getY() + vertex.getY() + topVertex.getY()) / 3.0f;
1565 break;
1566 }
1567 }
1568
1569 // check height
1570 if (height < foliageBrushPrototypes[prototypeIdx].heightMin || height > foliageBrushPrototypes[prototypeIdx].heightMax) continue;
1571
1572 //
1573 if (haveContact == false) {
1575 "Terrain2::applyFoliageBrush(): no contact@" +
1576 to_string(translation.getX()) + ", " +
1577 to_string(translation.getZ())
1578 );
1579 contact = translation;
1580 continue;
1581 }
1582
1583 // slope
1584 auto slope = Math::abs(180.0f / 3.14f * Math::acos(Math::clamp(Vector3::computeDotProduct(normal, Vector3(0.0, 1.0, 0.0)), -1.0, 1.0)));
1585 if (slope < foliageBrushPrototypes[prototypeIdx].slopeMin || slope > foliageBrushPrototypes[prototypeIdx].slopeMax) continue;
1586
1587 //
1588 Transformations transformations;
1589 transformations.setTranslation(translation);
1590 auto xAxisRotation = foliageBrushPrototypes[prototypeIdx].rotationXMin + ((foliageBrushPrototypes[prototypeIdx].rotationXMax - foliageBrushPrototypes[prototypeIdx].rotationXMin) * Math::random());
1591 auto yAxisRotation = foliageBrushPrototypes[prototypeIdx].rotationYMin + ((foliageBrushPrototypes[prototypeIdx].rotationYMax - foliageBrushPrototypes[prototypeIdx].rotationYMin) * Math::random());
1592 auto zAxisRotation = foliageBrushPrototypes[prototypeIdx].rotationZMin + ((foliageBrushPrototypes[prototypeIdx].rotationZMax - foliageBrushPrototypes[prototypeIdx].rotationZMin) * Math::random());
1593 if (foliageBrushPrototypes[prototypeIdx].normalAlign == true) {
1594 xAxisRotation = Vector3::computeAngle(normal, Vector3(0.0f, 1.0f, 0.0f), Vector3(-1.0f, 0.0f, 0.0f));
1595 zAxisRotation = Vector3::computeAngle(normal, Vector3(0.0f, 1.0f, 0.0f), Vector3(0.0f, 0.0f, -1.0f));
1596 Transformations _transformations;
1597 _transformations.addRotation(Rotation::Z_AXIS, zAxisRotation);
1598 _transformations.addRotation(Rotation::X_AXIS, xAxisRotation);
1599 _transformations.addRotation(Rotation::Y_AXIS, yAxisRotation);
1600 _transformations.update();
1601 auto euler = _transformations.getTransformationsMatrix().computeEulerAngles();
1602 zAxisRotation = euler.getZ();
1603 yAxisRotation = euler.getY();
1604 xAxisRotation = euler.getX();
1605 }
1606 transformations.addRotation(Rotation::Z_AXIS, zAxisRotation);
1607 transformations.addRotation(Rotation::Y_AXIS, yAxisRotation);
1608 transformations.addRotation(Rotation::X_AXIS, xAxisRotation);
1609 transformations.setScale(Vector3(prototypeScale, prototypeScale, prototypeScale));
1610
1611 transformations.setTranslation(translation.clone().setY(contact.getY()));
1612 transformations.update();
1613
1614 //
1615 foliageMaps[partitionIdx][prototypeId].push_back(transformations);
1616 newFoliageMaps[partitionIdx][prototypeId].push_back(transformations);
1617 }
1618 }
1619 break;
1620 }
1621
1622 //
1623 brushPosition.add(
1624 Vector3(
1625 STEP_SIZE,
1626 0.0f,
1627 0.0f
1628 )
1629 );
1630 }
1631 }
1632}
1633
1635 BoundingBox& terrainBoundingBox, // TODO: constness
1636 const Vector3& brushCenterPosition,
1637 const FoliageBrush& foliageBrush,
1638 const vector<FoliageBrushPrototype>& foliageBrushPrototypes,
1639 BrushOperation brushOperation,
1640 vector<unordered_map<int, vector<Transformations>>>& foliageMaps,
1641 unordered_set<int>& recreateFoliagePartitions
1642) {
1643 // check if we have a texture
1644 if (foliageBrush.brushTexture == nullptr) return;
1645
1646 // apply brush
1647 auto partitionsX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / PARTITION_SIZE));
1648 auto terrainHeightVectorVerticesPerX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / STEP_SIZE));
1649 auto terreinHeightVectorVerticesPerZ = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getZ() / STEP_SIZE));
1650
1651 // other operations
1652 auto textureData = foliageBrush.brushTexture->getTextureData();
1653 auto textureWidth = foliageBrush.brushTexture->getTextureWidth();
1654 auto textureHeight = foliageBrush.brushTexture->getTextureHeight();
1655 auto textureBytePerPixel = foliageBrush.brushTexture->getDepth() == 32?4:3;
1656
1657 auto heightMin = Float::MAX_VALUE;
1658 auto heightMax = Float::MIN_VALUE;
1659 auto prototypeCount = 0;
1660 for (auto& foliageBrushPrototype: foliageBrushPrototypes) {
1661 if (foliageBrushPrototype.prototypeId == -1) continue;
1662 heightMin = Math::min(heightMin, foliageBrushPrototype.heightMin);
1663 heightMax = Math::max(heightMax, foliageBrushPrototype.heightMax);
1664 prototypeCount++;
1665 }
1666 //
1667 for (auto z = 0.0f; z < textureHeight * foliageBrush.brushScale; z+= 1.0f) {
1668 auto brushPosition =
1669 brushCenterPosition.
1670 clone().
1671 sub(
1672 Vector3(
1673 (static_cast<float>(textureWidth) * foliageBrush.brushScale) / 2.0f,
1674 0.0f,
1675 ((static_cast<float>(textureHeight) * foliageBrush.brushScale) / 2.0f)
1676 )
1677 ).
1678 add(
1679 Vector3(
1680 0.0f,
1681 0.0f,
1682 z
1683 )
1684 );
1685 for (auto x = 0.0f; x < textureWidth * foliageBrush.brushScale; x+= 1.0f) {
1686 auto textureX = static_cast<int>(x / foliageBrush.brushScale);
1687 auto textureY = static_cast<int>(z / foliageBrush.brushScale);
1688 auto red = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 0);
1689 auto green = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 1);
1690 auto blue = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 2);
1691 auto alpha = textureBytePerPixel == 3?255:textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 3);
1692 auto appliedDensity = (static_cast<float>(red) + static_cast<float>(green) + static_cast<float>(blue)) / (255.0f * 3.0f);
1693
1694 //
1695 auto terrainHeightVectorX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / STEP_SIZE);
1696 auto terrainHeightVectorZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / STEP_SIZE);
1697
1698 //
1699 if (terrainHeightVectorX < 0 || terrainHeightVectorX >= terrainHeightVectorVerticesPerX ||
1700 terrainHeightVectorZ < 0 || terrainHeightVectorZ >= terreinHeightVectorVerticesPerZ) {
1701 //
1702 brushPosition.add(
1703 Vector3(
1704 STEP_SIZE,
1705 0.0f,
1706 0.0f
1707 )
1708 );
1709 continue;
1710 }
1711
1712 //
1713 auto partitionX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
1714 auto partitionZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
1715 auto partitionIdx = partitionZ * partitionsX + partitionX;
1716
1717 //
1718 switch(brushOperation) {
1720 {
1721 Vector3 topVertex;
1722 Vector3 topLeftVertex;
1723 Vector3 leftVertex;
1724 Vector3 vertex;
1725
1726 getTerrainVertex(terrainHeightVectorX, terrainHeightVectorZ - 1, topVertex);
1727 getTerrainVertex(terrainHeightVectorX - 1, terrainHeightVectorZ - 1, topLeftVertex);
1728 getTerrainVertex(terrainHeightVectorX - 1, terrainHeightVectorZ, leftVertex);
1729 getTerrainVertex(terrainHeightVectorX, terrainHeightVectorZ, vertex);
1730
1731 for (auto& foliageMapPartitionIt: foliageMaps[partitionIdx]) {
1732 auto& foliageMapPartitionPrototypeTransformations = foliageMapPartitionIt.second;
1733 for (auto i = 0; i < foliageMapPartitionPrototypeTransformations.size(); i++) {
1734 auto& translation = foliageMapPartitionPrototypeTransformations[i].getTranslation();
1735 if (appliedDensity > 0.0f &&
1736 translation.getX() >= leftVertex.getX() - 0.01f &&
1737 translation.getX() <= vertex.getX() + 0.01f &&
1738 translation.getZ() >= topVertex.getZ() - 0.01f &&
1739 translation.getZ() <= vertex.getZ() + 0.01f &&
1740 (prototypeCount == 0 ||
1741 (translation.getY() >= heightMin &&
1742 translation.getY() <= heightMax))) {
1743 //
1744 foliageMapPartitionPrototypeTransformations.erase(foliageMapPartitionPrototypeTransformations.begin() + i);
1745 recreateFoliagePartitions.insert(partitionIdx);
1746 i--;
1747 }
1748 }
1749 }
1750 }
1751 break;
1752 }
1753
1754 //
1755 brushPosition.add(
1756 Vector3(
1757 STEP_SIZE,
1758 0.0f,
1759 0.0f
1760 )
1761 );
1762 }
1763 }
1764}
1765
1767 BoundingBox& terrainBoundingBox, // TODO: constness
1768 vector<float>& terrainHeightVector,
1769 const Vector3& brushCenterPosition,
1770 const FoliageBrush& foliageBrush,
1771 vector<unordered_map<int, vector<Transformations>>>& foliageMaps,
1772 unordered_set<int>& updateFoliagePartitions
1773) {
1774 // check if we have a texture
1775 if (foliageBrush.brushTexture == nullptr) return;
1776
1777 // apply brush
1778 auto partitionsX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / PARTITION_SIZE));
1779 auto terrainHeightVectorVerticesPerX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / STEP_SIZE));
1780 auto terreinHeightVectorVerticesPerZ = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getZ() / STEP_SIZE));
1781
1782 // other operations
1783 auto textureData = foliageBrush.brushTexture->getTextureData();
1784 auto textureWidth = foliageBrush.brushTexture->getTextureWidth();
1785 auto textureHeight = foliageBrush.brushTexture->getTextureHeight();
1786 auto textureBytePerPixel = foliageBrush.brushTexture->getDepth() == 32?4:3;
1787
1788 //
1789 for (auto z = 0.0f; z < textureHeight * foliageBrush.brushScale; z+= 1.0f) {
1790 auto brushPosition =
1791 brushCenterPosition.
1792 clone().
1793 sub(
1794 Vector3(
1795 (static_cast<float>(textureWidth) * foliageBrush.brushScale) / 2.0f,
1796 0.0f,
1797 ((static_cast<float>(textureHeight) * foliageBrush.brushScale) / 2.0f)
1798 )
1799 ).
1800 add(
1801 Vector3(
1802 0.0f,
1803 0.0f,
1804 z
1805 )
1806 );
1807 for (auto x = 0.0f; x < textureWidth * foliageBrush.brushScale; x+= 1.0f) {
1808 auto brushMapCountMapX = static_cast<int>(x);
1809 auto brushMapCountMapZ = static_cast<int>(z);
1810 auto terrainHeightVectorX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / STEP_SIZE);
1811 auto terrainHeightVectorZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / STEP_SIZE);
1812 if (terrainHeightVectorX < 0 || terrainHeightVectorX >= terrainHeightVectorVerticesPerX ||
1813 terrainHeightVectorZ < 0 || terrainHeightVectorZ >= terreinHeightVectorVerticesPerZ) {
1814 //
1815 brushPosition.add(
1816 Vector3(
1817 STEP_SIZE,
1818 0.0f,
1819 0.0f
1820 )
1821 );
1822 continue;
1823 }
1824
1825 //
1826 auto textureX = static_cast<int>(x / foliageBrush.brushScale);
1827 auto textureY = static_cast<int>(z / foliageBrush.brushScale);
1828 auto red = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 0);
1829 auto green = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 1);
1830 auto blue = textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 2);
1831 auto alpha = textureBytePerPixel == 3?255:textureData->get(textureY * textureWidth * textureBytePerPixel + textureX * textureBytePerPixel + 3);
1832 auto brushTextureDensity = (static_cast<float>(red) + static_cast<float>(green) + static_cast<float>(blue)) / (255.0f * 3.0f);
1833
1834 //
1835 auto partitionX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
1836 auto partitionZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
1837 auto partitionIdx = partitionZ * partitionsX + partitionX;
1838
1839 //
1840 updateFoliagePartitions.insert(partitionIdx);
1841
1842 //
1843 Vector3 topVertex;
1844 Vector3 topLeftVertex;
1845 Vector3 leftVertex;
1846 Vector3 vertex;
1847
1848 //
1849 getTerrainVertex(terrainHeightVectorX, terrainHeightVectorZ - 1, topVertex);
1850 getTerrainVertex(terrainHeightVectorX - 1, terrainHeightVectorZ - 1, topLeftVertex);
1851 getTerrainVertex(terrainHeightVectorX - 1, terrainHeightVectorZ, leftVertex);
1852 getTerrainVertex(terrainHeightVectorX, terrainHeightVectorZ, vertex);
1853
1854 //
1855 for (auto& foliageMapPartitionIt: foliageMaps[partitionIdx]) {
1856 auto prototypeId = foliageMapPartitionIt.first;
1857 if (prototypeId == -1) continue;
1858 auto& foliageMapPartitionPrototypeTransformations = foliageMapPartitionIt.second;
1859
1860 //
1861 for (auto& transformations: foliageMapPartitionPrototypeTransformations) {
1862 auto& translation = transformations.getTranslation();
1863 if (brushTextureDensity > 0.0f &&
1864 translation.getX() >= leftVertex.getX() &&
1865 translation.getX() <= vertex.getX() &&
1866 translation.getZ() >= topVertex.getZ() &&
1867 translation.getZ() <= vertex.getZ()) {
1868 //
1869 auto haveContact = false;
1870 Vector3 contact;
1871 for (int _z = -1; _z < 2; _z++)
1872 for (int _x = -1; _x < 2; _x++) {
1873 Vector3 topVertex;
1874 Vector3 topLeftVertex;
1875 Vector3 leftVertex;
1876 Vector3 vertex;
1877
1878 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, _x + terrainHeightVectorX, _z + terrainHeightVectorZ - 1, topVertex);
1879 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, _x + terrainHeightVectorX - 1, _z + terrainHeightVectorZ - 1, topLeftVertex);
1880 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, _x + terrainHeightVectorX - 1, _z + terrainHeightVectorZ, leftVertex);
1881 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, _x + terrainHeightVectorX, _z + terrainHeightVectorZ, vertex);
1882
1883 if (LineSegment::doesLineSegmentCollideWithTriangle(topVertex, topLeftVertex, leftVertex, transformations.getTranslation().clone().setY(-10000.0f), transformations.getTranslation().clone().setY(+10000.0f), contact) == true) {
1884 haveContact = true;
1885 break;
1886 } else
1887 if (LineSegment::doesLineSegmentCollideWithTriangle(leftVertex, vertex, topVertex, transformations.getTranslation().clone().setY(-10000.0f), transformations.getTranslation().clone().setY(+10000.0f), contact) == true) {
1888 haveContact = true;
1889 break;
1890 }
1891 }
1892
1893 //
1894 if (haveContact == false) {
1896 "Terrain2::applyFoliageBrush(): no contact@" +
1897 to_string(transformations.getTranslation().getX()) + ", " +
1898 to_string(transformations.getTranslation().getZ())
1899 );
1900 contact = transformations.getTranslation();
1901 }
1902
1903 //
1904 transformations.setTranslation(transformations.getTranslation().clone().setY(contact.getY()));
1905 transformations.update();
1906 }
1907 }
1908 }
1909
1910 //
1911 brushPosition.add(
1912 Vector3(
1913 STEP_SIZE,
1914 0.0f,
1915 0.0f
1916 )
1917 );
1918 }
1919 }
1920}
1921
1923 BoundingBox& terrainBoundingBox, // TODO: constness
1924 vector<float>& terrainHeightVector,
1925 const Vector3& brushCenterPosition,
1926 Texture* brushTexture,
1927 float brushRotation,
1928 const Vector2& brushScale,
1929 vector<unordered_map<int, vector<Transformations>>>& foliageMaps,
1930 unordered_set<int>& updateFoliagePartitions
1931) {
1932 // check if we have a texture
1933 if (brushTexture == nullptr) return;
1934
1935 // apply brush
1936 vector<vector<Vector3>> partitionTerrainVertices;
1937 vector<vector<Vector3>> partitionTerrainNormals;
1938 auto partitionsX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / PARTITION_SIZE));
1939 auto terrainHeightVectorVerticesPerX = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getX() / STEP_SIZE));
1940 auto terreinHeightVectorVerticesPerZ = static_cast<int>(Math::ceil(terrainBoundingBox.getDimensions().getZ() / STEP_SIZE));
1941
1942 // texture
1943 auto textureData = brushTexture->getTextureData();
1944 auto textureWidth = brushTexture->getTextureWidth();
1945 auto textureHeight = brushTexture->getTextureHeight();
1946 auto textureBytePerPixel = brushTexture->getDepth() == 32?4:3;
1947
1948 // brush texture matrix
1949 Matrix2D3x3 brushTextureMatrix;
1950 brushTextureMatrix.identity();
1951 brushTextureMatrix.translate(Vector2(static_cast<float>(textureWidth) / 2.0f, static_cast<float>(textureHeight) / 2.0f));
1952 brushTextureMatrix.multiply((Matrix2D3x3()).identity().scale(Vector2(1.0f / brushScale.getX(), 1.0f / brushScale.getY())));
1953 brushTextureMatrix.multiply((Matrix2D3x3()).identity().rotate(brushRotation));
1954 auto brushScaleMax = Math::max(brushScale.getX(), brushScale.getY());
1955
1956 //
1957 for (auto z = -textureHeight * brushScaleMax * 2.0f; z < textureHeight * brushScaleMax * 2.0f; z+= STEP_SIZE) {
1958 auto brushPosition =
1959 brushCenterPosition.
1960 clone().
1961 sub(
1962 Vector3(
1963 static_cast<float>(textureWidth) * brushScaleMax * 2.0f,
1964 0.0f,
1965 0.0f
1966 )
1967 ).
1968 add(
1969 Vector3(
1970 0.0f,
1971 0.0f,
1972 z
1973 )
1974 );
1975 for (auto x = -textureWidth * brushScaleMax * 2.0f; x < textureWidth * brushScaleMax * 2.0f; x+= STEP_SIZE) {
1976 auto texturePositionUntransformed = Vector2(x, z);
1977 auto texturePosition = brushTextureMatrix.multiply(texturePositionUntransformed);
1978 auto textureX = static_cast<int>(texturePosition.getX());
1979 auto textureY = static_cast<int>(texturePosition.getY());
1980 if (textureX < 0 || textureX >= textureWidth ||
1981 textureY < 0 || textureY >= textureHeight) {
1982 brushPosition.add(
1983 Vector3(
1984 STEP_SIZE,
1985 0.0f,
1986 0.0f
1987 )
1988 );
1989 continue;
1990 }
1991 auto terrainHeightVectorX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / STEP_SIZE);
1992 auto terrainHeightVectorZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / STEP_SIZE);
1993 if (terrainHeightVectorX < 0 || terrainHeightVectorX >= terrainHeightVectorVerticesPerX ||
1994 terrainHeightVectorZ < 0 || terrainHeightVectorZ >= terreinHeightVectorVerticesPerZ) {
1995 brushPosition.add(
1996 Vector3(
1997 STEP_SIZE,
1998 0.0f,
1999 0.0f
2000 )
2001 );
2002 continue;
2003 }
2004
2005 //
2006 auto partitionX = static_cast<int>((brushPosition.getX() - terrainBoundingBox.getMin().getX()) / PARTITION_SIZE);
2007 auto partitionZ = static_cast<int>((brushPosition.getZ() - terrainBoundingBox.getMin().getZ()) / PARTITION_SIZE);
2008 auto partitionIdx = partitionZ * partitionsX + partitionX;
2009
2010 //
2011 updateFoliagePartitions.insert(partitionIdx);
2012
2013 //
2014 Vector3 topVertex;
2015 Vector3 topLeftVertex;
2016 Vector3 leftVertex;
2017 Vector3 vertex;
2018
2019 //
2020 getTerrainVertex(terrainHeightVectorX, terrainHeightVectorZ - 1, topVertex);
2021 getTerrainVertex(terrainHeightVectorX - 1, terrainHeightVectorZ - 1, topLeftVertex);
2022 getTerrainVertex(terrainHeightVectorX - 1, terrainHeightVectorZ, leftVertex);
2023 getTerrainVertex(terrainHeightVectorX, terrainHeightVectorZ, vertex);
2024
2025 //
2026 for (auto& foliageMapPartitionIt: foliageMaps[partitionIdx]) {
2027 auto prototypeId = foliageMapPartitionIt.first;
2028 if (prototypeId == -1) continue;
2029 auto& foliageMapPartitionPrototypeTransformations = foliageMapPartitionIt.second;
2030
2031 //
2032 for (auto& transformations: foliageMapPartitionPrototypeTransformations) {
2033 auto& translation = transformations.getTranslation();
2034 if (translation.getX() >= leftVertex.getX() &&
2035 translation.getX() <= vertex.getX() &&
2036 translation.getZ() >= topVertex.getZ() &&
2037 translation.getZ() <= vertex.getZ()) {
2038 //
2039 auto haveContact = false;
2040 Vector3 contact;
2041 for (int _z = -1; _z < 2; _z++)
2042 for (int _x = -1; _x < 2; _x++) {
2043 Vector3 topVertex;
2044 Vector3 topLeftVertex;
2045 Vector3 leftVertex;
2046 Vector3 vertex;
2047
2048 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, _x + terrainHeightVectorX, _z + terrainHeightVectorZ - 1, topVertex);
2049 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, _x + terrainHeightVectorX - 1, _z + terrainHeightVectorZ - 1, topLeftVertex);
2050 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, _x + terrainHeightVectorX - 1, _z + terrainHeightVectorZ, leftVertex);
2051 getTerrainVertex(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, _x + terrainHeightVectorX, _z + terrainHeightVectorZ, vertex);
2052
2053 if (LineSegment::doesLineSegmentCollideWithTriangle(topVertex, topLeftVertex, leftVertex, transformations.getTranslation().clone().setY(-10000.0f), transformations.getTranslation().clone().setY(+10000.0f), contact) == true) {
2054 haveContact = true;
2055 break;
2056 } else
2057 if (LineSegment::doesLineSegmentCollideWithTriangle(leftVertex, vertex, topVertex, transformations.getTranslation().clone().setY(-10000.0f), transformations.getTranslation().clone().setY(+10000.0f), contact) == true) {
2058 haveContact = true;
2059 break;
2060 }
2061 }
2062
2063 //
2064 if (haveContact == false) {
2066 "Terrain2::applyFoliageBrush(): no contact@" +
2067 to_string(transformations.getTranslation().getX()) + ", " +
2068 to_string(transformations.getTranslation().getZ())
2069 );
2070 contact = transformations.getTranslation();
2071 }
2072
2073 //
2074 transformations.setTranslation(transformations.getTranslation().clone().setY(contact.getY()));
2075 transformations.update();
2076 }
2077 }
2078 }
2079
2080 //
2081 brushPosition.add(
2082 Vector3(
2083 STEP_SIZE,
2084 0.0f,
2085 0.0f
2086 )
2087 );
2088 }
2089 }
2090}
2091
2092
2094 bool flipZ,
2095 float width,
2096 float depth,
2097 vector<float>& terrainHeightVector,
2098 unordered_map<int, float>& waterPositionMapsHeight,
2099 unordered_map<int, unordered_map<int, unordered_set<int>>>& waterPositionMaps,
2100 vector<unordered_map<int, vector<Transformations>>>& foliageMaps
2101) {
2102 auto terrainHeightVectorVerticesPerX = static_cast<int>(Math::ceil(width / STEP_SIZE));
2103 auto terreinHeightVectorVerticesPerZ = static_cast<int>(Math::ceil(depth / STEP_SIZE));
2104
2105 // terrain
2106 vector<float> terrainHeightVectorMirrored;
2107 terrainHeightVectorMirrored.resize(terrainHeightVectorVerticesPerX * 2 * terreinHeightVectorVerticesPerZ);
2108 for (auto z = 0; z < terreinHeightVectorVerticesPerZ; z++) {
2109 for (auto x = 0; x < terrainHeightVectorVerticesPerX; x++) {
2110 auto _z = flipZ == true?terreinHeightVectorVerticesPerZ - z - 1:z;
2111 terrainHeightVectorMirrored[z * terrainHeightVectorVerticesPerX * 2 + x] = terrainHeightVector[z * terrainHeightVectorVerticesPerX + x];
2112 terrainHeightVectorMirrored[_z * terrainHeightVectorVerticesPerX * 2 + (terrainHeightVectorVerticesPerX * 2 - x - 1)] = terrainHeightVector[z * terrainHeightVectorVerticesPerX + x];
2113 }
2114 }
2115 terrainHeightVector = terrainHeightVectorMirrored;
2116
2117 // water
2118 unordered_map<int, unordered_map<int, unordered_set<int>>> waterPositionMapsMirrored;
2119 unordered_map<int, float> waterPositionMapsHeightMirrored;
2120 auto idxMax = 0;
2121 for (auto& waterPositionMapsIt: waterPositionMaps) {
2122 auto idx = waterPositionMapsIt.first;
2123 if (idx > idxMax) idxMax = idx;
2124 }
2125 idxMax++;
2126 for (auto& waterPositionMapsIt: waterPositionMaps) {
2127 auto idx = waterPositionMapsIt.first;
2128 waterPositionMapsHeightMirrored[idx] = waterPositionMapsHeight[idx];
2129 waterPositionMapsHeightMirrored[idxMax + idx] = waterPositionMapsHeight[idx];
2130 for (auto& zIt: waterPositionMapsIt.second) {
2131 auto z = zIt.first;
2132 auto _z = flipZ == true?terreinHeightVectorVerticesPerZ - z - 1:z;
2133 for (auto& x: zIt.second) {
2134 waterPositionMapsMirrored[idx][z].insert(x);
2135 waterPositionMapsMirrored[idxMax + idx][_z].insert(terrainHeightVectorVerticesPerX * 2 - x - 1);
2136 }
2137 }
2138 }
2139 waterPositionMapsHeight = waterPositionMapsHeightMirrored;
2140 waterPositionMaps = waterPositionMapsMirrored;
2141
2142 // foliage
2143 auto partitionsX = static_cast<int>(Math::ceil(width * 2.0f / PARTITION_SIZE));
2144 auto partitionsZ = static_cast<int>(Math::ceil(depth / PARTITION_SIZE));
2145 vector<unordered_map<int, vector<Transformations>>> foliageMapsMirrored;
2146 createFoliageMaps(width * 2.0f, depth, foliageMapsMirrored);
2147 for (auto& foliageMapPartition: foliageMaps) {
2148 for (auto& foliageMapPartitionIt: foliageMapPartition) {
2149 auto foliagePrototypeId = foliageMapPartitionIt.first;
2150 for (auto& transformations: foliageMapPartitionIt.second) {
2151 {
2152 //
2153 auto partitionX = static_cast<int>((transformations.getTranslation().getX()) / PARTITION_SIZE);
2154 auto partitionZ = static_cast<int>((transformations.getTranslation().getZ()) / PARTITION_SIZE);
2155 auto partitionIdx = partitionZ * partitionsX + partitionX;
2156 foliageMapsMirrored[partitionIdx][foliagePrototypeId].push_back(transformations);
2157 }
2158 {
2159 auto transformationsMirrored = transformations;
2160 transformationsMirrored.setTranslation(
2161 Vector3(
2162 width * 2.0f - transformationsMirrored.getTranslation().getX(),
2163 transformationsMirrored.getTranslation().getY(),
2164 flipZ == true?depth - transformationsMirrored.getTranslation().getZ():transformationsMirrored.getTranslation().getZ()
2165 )
2166 );
2167 transformationsMirrored.addRotation(transformationsMirrored.getRotationAxis(0), -transformationsMirrored.getRotationAngle(0));
2168 transformationsMirrored.update();
2169 auto eulerAngles = transformationsMirrored.getTransformationsMatrix().computeEulerAngles();
2170 transformationsMirrored.removeRotation(3);
2171 transformationsMirrored.setRotationAngle(0, eulerAngles.getZ());
2172 transformationsMirrored.setRotationAngle(1, eulerAngles.getY());
2173 transformationsMirrored.setRotationAngle(2, eulerAngles.getX());
2174 transformationsMirrored.update();
2175 //
2176 auto partitionX = static_cast<int>((transformationsMirrored.getTranslation().getX()) / PARTITION_SIZE);
2177 auto partitionZ = static_cast<int>((transformationsMirrored.getTranslation().getZ()) / PARTITION_SIZE);
2178 if (partitionZ >= partitionsZ) partitionZ = partitionsZ - 1; // special case if translation = x, y, 0.0
2179 if (partitionX >= partitionsX) partitionX = partitionsX - 1; // special case if translation = 0.0, y, z
2180 auto partitionIdx = partitionZ * partitionsX + partitionX;
2181 foliageMapsMirrored[partitionIdx][foliagePrototypeId].push_back(transformationsMirrored);
2182 }
2183 }
2184 }
2185 }
2186 foliageMaps = foliageMapsMirrored;
2187}
2188
2190 bool flipX,
2191 float width,
2192 float depth,
2193 vector<float>& terrainHeightVector,
2194 unordered_map<int, float>& waterPositionMapsHeight,
2195 unordered_map<int, unordered_map<int, unordered_set<int>>>& waterPositionMaps,
2196 vector<unordered_map<int, vector<Transformations>>>& foliageMaps
2197) {
2198 auto terrainHeightVectorVerticesPerX = static_cast<int>(Math::ceil(width / STEP_SIZE));
2199 auto terreinHeightVectorVerticesPerZ = static_cast<int>(Math::ceil(depth / STEP_SIZE));
2200
2201 // terrain
2202 vector<float> terrainHeightVectorMirrored;
2203 terrainHeightVectorMirrored.resize(terrainHeightVectorVerticesPerX * terreinHeightVectorVerticesPerZ * 2);
2204 for (auto z = 0; z < terreinHeightVectorVerticesPerZ; z++) {
2205 for (auto x = 0; x < terrainHeightVectorVerticesPerX; x++) {
2206 auto _x = flipX == true?terrainHeightVectorVerticesPerX - x - 1:x;
2207 terrainHeightVectorMirrored[z * terrainHeightVectorVerticesPerX + x] = terrainHeightVector[z * terrainHeightVectorVerticesPerX + x];
2208 terrainHeightVectorMirrored[(terreinHeightVectorVerticesPerZ * 2 - z - 1) * terrainHeightVectorVerticesPerX + _x] = terrainHeightVector[z * terrainHeightVectorVerticesPerX + x];
2209 }
2210 }
2211 terrainHeightVector = terrainHeightVectorMirrored;
2212
2213 // water
2214 unordered_map<int, unordered_map<int, unordered_set<int>>> waterPositionMapsMirrored;
2215 unordered_map<int, float> waterPositionMapsHeightMirrored;
2216 auto idxMax = 0;
2217 for (auto& waterPositionMapsIt: waterPositionMaps) {
2218 auto idx = waterPositionMapsIt.first;
2219 if (idx > idxMax) idxMax = idx;
2220 }
2221 idxMax++;
2222 for (auto& waterPositionMapsIt: waterPositionMaps) {
2223 auto idx = waterPositionMapsIt.first;
2224 waterPositionMapsHeightMirrored[idx] = waterPositionMapsHeight[idx];
2225 waterPositionMapsHeightMirrored[idxMax + idx] = waterPositionMapsHeight[idx];
2226 for (auto& zIt: waterPositionMapsIt.second) {
2227 auto z = zIt.first;
2228 for (auto& x: zIt.second) {
2229 auto _x = flipX == true?terrainHeightVectorVerticesPerX - x - 1:x;
2230 waterPositionMapsMirrored[idx][z].insert(x);
2231 waterPositionMapsMirrored[idxMax + idx][terreinHeightVectorVerticesPerZ * 2 - z - 1].insert(_x);
2232 }
2233 }
2234 }
2235 waterPositionMapsHeight = waterPositionMapsHeightMirrored;
2236 waterPositionMaps = waterPositionMapsMirrored;
2237
2238 // foliage
2239 auto partitionsX = static_cast<int>(Math::ceil(width / PARTITION_SIZE));
2240 auto partitionsZ = static_cast<int>(Math::ceil(depth * 2.0f / PARTITION_SIZE));
2241 vector<unordered_map<int, vector<Transformations>>> foliageMapsMirrored;
2242 createFoliageMaps(width, depth * 2.0f, foliageMapsMirrored);
2243 for (auto& foliageMapPartition: foliageMaps) {
2244 for (auto& foliageMapPartitionIt: foliageMapPartition) {
2245 auto foliagePrototypeId = foliageMapPartitionIt.first;
2246 for (auto& transformations: foliageMapPartitionIt.second) {
2247 {
2248 //
2249 auto partitionX = static_cast<int>((transformations.getTranslation().getX()) / PARTITION_SIZE);
2250 auto partitionZ = static_cast<int>((transformations.getTranslation().getZ()) / PARTITION_SIZE);
2251 auto partitionIdx = partitionZ * partitionsX + partitionX;
2252 foliageMapsMirrored[partitionIdx][foliagePrototypeId].push_back(transformations);
2253 }
2254 {
2255 auto transformationsMirrored = transformations;
2256 transformationsMirrored.setTranslation(
2257 Vector3(
2258 flipX == true?width - transformationsMirrored.getTranslation().getX():transformationsMirrored.getTranslation().getX(),
2259 transformationsMirrored.getTranslation().getY(),
2260 depth * 2.0f - transformationsMirrored.getTranslation().getZ()
2261 )
2262 );
2263 transformationsMirrored.addRotation(transformationsMirrored.getRotationAxis(2), -transformationsMirrored.getRotationAngle(2));
2264 transformationsMirrored.update();
2265 auto eulerAngles = transformationsMirrored.getTransformationsMatrix().computeEulerAngles();
2266 transformationsMirrored.removeRotation(3);
2267 transformationsMirrored.setRotationAngle(0, eulerAngles.getZ());
2268 transformationsMirrored.setRotationAngle(1, eulerAngles.getY());
2269 transformationsMirrored.setRotationAngle(2, eulerAngles.getX());
2270 transformationsMirrored.update();
2271 //
2272 auto partitionX = static_cast<int>((transformationsMirrored.getTranslation().getX()) / PARTITION_SIZE);
2273 auto partitionZ = static_cast<int>((transformationsMirrored.getTranslation().getZ()) / PARTITION_SIZE);
2274 if (partitionX >= partitionsX) partitionX = partitionsX - 1; // special case if translation = 0.0, y, z
2275 if (partitionZ >= partitionsZ) partitionZ = partitionsZ - 1; // special case if translation = x, y, 0.0
2276 auto partitionIdx = partitionZ * partitionsX + partitionX;
2277 foliageMapsMirrored[partitionIdx][foliagePrototypeId].push_back(transformationsMirrored);
2278 }
2279 }
2280 }
2281 }
2282 foliageMaps = foliageMapsMirrored;
2283}
Rotation representation.
Definition: Rotation.h:18
Transformations which contain scale, rotations and translation.
const Matrix4x4 & getTransformationsMatrix() const
void setTranslation(const Vector3 &translation)
Set translation.
void setScale(const Vector3 &scale)
Set scale.
virtual void update()
Computes transformation matrix.
void addRotation(const Vector3 &axis, const float angle)
Add rotation.
Color 4 definition.
Definition: Color4.h:20
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
Definition: Face.h:19
Node faces entity A node can have multiple entities containing faces and a applied material.
Definition: FacesEntity.h:28
void setMaterial(Material *material)
Set up the entity's material.
Definition: FacesEntity.h:72
void setLOD1Distance(float lod1Distance)
Set LOD1 distance.
Definition: FacesEntity.h:121
void setLOD2Indices(const vector< int32_t > &lod2Indices)
Set LOD2 indices.
Definition: FacesEntity.cpp:80
void setLOD2Distance(float lod2Distance)
Set LOD2 distance.
Definition: FacesEntity.h:149
void setLOD3Distance(float lod3Distance)
Set LOD3 distance.
Definition: FacesEntity.h:177
void setLOD1Indices(const vector< int32_t > &lod1Indices)
Set LOD1 indices.
Definition: FacesEntity.cpp:72
void setLOD3Indices(const vector< int32_t > &lod3Indices)
Set LOD3 indices.
Definition: FacesEntity.cpp:88
void setFaces(const vector< Face > &faces)
Set up entity's faces.
Definition: FacesEntity.cpp:47
Represents a material.
Definition: Material.h:21
Representation of a 3d model.
Definition: Model.h:32
Model node.
Definition: Node.h:31
Represents rotation orders of a model.
Definition: RotationOrder.h:23
Represents specular material properties.
Model up vector.
Definition: UpVector.h:20
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Definition: BoundingBox.h:25
void extend(BoundingBox *boundingBox)
Extend bounding box with given bounding box.
Definition: BoundingBox.h:148
const Vector3 & getDimensions() const
Definition: BoundingBox.h:127
Line segment helper functions.
Definition: LineSegment.h:17
Standard math functions.
Definition: Math.h:21
3x3 2D Matrix class
Definition: Matrix2D3x3.h:22
Matrix2D3x3 & translate(const Vector2 &v)
Sets up a translation matrix.
Definition: Matrix2D3x3.h:164
Matrix2D3x3 & identity()
Setup identity matrix.
Definition: Matrix2D3x3.h:116
Matrix2D3x3 & multiply(const Matrix2D3x3 &m)
Multiplies this matrix with another matrix.
Definition: Matrix2D3x3.h:220
Matrix2D3x3 clone() const
Clones this matrix.
Definition: Matrix2D3x3.h:377
Vector3 computeEulerAngles() const
Compute Euler angles (rotation around x, y, z axes)
Definition: Matrix4x4.h:661
2D vector 2 class
Definition: Vector2.h:19
float getY() const
Definition: Vector2.h:111
float getX() const
Definition: Vector2.h:94
3D vector 3 class
Definition: Vector3.h:22
float getY() const
Definition: Vector3.h:119
float getX() const
Definition: Vector3.h:103
float getZ() const
Definition: Vector3.h:136
Vector3 & normalize()
Normalize the vector.
Definition: Vector3.h:288
Vector3 & set(float x, float y, float z)
Set up vector.
Definition: Vector3.h:73
Vector3 clone() const
Clones the vector.
Definition: Vector3.h:372
Vector3 & add(const Vector3 &v)
Adds a vector.
Definition: Vector3.h:301
Vector3 & setY(float y)
Set Y.
Definition: Vector3.h:128
Console class.
Definition: Console.h:26
static void println()
Print new line to console.
Definition: Console.cpp:78
static constexpr float MAX_VALUE
Definition: Float.h:25
static constexpr float MIN_VALUE
Definition: Float.h:26
Integer class.
Definition: Integer.h:26
static constexpr int MIN_VALUE
Definition: Integer.h:29
static constexpr int MAX_VALUE
Definition: Integer.h:28
Model tools functions class.
Definition: ModelTools.h:38
static void createDefaultAnimation(Model *model, int32_t frames)
Create default animation.
Definition: ModelTools.cpp:262
static void prepareForIndexedRendering(Model *model)
Prepare for indexed rendering.
Definition: ModelTools.cpp:84
static Vector3 computeNormal(const array< Vector3, 3 > &vertices)
Computes face normal for given face vertices.
Definition: ModelTools.h:55
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 hasWaterPosition(const unordered_map< int, unordered_set< int > > &waterPositionSet, int x, int z)
Definition: Terrain2.h:122
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
std::exception Exception
Exception base class.
Definition: Exception.h:19