/* * The ManaPlus Client * Copyright (C) 2004-2009 The Mana World Development Team * Copyright (C) 2009-2010 The Mana Developers * Copyright (C) 2011-2014 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/tmwa/playerhandler.h" #include "configuration.h" #include "game.h" #include "soundmanager.h" #include "units.h" #include "being/attributes.h" #include "being/beingflag.h" #include "being/localplayer.h" #include "gui/windows/statuswindow.h" #include "net/tmwa/attrs.h" #include "net/tmwa/inventoryhandler.h" #include "net/tmwa/messageout.h" #include "net/tmwa/protocol.h" #include "gui/windows/skilldialog.h" #include "gui/windows/whoisonline.h" #include "gui/onlineplayer.h" #include "gui/viewport.h" #include "resources/map/map.h" #include "debug.h" extern Net::PlayerHandler *playerHandler; extern int serverVersion; namespace TmwAthena { PlayerHandler::PlayerHandler() : MessageHandler(), Ea::PlayerHandler() { static const uint16_t _messages[] = { SMSG_WALK_RESPONSE, SMSG_PLAYER_WARP, SMSG_PLAYER_STAT_UPDATE_1, SMSG_PLAYER_STAT_UPDATE_2, SMSG_PLAYER_STAT_UPDATE_3, SMSG_PLAYER_STAT_UPDATE_4, SMSG_PLAYER_STAT_UPDATE_5, SMSG_PLAYER_STAT_UPDATE_6, SMSG_PLAYER_ARROW_MESSAGE, SMSG_ONLINE_LIST, SMSG_MAP_MASK, SMSG_MAP_MUSIC, 0 }; handledMessages = _messages; playerHandler = this; } void PlayerHandler::handleMessage(Net::MessageIn &msg) { BLOCK_START("PlayerHandler::handleMessage") switch (msg.getId()) { case SMSG_WALK_RESPONSE: processWalkResponse(msg); break; case SMSG_PLAYER_WARP: processPlayerWarp(msg); break; case SMSG_PLAYER_STAT_UPDATE_1: processPlayerStatUpdate1(msg); break; case SMSG_PLAYER_STAT_UPDATE_2: processPlayerStatUpdate2(msg); break; case SMSG_PLAYER_STAT_UPDATE_3: // Update a base attribute processPlayerStatUpdate3(msg); break; case SMSG_PLAYER_STAT_UPDATE_4: // Attribute increase ack processPlayerStatUpdate4(msg); break; // Updates stats and status points case SMSG_PLAYER_STAT_UPDATE_5: processPlayerStatUpdate5(msg); break; case SMSG_PLAYER_STAT_UPDATE_6: processPlayerStatUpdate6(msg); break; case SMSG_PLAYER_ARROW_MESSAGE: processPlayerArrowMessage(msg); break; case SMSG_ONLINE_LIST: processOnlineList(msg); break; case SMSG_MAP_MASK: processMapMask(msg); break; case SMSG_MAP_MUSIC: processMapMusic(msg); break; default: break; } BLOCK_END("PlayerHandler::handleMessage") } void PlayerHandler::attack(const int id, const bool keep) const { createOutPacket(CMSG_PLAYER_CHANGE_ACT); outMsg.writeInt32(id); if (keep) outMsg.writeInt8(7); else outMsg.writeInt8(0); } void PlayerHandler::stopAttack() const { createOutPacket(CMSG_PLAYER_STOP_ATTACK); } void PlayerHandler::emote(const uint8_t emoteId) const { createOutPacket(CMSG_PLAYER_EMOTE); outMsg.writeInt8(emoteId); } void PlayerHandler::increaseAttribute(const int attr) const { if (attr >= STR && attr <= LUK) { createOutPacket(CMSG_STAT_UPDATE_REQUEST); outMsg.writeInt16(static_cast(attr)); outMsg.writeInt8(1); } } void PlayerHandler::increaseSkill(const uint16_t skillId) const { if (PlayerInfo::getAttribute(Attributes::SKILL_POINTS) <= 0) return; createOutPacket(CMSG_SKILL_LEVELUP_REQUEST); outMsg.writeInt16(skillId); } void PlayerHandler::pickUp(const FloorItem *const floorItem) const { if (!floorItem) return; createOutPacket(CMSG_ITEM_PICKUP); outMsg.writeInt32(floorItem->getId()); TmwAthena::InventoryHandler *const handler = static_cast(inventoryHandler); if (handler) handler->pushPickup(floorItem->getId()); } void PlayerHandler::setDirection(const unsigned char direction) const { createOutPacket(CMSG_PLAYER_CHANGE_DIR); outMsg.writeInt16(0); outMsg.writeInt8(direction); } void PlayerHandler::setDestination(const int x, const int y, const int direction) const { createOutPacket(CMSG_PLAYER_CHANGE_DEST); outMsg.writeCoordinates(static_cast(x), static_cast(y), static_cast(direction)); } void PlayerHandler::changeAction(const BeingAction::Action &action) const { char type; switch (action) { case BeingAction::SIT: type = 2; break; case BeingAction::STAND: type = 3; break; default: case BeingAction::MOVE: case BeingAction::ATTACK: case BeingAction::DEAD: case BeingAction::HURT: case BeingAction::SPAWN: return; } createOutPacket(CMSG_PLAYER_CHANGE_ACT); outMsg.writeInt32(0); outMsg.writeInt8(type); } void PlayerHandler::respawn() const { createOutPacket(CMSG_PLAYER_RESTART); outMsg.writeInt8(0); } void PlayerHandler::requestOnlineList() const { createOutPacket(CMSG_ONLINE_LIST); } void PlayerHandler::removeOption() const { } void PlayerHandler::changeCart(const int type A_UNUSED) const { } void PlayerHandler::setMemo() const { } void PlayerHandler::processOnlineList(Net::MessageIn &msg) { if (!whoIsOnline) return; BLOCK_START("PlayerHandler::processOnlineList") const int size = msg.readInt16() - 4; std::vector arr; if (!size) { if (whoIsOnline) whoIsOnline->loadList(arr); BLOCK_END("PlayerHandler::processOnlineList") return; } char *const start = reinterpret_cast(msg.readBytes(size)); if (!start) { BLOCK_END("PlayerHandler::processOnlineList") return; } const char *buf = start; int addVal = 1; if (serverVersion >= 4) addVal = 3; while (buf - start + 1 < size && *(buf + static_cast(addVal))) { unsigned char status = 255; unsigned char ver = 0; unsigned char level = 0; if (serverVersion >= 4) { status = *buf; buf ++; level = *buf; buf ++; ver = *buf; } buf ++; unsigned char gender = Gender::UNSPECIFIED; if (serverVersion >= 4) { if (config.getBoolValue("showgender")) { if (status & BeingFlag::GENDER_MALE) gender = Gender::MALE; else if (status & BeingFlag::GENDER_OTHER) gender = Gender::OTHER; else gender = Gender::FEMALE; } } arr.push_back(new OnlinePlayer(static_cast(buf), status, level, gender, ver)); buf += strlen(buf) + 1; } if (whoIsOnline) whoIsOnline->loadList(arr); delete [] start; BLOCK_END("PlayerHandler::processOnlineList") } void PlayerHandler::updateStatus(const uint8_t status) const { createOutPacket(CMSG_SET_STATUS); outMsg.writeInt8(status); outMsg.writeInt8(0); } void PlayerHandler::processMapMask(Net::MessageIn &msg) { const int mask = msg.readInt32(); msg.readInt32(); // unused Map *const map = Game::instance()->getCurrentMap(); if (map) map->setMask(mask); } void PlayerHandler::processMapMusic(Net::MessageIn &msg) { const int size = msg.readInt16() - 5; const std::string music = msg.readString(size); soundManager.playMusic(music); Map *const map = viewport->getMap(); if (map) map->setMusicFile(music); } void PlayerHandler::processPlayerStatUpdate5(Net::MessageIn &msg) { BLOCK_START("PlayerHandler::processPlayerStatUpdate5") PlayerInfo::setAttribute(Attributes::CHAR_POINTS, msg.readInt16("char points")); unsigned int val = msg.readUInt8("str"); PlayerInfo::setStatBase(STR, val); if (statusWindow) statusWindow->setPointsNeeded(STR, msg.readUInt8("str cost")); else msg.readUInt8("str cost"); val = msg.readUInt8("agi"); PlayerInfo::setStatBase(AGI, val); if (statusWindow) statusWindow->setPointsNeeded(AGI, msg.readUInt8("agi cost")); else msg.readUInt8("agi cost"); val = msg.readUInt8("vit"); PlayerInfo::setStatBase(VIT, val); if (statusWindow) statusWindow->setPointsNeeded(VIT, msg.readUInt8("vit cost")); else msg.readUInt8("vit cost"); val = msg.readUInt8("int"); PlayerInfo::setStatBase(INT, val); if (statusWindow) statusWindow->setPointsNeeded(INT, msg.readUInt8("int cost")); else msg.readUInt8("int cost"); val = msg.readUInt8("dex"); PlayerInfo::setStatBase(DEX, val); if (statusWindow) statusWindow->setPointsNeeded(DEX, msg.readUInt8("dex cost")); else msg.readUInt8("dex cost"); val = msg.readUInt8("luk"); PlayerInfo::setStatBase(LUK, val); if (statusWindow) statusWindow->setPointsNeeded(LUK, msg.readUInt8("luk cost")); else msg.readUInt8("luk cost"); PlayerInfo::setStatBase(ATK, msg.readInt16("atk"), false); PlayerInfo::setStatMod(ATK, msg.readInt16("atk+")); PlayerInfo::updateAttrs(); val = msg.readInt16("matk"); PlayerInfo::setStatBase(Attributes::MATK, val, false); val = msg.readInt16("matk+"); PlayerInfo::setStatMod(Attributes::MATK, val); PlayerInfo::setStatBase(Attributes::DEF, msg.readInt16("def"), false); PlayerInfo::setStatMod(Attributes::DEF, msg.readInt16("def+")); PlayerInfo::setStatBase(Attributes::MDEF, msg.readInt16("mdef"), false); PlayerInfo::setStatMod(Attributes::MDEF, msg.readInt16("mdef+")); PlayerInfo::setStatBase(Attributes::HIT, msg.readInt16("hit")); PlayerInfo::setStatBase(Attributes::FLEE, msg.readInt16("flee"), false); PlayerInfo::setStatMod(Attributes::FLEE, msg.readInt16("flee+")); PlayerInfo::setStatBase(Attributes::CRIT, msg.readInt16("crit")); PlayerInfo::setStatBase(Attributes::MANNER, msg.readInt16("manner")); msg.readInt16("unused?"); BLOCK_END("PlayerHandler::processPlayerStatUpdate5") } void PlayerHandler::processWalkResponse(Net::MessageIn &msg) { BLOCK_START("PlayerHandler::processWalkResponse") /* * 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, "move path"); msg.readUInt8("unused"); if (localPlayer) localPlayer->setRealPos(dstX, dstY); BLOCK_END("PlayerHandler::processWalkResponse") } void PlayerHandler::setShortcut(const int idx A_UNUSED, const uint8_t type A_UNUSED, const int id A_UNUSED, const int level A_UNUSED) const { } void PlayerHandler::doriDori() const { } void PlayerHandler::explosionSpirits() const { } void PlayerHandler::requestPvpInfo() const { } void PlayerHandler::revive() const { } void PlayerHandler::setViewEquipment(const bool allow A_UNUSED) const { } void PlayerHandler::setStat(const int type, const int base, const int mod, const bool notify) const { if (type == 500) { localPlayer->setGMLevel(base); return; } Ea::PlayerHandler::setStat(type, base, mod, notify); } } // namespace TmwAthena