TDME2 1.9.121
PathFinding.h
Go to the documentation of this file.
1#pragma once
2
3#include <stack>
4#include <string>
5#include <unordered_map>
6#include <unordered_set>
7#include <vector>
8
9#include <tdme/tdme.h>
13#include <tdme/math/Math.h>
14#include <tdme/math/Vector3.h>
19
20using std::map;
21using std::stack;
22using std::string;
23using std::to_string;
24using std::unordered_map;
25using std::unordered_set;
26using std::vector;
27
37
38/**
39 * Path finding class
40 * @author Andreas Drewke
41 * @version $Id$
42 */
44{
45public:
46 static constexpr bool VERBOSE { true };
47
48 /**
49 * Public constructor
50 * @param world world
51 * @param sloping sloping
52 * @param stepsMax steps max
53 * @param actorHeight actor height
54 * @param stepSize step size
55 * @param stepSizeLast step size last
56 * @param actorStepUpMax actor step up max
57 * @param skipOnCollisionTypeIds skip cells with given collision type ids
58 * @param maxTries max tries
59 * @param flowMapStepSize flow map step size
60 * @param flowMapScaleActorBoundingVolumes flow map scale actor bounding volumes
61 */
63 World* world,
64 bool sloping = false,
65 int stepsMax = 1000,
66 float actorHeight = 2.0f,
67 float stepSize = 0.5f,
68 float stepSizeLast = 0.75f,
69 float actorStepUpMax = 0.25f,
70 uint16_t skipOnCollisionTypeIds = 0,
71 int maxTries = 5,
72 float flowMapStepSize = 0.5f,
74 );
75
76 /**
77 * Destructor
78 */
80
81 /**
82 * @return step size
83 */
84 inline float getStepSize() {
85 return stepSize;
86 }
87
88 /**
89 * Clear caches
90 */
91 inline void reset() {
92 walkableCache.clear();
93 }
94
95 /**
96 * Return string representation of given x,y,z for path finding id
97 * @param x x
98 * @param y y
99 * @param z z
100 * @param stepSize step size
101 * @return string representation
102 */
103 inline string toId(float x, float y, float z) {
104 return toId(x, y, z, stepSize);
105 }
106
107 /**
108 * Return string representation of given x,y,z for path finding id
109 * @param x x
110 * @param y y
111 * @param z z
112 * @return string representation
113 */
114 inline static string toId(float x, float y, float z, float stepSize) {
115 string result;
116 int32_t value = 0;
117 result.reserve(sizeof(value) * 3);
118 value = static_cast<int>(Math::ceil(x / stepSize));
119 result.append((char*)&value, sizeof(value));
120 value = static_cast<int>(Math::ceil(y / 0.1f));
121 result.append((char*)&value, sizeof(value));
122 value = static_cast<int>(Math::ceil(z / stepSize));
123 result.append((char*)&value, sizeof(value));
124 return result;
125 }
126
127 /**
128 * Align position component
129 * @param value value which is usually a position vector 3 position component
130 * @param stepSize step size
131 */
132 inline static float alignPositionComponent(float value, float stepSize) {
133 return Math::floor(value / stepSize) * stepSize;
134 }
135
136 /**
137 * Align position component
138 * @param value value which is usually a position vector 3 position component
139 */
140 inline float alignPositionComponent(float value) {
141 return alignPositionComponent(value, stepSize);
142 }
143
144 /**
145 * Returns integer position component
146 * @param value value
147 * @param stepSize step size
148 * @return integer position component
149 */
150 inline static int getIntegerPositionComponent(float value, float stepSize) {
151 return static_cast<int>(alignPositionComponent(value, stepSize) / stepSize);
152 }
153
154 /**
155 * Returns integer position component
156 * @param value value
157 * @return integer position component
158 */
159 inline int getIntegerPositionComponent(float value) {
161 }
162
163 /**
164 * Return string representation of given x,z integer flow map position representation for path finding id
165 * @param x x
166 * @param y y
167 * @param z z
168 * @return string representation
169 */
170 inline static string toIdInt(int x, int y, int z) {
171 string result;
172 result.reserve(sizeof(x) * 3);
173 result.append((char*)&x, sizeof(x));
174 result.append((char*)&y, sizeof(y));
175 result.append((char*)&z, sizeof(z));
176 return result;
177 }
178
179 /**
180 * Finds path to given end position
181 * @param startPosition start position
182 * @param endPosition end position
183 * @param collisionTypeIds collision type ids
184 * @param path path from actor to target
185 * @param alternativeEndSteps alternative end steps
186 * @param maxTriesOverride max tries override or -1 for default
187 * @param customTest custom test
188 * @return success
189 */
190 inline bool findPath(const Vector3& startPosition, const Vector3& endPosition, const uint16_t collisionTypeIds, vector<Vector3>& path, int alternativeEndSteps = 0, int maxTriesOverride = -1, PathFindingCustomTest* customTest = nullptr) {
191 return findPathCustom(startPosition, endPosition, stepSize, 1.0f, collisionTypeIds, path, alternativeEndSteps, maxTriesOverride, customTest);
192 }
193
194
195 /**
196 * Finds path to given end position for flow maps
197 * @param startPosition start position
198 * @param endPosition end position
199 * @param collisionTypeIds collision type ids
200 * @param path path from actor to target
201 * @param alternativeEndSteps alternative end steps
202 * @param maxTriesOverride max tries override or -1 for default
203 * @param customTest custom test
204 * @return success
205 */
206 inline bool findFlowMapPath(const Vector3& startPosition, const Vector3& endPosition, const uint16_t collisionTypeIds, vector<Vector3>& path, int alternativeEndSteps = 0, int maxTriesOverride = -1, PathFindingCustomTest* customTest = nullptr) {
207 return findPathCustom(startPosition, endPosition, flowMapStepSize, flowMapScaleActorBoundingVolumes, collisionTypeIds, path, alternativeEndSteps, maxTriesOverride, customTest);
208 }
209
210 /**
211 * Finds path to given end position
212 * @param startPosition start position
213 * @param endPosition end position
214 * @param stepSize step size
215 * @param scaleActorBoundingVolumes scale actor bounding volumes
216 * @param collisionTypeIds collision type ids
217 * @param path path from actor to target
218 * @param alternativeEndSteps alternative end steps
219 * @param maxTriesOverride max tries override or -1 for default
220 * @param customTest custom test
221 * @return success
222 */
223 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);
224
225 /**
226 * Checks if a cell is walkable
227 * @param x x
228 * @param y y
229 * @param z z
230 * @param height y stepped up
231 * @param stepSize step size
232 * @param scaleActorBoundingVolumes scale actor bounding volumes
233 * @param flowMapRequest flow map request
234 * @param collisionTypeIds collision type ids or 0 for default
235 * @param ignoreStepUpMax ignore step up max
236 * @return if cell is walkable
237 */
238 bool isWalkable(float x, float y, float z, float& height, float stepSize, float scaleActorBoundingVolumes, bool flowMapRequest, uint16_t collisionTypeIds = 0, bool ignoreStepUpMax = false);
239
240 /**
241 * Create flow map
242 * @param endPositions end positions
243 * @param center flow map center
244 * @param depth flow map depth
245 * @param width flow map width
246 * @param collisionTypeIds collision type ids
247 * @param path path to test along
248 * @param complete complete
249 * @param customTest custom test
250 * @return flow map
251 */
252 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);
253
254 /**
255 * Generate direct path from start to end
256 * @param start start
257 * @param end end
258 * @return path nodes
259 */
260 const vector<Vector3> generateDirectPath(const Vector3& start, const Vector3& end);
261private:
262 /**
263 * Path finding node
264 */
265 struct PathFindingNode final
266 {
267 /**
268 * Node id
269 */
270 string id;
271
272 /**
273 * Previous node
274 */
276
277 /**
278 * Position
279 */
281
282 /**
283 * Estimated costs to the end plus calculated costs from start to this node
284 */
285 float costsAll;
286
287 /**
288 * Calculated costs up to this point (from all previous nodes up to this node) to get to this coordinates from start
289 */
291
292 /**
293 * Estimated costs to get to the end
294 */
296
297 /**
298 * Integer position along x axis
299 */
300 int x;
301
302 /**
303 * Integer position along y axis
304 */
305 int y;
306
307 /**
308 * Integer position along z axis
309 */
310 int z;
311 };
312
313 /**
314 * Computes non square rooted distance between a and b
315 * @param a node a
316 * @param b node b
317 * @return non square rooted distance
318 */
319 inline float computeDistance(const PathFindingNode& a, const PathFindingNode& b) {
321 }
322
323 /**
324 * Computes minimal non square rooted distance between node and end point
325 * @param node node
326 * @return non square rooted distance
327 */
328 inline float computeDistanceToEnd(const PathFindingNode& node) {
330 }
331
332 /**
333 * Returns if nodes are equals
334 * @param a a
335 * @param bX b x coordinate
336 * @param bY b y coordinate
337 * @param bZ b z coordinate
338 * @return if node a == node b
339 */
340 inline bool equals(const PathFindingNode& a, float bX, float bY, float bZ) {
341 return a.position.clone().sub(Vector3(bX, bY, bZ)).computeLengthSquared() < Math::square(0.1f);
342 }
343
344 /**
345 * Returns if nodes are equals for (last node test)
346 * @param a a
347 * @param lastNode b
348 * @return if node a == node b
349 */
350 inline bool equalsLastNode(const PathFindingNode& a, const PathFindingNode& b) {
352 }
353
354 /**
355 * Checks if a cell is walkable
356 * @param x x
357 * @param y y
358 * @param z z
359 * @param height y stepped up
360 * @param stepSize step size
361 * @param scaleActorBoundingVolumes scale actor bounding volumes
362 * @param flowMapRequest flow map request
363 * @param collisionTypeIds collision type ids or 0 for default
364 * @param ignoreStepUpMax ignore step up max
365 * @return if cell is walkable
366 */
367 bool isWalkableInternal(float x, float y, float z, float& height, float stepSize, float scaleActorBoundingVolumes, bool flowMapRequest, uint16_t collisionTypeIds = 0, bool ignoreStepUpMax = false);
368
369 /**
370 * Checks if a cell is slope walkable
371 * @param x x
372 * @param y y
373 * @param z z
374 * @param successorX x
375 * @param successorY y
376 * @param successorZ z
377 * @param stepSize step size
378 * @param scaleActorBoundingVolumes scale actor bounding volumes
379 * @param collisionTypeIds collision type ids or 0 for default
380 * @return if cell is walkable
381 */
382 bool isSlopeWalkableInternal(float x, float y, float z, float successorX, float successorY, float successorZ, float stepSize, float scaleActorBoundingVolumes, bool flowMapRequest, uint16_t collisionTypeIds = 0);
383
384 /**
385 * Processes one step in AStar path finding
386 * @param node node
387 * @param stepSize step size
388 * @param scaleActorBoundingVolumes scale actor bounding volumes
389 * @param nodesToTest nodes to test or nullptr, applies to flow cost map generation
390 * @param flowMapRequest flow map request
391 * @return step status
392 */
393 void step(const PathFindingNode& node, float stepSize, float scaleActorBoundingVolumes, const unordered_set<string>* nodesToTest, bool flowMapRequest);
394
395 // properties
396 World* world { nullptr };
401 float stepSize;
410 unordered_map<string, PathFindingNode> openNodes;
411 unordered_map<string, PathFindingNode> closedNodes;
414 unordered_map<string, float> walkableCache;
415};
Transformations which contain scale, rotations and translation.
Dynamic physics world class.
Definition: World.h:38
Standard math functions.
Definition: Math.h:21
3D vector 3 class
Definition: Vector3.h:22
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 & setY(float y)
Set Y.
Definition: Vector3.h:128
Console class.
Definition: Console.h:26
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
bool findFlowMapPath(const Vector3 &startPosition, const Vector3 &endPosition, const uint16_t collisionTypeIds, vector< Vector3 > &path, int alternativeEndSteps=0, int maxTriesOverride=-1, PathFindingCustomTest *customTest=nullptr)
Finds path to given end position for flow maps.
Definition: PathFinding.h:206
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
PathFinding(World *world, bool sloping=false, int stepsMax=1000, float actorHeight=2.0f, float stepSize=0.5f, float stepSizeLast=0.75f, float actorStepUpMax=0.25f, uint16_t skipOnCollisionTypeIds=0, int maxTries=5, float flowMapStepSize=0.5f, float flowMapScaleActorBoundingVolumes=1.0f)
Public constructor.
Definition: PathFinding.cpp:47
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
bool findPath(const Vector3 &startPosition, const Vector3 &endPosition, const uint16_t collisionTypeIds, vector< Vector3 > &path, int alternativeEndSteps=0, int maxTriesOverride=-1, PathFindingCustomTest *customTest=nullptr)
Finds path to given end position.
Definition: PathFinding.h:190
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
void reset()
Clear caches.
Definition: PathFinding.h:91
string toId(float x, float y, float z)
Return string representation of given x,y,z for path finding id.
Definition: PathFinding.h:103
static string toId(float x, float y, float z, float stepSize)
Return string representation of given x,y,z for path finding id.
Definition: PathFinding.h:114
unordered_map< string, PathFindingNode > openNodes
Definition: PathFinding.h:410
float alignPositionComponent(float value)
Align position component.
Definition: PathFinding.h:140
int getIntegerPositionComponent(float value)
Returns integer position component.
Definition: PathFinding.h:159
BoundingVolume * actorBoundingVolumeSlopeTest
Definition: PathFinding.h:413
static float alignPositionComponent(float value, float stepSize)
Align position component.
Definition: PathFinding.h:132
Path finding custom test interface.
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