/* * The ManaPlus Client * Copyright (C) 2004-2009 The Mana World Development Team * Copyright (C) 2009-2010 The Mana Developers * Copyright (C) 2011-2012 The ManaPlus Developers * * This file is part of The ManaPlus Client. * * This program 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. * * This program 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 this program. If not, see . */ #include "net/ea/playerhandler.h" #include "game.h" #include "localplayer.h" #include "party.h" #include "units.h" #include "gui/ministatuswindow.h" #include "gui/okdialog.h" #include "gui/skilldialog.h" #include "gui/statuswindow.h" #include "gui/viewport.h" #include "net/npchandler.h" #include "net/ea/eaprotocol.h" #include "utils/gettext.h" #include "debug.h" extern OkDialog *weightNotice; extern OkDialog *deathNotice; extern int weightNoticeTime; // Max. distance we are willing to scroll after a teleport; // everything beyond will reset the port hard. static const int MAP_TELEPORT_SCROLL_DISTANCE = 8; // TODO Move somewhere else namespace { /** * Listener used for handling the overweigth message. */ struct WeightListener : public gcn::ActionListener { void action(const gcn::ActionEvent &event A_UNUSED) { weightNotice = nullptr; } } weightListener; /** * Listener used for handling death message. */ struct DeathListener : public gcn::ActionListener { void action(const gcn::ActionEvent &event A_UNUSED) { if (Net::getPlayerHandler()) Net::getPlayerHandler()->respawn(); deathNotice = nullptr; Game::closeDialogs(); if (viewport) viewport->closePopupMenu(); Net::NpcHandler *const handler = Net::getNpcHandler(); if (handler) handler->clearDialogs(); if (player_node) player_node->respawn(); } } deathListener; } // anonymous namespace static const char *randomDeathMessage() { static const char *const deadMsg[] = { N_("You are dead."), N_("We regret to inform you that your character was killed in " "battle."), N_("You are not that alive anymore."), N_("The cold hands of the grim reaper are grabbing for your soul."), N_("Game Over!"), N_("Insert coin to continue."), N_("No, kids. Your character did not really die. It... " "err... went to a better place."), N_("Your plan of breaking your enemies weapon by " "bashing it with your throat failed."), N_("I guess this did not run too well."), // NetHack reference: N_("Do you want your possessions identified?"), // Secret of Mana reference: N_("Sadly, no trace of you was ever found..."), // Final Fantasy VI reference: N_("Annihilated."), // Earthbound reference: N_("Looks like you got your head handed to you."), // Leisure Suit Larry 1 reference: N_("You screwed up again, dump your body down the tubes " "and get you another one."), // Monty Python references (Dead Parrot sketch mostly): N_("You're not dead yet. You're just resting."), N_("You are no more."), N_("You have ceased to be."), N_("You've expired and gone to meet your maker."), N_("You're a stiff."), N_("Bereft of life, you rest in peace."), N_("If you weren't so animated, you'd be pushing up the daisies."), N_("Your metabolic processes are now history."), N_("You're off the twig."), N_("You've kicked the bucket."), N_("You've shuffled off your mortal coil, run down the " "curtain and joined the bleedin' choir invisibile."), N_("You are an ex-player."), N_("You're pining for the fjords.") }; const int random = static_cast(rand() % (sizeof(deadMsg) / sizeof(deadMsg[0]))); return gettext(deadMsg[random]); } namespace Ea { PlayerHandler::PlayerHandler() { } void PlayerHandler::decreaseAttribute(int attr A_UNUSED) { // Supported by eA? } void PlayerHandler::ignorePlayer(const std::string &player A_UNUSED, bool ignore A_UNUSED) { // TODO } void PlayerHandler::ignoreAll(bool ignore A_UNUSED) { // TODO } bool PlayerHandler::canCorrectAttributes() const { return false; } Vector PlayerHandler::getDefaultWalkSpeed() const { // Return an normalized speed for any side // as the offset is calculated elsewhere. return Vector(150, 150, 0); } void PlayerHandler::processWalkResponse(Net::MessageIn &msg) { /* * This client assumes that all walk messages succeed, * and that the server will send a correction notice * otherwise. */ uint16_t srcX, srcY, dstX, dstY; msg.readInt32(); //tick msg.readCoordinatePair(srcX, srcY, dstX, dstY); if (player_node) player_node->setRealPos(dstX, dstY); } void PlayerHandler::processPlayerWarp(Net::MessageIn &msg) { std::string mapPath = msg.readString(16); int x = msg.readInt16(); int y = msg.readInt16(); logger->log("Warping to %s (%d, %d)", mapPath.c_str(), x, y); if (!player_node) logger->log1("SMSG_PLAYER_WARP player_node null"); /* * We must clear the local player's target *before* the call * to changeMap, as it deletes all beings. */ if (player_node) player_node->stopAttack(); Game *const game = Game::instance(); const std::string ¤tMapName = game->getCurrentMapName(); const bool sameMap = (currentMapName == mapPath); // Switch the actual map, deleting the previous one if necessary mapPath = mapPath.substr(0, mapPath.rfind(".")); game->changeMap(mapPath); int scrollOffsetX = 0; int scrollOffsetY = 0; if (player_node) { const Map *const map = game->getCurrentMap(); if (map) { if (x >= map->getWidth()) x = map->getWidth() - 1; if (y >= map->getHeight()) y = map->getHeight() - 1; if (x < 0) x = 0; if (y < 0) y = 0; /* Scroll if neccessary */ if (!sameMap || (abs(x - player_node->getTileX()) > MAP_TELEPORT_SCROLL_DISTANCE) || (abs(y - player_node->getTileY()) > MAP_TELEPORT_SCROLL_DISTANCE)) { scrollOffsetX = (x - player_node->getTileX()) * map->getTileWidth(); scrollOffsetY = (y - player_node->getTileY()) * map->getTileHeight(); } } player_node->setAction(Being::STAND); player_node->setTileCoords(x, y); player_node->navigateClean(); } logger->log("Adjust scrolling by %d:%d", scrollOffsetX, scrollOffsetY); if (viewport) { viewport->returnCamera(); if (miniStatusWindow) miniStatusWindow->updateStatus(); viewport->scrollBy(scrollOffsetX, scrollOffsetY); } } void PlayerHandler::processPlayerStatUpdate1(Net::MessageIn &msg) { const int type = msg.readInt16(); const int value = msg.readInt32(); if (!player_node) return; switch (type) { case 0x0000: player_node->setWalkSpeed(Vector(static_cast( value), static_cast(value), 0)); PlayerInfo::setStatBase(PlayerInfo::WALK_SPEED, value); PlayerInfo::setStatMod(PlayerInfo::WALK_SPEED, 0); break; case 0x0004: break; // manner case 0x0005: PlayerInfo::setAttribute(PlayerInfo::HP, value); if (player_node->isInParty() && Party::getParty(1)) { PartyMember *const m = Party::getParty(1) ->getMember(player_node->getId()); if (m) { m->setHp(value); m->setMaxHp(PlayerInfo::getAttribute(PlayerInfo::MAX_HP)); } } break; case 0x0006: PlayerInfo::setAttribute(PlayerInfo::MAX_HP, value); if (player_node->isInParty() && Party::getParty(1)) { PartyMember *const m = Party::getParty(1)->getMember( player_node->getId()); if (m) { m->setHp(PlayerInfo::getAttribute(PlayerInfo::HP)); m->setMaxHp(value); } } break; case 0x0007: PlayerInfo::setAttribute(PlayerInfo::MP, value); break; case 0x0008: PlayerInfo::setAttribute(PlayerInfo::MAX_MP, value); break; case 0x0009: PlayerInfo::setAttribute(PlayerInfo::CHAR_POINTS, value); break; case 0x000b: PlayerInfo::setAttribute(PlayerInfo::LEVEL, value); if (player_node) { player_node->setLevel(value); player_node->updateName(); } break; case 0x000c: PlayerInfo::setAttribute(PlayerInfo::SKILL_POINTS, value); if (skillDialog) skillDialog->update(); break; case 0x0018: if (!weightNotice) { const int max = PlayerInfo::getAttribute( PlayerInfo::MAX_WEIGHT) / 2; const int total = PlayerInfo::getAttribute( PlayerInfo::TOTAL_WEIGHT); if (value >= max && total < max) { weightNoticeTime = cur_time + 5; weightNotice = new OkDialog(_("Message"), _("You are carrying more than " "half your weight. You are " "unable to regain health."), DIALOG_OK, false); weightNotice->addActionListener( &weightListener); } else if (value < max && total >= max) { weightNoticeTime = cur_time + 5; weightNotice = new OkDialog(_("Message"), _("You are carrying less than " "half your weight. You " "can regain health."), DIALOG_OK, false); weightNotice->addActionListener( &weightListener); } } PlayerInfo::setAttribute(PlayerInfo::TOTAL_WEIGHT, value); break; case 0x0019: PlayerInfo::setAttribute(PlayerInfo::MAX_WEIGHT, value); break; case 0x0029: PlayerInfo::setStatBase(static_cast( EA_ATK), value); PlayerInfo::updateAttrs(); break; case 0x002a: PlayerInfo::setStatMod(static_cast( EA_ATK), value); PlayerInfo::updateAttrs(); break; case 0x002b: PlayerInfo::setStatBase(static_cast( EA_MATK), value); break; case 0x002c: PlayerInfo::setStatMod(static_cast( EA_MATK), value); break; case 0x002d: PlayerInfo::setStatBase(static_cast( EA_DEF), value); break; case 0x002e: PlayerInfo::setStatMod(static_cast( EA_DEF), value); break; case 0x002f: PlayerInfo::setStatBase(static_cast( EA_MDEF), value); break; case 0x0030: PlayerInfo::setStatMod(static_cast( EA_MDEF), value); break; case 0x0031: PlayerInfo::setStatBase(static_cast( EA_HIT), value); break; case 0x0032: PlayerInfo::setStatBase(static_cast( EA_FLEE), value); break; case 0x0033: PlayerInfo::setStatMod(static_cast( EA_FLEE), value); break; case 0x0034: PlayerInfo::setStatBase(static_cast( EA_CRIT), value); break; case 0x0035: player_node->setAttackSpeed(value); PlayerInfo::setStatBase(PlayerInfo::ATTACK_DELAY, value); PlayerInfo::setStatMod(PlayerInfo::ATTACK_DELAY, 0); PlayerInfo::updateAttrs(); break; case 0x0037: PlayerInfo::setStatBase(static_cast( EA_JOB), value); break; case 500: player_node->setGMLevel(value); break; default: logger->log("QQQQ PLAYER_STAT_UPDATE_1 " + toString(type) + "," + toString(value)); break; } if (PlayerInfo::getAttribute(PlayerInfo::HP) == 0 && !deathNotice) { deathNotice = new OkDialog(_("Message"), randomDeathMessage(), DIALOG_OK, false); deathNotice->addActionListener(&deathListener); if (player_node->getCurrentAction() != Being::DEAD) { player_node->setAction(Being::DEAD); player_node->recalcSpritesOrder(); } } } void PlayerHandler::processPlayerStatUpdate2(Net::MessageIn &msg) { const int type = msg.readInt16(); switch (type) { case 0x0001: PlayerInfo::setAttribute(PlayerInfo::EXP, msg.readInt32()); break; case 0x0002: PlayerInfo::setStatExperience(static_cast( EA_JOB), msg.readInt32(), PlayerInfo::getStatExperience(static_cast( EA_JOB)).second); break; case 0x0014: { const int oldMoney = PlayerInfo::getAttribute(PlayerInfo::MONEY); const int newMoney = msg.readInt32(); if (newMoney > oldMoney) { SERVER_NOTICE(strprintf(_("You picked up %s."), Units::formatCurrency(newMoney - oldMoney).c_str())) } else if (newMoney < oldMoney) { SERVER_NOTICE(strprintf(_("You spent %s."), Units::formatCurrency(oldMoney - newMoney).c_str())) } PlayerInfo::setAttribute(PlayerInfo::MONEY, newMoney); break; } case 0x0016: PlayerInfo::setAttribute(PlayerInfo::EXP_NEEDED, msg.readInt32()); break; case 0x0017: PlayerInfo::setStatExperience(static_cast( EA_JOB), PlayerInfo::getStatExperience( static_cast(EA_JOB)).first, msg.readInt32()); break; default: logger->log("QQQQ PLAYER_STAT_UPDATE_2 " + toString(type)); break; } } void PlayerHandler::processPlayerStatUpdate3(Net::MessageIn &msg) { const int type = msg.readInt32(); const int base = msg.readInt32(); const int bonus = msg.readInt32(); PlayerInfo::setStatBase(static_cast( type), base, false); PlayerInfo::setStatMod(static_cast( type), bonus); if (type == EA_ATK || type == PlayerInfo::ATTACK_DELAY) PlayerInfo::updateAttrs(); } void PlayerHandler::processPlayerStatUpdate4(Net::MessageIn &msg) { const int type = msg.readInt16(); const int ok = msg.readInt8(); const int value = msg.readInt8(); if (ok != 1) { const int oldValue = PlayerInfo::getStatBase( static_cast(type)); int points = PlayerInfo::getAttribute(PlayerInfo::CHAR_POINTS); points += oldValue - value; PlayerInfo::setAttribute(PlayerInfo::CHAR_POINTS, points); SERVER_NOTICE(_("Cannot raise skill!")) } PlayerInfo::setStatBase(static_cast(type), value); } void PlayerHandler::processPlayerStatUpdate5(Net::MessageIn &msg) { PlayerInfo::setAttribute(PlayerInfo::CHAR_POINTS, msg.readInt16()); int val = msg.readInt8(); PlayerInfo::setStatBase(static_cast(EA_STR), val); if (statusWindow) statusWindow->setPointsNeeded(EA_STR, msg.readInt8()); else msg.readInt8(); val = msg.readInt8(); PlayerInfo::setStatBase(static_cast(EA_AGI), val); if (statusWindow) statusWindow->setPointsNeeded(EA_AGI, msg.readInt8()); else msg.readInt8(); val = msg.readInt8(); PlayerInfo::setStatBase(static_cast(EA_VIT), val); if (statusWindow) statusWindow->setPointsNeeded(EA_VIT, msg.readInt8()); else msg.readInt8(); val = msg.readInt8(); PlayerInfo::setStatBase(static_cast(EA_INT), val); if (statusWindow) statusWindow->setPointsNeeded(EA_INT, msg.readInt8()); else msg.readInt8(); val = msg.readInt8(); PlayerInfo::setStatBase(static_cast(EA_DEX), val); if (statusWindow) statusWindow->setPointsNeeded(EA_DEX, msg.readInt8()); else msg.readInt8(); val = msg.readInt8(); PlayerInfo::setStatBase(static_cast(EA_LUK), val); if (statusWindow) statusWindow->setPointsNeeded(EA_LUK, msg.readInt8()); else msg.readInt8(); PlayerInfo::setStatBase(static_cast(EA_ATK), msg.readInt16(), false); PlayerInfo::setStatMod(static_cast(EA_ATK), msg.readInt16()); PlayerInfo::updateAttrs(); val = msg.readInt16(); PlayerInfo::setStatBase(static_cast(EA_MATK), val, false); val = msg.readInt16(); PlayerInfo::setStatMod(static_cast(EA_MATK), val); PlayerInfo::setStatBase(static_cast(EA_DEF), msg.readInt16(), false); PlayerInfo::setStatMod(static_cast(EA_DEF), msg.readInt16()); PlayerInfo::setStatBase(static_cast(EA_MDEF), msg.readInt16(), false); PlayerInfo::setStatMod(static_cast(EA_MDEF), msg.readInt16()); PlayerInfo::setStatBase(static_cast(EA_HIT), msg.readInt16()); PlayerInfo::setStatBase(static_cast(EA_FLEE), msg.readInt16(), false); PlayerInfo::setStatMod(static_cast(EA_FLEE), msg.readInt16()); PlayerInfo::setStatBase(static_cast(EA_CRIT), msg.readInt16()); msg.readInt16(); // manner } void PlayerHandler::processPlayerStatUpdate6(Net::MessageIn &msg) { const int type = msg.readInt16(); if (statusWindow) { switch (type) { case 0x0020: statusWindow->setPointsNeeded(EA_STR, msg.readInt8()); break; case 0x0021: statusWindow->setPointsNeeded(EA_AGI, msg.readInt8()); break; case 0x0022: statusWindow->setPointsNeeded(EA_VIT, msg.readInt8()); break; case 0x0023: statusWindow->setPointsNeeded(EA_INT, msg.readInt8()); break; case 0x0024: statusWindow->setPointsNeeded(EA_DEX, msg.readInt8()); break; case 0x0025: statusWindow->setPointsNeeded(EA_LUK, msg.readInt8()); break; default: logger->log("QQQQ PLAYER_STAT_UPDATE_6 " + toString(type)); break; } } } void PlayerHandler::processPlayerArrowMessage(Net::MessageIn &msg) { const int type = msg.readInt16(); switch (type) { case 0: SERVER_NOTICE(_("Equip arrows first.")) break; case 3: // arrows equiped break; default: logger->log("QQQQ 0x013b: Unhandled message %i", type); break; } } bool PlayerHandler::canUseMagic() const { return PlayerInfo::getStatEffective(static_cast( EA_MATK)) > 0; } PlayerInfo::Attribute PlayerHandler::getJobLocation() const { return static_cast(EA_JOB); } PlayerInfo::Attribute PlayerHandler::getAttackLocation() const { return static_cast(EA_ATK); } } // namespace Ea