TDME2 1.9.121
OctTreePartition.h
Go to the documentation of this file.
1#pragma once
2
3#include <list>
4#include <string>
5#include <unordered_map>
6#include <unordered_set>
7#include <vector>
8
9#include <tdme/tdme.h>
12#include <tdme/engine/Entity.h>
13#include <tdme/engine/Frustum.h>
15#include <tdme/math/fwd-tdme.h>
19
20using std::list;
21using std::string;
22using std::to_string;
23using std::unordered_map;
24using std::unordered_set;
25using std::vector;
26
33
34/**
35 * Oct tree partition implementation
36 * @author Andreas Drewke
37 * @version $Id$
38 */
40{
41private:
43 {
44 // partition size
46
47 // x, y, z position
48 int32_t x;
49 int32_t y;
50 int32_t z;
51
52 // parent node
54
55 // node bounding volume
57
58 // sub nodes of oct tree nodes
59 list<PartitionTreeNode> subNodes;
60
61 // sub nodes of oct tree nodes by partition coordinate, only used in root node
62 unordered_map<string, PartitionTreeNode*> subNodesByCoordinate;
63
64 // or finally our partition entities
65 vector<Entity*> partitionEntities;
66 };
67
68 static constexpr float PARTITION_SIZE_MIN { 64.0f };
69 static constexpr float PARTITION_SIZE_MAX { 512.0f };
70
72 unordered_map<string, vector<PartitionTreeNode*>> entityPartitionNodes;
73 vector<Entity*> visibleEntities;
74 unordered_set<Entity*> visibleEntitiesSet;
76
77 // overridden methods
78 void reset() override;
79 void addEntity(Entity* entity) override;
80 inline void updateEntity(Entity* entity) override {
81 addEntity(entity);
82 }
83 void removeEntity(Entity* entity) override;
84
85 /**
86 * Update partition tree
87 * @param parent parent
88 * @param x x
89 * @param y y
90 * @param z z
91 * @param partitionSize partition size
92 * @param entity entity
93 */
94 inline void updatePartitionTree(PartitionTreeNode* parent, int32_t x, int32_t y, int32_t z, float partitionSize, Entity* entity) {
95 // key
96 string key = to_string(x) + "," + to_string(y) + "," + to_string(z);
97 auto storedNodeIt = parent->subNodesByCoordinate.find(key);
98 auto storedNode = storedNodeIt != parent->subNodesByCoordinate.end()?storedNodeIt->second:nullptr;
99
100 // check if exists
101 if (storedNode == nullptr) {
102 // if not add
104 node.partitionSize = partitionSize;
105 node.x = x;
106 node.y = y;
107 node.z = z;
108 node.parent = parent;
109 node.bv.getMin().set(
110 x * partitionSize,
111 y * partitionSize,
112 z * partitionSize
113 );
114 node.bv.getMax().set(
115 x * partitionSize + partitionSize,
116 y * partitionSize + partitionSize,
117 z * partitionSize + partitionSize
118 );
119 node.bv.update();
120 // register in parent sub nodes
121 parent->subNodes.push_back(node);
122 storedNode = &parent->subNodes.back();
123 // register in parent sub nodes by coordinate, if root node
124 parent->subNodesByCoordinate[key] = storedNode;
125 }
126
127 // create sub nodes
128 if (partitionSize > PARTITION_SIZE_MIN) {
129 // frustum bounding box
130 auto boundingBox = entity->getBoundingBoxTransformed();
131 // create partitions if not yet done
132 auto minXPartition = static_cast<int32_t>((Math::floor(boundingBox->getMin().getX() - x * partitionSize) / (partitionSize / 2.0f)));
133 auto minYPartition = static_cast<int32_t>((Math::floor(boundingBox->getMin().getY() - y * partitionSize) / (partitionSize / 2.0f)));
134 auto minZPartition = static_cast<int32_t>((Math::floor(boundingBox->getMin().getZ() - z * partitionSize) / (partitionSize / 2.0f)));
135 auto maxXPartition = static_cast<int32_t>((Math::floor(boundingBox->getMax().getX() - x * partitionSize) / (partitionSize / 2.0f)));
136 auto maxYPartition = static_cast<int32_t>((Math::floor(boundingBox->getMax().getY() - y * partitionSize) / (partitionSize / 2.0f)));
137 auto maxZPartition = static_cast<int32_t>((Math::floor(boundingBox->getMax().getZ() - z * partitionSize) / (partitionSize / 2.0f)));
138 minXPartition = Math::clamp(minXPartition, 0, 1);
139 minYPartition = Math::clamp(minYPartition, 0, 1);
140 minZPartition = Math::clamp(minZPartition, 0, 1);
141 maxXPartition = Math::clamp(maxXPartition, 0, 1);
142 maxYPartition = Math::clamp(maxYPartition, 0, 1);
143 maxZPartition = Math::clamp(maxZPartition, 0, 1);
144 for (auto yPartition = minYPartition; yPartition <= maxYPartition; yPartition++)
145 for (auto xPartition = minXPartition; xPartition <= maxXPartition; xPartition++)
146 for (auto zPartition = minZPartition; zPartition <= maxZPartition; zPartition++) {
148 storedNode,
149 static_cast<int32_t>((x * partitionSize) / (partitionSize / 2.0f) + xPartition),
150 static_cast<int32_t>((y * partitionSize) / (partitionSize / 2.0f) + yPartition),
151 static_cast<int32_t>((z * partitionSize) / (partitionSize / 2.0f) + zPartition),
152 partitionSize / 2.0f, entity
153 );
154 }
155 } else {
156 storedNode->partitionEntities.push_back(entity);
157 entityPartitionNodes[entity->getId()].push_back(storedNode);
158 }
159 }
160
161 /**
162 * Is partition empty
163 * @param node node
164 * @return partition empty
165 */
167 // lowest level node has objects attached?
168 if (node->partitionEntities.size() > 0) {
169 return false;
170 } else {
171 for (auto& subNode: node->subNodes) {
172 if (isPartitionNodeEmpty(&subNode) == false)
173 return false;
174
175 }
176 return true;
177 }
178 }
179
180 /**
181 * Remove partition node, should be empty
182 * @param node node
183 */
185 // lowest level node has objects attached?
186 if (node->partitionEntities.size() > 0) {
187 Console::println("OctTreePartition::removePartitionNode(): partition has objects attached!!!");
188 node->partitionEntities.clear();
189 } else {
190 // otherwise check top level node sub nodes
191 for (auto& subNode: node->subNodes) {
192 removePartitionNode(&subNode);
193 }
194 node->subNodesByCoordinate.clear();
195 node->subNodes.clear();
196 }
197 }
198
199 /**
200 * Do partition tree lookup
201 * @param frustum frustum
202 * @param node node
203 * @return number of look ups
204 */
206 auto lookUps = 1;
207 // check if given cbv collides with partition node bv
208 if (frustum->isVisible(&node->bv) == false) {
209 return lookUps;
210 } else
211 // otherwise check sub nodes
212 if (node->subNodes.size() > 0) {
213 for (auto& subNode: node->subNodes) {
214 lookUps += doPartitionTreeLookUpVisibleObjects(frustum, &subNode);
215 }
216 return lookUps;
217 } else
218 // last check if this node has partition entities
219 if (node->partitionEntities.size() > 0) {
220 for (auto i = 0; i < node->partitionEntities.size(); i++) {
221 auto entity = node->partitionEntities[i];
222
223 // look up
224 lookUps++;
225 if (frustum->isVisible(entity->getBoundingBoxTransformed()) == false) continue;
226
227 // lets have this only once in result
228 if (visibleEntitiesSet.count(entity) == 1) {
229 continue;
230 }
231 visibleEntitiesSet.insert(entity);
232
233 // done
234 visibleEntities.push_back(entity);
235 }
236 return lookUps;
237 }
238 return lookUps;
239 }
240
241 /**
242 * Dump node to console
243 * @param node node
244 * @param indent indent
245 */
246 void dumpNode(PartitionTreeNode* node, int indent);
247
248 /**
249 * Find entity
250 * @param node node
251 * @param entity entity
252 */
253 void findEntity(PartitionTreeNode* node, Entity* entity);
254
255public:
256 /**
257 * Public constructor
258 */
260
261 // overridden methods
262 const vector<Entity*>& getVisibleEntities(Frustum* frustum) override;
263 inline bool isVisibleEntity(Entity* entity) override {
264 return visibleEntitiesSet.count(entity) == 1;
265 }
266
267 /**
268 * Dump oct tree to console
269 */
270 void dump();
271
272};
TDME engine entity.
Definition: Entity.h:31
virtual const string & getId()=0
virtual BoundingBox * getBoundingBoxTransformed()=0
Frustum class.
Definition: Frustum.h:30
bool isVisible(const Vector3 &vertex)
Checks if given vertex is in frustum.
Definition: Frustum.h:71
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.
void updateEntity(Entity *entity) override
Updates a entity.
unordered_set< Entity * > visibleEntitiesSet
static constexpr float PARTITION_SIZE_MIN
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
bool isVisibleEntity(Entity *entity) override
Check if entity is visible.
OctTreePartition()
Public constructor.
VectorIteratorMultiple< Entity * > entityIterator
vector< Entity * > visibleEntities
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Definition: BoundingBox.h:25
void update()
Updates this bounding box.
Vector3 & set(float x, float y, float z)
Set up vector.
Definition: Vector3.h:73
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
Partition interface.
Definition: Partition.h:19