Extracted Pathfinder class from World.

This commit is contained in:
Felix Ableitner 2013-04-28 18:11:39 +02:00
parent 51fdaebd0c
commit 5be4a65653
14 changed files with 337 additions and 298 deletions

View file

@ -27,9 +27,9 @@ Game::Game(sf::RenderWindow& window) :
mWindow.setKeyRepeatEnabled(true); mWindow.setKeyRepeatEnabled(true);
Generator generator; Generator generator;
generator.generateTiles(mTileManager, mWorld, generator.generateTiles(mTileManager, mPathfinder,
sf::IntRect(-32, -32, 64, 64)); sf::IntRect(-32, -32, 64, 64));
mPlayer = std::shared_ptr<Player>(new Player(mWorld, mTileManager, mPlayer = std::shared_ptr<Player>(new Player(mWorld, mTileManager, mPathfinder,
sf::Vector2f(0.0f, 0.0f), Yaml("player.yaml"))); sf::Vector2f(0.0f, 0.0f), Yaml("player.yaml")));
mWorld.insertCharacter(mPlayer); mWorld.insertCharacter(mPlayer);
} }

View file

@ -9,9 +9,11 @@
#define DG_GAME_H_ #define DG_GAME_H_
#include "sprites/TileManager.h" #include "sprites/TileManager.h"
#include "Pathfinder.h"
#include "World.h" #include "World.h"
class TileManager; class TileManager;
class Pathfinder;
class Player; class Player;
class World; class World;
@ -46,6 +48,7 @@ private:
World mWorld; World mWorld;
TileManager mTileManager; TileManager mTileManager;
Pathfinder mPathfinder;
std::shared_ptr<Player> mPlayer; std::shared_ptr<Player> mPlayer;
bool mQuit; bool mQuit;

242
source/Pathfinder.cpp Normal file
View file

@ -0,0 +1,242 @@
/*
* Pathfinder.cpp
*
* Created on: 28.04.2013
* Author: Felix
*/
#include "Pathfinder.h"
#include <algorithm>
#include <unordered_set>
#include <unordered_map>
#include <Thor/Vectors.hpp>
#include "util/Interval.h"
#include "sprites/TileManager.h"
const float Pathfinder::WALL_DISTANCE_MULTIPLIER = 1.5f;
/**
* Runs the A* path finding algorithm with areas as nodes and portals as edges.
*
* @warning Areas and portals must not be changed while this is running.
*
* @param start The area to start the path finding from. Must not be null.
* @param end The goal to reach. Must not be null.
* @return Path in reverse order (start being the last item and end the first).
*/
std::vector<Pathfinder::Portal*>
Pathfinder::astarArea(Area* start, Area* end) const {
assert(start);
assert(end);
std::unordered_set<Area*> closed;
std::unordered_map<Area*, float> openAreasEstimatedCost;
// Navigated areas with previous area/portal.
std::unordered_map<Area*, std::pair<Area*, Portal*>> previousAreaAndPortal;
std::unordered_map<Area*, float> bestPathCost;
openAreasEstimatedCost[start] = heuristic_cost_estimate(start, end);
bestPathCost[start] = 0;
while (!openAreasEstimatedCost.empty()) {
Area* current = std::min_element(openAreasEstimatedCost.begin(),
openAreasEstimatedCost.end())->first;
if (current == end) {
std::vector<Portal*> path;
auto previous = current;
while (previous != start) {
path.push_back(previousAreaAndPortal[previous].second);
previous = previousAreaAndPortal[previous].first;
}
return path;
}
openAreasEstimatedCost.erase(current);
closed.insert(current);
for (Portal& portal : current->portals) {
Area* neighbor = portal.area;
float tentative_g_score = bestPathCost[current] +
heuristic_cost_estimate(current,neighbor);
if (closed.find(neighbor) != closed.end()) {
if (tentative_g_score >= bestPathCost[neighbor])
continue;
}
if ((openAreasEstimatedCost.find(neighbor) ==
openAreasEstimatedCost.end()) ||
(tentative_g_score < bestPathCost[neighbor])) {
previousAreaAndPortal[neighbor] = std::make_pair(current,
&portal);
bestPathCost[neighbor] = tentative_g_score;
openAreasEstimatedCost[neighbor] = bestPathCost[neighbor] +
heuristic_cost_estimate(neighbor, end);
}
}
}
return std::vector<Portal*>();
}
/**
* Returns path in reverse order.
*
* @warning Areas and portals must not be changed while this running.
*
* @param start Position to start the path from.
* @param end Position to move to.
* @param radius Radius of the moving object.
* @return Path from end to start (path from start to end in reverse order).
*/
std::vector<sf::Vector2f>
Pathfinder::getPath(const sf::Vector2f& start, const sf::Vector2f& end,
float radius) const {
if (!getArea(end))
return std::vector<sf::Vector2f>();
std::vector<Portal*> portals = astarArea(getArea(start), getArea(end));
if (portals.empty())
return std::vector<sf::Vector2f>();
std::vector<sf::Vector2f> path;
path.push_back(end);
for (auto p : portals) {
// Find the point on the line of the portal closest to the previous point.
sf::Vector2f startToEnd = sf::Vector2f(p->end - p->start);
float percentage = thor::dotProduct(startToEnd, path.back() - sf::Vector2f(p->start)) /
thor::squaredLength(startToEnd);
sf::Vector2f point;
if (percentage < 0 || percentage > 1.0f) {
if (thor::squaredLength(sf::Vector2f(p->start) - path.back()) <
thor::squaredLength(sf::Vector2f(p->end) - path.back())) {
thor::setLength(startToEnd, WALL_DISTANCE_MULTIPLIER * radius);
point = sf::Vector2f(p->start) + startToEnd;
}
else {
thor::setLength(startToEnd, WALL_DISTANCE_MULTIPLIER * radius);
point = sf::Vector2f(p->end) - startToEnd;
}
}
else
point = sf::Vector2f(p->start) + startToEnd * percentage;
// Take two points on a line orthogonal to the portal.
thor::setLength(startToEnd, radius);
startToEnd = thor::perpendicularVector(startToEnd);
path.push_back(point + startToEnd);
path.push_back(point - startToEnd);
// Make sure the points are in the right order.
if (thor::squaredLength(*(path.end() - 1) - *(path.end() - 3) ) <
thor::squaredLength(*(path.end() - 2) - *(path.end() - 3) ))
std::swap(*(path.end() - 1), *(path.end() - 2));
}
return path;
}
/**
* Returns the linear distance between two areas (using their center).
*/
float
Pathfinder::heuristic_cost_estimate(Area* start, Area* end) const {
return thor::length(sf::Vector2f(end->center - start->center));
}
bool Pathfinder::Portal::operator==(const Portal& p) {
return start == p.start && end == p.end && area == p.area;
}
/**
* Inserts an area used for path finding.
*
* @parm rect Rectangle the area covers.
*/
void
Pathfinder::insertArea(const sf::IntRect& rect) {
Area a;
// Not sure why the offset of -50 is required, but with it, areas align
// with tiles perfectly.
a.area = sf::IntRect(rect.left * TileManager::TILE_SIZE.x - 50,
rect.top * TileManager::TILE_SIZE.y - 50,
rect.width * TileManager::TILE_SIZE.x,
rect.height * TileManager::TILE_SIZE.y);
a.center = sf::Vector2i(a.area.left + a.area.width / 2,
a.area.top + a.area.height / 2);
mAreas.push_back(a);
}
/**
* Generates portals that connect areas. Needs to be run after insertArea for
* path finding to work.
*/
void
Pathfinder::generatePortals() {
for (Area& it : mAreas) {
// We currently recreate portals for all existing areas, so we have
// to clear in case this was already generated.
it.portals.clear();
for (Area& other : mAreas) {
if (&it == &other)
continue;
Portal portal;
portal.area = &other;
if (it.area.left + it.area.width == other.area.left) {
Interval overlap = Interval::IntervalFromPoints(it.area.top,
it.area.top + it.area.height)
.getOverlap(Interval::IntervalFromPoints(other.area.top,
other.area.top + other.area.height));
if (overlap.getLength() > 0) {
portal.start = sf::Vector2i(other.area.left, overlap.start);
portal.end = sf::Vector2i(other.area.left, overlap.end);
it.portals.push_back(portal);
}
}
if (other.area.left + other.area.width == it.area.left) {
Interval overlap = Interval::IntervalFromPoints(it.area.top,
it.area.top + it.area.height)
.getOverlap(Interval::IntervalFromPoints(other.area.top,
other.area.top + other.area.height));
if (overlap.getLength() > 0) {
portal.start = sf::Vector2i(it.area.left, overlap.start);
portal.end = sf::Vector2i(it.area.left, overlap.end);
it.portals.push_back(portal);
}
}
else if (it.area.top + it.area.height == other.area.top) {
Interval overlap = Interval::IntervalFromPoints(it.area.left,
it.area.left + it.area.width)
.getOverlap(Interval::IntervalFromPoints(other.area.left,
other.area.left + other.area.width));
if (overlap.getLength() > 0) {
portal.start = sf::Vector2i(overlap.start, other.area.top);
portal.end = sf::Vector2i(overlap.end, other.area.top);
it.portals.push_back(portal);
}
}
else if (other.area.top + other.area.height == it.area.top) {
Interval overlap = Interval::IntervalFromPoints(it.area.left,
it.area.left + it.area.width)
.getOverlap(Interval::IntervalFromPoints(other.area.left,
other.area.left + other.area.width));
if (overlap.getLength() > 0) {
portal.start = sf::Vector2i(overlap.start, it.area.top);
portal.end = sf::Vector2i(overlap.end, it.area.top);
it.portals.push_back(portal);
}
}
}
}
}
/**
* Returns the area where point is in.
*/
Pathfinder::Area*
Pathfinder::getArea(const sf::Vector2f& point) const {
for (auto& area : mAreas) {
if (sf::FloatRect(area.area).contains(point))
// Make the return value non-const for convenience.
return &const_cast<Area&>(area);
}
return nullptr;
}

57
source/Pathfinder.h Normal file
View file

@ -0,0 +1,57 @@
/*
* Pathfinder.h
*
* Created on: 28.04.2013
* Author: Felix
*/
#ifndef DG_PATHFINDER_H_
#define DG_PATHFINDER_H_
#include <SFML/System.hpp>
#include <SFML/Graphics.hpp>
class Pathfinder {
private:
struct Area;
struct Portal;
public:
void insertArea(const sf::IntRect& rect);
void generatePortals();
std::vector<sf::Vector2f> getPath(const sf::Vector2f& start,
const sf::Vector2f& end, float radius) const;
private:
Area* getArea(const sf::Vector2f& point) const;
float heuristic_cost_estimate(Area* start, Area* end) const;
std::vector<Portal*> astarArea(Area* start, Area* end) const;
private:
static const float WALL_DISTANCE_MULTIPLIER;
std::vector<Area> mAreas; //< This has to be a vector as objects are compared by address.
};
/**
* Edges
*
* Redundant data as portals are saved twice.
*/
struct Pathfinder::Portal {
Portal() = default;
bool operator==(const Portal& p);
sf::Vector2i start;
sf::Vector2i end;
Area* area;
};
/**
* Nodes
*/
struct Pathfinder::Area {
sf::IntRect area;
sf::Vector2i center;
std::vector<Portal> portals;
};
#endif /* DG_PATHFINDER_H_ */

View file

@ -7,16 +7,11 @@
#include "World.h" #include "World.h"
#include <algorithm>
#include <unordered_set>
#include <unordered_map>
#include <Thor/Vectors.hpp> #include <Thor/Vectors.hpp>
#include "util/Interval.h"
#include "sprites/TileManager.h" #include "sprites/TileManager.h"
#include "util/Interval.h"
const float World::WALL_DISTANCE_MULTIPLIER = 1.5f;
/** /**
* Insert a drawable into the group. Drawables should only be handled with shared_ptr. * Insert a drawable into the group. Drawables should only be handled with shared_ptr.
* An object can't be inserted more than once at the same level. * An object can't be inserted more than once at the same level.
@ -45,122 +40,6 @@ World::insertCharacter(std::shared_ptr<Character> character) {
insert(character); insert(character);
} }
/**
* Runs the A* path finding algorithm with areas as nodes and portals as edges.
*
* @warning Areas and portals must not be changed while this is running.
*
* @param start The area to start the path finding from. Must not be null.
* @param end The goal to reach. Must not be null.
* @return Path in reverse order (start being the last item and end the first).
*/
std::vector<World::Portal*>
World::astarArea(Area* start, Area* end) const {
assert(start);
assert(end);
std::unordered_set<Area*> closed;
std::unordered_map<Area*, float> openAreasEstimatedCost;
// Navigated areas with previous area/portal.
std::unordered_map<Area*, std::pair<Area*, Portal*>> previousAreaAndPortal;
std::unordered_map<Area*, float> bestPathCost;
openAreasEstimatedCost[start] = heuristic_cost_estimate(start, end);
bestPathCost[start] = 0;
while (!openAreasEstimatedCost.empty()) {
Area* current = std::min_element(openAreasEstimatedCost.begin(),
openAreasEstimatedCost.end())->first;
if (current == end) {
std::vector<Portal*> path;
auto previous = current;
while (previous != start) {
path.push_back(previousAreaAndPortal[previous].second);
previous = previousAreaAndPortal[previous].first;
}
return path;
}
openAreasEstimatedCost.erase(current);
closed.insert(current);
for (Portal& portal : current->portals) {
Area* neighbor = portal.area;
float tentative_g_score = bestPathCost[current] +
heuristic_cost_estimate(current,neighbor);
if (closed.find(neighbor) != closed.end()) {
if (tentative_g_score >= bestPathCost[neighbor])
continue;
}
if ((openAreasEstimatedCost.find(neighbor) ==
openAreasEstimatedCost.end()) ||
(tentative_g_score < bestPathCost[neighbor])) {
previousAreaAndPortal[neighbor] = std::make_pair(current,
&portal);
bestPathCost[neighbor] = tentative_g_score;
openAreasEstimatedCost[neighbor] = bestPathCost[neighbor] +
heuristic_cost_estimate(neighbor, end);
}
}
}
return std::vector<Portal*>();
}
/**
* Returns path in reverse order.
*
* @warning Areas and portals must not be changed while this running.
*
* @param start Position to start the path from.
* @param end Position to move to.
* @param radius Radius of the moving object.
* @return Path from end to start (path from start to end in reverse order).
*/
std::vector<sf::Vector2f>
World::getPath(const sf::Vector2f& start, const sf::Vector2f& end,
float radius) const {
if (!getArea(end))
return std::vector<sf::Vector2f>();
std::vector<Portal*> portals = astarArea(getArea(start), getArea(end));
if (portals.empty())
return std::vector<sf::Vector2f>();
std::vector<sf::Vector2f> path;
path.push_back(end);
for (auto p : portals) {
// Find the point on the line of the portal closest to the previous point.
sf::Vector2f startToEnd = sf::Vector2f(p->end - p->start);
float percentage = thor::dotProduct(startToEnd, path.back() - sf::Vector2f(p->start)) /
thor::squaredLength(startToEnd);
sf::Vector2f point;
if (percentage < 0 || percentage > 1.0f) {
if (thor::squaredLength(sf::Vector2f(p->start) - path.back()) <
thor::squaredLength(sf::Vector2f(p->end) - path.back())) {
thor::setLength(startToEnd, WALL_DISTANCE_MULTIPLIER * radius);
point = sf::Vector2f(p->start) + startToEnd;
}
else {
thor::setLength(startToEnd, WALL_DISTANCE_MULTIPLIER * radius);
point = sf::Vector2f(p->end) - startToEnd;
}
}
else
point = sf::Vector2f(p->start) + startToEnd * percentage;
// Take two points on a line orthogonal to the portal.
thor::setLength(startToEnd, radius);
startToEnd = thor::perpendicularVector(startToEnd);
path.push_back(point + startToEnd);
path.push_back(point - startToEnd);
// Make sure the points are in the right order.
if (thor::squaredLength(*(path.end() - 1) - *(path.end() - 3) ) <
thor::squaredLength(*(path.end() - 2) - *(path.end() - 3) ))
std::swap(*(path.end() - 1), *(path.end() - 2));
}
return path;
}
/** /**
* Returns all characters that are within maxDistance from position. * Returns all characters that are within maxDistance from position.
*/ */
@ -177,23 +56,6 @@ std::vector<std::shared_ptr<Character> >
return visible; return visible;
} }
/**
* Initializes start and end of an area, sets area to null.
*/
World::Portal::Portal(const sf::Vector2i& start, const sf::Vector2i& end) :
start(start),
end(end),
area(nullptr) {
}
/**
* Returns the linear distance between two areas (using their center).
*/
float
World::heuristic_cost_estimate(Area* start, Area* end) const {
return thor::length(sf::Vector2f(end->center - start->center));
}
/** /**
* Checks for collisions and applies movement, also removes sprites if * Checks for collisions and applies movement, also removes sprites if
* Sprite::getDelete returns true. * Sprite::getDelete returns true.
@ -257,90 +119,6 @@ World::think(int elapsed) {
} }
} }
/**
* Inserts an area used for path finding.
*
* @parm rect Rectangle the area covers.
*/
void
World::insertArea(const sf::IntRect& rect) {
Area a;
// Not sure why the offset of -50 is required, but with it, areas align
// with tiles perfectly.
a.area = sf::IntRect(rect.left * TileManager::TILE_SIZE.x - 50,
rect.top * TileManager::TILE_SIZE.y - 50,
rect.width * TileManager::TILE_SIZE.x,
rect.height * TileManager::TILE_SIZE.y);
a.center = sf::Vector2i(a.area.left + a.area.width / 2,
a.area.top + a.area.height / 2);
mAreas.push_back(a);
}
/**
* Generates portals that connect areas. Needs to be run after insertArea for
* path finding to work.
*
* Could be improved by only checking nearby areas.
*/
void
World::generatePortals() {
for (Area& it : mAreas) {
// We currently recreate portals for all existing areas, so we have
// to clear in case this was already generated.
it.portals.clear();
for (Area& other : mAreas) {
if (&it == &other)
continue;
Portal portal;
portal.area = &other;
if (it.area.left + it.area.width == other.area.left) {
Interval overlap = Interval::IntervalFromPoints(it.area.top,
it.area.top + it.area.height)
.getOverlap(Interval::IntervalFromPoints(other.area.top,
other.area.top + other.area.height));
if (overlap.getLength() > 0) {
portal.start = sf::Vector2i(other.area.left, overlap.start);
portal.end = sf::Vector2i(other.area.left, overlap.end);
it.portals.push_back(portal);
}
}
if (other.area.left + other.area.width == it.area.left) {
Interval overlap = Interval::IntervalFromPoints(it.area.top,
it.area.top + it.area.height)
.getOverlap(Interval::IntervalFromPoints(other.area.top,
other.area.top + other.area.height));
if (overlap.getLength() > 0) {
portal.start = sf::Vector2i(it.area.left, overlap.start);
portal.end = sf::Vector2i(it.area.left, overlap.end);
it.portals.push_back(portal);
}
}
else if (it.area.top + it.area.height == other.area.top) {
Interval overlap = Interval::IntervalFromPoints(it.area.left,
it.area.left + it.area.width)
.getOverlap(Interval::IntervalFromPoints(other.area.left,
other.area.left + other.area.width));
if (overlap.getLength() > 0) {
portal.start = sf::Vector2i(overlap.start, other.area.top);
portal.end = sf::Vector2i(overlap.end, other.area.top);
it.portals.push_back(portal);
}
}
else if (other.area.top + other.area.height == it.area.top) {
Interval overlap = Interval::IntervalFromPoints(it.area.left,
it.area.left + it.area.width)
.getOverlap(Interval::IntervalFromPoints(other.area.left,
other.area.left + other.area.width));
if (overlap.getLength() > 0) {
portal.start = sf::Vector2i(overlap.start, it.area.top);
portal.end = sf::Vector2i(overlap.end, it.area.top);
it.portals.push_back(portal);
}
}
}
}
}
/** /**
* Tests for collisions using Seperating Axis Theorem (SAT). * Tests for collisions using Seperating Axis Theorem (SAT).
* *
@ -448,20 +226,6 @@ World::testCollision(std::shared_ptr<Sprite> spriteA,
return false; return false;
} }
/**
* Returns the area where point is in.
* Just iterates through all areas and tests each.
*/
World::Area*
World::getArea(const sf::Vector2f& point) const {
for (auto area = mAreas.begin(); area != mAreas.end(); area++) {
if (sf::FloatRect(area->area).contains(point))
// Make the return value non-const for convenience.
return &const_cast<Area&>(*area);
}
return nullptr;
}
/** /**
* Draws all elements in the group. * Draws all elements in the group.
*/ */

View file

@ -26,52 +26,18 @@ public:
void insertCharacter(std::shared_ptr<Character> character); void insertCharacter(std::shared_ptr<Character> character);
void step(int elapsed); void step(int elapsed);
void think(int elapsed); void think(int elapsed);
void insertArea(const sf::IntRect& rect);
void generatePortals();
std::vector<sf::Vector2f> getPath(const sf::Vector2f& start,
const sf::Vector2f& end, float radius) const;
std::vector<std::shared_ptr<Character> > std::vector<std::shared_ptr<Character> >
getCharacters(const sf::Vector2f& position, float maxDistance) const; getCharacters(const sf::Vector2f& position, float maxDistance) const;
private: private:
struct Area;
/**
* Edges
*
* Redundant data as portals are saved twice.
*/
struct Portal {
Portal() = default;
Portal(const sf::Vector2i& start, const sf::Vector2i& end);
bool operator==(const Portal& p) {
return start == p.start && end == p.end && area == p.area;
}
sf::Vector2i start;
sf::Vector2i end;
Area* area;
};
/**
* Nodes
*/
struct Area {
sf::IntRect area;
sf::Vector2i center;
std::vector<Portal> portals;
};
private: private:
void draw(sf::RenderTarget& target, sf::RenderStates states) const; void draw(sf::RenderTarget& target, sf::RenderStates states) const;
bool testCollision(std::shared_ptr<Sprite> spriteA, std::shared_ptr<Sprite> spriteB, bool testCollision(std::shared_ptr<Sprite> spriteA, std::shared_ptr<Sprite> spriteB,
int elapsed) const; int elapsed) const;
Area* getArea(const sf::Vector2f& point) const;
float heuristic_cost_estimate(Area* start, Area* end) const;
std::vector<Portal*> astarArea(Area* start, Area* end) const;
private: private:
static const float WALL_DISTANCE_MULTIPLIER;
std::map<Sprite::Category, std::vector<std::shared_ptr<Sprite> > > mDrawables; std::map<Sprite::Category, std::vector<std::shared_ptr<Sprite> > > mDrawables;
std::vector<Area> mAreas; //< This has to be a vector as objects are compared by address.
std::vector<std::shared_ptr<Character> > mCharacters; std::vector<std::shared_ptr<Character> > mCharacters;
}; };

View file

@ -15,23 +15,25 @@
#include "../util/Log.h" #include "../util/Log.h"
#include "../util/Yaml.h" #include "../util/Yaml.h"
#include "../World.h" #include "../World.h"
#include "../Pathfinder.h"
const float Character::VISION_DISTANCE = 500.0f; const float Character::VISION_DISTANCE = 500.0f;
/** /**
* Saves pointer to this instance in static var for think(). * Saves pointer to this instance in static var for think().
*/ */
Character::Character(World& world, TileManager& tileManager, const Data& data, Character::Character(World& world, TileManager& tileManager, Pathfinder& pathfinder,
const Yaml& config) : const Data& data, const Yaml& config) :
Sprite(data, config), Sprite(data, config),
mWorld(world), mWorld(world),
mTileManager(tileManager), mTileManager(tileManager),
mPathfinder(pathfinder),
mMaxHealth(config.get(YAML_KEY::HEALTH, YAML_DEFAULT::HEALTH)), mMaxHealth(config.get(YAML_KEY::HEALTH, YAML_DEFAULT::HEALTH)),
mCurrentHealth(mMaxHealth), mCurrentHealth(mMaxHealth),
mMovementSpeed(config.get(YAML_KEY::SPEED, YAML_DEFAULT::SPEED)), mMovementSpeed(config.get(YAML_KEY::SPEED, YAML_DEFAULT::SPEED)),
mWeapon(new Weapon(world, *this, mWeapon(new Weapon(world, *this,
Yaml(config.get(YAML_KEY::WEAPON, YAML_DEFAULT::WEAPON)))), Yaml(config.get(YAML_KEY::WEAPON, YAML_DEFAULT::WEAPON)))),
mLastPosition(getPosition()){ mLastPosition(getPosition()) {
} }
Character::~Character() { Character::~Character() {
@ -112,7 +114,7 @@ Character::setDestination(const sf::Vector2f& destination) {
mPath.clear(); mPath.clear();
return true; return true;
} }
mPath = mWorld.getPath(getPosition(), destination, getRadius()); mPath = mPathfinder.getPath(getPosition(), destination, getRadius());
if (!mPath.empty()) if (!mPath.empty())
setSpeed(mPath.back() - getPosition(), mMovementSpeed); setSpeed(mPath.back() - getPosition(), mMovementSpeed);
else { else {

View file

@ -10,6 +10,7 @@
#include "Sprite.h" #include "Sprite.h"
class Pathfinder;
class TileManager; class TileManager;
class World; class World;
class Weapon; class Weapon;
@ -21,7 +22,7 @@ class Yaml;
class Character : public Sprite { class Character : public Sprite {
public: public:
explicit Character(World& world, TileManager& tileManager, explicit Character(World& world, TileManager& tileManager,
const Data& data, const Yaml& config); Pathfinder& pathfinder, const Data& data, const Yaml& config);
virtual ~Character() = 0; virtual ~Character() = 0;
void onDamage(int damage); void onDamage(int damage);
@ -47,6 +48,7 @@ private:
friend class World; friend class World;
World& mWorld; World& mWorld;
TileManager& mTileManager; TileManager& mTileManager;
Pathfinder& mPathfinder;
const int mMaxHealth; const int mMaxHealth;
int mCurrentHealth; //< Current health. Between 0 and mMaxHealth. int mCurrentHealth; //< Current health. Between 0 and mMaxHealth.

View file

@ -13,7 +13,7 @@
#include "simplexnoise.h" #include "simplexnoise.h"
#include "../sprites/TileManager.h" #include "../sprites/TileManager.h"
#include "../util/Log.h" #include "../util/Log.h"
#include "../World.h" #include "../Pathfinder.h"
/// For usage with simplexnoise.h /// For usage with simplexnoise.h
uint8_t perm[512]; uint8_t perm[512];
@ -40,7 +40,7 @@ Generator::Generator() {
* power of two. * power of two.
*/ */
void void
Generator::generateTiles(TileManager& tm, World& world, Generator::generateTiles(TileManager& tm, Pathfinder& pathfinder,
const sf::IntRect& area) const { const sf::IntRect& area) const {
// Check if width and height are power of two. // Check if width and height are power of two.
assert(area.width && !(area.width & (area.width - 1))); assert(area.width && !(area.width & (area.width - 1)));
@ -74,8 +74,9 @@ Generator::generateTiles(TileManager& tm, World& world,
TileManager::Type::FLOOR); TileManager::Type::FLOOR);
} }
} }
generateAreas(world, filtered, area, sf::Vector2f(area.left, area.top)); generateAreas(pathfinder, filtered, area,
world.generatePortals(); sf::Vector2f(area.left, area.top));
pathfinder.generatePortals();
} }
/** /**
@ -155,13 +156,13 @@ Generator::filterWalls(std::vector<std::vector<bool> >& in,
* @param offset Offset of tiles[0][0] from World coordinate (0, 0). * @param offset Offset of tiles[0][0] from World coordinate (0, 0).
*/ */
void void
Generator::generateAreas(World& world, std::vector<std::vector<bool> >& tiles, Generator::generateAreas(Pathfinder& pathfinder, std::vector<std::vector<bool> >& tiles,
const sf::IntRect& area, const sf::Vector2f& offset) { const sf::IntRect& area, const sf::Vector2f& offset) {
assert(area.width > 0 && area.height > 0); assert(area.width > 0 && area.height > 0);
int count = countWalls(sf::IntRect(area.left - offset.y, area.top - offset.x, int count = countWalls(sf::IntRect(area.left - offset.y, area.top - offset.x,
area.width, area.height), tiles); area.width, area.height), tiles);
if (count == 0) { if (count == 0) {
world.insertArea(sf::IntRect(area)); pathfinder.insertArea(sf::IntRect(area));
} }
else if (count == area.width * area.height) { else if (count == area.width * area.height) {
return; return;
@ -169,13 +170,13 @@ Generator::generateAreas(World& world, std::vector<std::vector<bool> >& tiles,
else { else {
int halfWidth = area.width / 2.0f; int halfWidth = area.width / 2.0f;
int halfHeight = area.height / 2.0f; int halfHeight = area.height / 2.0f;
generateAreas(world, tiles, sf::IntRect(area.left, generateAreas(pathfinder, tiles, sf::IntRect(area.left,
area.top, halfWidth, halfHeight), offset); area.top, halfWidth, halfHeight), offset);
generateAreas(world, tiles, sf::IntRect(area.left + halfWidth, generateAreas(pathfinder, tiles, sf::IntRect(area.left + halfWidth,
area.top, halfWidth, halfHeight), offset); area.top, halfWidth, halfHeight), offset);
generateAreas(world, tiles, sf::IntRect(area.left, generateAreas(pathfinder, tiles, sf::IntRect(area.left,
area.top + halfHeight, halfWidth, halfHeight), offset); area.top + halfHeight, halfWidth, halfHeight), offset);
generateAreas(world, tiles, sf::IntRect(area.left + halfWidth, generateAreas(pathfinder, tiles, sf::IntRect(area.left + halfWidth,
area.top + halfHeight, halfWidth, halfHeight), offset); area.top + halfHeight, halfWidth, halfHeight), offset);
} }
} }

View file

@ -10,13 +10,13 @@
#include <SFML/Graphics.hpp> #include <SFML/Graphics.hpp>
class Pathfinder;
class TileManager; class TileManager;
class World;
class Generator { class Generator {
public: public:
explicit Generator(); explicit Generator();
void generateTiles(TileManager& tm, World& world, void generateTiles(TileManager& tm, Pathfinder& pathfinder,
const sf::IntRect& area) const; const sf::IntRect& area) const;
//void generateCharacters(World& world, const sf::IntRect& area) const; //void generateCharacters(World& world, const sf::IntRect& area) const;
sf::Vector2f getPlayerSpawn() const; sf::Vector2f getPlayerSpawn() const;
@ -29,7 +29,7 @@ private:
int x, int y, int longside, int shortside, int subtract); int x, int y, int longside, int shortside, int subtract);
static int countWalls(const sf::IntRect& area, static int countWalls(const sf::IntRect& area,
std::vector<std::vector<bool> >& tiles); std::vector<std::vector<bool> >& tiles);
static void generateAreas(World& world, static void generateAreas(Pathfinder& pathfinder,
std::vector<std::vector<bool> >& tiles, std::vector<std::vector<bool> >& tiles,
const sf::IntRect& area, const sf::Vector2f& offset); const sf::IntRect& area, const sf::Vector2f& offset);
}; };

View file

@ -9,9 +9,9 @@
#include <Thor/Vectors.hpp> #include <Thor/Vectors.hpp>
Enemy::Enemy(World& world, TileManager& tileManager, Enemy::Enemy(World& world, TileManager& tileManager, Pathfinder& pathfinder,
const sf::Vector2f& position, const Yaml& config) : const sf::Vector2f& position, const Yaml& config) :
Character(world, tileManager, Data(position, CATEGORY_ACTOR, MASK_ALL), Character(world, tileManager, pathfinder, Data(position, CATEGORY_ACTOR, MASK_ALL),
config) { config) {
} }

View file

@ -16,7 +16,8 @@ class Yaml;
class Enemy : public Character { class Enemy : public Character {
public: public:
explicit Enemy(World& world, TileManager& tileManager, explicit Enemy(World& world, TileManager& tileManager,
const sf::Vector2f& position, const Yaml& config); Pathfinder& pathfinder, const sf::Vector2f& position,
const Yaml& config);
protected: protected:
virtual void onThink(int elapsed); virtual void onThink(int elapsed);

View file

@ -12,10 +12,10 @@
/** /**
* Initializes Sprite. * Initializes Sprite.
*/ */
Player::Player(World& world, TileManager& tileManager, Player::Player(World& world, TileManager& tileManager, Pathfinder& pathfinder,
const sf::Vector2f& position, const Yaml& config) : const sf::Vector2f& position, const Yaml& config) :
Character(world, tileManager, Data(position, CATEGORY_ACTOR, MASK_ALL), Character(world, tileManager, pathfinder,
config), Data(position, CATEGORY_ACTOR, MASK_ALL), config),
mDirection(0) { mDirection(0) {
} }

View file

@ -30,7 +30,8 @@ public:
public: public:
explicit Player(World& world, TileManager& tileManager, explicit Player(World& world, TileManager& tileManager,
const sf::Vector2f& position, const Yaml& config); Pathfinder& pathfinder, const sf::Vector2f& position,
const Yaml& config);
void setCrosshairPosition(const sf::Vector2f& position); void setCrosshairPosition(const sf::Vector2f& position);
void pullTrigger(); void pullTrigger();