TDME2 1.9.121
PathFinding.cpp
Go to the documentation of this file.
2
3#include <algorithm>
4#include <map>
5#include <stack>
6#include <string>
7#include <unordered_map>
8#include <unordered_set>
9#include <vector>
10
11#include <tdme/tdme.h>
17#include <tdme/math/Math.h>
18#include <tdme/math/Vector3.h>
22#include <tdme/utilities/Time.h>
23
24using std::map;
25using std::reverse;
26using std::stack;
27using std::string;
28using std::to_string;
29using std::unordered_map;
30using std::unordered_set;
31using std::vector;
32
44
46
47PathFinding::PathFinding(
48 World* world,
49 bool sloping,
50 int stepsMax,
51 float actorHeight,
52 float stepSize,
53 float stepSizeLast,
54 float actorStepUpMax,
55 uint16_t skipOnCollisionTypeIds,
56 int maxTries,
57 float flowMapStepSize,
58 float flowMapScaleActorBoundingVolumes
59 ) {
60 this->world = world;
61 this->customTest = nullptr;
62 this->sloping = sloping;
63 this->actorBoundingVolume = nullptr;
64 this->actorBoundingVolumeSlopeTest = nullptr;
65 this->stepsMax = stepsMax;
66 this->actorHeight = actorHeight;
67 this->stepSize = stepSize;
68 this->stepSizeLast = stepSizeLast;
69 this->actorStepUpMax = actorStepUpMax;
70 this->skipOnCollisionTypeIds = skipOnCollisionTypeIds;
71 this->maxTries = maxTries;
72 this->collisionTypeIds = 0;
73 this->flowMapStepSize = flowMapStepSize;
74 this->flowMapScaleActorBoundingVolumes = flowMapScaleActorBoundingVolumes;
75}
76
78}
79
80bool PathFinding::isWalkableInternal(float x, float y, float z, float& height, float stepSize, float scaleActorBoundingVolumes, bool flowMapRequest, uint16_t collisionTypeIds, bool ignoreStepUpMax) {
81 string cacheId;
82 if (flowMapRequest == true) {
83 cacheId =
84 flowMapRequest == true?
85 "straight;" +
86 to_string(static_cast<int>(stepSize * 10.0f)) + ";" +
87 to_string(static_cast<int>(scaleActorBoundingVolumes * 10.0f)) + ";" +
88 toId(x, y, z, stepSize) + ";" +
89 to_string(collisionTypeIds) + ";" +
90 to_string(ignoreStepUpMax):
91 "";
92 auto walkableCacheIt = walkableCache.find(cacheId);
93 if (walkableCacheIt != walkableCache.end()) {
94 height = walkableCacheIt->second;
95 return height > -10000.0f;
96 }
97 }
98 auto walkable = isWalkable(x, y, z, height, stepSize, scaleActorBoundingVolumes, flowMapRequest, collisionTypeIds, ignoreStepUpMax);
99 if (flowMapRequest == true) walkableCache[cacheId] = walkable == false?-10000.0f:height;
100 if (walkable == false) return false;
101 return customTest == nullptr || customTest->isWalkable(this, x, height, z) == true;
102}
103
104bool PathFinding::isSlopeWalkableInternal(float x, float y, float z, float successorX, float successorY, float successorZ, float stepSize, float scaleActorBoundingVolumes, bool flowMapRequest, uint16_t collisionTypeIds) {
105 float slopeAngle = 0.0f;
106
107 // slope angle and center
108 auto toVector = Vector3(successorX, y, successorZ);
109 auto fromVector = Vector3(x, y, z);
110 auto axis = toVector.clone().sub(fromVector);
111 auto center = axis.clone().scale(0.5f).add(fromVector).setY(y + 0.1f);
112 axis.normalize();
113 slopeAngle = Vector3::computeAngle(
114 axis,
115 Vector3(0.0f, 0.0f, -1.0f),
116 Vector3(0.0f, 1.0f, 0.0f)
117 );
118
119 string cacheId;
120 if (flowMapRequest == true) {
121 auto cacheId =
122 "slope;" +
123 to_string(static_cast<int>(stepSize * 10.0f)) + ";" +
124 to_string(static_cast<int>(scaleActorBoundingVolumes * 10.0f)) + ";" +
125 toId(x, y, z, stepSize) + ";" +
126 to_string(collisionTypeIds) +
127 to_string(slopeAngle);
128 auto walkableCacheIt = walkableCache.find(cacheId);
129 if (walkableCacheIt != walkableCache.end()) {
130 auto height = walkableCacheIt->second;
131 return true;
132 }
133 }
134
135 // set up transformations
136 Transformations slopeTestTransformations;
137 slopeTestTransformations.setTranslation(center);
138 slopeTestTransformations.addRotation(Vector3(0.0f, 1.0f, 0.0f), slopeAngle);
139 slopeTestTransformations.update();
140
141 // update rigid body
142 auto actorSlopeTestCollisionBody = world->getBody("tdme.pathfinding.actor.slopetest");
143 actorSlopeTestCollisionBody->fromTransformations(slopeTestTransformations);
144
145 // check if actor collides with world
146 vector<Body*> collidedRigidBodies;
147 auto walkable = world->doesCollideWith(collisionTypeIds == 0?this->collisionTypeIds:collisionTypeIds, actorSlopeTestCollisionBody, collidedRigidBodies) == false;
148 if (flowMapRequest == true) walkableCache[cacheId] = walkable == false?-10000.0f:0.0f;
149 return walkable;
150}
151
152bool PathFinding::isWalkable(float x, float y, float z, float& height, float stepSize, float scaleActorBoundingVolumes, bool flowMapRequest, uint16_t collisionTypeIds, bool ignoreStepUpMax) {
153 // determine y height of ground plate of actor bounding volume
154 float _z = z - stepSize / 2.0f;
155 height = -10000.0f;
156 Vector3 actorPosition;
157 for (auto i = 0; i <= 4; i++) {
158 float _x = x - stepSize / 2.0f;
159 for (auto j = 0; j <= 4; j++) {
160 Vector3 actorPositionCandidate;
161 auto body = world->determineHeight(
162 collisionTypeIds == 0?this->collisionTypeIds:collisionTypeIds,
163 ignoreStepUpMax == true?10000.0f:actorStepUpMax,
164 actorPositionCandidate.set(_x, y, _z),
165 actorPosition,
166 -10000.0f,
167 ignoreStepUpMax == true?10000.0f:y + actorStepUpMax + 0.2f
168 );
169 if (body == nullptr || ((body->getCollisionTypeId() & skipOnCollisionTypeIds) != 0)) {
170 return false;
171 }
172 if (actorPosition.getY() > height) height = actorPosition.getY();
173 _x+= stepSize / 4.0f;
174 }
175 _z+= stepSize / 4.0f;
176 }
177
178 // set up transformations
179 Transformations actorTransformations;
180 actorTransformations.setTranslation(Vector3(x, ignoreStepUpMax == true?height + 0.2f:Math::min(y + actorStepUpMax, height + 0.2f), z));
181 actorTransformations.update();
182
183 // update rigid body
184 auto actorCollisionBody = world->getBody("tdme.pathfinding.actor");
185 actorCollisionBody->fromTransformations(actorTransformations);
186
187 // check if actor collides with world
188 vector<Body*> collidedRigidBodies;
189 return world->doesCollideWith(collisionTypeIds == 0?this->collisionTypeIds:collisionTypeIds, actorCollisionBody, collidedRigidBodies) == false;
190}
191
192void PathFinding::step(const PathFindingNode& node, float stepSize, float scaleActorBoundingVolumes, const unordered_set<string>* nodesToTestPtr, bool flowMapRequest) {
193 auto nodeId = node.id;
194
195 // Find valid successors
196 stack<PathFindingNode> successorNodes;
197 for (auto z = -1; z <= 1; z++)
198 for (auto x = -1; x <= 1; x++)
199 if ((z != 0 || x != 0) &&
200 (sloping == true ||
201 (Math::abs(x) == 1 && Math::abs(z) == 1) == false)) {
202 auto successorX = static_cast<float>(x) * stepSize + (nodesToTestPtr != nullptr?static_cast<float>(node.x) * stepSize:node.position.getX());
203 auto successorZ = static_cast<float>(z) * stepSize + (nodesToTestPtr != nullptr?static_cast<float>(node.z) * stepSize:node.position.getZ());
204 if (nodesToTestPtr != nullptr) {
205 auto successorNodeId = toIdInt(
206 node.x + x,
207 0,
208 node.z + z
209 );
210 if (nodesToTestPtr->find(successorNodeId) == nodesToTestPtr->end()) continue;
211 }
212 auto slopeWalkable = Math::abs(x) == 1 && Math::abs(z) == 1?isSlopeWalkableInternal(node.position.getX(), node.position.getY(), node.position.getZ(), successorX, node.position.getY(), successorZ, stepSize, scaleActorBoundingVolumes, flowMapRequest):true;
213 //
214 float yHeight;
215 // first node or walkable?
216 if (slopeWalkable == true && isWalkableInternal(successorX, node.position.getY(), successorZ, yHeight, stepSize, scaleActorBoundingVolumes, flowMapRequest) == true) {
217 // check if successor node equals previous node / node
218 if (equals(node, successorX, yHeight, successorZ)) {
219 continue;
220 }
221 // Add the node to the available sucessorNodes
222 PathFindingNode successorNode;
223 successorNode.position = Vector3(successorX, yHeight, successorZ);
224 successorNode.costsAll = 0.0f;
225 successorNode.costsReachPoint = 0.0f;
226 successorNode.costsEstimated = 0.0f;
227 successorNode.x = node.x + x;
228 successorNode.y = flowMapRequest == true?0:getIntegerPositionComponent(successorNode.position.getY(), 0.1f);
229 successorNode.z = node.z + z;
230 successorNode.id = toIdInt(
231 successorNode.x,
232 successorNode.y,
233 successorNode.z
234 );
235 successorNodes.push(successorNode);
236 }
237 }
238
239 // Check successor nodes
240 while (successorNodes.empty() == false) {
241 auto& successorNode = successorNodes.top();
242
243 // Compute successor node's costs by costs to reach nodes point and the computed distance from node to successor node
244 float successorCostsReachPoint = node.costsReachPoint + computeDistance(successorNode, node);
245
246 // Find sucessor node in open nodes list
247 PathFindingNode* openListNode = nullptr;
248 auto openListNodeIt = openNodes.find(successorNode.id);
249 if (openListNodeIt != openNodes.end()) {
250 openListNode = &openListNodeIt->second;
251 }
252
253 // found it in open nodes list
254 if (openListNode != nullptr) {
255 // is the node from open nodes less expensive, discard successor node
256 if (openListNode->costsReachPoint <= successorCostsReachPoint) {
257 // remove it from stack
258 successorNodes.pop();
259 // discard it
260 continue;
261 }
262 }
263
264 // Find successor node in closed nodes list
265 PathFindingNode* closedListNode = nullptr;
266 auto closedListNodeIt = closedNodes.find(successorNode.id);
267 if (closedListNodeIt != closedNodes.end()) {
268 closedListNode = &closedListNodeIt->second;
269 }
270
271 // found it in closed nodes list
272 if (closedListNode != nullptr) {
273 // is the node from closed nodes list less expensive, discard successor node
274 if (closedListNode->costsReachPoint <= successorCostsReachPoint) {
275 // remove it from stack
276 successorNodes.pop();
277 // discard it
278 continue;
279 }
280 }
281
282 // Sucessor node is the node with least cost to this point
283 successorNode.previousNodeId = nodeId;
284 successorNode.costsReachPoint = successorCostsReachPoint;
285 successorNode.costsEstimated = computeDistanceToEnd(successorNode);
286 successorNode.costsAll = successorNode.costsReachPoint + successorNode.costsEstimated;
287
288 // Remove found node from open nodes list, since it was less optimal
289 if (openListNode != nullptr) {
290 // remove open list node
291 openNodes.erase(openListNodeIt);
292 }
293
294 // Remove found node from closed nodes list, since it was less optimal
295 if (closedListNode != nullptr) {
296 closedNodes.erase(closedListNodeIt);
297 }
298
299 // Add successor node to open nodes list, as we might want to check its successors to find a path to the end
300 openNodes[successorNode.id] = successorNode;
301
302 // remove it from stack
303 successorNodes.pop();
304 }
305
306 // add node to closed nodes list, as we checked its successors
307 closedNodes[nodeId] = node;
308
309 // Remove node from open nodes, as we checked its successors
310 openNodes.erase(nodeId);
311}
312
314 const Vector3& startPosition,
315 const Vector3& endPosition,
316 float stepSize,
317 float scaleActorBoundingVolumes,
318 const uint16_t collisionTypeIds,
319 vector<Vector3>& path,
320 int alternativeEndSteps,
321 int maxTriesOverride,
322 PathFindingCustomTest* customTest) {
323 // clear path
324 path.clear();
325
326 //
327 auto now = Time::getCurrentMillis();
328
329 //
330 auto currentMaxTries = maxTriesOverride == -1?this->maxTries:maxTriesOverride;
331
332 // set up custom test
333 this->customTest = customTest;
334
335 // initialize custom test
336 if (this->customTest != nullptr) this->customTest->initialize();
337
338 //
339 this->collisionTypeIds = collisionTypeIds;
340
341 // init bounding volume, transformations, collision body
343 Vector3(0.0f, actorHeight / 2.0f, 0.0f),
344 OrientedBoundingBox::AABB_AXIS_X,
345 OrientedBoundingBox::AABB_AXIS_Y,
346 OrientedBoundingBox::AABB_AXIS_Z,
347 Vector3(stepSize * scaleActorBoundingVolumes, actorHeight / 2.0f, stepSize * scaleActorBoundingVolumes)
348 );
349 // set up transformations
350 Transformations actorTransformations;
351 actorTransformations.setTranslation(startPosition);
352 actorTransformations.update();
353 world->addCollisionBody("tdme.pathfinding.actor", true, 32768, actorTransformations, {actorBoundingVolume});
354
355 // init bounding volume for slope testcollision body
357 Vector3(0.0f, actorHeight / 2.0f, 0.0f),
358 OrientedBoundingBox::AABB_AXIS_X,
359 OrientedBoundingBox::AABB_AXIS_Y,
360 OrientedBoundingBox::AABB_AXIS_Z,
361 Vector3(stepSize * scaleActorBoundingVolumes * 2.5f, actorHeight / 2.0f, stepSize * scaleActorBoundingVolumes * 2.5f)
362 );
363 world->addCollisionBody("tdme.pathfinding.actor.slopetest", true, 32768, actorTransformations, {actorBoundingVolumeSlopeTest});
364
365 //
366 bool success = false;
367
368 // equal start and end position?
369 if (endPosition.clone().sub(startPosition).computeLengthSquared() < Math::square(0.1f)) {
370 if (VERBOSE == true) Console::println("PathFinding::findPath(): start position == end position! Exiting!");
371 path.push_back(startPosition);
372 float endYHeight = endPosition.getY();
374 endPosition.getX(),
375 endPosition.getY(),
376 endPosition.getZ(),
377 endYHeight,
378 stepSize,
379 scaleActorBoundingVolumes,
380 false
381 ) == false) {
382 path.push_back(endPosition);
383 success = true;
384 }
385 } else
386 // equal start and end position?
387 if (startPosition.clone().sub(endPosition).computeLengthSquared() < stepSizeLast * stepSizeLast + stepSizeLast * stepSizeLast + 0.1f) {
388 if (VERBOSE == true) Console::println("PathFinding::findPath(): end - start position < stepSizeLast! Exiting!");
389 path.push_back(startPosition);
390 float endYHeight = endPosition.getY();
392 endPosition.getX(),
393 endPosition.getY(),
394 endPosition.getZ(),
395 endYHeight,
396 stepSize,
397 scaleActorBoundingVolumes,
398 false
399 ) == false) {
400 path.push_back(endPosition);
401 success = true;
402 }
403 }
404
405 //
406 auto tries = 0;
407 auto endPositionIdx = 0;
408
409 //
410 if (success == false) {
411 //
412 auto forcedAlternativeEndSteps = 0;
413
414 // compute possible end positions
415 vector<Vector3> startPositionCandidates;
416 vector<Vector3> endPositionCandidates;
417 {
418 auto forwardVector = endPosition.clone().sub(startPosition).setY(0.0f).normalize();
419 auto sideVector = Vector3::computeCrossProduct(forwardVector, Vector3(0.0f, 1.0f, 0.0f)).setY(0.0f).normalize();
420 if (Float::isNaN(sideVector.getX()) ||
421 Float::isNaN(sideVector.getY()) ||
422 Float::isNaN(sideVector.getZ())) {
423 if (VERBOSE == true) Console::println("PathFinding::findPath(): side vector = NaN!");
424 endPositionCandidates.push_back(endPosition);
425 startPositionCandidates.push_back(startPosition);
426 } else {
427 {
428 auto sideDistance = stepSize;
429 auto forwardDistance = 0.0f;
430 auto i = 0;
431 while (true == true) {
432 endPositionCandidates.push_back(Vector3().set(sideVector).scale(0.0f).add(forwardVector.clone().scale(-forwardDistance)).add(endPosition));
433 i++; if (i >= alternativeEndSteps) break;
434 endPositionCandidates.push_back(Vector3().set(sideVector).scale(-sideDistance).add(forwardVector.clone().scale(-forwardDistance)).add(endPosition));
435 i++; if (i >= alternativeEndSteps) break;
436 endPositionCandidates.push_back(Vector3().set(sideVector).scale(+sideDistance).add(forwardVector.clone().scale(-forwardDistance)).add(endPosition));
437 i++; if (i >= alternativeEndSteps) break;
438 forwardDistance+= stepSize;
439 }
440 forwardDistance = 0.0f;
441 forwardVector.scale(-1.0f);
442 sideVector.scale(-1.0f);
443 startPositionCandidates.push_back(Vector3().set(sideVector).scale(0.0f).add(forwardVector.clone().scale(-forwardDistance)).add(startPosition));
444 startPositionCandidates.push_back(Vector3().set(sideVector).scale(-sideDistance).add(forwardVector.clone().scale(-forwardDistance)).add(startPosition));
445 startPositionCandidates.push_back(Vector3().set(sideVector).scale(+sideDistance).add(forwardVector.clone().scale(-forwardDistance)).add(startPosition));
446 startPositionCandidates.push_back(Vector3().set(sideVector).scale(0.0f).add(forwardVector.clone().scale(forwardDistance)).add(startPosition));
447 startPositionCandidates.push_back(Vector3().set(sideVector).scale(-sideDistance).add(forwardVector.clone().scale(forwardDistance)).add(startPosition));
448 startPositionCandidates.push_back(Vector3().set(sideVector).scale(+sideDistance).add(forwardVector.clone().scale(forwardDistance)).add(startPosition));
449 forwardDistance+= stepSize;
450 startPositionCandidates.push_back(Vector3().set(sideVector).scale(0.0f).add(forwardVector.clone().scale(-forwardDistance)).add(startPosition));
451 startPositionCandidates.push_back(Vector3().set(sideVector).scale(-sideDistance).add(forwardVector.clone().scale(-forwardDistance)).add(startPosition));
452 startPositionCandidates.push_back(Vector3().set(sideVector).scale(+sideDistance).add(forwardVector.clone().scale(-forwardDistance)).add(startPosition));
453 startPositionCandidates.push_back(Vector3().set(sideVector).scale(0.0f).add(forwardVector.clone().scale(forwardDistance)).add(startPosition));
454 startPositionCandidates.push_back(Vector3().set(sideVector).scale(-sideDistance).add(forwardVector.clone().scale(forwardDistance)).add(startPosition));
455 startPositionCandidates.push_back(Vector3().set(sideVector).scale(+sideDistance).add(forwardVector.clone().scale(forwardDistance)).add(startPosition));
456 forwardDistance+= stepSize;
457 startPositionCandidates.push_back(Vector3().set(sideVector).scale(0.0f).add(forwardVector.clone().scale(-forwardDistance)).add(startPosition));
458 startPositionCandidates.push_back(Vector3().set(sideVector).scale(-sideDistance).add(forwardVector.clone().scale(-forwardDistance)).add(startPosition));
459 startPositionCandidates.push_back(Vector3().set(sideVector).scale(+sideDistance).add(forwardVector.clone().scale(-forwardDistance)).add(startPosition));
460 startPositionCandidates.push_back(Vector3().set(sideVector).scale(0.0f).add(forwardVector.clone().scale(forwardDistance)).add(startPosition));
461 startPositionCandidates.push_back(Vector3().set(sideVector).scale(-sideDistance).add(forwardVector.clone().scale(forwardDistance)).add(startPosition));
462 startPositionCandidates.push_back(Vector3().set(sideVector).scale(+sideDistance).add(forwardVector.clone().scale(forwardDistance)).add(startPosition));
463 }
464 if (currentMaxTries == 0) {
465 forcedAlternativeEndSteps = 27 * 2;
466 auto forwardDistance = 0.0f;
467 auto i = 0;
468 while (true == true) {
469 endPositionCandidates.push_back(Vector3().set(sideVector).scale(0.0f).add(forwardVector.clone().scale(forwardDistance)).add(endPosition));
470 i++; if (i >= forcedAlternativeEndSteps) break;
471 endPositionCandidates.push_back(Vector3().set(sideVector).scale(-0.3f).add(forwardVector.clone().scale(forwardDistance)).add(endPosition));
472 i++; if (i >= forcedAlternativeEndSteps) break;
473 endPositionCandidates.push_back(Vector3().set(sideVector).scale(-0.2f).add(forwardVector.clone().scale(forwardDistance)).add(endPosition));
474 i++; if (i >= forcedAlternativeEndSteps) break;
475 endPositionCandidates.push_back(Vector3().set(sideVector).scale(-0.1f).add(forwardVector.clone().scale(forwardDistance)).add(endPosition));
476 i++; if (i >= forcedAlternativeEndSteps) break;
477 endPositionCandidates.push_back(Vector3().set(sideVector).scale(+0.1f).add(forwardVector.clone().scale(forwardDistance)).add(endPosition));
478 i++; if (i >= forcedAlternativeEndSteps) break;
479 endPositionCandidates.push_back(Vector3().set(sideVector).scale(+0.2f).add(forwardVector.clone().scale(forwardDistance)).add(endPosition));
480 i++; if (i >= forcedAlternativeEndSteps) break;
481 endPositionCandidates.push_back(Vector3().set(sideVector).scale(+0.3f).add(forwardVector.clone().scale(forwardDistance)).add(endPosition));
482 i++; if (i >= forcedAlternativeEndSteps) break;
483 endPositionCandidates.push_back(Vector3().set(sideVector).scale(0.0f).add(forwardVector.clone().scale(-forwardDistance)).add(endPosition));
484 i++; if (i >= forcedAlternativeEndSteps) break;
485 endPositionCandidates.push_back(Vector3().set(sideVector).scale(-0.3f).add(forwardVector.clone().scale(-forwardDistance)).add(endPosition));
486 i++; if (i >= forcedAlternativeEndSteps) break;
487 endPositionCandidates.push_back(Vector3().set(sideVector).scale(-0.2f).add(forwardVector.clone().scale(-forwardDistance)).add(endPosition));
488 i++; if (i >= forcedAlternativeEndSteps) break;
489 endPositionCandidates.push_back(Vector3().set(sideVector).scale(-0.1f).add(forwardVector.clone().scale(-forwardDistance)).add(endPosition));
490 i++; if (i >= forcedAlternativeEndSteps) break;
491 endPositionCandidates.push_back(Vector3().set(sideVector).scale(+0.1f).add(forwardVector.clone().scale(-forwardDistance)).add(endPosition));
492 i++; if (i >= forcedAlternativeEndSteps) break;
493 endPositionCandidates.push_back(Vector3().set(sideVector).scale(+0.2f).add(forwardVector.clone().scale(-forwardDistance)).add(endPosition));
494 i++; if (i >= forcedAlternativeEndSteps) break;
495 endPositionCandidates.push_back(Vector3().set(sideVector).scale(+0.3f).add(forwardVector.clone().scale(-forwardDistance)).add(endPosition));
496 i++; if (i >= forcedAlternativeEndSteps) break;
497 forwardDistance+= 0.1f;
498 }
499 }
500 }
501 }
502
503 Vector3 startPositionComputed = startPosition;
504 for (auto& startPositionCandidate: startPositionCandidates) {
505 float startYHeight = startPositionCandidate.getY();
507 startPositionCandidate.getX(),
508 startPositionCandidate.getY(),
509 startPositionCandidate.getZ(),
510 startYHeight,
511 stepSize,
512 scaleActorBoundingVolumes,
513 false
514 ) == false) {
515 if (VERBOSE == true) {
517 "PathFinding::findPath(): Start position not walkable: " +
518 to_string(startPositionCandidate.getX()) + ", " +
519 to_string(startPositionCandidate.getY()) + ", " +
520 to_string(startPositionCandidate.getZ()) + " / " +
521 to_string(startYHeight)
522 );
523 }
524 //
525 continue;
526 } else {
527 startPositionComputed = startPositionCandidate;
528 startPositionComputed.setY(startYHeight);
529 break;
530 }
531 }
532
533 for (auto& endPositionCandidate: endPositionCandidates) {
534 Vector3 endPositionComputed = endPositionCandidate;
535 float endYHeight = endPositionComputed.getY();
536 if (alternativeEndSteps > 0 &&
538 endPositionComputed.getX(),
539 endPositionComputed.getY(),
540 endPositionComputed.getZ(),
541 endYHeight,
542 stepSize,
543 scaleActorBoundingVolumes,
544 false
545 ) == false) {
546 if (VERBOSE == true) {
548 "PathFinding::findPath(): End position not walkable: " +
549 to_string(endPositionComputed.getX()) + ", " +
550 to_string(endPositionComputed.getY()) + ", " +
551 to_string(endPositionComputed.getZ()) + " / " +
552 to_string(endYHeight)
553 );
554 }
555 //
556 continue;
557 } else {
558 endPositionComputed.setY(endYHeight);
559 }
560
561 if (VERBOSE == true) {
563 "Finding path: " +
564 to_string(startPositionComputed.getX()) + ", " +
565 to_string(startPositionComputed.getY()) + ", " +
566 to_string(startPositionComputed.getZ()) + " --> " +
567 to_string(endPositionComputed.getX()) + ", " +
568 to_string(endPositionComputed.getY()) + ", " +
569 to_string(endPositionComputed.getZ())
570 );
571 }
572
573 // end node + start node
574 {
575 // start node
576 PathFindingNode start;
577 start.position = startPositionComputed;
578 start.costsAll = 0.0f;
579 start.costsReachPoint = 0.0f;
580 start.costsEstimated = 0.0f;
582 start.y = 0;
584 start.id = toId(
585 start.position.getX(),
586 start.position.getY(),
587 start.position.getZ(),
589 );
590
591 // end node
592 end.position = endPositionComputed;
593 end.costsAll = 0.0f;
594 end.costsReachPoint = 0.0f;
595 end.costsEstimated = 0.0f;
597 end.y = 0;
599 end.id = toId(
600 end.position.getX(),
601 end.position.getY(),
602 end.position.getZ(),
604 );
605
606 // set up start node costs
608 start.costsAll = start.costsEstimated;
609
610 // put to open nodes
611 openNodes[start.id] = start;
612 }
613
614 // do the steps
615 bool done = false;
616 for (auto stepIdx = 0; done == false && stepIdx < stepsMax; stepIdx++) {
617 // check if there are still open nodes available
618 if (openNodes.size() == 0) {
619 done = true;
620 break;
621 }
622
623 // Choose node from open nodes thats least expensive to check its successors
624 PathFindingNode* nodePtr = nullptr;
625 PathFindingNode* endNodePtr = nullptr;
626 for (auto nodeIt = openNodes.begin(); nodeIt != openNodes.end(); ++nodeIt) {
627 if (equalsLastNode(nodeIt->second, end) == true && (endNodePtr == nullptr || nodeIt->second.costsAll < endNodePtr->costsAll)) endNodePtr = &nodeIt->second;
628 if (nodePtr == nullptr || nodeIt->second.costsAll < nodePtr->costsAll) nodePtr = &nodeIt->second;
629 }
630
631 //
632 if (nodePtr == nullptr) {
633 done = true;
634 break;
635 }
636
637 //
638 if (endNodePtr != nullptr) {
639 const auto& node = *endNodePtr;
640 end.previousNodeId = node.previousNodeId;
641 // Console::println("PathFinding::findPath(): path found with steps: " + to_string(stepIdx));
642 int nodesCount = 0;
643 unordered_map<string, PathFindingNode>::iterator nodeIt;
644 for (auto nodePtr = &end; nodePtr != nullptr; nodePtr = (nodeIt = closedNodes.find(nodePtr->previousNodeId)) != closedNodes.end()?&nodeIt->second:nullptr) {
645 nodesCount++;
646 // if (nodesCount > 0 && nodesCount % 100 == 0) {
647 // Console::println("PathFinding::findPath(): compute path: steps: " + to_string(nodesCount) + " / " + to_string(path.size()) + ": " + to_string((uint64_t)node));
648 // }
649 if (Float::isNaN(nodePtr->position.getX()) ||
650 Float::isNaN(nodePtr->position.getY()) ||
651 Float::isNaN(nodePtr->position.getZ())) {
652 Console::println("PathFinding::findPath(): compute path: step: NaN");
653 done = true;
654 break;
655 }
656 path.push_back(nodePtr->position);
657 }
658 reverse(path.begin(), path.end());
659 if (path.size() > 1) path.erase(path.begin());
660 if (path.size() == 0) {
661 // Console::println("PathFinding::findPath(): path with 0 steps: Fixing!");
662 path.push_back(endPositionComputed);
663 }
664 done = true;
665 success = true;
666 break;
667 } else {
668 const auto& node = *nodePtr;
669 // do a step
670 step(node, stepSize, scaleActorBoundingVolumes, nullptr, false);
671 }
672 }
673
674 // reset
675 openNodes.clear();
676 closedNodes.clear();
677
678 //
679 tries++;
680
681 //
682 if (success == true || tries >= currentMaxTries + forcedAlternativeEndSteps) break;
683 }
684
685 //
686 if (tries == 0) {
687 Console::println("PathFinding::findPath(): end position were not walkable!");
688 }
689 }
690
691 //
692 /*
693 if (stepIdx == stepsMax) {
694 Console::println("PathFinding::findPath(): steps == stepsMax: " + to_string(stepIdx));
695 }
696 */
697
698 // unset actor bounding volume and remove rigid body
699 world->removeBody("tdme.pathfinding.actor");
700 world->removeBody("tdme.pathfinding.actor.slopetest");
701 delete actorBoundingVolume;
702 actorBoundingVolume = nullptr;
705
706 //
707 if (VERBOSE == true && tries > 1) Console::println("PathFinding::findPath(): time: " + to_string(Time::getCurrentMillis() - now) + "ms / " + to_string(tries) + " tries, success = " + to_string(success) + ", path steps: " + to_string(path.size()));
708
709 // dispose custom test
710 if (this->customTest != nullptr) {
711 this->customTest->dispose();
712 this->customTest = nullptr;
713 }
714
715 // return success
716 return success;
717}
718
719FlowMap* PathFinding::createFlowMap(const vector<Vector3>& endPositions, const Vector3& center, float depth, float width, const uint16_t collisionTypeIds, const vector<Vector3>& path, bool complete, PathFindingCustomTest* customTest) {
720 // set up end position in costs map
721 if (path.empty() == true) {
722 Console::println("PathFinding::createFlowMap(): no path given");
723 return nullptr;
724 }
725
726 // set up custom test
727 this->customTest = customTest;
728
729 // initialize custom test
730 if (this->customTest != nullptr) this->customTest->initialize();
731
732 //
733 this->collisionTypeIds = collisionTypeIds;
734
735 // init bounding volume, transformations, collision body
737 Vector3(0.0f, actorHeight / 2.0f, 0.0f),
738 OrientedBoundingBox::AABB_AXIS_X,
739 OrientedBoundingBox::AABB_AXIS_Y,
740 OrientedBoundingBox::AABB_AXIS_Z,
742 );
743 // set up transformations
744 Transformations actorTransformations;
745 actorTransformations.setTranslation(endPositions[0]);
746 actorTransformations.update();
747 world->addCollisionBody("tdme.pathfinding.actor", true, 32768, actorTransformations, {actorBoundingVolume});
748
749 // init bounding volume for slope testcollision body
751 Vector3(0.0f, actorHeight / 2.0f, 0.0f),
752 OrientedBoundingBox::AABB_AXIS_X,
753 OrientedBoundingBox::AABB_AXIS_Y,
754 OrientedBoundingBox::AABB_AXIS_Z,
756 );
757 world->addCollisionBody("tdme.pathfinding.actor.slopetest", true, 32768, actorTransformations, {actorBoundingVolumeSlopeTest});
758
759 //
760 auto zMin = static_cast<int>(Math::ceil(-depth / 2.0f / flowMapStepSize));
761 auto zMax = static_cast<int>(Math::ceil(depth / 2.0f / flowMapStepSize));
762 auto xMin = static_cast<int>(Math::ceil(-width / 2.0f / flowMapStepSize));
763 auto xMax = static_cast<int>(Math::ceil(width / 2.0f / flowMapStepSize));
764 const vector<Vector3> emptyPath = { center };
765 const vector<Vector3>& pathToUse = path.empty() == false?path:emptyPath;
766
767 // set up end position in costs map
768 if (endPositions.size() == 0) {
769 Console::println("PathFinding::createFlowMap(): no end positions given");
770 }
771
772 // end node
773 end.position = path[0];
774 end.costsAll = 0.0f;
775 end.costsReachPoint = 0.0f;
776 end.costsEstimated = 0.0f;
778 end.y = 0;
780 end.id = toId(
781 end.position.getX(),
782 end.position.getY(),
784 );
785
786 // add end position to open nodes/start nodes
787 for (auto& endPosition: endPositions) {
788 auto nodePosition = Vector3(
790 endPosition.getY(),
792 );
793 float nodeYHeight;
794 isWalkableInternal(nodePosition.getX(), nodePosition.getY(), nodePosition.getZ(), nodeYHeight, flowMapStepSize, flowMapScaleActorBoundingVolumes, true);
795 PathFindingNode node;
796 node.position.set(nodePosition).setY(nodeYHeight);
797 node.costsAll = 0.0f;
798 node.costsReachPoint = 0.0f;
799 node.costsEstimated = 0.0f;
800 node.x = FlowMap::getIntegerPositionComponent(nodePosition.getX(), flowMapStepSize);
801 node.y = 0;
802 node.z = FlowMap::getIntegerPositionComponent(nodePosition.getZ(), flowMapStepSize);
803 node.id = toIdInt(node.x, node.y, node.z);
804 // set up start node costs
806 node.costsAll = node.costsEstimated;
807 // put to open nodes
808 openNodes[node.id] = node;
809 }
810
811 // nodes to test
812 unordered_set<string> nodesToTest;
813 for (auto& _centerPathNode: pathToUse) {
814 auto centerPathNode = Vector3(
815 FlowMap::alignPositionComponent(_centerPathNode.getX(), flowMapStepSize),
816 _centerPathNode.getY(),
817 FlowMap::alignPositionComponent(_centerPathNode.getZ(), flowMapStepSize)
818 );
819 auto centerPathNodeX = FlowMap::getIntegerPositionComponent(centerPathNode.getX(), flowMapStepSize);
820 auto centerPathNodeZ = FlowMap::getIntegerPositionComponent(centerPathNode.getZ(), flowMapStepSize);
821 for (auto z = zMin; z <= zMax; z++) {
822 for (auto x = xMin; x <= xMax; x++) {
823 auto nodeId = toIdInt(
824 centerPathNodeX + x,
825 0,
826 centerPathNodeZ + z
827 );
828 nodesToTest.insert(nodeId);
829 }
830 }
831 }
832
833 // Do A* on open nodes
834 while (openNodes.size() > 0) {
835 // Choose node from open nodes thats least expensive to check its successors
836 PathFindingNode* nodePtr = nullptr;
837 for (auto nodeIt = openNodes.begin(); nodeIt != openNodes.end(); ++nodeIt) {
838 if (nodePtr == nullptr || nodeIt->second.costsAll < nodePtr->costsAll) nodePtr = &nodeIt->second;
839 }
840 if (nodePtr == nullptr) break;
841 const auto& node = *nodePtr;
842
843 // do a step
844 step(node, flowMapStepSize, flowMapScaleActorBoundingVolumes, &nodesToTest, true);
845 }
846
847 // clear nodes to test
848 nodesToTest.clear();
849
850 // generate flow map, which is based on
851 // see: https://howtorts.github.io/2014/01/04/basic-flow-fields.html
852 auto flowMap = new FlowMap(pathToUse, endPositions, flowMapStepSize, complete);
853 flowMap->acquireReference();
854 for (auto& _centerPathNode: pathToUse) {
855 auto centerPathNode = Vector3(
856 FlowMap::alignPositionComponent(_centerPathNode.getX(), flowMapStepSize),
857 _centerPathNode.getY(),
858 FlowMap::alignPositionComponent(_centerPathNode.getZ(), flowMapStepSize)
859 );
860 auto centerPathNodeX = FlowMap::getIntegerPositionComponent(centerPathNode.getX(), flowMapStepSize);
861 auto centerPathNodeZ = FlowMap::getIntegerPositionComponent(centerPathNode.getZ(), flowMapStepSize);
862 for (auto z = zMin; z <= zMax; z++) {
863 for (auto x = xMin; x <= xMax; x++) {
864 auto cellPosition = Vector3(
865 static_cast<float>(centerPathNodeX) * flowMapStepSize + static_cast<float>(x) * flowMapStepSize,
866 0.0f,
867 static_cast<float>(centerPathNodeZ) * flowMapStepSize + static_cast<float>(z) * flowMapStepSize
868 );
869 auto cellId = FlowMap::toIdInt(
870 centerPathNodeX + x,
871 centerPathNodeZ + z
872 );
873 auto nodeId = toIdInt(
874 centerPathNodeX + x,
875 0,
876 centerPathNodeZ + z
877 );
878
879 // do we already have this cell?
880 if (flowMap->hasCell(cellId) == true) continue;
881
882 // walkable?
883 auto nodeIt = closedNodes.find(nodeId);
884 if (nodeIt == closedNodes.end()) {
885 continue;
886 }
887 auto& node = nodeIt->second;
888 // set y
889 cellPosition.setY(node.position.getY());
890
891 // check neighbours around our current cell
892 PathFindingNode* minCostsNode = nullptr;
893 auto minCosts = Float::MAX_VALUE;
894 for (auto _z = -1; _z <= 1; _z++)
895 for (auto _x = -1; _x <= 1; _x++)
896 if (_z != 0 || _x != 0) {
897 //
898 auto neighbourNodeId = toIdInt(
899 centerPathNodeX + x + _x,
900 0,
901 centerPathNodeZ + z + _z
902 );
903 // same node?
904 if (neighbourNodeId == nodeId) continue;
905 // do we have this cell?
906 auto neighbourNodeIt = closedNodes.find(neighbourNodeId);
907 if (neighbourNodeIt == closedNodes.end()) {
908 // nope
909 continue;
910 } else {
911 // yes && walkable
912 auto& neighbourNode = neighbourNodeIt->second;
913 if (minCostsNode == nullptr || neighbourNode.costsReachPoint < minCosts) {
914 minCostsNode = &neighbourNode;
915 minCosts = neighbourNode.costsReachPoint;
916 }
917 }
918 }
919 if (minCostsNode != nullptr) {
920 auto direction = minCostsNode->position.clone().sub(node.position).setY(0.0f).normalize();
921 if (Float::isNaN(direction.getX()) || Float::isNaN(direction.getY()) || Float::isNaN(direction.getZ())) {
923 minCostsNode->id + "; " +
924 to_string(minCostsNode->position.getX()) + ", " +
925 to_string(minCostsNode->position.getY()) + ", " +
926 to_string(minCostsNode->position.getZ()) + " -> " +
927 node.id + "; " +
928 to_string(node.position.getX()) + ", " +
929 to_string(node.position.getY()) + ", " +
930 to_string(node.position.getZ()) + ": " +
931 to_string(minCostsNode == &node) + "; " +
932 to_string(cellPosition.getX()) + ", " +
933 to_string(cellPosition.getY()) + ", " +
934 to_string(cellPosition.getZ()) + "; " +
935 cellId
936 );
937 }
938 flowMap->addCell(
939 cellId,
940 cellPosition,
941 true,
942 direction,
943 -1
944 );
945 }
946 }
947 }
948 }
949
950 // reset
951 openNodes.clear();
952 closedNodes.clear();
953
954 // do some post adjustments
955 for (auto& _centerPathNode: pathToUse) {
956 auto centerPathNode = Vector3(
957 FlowMap::alignPositionComponent(_centerPathNode.getX(), flowMapStepSize),
958 _centerPathNode.getY(),
959 FlowMap::alignPositionComponent(_centerPathNode.getZ(), flowMapStepSize)
960 );
961 auto centerPathNodeX = FlowMap::getIntegerPositionComponent(centerPathNode.getX(), flowMapStepSize);
962 auto centerPathNodeZ = FlowMap::getIntegerPositionComponent(centerPathNode.getZ(), flowMapStepSize);
963 for (auto z = zMin; z <= zMax; z++) {
964 for (auto x = xMin; x <= xMax; x++) {
965 auto cellPosition = Vector3(
966 static_cast<float>(centerPathNodeX) * flowMapStepSize + static_cast<float>(x) * flowMapStepSize,
967 0.0f,
968 static_cast<float>(centerPathNodeZ) * flowMapStepSize + static_cast<float>(z) * flowMapStepSize
969 );
970 auto cellId = FlowMap::toIdInt(
971 centerPathNodeX + x,
972 centerPathNodeZ + z
973 );
974 auto cell = flowMap->getCell(cellId);
975 if (cell == nullptr) continue;
976 auto topCell = flowMap->getCell(FlowMap::toIdInt(centerPathNodeX + x, centerPathNodeZ + z - 1));
977 if (topCell == nullptr && Math::abs(cell->getDirection().getX()) > 0.0f && cell->getDirection().getZ() < 0.0f){
978 cell->setDirection(cell->getDirection().clone().setZ(0.0f).normalize());
979 }
980 auto bottomCell = flowMap->getCell(FlowMap::toIdInt(centerPathNodeX + x, centerPathNodeZ + z + 1));
981 if (bottomCell == nullptr && Math::abs(cell->getDirection().getX()) > 0.0f && cell->getDirection().getZ() > 0.0f){
982 cell->setDirection(cell->getDirection().clone().setZ(0.0f).normalize());
983 }
984 auto leftCell = flowMap->getCell(FlowMap::toIdInt(centerPathNodeX + x - 1, centerPathNodeZ + z));
985 if (leftCell == nullptr && cell->getDirection().getX() < 0.0f && Math::abs(cell->getDirection().getZ()) > 0.0f){
986 cell->setDirection(cell->getDirection().clone().setX(0.0f).normalize());
987 }
988 auto rightCell = flowMap->getCell(FlowMap::toIdInt(centerPathNodeX + x + 1, centerPathNodeZ + z));
989 if (rightCell == nullptr && cell->getDirection().getX() > 0.0f && Math::abs(cell->getDirection().getZ()) > 0.0f){
990 cell->setDirection(cell->getDirection().clone().setX(0.0f).normalize());
991 }
992 }
993 }
994 }
995
996 // do some more post adjustments
997 Vector3 lastCenterPathNode = pathToUse.size() < 2?Vector3():pathToUse[0] - (pathToUse[1] - pathToUse[0]);
998 unordered_set<string> cellsProcessed;
999 for (int i = pathToUse.size() - 1; i >= 0; i--) {
1000 auto _centerPathNode = pathToUse[i];
1001 auto centerPathNode = Vector3(
1002 FlowMap::alignPositionComponent(_centerPathNode.getX(), flowMapStepSize),
1003 _centerPathNode.getY(),
1004 FlowMap::alignPositionComponent(_centerPathNode.getZ(), flowMapStepSize)
1005 );
1006 auto pathDirection = (lastCenterPathNode - centerPathNode).setY(0.0f).normalize();
1007 auto centerPathNodeX = FlowMap::getIntegerPositionComponent(centerPathNode.getX(), flowMapStepSize);
1008 auto centerPathNodeZ = FlowMap::getIntegerPositionComponent(centerPathNode.getZ(), flowMapStepSize);
1009 for (auto z = zMin; z <= zMax; z++) {
1010 for (auto x = xMin; x <= xMax; x++) {
1011 auto cellId = FlowMap::toIdInt(
1012 centerPathNodeX + x,
1013 centerPathNodeZ + z
1014 );
1015 auto cell = flowMap->getCell(cellId);
1016 if (cell == nullptr) continue;
1017
1018 // check if we have missing neighbour cells
1019 auto hadMissingNeighborCell = false;
1020 for (auto nZ = -1; nZ < 2 && hadMissingNeighborCell == false; nZ++) {
1021 for (auto nX = -1; nX < 2 && hadMissingNeighborCell == false; nX++) {
1022 if (nZ == 0 && nX == 0) continue;
1023 auto neighbourCellId = FlowMap::toIdInt(
1024 centerPathNodeX + x + nX,
1025 centerPathNodeZ + z + nZ
1026 );
1027 auto neighbourCell = flowMap->getCell(neighbourCellId);
1028 if (neighbourCell == nullptr) {
1029 cell->setMissingNeighborCell(true);
1030 hadMissingNeighborCell = true;
1031 break;
1032 }
1033 }
1034 }
1035
1036 // determine path node index
1037 {
1038 auto i = 0;
1039 auto pathNodeIdx = -1;
1040 auto pathNodeNodeDistanceSquared = Float::MAX_VALUE;
1041 for (auto& pathNode: pathToUse) {
1042 auto pathNodeCellAxis = pathNode - cell->getPosition();
1043 auto pathNodeCandidateDistanceSquared = pathNodeCellAxis.computeLengthSquared();
1044 if (pathNodeIdx == -1 || pathNodeCandidateDistanceSquared < pathNodeNodeDistanceSquared) {
1045 pathNodeIdx = i;
1046 pathNodeNodeDistanceSquared = pathNodeCandidateDistanceSquared;
1047 }
1048 i++;
1049 }
1050 cell->setPathNodeIdx(pathNodeIdx < pathToUse.size() - 1?pathNodeIdx + 1:pathNodeIdx);
1051 }
1052 }
1053 }
1054 lastCenterPathNode = centerPathNode;
1055 }
1056
1057 // unset actor bounding volume and remove rigid body
1058 world->removeBody("tdme.pathfinding.actor");
1059 world->removeBody("tdme.pathfinding.actor.slopetest");
1060 delete actorBoundingVolume;
1061 actorBoundingVolume = nullptr;
1064
1065 // dispose custom test
1066 if (this->customTest != nullptr) {
1067 this->customTest->dispose();
1068 this->customTest = nullptr;
1069 }
1070
1071 //
1072 return flowMap;
1073}
1074
1075const vector<Vector3> PathFinding::generateDirectPath(const Vector3& start, const Vector3& end) {
1076 vector<Vector3> path;
1077 Vector3 axis;
1078 axis.set(end).sub(start);
1079 auto length = axis.computeLength();
1080 auto step = axis.clone().normalize() * stepSize;
1081 Vector3 current = start;
1082 for (auto i = stepSize; i < length; i+= stepSize) {
1083 path.push_back(current.add(step));
1084 }
1085 path.push_back(end);
1086 return path;
1087}
Transformations which contain scale, rotations and translation.
void setTranslation(const Vector3 &translation)
Set translation.
virtual void update()
Computes transformation matrix.
void addRotation(const Vector3 &axis, const float angle)
Add rotation.
Dynamic rigid/static rigid/collision body class.
Definition: Body.h:43
void fromTransformations(const Transformations &transformations)
Synchronizes this rigid body with transformations.
Definition: Body.cpp:461
Dynamic physics world class.
Definition: World.h:38
Body * getBody(const string &id)
Returns body identified by id.
Definition: World.cpp:122
Body * determineHeight(uint16_t collisionTypeId, float stepUpMax, const Vector3 &point, Vector3 &dest, float minHeight=-10000.0f, float maxHeight=10000.0f)
Determine height on x,y,u while respecting step up max.
Definition: World.cpp:300
void removeBody(const string &id)
Removes body identified by id.
Definition: World.cpp:131
bool doesCollideWith(uint16_t collisionTypeIds, Body *body, vector< Body * > &rigidBodies)
Check if world collides with given body.
Definition: World.cpp:381
Body * addCollisionBody(const string &id, bool enabled, uint16_t collisionTypeId, const Transformations &transformations, vector< BoundingVolume * > boundingVolumes)
Add a collision body.
Definition: World.cpp:99
Oriented bounding box physics primitive.
Standard math functions.
Definition: Math.h:21
3D vector 3 class
Definition: Vector3.h:22
float getY() const
Definition: Vector3.h:119
float getX() const
Definition: Vector3.h:103
float computeLength() const
Definition: Vector3.h:202
float getZ() const
Definition: Vector3.h:136
Vector3 & normalize()
Normalize the vector.
Definition: Vector3.h:288
Vector3 & set(float x, float y, float z)
Set up vector.
Definition: Vector3.h:73
float computeLengthSquared() const
Definition: Vector3.h:209
Vector3 clone() const
Clones the vector.
Definition: Vector3.h:372
Vector3 & sub(const Vector3 &v)
Subtracts a vector.
Definition: Vector3.h:325
Vector3 & add(const Vector3 &v)
Adds a vector.
Definition: Vector3.h:301
Vector3 & setY(float y)
Set Y.
Definition: Vector3.h:128
Console class.
Definition: Console.h:26
static void println()
Print new line to console.
Definition: Console.cpp:78
Float class.
Definition: Float.h:23
static constexpr float MAX_VALUE
Definition: Float.h:25
static bool isNaN(float value)
Check if float is not a number.
Definition: Float.h:62
int getIntegerPositionComponent(float value) const
Returns integer position component.
Definition: FlowMap.h:126
static string toIdInt(int x, int z)
Return string representation of given x,z integer flow map position representation for flow map cell ...
Definition: FlowMap.h:146
static float alignPositionComponent(float value, float stepSize)
Align position component.
Definition: FlowMap.h:109
Path finding class.
Definition: PathFinding.h:44
BoundingVolume * actorBoundingVolume
Definition: PathFinding.h:412
const vector< Vector3 > generateDirectPath(const Vector3 &start, const Vector3 &end)
Generate direct path from start to end.
FlowMap * createFlowMap(const vector< Vector3 > &endPositions, const Vector3 &center, float depth, float width, const uint16_t collisionTypeIds, const vector< Vector3 > &path=vector< Vector3 >(), bool complete=true, PathFindingCustomTest *customTest=nullptr)
Create flow map.
bool isWalkable(float x, float y, float z, float &height, float stepSize, float scaleActorBoundingVolumes, bool flowMapRequest, uint16_t collisionTypeIds=0, bool ignoreStepUpMax=false)
Checks if a cell is walkable.
PathFindingCustomTest * customTest
Definition: PathFinding.h:397
static string toIdInt(int x, int y, int z)
Return string representation of given x,z integer flow map position representation for path finding i...
Definition: PathFinding.h:170
unordered_map< string, float > walkableCache
Definition: PathFinding.h:414
bool isSlopeWalkableInternal(float x, float y, float z, float successorX, float successorY, float successorZ, float stepSize, float scaleActorBoundingVolumes, bool flowMapRequest, uint16_t collisionTypeIds=0)
Checks if a cell is slope walkable.
void step(const PathFindingNode &node, float stepSize, float scaleActorBoundingVolumes, const unordered_set< string > *nodesToTest, bool flowMapRequest)
Processes one step in AStar path finding.
float computeDistanceToEnd(const PathFindingNode &node)
Computes minimal non square rooted distance between node and end point.
Definition: PathFinding.h:328
bool isWalkableInternal(float x, float y, float z, float &height, float stepSize, float scaleActorBoundingVolumes, bool flowMapRequest, uint16_t collisionTypeIds=0, bool ignoreStepUpMax=false)
Checks if a cell is walkable.
Definition: PathFinding.cpp:80
float computeDistance(const PathFindingNode &a, const PathFindingNode &b)
Computes non square rooted distance between a and b.
Definition: PathFinding.h:319
static constexpr bool VERBOSE
Definition: PathFinding.h:46
static int getIntegerPositionComponent(float value, float stepSize)
Returns integer position component.
Definition: PathFinding.h:150
unordered_map< string, PathFindingNode > closedNodes
Definition: PathFinding.h:411
bool findPathCustom(const Vector3 &startPosition, const Vector3 &endPosition, float stepSize, float scaleActorBoundingVolumes, const uint16_t collisionTypeIds, vector< Vector3 > &path, int alternativeEndSteps=0, int maxTriesOverride=-1, PathFindingCustomTest *customTest=nullptr)
Finds path to given end position.
bool equals(const PathFindingNode &a, float bX, float bY, float bZ)
Returns if nodes are equals.
Definition: PathFinding.h:340
bool equalsLastNode(const PathFindingNode &a, const PathFindingNode &b)
Returns if nodes are equals for (last node test)
Definition: PathFinding.h:350
string toId(float x, float y, float z)
Return string representation of given x,y,z for path finding id.
Definition: PathFinding.h:103
unordered_map< string, PathFindingNode > openNodes
Definition: PathFinding.h:410
BoundingVolume * actorBoundingVolumeSlopeTest
Definition: PathFinding.h:413
Time utility class.
Definition: Time.h:21
static int64_t getCurrentMillis()
Retrieve current time in milliseconds.
Definition: Time.h:28
Path finding custom test interface.
virtual void initialize()=0
Initialize custom test for a sequence of isWalkable calls for the current path finding execution.
virtual void dispose()=0
Disposes custom test after a sequence of isWalkable calls for the last path finding execution.
virtual bool isWalkable(PathFinding *pathFinding, float x, float y, float z)=0
Is walkable user test.
int y
Integer position along y axis.
Definition: PathFinding.h:305
int z
Integer position along z axis.
Definition: PathFinding.h:310
int x
Integer position along x axis.
Definition: PathFinding.h:300
float costsAll
Estimated costs to the end plus calculated costs from start to this node.
Definition: PathFinding.h:285
float costsEstimated
Estimated costs to get to the end.
Definition: PathFinding.h:295
float costsReachPoint
Calculated costs up to this point (from all previous nodes up to this node) to get to this coordinate...
Definition: PathFinding.h:290