TDME2 1.9.121
OctTreePartition.cpp
Go to the documentation of this file.
2
3#include <algorithm>
4#include <list>
5#include <string>
6#include <unordered_map>
7#include <unordered_set>
8#include <vector>
9
10#include <tdme/tdme.h>
12#include <tdme/engine/Entity.h>
13#include <tdme/engine/Frustum.h>
14#include <tdme/math/Math.h>
17
18using std::list;
19using std::remove;
20using std::string;
21using std::to_string;
22using std::unordered_map;
23using std::unordered_set;
24using std::vector;
25
33
34OctTreePartition::OctTreePartition()
35{
36 reset();
37}
38
40{
41 this->entityPartitionNodes.clear();
42 this->visibleEntities.clear();
43 this->treeRoot.partitionSize = -1;
44 this->treeRoot.x = -1;
45 this->treeRoot.y = -1;
46 this->treeRoot.z = -1;
47 this->treeRoot.parent = nullptr;
48}
49
51{
52 // update if already exists
53 vector<PartitionTreeNode*>* thisEntityPartitions = nullptr;
54 auto thisEntityPartitionsIt = entityPartitionNodes.find(entity->getId());
55 if (thisEntityPartitionsIt != entityPartitionNodes.end()) {
56 thisEntityPartitions = &thisEntityPartitionsIt->second;
57 }
58 if (thisEntityPartitions != nullptr && thisEntityPartitions->empty() == false) {
59 removeEntity(entity);
60 }
61 // frustum bounding box
62 auto boundingBox = entity->getBoundingBoxTransformed();
63 // find, create root nodes if not exists
64 auto minXPartition = static_cast< int32_t >(Math::floor(boundingBox->getMin().getX() / PARTITION_SIZE_MAX));
65 auto minYPartition = static_cast< int32_t >(Math::floor(boundingBox->getMin().getY() / PARTITION_SIZE_MAX));
66 auto minZPartition = static_cast< int32_t >(Math::floor(boundingBox->getMin().getZ() / PARTITION_SIZE_MAX));
67 auto maxXPartition = static_cast< int32_t >(Math::floor(boundingBox->getMax().getX() / PARTITION_SIZE_MAX));
68 auto maxYPartition = static_cast< int32_t >(Math::floor(boundingBox->getMax().getY() / PARTITION_SIZE_MAX));
69 auto maxZPartition = static_cast< int32_t >(Math::floor(boundingBox->getMax().getZ() / PARTITION_SIZE_MAX));
70 for (auto yPartition = minYPartition; yPartition <= maxYPartition; yPartition++)
71 for (auto xPartition = minXPartition; xPartition <= maxXPartition; xPartition++)
72 for (auto zPartition = minZPartition; zPartition <= maxZPartition; zPartition++) {
73 updatePartitionTree(&treeRoot, xPartition, yPartition, zPartition, PARTITION_SIZE_MAX, entity);
74 }
75}
76
78{
79 // check if we have entity in oct tree
80 vector<PartitionTreeNode*>* objectPartitionsVector = nullptr;
81 auto objectPartitionsVectorIt = entityPartitionNodes.find(entity->getId());
82 if (objectPartitionsVectorIt != entityPartitionNodes.end()) {
83 objectPartitionsVector = &objectPartitionsVectorIt->second;
84 }
85 if (objectPartitionsVector == nullptr || objectPartitionsVector->empty() == true) {
86 Console::println(
87 "OctTreePartition::removeEntity(): '" +
88 entity->getId() +
89 "' not registered"
90 );
91 return;
92 }
93 // remove object from assigned partitions
94 // TODO: remove tree root sub nodes as well not only empty root nodes
95 while (objectPartitionsVector->size() > 0) {
96 auto lastIdx = objectPartitionsVector->size() - 1;
97 auto partitionTreeNode = (*objectPartitionsVector)[lastIdx];
98 auto& partitionObjects = partitionTreeNode->partitionEntities;
99 partitionObjects.erase(remove(partitionObjects.begin(), partitionObjects.end(), entity), partitionObjects.end());
100 objectPartitionsVector->erase(objectPartitionsVector->begin() + lastIdx);
101 if (partitionObjects.empty() == true) {
102 auto rootPartitionTreeNode = partitionTreeNode;
103 while (rootPartitionTreeNode->parent != nullptr) rootPartitionTreeNode = rootPartitionTreeNode->parent;
104 // check if whole top level partition is empty
105 if (isPartitionNodeEmpty(rootPartitionTreeNode) == true) {
106 // yep, remove it
107 removePartitionNode(rootPartitionTreeNode);
108 for (auto treeRootSubNodeIt = treeRoot.subNodes.begin(); treeRootSubNodeIt != treeRoot.subNodes.end(); ++treeRootSubNodeIt) {
109 if ((void*)&treeRootSubNodeIt == (void*)rootPartitionTreeNode) {
110 treeRoot.subNodes.erase(treeRootSubNodeIt);
111 break;
112 }
113 }
114 string key = to_string(rootPartitionTreeNode->x) + "," + to_string(rootPartitionTreeNode->y) + "," + to_string(rootPartitionTreeNode->z);
116 }
117 }
118 }
119 entityPartitionNodes.erase(objectPartitionsVectorIt);
120}
121
122const vector<Entity*>& OctTreePartition::getVisibleEntities(Frustum* frustum)
123{
124 visibleEntities.clear();
125 visibleEntitiesSet.clear();
126 auto lookUps = 0;
127 for (auto& subNode: treeRoot.subNodes) {
128 lookUps += doPartitionTreeLookUpVisibleObjects(frustum, &subNode);
129 }
130 return visibleEntities;
131}
132
134 for (auto i = 0; i < indent; i++) Console::print("\t");
135 Console::println(to_string(node->x) + "/" + to_string(node->y) + "/" + to_string(node->z) + ": ");
136 for (auto entity: node->partitionEntities) {
137 for (auto i = 0; i < indent + 1; i++) Console::print("\t");
138 Console::println(entity->getId());
139 }
140 for (auto subNode: node->subNodes) dumpNode(&subNode, indent + 1);
141}
142
144 for (auto nodeEntity: node->partitionEntities) {
145 if (nodeEntity == entity) Console::println("OctTreePartition::findEntity(): found entity: " + entity->getId());
146 }
147 for (auto subNode: node->subNodes) findEntity(&subNode, entity);
148}
149
151 dumpNode(&treeRoot, 0);
152}
TDME engine entity.
Definition: Entity.h:31
virtual const string & getId()=0
virtual BoundingBox * getBoundingBoxTransformed()=0
Frustum class.
Definition: Frustum.h:30
Oct tree partition implementation.
bool isPartitionNodeEmpty(PartitionTreeNode *node)
Is partition empty.
void removeEntity(Entity *entity) override
Removes a entity.
void dumpNode(PartitionTreeNode *node, int indent)
Dump node to console.
const vector< Entity * > & getVisibleEntities(Frustum *frustum) override
Get visible entities.
void removePartitionNode(PartitionTreeNode *node)
Remove partition node, should be empty.
unordered_set< Entity * > visibleEntitiesSet
void addEntity(Entity *entity) override
Adds a entity.
void dump()
Dump oct tree to console.
void updatePartitionTree(PartitionTreeNode *parent, int32_t x, int32_t y, int32_t z, float partitionSize, Entity *entity)
Update partition tree.
int32_t doPartitionTreeLookUpVisibleObjects(Frustum *frustum, PartitionTreeNode *node)
Do partition tree lookup.
void findEntity(PartitionTreeNode *node, Entity *entity)
Find entity.
unordered_map< string, vector< PartitionTreeNode * > > entityPartitionNodes
static constexpr float PARTITION_SIZE_MAX
vector< Entity * > visibleEntities
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Definition: BoundingBox.h:25
Standard math functions.
Definition: Math.h:21
Console class.
Definition: Console.h:26
Vector iterator template to be used to iterate multiple vectors at a single invocation.
unordered_map< string, PartitionTreeNode * > subNodesByCoordinate