/* * The Mana World * Copyright 2004 The Mana World Development Team * * This file is part of The Mana World. * * The Mana World is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * any later version. * * The Mana World is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with The Mana World; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * * $Id$ */ #include "being.h" #include <sstream> #include <guichan/imagefont.hpp> #include "game.h" #include "graphics.h" #include "log.h" #include "map.h" #include "graphic/spriteset.h" #include "gui/gui.h" #include "net/messageout.h" #include "net/network.h" #include "net/protocol.h" #include "resources/resourcemanager.h" extern Being* autoTarget; extern std::map<int, Spriteset*> monsterset; Being *player_node = NULL; std::list<Being*> beings; PATH_NODE::PATH_NODE(unsigned short x, unsigned short y): x(x), y(y) { } Being* createBeing(unsigned int id, unsigned short job, Map *map) { Being *being = new Being; being->setId(id); being->job = job; being->setMap(map); beings.push_back(being); // If the being is a player, request the name if (being->getType() == Being::PLAYER) { MessageOut outMsg; outMsg.writeShort(0x0094); outMsg.writeLong(being->getId());//readLong(2)); writeSet(6); } // If the being is a monster then load the monsterset else if (being->job >= 1002 && monsterset.find(being->job - 1002) == monsterset.end()) { std::stringstream filename; filename << "graphics/sprites/monster" << (being->job - 1002) << ".png"; logger->log("%s",filename.str().c_str()); Image *monsterbitmap = ResourceManager::getInstance()->getImage(filename.str()); if (!monsterbitmap) { logger->error("Unable to load monster.png"); } else { monsterset[being->job - 1002] = new Spriteset(monsterbitmap, 60, 60); } } return being; } void remove_node(Being *being) { delete being; beings.remove(being); } Being *findNode(unsigned int id) { std::list<Being*>::iterator i; for (i = beings.begin(); i != beings.end(); i++) { Being *being = (*i); if (being->getId() == id) { return being; } } return NULL; } Being *findNode(unsigned short x, unsigned short y) { std::list<Being*>::iterator i; for (i = beings.begin(); i != beings.end(); i++) { Being *being = (*i); // Return being if found and it is not a dead monster if (being->x == x && being->y == y && being->action != Being::MONSTER_DEAD) { return being; } } return NULL; } Being* findNode(unsigned short x, unsigned short y, Being::Type type) { std::list<Being *>::iterator i; for (i = beings.begin(); i != beings.end(); i++) { Being *being = (*i); // Check if is a NPC (only low job ids) if (being->x == x && being->y == y && being->getType() == type && being->action != Being::MONSTER_DEAD) { return being; } } return NULL; } class BeingCompare { public: bool operator() (const Being *a, const Being *b) const { return a->y < b->y; } }; void sort() { beings.sort(BeingCompare()); } Being::Being(): job(0), x(0), y(0), direction(SOUTH), action(0), frame(0), speech_color(0), walk_time(0), speed(150), emotion(0), emotion_time(0), text_x(0), text_y(0), aspd(350), m_weapon(0), m_id(0), map(0), hairStyle(1), hairColor(1), speech_time(0), damage_time(0), showSpeech(false), showDamage(false) { } Being::~Being() { clearPath(); } void Being::setDestination(int destX, int destY) { if (!map) return; setPath(map->findPath(x, y, destX, destY)); } void Being::clearPath() { path.clear(); } void Being::setPath(std::list<PATH_NODE> path) { this->path = path; if (action != WALK && action != DEAD) { nextStep(); walk_time = tick_time; } } void Being::setHairColor(int color) { hairColor = color; if (hairColor < 1 || hairColor > NR_HAIR_COLORS + 1) { hairColor = 1; } } void Being::setHairStyle(int style) { hairStyle = style; if (hairStyle < 1 || hairStyle > NR_HAIR_STYLES) { hairStyle = 1; } } unsigned short Being::getHairColor() { return hairColor; } unsigned short Being::getHairStyle() { return hairStyle; } void Being::setSpeech(const std::string &text, int time) { speech = text; speech_time = tick_time; showSpeech = true; } void Being::setDamage(short amount, int time) { if (!amount) { damage = "miss"; } else { std::stringstream damageString; damageString << amount; damage = damageString.str(); } damage_time = tick_time; showDamage = true; } void Being::setMap(Map *map) { this->map = map; } void Being::nextStep() { if (!path.empty()) { PATH_NODE node = path.front(); path.pop_front(); int oldX = x; int oldY = y; int newX = node.x; int newY = node.y; if (newX > oldX) { if (newY > oldY) direction = SE; else if (newY < oldY) direction = NE; else direction = EAST; } else if (newX < oldX) { if (newY > oldY) direction = SW; else if (newY < oldY) direction = NW; else direction = WEST; } else { if (newY > oldY) direction = SOUTH; else if (newY < oldY) direction = NORTH; } x = newX; y = newY; action = WALK; walk_time += speed / 10; } else { action = STAND; } frame = 0; } void Being::logic() { if (getType() == PLAYER) { switch (action) { case WALK: frame = (get_elapsed_time(walk_time) * 4) / speed; if (frame >= 4) { nextStep(); } break; case ATTACK: frame = (get_elapsed_time(walk_time) * 4) / aspd; if (frame >= 4) { nextStep(); if (autoTarget && this == player_node) { attack(autoTarget); } } break; } if (emotion != 0) { emotion_time--; if (emotion_time == 0) { emotion = 0; } } } if (get_elapsed_time(speech_time) > 5000) { showSpeech = false; } if (get_elapsed_time(damage_time) > 3000) { showDamage = false; } } void Being::drawSpeech(Graphics *graphics) { // Draw speech above this being if (showSpeech) { graphics->setFont(speechFont); graphics->drawText(speech, text_x + 18, text_y - 60, gcn::Graphics::CENTER); // Backing to default font graphics->setFont(gui->getFont()); } if (showDamage) { // Selecting the right color if (damage == "miss") { graphics->setFont(hitYellowFont); } else if (getType() == MONSTER) { graphics->setFont(hitBlueFont); } else { graphics->setFont(hitRedFont); } int textX = 0; int textY = 0; if (getType() == PLAYER) { textX = 16; textY = 70; } else { textX = 60; textY = 0; } graphics->drawText(damage, text_x + textX, text_y - textY - get_elapsed_time(damage_time) / 100, gcn::Graphics::CENTER); // Backing to default font graphics->setFont(gui->getFont()); } } Being::Type Being::getType() { if (job < 10) { return PLAYER; } else if (job >= 100 & job < 200) { return NPC; } else if (job >= 1000 && job < 1200) { return MONSTER; } else { return UNKNOWN; } } void Being::setWeapon(unsigned short weapon) { m_weapon = weapon; } void Being::setWeaponById(unsigned short weapon) { switch (weapon) { case 529: // iron arrows case 1199: // arrows break; case 1200: // bow case 530: // short bow case 545: // forest bow setWeapon(2); break; case 521: // sharp knife case 522: // dagger case 536: // short sword case 1201: // knife setWeapon(1); break; case 0: // unequip setWeapon(0); break; default: logger->log("unknown item equiped : %d", weapon); } } void Being::setId(unsigned int id) { m_id = id; }