5#include <unordered_map>
17using std::unordered_map;
36 unordered_map<string, FlowMapCell>
cells;
62 inline void addCell(
const string&
id,
const Vector3& position,
bool walkable,
const Vector3& direction,
int pathIdx) {
70 inline bool hasCell(
const string&
id)
const {
71 auto cellIt =
cells.find(
id);
72 return cellIt !=
cells.end();
82 inline string toId(
float x,
float z)
const {
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));
149 result.reserve(
sizeof(x) * 2);
150 result.append((
char*)&x,
sizeof(x));
151 result.append((
char*)&z,
sizeof(z));
190 inline const vector<Vector3>&
getPath()
const {
200 auto cellIt =
cells.find(
id);
201 if (cellIt ==
cells.end())
return nullptr;
202 return &cellIt->second;
211 auto cellIt =
cells.find(
id);
212 if (cellIt ==
cells.end())
return nullptr;
213 return &cellIt->second;
227 auto cellIt =
cells.find(
id);
228 if (cellIt ==
cells.end())
return nullptr;
229 return &cellIt->second;
243 auto cellIt =
cells.find(cellId);
244 if (cellIt ==
cells.end())
return nullptr;
245 return &cellIt->second;
260 if (f00 ==
nullptr)
return Vector3();
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;
268 auto direction = top.clone().scale(1.0f - yWeight).add(bottom.clone().scale(yWeight)).normalize();
276 inline const unordered_map<string, FlowMapCell>&
getCellMap()
const {
285 auto cellIt =
cells.find(
id);
286 if (cellIt ==
cells.end())
return;
299 auto pathSize =
path.size();
300 for (
auto& pathNode: flowMap->
path) {
301 path.push_back(pathNode);
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;
311 for (
auto& cellIt: flowMap->
cells) {
312 auto cell =
getCell(cellIt.first);
313 cell->setMissingNeighborCell(
false);
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;
324 auto neighbourCell =
getCell(neighbourCellId);
325 if (neighbourCell ==
nullptr) {
326 cell->setMissingNeighborCell(
true);
327 hadMissingNeighborCell =
true;
347 auto halfSteps = steps / 2;
350 for (
auto nZ = -halfSteps; nZ < halfSteps; nZ++) {
351 for (
auto nX = -halfSteps; nX < halfSteps; nX++) {
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;
static constexpr float MAX_VALUE
const vector< Vector3 > & getEndPositions() const
Returns end positions.
static string toId(float x, float z, float stepSize)
Return string representation of given x,z for flow map cell id.
void addCell(const string &id, const Vector3 &position, bool walkable, const Vector3 &direction, int pathIdx)
Adds a cell to flow map.
const Vector3 computeDirection(float x, float z) const
Compute direction also taking neighbour cells into account.
const FlowMapCell * getCell(const string &id) const
Get cell by id.
int getIntegerPositionComponent(float value) const
Returns integer position component.
unordered_map< string, FlowMapCell > cells
FlowMap(const vector< Vector3 > &path, const vector< Vector3 > &endPositions, float stepSize, bool complete=true)
Constructor.
const unordered_map< string, FlowMapCell > & getCellMap() const
Cell map getter.
void setComplete(bool complete)
Set flow map complete flag.
const vector< Vector3 > & getPath() const
Returns path flow map is generated on.
FlowMapCell * getCell(const string &id)
Get cell by id.
FlowMapCell * getCell(float x, float z)
Get cell by position.
vector< Vector3 > endPositions
static int getIntegerPositionComponent(float value, float stepSize)
Returns integer position component.
bool hasCell(const string &id) const
Checks if a cell exists in flow map.
const FlowMapCell * getCell(float x, float z) const
Get cell by position.
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...
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.
void removeCell(const string &id)
Remove cell by id.
static string toIdInt(int x, int z)
Return string representation of given x,z integer flow map position representation for flow map cell ...
float getStepSize() const
string toId(float x, float z) const
Return string representation of given x,z for flow map cell id.
float alignPositionComponent(float value) const
Align position component.
~FlowMap()
Private destructor.
static float alignPositionComponent(float value, float stepSize)
Align position component.
Reference counter implementation to be used with inheritance.