7#include <unordered_map>
8#include <unordered_set>
29using std::unordered_map;
30using std::unordered_set;
47PathFinding::PathFinding(
55 uint16_t skipOnCollisionTypeIds,
57 float flowMapStepSize,
58 float flowMapScaleActorBoundingVolumes
80bool PathFinding::isWalkableInternal(
float x,
float y,
float z,
float& height,
float stepSize,
float scaleActorBoundingVolumes,
bool flowMapRequest, uint16_t collisionTypeIds,
bool ignoreStepUpMax) {
82 if (flowMapRequest ==
true) {
84 flowMapRequest ==
true?
86 to_string(
static_cast<int>(
stepSize * 10.0f)) +
";" +
87 to_string(
static_cast<int>(scaleActorBoundingVolumes * 10.0f)) +
";" +
90 to_string(ignoreStepUpMax):
94 height = walkableCacheIt->second;
95 return height > -10000.0f;
99 if (flowMapRequest ==
true)
walkableCache[cacheId] = walkable ==
false?-10000.0f:height;
100 if (walkable ==
false)
return false;
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;
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);
113 slopeAngle = Vector3::computeAngle(
120 if (flowMapRequest ==
true) {
123 to_string(
static_cast<int>(
stepSize * 10.0f)) +
";" +
124 to_string(
static_cast<int>(scaleActorBoundingVolumes * 10.0f)) +
";" +
127 to_string(slopeAngle);
130 auto height = walkableCacheIt->second;
139 slopeTestTransformations.
update();
142 auto actorSlopeTestCollisionBody =
world->
getBody(
"tdme.pathfinding.actor.slopetest");
146 vector<Body*> collidedRigidBodies;
148 if (flowMapRequest ==
true)
walkableCache[cacheId] = walkable ==
false?-10000.0f:0.0f;
152bool PathFinding::isWalkable(
float x,
float y,
float z,
float& height,
float stepSize,
float scaleActorBoundingVolumes,
bool flowMapRequest, uint16_t collisionTypeIds,
bool ignoreStepUpMax) {
157 for (
auto i = 0; i <= 4; i++) {
159 for (
auto j = 0; j <= 4; j++) {
160 Vector3 actorPositionCandidate;
164 actorPositionCandidate.
set(_x, y, _z),
172 if (actorPosition.
getY() > height) height = actorPosition.
getY();
181 actorTransformations.
update();
184 auto actorCollisionBody =
world->
getBody(
"tdme.pathfinding.actor");
188 vector<Body*> collidedRigidBodies;
193 auto nodeId = node.
id;
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) &&
201 (Math::abs(x) == 1 && Math::abs(z) == 1) ==
false)) {
204 if (nodesToTestPtr !=
nullptr) {
205 auto successorNodeId =
toIdInt(
210 if (nodesToTestPtr->find(successorNodeId) == nodesToTestPtr->end())
continue;
218 if (
equals(node, successorX, yHeight, successorZ)) {
227 successorNode.
x = node.
x + x;
229 successorNode.
z = node.
z + z;
235 successorNodes.push(successorNode);
240 while (successorNodes.empty() ==
false) {
241 auto& successorNode = successorNodes.top();
248 auto openListNodeIt =
openNodes.find(successorNode.id);
250 openListNode = &openListNodeIt->second;
254 if (openListNode !=
nullptr) {
258 successorNodes.pop();
266 auto closedListNodeIt =
closedNodes.find(successorNode.id);
268 closedListNode = &closedListNodeIt->second;
272 if (closedListNode !=
nullptr) {
276 successorNodes.pop();
284 successorNode.costsReachPoint = successorCostsReachPoint;
286 successorNode.costsAll = successorNode.costsReachPoint + successorNode.costsEstimated;
289 if (openListNode !=
nullptr) {
295 if (closedListNode !=
nullptr) {
300 openNodes[successorNode.id] = successorNode;
303 successorNodes.pop();
317 float scaleActorBoundingVolumes,
318 const uint16_t collisionTypeIds,
319 vector<Vector3>& path,
320 int alternativeEndSteps,
321 int maxTriesOverride,
330 auto currentMaxTries = maxTriesOverride == -1?this->
maxTries:maxTriesOverride;
336 if (this->customTest !=
nullptr) this->customTest->
initialize();
344 OrientedBoundingBox::AABB_AXIS_X,
345 OrientedBoundingBox::AABB_AXIS_Y,
346 OrientedBoundingBox::AABB_AXIS_Z,
352 actorTransformations.
update();
358 OrientedBoundingBox::AABB_AXIS_X,
359 OrientedBoundingBox::AABB_AXIS_Y,
360 OrientedBoundingBox::AABB_AXIS_Z,
366 bool success =
false;
371 path.push_back(startPosition);
372 float endYHeight = endPosition.
getY();
379 scaleActorBoundingVolumes,
382 path.push_back(endPosition);
388 if (
VERBOSE ==
true)
Console::println(
"PathFinding::findPath(): end - start position < stepSizeLast! Exiting!");
389 path.push_back(startPosition);
390 float endYHeight = endPosition.
getY();
397 scaleActorBoundingVolumes,
400 path.push_back(endPosition);
407 auto endPositionIdx = 0;
410 if (success ==
false) {
412 auto forcedAlternativeEndSteps = 0;
415 vector<Vector3> startPositionCandidates;
416 vector<Vector3> endPositionCandidates;
419 auto sideVector = Vector3::computeCrossProduct(forwardVector,
Vector3(0.0f, 1.0f, 0.0f)).
setY(0.0f).
normalize();
424 endPositionCandidates.push_back(endPosition);
425 startPositionCandidates.push_back(startPosition);
429 auto forwardDistance = 0.0f;
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;
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));
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));
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));
464 if (currentMaxTries == 0) {
465 forcedAlternativeEndSteps = 27 * 2;
466 auto forwardDistance = 0.0f;
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;
503 Vector3 startPositionComputed = startPosition;
504 for (
auto& startPositionCandidate: startPositionCandidates) {
505 float startYHeight = startPositionCandidate.
getY();
507 startPositionCandidate.getX(),
508 startPositionCandidate.getY(),
509 startPositionCandidate.getZ(),
512 scaleActorBoundingVolumes,
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)
527 startPositionComputed = startPositionCandidate;
528 startPositionComputed.
setY(startYHeight);
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(),
543 scaleActorBoundingVolumes,
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)
558 endPositionComputed.
setY(endYHeight);
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())
577 start.
position = startPositionComputed;
616 for (
auto stepIdx = 0; done ==
false && stepIdx <
stepsMax; stepIdx++) {
628 if (nodePtr ==
nullptr || nodeIt->second.
costsAll < nodePtr->
costsAll) nodePtr = &nodeIt->second;
632 if (nodePtr ==
nullptr) {
638 if (endNodePtr !=
nullptr) {
639 const auto& node = *endNodePtr;
643 unordered_map<string, PathFindingNode>::iterator nodeIt;
658 reverse(path.begin(), path.end());
659 if (path.size() > 1) path.erase(path.begin());
660 if (path.size() == 0) {
662 path.push_back(endPositionComputed);
668 const auto& node = *nodePtr;
670 step(node,
stepSize, scaleActorBoundingVolumes,
nullptr,
false);
682 if (success ==
true || tries >= currentMaxTries + forcedAlternativeEndSteps)
break;
687 Console::println(
"PathFinding::findPath(): end position were not walkable!");
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()));
710 if (this->customTest !=
nullptr) {
712 this->customTest =
nullptr;
721 if (path.empty() ==
true) {
730 if (this->customTest !=
nullptr) this->customTest->
initialize();
738 OrientedBoundingBox::AABB_AXIS_X,
739 OrientedBoundingBox::AABB_AXIS_Y,
740 OrientedBoundingBox::AABB_AXIS_Z,
746 actorTransformations.
update();
752 OrientedBoundingBox::AABB_AXIS_X,
753 OrientedBoundingBox::AABB_AXIS_Y,
754 OrientedBoundingBox::AABB_AXIS_Z,
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;
768 if (endPositions.size() == 0) {
787 for (
auto& endPosition: endPositions) {
812 unordered_set<string> nodesToTest;
813 for (
auto& _centerPathNode: pathToUse) {
816 _centerPathNode.getY(),
821 for (
auto z = zMin; z <= zMax; z++) {
822 for (
auto x = xMin; x <= xMax; x++) {
828 nodesToTest.insert(nodeId);
838 if (nodePtr ==
nullptr || nodeIt->second.
costsAll < nodePtr->
costsAll) nodePtr = &nodeIt->second;
840 if (nodePtr ==
nullptr)
break;
841 const auto& node = *nodePtr;
853 flowMap->acquireReference();
854 for (
auto& _centerPathNode: pathToUse) {
857 _centerPathNode.getY(),
862 for (
auto z = zMin; z <= zMax; z++) {
863 for (
auto x = xMin; x <= xMax; x++) {
880 if (flowMap->hasCell(cellId) ==
true)
continue;
887 auto& node = nodeIt->second;
889 cellPosition.setY(node.position.getY());
894 for (
auto _z = -1; _z <= 1; _z++)
895 for (
auto _x = -1; _x <= 1; _x++)
896 if (_z != 0 || _x != 0) {
898 auto neighbourNodeId =
toIdInt(
899 centerPathNodeX + x + _x,
901 centerPathNodeZ + z + _z
904 if (neighbourNodeId == nodeId)
continue;
906 auto neighbourNodeIt =
closedNodes.find(neighbourNodeId);
912 auto& neighbourNode = neighbourNodeIt->second;
913 if (minCostsNode ==
nullptr || neighbourNode.
costsReachPoint < minCosts) {
914 minCostsNode = &neighbourNode;
919 if (minCostsNode !=
nullptr) {
923 minCostsNode->
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()) +
"; " +
955 for (
auto& _centerPathNode: pathToUse) {
958 _centerPathNode.getY(),
963 for (
auto z = zMin; z <= zMax; z++) {
964 for (
auto x = xMin; x <= xMax; x++) {
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());
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());
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());
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());
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(
1003 _centerPathNode.getY(),
1006 auto pathDirection = (lastCenterPathNode - centerPathNode).setY(0.0f).normalize();
1009 for (
auto z = zMin; z <= zMax; z++) {
1010 for (
auto x = xMin; x <= xMax; x++) {
1012 centerPathNodeX + x,
1015 auto cell = flowMap->getCell(cellId);
1016 if (cell ==
nullptr)
continue;
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;
1024 centerPathNodeX + x + nX,
1025 centerPathNodeZ + z + nZ
1027 auto neighbourCell = flowMap->getCell(neighbourCellId);
1028 if (neighbourCell ==
nullptr) {
1029 cell->setMissingNeighborCell(
true);
1030 hadMissingNeighborCell =
true;
1039 auto pathNodeIdx = -1;
1041 for (
auto& pathNode: pathToUse) {
1042 auto pathNodeCellAxis = pathNode - cell->getPosition();
1043 auto pathNodeCandidateDistanceSquared = pathNodeCellAxis.computeLengthSquared();
1044 if (pathNodeIdx == -1 || pathNodeCandidateDistanceSquared < pathNodeNodeDistanceSquared) {
1046 pathNodeNodeDistanceSquared = pathNodeCandidateDistanceSquared;
1050 cell->setPathNodeIdx(pathNodeIdx < pathToUse.size() - 1?pathNodeIdx + 1:pathNodeIdx);
1054 lastCenterPathNode = centerPathNode;
1066 if (this->customTest !=
nullptr) {
1068 this->customTest =
nullptr;
1076 vector<Vector3> path;
1083 path.push_back(current.
add(
step));
1085 path.push_back(
end);
Dynamic rigid/static rigid/collision body class.
void fromTransformations(const Transformations &transformations)
Synchronizes this rigid body with transformations.
Dynamic physics world class.
Body * getBody(const string &id)
Returns body identified by id.
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.
void removeBody(const string &id)
Removes body identified by id.
bool doesCollideWith(uint16_t collisionTypeIds, Body *body, vector< Body * > &rigidBodies)
Check if world collides with given body.
Body * addCollisionBody(const string &id, bool enabled, uint16_t collisionTypeId, const Transformations &transformations, vector< BoundingVolume * > boundingVolumes)
Add a collision body.
Bounding volume interface.
Oriented bounding box physics primitive.
float computeLength() const
Vector3 & normalize()
Normalize the vector.
Vector3 & set(float x, float y, float z)
Set up vector.
float computeLengthSquared() const
Vector3 clone() const
Clones the vector.
Vector3 & sub(const Vector3 &v)
Subtracts a 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 bool isNaN(float value)
Check if float is not a number.
int getIntegerPositionComponent(float value) const
Returns integer position component.
static string toIdInt(int x, int z)
Return string representation of given x,z integer flow map position representation for flow map cell ...
static float alignPositionComponent(float value, float stepSize)
Align position component.
BoundingVolume * actorBoundingVolume
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 ¢er, 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
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...
uint16_t skipOnCollisionTypeIds
unordered_map< string, float > walkableCache
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.
~PathFinding()
Destructor.
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.
float computeDistance(const PathFindingNode &a, const PathFindingNode &b)
Computes non square rooted distance between a and b.
uint16_t collisionTypeIds
static constexpr bool VERBOSE
static int getIntegerPositionComponent(float value, float stepSize)
Returns integer position component.
unordered_map< string, PathFindingNode > closedNodes
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.
float flowMapScaleActorBoundingVolumes
bool equalsLastNode(const PathFindingNode &a, const PathFindingNode &b)
Returns if nodes are equals for (last node test)
string toId(float x, float y, float z)
Return string representation of given x,y,z for path finding id.
unordered_map< string, PathFindingNode > openNodes
BoundingVolume * actorBoundingVolumeSlopeTest
static int64_t getCurrentMillis()
Retrieve current time in milliseconds.
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.
int z
Integer position along z axis.
int x
Integer position along x axis.
float costsAll
Estimated costs to the end plus calculated costs from start to this node.
float costsEstimated
Estimated costs to get to the end.
string previousNodeId
Previous node.
float costsReachPoint
Calculated costs up to this point (from all previous nodes up to this node) to get to this coordinate...
Vector3 position
Position.