This repository has been archived on 2019-12-07. You can view files and clone it, but cannot push or open issues or pull requests.
dungeon-gunner/source/abstract/Character.cpp

173 lines
4 KiB
C++
Raw Normal View History

2012-10-01 09:02:44 +00:00
/*
* Actor.cpp
*
* Created on: 02.09.2012
* Author: Felix
2012-09-12 12:21:57 +00:00
*/
2012-10-01 09:02:44 +00:00
#include "Character.h"
#include <Thor/Vectors.hpp>
#include "../items/Weapon.h"
#include "../sprites/Corpse.h"
#include "../sprites/TileManager.h"
#include "../util/Log.h"
#include "../util/Yaml.h"
2013-03-29 17:34:51 +00:00
#include "../World.h"
2012-12-22 14:10:26 +00:00
const std::string Character::KEY_HEALTH = "health";
const int Character::DEFAULT_HEALTH = 100;
2012-12-22 14:10:26 +00:00
const std::string Character::KEY_SPEED = "speed";
const float Character::DEFAULT_SPEED = 100;
const std::string Character::KEY_WEAPON = "weapon";
const std::string Character::DEFAULT_WEAPON = "weapon.yaml";
const float Character::VISION_DISTANCE = 500.0f;
2012-10-01 09:02:44 +00:00
/**
* Saves pointer to this instance in static var for think().
*/
Character::Character(World& world, TileManager& tileManager, const Data& data,
const Yaml& config) :
2012-12-23 14:50:49 +00:00
Sprite(data, config),
2012-12-22 00:44:36 +00:00
mWorld(world),
mTileManager(tileManager),
mMaxHealth(config.get(KEY_HEALTH, DEFAULT_HEALTH)),
mCurrentHealth(mMaxHealth),
mMovementSpeed(config.get(KEY_SPEED, DEFAULT_SPEED)),
mWeapon(new Weapon(world, *this,
Yaml(config.get(KEY_WEAPON, DEFAULT_WEAPON)))),
mLastPosition(getPosition()){
2012-10-01 09:02:44 +00:00
}
Character::~Character() {
}
/**
* Subtracts health from Actor. Calls onDeath() when health reaches zero and marks
* object for deletion.
2012-10-01 09:02:44 +00:00
*
* @param damage Amount of health to subtract.
*/
void
Character::onDamage(int damage) {
mCurrentHealth -= damage;
if (mCurrentHealth <= 0) {
mCurrentHealth = 0;
onDeath();
setDelete(true);
2012-10-01 09:02:44 +00:00
}
2012-09-12 12:21:57 +00:00
}
2012-10-01 09:02:44 +00:00
/**
* Implement this function for any (periodical) AI computations.
2013-03-09 15:25:04 +00:00
* If overwritten, this function should always be called from the overwriting function.
*
2013-03-09 15:25:04 +00:00
* @param elapsed Amount of time to simulate.
*/
void
2013-03-09 15:25:04 +00:00
Character::onThink(int elapsed) {
mWeapon->onThink(elapsed);
move();
}
2012-10-01 09:02:44 +00:00
/**
* Called when health reaches zero. Default implementation drops a corpse at
* the current position.
2012-10-01 09:02:44 +00:00
*/
void
Character::onDeath() {
mWorld.insert(std::shared_ptr<Sprite>(new Corpse(getPosition(), Yaml("body.yaml"))));
2012-09-12 12:21:57 +00:00
}
/**
* Gets the default movement speed (walking) of the character.
*/
float
Character::getMovementSpeed() const {
return mMovementSpeed;
}
2012-10-13 16:57:12 +00:00
/**
2012-12-24 00:14:22 +00:00
* Pull the trigger on the attached weapon.
2012-10-13 16:57:12 +00:00
*/
void
2012-12-24 00:14:22 +00:00
Character::pullTrigger() {
mWeapon->pullTrigger();
2012-12-24 00:14:22 +00:00
}
/**
* Release the trigger on the attached weapon.
*/
void
Character::releaseTrigger() {
mWeapon->releaseTrigger();
2012-10-13 16:57:12 +00:00
}
/**
2013-04-18 10:45:37 +00:00
* Set a destination to be walked to. Call move() to actually
* perform the movement.
*
2013-04-18 10:45:37 +00:00
* @param destination An absolute point to move towards. Set to current
* position to stop movement.
* @return True if a path was found.
*/
bool
2012-12-22 14:10:26 +00:00
Character::setDestination(const sf::Vector2f& destination) {
2013-04-18 10:45:37 +00:00
if (destination == getPosition()) {
mPath.clear();
return true;
}
2013-03-27 15:09:01 +00:00
mPath = mWorld.getPath(getPosition(), destination, getRadius());
if (!mPath.empty())
setSpeed(mPath.back() - getPosition(), mMovementSpeed);
else {
setSpeed(sf::Vector2f(), 0);
LOG_W("No path found to destination.");
}
return !mPath.empty();
}
/**
* Move towards a destination. Call setDestination() for setting the destination.
* This is automatically called from onThink().
*/
void
Character::move() {
sf::Vector2f distanceTraveled = mLastPosition - getPosition();
mLastPosition = getPosition();
if (!mPath.empty()) {
// Point reached (during next time step).
if (thor::length(mPath.back() - getPosition()) < thor::length(distanceTraveled)) {
mPath.pop_back();
(!mPath.empty())
? setSpeed(mPath.back() - getPosition(), mMovementSpeed)
: setSpeed(sf::Vector2f(), 0);
}
}
}
/**
* Returns true if the character is currently moving.
*/
bool
Character::isMoving() const {
return !mPath.empty();
}
/**
* Tests if a target is visible from the current position.
*/
bool
Character::isVisible(const sf::Vector2f& target) const {
return mTileManager.raycast(getPosition(), target);
}
/**
* Calls World::getCharacters with current position.
*/
std::vector<std::shared_ptr<Character> >
Character::getCharacters() const {
return mWorld.getCharacters(getPosition(), VISION_DISTANCE);
}