7#include <unordered_map>
8#include <unordered_set>
41using std::unordered_map;
42using std::unordered_set;
70void Terrain2::createTerrainModels(
float width,
float depth,
float y, vector<float>& terrainHeightVector,
BoundingBox& terrainBoundingBox, vector<Model*>& terrainModels,
bool createLODLevels)
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);
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) {
100 auto partitionIdx = partitionZ * partitionsX + partitionX;
102 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
103 auto& terrainNormals = partitionTerrainNormals[partitionIdx];
104 auto& terrainFaces = partitionTerrainFaces[partitionIdx];
105 auto& terrainTriangles = partitionTerrainTriangles[partitionIdx];
107 int normalIdx = terrainNormals.size();
108 int vertexIdx = terrainVertices.size();
110 auto terrainHeightVectorX =
static_cast<int>(x /
STEP_SIZE);
111 auto terrainHeightVectorZ =
static_cast<int>(z /
STEP_SIZE);
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);
123 terrainVertices.push_back(topVertex);
124 terrainVertices.push_back(topLeftVertex);
125 terrainVertices.push_back(leftVertex);
126 terrainVertices.push_back(vertex);
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);
133 terrainNormals.push_back(topNormal);
134 terrainNormals.push_back(topLeftNormal);
135 terrainNormals.push_back(leftNormal);
136 terrainNormals.push_back(normal);
138 terrainFaces.push_back(
148 terrainFaces.push_back(
158 terrainTriangles[terrainHeightVectorZ]+= 2;
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");
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]) {
195 if (createLODLevels ==
true) {
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;
206 auto finishedX =
false;
207 for (
auto x = 0; finishedX ==
false && x < trianglesPerX; x+= 8) {
208 if (x > trianglesPerX - 8) {
209 x = trianglesPerX - 8;
212 lod1Indices.push_back(facesIndices[z * trianglesPerX + x + 6][0]);
213 lod1Indices.push_back(facesIndices[z * trianglesPerX + x][1]);
214 lod1Indices.push_back(facesIndices[(z + 3) * trianglesPerX + x][2]);
215 lod1Indices.push_back(facesIndices[(z + 3) * trianglesPerX + x][2]);
216 lod1Indices.push_back(facesIndices[(z + 3) * trianglesPerX + x + 7][1]);
217 lod1Indices.push_back(facesIndices[z * trianglesPerX + x + 6][0]);
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;
234 auto finishedX =
false;
235 for (
auto x = 0; finishedX ==
false && x < trianglesPerX; x+= 16) {
236 if (x > trianglesPerX - 16) {
237 x = trianglesPerX - 16;
240 lod2Indices.push_back(facesIndices[z * trianglesPerX + x + 14][0]);
241 lod2Indices.push_back(facesIndices[z * trianglesPerX + x][1]);
242 lod2Indices.push_back(facesIndices[(z + 7) * trianglesPerX + x][2]);
243 lod2Indices.push_back(facesIndices[(z + 7) * trianglesPerX + x][2]);
244 lod2Indices.push_back(facesIndices[(z + 7) * trianglesPerX + x + 15][1]);
245 lod2Indices.push_back(facesIndices[z * trianglesPerX + x + 14][0]);
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;
262 auto finishedX =
false;
263 for (
auto x = 0; finishedX ==
false && x < trianglesPerX; x+= 32) {
264 if (x > trianglesPerX - 32) {
265 x = trianglesPerX - 32;
268 lod3Indices.push_back(facesIndices[z * trianglesPerX + x + 30][0]);
269 lod3Indices.push_back(facesIndices[z * trianglesPerX + x][1]);
270 lod3Indices.push_back(facesIndices[(z + 15) * trianglesPerX + x][2]);
271 lod3Indices.push_back(facesIndices[(z + 15) * trianglesPerX + x][2]);
272 lod3Indices.push_back(facesIndices[(z + 15) * trianglesPerX + x + 31][1]);
273 lod3Indices.push_back(facesIndices[z * trianglesPerX + x + 30][0]);
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();
292 terrainBoundingBox.
extend(terrainModel->getBoundingBox());
295 terrainModels.push_back(terrainModel);
303 auto haveVertex =
getTerrainVertex(terrainHeightVector, verticesPerX, verticesPerZ, x, z - 1, vertex);
304 if (haveVertex ==
false)
return Vector3(0.0f, 1.0f, 0.0f);
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);
322 if (haveTopVertex ==
true && haveTopLeftVertex ==
true) {
330 vertexNormal.
add(triangleNormal);
333 if (haveTopLeftVertex ==
true && haveLeftVertex ==
true) {
341 vertexNormal.
add(triangleNormal);
344 if (haveLeftVertex ==
true && haveBottomVertex ==
true) {
352 vertexNormal.
add(triangleNormal);
355 if (haveBottomVertex ==
true && haveBottomRightVertex ==
true) {
363 vertexNormal.
add(triangleNormal);
366 if (haveBottomRightVertex ==
true && haveRightVertex ==
true) {
374 vertexNormal.
add(triangleNormal);
377 if (haveRightVertex ==
true && haveTopVertex ==
true) {
385 vertexNormal.
add(triangleNormal);
388 if (normalCount > 0) {
391 Console::println(
"Terrain2::computeTerrainVertexNormal(): no vertex normal available: normal count == 0");
392 return vertexNormal.
set(0.0f, 1.0f, 0.0f);
397 vector<Model*>& terrainModels,
398 vector<float>& terrainHeightVector,
399 const Vector3& brushCenterPosition,
407 if (brushTexture ==
nullptr)
return;
409 if (terrainModels.empty() ==
true)
return;
412 vector<vector<Vector3>> partitionTerrainVertices;
413 vector<vector<Vector3>> partitionTerrainNormals;
414 partitionTerrainVertices.resize(terrainModels.size());
415 partitionTerrainNormals.resize(terrainModels.size());
427 auto textureBytePerPixel = brushTexture->
getDepth() == 32?4:3;
428 for (
auto z = 0.0f; z < textureHeight * brushScale; z+=
STEP_SIZE) {
434 (
static_cast<float>(textureWidth) * brushScale) / 2.0f,
436 (
static_cast<float>(textureHeight) * brushScale) / 2.0f
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) {
467 auto vertexIdx = terrainHeightVectorZ * terrainHeightVectorVerticesPerX + terrainHeightVectorX;
468 auto terrainVertexHeight = terrainHeightVector[vertexIdx];
469 switch(brushOperation) {
471 terrainVertexHeight+= appliedStrength;
474 terrainVertexHeight-= appliedStrength;
477 terrainVertexHeight = terrainVertexHeight * (1.0f - Math::clamp(appliedStrength, 0.0f, 1.0f)) + brushHeight * Math::clamp(appliedStrength, 0.0f, 1.0f);
480 terrainVertexHeight = terrainVertexHeight * (1.0f - Math::clamp(appliedStrength, 0.0f, 1.0f)) + 0.0f * Math::clamp(appliedStrength, 0.0f, 1.0f);
483 auto terrainVertexHeightNeighbours = 0.0f;
484 auto terrainVertexHeightNeighbourCount = 0;
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];
497 if (haveLeftVertex ==
true) {
498 terrainVertexHeightNeighbourCount++;
499 terrainVertexHeightNeighbours+= leftVertex[1];
501 if (haveBottomVertex ==
true) {
502 terrainVertexHeightNeighbourCount++;
503 terrainVertexHeightNeighbours+= bottomVertex[1];
505 if (haveRightVertex ==
true) {
506 terrainVertexHeightNeighbourCount++;
507 terrainVertexHeightNeighbours+= rightVertex[1];
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);
515 terrainHeightVector[vertexIdx] = terrainVertexHeight;
519 auto _brushPosition = brushPosition;
522 auto partitionIdx = partitionZ * partitionsX + partitionX;
523 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:
nullptr;
524 auto terrainNode = terrainModel !=
nullptr?terrainModel->getNodeById(
"terrain"):
nullptr;
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;
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();
539 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
540 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
541 terrainVertices[vertexIdx + 3][1] = terrainVertexHeight;
547 auto _brushPosition = brushPosition.clone().sub(
Vector3(0.0f, 0.0f, -
STEP_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;
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;
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();
567 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
568 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
569 terrainVertices[vertexIdx + 0][1] = terrainVertexHeight;
578 auto partitionIdx = partitionZ * partitionsX + partitionX;
579 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:
nullptr;
580 auto terrainNode = terrainModel !=
nullptr?terrainModel->getNodeById(
"terrain"):
nullptr;
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;
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();
595 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
596 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
597 terrainVertices[vertexIdx + 1][1] = terrainVertexHeight;
603 auto _brushPosition = brushPosition.clone().sub(
Vector3(-
STEP_SIZE, 0.0f, 0.0f));
606 auto partitionIdx = partitionZ * partitionsX + partitionX;
607 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:
nullptr;
608 auto terrainNode = terrainModel !=
nullptr?terrainModel->getNodeById(
"terrain"):
nullptr;
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;
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();
623 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
624 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
625 terrainVertices[vertexIdx + 2][1] = terrainVertexHeight;
647 (
static_cast<float>(textureWidth) * brushScale) / 2.0f,
649 (
static_cast<float>(textureHeight) * brushScale) / 2.0f
662 auto partitionIdx = partitionZ * partitionsX + partitionX;
663 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:
nullptr;
664 auto terrainNode = terrainModel !=
nullptr?terrainModel->getNodeById(
"terrain"):
nullptr;
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;
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();
681 auto& terrainNormals = partitionTerrainNormals[partitionIdx];
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);
689 terrainNormals[normalIdx + 0] = topNormal;
690 terrainNormals[normalIdx + 1] = topLeftNormal;
691 terrainNormals[normalIdx + 2] = leftNormal;
692 terrainNormals[normalIdx + 3] = normal;
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();
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();
738 vector<Model*>& terrainModels,
739 vector<float>& terrainHeightVector,
740 const Vector3& brushCenterPosition,
748 if (brushTexture ==
nullptr)
return;
750 if (terrainModels.empty() ==
true)
return;
753 vector<vector<Vector3>> partitionTerrainVertices;
754 vector<vector<Vector3>> partitionTerrainNormals;
755 partitionTerrainVertices.resize(terrainModels.size());
756 partitionTerrainNormals.resize(terrainModels.size());
765 auto textureBytePerPixel = brushTexture->
getDepth() == 32?4:3;
770 brushTextureMatrix.
translate(
Vector2(
static_cast<float>(textureWidth) / 2.0f,
static_cast<float>(textureHeight) / 2.0f));
773 auto brushScaleMax = Math::max(brushScale.
getX(), brushScale.
getY());
776 for (
auto z = -textureHeight * brushScaleMax * 2.0f; z < textureHeight * brushScaleMax * 2.0f; z+=
STEP_SIZE) {
782 static_cast<float>(textureWidth) * brushScaleMax * 2.0f,
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) {
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) {
828 auto terrainVertexHeight = 0.0f;
830 auto vertexIdx = terrainHeightVectorZ * terrainHeightVectorVerticesPerX + terrainHeightVectorX;
831 terrainVertexHeight = terrainHeightVector[vertexIdx];
832 terrainVertexHeight = height > terrainVertexHeight?height:terrainVertexHeight;
833 terrainHeightVector[vertexIdx] = terrainVertexHeight;
838 auto _brushPosition = brushPosition;
841 auto partitionIdx = partitionZ * partitionsX + partitionX;
842 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:
nullptr;
843 auto terrainNode = terrainModel !=
nullptr?terrainModel->getNodeById(
"terrain"):
nullptr;
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;
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();
858 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
859 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
860 terrainVertices[vertexIdx + 3][1] = terrainVertexHeight;
869 auto partitionIdx = partitionZ * partitionsX + partitionX;
870 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:
nullptr;
871 auto terrainNode = terrainModel !=
nullptr?terrainModel->getNodeById(
"terrain"):
nullptr;
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;
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();
886 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
887 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
888 terrainVertices[vertexIdx + 0][1] = terrainVertexHeight;
897 auto partitionIdx = partitionZ * partitionsX + partitionX;
898 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:
nullptr;
899 auto terrainNode = terrainModel !=
nullptr?terrainModel->getNodeById(
"terrain"):
nullptr;
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;
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();
914 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
915 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
916 terrainVertices[vertexIdx + 1][1] = terrainVertexHeight;
925 auto partitionIdx = partitionZ * partitionsX + partitionX;
926 auto terrainModel = partitionIdx < terrainModels.size()?terrainModels[partitionIdx]:
nullptr;
927 auto terrainNode = terrainModel !=
nullptr?terrainModel->getNodeById(
"terrain"):
nullptr;
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;
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();
942 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
943 auto vertexIdx = (terrainModelZ * terrainModelVerticesPerX * 4) + (terrainModelX * 4);
944 terrainVertices[vertexIdx + 2][1] = terrainVertexHeight;
960 for (
auto z = -textureHeight * brushScaleMax * 2.0f; z < textureHeight * brushScaleMax * 2.0f; z+=
STEP_SIZE) {
966 static_cast<float>(textureWidth) * brushScaleMax * 2.0f,
978 for (
auto x = -textureWidth * brushScaleMax * 2.0f; x < textureWidth * brushScaleMax * 2.0f; x+=
STEP_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;
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;
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();
1000 auto& terrainNormals = partitionTerrainNormals[partitionIdx];
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);
1008 terrainNormals[normalIdx + 0] = topNormal;
1009 terrainNormals[normalIdx + 1] = topLeftNormal;
1010 terrainNormals[normalIdx + 2] = leftNormal;
1011 terrainNormals[normalIdx + 3] = normal;
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();
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();
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);
1064 waterPositionMap[terrainHeightVectorZCenter].insert(terrainHeightVectorXCenter);
1067 Console::println(
"Terrain2::determineWaterPositionSet: " + to_string(terrainHeightVectorXCenter) +
" / " + to_string(terrainHeightVectorZCenter) +
" @ " + to_string(waterHeight));
1070 determineWaterXPositionSet(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, terrainHeightVectorXCenter, terrainHeightVectorZCenter, waterHeight, waterPositionMap[terrainHeightVectorZCenter]);
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]);
1084 if (waterPositionMap[terrainHeightVectorZ].empty() ==
true)
break;
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]);
1102 if (waterPositionMap[terrainHeightVectorZ].empty() ==
true)
break;
1109 auto waterPositionMapCopy = waterPositionMap;
1110 auto zMin = waterPositionMap.begin()->first;
1111 auto zMax = waterPositionMap.begin()->first;
1113 waterPositionMapCopy = waterPositionMap;
1114 for (
auto& waterPositionMapIt: waterPositionMap) {
1115 auto z = waterPositionMapIt.first;
1116 for (
auto x: waterPositionMapIt.second) {
1118 determineWater(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, x, z - 1, waterHeight) ==
true) {
1121 terrainHeightVector,
1132 determineWater(terrainHeightVector, terrainHeightVectorVerticesPerX, terreinHeightVectorVerticesPerZ, x, z + 1, waterHeight) ==
true) {
1135 terrainHeightVector,
1147 }
while (waterPositionMapCopy.size() != waterPositionMap.size());
1148 waterPositionMap[zMax] = waterPositionMap[zMax - 1];
1152 auto haveWaterPositionSet = waterPositionMap.empty() ==
false;
1153 Console::println(
"Terrain2::determineWaterPositionSet: Have water position set: " + to_string(haveWaterPositionSet));
1154 return haveWaterPositionSet;
1163 for (
auto& mIt: waterPositionMap) {
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;
1174 (
static_cast<float>(xMin + xMax) / 2.0f) *
STEP_SIZE,
1176 (
static_cast<float>(zMin + zMax) / 2.0f) *
STEP_SIZE
1182 const unordered_map<
int, unordered_set<int>>& waterPositionMap,
1185 vector<Model*>& waterModels
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);
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);
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;
1222 auto partitionIdx = partitionZ * partitionsX + partitionX;
1224 auto& terrainVertices = partitionTerrainVertices[partitionIdx];
1225 auto& terrainNormals = partitionTerrainNormals[partitionIdx];
1226 auto& terrainFaces = partitionWaterFaces[partitionIdx];
1228 int normalIdx = terrainNormals.size();
1229 int vertexIdx = terrainVertices.size();
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);
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);
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);
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);
1251 if (hasTopLeft ==
false ||
1254 hasOrigin ==
false) {
1255 terrainFaces.push_back(
1266 terrainFaces.push_back(
1276 terrainFaces.push_back(
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);
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");
1304 vector<FacesEntity> nodeFacesEntities;
1305 vector<Face> nodeFaces;
1306 for (
auto faceIndices: partitionWaterFaces[partitionIdx]) {
1307 nodeFaces.push_back(
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);
1334 vector<Model*>& terrainModels,
1335 vector<float>& terrainHeightVector,
1336 const Vector3& brushCenterPosition,
1340 if (terrainModels.empty() ==
true)
return false;
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];
1357 vector<unordered_map<
int, vector<Transformations>>>& foliageMaps
1368 vector<unordered_map<
int, vector<Transformations>>>& foliageMaps
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();
1379 vector<unordered_map<
int, vector<Transformations>>>& foliageMaps
1382 for (
auto& foliageMap: foliageMaps) foliageMap.clear();
1387 vector<float>& terrainHeightVector,
1388 const Vector3& brushCenterPosition,
1390 const vector<FoliageBrushPrototype>& foliageBrushPrototypes,
1392 vector<unordered_map<
int, vector<Transformations>>>& foliageMaps,
1393 vector<unordered_map<
int, vector<Transformations>>>& newFoliageMaps
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;
1428 brushMapCountMapTemplate.push_back(brushMapCountMapEntity);
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;
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);
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;
1470 for (
auto z = 0.0f; z < textureHeight * foliageBrush.
brushScale; z+= 1.0f) {
1471 auto brushPosition =
1472 brushCenterPosition.
1476 (
static_cast<float>(textureWidth) * foliageBrush.
brushScale) / 2.0f,
1478 ((
static_cast<float>(textureHeight) * foliageBrush.
brushScale) / 2.0f)
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) {
1509 switch(brushOperation) {
1511 for (
auto& brushMapCountMapEntityIt: brushMapCountMapEntity) {
1512 auto prototypeId = brushMapCountMapEntityIt.first;
1513 if (prototypeId == -1)
continue;
1514 auto prototypeCount = brushMapCountMapEntityIt.second;
1516 auto prototypeIdx = -1;
1517 for (
auto i = 0; i < foliageBrushPrototypes.size(); i++) {
1518 if (foliageBrushPrototypes[i].prototypeId == prototypeId) prototypeIdx = i;
1520 if (prototypeIdx == -1)
continue;
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());
1528 Math::floor(brushPosition.getX()) + Math::random(),
1530 Math::floor(brushPosition.getZ()) + Math::random()
1536 auto partitionIdx = partitionZ * partitionsX + partitionX;
1539 auto haveContact =
false;
1543 for (
int _z = -1; _z < 2; _z++)
1544 for (
int _x = -1; _x < 2; _x++) {
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);
1555 if (LineSegment::doesLineSegmentCollideWithTriangle(topVertex, topLeftVertex, leftVertex, translation.
clone().
setY(-10000.0f), translation.
clone().
setY(+10000.0f), contact) ==
true) {
1558 height = (topVertex.
getY() + topLeftVertex.
getY() + leftVertex.
getY()) / 3.0f;
1561 if (LineSegment::doesLineSegmentCollideWithTriangle(leftVertex, vertex, topVertex, translation.
clone().
setY(-10000.0f), translation.
clone().
setY(+10000.0f), contact) ==
true) {
1564 height = (leftVertex.
getY() + vertex.
getY() + topVertex.
getY()) / 3.0f;
1570 if (height < foliageBrushPrototypes[prototypeIdx].heightMin || height > foliageBrushPrototypes[prototypeIdx].heightMax)
continue;
1573 if (haveContact ==
false) {
1575 "Terrain2::applyFoliageBrush(): no contact@" +
1576 to_string(translation.
getX()) +
", " +
1577 to_string(translation.
getZ())
1579 contact = translation;
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;
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));
1597 _transformations.
addRotation(Rotation::Z_AXIS, zAxisRotation);
1598 _transformations.
addRotation(Rotation::X_AXIS, xAxisRotation);
1599 _transformations.
addRotation(Rotation::Y_AXIS, yAxisRotation);
1600 _transformations.
update();
1602 zAxisRotation = euler.
getZ();
1603 yAxisRotation = euler.getY();
1604 xAxisRotation = euler.getX();
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));
1612 transformations.
update();
1615 foliageMaps[partitionIdx][prototypeId].push_back(transformations);
1616 newFoliageMaps[partitionIdx][prototypeId].push_back(transformations);
1636 const Vector3& brushCenterPosition,
1638 const vector<FoliageBrushPrototype>& foliageBrushPrototypes,
1640 vector<unordered_map<
int, vector<Transformations>>>& foliageMaps,
1641 unordered_set<int>& recreateFoliagePartitions
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);
1667 for (
auto z = 0.0f; z < textureHeight * foliageBrush.
brushScale; z+= 1.0f) {
1668 auto brushPosition =
1669 brushCenterPosition.
1673 (
static_cast<float>(textureWidth) * foliageBrush.
brushScale) / 2.0f,
1675 ((
static_cast<float>(textureHeight) * foliageBrush.
brushScale) / 2.0f)
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);
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);
1699 if (terrainHeightVectorX < 0 || terrainHeightVectorX >= terrainHeightVectorVerticesPerX ||
1700 terrainHeightVectorZ < 0 || terrainHeightVectorZ >= terreinHeightVectorVerticesPerZ) {
1715 auto partitionIdx = partitionZ * partitionsX + partitionX;
1718 switch(brushOperation) {
1726 getTerrainVertex(terrainHeightVectorX, terrainHeightVectorZ - 1, topVertex);
1727 getTerrainVertex(terrainHeightVectorX - 1, terrainHeightVectorZ - 1, topLeftVertex);
1728 getTerrainVertex(terrainHeightVectorX - 1, terrainHeightVectorZ, leftVertex);
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))) {
1744 foliageMapPartitionPrototypeTransformations.erase(foliageMapPartitionPrototypeTransformations.begin() + i);
1745 recreateFoliagePartitions.insert(partitionIdx);
1768 vector<float>& terrainHeightVector,
1769 const Vector3& brushCenterPosition,
1771 vector<unordered_map<
int, vector<Transformations>>>& foliageMaps,
1772 unordered_set<int>& updateFoliagePartitions
1789 for (
auto z = 0.0f; z < textureHeight * foliageBrush.
brushScale; z+= 1.0f) {
1790 auto brushPosition =
1791 brushCenterPosition.
1795 (
static_cast<float>(textureWidth) * foliageBrush.
brushScale) / 2.0f,
1797 ((
static_cast<float>(textureHeight) * foliageBrush.
brushScale) / 2.0f)
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) {
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);
1837 auto partitionIdx = partitionZ * partitionsX + partitionX;
1840 updateFoliagePartitions.insert(partitionIdx);
1849 getTerrainVertex(terrainHeightVectorX, terrainHeightVectorZ - 1, topVertex);
1850 getTerrainVertex(terrainHeightVectorX - 1, terrainHeightVectorZ - 1, topLeftVertex);
1851 getTerrainVertex(terrainHeightVectorX - 1, terrainHeightVectorZ, leftVertex);
1855 for (
auto& foliageMapPartitionIt: foliageMaps[partitionIdx]) {
1856 auto prototypeId = foliageMapPartitionIt.first;
1857 if (prototypeId == -1)
continue;
1858 auto& foliageMapPartitionPrototypeTransformations = foliageMapPartitionIt.second;
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()) {
1869 auto haveContact =
false;
1871 for (
int _z = -1; _z < 2; _z++)
1872 for (
int _x = -1; _x < 2; _x++) {
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);
1883 if (LineSegment::doesLineSegmentCollideWithTriangle(topVertex, topLeftVertex, leftVertex, transformations.getTranslation().
clone().
setY(-10000.0f), transformations.getTranslation().
clone().
setY(+10000.0f), contact) ==
true) {
1887 if (LineSegment::doesLineSegmentCollideWithTriangle(leftVertex, vertex, topVertex, transformations.getTranslation().
clone().
setY(-10000.0f), transformations.getTranslation().
clone().
setY(+10000.0f), contact) ==
true) {
1894 if (haveContact ==
false) {
1896 "Terrain2::applyFoliageBrush(): no contact@" +
1897 to_string(transformations.getTranslation().getX()) +
", " +
1898 to_string(transformations.getTranslation().getZ())
1900 contact = transformations.getTranslation();
1904 transformations.setTranslation(transformations.getTranslation().clone().setY(contact.
getY()));
1905 transformations.update();
1924 vector<float>& terrainHeightVector,
1925 const Vector3& brushCenterPosition,
1927 float brushRotation,
1929 vector<unordered_map<
int, vector<Transformations>>>& foliageMaps,
1930 unordered_set<int>& updateFoliagePartitions
1933 if (brushTexture ==
nullptr)
return;
1936 vector<vector<Vector3>> partitionTerrainVertices;
1937 vector<vector<Vector3>> partitionTerrainNormals;
1946 auto textureBytePerPixel = brushTexture->
getDepth() == 32?4:3;
1951 brushTextureMatrix.
translate(
Vector2(
static_cast<float>(textureWidth) / 2.0f,
static_cast<float>(textureHeight) / 2.0f));
1954 auto brushScaleMax = Math::max(brushScale.
getX(), brushScale.
getY());
1957 for (
auto z = -textureHeight * brushScaleMax * 2.0f; z < textureHeight * brushScaleMax * 2.0f; z+=
STEP_SIZE) {
1958 auto brushPosition =
1959 brushCenterPosition.
1963 static_cast<float>(textureWidth) * brushScaleMax * 2.0f,
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) {
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) {
2008 auto partitionIdx = partitionZ * partitionsX + partitionX;
2011 updateFoliagePartitions.insert(partitionIdx);
2020 getTerrainVertex(terrainHeightVectorX, terrainHeightVectorZ - 1, topVertex);
2021 getTerrainVertex(terrainHeightVectorX - 1, terrainHeightVectorZ - 1, topLeftVertex);
2022 getTerrainVertex(terrainHeightVectorX - 1, terrainHeightVectorZ, leftVertex);
2026 for (
auto& foliageMapPartitionIt: foliageMaps[partitionIdx]) {
2027 auto prototypeId = foliageMapPartitionIt.first;
2028 if (prototypeId == -1)
continue;
2029 auto& foliageMapPartitionPrototypeTransformations = foliageMapPartitionIt.second;
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()) {
2039 auto haveContact =
false;
2041 for (
int _z = -1; _z < 2; _z++)
2042 for (
int _x = -1; _x < 2; _x++) {
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);
2053 if (LineSegment::doesLineSegmentCollideWithTriangle(topVertex, topLeftVertex, leftVertex, transformations.getTranslation().
clone().
setY(-10000.0f), transformations.getTranslation().
clone().
setY(+10000.0f), contact) ==
true) {
2057 if (LineSegment::doesLineSegmentCollideWithTriangle(leftVertex, vertex, topVertex, transformations.getTranslation().
clone().
setY(-10000.0f), transformations.getTranslation().
clone().
setY(+10000.0f), contact) ==
true) {
2064 if (haveContact ==
false) {
2066 "Terrain2::applyFoliageBrush(): no contact@" +
2067 to_string(transformations.getTranslation().getX()) +
", " +
2068 to_string(transformations.getTranslation().getZ())
2070 contact = transformations.getTranslation();
2074 transformations.setTranslation(transformations.getTranslation().clone().setY(contact.
getY()));
2075 transformations.update();
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
2102 auto terrainHeightVectorVerticesPerX =
static_cast<int>(Math::ceil(width /
STEP_SIZE));
2103 auto terreinHeightVectorVerticesPerZ =
static_cast<int>(Math::ceil(depth /
STEP_SIZE));
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];
2115 terrainHeightVector = terrainHeightVectorMirrored;
2118 unordered_map<int, unordered_map<int, unordered_set<int>>> waterPositionMapsMirrored;
2119 unordered_map<int, float> waterPositionMapsHeightMirrored;
2121 for (
auto& waterPositionMapsIt: waterPositionMaps) {
2122 auto idx = waterPositionMapsIt.first;
2123 if (idx > idxMax) idxMax = idx;
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) {
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);
2139 waterPositionMapsHeight = waterPositionMapsHeightMirrored;
2140 waterPositionMaps = waterPositionMapsMirrored;
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;
2147 for (
auto& foliageMapPartition: foliageMaps) {
2148 for (
auto& foliageMapPartitionIt: foliageMapPartition) {
2149 auto foliagePrototypeId = foliageMapPartitionIt.first;
2150 for (
auto& transformations: foliageMapPartitionIt.second) {
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);
2159 auto transformationsMirrored = transformations;
2160 transformationsMirrored.setTranslation(
2162 width * 2.0f - transformationsMirrored.getTranslation().getX(),
2163 transformationsMirrored.getTranslation().getY(),
2164 flipZ ==
true?depth - transformationsMirrored.getTranslation().getZ():transformationsMirrored.getTranslation().getZ()
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();
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;
2179 if (partitionX >= partitionsX) partitionX = partitionsX - 1;
2180 auto partitionIdx = partitionZ * partitionsX + partitionX;
2181 foliageMapsMirrored[partitionIdx][foliagePrototypeId].push_back(transformationsMirrored);
2186 foliageMaps = foliageMapsMirrored;
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
2198 auto terrainHeightVectorVerticesPerX =
static_cast<int>(Math::ceil(width /
STEP_SIZE));
2199 auto terreinHeightVectorVerticesPerZ =
static_cast<int>(Math::ceil(depth /
STEP_SIZE));
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];
2211 terrainHeightVector = terrainHeightVectorMirrored;
2214 unordered_map<int, unordered_map<int, unordered_set<int>>> waterPositionMapsMirrored;
2215 unordered_map<int, float> waterPositionMapsHeightMirrored;
2217 for (
auto& waterPositionMapsIt: waterPositionMaps) {
2218 auto idx = waterPositionMapsIt.first;
2219 if (idx > idxMax) idxMax = idx;
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) {
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);
2235 waterPositionMapsHeight = waterPositionMapsHeightMirrored;
2236 waterPositionMaps = waterPositionMapsMirrored;
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;
2243 for (
auto& foliageMapPartition: foliageMaps) {
2244 for (
auto& foliageMapPartitionIt: foliageMapPartition) {
2245 auto foliagePrototypeId = foliageMapPartitionIt.first;
2246 for (
auto& transformations: foliageMapPartitionIt.second) {
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);
2255 auto transformationsMirrored = transformations;
2256 transformationsMirrored.setTranslation(
2258 flipX ==
true?width - transformationsMirrored.getTranslation().getX():transformationsMirrored.getTranslation().getX(),
2259 transformationsMirrored.getTranslation().getY(),
2260 depth * 2.0f - transformationsMirrored.getTranslation().getZ()
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();
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;
2275 if (partitionZ >= partitionsZ) partitionZ = partitionsZ - 1;
2276 auto partitionIdx = partitionZ * partitionsX + partitionX;
2277 foliageMapsMirrored[partitionIdx][foliagePrototypeId].push_back(transformationsMirrored);
2282 foliageMaps = foliageMapsMirrored;
ByteBuffer * getTextureData()
int32_t getTextureHeight() const
int32_t getTextureWidth() const
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
Node faces entity A node can have multiple entities containing faces and a applied material.
void setMaterial(Material *material)
Set up the entity's material.
void setLOD1Distance(float lod1Distance)
Set LOD1 distance.
void setLOD2Indices(const vector< int32_t > &lod2Indices)
Set LOD2 indices.
void setLOD2Distance(float lod2Distance)
Set LOD2 distance.
void setLOD3Distance(float lod3Distance)
Set LOD3 distance.
void setLOD1Indices(const vector< int32_t > &lod1Indices)
Set LOD1 indices.
void setLOD3Indices(const vector< int32_t > &lod3Indices)
Set LOD3 indices.
void setFaces(const vector< Face > &faces)
Set up entity's faces.
Representation of a 3d model.
Represents rotation orders of a model.
Represents specular material properties.
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
void extend(BoundingBox *boundingBox)
Extend bounding box with given bounding box.
const Vector3 & getDimensions() const
Line segment helper functions.
Matrix2D3x3 & translate(const Vector2 &v)
Sets up a translation matrix.
Matrix2D3x3 & identity()
Setup identity matrix.
Matrix2D3x3 & multiply(const Matrix2D3x3 &m)
Multiplies this matrix with another matrix.
Matrix2D3x3 clone() const
Clones this matrix.
Vector3 computeEulerAngles() const
Compute Euler angles (rotation around x, y, z axes)
Vector3 & normalize()
Normalize the vector.
Vector3 & set(float x, float y, float z)
Set up vector.
Vector3 clone() const
Clones the vector.
Vector3 & add(const Vector3 &v)
Adds a vector.
Vector3 & setY(float y)
Set Y.
static void println()
Print new line to console.
static constexpr float MAX_VALUE
static constexpr float MIN_VALUE
static constexpr int MIN_VALUE
static constexpr int MAX_VALUE
static void getWaterVertex(int x, int z, float waterHeight, Vector3 &vertex)
Get the terrain vertex for given x and z position.
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.
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.
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.
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.
static Vector3 computeWaterReflectionEnvironmentMappingPosition(const unordered_map< int, unordered_set< int > > &waterPositionMap, float waterHeight)
Compute water reflection environment mapping position.
static constexpr float PARTITION_SIZE
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.
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.
static bool determineWater(const vector< float > &terrainHeightVector, int verticesPerX, int verticesPerZ, int x, int z, float waterHeight)
Determine if water can be generated.
static bool hasWaterPosition(const unordered_map< int, unordered_set< int > > &waterPositionSet, int x, int z)
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.
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.
static bool getTerrainModelsHeight(BoundingBox &terrainBoundingBox, vector< Model * > &terrainModels, vector< float > &terrainHeightVector, const Vector3 &brushCenterPosition, float &brushHeight)
Get terrain models height for e.g.
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.
static constexpr float STEP_SIZE
static void emptyFoliageMaps(vector< unordered_map< int, vector< Transformations > > > &foliageMaps)
Empty foliage maps.
@ BRUSHOPERATION_WATER_ADD
@ BRUSHOPERATION_SUBTRACT
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.
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.
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.
static void getTerrainVertex(int x, int z, Vector3 &vertex)
Get the terrain vertex for given x and z position without providing y component.
static void createFoliageMaps(BoundingBox &terrainBoundingBox, vector< unordered_map< int, vector< Transformations > > > &foliageMaps)
Create foliage maps.
std::exception Exception
Exception base class.