TDME2 1.9.121
FlowMap.h
Go to the documentation of this file.
1#pragma once
2
3#include <map>
4#include <string>
5#include <unordered_map>
6#include <vector>
7
8#include <tdme/tdme.h>
9#include <tdme/math/Math.h>
10#include <tdme/math/Vector3.h>
14
15using std::string;
16using std::to_string;
17using std::unordered_map;
18using std::vector;
19
25
26/**
27 * Flow map
28 * @author Andreas Drewke
29 * @version $Id$
30 */
32friend class PathFinding;
33private:
35 float stepSize;
36 unordered_map<string, FlowMapCell> cells;
37 vector<Vector3> endPositions;
38 vector<Vector3> path;
39
40 /**
41 * Private destructor
42 */
43 inline ~FlowMap() {
44 }
45
46 /**
47 * Set flow map complete flag
48 * @param complete complete
49 */
50 inline void setComplete(bool complete) {
51 this->complete = complete;
52 }
53
54 /**
55 * Adds a cell to flow map
56 * @param id id
57 * @param position position
58 * @param walkable walkable
59 * @param direction direction
60 * @param pathIdx path index
61 */
62 inline void addCell(const string& id, const Vector3& position, bool walkable, const Vector3& direction, int pathIdx) {
63 cells[id] = FlowMapCell(position, walkable, direction, pathIdx);
64 }
65
66 /**
67 * Checks if a cell exists in flow map
68 * @param id id
69 */
70 inline bool hasCell(const string& id) const {
71 auto cellIt = cells.find(id);
72 return cellIt != cells.end();
73 }
74
75public:
76 /**
77 * Return string representation of given x,z for flow map cell id
78 * @param x x
79 * @param z z
80 * @return string representation
81 */
82 inline string toId(float x, float z) const {
83 return toId(x, z, stepSize);
84 }
85
86 /**
87 * Return string representation of given x,z for flow map cell id
88 * @param x x
89 * @param z z
90 * @param stepSize step size
91 * @return string representation
92 */
93 inline static string toId(float x, float z, float stepSize) {
94 string result;
95 int32_t value = 0;
96 result.reserve(sizeof(value) * 2);
97 value = static_cast<int>(Math::ceil(x / stepSize));
98 result.append((char*)&value, sizeof(value));
99 value = static_cast<int>(Math::ceil(z / stepSize));
100 result.append((char*)&value, sizeof(value));
101 return result;
102 }
103
104 /**
105 * Align position component
106 * @param value value which is usually a position vector 3 position component
107 * @param stepSize step size
108 */
109 inline static float alignPositionComponent(float value, float stepSize) {
110 return Math::floor(value / stepSize) * stepSize;
111 }
112
113 /**
114 * Align position component
115 * @param value value which is usually a position vector 3 position component
116 */
117 inline float alignPositionComponent(float value) const {
118 return alignPositionComponent(value, stepSize);
119 }
120
121 /**
122 * Returns integer position component
123 * @param value value
124 * @return integer position component
125 */
126 inline int getIntegerPositionComponent(float value) const {
127 return static_cast<int>(alignPositionComponent(value, stepSize) / stepSize);
128 }
129
130 /**
131 * Returns integer position component
132 * @param value value
133 * @param stepSize step size
134 * @return integer position component
135 */
136 inline static int getIntegerPositionComponent(float value, float stepSize) {
137 return static_cast<int>(alignPositionComponent(value, stepSize) / stepSize);
138 }
139
140 /**
141 * Return string representation of given x,z integer flow map position representation for flow map cell id
142 * @param x x
143 * @param z z
144 * @return string representation
145 */
146 inline static string toIdInt(int x, int z) {
147 string result;
148 int32_t value = 0;
149 result.reserve(sizeof(x) * 2);
150 result.append((char*)&x, sizeof(x));
151 result.append((char*)&z, sizeof(z));
152 return result;
153 }
154
155 /**
156 * Constructor
157 * @param path path
158 * @param endPositions end positions
159 * @param stepSize step size
160 */
161 inline FlowMap(const vector<Vector3>& path, const vector<Vector3>& endPositions, float stepSize, bool complete = true): path(path), endPositions(endPositions), stepSize(stepSize), complete(complete) {
162 }
163
164 /**
165 * @return if flow map is complete
166 */
167 inline bool isComplete() const {
168 return complete;
169 }
170
171 /**
172 * @return step size
173 */
174 inline float getStepSize() const {
175 return stepSize;
176 }
177
178 /**
179 * Returns end positions
180 * @return end positions
181 */
182 inline const vector<Vector3>& getEndPositions() const {
183 return endPositions;
184 }
185
186 /**
187 * Returns path flow map is generated on
188 * @return path
189 */
190 inline const vector<Vector3>& getPath() const {
191 return path;
192 }
193
194 /**
195 * Get cell by id
196 * @param id id
197 * @return cell
198 */
199 inline const FlowMapCell* getCell(const string& id) const {
200 auto cellIt = cells.find(id);
201 if (cellIt == cells.end()) return nullptr;
202 return &cellIt->second;
203 }
204
205 /**
206 * Get cell by id
207 * @param id id
208 * @return cell
209 */
210 inline FlowMapCell* getCell(const string& id) {
211 auto cellIt = cells.find(id);
212 if (cellIt == cells.end()) return nullptr;
213 return &cellIt->second;
214 }
215
216 /**
217 * Get cell by position
218 * @param x x
219 * @param z z
220 * @return cell
221 */
222 inline const FlowMapCell* getCell(float x, float z) const {
223 auto id = toId(
226 );
227 auto cellIt = cells.find(id);
228 if (cellIt == cells.end()) return nullptr;
229 return &cellIt->second;
230 }
231
232 /**
233 * Get cell by position
234 * @param x x
235 * @param z z
236 * @return cell
237 */
238 inline FlowMapCell* getCell(float x, float z) {
239 auto cellId = toId(
242 );
243 auto cellIt = cells.find(cellId);
244 if (cellIt == cells.end()) return nullptr;
245 return &cellIt->second;
246 }
247
248 /**
249 * Compute direction also taking neighbour cells into account
250 * @param x x
251 * @param y y
252 * @param direction direction
253 */
254 inline const Vector3 computeDirection(float x, float z) const {
255 // https://howtorts.github.io/2014/01/04/basic-flow-fields.html
256 auto cellCount = 0;
257 auto xInt = getIntegerPositionComponent(x);
258 auto zInt = getIntegerPositionComponent(z);
259 auto f00 = getCell(toIdInt(xInt, zInt));
260 if (f00 == nullptr) return Vector3();
261 auto f01 = getCell(toIdInt(xInt, zInt + 1));
262 auto f10 = getCell(toIdInt(xInt + 1, zInt));
263 auto f11 = getCell(toIdInt(xInt + 1, zInt + 1));
264 auto xWeight = x - xInt * stepSize;
265 auto top = f10 != nullptr?f00->getDirection().clone().scale(1.0f - xWeight).add(f10->getDirection().clone().scale(xWeight)):f00->getDirection();
266 auto bottom = f01 != nullptr && f11 != nullptr?f01->getDirection().clone().scale(1.0f - xWeight).add(f11->getDirection().clone().scale(xWeight)):top;
267 auto yWeight = z - zInt * stepSize;
268 auto direction = top.clone().scale(1.0f - yWeight).add(bottom.clone().scale(yWeight)).normalize();
269 return direction;
270 }
271
272 /**
273 * Cell map getter
274 * @returns cell map
275 */
276 inline const unordered_map<string, FlowMapCell>& getCellMap() const {
277 return cells;
278 }
279
280 /**
281 * Remove cell by id
282 * @param id id
283 */
284 inline void removeCell(const string& id) {
285 auto cellIt = cells.find(id);
286 if (cellIt == cells.end()) return;
287 cells.erase(cellIt);
288 }
289
290 /**
291 * Merge given flow map into this flow map, please note that given flow map step size needs to match this flow maps step size
292 * This only applies to a series of flow maps created sequentially and in correct order along a path
293 * @param flowMap flow map
294 */
295 inline void merge(const FlowMap* flowMap) {
296 // complete
297 complete = flowMap->complete;
298 // add path
299 auto pathSize = path.size();
300 for (auto& pathNode: flowMap->path) {
301 path.push_back(pathNode);
302 }
303 // add cells
304 // TODO: check again cell misssing neighbour cells
305 for (auto& cellIt: flowMap->cells) {
306 auto cellExists = cells.find(cellIt.first) != cells.end();
307 cells[cellIt.first] = cellIt.second;
308 cells[cellIt.first].pathNodeIdx+= pathSize;
309 }
310 // check if we have missing neighbour cells
311 for (auto& cellIt: flowMap->cells) {
312 auto cell = getCell(cellIt.first);
313 cell->setMissingNeighborCell(false);
314 auto cellX = getIntegerPositionComponent(cell->position.getX());
315 auto cellZ = getIntegerPositionComponent(cell->position.getZ());
316 auto hadMissingNeighborCell = false;
317 for (auto nZ = -1; nZ < 2 && hadMissingNeighborCell == false; nZ++) {
318 for (auto nX = -1; nX < 2 && hadMissingNeighborCell == false; nX++) {
319 if (nZ == 0 && nX == 0) continue;
320 auto neighbourCellId = FlowMap::toIdInt(
321 cellX + nX,
322 cellZ + nZ
323 );
324 auto neighbourCell = getCell(neighbourCellId);
325 if (neighbourCell == nullptr) {
326 cell->setMissingNeighborCell(true);
327 hadMissingNeighborCell = true;
328 break;
329 }
330 }
331 }
332 }
333 // end positions
334 endPositions = flowMap->endPositions;
335 }
336
337 /**
338 * Find nearest cell, which can be used if outside of flow map to find back in
339 * @param x x
340 * @param z z
341 * @param steps steps
342 */
343 inline FlowMapCell* findNearestCell(float x, float z, int steps = 8) {
344 Vector3 position = Vector3(x, 0.0f, z);
345 auto cellX = getIntegerPositionComponent(x);
346 auto cellZ = getIntegerPositionComponent(z);
347 auto halfSteps = steps / 2;
348 FlowMapCell* cellBestFit = nullptr;
349 float cellBestFitDistanceSquared = Float::MAX_VALUE;
350 for (auto nZ = -halfSteps; nZ < halfSteps; nZ++) {
351 for (auto nX = -halfSteps; nX < halfSteps; nX++) {
352 auto cellId = FlowMap::toIdInt(
353 cellX + nX,
354 cellZ + nZ
355 );
356 auto cellCandidate = getCell(cellId);
357 if (cellCandidate == nullptr) continue;
358 auto cellCandidateDistanceSquared = cellCandidate->getPosition().clone().sub(position).computeLengthSquared();
359 if (cellBestFit == nullptr || cellCandidateDistanceSquared < cellBestFitDistanceSquared) {
360 cellBestFit = cellCandidate;
361 cellBestFitDistanceSquared = cellCandidateDistanceSquared;
362 }
363 }
364 }
365 return cellBestFit;
366 }
367
368};
Standard math functions.
Definition: Math.h:21
3D vector 3 class
Definition: Vector3.h:22
Console class.
Definition: Console.h:26
static constexpr float MAX_VALUE
Definition: Float.h:25
const vector< Vector3 > & getEndPositions() const
Returns end positions.
Definition: FlowMap.h:182
static string toId(float x, float z, float stepSize)
Return string representation of given x,z for flow map cell id.
Definition: FlowMap.h:93
void addCell(const string &id, const Vector3 &position, bool walkable, const Vector3 &direction, int pathIdx)
Adds a cell to flow map.
Definition: FlowMap.h:62
const Vector3 computeDirection(float x, float z) const
Compute direction also taking neighbour cells into account.
Definition: FlowMap.h:254
bool isComplete() const
Definition: FlowMap.h:167
const FlowMapCell * getCell(const string &id) const
Get cell by id.
Definition: FlowMap.h:199
int getIntegerPositionComponent(float value) const
Returns integer position component.
Definition: FlowMap.h:126
unordered_map< string, FlowMapCell > cells
Definition: FlowMap.h:36
FlowMap(const vector< Vector3 > &path, const vector< Vector3 > &endPositions, float stepSize, bool complete=true)
Constructor.
Definition: FlowMap.h:161
const unordered_map< string, FlowMapCell > & getCellMap() const
Cell map getter.
Definition: FlowMap.h:276
void setComplete(bool complete)
Set flow map complete flag.
Definition: FlowMap.h:50
const vector< Vector3 > & getPath() const
Returns path flow map is generated on.
Definition: FlowMap.h:190
FlowMapCell * getCell(const string &id)
Get cell by id.
Definition: FlowMap.h:210
FlowMapCell * getCell(float x, float z)
Get cell by position.
Definition: FlowMap.h:238
vector< Vector3 > endPositions
Definition: FlowMap.h:37
static int getIntegerPositionComponent(float value, float stepSize)
Returns integer position component.
Definition: FlowMap.h:136
bool hasCell(const string &id) const
Checks if a cell exists in flow map.
Definition: FlowMap.h:70
const FlowMapCell * getCell(float x, float z) const
Get cell by position.
Definition: FlowMap.h:222
void merge(const FlowMap *flowMap)
Merge given flow map into this flow map, please note that given flow map step size needs to match thi...
Definition: FlowMap.h:295
FlowMapCell * findNearestCell(float x, float z, int steps=8)
Find nearest cell, which can be used if outside of flow map to find back in.
Definition: FlowMap.h:343
void removeCell(const string &id)
Remove cell by id.
Definition: FlowMap.h:284
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
float getStepSize() const
Definition: FlowMap.h:174
string toId(float x, float z) const
Return string representation of given x,z for flow map cell id.
Definition: FlowMap.h:82
vector< Vector3 > path
Definition: FlowMap.h:38
float alignPositionComponent(float value) const
Align position component.
Definition: FlowMap.h:117
~FlowMap()
Private destructor.
Definition: FlowMap.h:43
static float alignPositionComponent(float value, float stepSize)
Align position component.
Definition: FlowMap.h:109
Path finding class.
Definition: PathFinding.h:44
Reference counter implementation to be used with inheritance.
Definition: Reference.h:10